2011-05-26 17:07:38 +00:00
|
|
|
/*
|
2011-10-23 17:52:20 +00:00
|
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
* Contributor(s): Chingiz Dyussenov, Arystanbek Dyussenov, Jan Diederich, Tod Liverseed.
|
|
|
|
*
|
|
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
|
|
*/
|
2011-05-26 17:07:38 +00:00
|
|
|
|
|
|
|
#include "GeometryExporter.h"
|
|
|
|
#include "AnimationExporter.h"
|
2011-07-17 17:30:41 +00:00
|
|
|
#include "MaterialExporter.h"
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
Global G;
|
|
|
|
|
2011-05-26 17:07:38 +00:00
|
|
|
template<class Functor>
|
2012-06-12 21:25:29 +00:00
|
|
|
void forEachObjectInExportSet(Scene *sce, Functor &f, LinkNode *export_set)
|
2011-05-26 17:07:38 +00:00
|
|
|
{
|
2012-06-12 21:25:29 +00:00
|
|
|
LinkNode *node;
|
2012-06-12 22:05:33 +00:00
|
|
|
for (node = export_set; node; node = node->next) {
|
2012-06-12 21:25:29 +00:00
|
|
|
Object *ob = (Object *)node->link;
|
2011-05-26 17:07:38 +00:00
|
|
|
f(ob);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnimationExporter::exportAnimations(Scene *sce)
|
2011-08-26 15:16:27 +00:00
|
|
|
{
|
2012-03-24 07:52:14 +00:00
|
|
|
if (hasAnimations(sce)) {
|
2011-08-26 15:16:27 +00:00
|
|
|
this->scene = sce;
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
openLibrary();
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2012-06-12 21:25:29 +00:00
|
|
|
forEachObjectInExportSet(sce, *this, this->export_settings->export_set);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
closeLibrary();
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
2011-08-26 15:16:27 +00:00
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
// called for each exported object
|
2012-06-12 22:05:33 +00:00
|
|
|
void AnimationExporter::operator()(Object *ob)
|
2011-08-26 15:16:27 +00:00
|
|
|
{
|
|
|
|
FCurve *fcu;
|
2012-02-27 10:35:39 +00:00
|
|
|
char *transformName;
|
2011-09-10 09:38:38 +00:00
|
|
|
/* bool isMatAnim = false; */ /* UNUSED */
|
2011-08-26 15:16:27 +00:00
|
|
|
|
|
|
|
//Export transform animations
|
2012-04-28 06:31:57 +00:00
|
|
|
if (ob->adt && ob->adt->action) {
|
2012-06-12 22:05:33 +00:00
|
|
|
fcu = (FCurve *)ob->adt->action->curves.first;
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
//transform matrix export for bones are temporarily disabled here.
|
2012-06-12 22:05:33 +00:00
|
|
|
if (ob->type == OB_ARMATURE) {
|
|
|
|
bArmature *arm = (bArmature *)ob->data;
|
|
|
|
for (Bone *bone = (Bone *)arm->bonebase.first; bone; bone = bone->next)
|
2011-08-26 15:16:27 +00:00
|
|
|
write_bone_animation_matrix(ob, bone);
|
2011-07-06 17:41:14 +00:00
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
while (fcu) {
|
2011-08-28 18:30:18 +00:00
|
|
|
//for armature animations as objects
|
2012-06-12 22:05:33 +00:00
|
|
|
if (ob->type == OB_ARMATURE)
|
2011-08-28 18:30:18 +00:00
|
|
|
transformName = fcu->rna_path;
|
|
|
|
else
|
2012-06-12 22:05:33 +00:00
|
|
|
transformName = extract_transform_name(fcu->rna_path);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
if ((!strcmp(transformName, "location") || !strcmp(transformName, "scale")) ||
|
2012-06-12 22:05:33 +00:00
|
|
|
(!strcmp(transformName, "rotation_euler") && ob->rotmode == ROT_MODE_EUL) ||
|
2012-04-28 06:31:57 +00:00
|
|
|
(!strcmp(transformName, "rotation_quaternion")))
|
|
|
|
{
|
2012-04-29 15:47:02 +00:00
|
|
|
dae_animation(ob, fcu, transformName, false);
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2011-08-26 15:16:27 +00:00
|
|
|
fcu = fcu->next;
|
2011-07-06 17:41:14 +00:00
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
}
|
2013-01-21 13:45:49 +00:00
|
|
|
|
|
|
|
export_object_constraint_animation(ob);
|
|
|
|
|
|
|
|
//This needs to be handled by extra profiles, so postponed for now
|
|
|
|
//export_morph_animation(ob);
|
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
//Export Lamp parameter animations
|
2012-06-12 22:05:33 +00:00
|
|
|
if ( (ob->type == OB_LAMP) && ((Lamp *)ob->data)->adt && ((Lamp *)ob->data)->adt->action) {
|
|
|
|
fcu = (FCurve *)(((Lamp *)ob->data)->adt->action->curves.first);
|
2011-08-26 15:16:27 +00:00
|
|
|
while (fcu) {
|
2012-06-12 22:05:33 +00:00
|
|
|
transformName = extract_transform_name(fcu->rna_path);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if ((!strcmp(transformName, "color")) || (!strcmp(transformName, "spot_size")) ||
|
2012-04-28 06:31:57 +00:00
|
|
|
(!strcmp(transformName, "spot_blend")) || (!strcmp(transformName, "distance")))
|
|
|
|
{
|
2012-04-29 15:47:02 +00:00
|
|
|
dae_animation(ob, fcu, transformName, true);
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2011-08-26 15:16:27 +00:00
|
|
|
fcu = fcu->next;
|
2011-07-06 17:41:14 +00:00
|
|
|
}
|
2011-08-26 15:16:27 +00:00
|
|
|
}
|
2011-07-17 18:51:03 +00:00
|
|
|
|
2011-08-26 15:16:27 +00:00
|
|
|
//Export Camera parameter animations
|
2012-06-12 22:05:33 +00:00
|
|
|
if ( (ob->type == OB_CAMERA) && ((Camera *)ob->data)->adt && ((Camera *)ob->data)->adt->action) {
|
|
|
|
fcu = (FCurve *)(((Camera *)ob->data)->adt->action->curves.first);
|
2011-08-26 15:16:27 +00:00
|
|
|
while (fcu) {
|
2012-06-12 22:05:33 +00:00
|
|
|
transformName = extract_transform_name(fcu->rna_path);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if ((!strcmp(transformName, "lens")) ||
|
|
|
|
(!strcmp(transformName, "ortho_scale")) ||
|
2012-08-05 21:35:09 +00:00
|
|
|
(!strcmp(transformName, "clip_end")) ||
|
|
|
|
(!strcmp(transformName, "clip_start")))
|
2012-04-28 06:31:57 +00:00
|
|
|
{
|
2012-04-29 15:47:02 +00:00
|
|
|
dae_animation(ob, fcu, transformName, true);
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2011-08-26 15:16:27 +00:00
|
|
|
fcu = fcu->next;
|
2011-07-17 18:51:03 +00:00
|
|
|
}
|
2011-08-26 15:16:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//Export Material parameter animations.
|
2012-04-28 06:31:57 +00:00
|
|
|
for (int a = 0; a < ob->totcol; a++) {
|
2012-06-12 22:05:33 +00:00
|
|
|
Material *ma = give_current_material(ob, a + 1);
|
2011-08-26 15:16:27 +00:00
|
|
|
if (!ma) continue;
|
2012-04-28 06:31:57 +00:00
|
|
|
if (ma->adt && ma->adt->action) {
|
2011-09-10 09:38:38 +00:00
|
|
|
/* isMatAnim = true; */
|
2012-06-12 22:05:33 +00:00
|
|
|
fcu = (FCurve *)ma->adt->action->curves.first;
|
2011-08-26 15:16:27 +00:00
|
|
|
while (fcu) {
|
2012-06-12 22:05:33 +00:00
|
|
|
transformName = extract_transform_name(fcu->rna_path);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if ((!strcmp(transformName, "specular_hardness")) || (!strcmp(transformName, "specular_color")) ||
|
|
|
|
(!strcmp(transformName, "diffuse_color")) || (!strcmp(transformName, "alpha")) ||
|
|
|
|
(!strcmp(transformName, "ior")))
|
2012-04-28 06:31:57 +00:00
|
|
|
{
|
2012-06-12 22:05:33 +00:00
|
|
|
dae_animation(ob, fcu, transformName, true, ma);
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
fcu = fcu->next;
|
|
|
|
}
|
|
|
|
}
|
2013-01-21 13:45:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnimationExporter::export_object_constraint_animation(Object *ob)
|
|
|
|
{
|
|
|
|
std::vector<float> fra;
|
|
|
|
//Takes frames of target animations
|
|
|
|
make_anim_frames_from_targets(ob, fra);
|
|
|
|
|
|
|
|
if (fra.size())
|
|
|
|
dae_baked_object_animation(fra, ob);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnimationExporter::export_morph_animation(Object *ob)
|
|
|
|
{
|
|
|
|
FCurve *fcu;
|
|
|
|
char *transformName;
|
|
|
|
Key *key = BKE_key_from_object(ob);
|
|
|
|
if(!key) return;
|
|
|
|
|
|
|
|
if(key->adt && key->adt->action){
|
|
|
|
fcu = (FCurve *)key->adt->action->curves.first;
|
|
|
|
|
|
|
|
while (fcu) {
|
|
|
|
transformName = extract_transform_name(fcu->rna_path);
|
|
|
|
|
|
|
|
dae_animation(ob, fcu, transformName, true);
|
|
|
|
|
|
|
|
fcu = fcu->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
void AnimationExporter::make_anim_frames_from_targets(Object *ob, std::vector<float> &frames ){
|
|
|
|
|
|
|
|
ListBase *conlist = get_active_constraints(ob);
|
|
|
|
if(conlist == NULL) return;
|
|
|
|
bConstraint *con;
|
|
|
|
for (con = (bConstraint*)conlist->first; con; con = con->next) {
|
|
|
|
ListBase targets = {NULL, NULL};
|
|
|
|
|
|
|
|
bConstraintTypeInfo *cti = BKE_constraint_get_typeinfo(con);
|
|
|
|
|
|
|
|
if(!validateConstraints(con)) continue;
|
|
|
|
|
|
|
|
if (cti && cti->get_constraint_targets) {
|
|
|
|
bConstraintTarget *ct;
|
|
|
|
Object *obtar;
|
|
|
|
/* get targets
|
|
|
|
* - constraints should use ct->matrix, not directly accessing values
|
|
|
|
* - ct->matrix members have not yet been calculated here!
|
|
|
|
*/
|
|
|
|
cti->get_constraint_targets(con, &targets);
|
|
|
|
if(cti){
|
|
|
|
for (ct = (bConstraintTarget*)targets.first; ct; ct = ct->next){
|
|
|
|
obtar = ct->tar;
|
|
|
|
find_frames(obtar, frames);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
2011-08-26 15:16:27 +00:00
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//euler sources from quternion sources
|
2012-06-12 22:05:33 +00:00
|
|
|
float *AnimationExporter::get_eul_source_for_quat(Object *ob)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
2012-06-12 22:05:33 +00:00
|
|
|
FCurve *fcu = (FCurve *)ob->adt->action->curves.first;
|
2011-09-04 00:15:59 +00:00
|
|
|
const int keys = fcu->totvert;
|
2012-06-12 22:05:33 +00:00
|
|
|
float *quat = (float *)MEM_callocN(sizeof(float) * fcu->totvert * 4, "quat output source values");
|
|
|
|
float *eul = (float *)MEM_callocN(sizeof(float) * fcu->totvert * 3, "quat output source values");
|
2011-09-04 00:15:59 +00:00
|
|
|
float temp_quat[4];
|
|
|
|
float temp_eul[3];
|
2012-04-28 06:31:57 +00:00
|
|
|
while (fcu) {
|
2012-06-12 22:05:33 +00:00
|
|
|
char *transformName = extract_transform_name(fcu->rna_path);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if (!strcmp(transformName, "rotation_quaternion") ) {
|
|
|
|
for (int i = 0; i < fcu->totvert; i++) {
|
|
|
|
*(quat + (i * 4) + fcu->array_index) = fcu->bezt[i].vec[1][1];
|
2011-06-14 20:42:01 +00:00
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
|
|
|
fcu = fcu->next;
|
|
|
|
}
|
2011-06-14 20:42:01 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
for (int i = 0; i < keys; i++) {
|
|
|
|
for (int j = 0; j < 4; j++)
|
|
|
|
temp_quat[j] = quat[(i * 4) + j];
|
2011-06-16 15:04:37 +00:00
|
|
|
|
2012-04-29 15:47:02 +00:00
|
|
|
quat_to_eul(temp_eul, temp_quat);
|
2011-06-16 15:04:37 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
for (int k = 0; k < 3; k++)
|
|
|
|
eul[i * 3 + k] = temp_eul[k];
|
2011-06-14 20:42:01 +00:00
|
|
|
|
2011-06-16 15:04:37 +00:00
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
MEM_freeN(quat);
|
|
|
|
return eul;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get proper name for bones
|
2012-06-12 22:05:33 +00:00
|
|
|
std::string AnimationExporter::getObjectBoneName(Object *ob, const FCurve *fcu)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
//hard-way to derive the bone name from rna_path. Must find more compact method
|
|
|
|
std::string rna_path = std::string(fcu->rna_path);
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
char *boneName = strtok((char *)rna_path.c_str(), "\"");
|
2012-04-29 15:47:02 +00:00
|
|
|
boneName = strtok(NULL, "\"");
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if (boneName != NULL)
|
2011-09-04 00:15:59 +00:00
|
|
|
return /*id_name(ob) + "_" +*/ std::string(boneName);
|
2012-10-21 05:46:41 +00:00
|
|
|
else
|
2011-09-04 00:15:59 +00:00
|
|
|
return id_name(ob);
|
|
|
|
}
|
|
|
|
|
|
|
|
//convert f-curves to animation curves and write
|
2012-06-12 22:05:33 +00:00
|
|
|
void AnimationExporter::dae_animation(Object *ob, FCurve *fcu, char *transformName, bool is_param, Material *ma)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
const char *axis_name = NULL;
|
|
|
|
char anim_id[200];
|
|
|
|
|
|
|
|
bool has_tangents = false;
|
|
|
|
bool quatRotation = false;
|
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if (!strcmp(transformName, "rotation_quaternion") ) {
|
2011-09-04 00:15:59 +00:00
|
|
|
fprintf(stderr, "quaternion rotation curves are not supported. rotation curve will not be exported\n");
|
|
|
|
quatRotation = true;
|
|
|
|
return;
|
2011-06-05 18:58:22 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//axis names for colors
|
2012-08-05 21:35:09 +00:00
|
|
|
else if (!strcmp(transformName, "color") ||
|
|
|
|
!strcmp(transformName, "specular_color") ||
|
|
|
|
!strcmp(transformName, "diffuse_color") ||
|
|
|
|
!strcmp(transformName, "alpha"))
|
2011-06-05 18:58:22 +00:00
|
|
|
{
|
2011-09-04 00:15:59 +00:00
|
|
|
const char *axis_names[] = {"R", "G", "B"};
|
|
|
|
if (fcu->array_index < 3)
|
2011-06-30 18:24:45 +00:00
|
|
|
axis_name = axis_names[fcu->array_index];
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//axis names for transforms
|
2012-08-05 21:35:09 +00:00
|
|
|
else if (!strcmp(transformName, "location") ||
|
|
|
|
!strcmp(transformName, "scale") ||
|
|
|
|
!strcmp(transformName, "rotation_euler") ||
|
|
|
|
!strcmp(transformName, "rotation_quaternion"))
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
const char *axis_names[] = {"X", "Y", "Z"};
|
|
|
|
if (fcu->array_index < 3)
|
2011-06-05 18:58:22 +00:00
|
|
|
axis_name = axis_names[fcu->array_index];
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2012-04-28 06:31:57 +00:00
|
|
|
else {
|
|
|
|
/* no axis name. single parameter */
|
2011-09-04 00:15:59 +00:00
|
|
|
axis_name = "";
|
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::string ob_name = std::string("null");
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//Create anim Id
|
2012-04-28 06:31:57 +00:00
|
|
|
if (ob->type == OB_ARMATURE) {
|
2012-04-29 15:47:02 +00:00
|
|
|
ob_name = getObjectBoneName(ob, fcu);
|
2012-06-12 22:05:33 +00:00
|
|
|
BLI_snprintf(anim_id, sizeof(anim_id), "%s_%s.%s", (char *)translate_id(ob_name).c_str(),
|
|
|
|
transformName, axis_name);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2012-04-28 06:31:57 +00:00
|
|
|
else {
|
2011-09-04 00:15:59 +00:00
|
|
|
if (ma)
|
|
|
|
ob_name = id_name(ob) + "_material";
|
|
|
|
else
|
|
|
|
ob_name = id_name(ob);
|
2012-06-12 22:05:33 +00:00
|
|
|
BLI_snprintf(anim_id, sizeof(anim_id), "%s_%s_%s", (char *)translate_id(ob_name).c_str(),
|
|
|
|
fcu->rna_path, axis_name);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
openAnimation(anim_id, COLLADABU::Utils::EMPTY_STRING);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create input source
|
|
|
|
std::string input_id = create_source_from_fcurve(COLLADASW::InputSemantic::INPUT, fcu, anim_id, axis_name);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create output source
|
2012-02-27 10:35:39 +00:00
|
|
|
std::string output_id;
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//quat rotations are skipped for now, because of complications with determining axis.
|
2012-04-28 06:31:57 +00:00
|
|
|
if (quatRotation) {
|
|
|
|
float *eul = get_eul_source_for_quat(ob);
|
2012-06-12 22:05:33 +00:00
|
|
|
float *eul_axis = (float *)MEM_callocN(sizeof(float) * fcu->totvert, "quat output source values");
|
|
|
|
for (int i = 0; i < fcu->totvert; i++) {
|
|
|
|
eul_axis[i] = eul[i * 3 + fcu->array_index];
|
2012-04-28 06:31:57 +00:00
|
|
|
}
|
2012-06-12 22:05:33 +00:00
|
|
|
output_id = create_source_from_array(COLLADASW::InputSemantic::OUTPUT, eul_axis, fcu->totvert, quatRotation, anim_id, axis_name);
|
2011-09-04 00:15:59 +00:00
|
|
|
MEM_freeN(eul);
|
|
|
|
MEM_freeN(eul_axis);
|
|
|
|
}
|
2012-10-21 07:58:38 +00:00
|
|
|
else if (!strcmp(transformName, "lens") && (ob->type == OB_CAMERA)) {
|
2012-08-05 21:35:09 +00:00
|
|
|
output_id = create_lens_source_from_fcurve((Camera *) ob->data, COLLADASW::InputSemantic::OUTPUT, fcu, anim_id);
|
|
|
|
}
|
2012-04-28 06:31:57 +00:00
|
|
|
else {
|
2012-06-12 22:05:33 +00:00
|
|
|
output_id = create_source_from_fcurve(COLLADASW::InputSemantic::OUTPUT, fcu, anim_id, axis_name);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2012-08-05 21:35:09 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create interpolations source
|
|
|
|
std::string interpolation_id = create_interpolation_source(fcu, anim_id, axis_name, &has_tangents);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// handle tangents (if required)
|
|
|
|
std::string intangent_id;
|
|
|
|
std::string outtangent_id;
|
2011-07-02 05:05:03 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (has_tangents) {
|
|
|
|
// create in_tangent source
|
|
|
|
intangent_id = create_source_from_fcurve(COLLADASW::InputSemantic::IN_TANGENT, fcu, anim_id, axis_name);
|
2011-07-02 05:05:03 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create out_tangent source
|
|
|
|
outtangent_id = create_source_from_fcurve(COLLADASW::InputSemantic::OUT_TANGENT, fcu, anim_id, axis_name);
|
|
|
|
}
|
2011-07-18 17:33:03 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::string sampler_id = std::string(anim_id) + SAMPLER_ID_SUFFIX;
|
|
|
|
COLLADASW::LibraryAnimations::Sampler sampler(sw, sampler_id);
|
|
|
|
std::string empty;
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INPUT, COLLADABU::URI(empty, input_id));
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::OUTPUT, COLLADABU::URI(empty, output_id));
|
|
|
|
|
|
|
|
// this input is required
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INTERPOLATION, COLLADABU::URI(empty, interpolation_id));
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (has_tangents) {
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::IN_TANGENT, COLLADABU::URI(empty, intangent_id));
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::OUT_TANGENT, COLLADABU::URI(empty, outtangent_id));
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
addSampler(sampler);
|
|
|
|
|
2012-02-27 10:35:39 +00:00
|
|
|
std::string target;
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if (!is_param)
|
2012-05-09 09:24:15 +00:00
|
|
|
target = translate_id(ob_name) +
|
|
|
|
"/" + get_transform_sid(fcu->rna_path, -1, axis_name, true);
|
2012-04-28 06:31:57 +00:00
|
|
|
else {
|
2012-06-12 22:05:33 +00:00
|
|
|
if (ob->type == OB_LAMP)
|
2012-05-09 09:24:15 +00:00
|
|
|
target = get_light_id(ob) +
|
|
|
|
"/" + get_light_param_sid(fcu->rna_path, -1, axis_name, true);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if (ob->type == OB_CAMERA)
|
2012-05-09 09:24:15 +00:00
|
|
|
target = get_camera_id(ob) +
|
|
|
|
"/" + get_camera_param_sid(fcu->rna_path, -1, axis_name, true);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
if (ma)
|
2012-05-09 09:24:15 +00:00
|
|
|
target = translate_id(id_name(ma)) + "-effect" +
|
2012-06-12 22:05:33 +00:00
|
|
|
"/common/" /*profile common is only supported */ + get_transform_sid(fcu->rna_path, -1, axis_name, true);
|
2013-01-21 13:45:49 +00:00
|
|
|
//if shape key animation, this is the main problem, how to define the channel targets.
|
|
|
|
/*target = get_morph_id(ob) +
|
|
|
|
"/value" +*/
|
2011-08-03 19:12:18 +00:00
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
addChannel(COLLADABU::URI(empty, sampler_id), target);
|
2011-08-03 19:12:18 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
closeAnimation();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//write bone animations in transform matrix sources
|
|
|
|
void AnimationExporter::write_bone_animation_matrix(Object *ob_arm, Bone *bone)
|
|
|
|
{
|
|
|
|
if (!ob_arm->adt)
|
|
|
|
return;
|
|
|
|
|
|
|
|
//This will only export animations of bones in deform group.
|
2012-03-28 05:03:24 +00:00
|
|
|
/* if (!is_bone_deform_group(bone)) return; */
|
2011-09-04 00:15:59 +00:00
|
|
|
|
|
|
|
sample_and_write_bone_animation_matrix(ob_arm, bone);
|
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
for (Bone *child = (Bone *)bone->childbase.first; child; child = child->next)
|
2011-09-04 00:15:59 +00:00
|
|
|
write_bone_animation_matrix(ob_arm, child);
|
|
|
|
}
|
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
bool AnimationExporter::is_bone_deform_group(Bone *bone)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
bool is_def;
|
|
|
|
//Check if current bone is deform
|
2012-06-12 22:05:33 +00:00
|
|
|
if ((bone->flag & BONE_NO_DEFORM) == 0) return true;
|
2011-09-04 00:15:59 +00:00
|
|
|
//Check child bones
|
2012-04-28 06:31:57 +00:00
|
|
|
else {
|
2012-06-12 22:05:33 +00:00
|
|
|
for (Bone *child = (Bone *)bone->childbase.first; child; child = child->next) {
|
2011-09-04 00:15:59 +00:00
|
|
|
//loop through all the children until deform bone is found, and then return
|
|
|
|
is_def = is_bone_deform_group(child);
|
|
|
|
if (is_def) return true;
|
2011-08-21 13:51:04 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
//no deform bone found in children also
|
|
|
|
return false;
|
|
|
|
}
|
2011-08-21 13:51:04 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
void AnimationExporter::sample_and_write_bone_animation_matrix(Object *ob_arm, Bone *bone)
|
|
|
|
{
|
2012-06-12 22:05:33 +00:00
|
|
|
bArmature *arm = (bArmature *)ob_arm->data;
|
2011-09-04 00:15:59 +00:00
|
|
|
int flag = arm->flag;
|
|
|
|
std::vector<float> fra;
|
|
|
|
//char prefix[256];
|
2011-08-03 19:12:18 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
//Check if there is a fcurve in the armature for the bone in param
|
|
|
|
//when baking this check is not needed, solve every bone for every frame.
|
2013-02-01 15:17:51 +00:00
|
|
|
/*FCurve *fcu = (FCurve *)ob_arm->adt->action->curves.first;
|
|
|
|
|
|
|
|
while (fcu) {
|
2012-04-29 15:47:02 +00:00
|
|
|
std::string bone_name = getObjectBoneName(ob_arm, fcu);
|
2012-06-12 22:05:33 +00:00
|
|
|
int val = BLI_strcasecmp((char *)bone_name.c_str(), bone->name);
|
|
|
|
if (val == 0) break;
|
2011-09-04 00:15:59 +00:00
|
|
|
fcu = fcu->next;
|
|
|
|
}
|
2011-08-05 17:19:31 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
if (!(fcu)) return;*/
|
|
|
|
|
2012-05-05 16:03:57 +00:00
|
|
|
bPoseChannel *pchan = BKE_pose_channel_find_name(ob_arm->pose, bone->name);
|
2011-09-04 00:15:59 +00:00
|
|
|
if (!pchan)
|
|
|
|
return;
|
2011-08-05 17:19:31 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
//every inserted keyframe of bones.
|
2011-09-04 00:15:59 +00:00
|
|
|
find_frames(ob_arm, fra);
|
2011-08-07 19:22:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (flag & ARM_RESTPOS) {
|
|
|
|
arm->flag &= ~ARM_RESTPOS;
|
2012-05-05 16:03:57 +00:00
|
|
|
BKE_pose_where_is(scene, ob_arm);
|
2011-08-03 19:12:18 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (fra.size()) {
|
2012-06-12 22:05:33 +00:00
|
|
|
dae_baked_animation(fra, ob_arm, bone);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (flag & ARM_RESTPOS)
|
|
|
|
arm->flag = flag;
|
2012-05-05 16:03:57 +00:00
|
|
|
BKE_pose_where_is(scene, ob_arm);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2012-04-29 15:47:02 +00:00
|
|
|
void AnimationExporter::dae_baked_animation(std::vector<float> &fra, Object *ob_arm, Bone *bone)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
std::string ob_name = id_name(ob_arm);
|
|
|
|
std::string bone_name = bone->name;
|
|
|
|
char anim_id[200];
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (!fra.size())
|
|
|
|
return;
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
BLI_snprintf(anim_id, sizeof(anim_id), "%s_%s_%s", (char *)translate_id(ob_name).c_str(),
|
|
|
|
(char *)translate_id(bone_name).c_str(), "pose_matrix");
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
openAnimation(anim_id, COLLADABU::Utils::EMPTY_STRING);
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create input source
|
|
|
|
std::string input_id = create_source_from_vector(COLLADASW::InputSemantic::INPUT, fra, false, anim_id, "");
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create output source
|
|
|
|
std::string output_id;
|
2013-01-21 13:45:49 +00:00
|
|
|
|
|
|
|
output_id = create_4x4_source(fra, ob_arm, bone, anim_id);
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create interpolations source
|
|
|
|
std::string interpolation_id = fake_interpolation_source(fra.size(), anim_id, "");
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::string sampler_id = std::string(anim_id) + SAMPLER_ID_SUFFIX;
|
|
|
|
COLLADASW::LibraryAnimations::Sampler sampler(sw, sampler_id);
|
|
|
|
std::string empty;
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INPUT, COLLADABU::URI(empty, input_id));
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::OUTPUT, COLLADABU::URI(empty, output_id));
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// TODO create in/out tangents source
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// this input is required
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INTERPOLATION, COLLADABU::URI(empty, interpolation_id));
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
addSampler(sampler);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::string target = translate_id(bone_name) + "/transform";
|
|
|
|
addChannel(COLLADABU::URI(empty, sampler_id), target);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
closeAnimation();
|
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
void AnimationExporter::dae_baked_object_animation(std::vector<float> &fra, Object *ob)
|
|
|
|
{
|
|
|
|
std::string ob_name = id_name(ob);
|
|
|
|
char anim_id[200];
|
|
|
|
|
|
|
|
if (!fra.size())
|
|
|
|
return;
|
|
|
|
|
|
|
|
BLI_snprintf(anim_id, sizeof(anim_id), "%s_%s", (char*)translate_id(ob_name).c_str(),
|
|
|
|
"object_matrix");
|
|
|
|
|
|
|
|
openAnimation(anim_id, COLLADABU::Utils::EMPTY_STRING);
|
|
|
|
|
|
|
|
// create input source
|
|
|
|
std::string input_id = create_source_from_vector(COLLADASW::InputSemantic::INPUT, fra, false, anim_id, "");
|
|
|
|
|
|
|
|
// create output source
|
|
|
|
std::string output_id;
|
|
|
|
output_id = create_4x4_source( fra, ob, NULL, anim_id);
|
|
|
|
|
|
|
|
// create interpolations source
|
|
|
|
std::string interpolation_id = fake_interpolation_source(fra.size(), anim_id, "");
|
|
|
|
|
|
|
|
std::string sampler_id = std::string(anim_id) + SAMPLER_ID_SUFFIX;
|
|
|
|
COLLADASW::LibraryAnimations::Sampler sampler(sw, sampler_id);
|
|
|
|
std::string empty;
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INPUT, COLLADABU::URI(empty, input_id));
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::OUTPUT, COLLADABU::URI(empty, output_id));
|
|
|
|
|
|
|
|
// TODO create in/out tangents source
|
|
|
|
|
|
|
|
// this input is required
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INTERPOLATION, COLLADABU::URI(empty, interpolation_id));
|
|
|
|
|
|
|
|
addSampler(sampler);
|
|
|
|
|
|
|
|
std::string target = translate_id(ob_name) + "/transform";
|
|
|
|
addChannel(COLLADABU::URI(empty, sampler_id), target);
|
|
|
|
|
|
|
|
closeAnimation();
|
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// dae_bone_animation -> add_bone_animation
|
|
|
|
// (blend this into dae_bone_animation)
|
|
|
|
void AnimationExporter::dae_bone_animation(std::vector<float> &fra, float *values, int tm_type, int axis, std::string ob_name, std::string bone_name)
|
|
|
|
{
|
|
|
|
const char *axis_names[] = {"X", "Y", "Z"};
|
|
|
|
const char *axis_name = NULL;
|
|
|
|
char anim_id[200];
|
|
|
|
bool is_rot = tm_type == 0;
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (!fra.size())
|
|
|
|
return;
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
char rna_path[200];
|
|
|
|
BLI_snprintf(rna_path, sizeof(rna_path), "pose.bones[\"%s\"].%s", bone_name.c_str(),
|
2012-06-12 22:05:33 +00:00
|
|
|
tm_type == 0 ? "rotation_quaternion" : (tm_type == 1 ? "scale" : "location"));
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (axis > -1)
|
|
|
|
axis_name = axis_names[axis];
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::string transform_sid = get_transform_sid(NULL, tm_type, axis_name, false);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
BLI_snprintf(anim_id, sizeof(anim_id), "%s_%s_%s", (char *)translate_id(ob_name).c_str(),
|
|
|
|
(char *)translate_id(bone_name).c_str(), (char *)transform_sid.c_str());
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
openAnimation(anim_id, COLLADABU::Utils::EMPTY_STRING);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create input source
|
|
|
|
std::string input_id = create_source_from_vector(COLLADASW::InputSemantic::INPUT, fra, is_rot, anim_id, axis_name);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create output source
|
|
|
|
std::string output_id;
|
|
|
|
if (axis == -1)
|
|
|
|
output_id = create_xyz_source(values, fra.size(), anim_id);
|
|
|
|
else
|
|
|
|
output_id = create_source_from_array(COLLADASW::InputSemantic::OUTPUT, values, fra.size(), is_rot, anim_id, axis_name);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// create interpolations source
|
|
|
|
std::string interpolation_id = fake_interpolation_source(fra.size(), anim_id, axis_name);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::string sampler_id = std::string(anim_id) + SAMPLER_ID_SUFFIX;
|
|
|
|
COLLADASW::LibraryAnimations::Sampler sampler(sw, sampler_id);
|
|
|
|
std::string empty;
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INPUT, COLLADABU::URI(empty, input_id));
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::OUTPUT, COLLADABU::URI(empty, output_id));
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// TODO create in/out tangents source
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// this input is required
|
|
|
|
sampler.addInput(COLLADASW::InputSemantic::INTERPOLATION, COLLADABU::URI(empty, interpolation_id));
|
|
|
|
|
|
|
|
addSampler(sampler);
|
|
|
|
|
|
|
|
std::string target = translate_id(ob_name + "_" + bone_name) + "/" + transform_sid;
|
|
|
|
addChannel(COLLADABU::URI(empty, sampler_id), target);
|
|
|
|
|
|
|
|
closeAnimation();
|
|
|
|
}
|
|
|
|
|
|
|
|
float AnimationExporter::convert_time(float frame)
|
|
|
|
{
|
|
|
|
return FRA2TIME(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
float AnimationExporter::convert_angle(float angle)
|
|
|
|
{
|
|
|
|
return COLLADABU::Math::Utils::radToDegF(angle);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string AnimationExporter::get_semantic_suffix(COLLADASW::InputSemantic::Semantics semantic)
|
|
|
|
{
|
2012-04-28 06:31:57 +00:00
|
|
|
switch (semantic) {
|
2011-05-26 17:07:38 +00:00
|
|
|
case COLLADASW::InputSemantic::INPUT:
|
|
|
|
return INPUT_SOURCE_ID_SUFFIX;
|
|
|
|
case COLLADASW::InputSemantic::OUTPUT:
|
|
|
|
return OUTPUT_SOURCE_ID_SUFFIX;
|
|
|
|
case COLLADASW::InputSemantic::INTERPOLATION:
|
|
|
|
return INTERPOLATION_SOURCE_ID_SUFFIX;
|
|
|
|
case COLLADASW::InputSemantic::IN_TANGENT:
|
|
|
|
return INTANGENT_SOURCE_ID_SUFFIX;
|
|
|
|
case COLLADASW::InputSemantic::OUT_TANGENT:
|
|
|
|
return OUTTANGENT_SOURCE_ID_SUFFIX;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
return "";
|
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
void AnimationExporter::add_source_parameters(COLLADASW::SourceBase::ParameterNameList& param,
|
2012-06-12 22:05:33 +00:00
|
|
|
COLLADASW::InputSemantic::Semantics semantic, bool is_rot, const char *axis, bool transform)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
2012-04-28 06:31:57 +00:00
|
|
|
switch (semantic) {
|
2011-05-26 17:07:38 +00:00
|
|
|
case COLLADASW::InputSemantic::INPUT:
|
|
|
|
param.push_back("TIME");
|
|
|
|
break;
|
|
|
|
case COLLADASW::InputSemantic::OUTPUT:
|
|
|
|
if (is_rot) {
|
|
|
|
param.push_back("ANGLE");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (axis) {
|
|
|
|
param.push_back(axis);
|
|
|
|
}
|
2011-08-05 18:32:39 +00:00
|
|
|
else
|
2012-06-12 22:05:33 +00:00
|
|
|
if (transform) {
|
|
|
|
param.push_back("TRANSFORM");
|
|
|
|
}
|
|
|
|
else { //assumes if axis isn't specified all axises are added
|
|
|
|
param.push_back("X");
|
|
|
|
param.push_back("Y");
|
|
|
|
param.push_back("Z");
|
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case COLLADASW::InputSemantic::IN_TANGENT:
|
|
|
|
case COLLADASW::InputSemantic::OUT_TANGENT:
|
|
|
|
param.push_back("X");
|
|
|
|
param.push_back("Y");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2012-08-05 21:35:09 +00:00
|
|
|
void AnimationExporter::get_source_values(BezTriple *bezt, COLLADASW::InputSemantic::Semantics semantic, bool is_rotation, float *values, int *length)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
switch (semantic) {
|
2011-05-26 17:07:38 +00:00
|
|
|
case COLLADASW::InputSemantic::INPUT:
|
|
|
|
*length = 1;
|
|
|
|
values[0] = convert_time(bezt->vec[1][0]);
|
|
|
|
break;
|
|
|
|
case COLLADASW::InputSemantic::OUTPUT:
|
|
|
|
*length = 1;
|
2012-08-05 21:35:09 +00:00
|
|
|
if (is_rotation) {
|
2011-09-17 09:43:51 +00:00
|
|
|
values[0] = RAD2DEGF(bezt->vec[1][1]);
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
2011-06-16 19:25:21 +00:00
|
|
|
else {
|
2011-05-26 17:07:38 +00:00
|
|
|
values[0] = bezt->vec[1][1];
|
2011-06-16 19:25:21 +00:00
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-05-26 17:07:38 +00:00
|
|
|
case COLLADASW::InputSemantic::IN_TANGENT:
|
2011-09-04 00:15:59 +00:00
|
|
|
*length = 2;
|
2011-05-29 19:27:30 +00:00
|
|
|
values[0] = convert_time(bezt->vec[0][0]);
|
|
|
|
if (bezt->ipo != BEZT_IPO_BEZ) {
|
|
|
|
// We're in a mixed interpolation scenario, set zero as it's irrelevant but value might contain unused data
|
2012-10-21 05:46:41 +00:00
|
|
|
values[0] = 0;
|
|
|
|
values[1] = 0;
|
2011-08-17 18:28:01 +00:00
|
|
|
}
|
2012-08-05 21:35:09 +00:00
|
|
|
else if (is_rotation) {
|
2011-09-17 09:43:51 +00:00
|
|
|
values[1] = RAD2DEGF(bezt->vec[0][1]);
|
2012-03-24 06:38:07 +00:00
|
|
|
}
|
|
|
|
else {
|
2011-05-29 19:27:30 +00:00
|
|
|
values[1] = bezt->vec[0][1];
|
2011-06-16 19:25:21 +00:00
|
|
|
}
|
2011-05-29 19:27:30 +00:00
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-05-26 17:07:38 +00:00
|
|
|
case COLLADASW::InputSemantic::OUT_TANGENT:
|
|
|
|
*length = 2;
|
2011-05-29 19:27:30 +00:00
|
|
|
values[0] = convert_time(bezt->vec[2][0]);
|
|
|
|
if (bezt->ipo != BEZT_IPO_BEZ) {
|
|
|
|
// We're in a mixed interpolation scenario, set zero as it's irrelevant but value might contain unused data
|
2012-10-21 05:46:41 +00:00
|
|
|
values[0] = 0;
|
|
|
|
values[1] = 0;
|
2011-06-14 20:42:01 +00:00
|
|
|
}
|
2012-08-05 21:35:09 +00:00
|
|
|
else if (is_rotation) {
|
2011-09-17 09:43:51 +00:00
|
|
|
values[1] = RAD2DEGF(bezt->vec[2][1]);
|
2012-03-24 06:38:07 +00:00
|
|
|
}
|
|
|
|
else {
|
2011-05-29 19:27:30 +00:00
|
|
|
values[1] = bezt->vec[2][1];
|
2011-06-16 19:25:21 +00:00
|
|
|
}
|
2011-05-29 19:27:30 +00:00
|
|
|
break;
|
2011-05-26 17:07:38 +00:00
|
|
|
default:
|
|
|
|
*length = 0;
|
|
|
|
break;
|
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::string AnimationExporter::create_source_from_fcurve(COLLADASW::InputSemantic::Semantics semantic, FCurve *fcu, const std::string& anim_id, const char *axis_name)
|
|
|
|
{
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(semantic);
|
|
|
|
|
|
|
|
//bool is_rotation = !strcmp(fcu->rna_path, "rotation");
|
|
|
|
bool is_angle = false;
|
|
|
|
|
|
|
|
if (strstr(fcu->rna_path, "rotation")) is_angle = true;
|
|
|
|
|
|
|
|
COLLADASW::FloatSourceF source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(fcu->totvert);
|
|
|
|
|
|
|
|
switch (semantic) {
|
2011-05-29 19:27:30 +00:00
|
|
|
case COLLADASW::InputSemantic::INPUT:
|
|
|
|
case COLLADASW::InputSemantic::OUTPUT:
|
2012-10-21 05:46:41 +00:00
|
|
|
source.setAccessorStride(1);
|
2011-05-29 19:27:30 +00:00
|
|
|
break;
|
|
|
|
case COLLADASW::InputSemantic::IN_TANGENT:
|
|
|
|
case COLLADASW::InputSemantic::OUT_TANGENT:
|
2012-10-21 05:46:41 +00:00
|
|
|
source.setAccessorStride(2);
|
2011-05-29 19:27:30 +00:00
|
|
|
break;
|
2011-09-05 19:27:21 +00:00
|
|
|
default:
|
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
add_source_parameters(param, semantic, is_angle, axis_name, false);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.prepareToAppendValues();
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
for (unsigned int i = 0; i < fcu->totvert; i++) {
|
|
|
|
float values[3]; // be careful!
|
|
|
|
int length = 0;
|
|
|
|
get_source_values(&fcu->bezt[i], semantic, is_angle, values, &length);
|
|
|
|
for (int j = 0; j < length; j++)
|
|
|
|
source.appendValues(values[j]);
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
2011-06-14 20:42:01 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.finish();
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return source_id;
|
|
|
|
}
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2012-08-05 21:35:09 +00:00
|
|
|
/*
|
|
|
|
* Similar to create_source_from_fcurve, but adds conversion of lens
|
|
|
|
* animation data from focal length to FOV.
|
|
|
|
*/
|
|
|
|
std::string AnimationExporter::create_lens_source_from_fcurve(Camera *cam, COLLADASW::InputSemantic::Semantics semantic, FCurve *fcu, const std::string& anim_id)
|
|
|
|
{
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(semantic);
|
|
|
|
|
|
|
|
COLLADASW::FloatSourceF source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(fcu->totvert);
|
|
|
|
|
|
|
|
source.setAccessorStride(1);
|
|
|
|
|
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
add_source_parameters(param, semantic, false, "", false);
|
|
|
|
|
|
|
|
source.prepareToAppendValues();
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < fcu->totvert; i++) {
|
|
|
|
float values[3]; // be careful!
|
|
|
|
int length = 0;
|
|
|
|
get_source_values(&fcu->bezt[i], semantic, false, values, &length);
|
|
|
|
for (int j = 0; j < length; j++)
|
|
|
|
{
|
|
|
|
float val = RAD2DEGF(focallength_to_fov(values[j], cam->sensor_x));
|
|
|
|
source.appendValues(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
source.finish();
|
|
|
|
|
|
|
|
return source_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//Currently called only to get OUTPUT source values ( if rotation and hence the axis is also specified )
|
|
|
|
std::string AnimationExporter::create_source_from_array(COLLADASW::InputSemantic::Semantics semantic, float *v, int tot, bool is_rot, const std::string& anim_id, const char *axis_name)
|
|
|
|
{
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(semantic);
|
|
|
|
|
|
|
|
COLLADASW::FloatSourceF source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(tot);
|
|
|
|
source.setAccessorStride(1);
|
|
|
|
|
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
add_source_parameters(param, semantic, is_rot, axis_name, false);
|
|
|
|
|
|
|
|
source.prepareToAppendValues();
|
|
|
|
|
|
|
|
for (int i = 0; i < tot; i++) {
|
|
|
|
float val = v[i];
|
|
|
|
////if (semantic == COLLADASW::InputSemantic::INPUT)
|
|
|
|
// val = convert_time(val);
|
|
|
|
//else
|
|
|
|
if (is_rot)
|
2011-09-17 09:43:51 +00:00
|
|
|
val = RAD2DEGF(val);
|
2011-09-04 00:15:59 +00:00
|
|
|
source.appendValues(val);
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.finish();
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return source_id;
|
|
|
|
}
|
|
|
|
// only used for sources with INPUT semantic
|
|
|
|
std::string AnimationExporter::create_source_from_vector(COLLADASW::InputSemantic::Semantics semantic, std::vector<float> &fra, bool is_rot, const std::string& anim_id, const char *axis_name)
|
|
|
|
{
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(semantic);
|
|
|
|
|
|
|
|
COLLADASW::FloatSourceF source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(fra.size());
|
|
|
|
source.setAccessorStride(1);
|
|
|
|
|
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
add_source_parameters(param, semantic, is_rot, axis_name, false);
|
|
|
|
|
|
|
|
source.prepareToAppendValues();
|
|
|
|
|
|
|
|
std::vector<float>::iterator it;
|
|
|
|
for (it = fra.begin(); it != fra.end(); it++) {
|
|
|
|
float val = *it;
|
|
|
|
//if (semantic == COLLADASW::InputSemantic::INPUT)
|
|
|
|
val = convert_time(val);
|
|
|
|
/*else if (is_rot)
|
2012-06-12 22:05:33 +00:00
|
|
|
val = convert_angle(val);*/
|
2011-09-04 00:15:59 +00:00
|
|
|
source.appendValues(val);
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.finish();
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return source_id;
|
|
|
|
}
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
|
|
|
|
std::string AnimationExporter::create_4x4_source(std::vector<float> &frames, Object * ob, Bone *bone, const std::string& anim_id)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
COLLADASW::InputSemantic::Semantics semantic = COLLADASW::InputSemantic::OUTPUT;
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(semantic);
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
COLLADASW::Float4x4Source source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(frames.size());
|
|
|
|
source.setAccessorStride(16);
|
2011-08-17 18:28:01 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
add_source_parameters(param, semantic, false, NULL, true);
|
2011-08-17 18:28:01 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.prepareToAppendValues();
|
2013-01-21 13:45:49 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
bPoseChannel *parchan = NULL;
|
|
|
|
bPoseChannel *pchan = NULL;
|
2013-02-01 15:17:51 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
if (ob->type == OB_ARMATURE ){
|
|
|
|
bPose *pose = ob->pose;
|
|
|
|
pchan = BKE_pose_channel_find_name(pose, bone->name);
|
|
|
|
if (!pchan)
|
|
|
|
return "";
|
2011-08-17 18:28:01 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
parchan = pchan->parent;
|
2011-08-17 18:28:01 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
enable_fcurves(ob->adt->action, bone->name);
|
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::vector<float>::iterator it;
|
|
|
|
int j = 0;
|
|
|
|
for (it = frames.begin(); it != frames.end(); it++) {
|
|
|
|
float mat[4][4], ipar[4][4];
|
2013-01-21 13:45:49 +00:00
|
|
|
|
2012-05-05 14:33:36 +00:00
|
|
|
float ctime = BKE_scene_frame_get_from_ctime(scene, *it);
|
2013-01-21 13:45:49 +00:00
|
|
|
CFRA = BKE_scene_frame_get_from_ctime(scene, *it);
|
|
|
|
//BKE_scene_update_for_newframe(G.main,scene,scene->lay);
|
|
|
|
BKE_animsys_evaluate_animdata(scene, &ob->id, ob->adt, ctime, ADT_RECALC_ALL);
|
|
|
|
|
|
|
|
if (bone){
|
|
|
|
if( pchan->flag & POSE_CHAIN)
|
|
|
|
{
|
|
|
|
enable_fcurves(ob->adt->action, NULL);
|
|
|
|
BKE_animsys_evaluate_animdata(scene, &ob->id, ob->adt, ctime, ADT_RECALC_ALL);
|
|
|
|
BKE_pose_where_is(scene, ob);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
BKE_pose_where_is_bone(scene, ob, pchan, ctime, 1);
|
|
|
|
|
|
|
|
// compute bone local mat
|
|
|
|
if (bone->parent) {
|
|
|
|
invert_m4_m4(ipar, parchan->pose_mat);
|
|
|
|
mult_m4_m4m4(mat, ipar, pchan->pose_mat);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
copy_m4_m4(mat, pchan->pose_mat);
|
|
|
|
|
2012-02-05 16:19:28 +00:00
|
|
|
// SECOND_LIFE_COMPATIBILITY
|
|
|
|
// AFAIK animation to second life is via BVH, but no
|
|
|
|
// reason to not have the collada-animation be correct
|
2013-01-21 13:45:49 +00:00
|
|
|
if (export_settings->second_life) {
|
|
|
|
float temp[4][4];
|
|
|
|
copy_m4_m4(temp, bone->arm_mat);
|
|
|
|
temp[3][0] = temp[3][1] = temp[3][2] = 0.0f;
|
|
|
|
invert_m4(temp);
|
2012-02-05 16:19:28 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
mult_m4_m4m4(mat, mat, temp);
|
2012-02-05 16:19:28 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
if (bone->parent) {
|
|
|
|
copy_m4_m4(temp, bone->parent->arm_mat);
|
|
|
|
temp[3][0] = temp[3][1] = temp[3][2] = 0.0f;
|
2012-02-05 16:19:28 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
mult_m4_m4m4(mat, temp, mat);
|
|
|
|
}
|
2012-02-05 16:19:28 +00:00
|
|
|
}
|
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
calc_ob_mat_at_time(ob, ctime, mat);
|
|
|
|
}
|
|
|
|
|
|
|
|
UnitConverter converter;
|
2011-08-17 18:28:01 +00:00
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
double outmat[4][4];
|
|
|
|
converter.mat4_to_dae_double(outmat, mat);
|
2011-08-05 18:32:39 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.appendValues(outmat);
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
j++;
|
2013-01-21 13:45:49 +00:00
|
|
|
|
|
|
|
BIK_release_tree(scene, ob, ctime);
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
enable_fcurves(ob->adt->action, NULL);
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.finish();
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return source_id;
|
|
|
|
}
|
2013-01-21 13:45:49 +00:00
|
|
|
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// only used for sources with OUTPUT semantic ( locations and scale)
|
|
|
|
std::string AnimationExporter::create_xyz_source(float *v, int tot, const std::string& anim_id)
|
|
|
|
{
|
|
|
|
COLLADASW::InputSemantic::Semantics semantic = COLLADASW::InputSemantic::OUTPUT;
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(semantic);
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
COLLADASW::FloatSourceF source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(tot);
|
|
|
|
source.setAccessorStride(3);
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
add_source_parameters(param, semantic, false, NULL, false);
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.prepareToAppendValues();
|
2011-05-29 19:27:30 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
for (int i = 0; i < tot; i++) {
|
|
|
|
source.appendValues(*v, *(v + 1), *(v + 2));
|
|
|
|
v += 3;
|
2011-05-29 19:27:30 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.finish();
|
|
|
|
|
|
|
|
return source_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string AnimationExporter::create_interpolation_source(FCurve *fcu, const std::string& anim_id, const char *axis_name, bool *has_tangents)
|
|
|
|
{
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(COLLADASW::InputSemantic::INTERPOLATION);
|
|
|
|
|
|
|
|
COLLADASW::NameSource source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(fcu->totvert);
|
|
|
|
source.setAccessorStride(1);
|
|
|
|
|
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
param.push_back("INTERPOLATION");
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.prepareToAppendValues();
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
*has_tangents = false;
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
for (unsigned int i = 0; i < fcu->totvert; i++) {
|
2012-06-12 22:05:33 +00:00
|
|
|
if (fcu->bezt[i].ipo == BEZT_IPO_BEZ) {
|
2011-09-04 00:15:59 +00:00
|
|
|
source.appendValues(BEZIER_NAME);
|
|
|
|
*has_tangents = true;
|
2012-03-24 06:38:07 +00:00
|
|
|
}
|
2012-06-12 22:05:33 +00:00
|
|
|
else if (fcu->bezt[i].ipo == BEZT_IPO_CONST) {
|
2011-09-04 00:15:59 +00:00
|
|
|
source.appendValues(STEP_NAME);
|
2012-03-24 06:38:07 +00:00
|
|
|
}
|
|
|
|
else { // BEZT_IPO_LIN
|
2011-05-26 17:07:38 +00:00
|
|
|
source.appendValues(LINEAR_NAME);
|
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
|
|
|
// unsupported? -- HERMITE, CARDINAL, BSPLINE, NURBS
|
|
|
|
|
|
|
|
source.finish();
|
|
|
|
|
|
|
|
return source_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string AnimationExporter::fake_interpolation_source(int tot, const std::string& anim_id, const char *axis_name)
|
|
|
|
{
|
|
|
|
std::string source_id = anim_id + get_semantic_suffix(COLLADASW::InputSemantic::INTERPOLATION);
|
|
|
|
|
|
|
|
COLLADASW::NameSource source(mSW);
|
|
|
|
source.setId(source_id);
|
|
|
|
source.setArrayId(source_id + ARRAY_ID_SUFFIX);
|
|
|
|
source.setAccessorCount(tot);
|
|
|
|
source.setAccessorStride(1);
|
|
|
|
|
|
|
|
COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList();
|
|
|
|
param.push_back("INTERPOLATION");
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.prepareToAppendValues();
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
for (int i = 0; i < tot; i++) {
|
|
|
|
source.appendValues(LINEAR_NAME);
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
source.finish();
|
2011-08-13 16:21:41 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return source_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string AnimationExporter::get_light_param_sid(char *rna_path, int tm_type, const char *axis_name, bool append_axis)
|
|
|
|
{
|
|
|
|
std::string tm_name;
|
|
|
|
// when given rna_path, determine tm_type from it
|
|
|
|
if (rna_path) {
|
|
|
|
char *name = extract_transform_name(rna_path);
|
|
|
|
|
|
|
|
if (!strcmp(name, "color"))
|
|
|
|
tm_type = 1;
|
|
|
|
else if (!strcmp(name, "spot_size"))
|
|
|
|
tm_type = 2;
|
|
|
|
else if (!strcmp(name, "spot_blend"))
|
|
|
|
tm_type = 3;
|
|
|
|
else if (!strcmp(name, "distance"))
|
|
|
|
tm_type = 4;
|
|
|
|
else
|
|
|
|
tm_type = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (tm_type) {
|
2011-08-13 16:21:41 +00:00
|
|
|
case 1:
|
|
|
|
tm_name = "color";
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
tm_name = "fall_off_angle";
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
tm_name = "fall_off_exponent";
|
|
|
|
break;
|
|
|
|
case 4:
|
2011-08-14 16:13:35 +00:00
|
|
|
tm_name = "blender/blender_dist";
|
2011-08-13 16:21:41 +00:00
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-08-13 16:21:41 +00:00
|
|
|
default:
|
|
|
|
tm_name = "";
|
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-13 16:21:41 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (tm_name.size()) {
|
2011-09-05 19:27:21 +00:00
|
|
|
if (axis_name[0])
|
2011-09-04 00:15:59 +00:00
|
|
|
return tm_name + "." + std::string(axis_name);
|
|
|
|
else
|
|
|
|
return tm_name;
|
2011-08-13 16:21:41 +00:00
|
|
|
}
|
2011-08-16 16:03:37 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return std::string("");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string AnimationExporter::get_camera_param_sid(char *rna_path, int tm_type, const char *axis_name, bool append_axis)
|
|
|
|
{
|
|
|
|
std::string tm_name;
|
|
|
|
// when given rna_path, determine tm_type from it
|
|
|
|
if (rna_path) {
|
|
|
|
char *name = extract_transform_name(rna_path);
|
|
|
|
|
|
|
|
if (!strcmp(name, "lens"))
|
|
|
|
tm_type = 0;
|
|
|
|
else if (!strcmp(name, "ortho_scale"))
|
|
|
|
tm_type = 1;
|
|
|
|
else if (!strcmp(name, "clip_end"))
|
|
|
|
tm_type = 2;
|
|
|
|
else if (!strcmp(name, "clip_start"))
|
|
|
|
tm_type = 3;
|
|
|
|
|
|
|
|
else
|
|
|
|
tm_type = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (tm_type) {
|
2011-08-16 16:03:37 +00:00
|
|
|
case 0:
|
|
|
|
tm_name = "xfov";
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
tm_name = "xmag";
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
tm_name = "zfar";
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
tm_name = "znear";
|
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-08-16 16:03:37 +00:00
|
|
|
default:
|
|
|
|
tm_name = "";
|
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-16 16:03:37 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (tm_name.size()) {
|
2011-09-05 19:27:21 +00:00
|
|
|
if (axis_name[0])
|
2011-09-04 00:15:59 +00:00
|
|
|
return tm_name + "." + std::string(axis_name);
|
|
|
|
else
|
|
|
|
return tm_name;
|
2011-08-16 16:03:37 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return std::string("");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assign sid of the animated parameter or transform
|
|
|
|
// for rotation, axis name is always appended and the value of append_axis is ignored
|
|
|
|
std::string AnimationExporter::get_transform_sid(char *rna_path, int tm_type, const char *axis_name, bool append_axis)
|
|
|
|
{
|
|
|
|
std::string tm_name;
|
2012-06-12 22:05:33 +00:00
|
|
|
bool is_rotation = false;
|
2011-09-04 00:15:59 +00:00
|
|
|
// when given rna_path, determine tm_type from it
|
|
|
|
if (rna_path) {
|
|
|
|
char *name = extract_transform_name(rna_path);
|
|
|
|
|
|
|
|
if (!strcmp(name, "rotation_euler"))
|
|
|
|
tm_type = 0;
|
|
|
|
else if (!strcmp(name, "rotation_quaternion"))
|
|
|
|
tm_type = 1;
|
|
|
|
else if (!strcmp(name, "scale"))
|
|
|
|
tm_type = 2;
|
|
|
|
else if (!strcmp(name, "location"))
|
|
|
|
tm_type = 3;
|
|
|
|
else if (!strcmp(name, "specular_hardness"))
|
|
|
|
tm_type = 4;
|
|
|
|
else if (!strcmp(name, "specular_color"))
|
|
|
|
tm_type = 5;
|
|
|
|
else if (!strcmp(name, "diffuse_color"))
|
|
|
|
tm_type = 6;
|
|
|
|
else if (!strcmp(name, "alpha"))
|
|
|
|
tm_type = 7;
|
|
|
|
else if (!strcmp(name, "ior"))
|
|
|
|
tm_type = 8;
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
else
|
|
|
|
tm_type = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (tm_type) {
|
2011-05-26 17:07:38 +00:00
|
|
|
case 0:
|
|
|
|
case 1:
|
2011-06-14 20:42:01 +00:00
|
|
|
tm_name = "rotation";
|
|
|
|
is_rotation = true;
|
|
|
|
break;
|
2011-08-17 18:28:01 +00:00
|
|
|
case 2:
|
2011-05-26 17:07:38 +00:00
|
|
|
tm_name = "scale";
|
|
|
|
break;
|
2011-06-05 18:58:22 +00:00
|
|
|
case 3:
|
2011-05-26 17:07:38 +00:00
|
|
|
tm_name = "location";
|
|
|
|
break;
|
2011-07-02 05:05:03 +00:00
|
|
|
case 4:
|
2011-07-18 17:33:03 +00:00
|
|
|
tm_name = "shininess";
|
|
|
|
break;
|
2011-08-16 17:17:13 +00:00
|
|
|
case 5:
|
2011-07-27 16:29:28 +00:00
|
|
|
tm_name = "specular";
|
|
|
|
break;
|
2011-08-16 17:17:13 +00:00
|
|
|
case 6:
|
2011-07-27 18:38:44 +00:00
|
|
|
tm_name = "diffuse";
|
2012-10-21 05:46:41 +00:00
|
|
|
break;
|
2011-08-16 17:17:13 +00:00
|
|
|
case 7:
|
2011-07-28 18:25:23 +00:00
|
|
|
tm_name = "transparency";
|
2012-10-21 05:46:41 +00:00
|
|
|
break;
|
2011-08-16 17:17:13 +00:00
|
|
|
case 8:
|
2011-08-06 06:11:31 +00:00
|
|
|
tm_name = "index_of_refraction";
|
2012-10-21 05:46:41 +00:00
|
|
|
break;
|
2011-09-04 00:15:59 +00:00
|
|
|
|
2011-05-26 17:07:38 +00:00
|
|
|
default:
|
|
|
|
tm_name = "";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (tm_name.size()) {
|
|
|
|
if (is_rotation)
|
|
|
|
return tm_name + std::string(axis_name) + ".ANGLE";
|
|
|
|
else
|
2012-06-12 22:05:33 +00:00
|
|
|
if (axis_name[0])
|
|
|
|
return tm_name + "." + std::string(axis_name);
|
|
|
|
else
|
|
|
|
return tm_name;
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
return std::string("");
|
|
|
|
}
|
2011-08-07 18:15:40 +00:00
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
char *AnimationExporter::extract_transform_name(char *rna_path)
|
2011-09-04 00:15:59 +00:00
|
|
|
{
|
|
|
|
char *dot = strrchr(rna_path, '.');
|
|
|
|
return dot ? (dot + 1) : rna_path;
|
|
|
|
}
|
2011-08-07 18:15:40 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//find keyframes of all the objects animations
|
|
|
|
void AnimationExporter::find_frames(Object *ob, std::vector<float> &fra)
|
|
|
|
{
|
2012-06-12 22:05:33 +00:00
|
|
|
FCurve *fcu = (FCurve *)ob->adt->action->curves.first;
|
2011-08-07 18:15:40 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
for (; fcu; fcu = fcu->next) {
|
2011-05-26 17:07:38 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
for (unsigned int i = 0; i < fcu->totvert; i++) {
|
|
|
|
float f = fcu->bezt[i].vec[1][0];
|
|
|
|
if (std::find(fra.begin(), fra.end(), f) == fra.end())
|
|
|
|
fra.push_back(f);
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
}
|
2011-07-17 17:30:41 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// keep the keys in ascending order
|
|
|
|
std::sort(fra.begin(), fra.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// enable fcurves driving a specific bone, disable all the rest
|
|
|
|
// if bone_name = NULL enable all fcurves
|
|
|
|
void AnimationExporter::enable_fcurves(bAction *act, char *bone_name)
|
|
|
|
{
|
|
|
|
FCurve *fcu;
|
|
|
|
char prefix[200];
|
|
|
|
|
|
|
|
if (bone_name)
|
|
|
|
BLI_snprintf(prefix, sizeof(prefix), "pose.bones[\"%s\"]", bone_name);
|
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
for (fcu = (FCurve *)act->curves.first; fcu; fcu = fcu->next) {
|
2011-09-04 00:15:59 +00:00
|
|
|
if (bone_name) {
|
|
|
|
if (!strncmp(fcu->rna_path, prefix, strlen(prefix)))
|
|
|
|
fcu->flag &= ~FCURVE_DISABLED;
|
|
|
|
else
|
|
|
|
fcu->flag |= FCURVE_DISABLED;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
fcu->flag &= ~FCURVE_DISABLED;
|
2011-05-26 17:07:38 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-16 16:03:37 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
bool AnimationExporter::hasAnimations(Scene *sce)
|
|
|
|
{
|
2012-06-12 23:19:34 +00:00
|
|
|
LinkNode *node;
|
2011-08-16 16:03:37 +00:00
|
|
|
|
2012-06-17 09:58:26 +00:00
|
|
|
for (node=this->export_settings->export_set; node; node=node->next) {
|
2012-06-12 23:19:34 +00:00
|
|
|
Object *ob = (Object *)node->link;
|
2011-09-04 00:15:59 +00:00
|
|
|
|
|
|
|
FCurve *fcu = 0;
|
|
|
|
//Check for object transform animations
|
2012-03-24 07:52:14 +00:00
|
|
|
if (ob->adt && ob->adt->action)
|
2012-06-12 22:05:33 +00:00
|
|
|
fcu = (FCurve *)ob->adt->action->curves.first;
|
2011-09-04 00:15:59 +00:00
|
|
|
//Check for Lamp parameter animations
|
2012-06-12 22:05:33 +00:00
|
|
|
else if ( (ob->type == OB_LAMP) && ((Lamp *)ob->data)->adt && ((Lamp *)ob->data)->adt->action)
|
|
|
|
fcu = (FCurve *)(((Lamp *)ob->data)->adt->action->curves.first);
|
2011-09-04 00:15:59 +00:00
|
|
|
//Check for Camera parameter animations
|
2012-06-12 22:05:33 +00:00
|
|
|
else if ( (ob->type == OB_CAMERA) && ((Camera *)ob->data)->adt && ((Camera *)ob->data)->adt->action)
|
|
|
|
fcu = (FCurve *)(((Camera *)ob->data)->adt->action->curves.first);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
|
|
|
//Check Material Effect parameter animations.
|
2012-04-28 06:31:57 +00:00
|
|
|
for (int a = 0; a < ob->totcol; a++) {
|
2012-06-12 22:05:33 +00:00
|
|
|
Material *ma = give_current_material(ob, a + 1);
|
2011-09-04 00:15:59 +00:00
|
|
|
if (!ma) continue;
|
2012-04-28 06:31:57 +00:00
|
|
|
if (ma->adt && ma->adt->action) {
|
2012-06-12 22:05:33 +00:00
|
|
|
fcu = (FCurve *)ma->adt->action->curves.first;
|
2011-08-16 16:03:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-21 13:45:49 +00:00
|
|
|
//check shape key animation
|
|
|
|
if(!fcu){
|
|
|
|
Key *key = BKE_key_from_object(ob);
|
|
|
|
if(key && key->adt && key->adt->action)
|
|
|
|
fcu = (FCurve *)key->adt->action->curves.first;
|
|
|
|
}
|
2012-06-12 22:05:33 +00:00
|
|
|
if (fcu)
|
2011-09-04 00:15:59 +00:00
|
|
|
return true;
|
2011-08-16 16:03:37 +00:00
|
|
|
}
|
2011-09-04 00:15:59 +00:00
|
|
|
return false;
|
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
//------------------------------- Not used in the new system.--------------------------------------------------------
|
|
|
|
void AnimationExporter::find_rotation_frames(Object *ob, std::vector<float> &fra, const char *prefix, int rotmode)
|
|
|
|
{
|
|
|
|
if (rotmode > 0)
|
|
|
|
find_frames(ob, fra, prefix, "rotation_euler");
|
|
|
|
else if (rotmode == ROT_MODE_QUAT)
|
|
|
|
find_frames(ob, fra, prefix, "rotation_quaternion");
|
|
|
|
/*else if (rotmode == ROT_MODE_AXISANGLE)
|
2012-06-12 22:05:33 +00:00
|
|
|
;*/
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AnimationExporter::find_frames(Object *ob, std::vector<float> &fra, const char *prefix, const char *tm_name)
|
|
|
|
{
|
2012-06-12 22:05:33 +00:00
|
|
|
FCurve *fcu = (FCurve *)ob->adt->action->curves.first;
|
2011-09-04 00:15:59 +00:00
|
|
|
|
|
|
|
for (; fcu; fcu = fcu->next) {
|
|
|
|
if (prefix && strncmp(prefix, fcu->rna_path, strlen(prefix)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
char *name = extract_transform_name(fcu->rna_path);
|
|
|
|
if (!strcmp(name, tm_name)) {
|
|
|
|
for (unsigned int i = 0; i < fcu->totvert; i++) {
|
|
|
|
float f = fcu->bezt[i].vec[1][0];
|
|
|
|
if (std::find(fra.begin(), fra.end(), f) == fra.end())
|
|
|
|
fra.push_back(f);
|
|
|
|
}
|
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// keep the keys in ascending order
|
|
|
|
std::sort(fra.begin(), fra.end());
|
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
void AnimationExporter::write_bone_animation(Object *ob_arm, Bone *bone)
|
|
|
|
{
|
|
|
|
if (!ob_arm->adt)
|
|
|
|
return;
|
|
|
|
|
|
|
|
//write bone animations for 3 transform types
|
|
|
|
//i=0 --> rotations
|
|
|
|
//i=1 --> scale
|
|
|
|
//i=2 --> location
|
|
|
|
for (int i = 0; i < 3; i++)
|
|
|
|
sample_and_write_bone_animation(ob_arm, bone, i);
|
|
|
|
|
2012-06-12 22:05:33 +00:00
|
|
|
for (Bone *child = (Bone *)bone->childbase.first; child; child = child->next)
|
2011-09-04 00:15:59 +00:00
|
|
|
write_bone_animation(ob_arm, child);
|
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
void AnimationExporter::sample_and_write_bone_animation(Object *ob_arm, Bone *bone, int transform_type)
|
|
|
|
{
|
2012-06-12 22:05:33 +00:00
|
|
|
bArmature *arm = (bArmature *)ob_arm->data;
|
2011-09-04 00:15:59 +00:00
|
|
|
int flag = arm->flag;
|
|
|
|
std::vector<float> fra;
|
|
|
|
char prefix[256];
|
|
|
|
|
|
|
|
BLI_snprintf(prefix, sizeof(prefix), "pose.bones[\"%s\"]", bone->name);
|
|
|
|
|
2012-05-05 16:03:57 +00:00
|
|
|
bPoseChannel *pchan = BKE_pose_channel_find_name(ob_arm->pose, bone->name);
|
2011-09-04 00:15:59 +00:00
|
|
|
if (!pchan)
|
|
|
|
return;
|
2012-03-02 16:05:54 +00:00
|
|
|
//Fill frame array with key frame values framed at \param:transform_type
|
2011-09-04 00:15:59 +00:00
|
|
|
switch (transform_type) {
|
2011-08-16 17:17:13 +00:00
|
|
|
case 0:
|
|
|
|
find_rotation_frames(ob_arm, fra, prefix, pchan->rotmode);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
find_frames(ob_arm, fra, prefix, "scale");
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
find_frames(ob_arm, fra, prefix, "location");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return;
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// exit rest position
|
|
|
|
if (flag & ARM_RESTPOS) {
|
|
|
|
arm->flag &= ~ARM_RESTPOS;
|
2012-05-05 16:03:57 +00:00
|
|
|
BKE_pose_where_is(scene, ob_arm);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
|
|
|
//v array will hold all values which will be exported.
|
|
|
|
if (fra.size()) {
|
2012-06-12 22:05:33 +00:00
|
|
|
float *values = (float *)MEM_callocN(sizeof(float) * 3 * fra.size(), "temp. anim frames");
|
2011-09-04 00:15:59 +00:00
|
|
|
sample_animation(values, fra, transform_type, bone, ob_arm, pchan);
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (transform_type == 0) {
|
|
|
|
// write x, y, z curves separately if it is rotation
|
2012-06-12 22:05:33 +00:00
|
|
|
float *axisValues = (float *)MEM_callocN(sizeof(float) * fra.size(), "temp. anim frames");
|
2011-09-04 00:15:59 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
for (unsigned int j = 0; j < fra.size(); j++)
|
|
|
|
axisValues[j] = values[j * 3 + i];
|
|
|
|
|
|
|
|
dae_bone_animation(fra, axisValues, transform_type, i, id_name(ob_arm), bone->name);
|
|
|
|
}
|
|
|
|
MEM_freeN(axisValues);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// write xyz at once if it is location or scale
|
|
|
|
dae_bone_animation(fra, values, transform_type, -1, id_name(ob_arm), bone->name);
|
2011-08-16 17:17:13 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
MEM_freeN(values);
|
2011-08-16 17:17:13 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
// restore restpos
|
|
|
|
if (flag & ARM_RESTPOS)
|
|
|
|
arm->flag = flag;
|
2012-05-05 16:03:57 +00:00
|
|
|
BKE_pose_where_is(scene, ob_arm);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
void AnimationExporter::sample_animation(float *v, std::vector<float> &frames, int type, Bone *bone, Object *ob_arm, bPoseChannel *pchan)
|
|
|
|
{
|
|
|
|
bPoseChannel *parchan = NULL;
|
|
|
|
bPose *pose = ob_arm->pose;
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2012-05-05 16:03:57 +00:00
|
|
|
pchan = BKE_pose_channel_find_name(pose, bone->name);
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
if (!pchan)
|
|
|
|
return;
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
parchan = pchan->parent;
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
enable_fcurves(ob_arm->adt->action, bone->name);
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
std::vector<float>::iterator it;
|
|
|
|
for (it = frames.begin(); it != frames.end(); it++) {
|
|
|
|
float mat[4][4], ipar[4][4];
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2012-05-05 14:33:36 +00:00
|
|
|
float ctime = BKE_scene_frame_get_from_ctime(scene, *it);
|
2011-08-17 18:28:01 +00:00
|
|
|
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2012-04-29 15:47:02 +00:00
|
|
|
BKE_animsys_evaluate_animdata(scene, &ob_arm->id, ob_arm->adt, ctime, ADT_RECALC_ANIM);
|
2012-05-05 16:03:57 +00:00
|
|
|
BKE_pose_where_is_bone(scene, ob_arm, pchan, ctime, 1);
|
2011-09-04 00:15:59 +00:00
|
|
|
|
|
|
|
// compute bone local mat
|
|
|
|
if (bone->parent) {
|
|
|
|
invert_m4_m4(ipar, parchan->pose_mat);
|
Math lib: matrix multiplication order fix for two functions that were
inconsistent with similar functions & math notation:
mul_m4_m4m4(R, B, A) => mult_m4_m4m4(R, A, B)
mul_m3_m3m4(R, B, A) => mult_m3_m3m4(R, A, B)
For branch maintainers, it should be relatively simple to fix things manually,
it's also possible run this script after merging to do automatic replacement:
http://www.pasteall.org/27459/python
2011-12-16 19:53:12 +00:00
|
|
|
mult_m4_m4m4(mat, ipar, pchan->pose_mat);
|
2011-09-04 00:15:59 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
copy_m4_m4(mat, pchan->pose_mat);
|
2011-08-16 17:17:13 +00:00
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
switch (type) {
|
2011-08-16 17:17:13 +00:00
|
|
|
case 0:
|
|
|
|
mat4_to_eul(v, mat);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
mat4_to_size(v, mat);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
copy_v3_v3(v, mat[3]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
v += 3;
|
2011-08-16 17:17:13 +00:00
|
|
|
}
|
|
|
|
|
2011-09-04 00:15:59 +00:00
|
|
|
enable_fcurves(ob_arm->adt->action, NULL);
|
|
|
|
}
|
2013-01-21 13:45:49 +00:00
|
|
|
|
|
|
|
bool AnimationExporter::validateConstraints(bConstraint *con){
|
|
|
|
|
|
|
|
bool valid = true;
|
|
|
|
bConstraintTypeInfo *cti = BKE_constraint_get_typeinfo(con);
|
|
|
|
/* these we can skip completely (invalid constraints...) */
|
|
|
|
if (cti == NULL) valid = false;
|
|
|
|
if (con->flag & (CONSTRAINT_DISABLE | CONSTRAINT_OFF)) valid = false;
|
|
|
|
/* these constraints can't be evaluated anyway */
|
|
|
|
if (cti->evaluate_constraint == NULL) valid = false;
|
|
|
|
/* influence == 0 should be ignored */
|
|
|
|
if (con->enforce == 0.0f) valid = false;
|
|
|
|
|
|
|
|
return valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnimationExporter::calc_ob_mat_at_time(Object *ob, float ctime , float mat[][4]){
|
|
|
|
ListBase *conlist = get_active_constraints(ob);
|
|
|
|
bConstraint *con;
|
|
|
|
for (con = (bConstraint*)conlist->first; con; con = con->next) {
|
|
|
|
ListBase targets = {NULL, NULL};
|
|
|
|
|
|
|
|
bConstraintTypeInfo *cti = BKE_constraint_get_typeinfo(con);
|
|
|
|
|
|
|
|
if (cti && cti->get_constraint_targets) {
|
|
|
|
bConstraintTarget *ct;
|
|
|
|
Object *obtar;
|
|
|
|
cti->get_constraint_targets(con, &targets);
|
|
|
|
for (ct = (bConstraintTarget*)targets.first; ct; ct = ct->next){
|
|
|
|
obtar = ct->tar;
|
|
|
|
BKE_animsys_evaluate_animdata(scene, &obtar->id, obtar->adt, ctime, ADT_RECALC_ANIM);
|
|
|
|
BKE_object_where_is_calc_time(scene, obtar, ctime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
BKE_object_where_is_calc_time(scene, ob, ctime);
|
|
|
|
copy_m4_m4(mat, ob->obmat);
|
|
|
|
}
|
|
|
|
|