| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  | 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); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 17:10:13 +01:00
										 |  |  | bool AnimationExporter::exportAnimations(Scene *sce) | 
					
						
							| 
									
										
										
										
											2011-08-26 15:16:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-06 23:21:52 +00:00
										 |  |  | 	bool has_animations = hasAnimations(sce); | 
					
						
							|  |  |  | 	if (has_animations) { | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 		this->eval_ctx = eval_ctx; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-03-06 23:21:52 +00:00
										 |  |  | 	return has_animations; | 
					
						
							| 
									
										
										
										
											2011-08-26 15:16:27 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2011-05-26 17:07:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | bool AnimationExporter::is_flat_line(std::vector<float> &values, int channel_count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	for (int i = 0; i < values.size(); i += channel_count) { | 
					
						
							|  |  |  | 		for (int j = 0; j < channel_count; j++) { | 
					
						
							|  |  |  | 			if (!bc_in_range(values[j], values[i+j], 0.000001)) | 
					
						
							|  |  |  | 				return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  This function creates a complete LINEAR Collada <Animation> Entry with all needed  | 
					
						
							|  |  |  |  *  <source>, <sampler>, and <channel> entries. | 
					
						
							|  |  |  |  *  This is is used for creating sampled Transformation Animations for either: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *		1-axis animation: | 
					
						
							|  |  |  |  *		    times contains the time points in seconds from within the timeline | 
					
						
							|  |  |  |  *			values contains the data (list of single floats) | 
					
						
							|  |  |  |  *			channel_count = 1 | 
					
						
							|  |  |  |  *			axis_name = ['X' | 'Y' | 'Z'] | 
					
						
							|  |  |  |  *			is_rot indicates if the animation is a rotation | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *		3-axis animation: | 
					
						
							|  |  |  |  *			times contains the time points in seconds from within the timeline | 
					
						
							|  |  |  |  *			values contains the data (list of floats where each 3 entries are one vector) | 
					
						
							|  |  |  |  *			channel_count = 3 | 
					
						
							|  |  |  |  *			axis_name = "" (actually not used) | 
					
						
							|  |  |  |  *			is_rot = false (see xxx below) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	xxx: I tried to create a 3 axis rotation animation  | 
					
						
							|  |  |  |  *		 like for translation or scale. But i could not  | 
					
						
							|  |  |  |  *		 figure out how to setup the channel for this case. | 
					
						
							|  |  |  |  *		 So for now rotations are exported as 3 separate 1-axis collada animations | 
					
						
							|  |  |  |  *		 See export_sampled_animation() further down. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void AnimationExporter::create_sampled_animation(int channel_count, | 
					
						
							|  |  |  | 	std::vector<float> ×, | 
					
						
							|  |  |  | 	std::vector<float> &values, | 
					
						
							|  |  |  | 	std::string ob_name, | 
					
						
							|  |  |  | 	std::string label, | 
					
						
							|  |  |  | 	std::string axis_name, | 
					
						
							|  |  |  | 	bool is_rot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char anim_id[200]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	if (is_flat_line(values, channel_count)) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 	BLI_snprintf(anim_id, sizeof(anim_id), "%s_%s_%s", (char *)translate_id(ob_name).c_str(), label.c_str(), axis_name.c_str()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	openAnimation(anim_id, COLLADABU::Utils::EMPTY_STRING); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create input source */ | 
					
						
							|  |  |  | 	std::string input_id = create_source_from_vector(COLLADASW::InputSemantic::INPUT, times, false, anim_id, ""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create output source */ | 
					
						
							|  |  |  | 	std::string output_id; | 
					
						
							|  |  |  | 	if (channel_count == 1) | 
					
						
							|  |  |  | 		output_id = create_source_from_array(COLLADASW::InputSemantic::OUTPUT, &values[0], values.size(), is_rot, anim_id, axis_name.c_str()); | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	else if (channel_count == 3) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		output_id = create_xyz_source(&values[0], times.size(), anim_id); | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	else if (channel_count == 16) | 
					
						
							|  |  |  | 		output_id = create_4x4_source(times, values, anim_id); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01: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)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* TODO create in/out tangents source (LINEAR) */ | 
					
						
							|  |  |  | 	std::string interpolation_id = fake_interpolation_source(times.size(), anim_id, ""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Create Sampler */ | 
					
						
							|  |  |  | 	sampler.addInput(COLLADASW::InputSemantic::INTERPOLATION, COLLADABU::URI(empty, interpolation_id)); | 
					
						
							|  |  |  | 	addSampler(sampler); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Create channel */ | 
					
						
							|  |  |  | 	std::string target = translate_id(ob_name) + "/" + label + axis_name + ((is_rot) ? ".ANGLE" : ""); | 
					
						
							|  |  |  | 	addChannel(COLLADABU::URI(empty, sampler_id), target); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	closeAnimation(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Export all animation FCurves of an Object. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Note: This uses the keyframes as sample points, | 
					
						
							|  |  |  |  * and exports "baked keyframes" while keeping the tangent infromation | 
					
						
							|  |  |  |  * of the FCurves intact. This works for simple cases, but breaks | 
					
						
							|  |  |  |  * especially when negative scales are involved in the animation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * If it is necessary to conserve the Animation precisely then | 
					
						
							|  |  |  |  * use export_sampled_animation_set() instead. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void AnimationExporter::export_keyframed_animation_set(Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FCurve *fcu = (FCurve *)ob->adt->action->curves.first; | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	if (!fcu) { | 
					
						
							|  |  |  | 		return; /* object has no animation */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	if (this->export_settings->export_transformation_type == BC_TRANSFORMATION_TYPE_MATRIX) { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 		std::vector<float> ctimes; | 
					
						
							|  |  |  | 		find_keyframes(ob, ctimes); | 
					
						
							|  |  |  | 		if (ctimes.size() > 0) | 
					
						
							|  |  |  | 			export_sampled_matrix_animation(ob, ctimes); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		char *transformName; | 
					
						
							|  |  |  | 		while (fcu) { | 
					
						
							|  |  |  | 			//for armature animations as objects
 | 
					
						
							|  |  |  | 			if (ob->type == OB_ARMATURE) | 
					
						
							|  |  |  | 				transformName = fcu->rna_path; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				transformName = extract_transform_name(fcu->rna_path); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 			if ( | 
					
						
							|  |  |  | 				STREQ(transformName, "location") || | 
					
						
							|  |  |  | 				STREQ(transformName, "scale") || | 
					
						
							|  |  |  | 				(STREQ(transformName, "rotation_euler") && ob->rotmode == ROT_MODE_EUL) || | 
					
						
							|  |  |  | 				STREQ(transformName, "rotation_quaternion")) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				create_keyframed_animation(ob, fcu, transformName, false); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			fcu = fcu->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Export the sampled animation of an Object. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Note: This steps over all animation frames (step size is given in export_settings.sample_size) | 
					
						
							|  |  |  |  * and then evaluates the transformation, | 
					
						
							|  |  |  |  * and exports "baked samples" This works always, however currently the interpolation type is set | 
					
						
							|  |  |  |  * to LINEAR for now. (maybe later this can be changed to BEZIER) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Note: If it is necessary to keep the FCurves intact, then use export_keyframed_animation_set() instead. | 
					
						
							|  |  |  |  * However be aware that exporting keyframed animation may modify the animation slightly. | 
					
						
							|  |  |  |  * Also keyframed animation exports tend to break when negative scales are involved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void AnimationExporter::export_sampled_animation_set(Object *ob) | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	std::vector<float>ctimes; | 
					
						
							|  |  |  | 	find_sampleframes(ob, ctimes); | 
					
						
							|  |  |  | 	if (ctimes.size() > 0) { | 
					
						
							|  |  |  | 		if (this->export_settings->export_transformation_type == BC_TRANSFORMATION_TYPE_MATRIX) | 
					
						
							|  |  |  | 			export_sampled_matrix_animation(ob, ctimes); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			export_sampled_transrotloc_animation(ob, ctimes); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AnimationExporter::export_sampled_matrix_animation(Object *ob, std::vector<float> &ctimes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	UnitConverter converter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	std::vector<float> values; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (std::vector<float>::iterator ctime = ctimes.begin(); ctime != ctimes.end(); ++ctime) { | 
					
						
							|  |  |  | 		float fmat[4][4]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 22:49:30 +01:00
										 |  |  | 		bc_update_scene(eval_ctx, scene, *ctime); | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 		BKE_object_matrix_local_get(ob, fmat); | 
					
						
							|  |  |  | 		if (this->export_settings->limit_precision) | 
					
						
							| 
									
										
										
										
											2018-02-27 16:55:18 +01:00
										 |  |  | 			bc_sanitize_mat(fmat, 6); | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		for (int i = 0; i < 4; i++) | 
					
						
							|  |  |  | 			for (int j = 0; j < 4; j++) | 
					
						
							| 
									
										
										
										
											2018-02-27 16:55:18 +01:00
										 |  |  | 				values.push_back(fmat[i][j]); | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	std::string ob_name = id_name(ob); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	create_sampled_animation(16, ctimes, values, ob_name, "transform", "", false); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AnimationExporter::export_sampled_transrotloc_animation(Object *ob, std::vector<float> &ctimes) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	static int LOC   = 0; | 
					
						
							|  |  |  | 	static int EULX  = 1; | 
					
						
							|  |  |  | 	static int EULY  = 2; | 
					
						
							|  |  |  | 	static int EULZ  = 3; | 
					
						
							|  |  |  | 	static int SCALE = 4; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	std::vector<float> baked_curves[5]; | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (std::vector<float>::iterator ctime = ctimes.begin(); ctime != ctimes.end(); ++ctime ) { | 
					
						
							|  |  |  | 		float fmat[4][4]; | 
					
						
							|  |  |  | 		float floc[3]; | 
					
						
							|  |  |  | 		float fquat[4]; | 
					
						
							|  |  |  | 		float fsize[3]; | 
					
						
							|  |  |  | 		float feul[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 22:49:30 +01:00
										 |  |  | 		bc_update_scene(eval_ctx, scene, *ctime); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		BKE_object_matrix_local_get(ob, fmat); | 
					
						
							|  |  |  | 		mat4_decompose(floc, fquat, fsize, fmat); | 
					
						
							|  |  |  | 		quat_to_eul(feul, fquat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		baked_curves[LOC].push_back(floc[0]); | 
					
						
							|  |  |  | 		baked_curves[LOC].push_back(floc[1]); | 
					
						
							|  |  |  | 		baked_curves[LOC].push_back(floc[2]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		baked_curves[EULX].push_back(feul[0]); | 
					
						
							|  |  |  | 		baked_curves[EULY].push_back(feul[1]); | 
					
						
							|  |  |  | 		baked_curves[EULZ].push_back(feul[2]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		baked_curves[SCALE].push_back(fsize[0]); | 
					
						
							|  |  |  | 		baked_curves[SCALE].push_back(fsize[1]); | 
					
						
							|  |  |  | 		baked_curves[SCALE].push_back(fsize[2]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	std::string ob_name = id_name(ob); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	create_sampled_animation(3, ctimes, baked_curves[SCALE], ob_name, "scale",   "", false); | 
					
						
							|  |  |  | 	create_sampled_animation(3, ctimes, baked_curves[LOC],  ob_name, "location", "", false); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Not sure how to export rotation as a 3channel animation, 
 | 
					
						
							|  |  |  | 	 * so separate into 3 single animations for now: | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	create_sampled_animation(1, ctimes, baked_curves[EULX], ob_name, "rotation", "X", true); | 
					
						
							|  |  |  | 	create_sampled_animation(1, ctimes, baked_curves[EULY], ob_name, "rotation", "Y", true); | 
					
						
							|  |  |  | 	create_sampled_animation(1, ctimes, baked_curves[EULZ], ob_name, "rotation", "Z", true); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 23:37:27 +01:00
										 |  |  | 	fprintf(stdout, "Animation Export: Baked %d frames for %s (sampling rate: %d)\n", | 
					
						
							|  |  |  | 		(int)baked_curves[0].size(), | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		ob->id.name, | 
					
						
							|  |  |  | 		this->export_settings->sampling_rate); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-27 10:35:39 +00:00
										 |  |  | 	char *transformName; | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2011-08-16 17:17:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 		if (ob->type == OB_ARMATURE) { | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 			/* Export skeletal animation (if any)*/ | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 			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
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* Armatures can have object animation and skeletal animation*/ | 
					
						
							|  |  |  | 		if (this->export_settings->sampling_rate < 1) { | 
					
						
							|  |  |  | 			export_keyframed_animation_set(ob); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 			export_sampled_animation_set(ob); | 
					
						
							| 
									
										
										
										
											2011-07-06 17:41:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-08-26 15:16:27 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-02-04 00:18:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	export_object_constraint_animation(ob); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 	//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) { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		FCurve *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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 			if ((STREQ(transformName, "color")) || (STREQ(transformName, "spot_size")) || | 
					
						
							|  |  |  | 			    (STREQ(transformName, "spot_blend")) || (STREQ(transformName, "distance"))) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 				create_keyframed_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) { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		FCurve *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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 			if ((STREQ(transformName, "lens")) || | 
					
						
							|  |  |  | 			    (STREQ(transformName, "ortho_scale")) || | 
					
						
							|  |  |  | 			    (STREQ(transformName, "clip_end")) ||  | 
					
						
							|  |  |  | 				(STREQ(transformName, "clip_start"))) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 				create_keyframed_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; */ | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			FCurve *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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 				if ((STREQ(transformName, "specular_hardness")) || (STREQ(transformName, "specular_color")) || | 
					
						
							|  |  |  | 				    (STREQ(transformName, "diffuse_color")) || (STREQ(transformName, "alpha")) || | 
					
						
							|  |  |  | 				    (STREQ(transformName, "ior"))) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 				{ | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 					create_keyframed_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; | 
					
						
							| 
									
										
										
										
											2013-02-04 00:18:09 +00:00
										 |  |  | 	//Takes frames of target animations
 | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 	make_anim_frames_from_targets(ob, fra); | 
					
						
							| 
									
										
										
										
											2013-02-04 00:18:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 	if (fra.size()) | 
					
						
							| 
									
										
										
										
											2013-02-04 00:18:09 +00:00
										 |  |  | 		dae_baked_object_animation(fra, ob); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AnimationExporter::export_morph_animation(Object *ob) | 
					
						
							|  |  |  | {  | 
					
						
							|  |  |  | 	FCurve *fcu; | 
					
						
							|  |  |  | 	char *transformName; | 
					
						
							|  |  |  | 	Key *key = BKE_key_from_object(ob); | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 	if (!key) return; | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 	if (key->adt && key->adt->action) { | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 		fcu = (FCurve *)key->adt->action->curves.first; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		while (fcu) { | 
					
						
							|  |  |  | 			transformName = extract_transform_name(fcu->rna_path); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			create_keyframed_animation(ob, fcu, transformName, true); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			fcu = fcu->next; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | void AnimationExporter::make_anim_frames_from_targets(Object *ob, std::vector<float> &frames ) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-04-05 18:20:27 +02:00
										 |  |  | 	ListBase *conlist = get_active_constraints(ob); | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 	if (conlist == NULL) return; | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 	bConstraint *con; | 
					
						
							| 
									
										
										
										
											2013-03-18 11:44:56 +00:00
										 |  |  | 	for (con = (bConstraint *)conlist->first; con; con = con->next) { | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 		ListBase targets = {NULL, NULL}; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2015-03-30 21:17:07 +11:00
										 |  |  | 		const bConstraintTypeInfo *cti = BKE_constraint_typeinfo_get(con); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 		if (!validateConstraints(con)) continue; | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		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); | 
					
						
							| 
									
										
										
										
											2013-05-06 11:12:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			for (ct = (bConstraintTarget *)targets.first; ct; ct = ct->next) { | 
					
						
							|  |  |  | 				obtar = ct->tar; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (obtar) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 					find_keyframes(obtar, frames); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-05-06 11:12:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (cti->flush_constraint_targets) | 
					
						
							|  |  |  | 				cti->flush_constraint_targets(con, &targets, 1); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		if (STREQ(transformName, "rotation_quaternion") ) { | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 			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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-07 10:26:04 +00:00
										 |  |  | std::string AnimationExporter::getAnimationPathId(const FCurve *fcu) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	std::string rna_path = std::string(fcu->rna_path); | 
					
						
							|  |  |  | 	return translate_id(rna_path); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /* convert f-curves to animation curves and write */ | 
					
						
							|  |  |  | void AnimationExporter::create_keyframed_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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 	Object *obj = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 	if (STREQ(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
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 	else if (STREQ(transformName, "color") || | 
					
						
							|  |  |  | 	         STREQ(transformName, "specular_color") || | 
					
						
							|  |  |  | 	         STREQ(transformName, "diffuse_color") || | 
					
						
							|  |  |  | 	         STREQ(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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Note: Handle transformation animations separately (to apply matrix inverse to fcurves) | 
					
						
							|  |  |  | 	 * We will use the object to evaluate the animation on all keyframes and calculate the  | 
					
						
							|  |  |  | 	 * resulting object matrix. We need this to incorporate the | 
					
						
							|  |  |  | 	 * effects of the parent inverse matrix (when it contains a rotation component) | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * TODO: try to combine exported fcurves into 3 channel animations like done  | 
					
						
							|  |  |  | 	 * in export_sampled_animation(). For now each channel is exported as separate <Animation>. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	else if ( | 
					
						
							|  |  |  | 		STREQ(transformName, "scale") || | 
					
						
							|  |  |  | 		STREQ(transformName, "location") || | 
					
						
							|  |  |  | 		STREQ(transformName, "rotation_euler")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		const char *axis_names[] = {"X", "Y", "Z"}; | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		if (fcu->array_index < 3) { | 
					
						
							| 
									
										
										
										
											2011-06-05 18:58:22 +00:00
										 |  |  | 			axis_name = axis_names[fcu->array_index]; | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			obj = ob; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01: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); | 
					
						
							| 
									
										
										
										
											2013-03-07 10:26:04 +00:00
										 |  |  | 		BLI_snprintf( | 
					
						
							| 
									
										
										
										
											2017-09-27 00:17:28 +10:00
										 |  |  | 		        anim_id, | 
					
						
							|  |  |  | 		        sizeof(anim_id), | 
					
						
							|  |  |  | 		        "%s_%s.%s", | 
					
						
							|  |  |  | 		        (char *)translate_id(ob_name).c_str(), | 
					
						
							|  |  |  | 		        (char *)translate_id(transformName).c_str(), | 
					
						
							|  |  |  | 		        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); | 
					
						
							| 
									
										
										
										
											2013-03-07 10:26:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		BLI_snprintf( | 
					
						
							| 
									
										
										
										
											2017-09-27 00:17:28 +10:00
										 |  |  | 		        anim_id, | 
					
						
							|  |  |  | 		        sizeof(anim_id), | 
					
						
							|  |  |  | 		        "%s_%s_%s", | 
					
						
							|  |  |  | 		        (char *)translate_id(ob_name).c_str(), | 
					
						
							|  |  |  | 		        (char *)getAnimationPathId(fcu).c_str(), | 
					
						
							|  |  |  | 		        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); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 	else if (STREQ(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 { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		output_id = create_source_from_fcurve(COLLADASW::InputSemantic::OUTPUT, fcu, anim_id, axis_name, obj); | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		intangent_id = create_source_from_fcurve(COLLADASW::InputSemantic::IN_TANGENT, fcu, anim_id, axis_name, obj); | 
					
						
							| 
									
										
										
										
											2011-07-02 05:05:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 		// create out_tangent source
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		outtangent_id = create_source_from_fcurve(COLLADASW::InputSemantic::OUT_TANGENT, fcu, anim_id, axis_name, obj); | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (this->export_settings->sampling_rate < 1) | 
					
						
							|  |  |  | 		find_keyframes(ob_arm, fra); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		find_sampleframes(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; | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 		BKE_pose_where_is(eval_ctx, 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; | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 	BKE_pose_where_is(eval_ctx, 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-17 14:16:19 +01:00
										 |  |  | 	std::string target = get_joint_id(ob_arm, bone) + "/transform"; | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 	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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-18 11:44:56 +00:00
										 |  |  | 	BLI_snprintf(anim_id, sizeof(anim_id), "%s_%s", (char *)translate_id(ob_name).c_str(), | 
					
						
							| 
									
										
										
										
											2013-02-04 00:18:09 +00:00
										 |  |  | 	             "object_matrix"); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-31 09:35:00 +01:00
										 |  |  | void AnimationExporter::get_source_values(BezTriple *bezt, COLLADASW::InputSemantic::Semantics semantic, bool is_angle, 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; | 
					
						
							| 
									
										
										
										
											2014-01-31 09:35:00 +01:00
										 |  |  | 			if (is_angle) { | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-01-31 09:35:00 +01:00
										 |  |  | 			else if (is_angle) { | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-01-31 09:35:00 +01:00
										 |  |  | 			else if (is_angle) { | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | // old function to keep compatibility for calls where offset and object are not needed
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 	return create_source_from_fcurve(semantic, fcu, anim_id, axis_name, NULL); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | void AnimationExporter::evaluate_anim_with_constraints(Object *ob, float ctime) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BKE_animsys_evaluate_animdata(scene, &ob->id, ob->adt, ctime, ADT_RECALC_ALL); | 
					
						
							| 
									
										
										
										
											2018-04-05 18:20:27 +02:00
										 |  |  | 	ListBase *conlist = get_active_constraints(ob); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 	bConstraint *con; | 
					
						
							|  |  |  | 	for (con = (bConstraint *)conlist->first; con; con = con->next) { | 
					
						
							|  |  |  | 		ListBase targets = { NULL, NULL }; | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		const bConstraintTypeInfo *cti = BKE_constraint_typeinfo_get(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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (obtar) { | 
					
						
							|  |  |  | 					BKE_animsys_evaluate_animdata(scene, &obtar->id, obtar->adt, ctime, ADT_RECALC_ANIM); | 
					
						
							| 
									
										
										
										
											2018-02-25 00:06:41 +01:00
										 |  |  | 					BKE_object_where_is_calc_time(this->eval_ctx, scene, obtar, ctime); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (cti->flush_constraint_targets) | 
					
						
							|  |  |  | 				cti->flush_constraint_targets(con, &targets, 1); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-02-25 00:06:41 +01:00
										 |  |  | 	BKE_object_where_is_calc_time(this->eval_ctx, scene, ob, ctime); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * ob is needed to aply parent inverse information to fcurve. | 
					
						
							|  |  |  |  * TODO: Here we have to step over all keyframes for each object and for each fcurve. | 
					
						
							|  |  |  |  * Instead of processing each fcurve one by one,  | 
					
						
							|  |  |  |  * step over the animation from keyframe to keyframe,  | 
					
						
							|  |  |  |  * then create adjusted fcurves (and entries) for all affected objects. | 
					
						
							|  |  |  |  * Then we would need to step through the scene only once. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | std::string AnimationExporter::create_source_from_fcurve(COLLADASW::InputSemantic::Semantics semantic, FCurve *fcu, const std::string& anim_id, const char *axis_name, Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	std::string source_id = anim_id + get_semantic_suffix(semantic); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool is_angle = (strstr(fcu->rna_path, "rotation") || strstr(fcu->rna_path, "spot_size")); | 
					
						
							|  |  |  | 	bool is_euler = strstr(fcu->rna_path, "rotation_euler"); | 
					
						
							|  |  |  | 	bool is_translation = strstr(fcu->rna_path, "location"); | 
					
						
							|  |  |  | 	bool is_scale = strstr(fcu->rna_path, "scale"); | 
					
						
							|  |  |  | 	bool is_tangent = false; | 
					
						
							|  |  |  | 	int offset_index = 0; | 
					
						
							| 
									
										
										
										
											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(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); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			offset_index = 0; | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			offset_index = 1; | 
					
						
							|  |  |  | 			is_tangent = true; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 	for (unsigned int frame_index = 0; frame_index < fcu->totvert; frame_index++) { | 
					
						
							|  |  |  | 		float fixed_val = 0; | 
					
						
							|  |  |  | 		if (ob) { | 
					
						
							|  |  |  | 			float fmat[4][4]; | 
					
						
							|  |  |  | 			float frame = fcu->bezt[frame_index].vec[1][0]; | 
					
						
							|  |  |  | 			float ctime = BKE_scene_frame_get_from_ctime(scene, frame); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			evaluate_anim_with_constraints(ob, ctime); // set object transforms to fcurve's i'th keyframe
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			BKE_object_matrix_local_get(ob, fmat); | 
					
						
							|  |  |  | 			float floc[3]; | 
					
						
							|  |  |  | 			float fquat[4]; | 
					
						
							|  |  |  | 			float fsize[3]; | 
					
						
							|  |  |  | 			mat4_decompose(floc, fquat, fsize, fmat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (is_euler) { | 
					
						
							|  |  |  | 				float eul[3]; | 
					
						
							|  |  |  | 				quat_to_eul(eul, fquat); | 
					
						
							|  |  |  | 				fixed_val = RAD2DEGF(eul[fcu->array_index]); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if (is_translation) { | 
					
						
							|  |  |  | 				fixed_val = floc[fcu->array_index]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if (is_scale) { | 
					
						
							|  |  |  | 				fixed_val = fsize[fcu->array_index]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 		float values[3]; // be careful!
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		float offset = 0; | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 		int length = 0; | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		get_source_values(&fcu->bezt[frame_index], semantic, is_angle, values, &length); | 
					
						
							|  |  |  | 		if (is_tangent) { | 
					
						
							|  |  |  | 			float bases[3]; | 
					
						
							|  |  |  | 			int len = 0; | 
					
						
							|  |  |  | 			get_source_values(&fcu->bezt[frame_index], COLLADASW::InputSemantic::OUTPUT, is_angle, bases, &len); | 
					
						
							|  |  |  | 			offset = values[offset_index] - bases[0]; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (int j = 0; j < length; j++) { | 
					
						
							|  |  |  | 			float val; | 
					
						
							|  |  |  | 			if (j == offset_index) { | 
					
						
							|  |  |  | 				if (ob) { | 
					
						
							|  |  |  | 					val = fixed_val + offset; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					val = values[j] + offset; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				val = values[j]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			source.appendValues(val); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * only to get OUTPUT source values ( if rotation and hence the axis is also specified ) | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 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; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * only used for sources with INPUT semantic | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | std::string AnimationExporter::create_4x4_source(std::vector<float> &ctimes, std::vector<float> &values , const std::string &anim_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	COLLADASW::InputSemantic::Semantics semantic = COLLADASW::InputSemantic::OUTPUT; | 
					
						
							|  |  |  | 	std::string source_id = anim_id + get_semantic_suffix(semantic); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	COLLADASW::Float4x4Source source(mSW); | 
					
						
							|  |  |  | 	source.setId(source_id); | 
					
						
							|  |  |  | 	source.setArrayId(source_id + ARRAY_ID_SUFFIX); | 
					
						
							|  |  |  | 	source.setAccessorCount(ctimes.size()); | 
					
						
							|  |  |  | 	source.setAccessorStride(16); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	COLLADASW::SourceBase::ParameterNameList ¶m = source.getParameterNameList(); | 
					
						
							|  |  |  | 	add_source_parameters(param, semantic, false, NULL, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	source.prepareToAppendValues(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	std::vector<float>::iterator it; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (it = values.begin(); it != values.end(); it+=16) { | 
					
						
							|  |  |  | 		float mat[4][4]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		bc_copy_m4_farray(mat, &*it); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		UnitConverter converter; | 
					
						
							|  |  |  | 		double outmat[4][4]; | 
					
						
							|  |  |  | 		converter.mat4_to_dae_double(outmat, mat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (this->export_settings->limit_precision) | 
					
						
							|  |  |  | 			bc_sanitize_mat(outmat, 6); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		source.appendValues(outmat); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	source.finish(); | 
					
						
							|  |  |  | 	return source_id; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-08 04:00:06 +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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	bool is_bone_animation = ob->type == OB_ARMATURE && bone; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-02-04 00:18:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 	bPoseChannel *parchan = NULL; | 
					
						
							|  |  |  | 	bPoseChannel *pchan = NULL; | 
					
						
							| 
									
										
										
										
											2013-02-01 15:17:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 	if (is_bone_animation) { | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 		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]; | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		float frame = *it; | 
					
						
							| 
									
										
										
										
											2013-02-04 00:18:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		float ctime = BKE_scene_frame_get_from_ctime(scene, frame); | 
					
						
							| 
									
										
										
										
											2018-02-26 22:49:30 +01:00
										 |  |  | 		bc_update_scene(eval_ctx, scene, ctime); | 
					
						
							| 
									
										
										
										
											2018-02-26 16:34:45 +01:00
										 |  |  | 		if (is_bone_animation) { | 
					
						
							| 
									
										
										
										
											2018-02-26 22:49:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 			if (pchan->flag & POSE_CHAIN) { | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 				enable_fcurves(ob->adt->action, NULL); | 
					
						
							|  |  |  | 				BKE_animsys_evaluate_animdata(scene, &ob->id, ob->adt, ctime, ADT_RECALC_ALL); | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 				BKE_pose_where_is(eval_ctx, scene, ob); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 				BKE_pose_where_is_bone(eval_ctx, scene, ob, pchan, ctime, 1); | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			// compute bone local mat
 | 
					
						
							|  |  |  | 			if (bone->parent) { | 
					
						
							|  |  |  | 				invert_m4_m4(ipar, parchan->pose_mat); | 
					
						
							| 
									
										
										
										
											2013-05-26 18:36:25 +00:00
										 |  |  | 				mul_m4_m4m4(mat, ipar, pchan->pose_mat); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				copy_m4_m4(mat, pchan->pose_mat); | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			/* OPEN_SIM_COMPATIBILITY
 | 
					
						
							|  |  |  | 			 * AFAIK animation to second life is via BVH, but no | 
					
						
							|  |  |  | 			 * reason to not have the collada-animation be correct | 
					
						
							|  |  |  | 			 */ | 
					
						
							| 
									
										
										
										
											2013-10-03 14:24:54 +00:00
										 |  |  | 			if (export_settings->open_sim) { | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 				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-05-26 18:36:25 +00:00
										 |  |  | 				mul_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-05-26 18:36:25 +00:00
										 |  |  | 					mul_m4_m4m4(mat, temp, mat); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-02-05 16:19:28 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			copy_m4_m4(mat, ob->obmat); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-16 16:53:16 +01:00
										 |  |  | 		if (this->export_settings->limit_precision) | 
					
						
							|  |  |  | 			bc_sanitize_mat(outmat, 6); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-27 18:27:07 +02:00
										 |  |  | 	if (ob->adt) { | 
					
						
							|  |  |  | 		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
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * only used for sources with OUTPUT semantic ( locations and scale) | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		if (STREQ(name, "color")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 1; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "spot_size")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 2; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "spot_blend")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 3; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "distance")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		if (STREQ(name, "lens")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 0; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "ortho_scale")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 1; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "clip_end")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 2; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "clip_start")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			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(""); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Assign sid of the animated parameter or transform for rotation,  | 
					
						
							|  |  |  |  * axis name is always appended and the value of append_axis is ignored | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | std::string AnimationExporter::get_transform_sid(char *rna_path, int tm_type, const char *axis_name, bool append_axis) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	std::string tm_name; | 
					
						
							| 
									
										
										
										
											2014-01-31 09:35:00 +01:00
										 |  |  | 	bool is_angle = 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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		if (STREQ(name, "rotation_euler")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 0; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "rotation_quaternion")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 1; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "scale")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 2; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "location")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 3; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "specular_hardness")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 4; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "specular_color")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 5; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "diffuse_color")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 6; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "alpha")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			tm_type = 7; | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 		else if (STREQ(name, "ior")) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			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"; | 
					
						
							| 
									
										
										
										
											2014-01-31 09:35:00 +01:00
										 |  |  | 			is_angle = true; | 
					
						
							| 
									
										
										
										
											2011-06-14 20:42:01 +00:00
										 |  |  | 			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()) { | 
					
						
							| 
									
										
										
										
											2014-01-31 09:35:00 +01:00
										 |  |  | 		if (is_angle) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 			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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * enable fcurves driving a specific bone, disable all the rest | 
					
						
							|  |  |  |  * if bone_name = NULL enable all fcurves | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 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) { | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 			if (STREQLEN(fcu->rna_path, prefix, strlen(prefix))) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 				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
 | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 		if (!fcu) { | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 			Key *key = BKE_key_from_object(ob); | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | 			if (key && key->adt && key->adt->action) | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 				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) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		find_keyframes(ob, fra, prefix, "rotation_euler"); | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 	else if (rotmode == ROT_MODE_QUAT) | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 		find_keyframes(ob, fra, prefix, "rotation_quaternion"); | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | 	/*else if (rotmode == ROT_MODE_AXISANGLE)
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	   ;*/ | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | /* Take care to always have the first frame and the last frame in the animation
 | 
					
						
							|  |  |  |  * regardless of the sampling_rate setting | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void AnimationExporter::find_sampleframes(Object *ob, std::vector<float> &fra) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int frame = scene->r.sfra; | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		float ctime = BKE_scene_frame_get_from_ctime(scene, frame); | 
					
						
							|  |  |  | 		fra.push_back(ctime); | 
					
						
							|  |  |  | 		if (frame == scene->r.efra) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		frame += this->export_settings->sampling_rate; | 
					
						
							|  |  |  | 		if (frame > scene->r.efra) | 
					
						
							|  |  |  | 			frame = scene->r.efra; // make sure the last frame is always exported
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} while (true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 
 | 
					
						
							|  |  |  |  * find keyframes of all the objects animations | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void AnimationExporter::find_keyframes(Object *ob, std::vector<float> &fra) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (ob->adt && ob->adt->action) { | 
					
						
							|  |  |  | 		FCurve *fcu = (FCurve *)ob->adt->action->curves.first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (; fcu; fcu = fcu->next) { | 
					
						
							|  |  |  | 			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); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// keep the keys in ascending order
 | 
					
						
							|  |  |  | 		std::sort(fra.begin(), fra.end()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AnimationExporter::find_keyframes(Object *ob, std::vector<float> &fra, const char *prefix, const char *tm_name) | 
					
						
							| 
									
										
										
										
											2011-09-04 00:15:59 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-01-26 16:06:47 +01:00
										 |  |  | 	if (ob->adt && ob->adt->action) { | 
					
						
							|  |  |  | 		FCurve *fcu = (FCurve *)ob->adt->action->curves.first; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (; fcu; fcu = fcu->next) { | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 			if (prefix && !STREQLEN(prefix, fcu->rna_path, strlen(prefix))) | 
					
						
							| 
									
										
										
										
											2014-01-26 16:06:47 +01:00
										 |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			char *name = extract_transform_name(fcu->rna_path); | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 			if (STREQ(name, tm_name)) { | 
					
						
							| 
									
										
										
										
											2014-01-26 16:06:47 +01: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-09-04 00:15:59 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-08-16 17:17:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-26 16:06:47 +01:00
										 |  |  | 		// keep the keys in ascending order
 | 
					
						
							|  |  |  | 		std::sort(fra.begin(), fra.end()); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											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::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: | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			find_keyframes(ob_arm, fra, prefix, "scale"); | 
					
						
							| 
									
										
										
										
											2011-08-16 17:17:13 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case 2: | 
					
						
							| 
									
										
										
										
											2018-02-24 13:11:30 +01:00
										 |  |  | 			find_keyframes(ob_arm, fra, prefix, "location"); | 
					
						
							| 
									
										
										
										
											2011-08-16 17:17:13 +00:00
										 |  |  | 			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; | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 		BKE_pose_where_is(eval_ctx, 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; | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 	BKE_pose_where_is(eval_ctx, 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); | 
					
						
							| 
									
										
										
										
											2017-07-21 17:44:11 +02:00
										 |  |  | 		BKE_pose_where_is_bone(eval_ctx, 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); | 
					
						
							| 
									
										
										
										
											2013-05-26 18:36:25 +00:00
										 |  |  | 			mul_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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-10 17:06:05 +00:00
										 |  |  | bool AnimationExporter::validateConstraints(bConstraint *con) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-03-30 21:17:07 +11:00
										 |  |  | 	const bConstraintTypeInfo *cti = BKE_constraint_typeinfo_get(con); | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 	/* these we can skip completely (invalid constraints...) */ | 
					
						
							| 
									
										
										
										
											2018-03-11 20:00:46 +01:00
										 |  |  | 	if (cti == NULL) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	if (con->flag & (CONSTRAINT_DISABLE | CONSTRAINT_OFF)) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 	/* these constraints can't be evaluated anyway */ | 
					
						
							| 
									
										
										
										
											2018-03-11 20:00:46 +01:00
										 |  |  | 	if (cti->evaluate_constraint == NULL) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 	/* influence == 0 should be ignored */ | 
					
						
							| 
									
										
										
										
											2018-03-11 20:00:46 +01:00
										 |  |  | 	if (con->enforce == 0.0f) | 
					
						
							|  |  |  | 		return false; | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-11 20:00:46 +01:00
										 |  |  | 	/* validation passed */ | 
					
						
							|  |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2013-01-21 13:45:49 +00:00
										 |  |  | } |