| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup collada | 
					
						
							| 
									
										
										
										
											2011-02-27 20:30:35 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-27 15:34:55 -04:00
										 |  |  | #if !defined(WIN32)
 | 
					
						
							| 
									
										
										
										
											2020-12-04 11:28:09 +01:00
										 |  |  | #  include <cstdint>
 | 
					
						
							| 
									
										
										
										
											2010-10-06 12:04:56 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-12 06:25:04 +00:00
										 |  |  | /* COLLADABU_ASSERT, may be able to remove later */ | 
					
						
							|  |  |  | #include "COLLADABUPlatform.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BLI_compiler_attrs.h"
 | 
					
						
							| 
									
										
										
										
											2014-09-01 14:33:05 +02:00
										 |  |  | #include "BLI_listbase.h"
 | 
					
						
							|  |  |  | #include "BLI_math.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | #include "DNA_armature_types.h"
 | 
					
						
							|  |  |  | #include "DNA_modifier_types.h"
 | 
					
						
							| 
									
										
										
										
											2013-06-24 13:17:21 +00:00
										 |  |  | #include "DNA_scene_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-18 23:52:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BKE_action.h"
 | 
					
						
							|  |  |  | #include "BKE_object.h"
 | 
					
						
							|  |  |  | #include "BKE_object_deform.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | #include "ED_mesh.h"
 | 
					
						
							|  |  |  | #include "ED_object.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "SkinInfo.h"
 | 
					
						
							|  |  |  | #include "collada_utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  | /* use name, or fall back to original id if name not present (name is optional) */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | template<class T> static const char *bc_get_joint_name(T *node) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const std::string &id = node->getName(); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:59:27 +02:00
										 |  |  |   return id.empty() ? node->getOriginalId().c_str() : id.c_str(); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  | /* This is used to store data passed in write_controller_data.
 | 
					
						
							|  |  |  |  * Arrays from COLLADAFW::SkinControllerData lose ownership, so do this class members | 
					
						
							|  |  |  |  * so that arrays don't get freed until we free them explicitly. */ | 
					
						
							| 
									
										
										
										
											2021-04-08 11:07:12 +02:00
										 |  |  | SkinInfo::SkinInfo() = default; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | SkinInfo::SkinInfo(const SkinInfo &skin) | 
					
						
							|  |  |  |     : weights(skin.weights), | 
					
						
							|  |  |  |       joint_data(skin.joint_data), | 
					
						
							|  |  |  |       unit_converter(skin.unit_converter), | 
					
						
							|  |  |  |       ob_arm(skin.ob_arm), | 
					
						
							|  |  |  |       controller_uid(skin.controller_uid), | 
					
						
							|  |  |  |       parent(skin.parent) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   copy_m4_m4(bind_shape_matrix, (float(*)[4])skin.bind_shape_matrix); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   transfer_uint_array_data_const(skin.joints_per_vertex, joints_per_vertex); | 
					
						
							|  |  |  |   transfer_uint_array_data_const(skin.weight_indices, weight_indices); | 
					
						
							|  |  |  |   transfer_int_array_data_const(skin.joint_indices, joint_indices); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  | SkinInfo::SkinInfo(UnitConverter *conv) : unit_converter(conv), ob_arm(nullptr), parent(nullptr) | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  | /* nobody owns the data after this, so it should be freed manually with releaseMemory */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | template<class T> void SkinInfo::transfer_array_data(T &src, T &dest) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   dest.setData(src.getData(), src.getCount()); | 
					
						
							|  |  |  |   src.yieldOwnerShip(); | 
					
						
							|  |  |  |   dest.yieldOwnerShip(); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  | /* when src is const we cannot src.yieldOwnerShip, this is used by copy constructor */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void SkinInfo::transfer_int_array_data_const(const COLLADAFW::IntValuesArray &src, | 
					
						
							|  |  |  |                                              COLLADAFW::IntValuesArray &dest) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   dest.setData((int *)src.getData(), src.getCount()); | 
					
						
							|  |  |  |   dest.yieldOwnerShip(); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void SkinInfo::transfer_uint_array_data_const(const COLLADAFW::UIntValuesArray &src, | 
					
						
							|  |  |  |                                               COLLADAFW::UIntValuesArray &dest) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   dest.setData((unsigned int *)src.getData(), src.getCount()); | 
					
						
							|  |  |  |   dest.yieldOwnerShip(); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | void SkinInfo::borrow_skin_controller_data(const COLLADAFW::SkinControllerData *skin) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   transfer_array_data((COLLADAFW::UIntValuesArray &)skin->getJointsPerVertex(), joints_per_vertex); | 
					
						
							|  |  |  |   transfer_array_data((COLLADAFW::UIntValuesArray &)skin->getWeightIndices(), weight_indices); | 
					
						
							|  |  |  |   transfer_array_data((COLLADAFW::IntValuesArray &)skin->getJointIndices(), joint_indices); | 
					
						
							|  |  |  |   // transfer_array_data(skin->getWeights(), weights);
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |   /* cannot transfer data for FloatOrDoubleArray, copy values manually */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const COLLADAFW::FloatOrDoubleArray &weight = skin->getWeights(); | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |   for (unsigned int i = 0; i < weight.getValuesCount(); i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     weights.push_back(bc_get_float_value(weight, i)); | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 12:26:24 +02:00
										 |  |  |   UnitConverter::dae_matrix_to_mat4_(bind_shape_matrix, skin->getBindShapeMatrix()); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-06-08 08:07:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | void SkinInfo::free() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   joints_per_vertex.releaseMemory(); | 
					
						
							|  |  |  |   weight_indices.releaseMemory(); | 
					
						
							|  |  |  |   joint_indices.releaseMemory(); | 
					
						
							|  |  |  |   // weights.releaseMemory();
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  | /* using inverse bind matrices to construct armature
 | 
					
						
							|  |  |  |  * it is safe to invert them to get the original matrices | 
					
						
							|  |  |  |  * because if they are inverse matrices, they can be inverted */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void SkinInfo::add_joint(const COLLADABU::Math::Matrix4 &matrix) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   JointData jd; | 
					
						
							| 
									
										
										
										
											2020-09-04 12:26:24 +02:00
										 |  |  |   UnitConverter::dae_matrix_to_mat4_(jd.inv_bind_mat, matrix); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   joint_data.push_back(jd); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | void SkinInfo::set_controller(const COLLADAFW::SkinController *co) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   controller_uid = co->getUniqueId(); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |   /* fill in joint UIDs */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const COLLADAFW::UniqueIdArray &joint_uids = co->getJoints(); | 
					
						
							|  |  |  |   for (unsigned int i = 0; i < joint_uids.getCount(); i++) { | 
					
						
							|  |  |  |     joint_data[i].joint_uid = joint_uids[i]; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |     /* store armature pointer */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     // JointData& jd = joint_index_to_joint_info_map[i];
 | 
					
						
							|  |  |  |     // jd.ob_arm = ob_arm;
 | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |     /* now we'll be able to get inv bind matrix from joint id */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     // joint_id_to_joint_index_map[joint_ids[i]] = i;
 | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  | /* called from write_controller */ | 
					
						
							| 
									
										
										
										
											2018-06-14 15:55:51 +02:00
										 |  |  | Object *SkinInfo::create_armature(Main *bmain, Scene *scene, ViewLayer *view_layer) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |   ob_arm = bc_add_object(bmain, scene, view_layer, OB_ARMATURE, nullptr); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return ob_arm; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | Object *SkinInfo::set_armature(Object *ob_arm) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |   if (this->ob_arm) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return this->ob_arm; | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   this->ob_arm = ob_arm; | 
					
						
							|  |  |  |   return ob_arm; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:29:01 +00:00
										 |  |  | bool SkinInfo::get_joint_inv_bind_matrix(float inv_bind_mat[4][4], COLLADAFW::Node *node) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const COLLADAFW::UniqueId &uid = node->getUniqueId(); | 
					
						
							|  |  |  |   std::vector<JointData>::iterator it; | 
					
						
							|  |  |  |   for (it = joint_data.begin(); it != joint_data.end(); it++) { | 
					
						
							|  |  |  |     if ((*it).joint_uid == uid) { | 
					
						
							|  |  |  |       copy_m4_m4(inv_bind_mat, (*it).inv_bind_mat); | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 16:03:57 +00:00
										 |  |  | Object *SkinInfo::BKE_armature_from_object() | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return ob_arm; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | const COLLADAFW::UniqueId &SkinInfo::get_controller_uid() | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return controller_uid; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  | /* check if this skin controller references a joint or any descendant of it
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * some nodes may not be referenced by SkinController, | 
					
						
							|  |  |  |  * in this case to determine if the node belongs to this armature, | 
					
						
							|  |  |  |  * we need to search down the tree */ | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | bool SkinInfo::uses_joint_or_descendant(COLLADAFW::Node *node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const COLLADAFW::UniqueId &uid = node->getUniqueId(); | 
					
						
							|  |  |  |   std::vector<JointData>::iterator it; | 
					
						
							|  |  |  |   for (it = joint_data.begin(); it != joint_data.end(); it++) { | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     if ((*it).joint_uid == uid) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   COLLADAFW::NodePointerArray &children = node->getChildNodes(); | 
					
						
							|  |  |  |   for (unsigned int i = 0; i < children.getCount(); i++) { | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     if (uses_joint_or_descendant(children[i])) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void SkinInfo::link_armature(bContext *C, | 
					
						
							|  |  |  |                              Object *ob, | 
					
						
							|  |  |  |                              std::map<COLLADAFW::UniqueId, COLLADAFW::Node *> &joint_by_uid, | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  |                              TransformReader *tm) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Main *bmain = CTX_data_main(C); | 
					
						
							|  |  |  |   Scene *scene = CTX_data_scene(C); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 18:37:57 +05:30
										 |  |  |   ModifierData *md = ED_object_modifier_add( | 
					
						
							|  |  |  |       nullptr, bmain, scene, ob, nullptr, eModifierType_Armature); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ArmatureModifierData *amd = (ArmatureModifierData *)md; | 
					
						
							|  |  |  |   amd->object = ob_arm; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* XXX Why do we enforce objects to be children of Armatures if they weren't so before ?*/ | 
					
						
							|  |  |  |   if (!BKE_object_is_child_recursive(ob_arm, ob)) { | 
					
						
							|  |  |  |     bc_set_parent(ob, ob_arm, C); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Object workob; | 
					
						
							|  |  |  |   ob->parent = ob_arm; | 
					
						
							|  |  |  |   ob->partype = PAROBJECT; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BKE_object_workob_calc_parent(scene, ob, &workob); | 
					
						
							|  |  |  |   invert_m4_m4(ob->parentinv, workob.obmat); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DEG_id_tag_update(&obn->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   copy_m4_m4(ob->obmat, bind_shape_matrix); | 
					
						
							| 
									
										
										
										
											2020-11-06 14:25:30 +01:00
										 |  |  |   BKE_object_apply_mat4(ob, ob->obmat, false, false); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   amd->deformflag = ARM_DEF_VGROUP; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |   /* create all vertex groups */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   std::vector<JointData>::iterator it; | 
					
						
							|  |  |  |   int joint_index; | 
					
						
							|  |  |  |   for (it = joint_data.begin(), joint_index = 0; it != joint_data.end(); it++, joint_index++) { | 
					
						
							|  |  |  |     const char *name = "Group"; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |     /* skip joints that have invalid UID */ | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     if ((*it).joint_uid == COLLADAFW::UniqueId::INVALID) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       continue; | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-08 08:07:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |     /* name group by joint node name */ | 
					
						
							| 
									
										
										
										
											2018-06-08 08:07:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (joint_by_uid.find((*it).joint_uid) != joint_by_uid.end()) { | 
					
						
							|  |  |  |       name = bc_get_joint_name(joint_by_uid[(*it).joint_uid]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     BKE_object_defgroup_add_name(ob, name); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |   /* <vcount> - number of joints per vertex - joints_per_vertex
 | 
					
						
							|  |  |  |    * <v> - [[bone index, weight index] * joints per vertex] * vertices - weight indices | 
					
						
							|  |  |  |    * ^ bone index can be -1 meaning weight toward bind shape, how to express this in Blender? | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * for each vertex in weight indices | 
					
						
							|  |  |  |    *  for each bone index in vertex | 
					
						
							|  |  |  |    *      add vertex to group at group index | 
					
						
							|  |  |  |    *      treat group index -1 specially | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * get def group by index with BLI_findlink */ | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (unsigned int vertex = 0, weight = 0; vertex < joints_per_vertex.getCount(); vertex++) { | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     unsigned int limit = weight + joints_per_vertex[vertex]; | 
					
						
							|  |  |  |     for (; weight < limit; weight++) { | 
					
						
							|  |  |  |       int joint = joint_indices[weight], joint_weight = weight_indices[weight]; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |       /* -1 means "weight towards the bind shape", we just don't assign it to any group */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (joint != -1) { | 
					
						
							|  |  |  |         bDeformGroup *def = (bDeformGroup *)BLI_findlink(&ob->defbase, joint); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         ED_vgroup_vert_add(ob, def, vertex, weights[joint_weight], WEIGHT_REPLACE); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bPoseChannel *SkinInfo::get_pose_channel_from_node(COLLADAFW::Node *node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return BKE_pose_channel_find_name(ob_arm->pose, bc_get_joint_name(node)); | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SkinInfo::set_parent(Object *_parent) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   parent = _parent; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | Object *SkinInfo::get_parent() | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return parent; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | void SkinInfo::find_root_joints(const std::vector<COLLADAFW::Node *> &root_joints, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                                 std::map<COLLADAFW::UniqueId, COLLADAFW::Node *> &joint_by_uid, | 
					
						
							|  |  |  |                                 std::vector<COLLADAFW::Node *> &result) | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   std::vector<COLLADAFW::Node *>::const_iterator it; | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |   /* for each root_joint */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (it = root_joints.begin(); it != root_joints.end(); it++) { | 
					
						
							|  |  |  |     COLLADAFW::Node *root = *it; | 
					
						
							|  |  |  |     std::vector<JointData>::iterator ji; | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |     /* for each joint_data in this skin */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     for (ji = joint_data.begin(); ji != joint_data.end(); ji++) { | 
					
						
							|  |  |  |       if (joint_by_uid.find((*ji).joint_uid) != joint_by_uid.end()) { | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |         /* get joint node from joint map */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         COLLADAFW::Node *joint = joint_by_uid[(*ji).joint_uid]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-30 13:41:21 +10:00
										 |  |  |         /* find if joint node is in the tree belonging to the root_joint */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         if (find_node_in_tree(joint, root)) { | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |           if (std::find(result.begin(), result.end(), root) == result.end()) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             result.push_back(root); | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool SkinInfo::find_node_in_tree(COLLADAFW::Node *node, COLLADAFW::Node *tree_root) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |   if (node == tree_root) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   COLLADAFW::NodePointerArray &children = tree_root->getChildNodes(); | 
					
						
							|  |  |  |   for (unsigned int i = 0; i < children.getCount(); i++) { | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     if (find_node_in_tree(node, children[i])) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2019-05-31 22:51:19 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-10-05 00:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2010-10-05 00:49:39 +00:00
										 |  |  | } |