| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2011-10-23 17:52:20 +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 | 
					
						
							| 
									
										
										
										
											2020-05-09 17:14:35 +10:00
										 |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							| 
									
										
										
										
											2011-10-23 17:52:20 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2005 by the Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup modifiers | 
					
						
							| 
									
										
										
										
											2011-02-25 13:57:17 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-25 11:56:24 +01:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_math.h"
 | 
					
						
							|  |  |  | #include "BLI_task.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "BLT_translation.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-01 09:38:00 -05:00
										 |  |  | #include "DNA_defaults.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-08 11:33:31 +02:00
										 |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-12 00:36:50 +00:00
										 |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							| 
									
										
										
										
											2010-08-04 04:01:27 +00:00
										 |  |  | #include "DNA_object_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-15 18:36:24 +06:00
										 |  |  | #include "DNA_scene_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "DNA_screen_types.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "BKE_context.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BKE_deform.h"
 | 
					
						
							|  |  |  | #include "BKE_editmesh.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-10 12:58:59 +01:00
										 |  |  | #include "BKE_lib_id.h"
 | 
					
						
							|  |  |  | #include "BKE_lib_query.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-08 11:33:31 +02:00
										 |  |  | #include "BKE_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-07 11:17:25 +01:00
										 |  |  | #include "BKE_mesh_runtime.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-10 22:32:06 +10:00
										 |  |  | #include "BKE_mesh_wrapper.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | #include "BKE_modifier.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "BKE_screen.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "UI_interface.h"
 | 
					
						
							|  |  |  | #include "UI_resources.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-16 17:17:43 +02:00
										 |  |  | #include "BLO_read_write.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "RNA_access.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-12 22:33:43 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  | #include "DEG_depsgraph.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-22 15:03:42 +02:00
										 |  |  | #include "DEG_depsgraph_query.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "MOD_ui_common.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | #include "MOD_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #ifdef __SSE2__
 | 
					
						
							|  |  |  | #  include <emmintrin.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void initData(ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-01 09:38:00 -05:00
										 |  |  |   BLI_assert(MEMCMP_STRUCT_AFTER_IS_ZERO(mmd, modifier)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEMCPY_STRUCT_AFTER(mmd, DNA_struct_default_get(MeshDeformModifierData), modifier); | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void freeData(ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   if (mmd->bindinfluences) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->bindinfluences); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindoffsets) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->bindoffsets); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindcagecos) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->bindcagecos); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->dyngrid) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->dyngrid); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->dyninfluences) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->dyninfluences); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->dynverts) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->dynverts); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindweights) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->bindweights); /* deprecated */ | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindcos) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     MEM_freeN(mmd->bindcos); /* deprecated */ | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-04 12:45:30 +02:00
										 |  |  | static void copyData(const ModifierData *md, ModifierData *target, const int flag) | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const MeshDeformModifierData *mmd = (const MeshDeformModifierData *)md; | 
					
						
							|  |  |  |   MeshDeformModifierData *tmmd = (MeshDeformModifierData *)target; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-08 10:14:02 +02:00
										 |  |  |   BKE_modifier_copydata_generic(md, target, flag); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   if (mmd->bindinfluences) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->bindinfluences = MEM_dupallocN(mmd->bindinfluences); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindoffsets) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->bindoffsets = MEM_dupallocN(mmd->bindoffsets); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindcagecos) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->bindcagecos = MEM_dupallocN(mmd->bindcagecos); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->dyngrid) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->dyngrid = MEM_dupallocN(mmd->dyngrid); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->dyninfluences) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->dyninfluences = MEM_dupallocN(mmd->dyninfluences); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->dynverts) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->dynverts = MEM_dupallocN(mmd->dynverts); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindweights) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->bindweights = MEM_dupallocN(mmd->bindweights); /* deprecated */ | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (mmd->bindcos) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     tmmd->bindcos = MEM_dupallocN(mmd->bindcos); /* deprecated */ | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void requiredDataMask(Object *UNUSED(ob), | 
					
						
							|  |  |  |                              ModifierData *md, | 
					
						
							|  |  |  |                              CustomData_MeshMasks *r_cddata_masks) | 
					
						
							| 
									
										
										
										
											2018-06-17 17:04:27 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* ask for vertexgroups if we need them */ | 
					
						
							|  |  |  |   if (mmd->defgrp_name[0] != '\0') { | 
					
						
							|  |  |  |     r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static bool isDisabled(const struct Scene *UNUSED(scene), | 
					
						
							|  |  |  |                        ModifierData *md, | 
					
						
							|  |  |  |                        bool UNUSED(useRenderParams)) | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-23 15:50:53 +02:00
										 |  |  |   /* The object type check is only needed here in case we have a placeholder
 | 
					
						
							|  |  |  |    * object assigned (because the library containing the mesh is missing). | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2019-08-31 01:19:22 +10:00
										 |  |  |    * In other cases it should be impossible to have a type mismatch. | 
					
						
							| 
									
										
										
										
											2019-08-23 15:50:53 +02:00
										 |  |  |    */ | 
					
						
							|  |  |  |   return !mmd->object || mmd->object->type != OB_MESH; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-01 18:05:23 +02:00
										 |  |  | static void foreachIDLink(ModifierData *md, Object *ob, IDWalkFunc walk, void *userData) | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-01 18:05:23 +02:00
										 |  |  |   walk(userData, ob, (ID **)&mmd->object, IDWALK_CB_NOP); | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-22 12:54:06 +01:00
										 |  |  | static void updateDepsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx) | 
					
						
							| 
									
										
											  
											
												Depsgraph: New dependency graph integration commit
This commit integrates the work done so far on the new dependency graph system,
where goal was to replace legacy depsgraph with the new one, supporting loads of
neat features like:
- More granular dependency relation nature, which solves issues with fake cycles
  in the dependencies.
- Move towards all-animatable, by better integration of drivers into the system.
- Lay down some basis for upcoming copy-on-write, overrides and so on.
The new system is living side-by-side with the previous one and disabled by
default, so nothing will become suddenly broken. The way to enable new depsgraph
is to pass `--new-depsgraph` command line argument.
It's a bit early to consider the system production-ready, there are some TODOs
and issues were discovered during the merge period, they'll be addressed ASAP.
But it's important to merge, because it's the only way to attract artists to
really start testing this system.
There are number of assorted documents related on the design of the new system:
* http://wiki.blender.org/index.php/User:Aligorith/GSoC2013_Depsgraph#Design_Documents
* http://wiki.blender.org/index.php/User:Nazg-gul/DependencyGraph
There are also some user-related information online:
* http://code.blender.org/2015/02/blender-dependency-graph-branch-for-users/
* http://code.blender.org/2015/03/more-dependency-graph-tricks/
Kudos to everyone who was involved into the project:
- Joshua "Aligorith" Leung -- design specification, initial code
- Lukas "lukas_t" Toenne -- integrating code into blender, with further fixes
- Sergey "Sergey" "Sharybin" -- some mocking around, trying to wrap up the
  project and so
- Bassam "slikdigit" Kurdali -- stressing the new system, reporting all the
  issues and recording/writing documentation.
- Everyone else who i forgot to mention here :)
											
										 
											2015-05-12 15:05:57 +05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							|  |  |  |   if (mmd->object != NULL) { | 
					
						
							| 
									
										
										
										
											2019-11-21 12:20:49 +01:00
										 |  |  |     DEG_add_object_relation(ctx->node, mmd->object, DEG_OB_COMP_TRANSFORM, "Mesh Deform Modifier"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     DEG_add_object_relation(ctx->node, mmd->object, DEG_OB_COMP_GEOMETRY, "Mesh Deform Modifier"); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-04-20 11:08:49 +02:00
										 |  |  |   /* We need own transformation as well. */ | 
					
						
							|  |  |  |   DEG_add_modifier_to_transform_relation(ctx->node, "Mesh Deform Modifier"); | 
					
						
							| 
									
										
											  
											
												Depsgraph: New dependency graph integration commit
This commit integrates the work done so far on the new dependency graph system,
where goal was to replace legacy depsgraph with the new one, supporting loads of
neat features like:
- More granular dependency relation nature, which solves issues with fake cycles
  in the dependencies.
- Move towards all-animatable, by better integration of drivers into the system.
- Lay down some basis for upcoming copy-on-write, overrides and so on.
The new system is living side-by-side with the previous one and disabled by
default, so nothing will become suddenly broken. The way to enable new depsgraph
is to pass `--new-depsgraph` command line argument.
It's a bit early to consider the system production-ready, there are some TODOs
and issues were discovered during the merge period, they'll be addressed ASAP.
But it's important to merge, because it's the only way to attract artists to
really start testing this system.
There are number of assorted documents related on the design of the new system:
* http://wiki.blender.org/index.php/User:Aligorith/GSoC2013_Depsgraph#Design_Documents
* http://wiki.blender.org/index.php/User:Nazg-gul/DependencyGraph
There are also some user-related information online:
* http://code.blender.org/2015/02/blender-dependency-graph-branch-for-users/
* http://code.blender.org/2015/03/more-dependency-graph-tricks/
Kudos to everyone who was involved into the project:
- Joshua "Aligorith" Leung -- design specification, initial code
- Lukas "lukas_t" Toenne -- integrating code into blender, with further fixes
- Sergey "Sergey" "Sharybin" -- some mocking around, trying to wrap up the
  project and so
- Bassam "slikdigit" Kurdali -- stressing the new system, reporting all the
  issues and recording/writing documentation.
- Everyone else who i forgot to mention here :)
											
										 
											2015-05-12 15:05:57 +05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-30 11:08:33 +11:00
										 |  |  | static float meshdeform_dynamic_bind(MeshDeformModifierData *mmd, float (*dco)[3], float vec[3]) | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MDefCell *cell; | 
					
						
							|  |  |  |   MDefInfluence *inf; | 
					
						
							|  |  |  |   float gridvec[3], dvec[3], ivec[3], wx, wy, wz; | 
					
						
							|  |  |  |   float weight, cageweight, totweight, *cageco; | 
					
						
							|  |  |  |   int i, j, a, x, y, z, size; | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #ifdef __SSE2__
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   __m128 co = _mm_setzero_ps(); | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   float co[3] = {0.0f, 0.0f, 0.0f}; | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   totweight = 0.0f; | 
					
						
							|  |  |  |   size = mmd->dyngridsize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < 3; i++) { | 
					
						
							|  |  |  |     gridvec[i] = (vec[i] - mmd->dyncellmin[i] - mmd->dyncellwidth * 0.5f) / mmd->dyncellwidth; | 
					
						
							|  |  |  |     ivec[i] = (int)gridvec[i]; | 
					
						
							|  |  |  |     dvec[i] = gridvec[i] - ivec[i]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < 8; i++) { | 
					
						
							|  |  |  |     if (i & 1) { | 
					
						
							|  |  |  |       x = ivec[0] + 1; | 
					
						
							|  |  |  |       wx = dvec[0]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       x = ivec[0]; | 
					
						
							|  |  |  |       wx = 1.0f - dvec[0]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (i & 2) { | 
					
						
							|  |  |  |       y = ivec[1] + 1; | 
					
						
							|  |  |  |       wy = dvec[1]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       y = ivec[1]; | 
					
						
							|  |  |  |       wy = 1.0f - dvec[1]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (i & 4) { | 
					
						
							|  |  |  |       z = ivec[2] + 1; | 
					
						
							|  |  |  |       wz = dvec[2]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       z = ivec[2]; | 
					
						
							|  |  |  |       wz = 1.0f - dvec[2]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CLAMP(x, 0, size - 1); | 
					
						
							|  |  |  |     CLAMP(y, 0, size - 1); | 
					
						
							|  |  |  |     CLAMP(z, 0, size - 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     a = x + y * size + z * size * size; | 
					
						
							|  |  |  |     weight = wx * wy * wz; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cell = &mmd->dyngrid[a]; | 
					
						
							|  |  |  |     inf = mmd->dyninfluences + cell->offset; | 
					
						
							|  |  |  |     for (j = 0; j < cell->totinfluence; j++, inf++) { | 
					
						
							|  |  |  |       cageco = dco[inf->vertex]; | 
					
						
							|  |  |  |       cageweight = weight * inf->weight; | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #ifdef __SSE2__
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       { | 
					
						
							|  |  |  |         __m128 cageweight_r = _mm_set1_ps(cageweight); | 
					
						
							|  |  |  |         /* This will load one extra element, this is ok because
 | 
					
						
							|  |  |  |          * we ignore that part of register anyway. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         __m128 cageco_r = _mm_loadu_ps(cageco); | 
					
						
							|  |  |  |         co = _mm_add_ps(co, _mm_mul_ps(cageco_r, cageweight_r)); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       co[0] += cageweight * cageco[0]; | 
					
						
							|  |  |  |       co[1] += cageweight * cageco[1]; | 
					
						
							|  |  |  |       co[2] += cageweight * cageco[2]; | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       totweight += cageweight; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #ifdef __SSE2__
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   copy_v3_v3(vec, (float *)&co); | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   copy_v3_v3(vec, co); | 
					
						
							| 
									
										
										
										
											2014-10-21 11:37:29 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return totweight; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-22 11:56:52 +02:00
										 |  |  | typedef struct MeshdeformUserdata { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /*const*/ MeshDeformModifierData *mmd; | 
					
						
							|  |  |  |   const MDeformVert *dvert; | 
					
						
							|  |  |  |   /*const*/ float (*dco)[3]; | 
					
						
							|  |  |  |   int defgrp_index; | 
					
						
							|  |  |  |   float (*vertexCos)[3]; | 
					
						
							|  |  |  |   float (*cagemat)[4]; | 
					
						
							|  |  |  |   float (*icagemat)[3]; | 
					
						
							| 
									
										
										
										
											2014-10-22 11:56:52 +02:00
										 |  |  | } MeshdeformUserdata; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void meshdeform_vert_task(void *__restrict userdata, | 
					
						
							|  |  |  |                                  const int iter, | 
					
						
							| 
									
										
										
										
											2019-07-30 14:56:47 +02:00
										 |  |  |                                  const TaskParallelTLS *__restrict UNUSED(tls)) | 
					
						
							| 
									
										
										
										
											2014-10-22 11:56:52 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshdeformUserdata *data = userdata; | 
					
						
							|  |  |  |   /*const*/ MeshDeformModifierData *mmd = data->mmd; | 
					
						
							|  |  |  |   const MDeformVert *dvert = data->dvert; | 
					
						
							|  |  |  |   const int defgrp_index = data->defgrp_index; | 
					
						
							|  |  |  |   const int *offsets = mmd->bindoffsets; | 
					
						
							|  |  |  |   const MDefInfluence *__restrict influences = mmd->bindinfluences; | 
					
						
							|  |  |  |   /*const*/ float(*__restrict dco)[3] = data->dco; | 
					
						
							|  |  |  |   float(*vertexCos)[3] = data->vertexCos; | 
					
						
							|  |  |  |   float co[3]; | 
					
						
							|  |  |  |   float weight, totweight, fac = 1.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   if (mmd->flag & MOD_MDEF_DYNAMIC_BIND) { | 
					
						
							|  |  |  |     if (!mmd->dynverts[iter]) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       return; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (dvert) { | 
					
						
							| 
									
										
										
										
											2020-03-06 12:50:56 +11:00
										 |  |  |     fac = BKE_defvert_find_weight(&dvert[iter], defgrp_index); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (mmd->flag & MOD_MDEF_INVERT_VGROUP) { | 
					
						
							|  |  |  |       fac = 1.0f - fac; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fac <= 0.0f) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (mmd->flag & MOD_MDEF_DYNAMIC_BIND) { | 
					
						
							|  |  |  |     /* transform coordinate into cage's local space */ | 
					
						
							|  |  |  |     mul_v3_m4v3(co, data->cagemat, vertexCos[iter]); | 
					
						
							|  |  |  |     totweight = meshdeform_dynamic_bind(mmd, dco, co); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     totweight = 0.0f; | 
					
						
							|  |  |  |     zero_v3(co); | 
					
						
							|  |  |  |     int start = offsets[iter]; | 
					
						
							|  |  |  |     int end = offsets[iter + 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int a = start; a < end; a++) { | 
					
						
							|  |  |  |       weight = influences[a].weight; | 
					
						
							|  |  |  |       madd_v3_v3fl(co, dco[influences[a].vertex], weight); | 
					
						
							|  |  |  |       totweight += weight; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (totweight > 0.0f) { | 
					
						
							|  |  |  |     mul_v3_fl(co, fac / totweight); | 
					
						
							|  |  |  |     mul_m3_v3(data->icagemat, co); | 
					
						
							| 
									
										
										
										
											2020-08-26 21:01:38 +10:00
										 |  |  |     add_v3_v3(vertexCos[iter], co); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-10-22 11:56:52 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void meshdeformModifier_do(ModifierData *md, | 
					
						
							|  |  |  |                                   const ModifierEvalContext *ctx, | 
					
						
							|  |  |  |                                   Mesh *mesh, | 
					
						
							|  |  |  |                                   float (*vertexCos)[3], | 
					
						
							|  |  |  |                                   int numVerts) | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							|  |  |  |   Object *ob = ctx->object; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Mesh *cagemesh; | 
					
						
							|  |  |  |   MDeformVert *dvert = NULL; | 
					
						
							|  |  |  |   float imat[4][4], cagemat[4][4], iobmat[4][4], icagemat[3][3], cmat[4][4]; | 
					
						
							| 
									
										
										
										
											2020-08-26 23:20:37 +10:00
										 |  |  |   float(*dco)[3] = NULL, (*bindcagecos)[3]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int a, totvert, totcagevert, defgrp_index; | 
					
						
							|  |  |  |   MeshdeformUserdata data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   static int recursive_bind_sentinel = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   if (mmd->object == NULL || (mmd->bindcagecos == NULL && mmd->bindfunc == NULL)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Get cage mesh.
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Only do this is the target object is in edit mode by itself, meaning | 
					
						
							|  |  |  |    * we don't allow linked edit meshes here. | 
					
						
							|  |  |  |    * This is because editbmesh_get_mesh_cage_and_final() might easily | 
					
						
							|  |  |  |    * conflict with the thread which evaluates object which is in the edit | 
					
						
							|  |  |  |    * mode for this mesh. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * We'll support this case once granular dependency graph is landed. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   Object *ob_target = mmd->object; | 
					
						
							|  |  |  |   cagemesh = BKE_modifier_get_evaluated_mesh_from_evaluated_object(ob_target, false); | 
					
						
							|  |  |  |   if (cagemesh == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ctx->object, md, "Cannot get mesh from cage object"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* compute matrices to go in and out of cage object space */ | 
					
						
							|  |  |  |   invert_m4_m4(imat, ob_target->obmat); | 
					
						
							|  |  |  |   mul_m4_m4m4(cagemat, imat, ob->obmat); | 
					
						
							|  |  |  |   mul_m4_m4m4(cmat, mmd->bindmat, cagemat); | 
					
						
							|  |  |  |   invert_m4_m4(iobmat, cmat); | 
					
						
							|  |  |  |   copy_m3_m4(icagemat, iobmat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* bind weights if needed */ | 
					
						
							|  |  |  |   if (!mmd->bindcagecos) { | 
					
						
							|  |  |  |     /* progress bar redraw can make this recursive .. */ | 
					
						
							|  |  |  |     if (!DEG_is_active(ctx->depsgraph)) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |       BKE_modifier_set_error(ob, md, "Attempt to bind from inactive dependency graph"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       goto finally; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!recursive_bind_sentinel) { | 
					
						
							|  |  |  |       recursive_bind_sentinel = 1; | 
					
						
							|  |  |  |       mmd->bindfunc(mmd, cagemesh, (float *)vertexCos, numVerts, cagemat); | 
					
						
							|  |  |  |       recursive_bind_sentinel = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     goto finally; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* verify we have compatible weights */ | 
					
						
							|  |  |  |   totvert = numVerts; | 
					
						
							| 
									
										
										
										
											2020-08-26 23:20:37 +10:00
										 |  |  |   totcagevert = BKE_mesh_wrapper_vert_len(cagemesh); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (mmd->totvert != totvert) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, md, "Vertices changed from %d to %d", mmd->totvert, totvert); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     goto finally; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (mmd->totcagevert != totcagevert) { | 
					
						
							| 
									
										
										
										
											2020-07-22 16:21:33 +10:00
										 |  |  |     BKE_modifier_set_error( | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |         ob, md, "Cage vertices changed from %d to %d", mmd->totcagevert, totcagevert); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     goto finally; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (mmd->bindcagecos == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, md, "Bind data missing"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     goto finally; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* We allocate 1 element extra to make it possible to
 | 
					
						
							|  |  |  |    * load the values to SSE registers, which are float4. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   dco = MEM_calloc_arrayN((totcagevert + 1), sizeof(*dco), "MDefDco"); | 
					
						
							|  |  |  |   zero_v3(dco[totcagevert]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 23:20:37 +10:00
										 |  |  |   /* setup deformation data */ | 
					
						
							|  |  |  |   BKE_mesh_wrapper_vert_coords_copy(cagemesh, dco, totcagevert); | 
					
						
							|  |  |  |   bindcagecos = (float(*)[3])mmd->bindcagecos; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (a = 0; a < totcagevert; a++) { | 
					
						
							|  |  |  |     /* get cage vertex in world space with binding transform */ | 
					
						
							| 
									
										
										
										
											2020-08-26 23:28:44 +10:00
										 |  |  |     float co[3]; | 
					
						
							|  |  |  |     mul_v3_m4v3(co, mmd->bindmat, dco[a]); | 
					
						
							| 
									
										
										
										
											2020-08-26 21:01:38 +10:00
										 |  |  |     /* compute difference with world space bind coord */ | 
					
						
							|  |  |  |     sub_v3_v3v3(dco[a], co, bindcagecos[a]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MOD_get_vgroup(ob, mesh, mmd->defgrp_name, &dvert, &defgrp_index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Initialize data to be pass to the for body function. */ | 
					
						
							|  |  |  |   data.mmd = mmd; | 
					
						
							|  |  |  |   data.dvert = dvert; | 
					
						
							|  |  |  |   data.dco = dco; | 
					
						
							|  |  |  |   data.defgrp_index = defgrp_index; | 
					
						
							|  |  |  |   data.vertexCos = vertexCos; | 
					
						
							|  |  |  |   data.cagemat = cagemat; | 
					
						
							|  |  |  |   data.icagemat = icagemat; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Do deformation. */ | 
					
						
							| 
									
										
										
										
											2019-07-30 14:56:47 +02:00
										 |  |  |   TaskParallelSettings settings; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_parallel_range_settings_defaults(&settings); | 
					
						
							|  |  |  |   settings.min_iter_per_thread = 16; | 
					
						
							|  |  |  |   BLI_task_parallel_range(0, totvert, &data, meshdeform_vert_task, &settings); | 
					
						
							| 
									
										
										
										
											2014-10-22 11:56:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 11:17:25 +01:00
										 |  |  | finally: | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MEM_SAFE_FREE(dco); | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void deformVerts(ModifierData *md, | 
					
						
							|  |  |  |                         const ModifierEvalContext *ctx, | 
					
						
							|  |  |  |                         Mesh *mesh, | 
					
						
							|  |  |  |                         float (*vertexCos)[3], | 
					
						
							|  |  |  |                         int numVerts) | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mesh *mesh_src = MOD_deform_mesh_eval_get(ctx->object, NULL, mesh, NULL, numVerts, false, false); | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MOD_previous_vcos_store(md, vertexCos); /* if next modifier needs original vertices */ | 
					
						
							| 
									
										
										
										
											2013-11-04 23:33:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   meshdeformModifier_do(md, ctx, mesh_src, vertexCos, numVerts); | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (!ELEM(mesh_src, NULL, mesh)) { | 
					
						
							|  |  |  |     BKE_id_free(NULL, mesh_src); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void deformVertsEM(ModifierData *md, | 
					
						
							|  |  |  |                           const ModifierEvalContext *ctx, | 
					
						
							|  |  |  |                           struct BMEditMesh *editData, | 
					
						
							|  |  |  |                           Mesh *mesh, | 
					
						
							|  |  |  |                           float (*vertexCos)[3], | 
					
						
							|  |  |  |                           int numVerts) | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mesh *mesh_src = MOD_deform_mesh_eval_get( | 
					
						
							|  |  |  |       ctx->object, editData, mesh, NULL, numVerts, false, false); | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-25 20:16:42 +10:00
										 |  |  |   /* TODO(Campbell): use edit-mode data only (remove this line). */ | 
					
						
							|  |  |  |   if (mesh_src != NULL) { | 
					
						
							|  |  |  |     BKE_mesh_wrapper_ensure_mdata(mesh_src); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   meshdeformModifier_do(md, ctx, mesh_src, vertexCos, numVerts); | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (!ELEM(mesh_src, NULL, mesh)) { | 
					
						
							|  |  |  |     BKE_id_free(NULL, mesh_src); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-27 13:49:53 +00:00
										 |  |  | #define MESHDEFORM_MIN_INFLUENCE 0.00001f
 | 
					
						
							| 
									
										
										
										
											2010-04-23 11:19:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-08 10:14:02 +02:00
										 |  |  | void BKE_modifier_mdef_compact_influences(ModifierData *md) | 
					
						
							| 
									
										
										
										
											2010-04-23 11:19:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							|  |  |  |   float weight, *weights, totweight; | 
					
						
							|  |  |  |   int totinfluence, totvert, totcagevert, a, b; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   weights = mmd->bindweights; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   if (!weights) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   totvert = mmd->totvert; | 
					
						
							|  |  |  |   totcagevert = mmd->totcagevert; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* count number of influences above threshold */ | 
					
						
							|  |  |  |   for (b = 0; b < totvert; b++) { | 
					
						
							|  |  |  |     for (a = 0; a < totcagevert; a++) { | 
					
						
							|  |  |  |       weight = weights[a + b * totcagevert]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |       if (weight > MESHDEFORM_MIN_INFLUENCE) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         mmd->totinfluence++; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* allocate bind influences */ | 
					
						
							|  |  |  |   mmd->bindinfluences = MEM_calloc_arrayN( | 
					
						
							|  |  |  |       mmd->totinfluence, sizeof(MDefInfluence), "MDefBindInfluence"); | 
					
						
							|  |  |  |   mmd->bindoffsets = MEM_calloc_arrayN((totvert + 1), sizeof(int), "MDefBindOffset"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* write influences */ | 
					
						
							|  |  |  |   totinfluence = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (b = 0; b < totvert; b++) { | 
					
						
							|  |  |  |     mmd->bindoffsets[b] = totinfluence; | 
					
						
							|  |  |  |     totweight = 0.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* sum total weight */ | 
					
						
							|  |  |  |     for (a = 0; a < totcagevert; a++) { | 
					
						
							|  |  |  |       weight = weights[a + b * totcagevert]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |       if (weight > MESHDEFORM_MIN_INFLUENCE) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         totweight += weight; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:15:10 +10:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* assign weights normalized */ | 
					
						
							|  |  |  |     for (a = 0; a < totcagevert; a++) { | 
					
						
							|  |  |  |       weight = weights[a + b * totcagevert]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (weight > MESHDEFORM_MIN_INFLUENCE) { | 
					
						
							|  |  |  |         mmd->bindinfluences[totinfluence].weight = weight / totweight; | 
					
						
							|  |  |  |         mmd->bindinfluences[totinfluence].vertex = a; | 
					
						
							|  |  |  |         totinfluence++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   mmd->bindoffsets[b] = totinfluence; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* free */ | 
					
						
							|  |  |  |   MEM_freeN(mmd->bindweights); | 
					
						
							|  |  |  |   mmd->bindweights = NULL; | 
					
						
							| 
									
										
										
										
											2010-04-23 11:19:06 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  | static void panel_draw(const bContext *UNUSED(C), Panel *panel) | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | { | 
					
						
							|  |  |  |   uiLayout *col; | 
					
						
							|  |  |  |   uiLayout *layout = panel->layout; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   PointerRNA ob_ptr; | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  |   PointerRNA *ptr = modifier_panel_get_property_pointers(panel, &ob_ptr); | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  |   bool is_bound = RNA_boolean_get(ptr, "is_bound"); | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   uiLayoutSetPropSep(layout, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   col = uiLayoutColumn(layout, true); | 
					
						
							|  |  |  |   uiLayoutSetEnabled(col, !is_bound); | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  |   uiItemR(col, ptr, "object", 0, NULL, ICON_NONE); | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  |   modifier_vgroup_ui(layout, ptr, &ob_ptr, "vertex_group", "invert_vertex_group", NULL); | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   col = uiLayoutColumn(layout, false); | 
					
						
							|  |  |  |   uiLayoutSetEnabled(col, !is_bound); | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  |   uiItemR(col, ptr, "precision", 0, NULL, ICON_NONE); | 
					
						
							|  |  |  |   uiItemR(col, ptr, "use_dynamic_bind", 0, NULL, ICON_NONE); | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   uiItemO(layout, | 
					
						
							|  |  |  |           is_bound ? IFACE_("Unbind") : IFACE_("Bind"), | 
					
						
							|  |  |  |           ICON_NONE, | 
					
						
							|  |  |  |           "OBJECT_OT_meshdeform_bind"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  |   modifier_panel_end(layout, ptr); | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void panelRegister(ARegionType *region_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   modifier_panel_register(region_type, eModifierType_MeshDeform, panel_draw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-16 17:17:43 +02:00
										 |  |  | static void blendWrite(BlendWriter *writer, const ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							|  |  |  |   int size = mmd->dyngridsize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLO_write_struct_array(writer, MDefInfluence, mmd->totinfluence, mmd->bindinfluences); | 
					
						
							|  |  |  |   BLO_write_int32_array(writer, mmd->totvert + 1, mmd->bindoffsets); | 
					
						
							|  |  |  |   BLO_write_float3_array(writer, mmd->totcagevert, mmd->bindcagecos); | 
					
						
							|  |  |  |   BLO_write_struct_array(writer, MDefCell, size * size * size, mmd->dyngrid); | 
					
						
							|  |  |  |   BLO_write_struct_array(writer, MDefInfluence, mmd->totinfluence, mmd->dyninfluences); | 
					
						
							|  |  |  |   BLO_write_int32_array(writer, mmd->totvert, mmd->dynverts); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void blendRead(BlendDataReader *reader, ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   MeshDeformModifierData *mmd = (MeshDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLO_read_data_address(reader, &mmd->bindinfluences); | 
					
						
							|  |  |  |   BLO_read_int32_array(reader, mmd->totvert + 1, &mmd->bindoffsets); | 
					
						
							|  |  |  |   BLO_read_float3_array(reader, mmd->totcagevert, &mmd->bindcagecos); | 
					
						
							|  |  |  |   BLO_read_data_address(reader, &mmd->dyngrid); | 
					
						
							|  |  |  |   BLO_read_data_address(reader, &mmd->dyninfluences); | 
					
						
							|  |  |  |   BLO_read_int32_array(reader, mmd->totvert, &mmd->dynverts); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Deprecated storage. */ | 
					
						
							|  |  |  |   BLO_read_float_array(reader, mmd->totvert, &mmd->bindweights); | 
					
						
							|  |  |  |   BLO_read_float3_array(reader, mmd->totcagevert, &mmd->bindcos); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | ModifierTypeInfo modifierType_MeshDeform = { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* name */ "MeshDeform", | 
					
						
							|  |  |  |     /* structName */ "MeshDeformModifierData", | 
					
						
							|  |  |  |     /* structSize */ sizeof(MeshDeformModifierData), | 
					
						
							| 
									
										
										
										
											2020-09-25 12:49:18 +02:00
										 |  |  |     /* srna */ &RNA_MeshDeformModifier, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* type */ eModifierTypeType_OnlyDeform, | 
					
						
							| 
									
										
										
										
											2020-04-21 13:09:41 +02:00
										 |  |  |     /* flags */ eModifierTypeFlag_AcceptsCVs | eModifierTypeFlag_AcceptsVertexCosOnly | | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         eModifierTypeFlag_SupportsEditmode, | 
					
						
							| 
									
										
										
										
											2020-09-25 12:45:30 +02:00
										 |  |  |     /* icon */ ICON_MOD_MESHDEFORM, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* copyData */ copyData, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* deformVerts */ deformVerts, | 
					
						
							|  |  |  |     /* deformMatrices */ NULL, | 
					
						
							|  |  |  |     /* deformVertsEM */ deformVertsEM, | 
					
						
							|  |  |  |     /* deformMatricesEM */ NULL, | 
					
						
							| 
									
										
										
										
											2020-04-21 13:09:41 +02:00
										 |  |  |     /* modifyMesh */ NULL, | 
					
						
							|  |  |  |     /* modifyHair */ NULL, | 
					
						
							| 
									
										
										
										
											2020-12-10 14:35:02 +01:00
										 |  |  |     /* modifyGeometrySet */ NULL, | 
					
						
							| 
									
										
										
										
											2020-04-21 13:09:41 +02:00
										 |  |  |     /* modifyVolume */ NULL, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* initData */ initData, | 
					
						
							|  |  |  |     /* requiredDataMask */ requiredDataMask, | 
					
						
							|  |  |  |     /* freeData */ freeData, | 
					
						
							|  |  |  |     /* isDisabled */ isDisabled, | 
					
						
							|  |  |  |     /* updateDepsgraph */ updateDepsgraph, | 
					
						
							|  |  |  |     /* dependsOnTime */ NULL, | 
					
						
							|  |  |  |     /* dependsOnNormals */ NULL, | 
					
						
							| 
									
										
										
										
											2020-10-01 18:05:23 +02:00
										 |  |  |     /* foreachIDLink */ foreachIDLink, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* foreachTexLink */ NULL, | 
					
						
							|  |  |  |     /* freeRuntimeData */ NULL, | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  |     /* panelRegister */ panelRegister, | 
					
						
							| 
									
										
										
										
											2020-06-16 17:17:43 +02:00
										 |  |  |     /* blendWrite */ blendWrite, | 
					
						
							|  |  |  |     /* blendRead */ blendRead, | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | }; |