| 
									
										
										
										
											2019-01-23 11:28:59 +11: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, | 
					
						
							| 
									
										
										
										
											2019-01-23 11:28:59 +11:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright 2017, Blender Foundation. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 11:23:17 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup modifiers | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  | #include "BLI_alloca.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-25 11:39:14 +01:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							|  |  |  | #include "BLI_math_geom.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-09 11:33:32 +02:00
										 |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							|  |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | #include "DNA_object_types.h"
 | 
					
						
							|  |  |  | #include "DNA_scene_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "DNA_screen_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-09 11:33:32 +02:00
										 |  |  | #include "BKE_bvhutils.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "BKE_context.h"
 | 
					
						
							|  |  |  | #include "BKE_deform.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | #include "BKE_editmesh.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-10 12:58:59 +01:00
										 |  |  | #include "BKE_lib_id.h"
 | 
					
						
							|  |  |  | #include "BKE_lib_query.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-10 15:54:05 +10:00
										 |  |  | #include "BKE_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BKE_mesh_runtime.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-10 22:32:06 +10:00
										 |  |  | #include "BKE_mesh_wrapper.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | #include "BKE_modifier.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "BKE_screen.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "UI_interface.h"
 | 
					
						
							|  |  |  | #include "UI_resources.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-23 16:51:43 +02:00
										 |  |  | #include "BLO_read_write.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "RNA_access.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  | #include "DEG_depsgraph.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  | #include "DEG_depsgraph_query.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  | #include "MOD_ui_common.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | #include "MOD_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct SDefAdjacency { | 
					
						
							|  |  |  |   struct SDefAdjacency *next; | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint index; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } SDefAdjacency; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct SDefAdjacencyArray { | 
					
						
							|  |  |  |   SDefAdjacency *first; | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint num; /* Careful, this is twice the number of polygons (avoids an extra loop) */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } SDefAdjacencyArray; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Polygons per edge (only 2, any more will exit calculation). | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | typedef struct SDefEdgePolys { | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint polys[2], num; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } SDefEdgePolys; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct SDefBindCalcData { | 
					
						
							|  |  |  |   BVHTreeFromMesh *const treeData; | 
					
						
							|  |  |  |   const SDefAdjacencyArray *const vert_edges; | 
					
						
							|  |  |  |   const SDefEdgePolys *const edge_polys; | 
					
						
							|  |  |  |   SDefVert *const bind_verts; | 
					
						
							|  |  |  |   const MLoopTri *const looptri; | 
					
						
							|  |  |  |   const MPoly *const mpoly; | 
					
						
							|  |  |  |   const MEdge *const medge; | 
					
						
							|  |  |  |   const MLoop *const mloop; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Coordinates to bind to, transformed into local space (compatible with `vertexCos`). */ | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   float (*const targetCos)[3]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Coordinates to bind (reference to the modifiers input argument). */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float (*const vertexCos)[3]; | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   float imat[4][4]; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   const float falloff; | 
					
						
							|  |  |  |   int success; | 
					
						
							|  |  |  | } SDefBindCalcData; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This represents the relationship between a point (a source coordinate) | 
					
						
							|  |  |  |  * and the face-corner it's being bound to (from the target mesh). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note Some of these values could be de-duplicated however these are only | 
					
						
							|  |  |  |  * needed once when running bind, so optimizing this structure isn't a priority. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | typedef struct SDefBindPoly { | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Coordinates copied directly from the modifiers inptut. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float (*coords)[3]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Coordinates projected into 2D space using `normal`. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float (*coords_v2)[2]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** The point being queried projected into 2D space using `normal`. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float point_v2[2]; | 
					
						
							|  |  |  |   float weight_angular; | 
					
						
							|  |  |  |   float weight_dist_proj; | 
					
						
							|  |  |  |   float weight_dist; | 
					
						
							|  |  |  |   float weight; | 
					
						
							|  |  |  |   float scales[2]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Center of `coords` */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float centroid[3]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Center of `coords_v2` */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float centroid_v2[2]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * The calculated normal of coords (could be shared between faces). | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float normal[3]; | 
					
						
							|  |  |  |   float cent_edgemid_vecs_v2[2][2]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * The unsigned angle of this face-corner in `[0.0 .. PI]` range, | 
					
						
							| 
									
										
										
										
											2021-01-05 13:38:48 +11:00
										 |  |  |    * where a small value is a thin corner. PI is a straight line. | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |    * Take care dividing by this value as it can approach zero. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float edgemid_angle; | 
					
						
							|  |  |  |   float point_edgemid_angles[2]; | 
					
						
							|  |  |  |   float corner_edgemid_angles[2]; | 
					
						
							|  |  |  |   float dominant_angle_weight; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Index of the input polygon. */ | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint index; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** Number of vertices in this face. */ | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint numverts; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |    * This polygons loop-start. | 
					
						
							|  |  |  |    * \note that we could look this up from the polygon. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint loopstart; | 
					
						
							|  |  |  |   uint edge_inds[2]; | 
					
						
							|  |  |  |   uint edge_vert_inds[2]; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** The index of this corner in the face (starting at zero). */ | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint corner_ind; | 
					
						
							|  |  |  |   uint dominant_edge; | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /** When true `point_v2` is inside `coords_v2`. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   bool inside; | 
					
						
							|  |  |  | } SDefBindPoly; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct SDefBindWeightData { | 
					
						
							|  |  |  |   SDefBindPoly *bind_polys; | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint numpoly; | 
					
						
							|  |  |  |   uint numbinds; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } SDefBindWeightData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct SDefDeformData { | 
					
						
							|  |  |  |   const SDefVert *const bind_verts; | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   float (*const targetCos)[3]; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float (*const vertexCos)[3]; | 
					
						
							| 
									
										
										
										
											2021-01-06 12:47:01 +03:00
										 |  |  |   const MDeformVert *const dvert; | 
					
						
							|  |  |  |   int const defgrp_index; | 
					
						
							|  |  |  |   bool const invert_vgroup; | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   float const strength; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } SDefDeformData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Bind result values */ | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |   MOD_SDEF_BIND_RESULT_SUCCESS = 1, | 
					
						
							|  |  |  |   MOD_SDEF_BIND_RESULT_GENERIC_ERR = 0, | 
					
						
							|  |  |  |   MOD_SDEF_BIND_RESULT_MEM_ERR = -1, | 
					
						
							|  |  |  |   MOD_SDEF_BIND_RESULT_NONMANY_ERR = -2, | 
					
						
							|  |  |  |   MOD_SDEF_BIND_RESULT_CONCAVE_ERR = -3, | 
					
						
							|  |  |  |   MOD_SDEF_BIND_RESULT_OVERLAP_ERR = -4, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Infinite weight flags */ | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |   MOD_SDEF_INFINITE_WEIGHT_ANGULAR = (1 << 0), | 
					
						
							|  |  |  |   MOD_SDEF_INFINITE_WEIGHT_DIST_PROJ = (1 << 1), | 
					
						
							|  |  |  |   MOD_SDEF_INFINITE_WEIGHT_DIST = (1 << 2), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void initData(ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2020-10-01 09:38:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   BLI_assert(MEMCMP_STRUCT_AFTER_IS_ZERO(smd, modifier)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEMCPY_STRUCT_AFTER(smd, DNA_struct_default_get(SurfaceDeformModifierData), modifier); | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void requiredDataMask(Object *UNUSED(ob), | 
					
						
							|  |  |  |                              ModifierData *md, | 
					
						
							|  |  |  |                              CustomData_MeshMasks *r_cddata_masks) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Ask for vertex groups if we need them. */ | 
					
						
							|  |  |  |   if (smd->defgrp_name[0] != '\0') { | 
					
						
							|  |  |  |     r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void freeData(ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (smd->verts) { | 
					
						
							|  |  |  |     for (int i = 0; i < smd->numverts; i++) { | 
					
						
							|  |  |  |       if (smd->verts[i].binds) { | 
					
						
							|  |  |  |         for (int j = 0; j < smd->verts[i].numbinds; j++) { | 
					
						
							|  |  |  |           MEM_SAFE_FREE(smd->verts[i].binds[j].vert_inds); | 
					
						
							|  |  |  |           MEM_SAFE_FREE(smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-08 14:21:02 +02:00
										 |  |  |         MEM_SAFE_FREE(smd->verts[i].binds); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-08 14:21:02 +02:00
										 |  |  |     MEM_SAFE_FREE(smd->verts); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-04 12:45:30 +02:00
										 |  |  | static void copyData(const ModifierData *md, ModifierData *target, const int flag) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-05-08 15:04:10 +02:00
										 |  |  |   const SurfaceDeformModifierData *smd = (const SurfaceDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   SurfaceDeformModifierData *tsmd = (SurfaceDeformModifierData *)target; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-08 10:14:02 +02:00
										 |  |  |   BKE_modifier_copydata_generic(md, target, flag); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (smd->verts) { | 
					
						
							|  |  |  |     tsmd->verts = MEM_dupallocN(smd->verts); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     for (int i = 0; i < smd->numverts; i++) { | 
					
						
							|  |  |  |       if (smd->verts[i].binds) { | 
					
						
							|  |  |  |         tsmd->verts[i].binds = MEM_dupallocN(smd->verts[i].binds); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         for (int j = 0; j < smd->verts[i].numbinds; j++) { | 
					
						
							|  |  |  |           if (smd->verts[i].binds[j].vert_inds) { | 
					
						
							|  |  |  |             tsmd->verts[i].binds[j].vert_inds = MEM_dupallocN(smd->verts[i].binds[j].vert_inds); | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           if (smd->verts[i].binds[j].vert_weights) { | 
					
						
							|  |  |  |             tsmd->verts[i].binds[j].vert_weights = MEM_dupallocN( | 
					
						
							|  |  |  |                 smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-01 18:05:23 +02:00
										 |  |  | static void foreachIDLink(ModifierData *md, Object *ob, IDWalkFunc walk, void *userData) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-01 18:05:23 +02:00
										 |  |  |   walk(userData, ob, (ID **)&smd->target, IDWALK_NOP); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-22 12:54:06 +01:00
										 |  |  | static void updateDepsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  |   if (smd->target != NULL) { | 
					
						
							| 
									
										
										
										
											2018-02-22 12:54:06 +01:00
										 |  |  |     DEG_add_object_relation( | 
					
						
							|  |  |  |         ctx->node, smd->target, DEG_OB_COMP_GEOMETRY, "Surface Deform Modifier"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void freeAdjacencyMap(SDefAdjacencyArray *const vert_edges, | 
					
						
							|  |  |  |                              SDefAdjacency *const adj_ref, | 
					
						
							|  |  |  |                              SDefEdgePolys *const edge_polys) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   MEM_freeN(edge_polys); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEM_freeN(adj_ref); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEM_freeN(vert_edges); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-12 08:04:56 +02:00
										 |  |  | static int buildAdjacencyMap(const MPoly *poly, | 
					
						
							|  |  |  |                              const MEdge *edge, | 
					
						
							|  |  |  |                              const MLoop *const mloop, | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |                              const uint numpoly, | 
					
						
							|  |  |  |                              const uint numedges, | 
					
						
							| 
									
										
										
										
											2018-05-12 08:04:56 +02:00
										 |  |  |                              SDefAdjacencyArray *const vert_edges, | 
					
						
							|  |  |  |                              SDefAdjacency *adj, | 
					
						
							|  |  |  |                              SDefEdgePolys *const edge_polys) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   const MLoop *loop; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /* Find polygons adjacent to edges. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (int i = 0; i < numpoly; i++, poly++) { | 
					
						
							|  |  |  |     loop = &mloop[poly->loopstart]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     for (int j = 0; j < poly->totloop; j++, loop++) { | 
					
						
							|  |  |  |       if (edge_polys[loop->e].num == 0) { | 
					
						
							|  |  |  |         edge_polys[loop->e].polys[0] = i; | 
					
						
							|  |  |  |         edge_polys[loop->e].polys[1] = -1; | 
					
						
							|  |  |  |         edge_polys[loop->e].num++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (edge_polys[loop->e].num == 1) { | 
					
						
							|  |  |  |         edge_polys[loop->e].polys[1] = i; | 
					
						
							|  |  |  |         edge_polys[loop->e].num++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         return MOD_SDEF_BIND_RESULT_NONMANY_ERR; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   /* Find edges adjacent to vertices */ | 
					
						
							|  |  |  |   for (int i = 0; i < numedges; i++, edge++) { | 
					
						
							|  |  |  |     adj->next = vert_edges[edge->v1].first; | 
					
						
							|  |  |  |     adj->index = i; | 
					
						
							|  |  |  |     vert_edges[edge->v1].first = adj; | 
					
						
							|  |  |  |     vert_edges[edge->v1].num += edge_polys[i].num; | 
					
						
							|  |  |  |     adj++; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     adj->next = vert_edges[edge->v2].first; | 
					
						
							|  |  |  |     adj->index = i; | 
					
						
							|  |  |  |     vert_edges[edge->v2].first = adj; | 
					
						
							|  |  |  |     vert_edges[edge->v2].num += edge_polys[i].num; | 
					
						
							|  |  |  |     adj++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   return MOD_SDEF_BIND_RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  | BLI_INLINE void sortPolyVertsEdge(uint *indices, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |                                   const MLoop *const mloop, | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |                                   const uint edge, | 
					
						
							|  |  |  |                                   const uint num) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   bool found = false; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (int i = 0; i < num; i++) { | 
					
						
							|  |  |  |     if (mloop[i].e == edge) { | 
					
						
							|  |  |  |       found = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (found) { | 
					
						
							|  |  |  |       *indices = mloop[i].v; | 
					
						
							|  |  |  |       indices++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   /* Fill in remaining vertex indices that occur before the edge */ | 
					
						
							|  |  |  |   for (int i = 0; mloop[i].e != edge; i++) { | 
					
						
							|  |  |  |     *indices = mloop[i].v; | 
					
						
							|  |  |  |     indices++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  | BLI_INLINE void sortPolyVertsTri(uint *indices, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |                                  const MLoop *const mloop, | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |                                  const uint loopstart, | 
					
						
							|  |  |  |                                  const uint num) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   for (int i = loopstart; i < num; i++) { | 
					
						
							|  |  |  |     *indices = mloop[i].v; | 
					
						
							|  |  |  |     indices++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (int i = 0; i < loopstart; i++) { | 
					
						
							|  |  |  |     *indices = mloop[i].v; | 
					
						
							|  |  |  |     indices++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  | BLI_INLINE uint nearestVert(SDefBindCalcData *const data, const float point_co[3]) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-01-07 00:06:58 +11:00
										 |  |  |   BVHTreeNearest nearest = { | 
					
						
							|  |  |  |       .dist_sq = FLT_MAX, | 
					
						
							|  |  |  |       .index = -1, | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   const MPoly *poly; | 
					
						
							|  |  |  |   const MEdge *edge; | 
					
						
							|  |  |  |   const MLoop *loop; | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   float t_point[3]; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float max_dist = FLT_MAX; | 
					
						
							|  |  |  |   float dist; | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint index = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   mul_v3_m4v3(t_point, data->imat, point_co); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   BLI_bvhtree_find_nearest( | 
					
						
							|  |  |  |       data->treeData->tree, t_point, &nearest, data->treeData->nearest_callback, data->treeData); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   poly = &data->mpoly[data->looptri[nearest.index].poly]; | 
					
						
							|  |  |  |   loop = &data->mloop[poly->loopstart]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (int i = 0; i < poly->totloop; i++, loop++) { | 
					
						
							|  |  |  |     edge = &data->medge[loop->e]; | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |     dist = dist_squared_to_line_segment_v3( | 
					
						
							|  |  |  |         point_co, data->targetCos[edge->v1], data->targetCos[edge->v2]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     if (dist < max_dist) { | 
					
						
							|  |  |  |       max_dist = dist; | 
					
						
							|  |  |  |       index = loop->e; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   edge = &data->medge[index]; | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   if (len_squared_v3v3(point_co, data->targetCos[edge->v1]) < | 
					
						
							|  |  |  |       len_squared_v3v3(point_co, data->targetCos[edge->v2])) { | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     return edge->v1; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 12:40:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return edge->v2; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  | BLI_INLINE int isPolyValid(const float coords[][2], const uint nr) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   float prev_co[2]; | 
					
						
							|  |  |  |   float curr_vec[2], prev_vec[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!is_poly_convex_v2(coords, nr)) { | 
					
						
							|  |  |  |     return MOD_SDEF_BIND_RESULT_CONCAVE_ERR; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   copy_v2_v2(prev_co, coords[nr - 1]); | 
					
						
							|  |  |  |   sub_v2_v2v2(prev_vec, prev_co, coords[nr - 2]); | 
					
						
							| 
									
										
										
										
											2017-12-18 15:16:22 +01:00
										 |  |  |   normalize_v2(prev_vec); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (int i = 0; i < nr; i++) { | 
					
						
							|  |  |  |     sub_v2_v2v2(curr_vec, coords[i], prev_co); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-18 15:16:22 +01:00
										 |  |  |     const float curr_len = normalize_v2(curr_vec); | 
					
						
							|  |  |  |     if (curr_len < FLT_EPSILON) { | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       return MOD_SDEF_BIND_RESULT_OVERLAP_ERR; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (1.0f - dot_v2v2(prev_vec, curr_vec) < FLT_EPSILON) { | 
					
						
							|  |  |  |       return MOD_SDEF_BIND_RESULT_CONCAVE_ERR; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     copy_v2_v2(prev_co, coords[i]); | 
					
						
							|  |  |  |     copy_v2_v2(prev_vec, curr_vec); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return MOD_SDEF_BIND_RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void freeBindData(SDefBindWeightData *const bwdata) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   SDefBindPoly *bpoly = bwdata->bind_polys; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (bwdata->bind_polys) { | 
					
						
							|  |  |  |     for (int i = 0; i < bwdata->numpoly; bpoly++, i++) { | 
					
						
							|  |  |  |       MEM_SAFE_FREE(bpoly->coords); | 
					
						
							|  |  |  |       MEM_SAFE_FREE(bpoly->coords_v2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MEM_freeN(bwdata->bind_polys); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEM_freeN(bwdata); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-03 20:26:36 +11:00
										 |  |  | BLI_INLINE float computeAngularWeight(const float point_angle) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-03 20:26:36 +11:00
										 |  |  |   return sinf(point_angle * M_PI_2); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | BLI_INLINE SDefBindWeightData *computeBindWeights(SDefBindCalcData *const data, | 
					
						
							|  |  |  |                                                   const float point_co[3]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   const uint nearest = nearestVert(data, point_co); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   const SDefAdjacency *const vert_edges = data->vert_edges[nearest].first; | 
					
						
							|  |  |  |   const SDefEdgePolys *const edge_polys = data->edge_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   const SDefAdjacency *vedge; | 
					
						
							|  |  |  |   const MPoly *poly; | 
					
						
							|  |  |  |   const MLoop *loop; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   SDefBindWeightData *bwdata; | 
					
						
							|  |  |  |   SDefBindPoly *bpoly; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-07 22:36:11 +10:00
										 |  |  |   const float world[3] = {0.0f, 0.0f, 1.0f}; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float avg_point_dist = 0.0f; | 
					
						
							|  |  |  |   float tot_weight = 0.0f; | 
					
						
							|  |  |  |   int inf_weight_flags = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   bwdata = MEM_callocN(sizeof(*bwdata), "SDefBindWeightData"); | 
					
						
							|  |  |  |   if (bwdata == NULL) { | 
					
						
							|  |  |  |     data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   bwdata->numpoly = data->vert_edges[nearest].num / 2; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |   bpoly = MEM_calloc_arrayN(bwdata->numpoly, sizeof(*bpoly), "SDefBindPoly"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (bpoly == NULL) { | 
					
						
							|  |  |  |     freeBindData(bwdata); | 
					
						
							|  |  |  |     data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   bwdata->bind_polys = bpoly; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-01 07:40:07 +10:00
										 |  |  |   /* Loop over all adjacent edges,
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |    * and build the #SDefBindPoly data for each poly adjacent to those. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (vedge = vert_edges; vedge; vedge = vedge->next) { | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |     uint edge_ind = vedge->index; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     for (int i = 0; i < edge_polys[edge_ind].num; i++) { | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         bpoly = bwdata->bind_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         for (int j = 0; j < bwdata->numpoly; bpoly++, j++) { | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |           /* If coords isn't allocated, we have reached the first uninitialized `bpoly`. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           if ((bpoly->index == edge_polys[edge_ind].polys[i]) || (!bpoly->coords)) { | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Check if poly was already created by another edge or still has to be initialized */ | 
					
						
							|  |  |  |       if (!bpoly->coords) { | 
					
						
							|  |  |  |         float angle; | 
					
						
							|  |  |  |         float axis[3]; | 
					
						
							|  |  |  |         float tmp_vec_v2[2]; | 
					
						
							|  |  |  |         int is_poly_valid; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         bpoly->index = edge_polys[edge_ind].polys[i]; | 
					
						
							|  |  |  |         bpoly->coords = NULL; | 
					
						
							|  |  |  |         bpoly->coords_v2 = NULL; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         /* Copy poly data */ | 
					
						
							|  |  |  |         poly = &data->mpoly[bpoly->index]; | 
					
						
							|  |  |  |         loop = &data->mloop[poly->loopstart]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         bpoly->numverts = poly->totloop; | 
					
						
							|  |  |  |         bpoly->loopstart = poly->loopstart; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |         bpoly->coords = MEM_malloc_arrayN( | 
					
						
							|  |  |  |             poly->totloop, sizeof(*bpoly->coords), "SDefBindPolyCoords"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (bpoly->coords == NULL) { | 
					
						
							|  |  |  |           freeBindData(bwdata); | 
					
						
							|  |  |  |           data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |           return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |         bpoly->coords_v2 = MEM_malloc_arrayN( | 
					
						
							|  |  |  |             poly->totloop, sizeof(*bpoly->coords_v2), "SDefBindPolyCoords_v2"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (bpoly->coords_v2 == NULL) { | 
					
						
							|  |  |  |           freeBindData(bwdata); | 
					
						
							|  |  |  |           data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |           return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         for (int j = 0; j < poly->totloop; j++, loop++) { | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |           copy_v3_v3(bpoly->coords[j], data->targetCos[loop->v]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           /* Find corner and edge indices within poly loop array */ | 
					
						
							|  |  |  |           if (loop->v == nearest) { | 
					
						
							|  |  |  |             bpoly->corner_ind = j; | 
					
						
							|  |  |  |             bpoly->edge_vert_inds[0] = (j == 0) ? (poly->totloop - 1) : (j - 1); | 
					
						
							|  |  |  |             bpoly->edge_vert_inds[1] = (j == poly->totloop - 1) ? (0) : (j + 1); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |             bpoly->edge_inds[0] = data->mloop[poly->loopstart + bpoly->edge_vert_inds[0]].e; | 
					
						
							|  |  |  |             bpoly->edge_inds[1] = loop->e; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |         /* Compute polygons parametric data. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         mid_v3_v3_array(bpoly->centroid, bpoly->coords, poly->totloop); | 
					
						
							|  |  |  |         normal_poly_v3(bpoly->normal, bpoly->coords, poly->totloop); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         /* Compute poly skew angle and axis */ | 
					
						
							|  |  |  |         angle = angle_normalized_v3v3(bpoly->normal, world); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         cross_v3_v3v3(axis, bpoly->normal, world); | 
					
						
							|  |  |  |         normalize_v3(axis); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |         /* Map coords onto 2d normal plane. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         map_to_plane_axis_angle_v2_v3v3fl(bpoly->point_v2, point_co, axis, angle); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         zero_v2(bpoly->centroid_v2); | 
					
						
							|  |  |  |         for (int j = 0; j < poly->totloop; j++) { | 
					
						
							|  |  |  |           map_to_plane_axis_angle_v2_v3v3fl(bpoly->coords_v2[j], bpoly->coords[j], axis, angle); | 
					
						
							|  |  |  |           madd_v2_v2fl(bpoly->centroid_v2, bpoly->coords_v2[j], 1.0f / poly->totloop); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         is_poly_valid = isPolyValid(bpoly->coords_v2, poly->totloop); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (is_poly_valid != MOD_SDEF_BIND_RESULT_SUCCESS) { | 
					
						
							|  |  |  |           freeBindData(bwdata); | 
					
						
							|  |  |  |           data->success = is_poly_valid; | 
					
						
							|  |  |  |           return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         bpoly->inside = isect_point_poly_v2( | 
					
						
							|  |  |  |             bpoly->point_v2, bpoly->coords_v2, poly->totloop, false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         /* Initialize weight components */ | 
					
						
							|  |  |  |         bpoly->weight_angular = 1.0f; | 
					
						
							|  |  |  |         bpoly->weight_dist_proj = len_v2v2(bpoly->centroid_v2, bpoly->point_v2); | 
					
						
							|  |  |  |         bpoly->weight_dist = len_v3v3(bpoly->centroid, point_co); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         avg_point_dist += bpoly->weight_dist; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         /* Compute centroid to mid-edge vectors */ | 
					
						
							|  |  |  |         mid_v2_v2v2(bpoly->cent_edgemid_vecs_v2[0], | 
					
						
							|  |  |  |                     bpoly->coords_v2[bpoly->edge_vert_inds[0]], | 
					
						
							|  |  |  |                     bpoly->coords_v2[bpoly->corner_ind]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         mid_v2_v2v2(bpoly->cent_edgemid_vecs_v2[1], | 
					
						
							|  |  |  |                     bpoly->coords_v2[bpoly->edge_vert_inds[1]], | 
					
						
							|  |  |  |                     bpoly->coords_v2[bpoly->corner_ind]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         sub_v2_v2(bpoly->cent_edgemid_vecs_v2[0], bpoly->centroid_v2); | 
					
						
							|  |  |  |         sub_v2_v2(bpoly->cent_edgemid_vecs_v2[1], bpoly->centroid_v2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         /* Compute poly scales with respect to mid-edges, and normalize the vectors */ | 
					
						
							|  |  |  |         bpoly->scales[0] = normalize_v2(bpoly->cent_edgemid_vecs_v2[0]); | 
					
						
							|  |  |  |         bpoly->scales[1] = normalize_v2(bpoly->cent_edgemid_vecs_v2[1]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         /* Compute the required polygon angles */ | 
					
						
							|  |  |  |         bpoly->edgemid_angle = angle_normalized_v2v2(bpoly->cent_edgemid_vecs_v2[0], | 
					
						
							|  |  |  |                                                      bpoly->cent_edgemid_vecs_v2[1]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         sub_v2_v2v2(tmp_vec_v2, bpoly->coords_v2[bpoly->corner_ind], bpoly->centroid_v2); | 
					
						
							|  |  |  |         normalize_v2(tmp_vec_v2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         bpoly->corner_edgemid_angles[0] = angle_normalized_v2v2(tmp_vec_v2, | 
					
						
							|  |  |  |                                                                 bpoly->cent_edgemid_vecs_v2[0]); | 
					
						
							|  |  |  |         bpoly->corner_edgemid_angles[1] = angle_normalized_v2v2(tmp_vec_v2, | 
					
						
							|  |  |  |                                                                 bpoly->cent_edgemid_vecs_v2[1]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |         /* Check for infinite weights, and compute angular data otherwise. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (bpoly->weight_dist < FLT_EPSILON) { | 
					
						
							|  |  |  |           inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_DIST_PROJ; | 
					
						
							|  |  |  |           inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_DIST; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (bpoly->weight_dist_proj < FLT_EPSILON) { | 
					
						
							|  |  |  |           inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_DIST_PROJ; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           float cent_point_vec[2]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sub_v2_v2v2(cent_point_vec, bpoly->point_v2, bpoly->centroid_v2); | 
					
						
							|  |  |  |           normalize_v2(cent_point_vec); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           bpoly->point_edgemid_angles[0] = angle_normalized_v2v2(cent_point_vec, | 
					
						
							|  |  |  |                                                                  bpoly->cent_edgemid_vecs_v2[0]); | 
					
						
							|  |  |  |           bpoly->point_edgemid_angles[1] = angle_normalized_v2v2(cent_point_vec, | 
					
						
							|  |  |  |                                                                  bpoly->cent_edgemid_vecs_v2[1]); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   avg_point_dist /= bwdata->numpoly; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   /* If weights 1 and 2 are not infinite, loop over all adjacent edges again,
 | 
					
						
							|  |  |  |    * and build adjacency dependent angle data (depends on all polygons having been computed) */ | 
					
						
							|  |  |  |   if (!inf_weight_flags) { | 
					
						
							|  |  |  |     for (vedge = vert_edges; vedge; vedge = vedge->next) { | 
					
						
							|  |  |  |       SDefBindPoly *bpolys[2]; | 
					
						
							|  |  |  |       const SDefEdgePolys *epolys; | 
					
						
							|  |  |  |       float ang_weights[2]; | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |       uint edge_ind = vedge->index; | 
					
						
							|  |  |  |       uint edge_on_poly[2]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       epolys = &edge_polys[edge_ind]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Find bind polys corresponding to the edge's adjacent polys */ | 
					
						
							|  |  |  |       bpoly = bwdata->bind_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       for (int i = 0, j = 0; (i < bwdata->numpoly) && (j < epolys->num); bpoly++, i++) { | 
					
						
							|  |  |  |         if (ELEM(bpoly->index, epolys->polys[0], epolys->polys[1])) { | 
					
						
							|  |  |  |           bpolys[j] = bpoly; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           if (bpoly->edge_inds[0] == edge_ind) { | 
					
						
							|  |  |  |             edge_on_poly[j] = 0; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           else { | 
					
						
							|  |  |  |             edge_on_poly[j] = 1; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           j++; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Compute angular weight component */ | 
					
						
							|  |  |  |       if (epolys->num == 1) { | 
					
						
							| 
									
										
										
										
											2020-11-03 20:26:36 +11:00
										 |  |  |         ang_weights[0] = computeAngularWeight(bpolys[0]->point_edgemid_angles[edge_on_poly[0]]); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         bpolys[0]->weight_angular *= ang_weights[0] * ang_weights[0]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (epolys->num == 2) { | 
					
						
							| 
									
										
										
										
											2020-11-03 20:26:36 +11:00
										 |  |  |         ang_weights[0] = computeAngularWeight(bpolys[0]->point_edgemid_angles[edge_on_poly[0]]); | 
					
						
							|  |  |  |         ang_weights[1] = computeAngularWeight(bpolys[1]->point_edgemid_angles[edge_on_poly[1]]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         bpolys[0]->weight_angular *= ang_weights[0] * ang_weights[1]; | 
					
						
							|  |  |  |         bpolys[1]->weight_angular *= ang_weights[0] * ang_weights[1]; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /* Compute scaling and falloff:
 | 
					
						
							|  |  |  |    * - Scale all weights if no infinite weight is found. | 
					
						
							|  |  |  |    * - Scale only un-projected weight if projected weight is infinite. | 
					
						
							|  |  |  |    * - Scale none if both are infinite. */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (!inf_weight_flags) { | 
					
						
							|  |  |  |     bpoly = bwdata->bind_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     for (int i = 0; i < bwdata->numpoly; bpoly++, i++) { | 
					
						
							|  |  |  |       float corner_angle_weights[2]; | 
					
						
							|  |  |  |       float scale_weight, sqr, inv_sqr; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       corner_angle_weights[0] = bpoly->point_edgemid_angles[0] / bpoly->corner_edgemid_angles[0]; | 
					
						
							|  |  |  |       corner_angle_weights[1] = bpoly->point_edgemid_angles[1] / bpoly->corner_edgemid_angles[1]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       if (isnan(corner_angle_weights[0]) || isnan(corner_angle_weights[1])) { | 
					
						
							|  |  |  |         freeBindData(bwdata); | 
					
						
							|  |  |  |         data->success = MOD_SDEF_BIND_RESULT_GENERIC_ERR; | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Find which edge the point is closer to */ | 
					
						
							|  |  |  |       if (corner_angle_weights[0] < corner_angle_weights[1]) { | 
					
						
							|  |  |  |         bpoly->dominant_edge = 0; | 
					
						
							|  |  |  |         bpoly->dominant_angle_weight = corner_angle_weights[0]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         bpoly->dominant_edge = 1; | 
					
						
							|  |  |  |         bpoly->dominant_angle_weight = corner_angle_weights[1]; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       bpoly->dominant_angle_weight = sinf(bpoly->dominant_angle_weight * M_PI_2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Compute quadratic angular scale interpolation weight */ | 
					
						
							| 
									
										
										
										
											2020-11-03 20:26:36 +11:00
										 |  |  |       { | 
					
						
							|  |  |  |         const float edge_angle_a = bpoly->point_edgemid_angles[bpoly->dominant_edge]; | 
					
						
							|  |  |  |         const float edge_angle_b = bpoly->point_edgemid_angles[!bpoly->dominant_edge]; | 
					
						
							|  |  |  |         /* Clamp so skinny faces with near zero `edgemid_angle`
 | 
					
						
							|  |  |  |          * won't cause numeric problems. see T81988. */ | 
					
						
							|  |  |  |         scale_weight = edge_angle_a / max_ff(edge_angle_a, bpoly->edgemid_angle); | 
					
						
							|  |  |  |         scale_weight /= scale_weight + (edge_angle_b / max_ff(edge_angle_b, bpoly->edgemid_angle)); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       sqr = scale_weight * scale_weight; | 
					
						
							|  |  |  |       inv_sqr = 1.0f - scale_weight; | 
					
						
							|  |  |  |       inv_sqr *= inv_sqr; | 
					
						
							|  |  |  |       scale_weight = sqr / (sqr + inv_sqr); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Compute interpolated scale (no longer need the individual scales,
 | 
					
						
							|  |  |  |        * so simply storing the result over the scale in index zero) */ | 
					
						
							|  |  |  |       bpoly->scales[0] = bpoly->scales[bpoly->dominant_edge] * (1.0f - scale_weight) + | 
					
						
							|  |  |  |                          bpoly->scales[!bpoly->dominant_edge] * scale_weight; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Scale the point distance weights, and introduce falloff */ | 
					
						
							|  |  |  |       bpoly->weight_dist_proj /= bpoly->scales[0]; | 
					
						
							|  |  |  |       bpoly->weight_dist_proj = powf(bpoly->weight_dist_proj, data->falloff); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       bpoly->weight_dist /= avg_point_dist; | 
					
						
							|  |  |  |       bpoly->weight_dist = powf(bpoly->weight_dist, data->falloff); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Re-check for infinite weights, now that all scalings and interpolations are computed */ | 
					
						
							|  |  |  |       if (bpoly->weight_dist < FLT_EPSILON) { | 
					
						
							|  |  |  |         inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_DIST_PROJ; | 
					
						
							|  |  |  |         inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_DIST; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (bpoly->weight_dist_proj < FLT_EPSILON) { | 
					
						
							|  |  |  |         inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_DIST_PROJ; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (bpoly->weight_angular < FLT_EPSILON) { | 
					
						
							|  |  |  |         inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_ANGULAR; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   else if (!(inf_weight_flags & MOD_SDEF_INFINITE_WEIGHT_DIST)) { | 
					
						
							|  |  |  |     bpoly = bwdata->bind_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     for (int i = 0; i < bwdata->numpoly; bpoly++, i++) { | 
					
						
							|  |  |  |       /* Scale the point distance weight by average point distance, and introduce falloff */ | 
					
						
							|  |  |  |       bpoly->weight_dist /= avg_point_dist; | 
					
						
							|  |  |  |       bpoly->weight_dist = powf(bpoly->weight_dist, data->falloff); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* Re-check for infinite weights, now that all scalings and interpolations are computed */ | 
					
						
							|  |  |  |       if (bpoly->weight_dist < FLT_EPSILON) { | 
					
						
							|  |  |  |         inf_weight_flags |= MOD_SDEF_INFINITE_WEIGHT_DIST; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   /* Final loop, to compute actual weights */ | 
					
						
							|  |  |  |   bpoly = bwdata->bind_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (int i = 0; i < bwdata->numpoly; bpoly++, i++) { | 
					
						
							|  |  |  |     /* Weight computation from components */ | 
					
						
							|  |  |  |     if (inf_weight_flags & MOD_SDEF_INFINITE_WEIGHT_DIST) { | 
					
						
							|  |  |  |       bpoly->weight = bpoly->weight_dist < FLT_EPSILON ? 1.0f : 0.0f; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (inf_weight_flags & MOD_SDEF_INFINITE_WEIGHT_DIST_PROJ) { | 
					
						
							|  |  |  |       bpoly->weight = bpoly->weight_dist_proj < FLT_EPSILON ? 1.0f / bpoly->weight_dist : 0.0f; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (inf_weight_flags & MOD_SDEF_INFINITE_WEIGHT_ANGULAR) { | 
					
						
							|  |  |  |       bpoly->weight = bpoly->weight_angular < FLT_EPSILON ? | 
					
						
							|  |  |  |                           1.0f / bpoly->weight_dist_proj / bpoly->weight_dist : | 
					
						
							|  |  |  |                           0.0f; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       bpoly->weight = 1.0f / bpoly->weight_angular / bpoly->weight_dist_proj / bpoly->weight_dist; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-03 20:26:36 +11:00
										 |  |  |     /* Apply after other kinds of scaling so the faces corner angle is always
 | 
					
						
							|  |  |  |      * scaled in a uniform way, preventing heavily sub-divided triangle fans | 
					
						
							|  |  |  |      * from having a lop-sided influence on the weighting, see T81988. */ | 
					
						
							|  |  |  |     bpoly->weight *= bpoly->edgemid_angle / M_PI; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     tot_weight += bpoly->weight; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   bpoly = bwdata->bind_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (int i = 0; i < bwdata->numpoly; bpoly++, i++) { | 
					
						
							|  |  |  |     bpoly->weight /= tot_weight; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* Evaluate if this poly is relevant to bind */ | 
					
						
							|  |  |  |     /* Even though the weights should add up to 1.0,
 | 
					
						
							|  |  |  |      * the losses of weights smaller than epsilon here | 
					
						
							|  |  |  |      * should be negligible... */ | 
					
						
							|  |  |  |     if (bpoly->weight >= FLT_EPSILON) { | 
					
						
							|  |  |  |       if (bpoly->inside) { | 
					
						
							|  |  |  |         bwdata->numbinds += 1; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         if (bpoly->dominant_angle_weight < FLT_EPSILON || | 
					
						
							|  |  |  |             1.0f - bpoly->dominant_angle_weight < FLT_EPSILON) { | 
					
						
							|  |  |  |           bwdata->numbinds += 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           bwdata->numbinds += 2; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   return bwdata; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | BLI_INLINE float computeNormalDisplacement(const float point_co[3], | 
					
						
							|  |  |  |                                            const float point_co_proj[3], | 
					
						
							|  |  |  |                                            const float normal[3]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   float disp_vec[3]; | 
					
						
							|  |  |  |   float normal_dist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sub_v3_v3v3(disp_vec, point_co, point_co_proj); | 
					
						
							|  |  |  |   normal_dist = len_v3(disp_vec); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (dot_v3v3(disp_vec, normal) < 0) { | 
					
						
							|  |  |  |     normal_dist *= -1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return normal_dist; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-10 12:49:51 +01:00
										 |  |  | static void bindVert(void *__restrict userdata, | 
					
						
							|  |  |  |                      const int index, | 
					
						
							| 
									
										
										
										
											2019-07-30 14:56:47 +02:00
										 |  |  |                      const TaskParallelTLS *__restrict UNUSED(tls)) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   SDefBindCalcData *const data = (SDefBindCalcData *)userdata; | 
					
						
							|  |  |  |   float point_co[3]; | 
					
						
							|  |  |  |   float point_co_proj[3]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   SDefBindWeightData *bwdata; | 
					
						
							|  |  |  |   SDefVert *sdvert = data->bind_verts + index; | 
					
						
							|  |  |  |   SDefBindPoly *bpoly; | 
					
						
							|  |  |  |   SDefBind *sdbind; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (data->success != MOD_SDEF_BIND_RESULT_SUCCESS) { | 
					
						
							|  |  |  |     sdvert->binds = NULL; | 
					
						
							|  |  |  |     sdvert->numbinds = 0; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   copy_v3_v3(point_co, data->vertexCos[index]); | 
					
						
							|  |  |  |   bwdata = computeBindWeights(data, point_co); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (bwdata == NULL) { | 
					
						
							|  |  |  |     sdvert->binds = NULL; | 
					
						
							|  |  |  |     sdvert->numbinds = 0; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |   sdvert->binds = MEM_calloc_arrayN(bwdata->numbinds, sizeof(*sdvert->binds), "SDefVertBindData"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (sdvert->binds == NULL) { | 
					
						
							|  |  |  |     data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |     sdvert->numbinds = 0; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   sdvert->numbinds = bwdata->numbinds; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   sdbind = sdvert->binds; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   bpoly = bwdata->bind_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   for (int i = 0; i < bwdata->numbinds; bpoly++) { | 
					
						
							|  |  |  |     if (bpoly->weight >= FLT_EPSILON) { | 
					
						
							|  |  |  |       if (bpoly->inside) { | 
					
						
							|  |  |  |         const MLoop *loop = &data->mloop[bpoly->loopstart]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         sdbind->influence = bpoly->weight; | 
					
						
							|  |  |  |         sdbind->numverts = bpoly->numverts; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         sdbind->mode = MOD_SDEF_MODE_NGON; | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |         sdbind->vert_weights = MEM_malloc_arrayN( | 
					
						
							|  |  |  |             bpoly->numverts, sizeof(*sdbind->vert_weights), "SDefNgonVertWeights"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (sdbind->vert_weights == NULL) { | 
					
						
							|  |  |  |           data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |         sdbind->vert_inds = MEM_malloc_arrayN( | 
					
						
							|  |  |  |             bpoly->numverts, sizeof(*sdbind->vert_inds), "SDefNgonVertInds"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (sdbind->vert_inds == NULL) { | 
					
						
							|  |  |  |           data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         interp_weights_poly_v2( | 
					
						
							|  |  |  |             sdbind->vert_weights, bpoly->coords_v2, bpoly->numverts, bpoly->point_v2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |         /* Re-project vert based on weights and original poly verts,
 | 
					
						
							| 
									
										
										
										
											2019-05-01 07:40:07 +10:00
										 |  |  |          * to reintroduce poly non-planarity */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         zero_v3(point_co_proj); | 
					
						
							|  |  |  |         for (int j = 0; j < bpoly->numverts; j++, loop++) { | 
					
						
							|  |  |  |           madd_v3_v3fl(point_co_proj, bpoly->coords[j], sdbind->vert_weights[j]); | 
					
						
							|  |  |  |           sdbind->vert_inds[j] = loop->v; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         sdbind->normal_dist = computeNormalDisplacement(point_co, point_co_proj, bpoly->normal); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         sdbind++; | 
					
						
							|  |  |  |         i++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         float tmp_vec[3]; | 
					
						
							|  |  |  |         float cent[3], norm[3]; | 
					
						
							|  |  |  |         float v1[3], v2[3], v3[3]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (1.0f - bpoly->dominant_angle_weight >= FLT_EPSILON) { | 
					
						
							|  |  |  |           sdbind->influence = bpoly->weight * (1.0f - bpoly->dominant_angle_weight); | 
					
						
							|  |  |  |           sdbind->numverts = bpoly->numverts; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sdbind->mode = MOD_SDEF_MODE_CENTROID; | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |           sdbind->vert_weights = MEM_malloc_arrayN( | 
					
						
							|  |  |  |               3, sizeof(*sdbind->vert_weights), "SDefCentVertWeights"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           if (sdbind->vert_weights == NULL) { | 
					
						
							|  |  |  |             data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |           sdbind->vert_inds = MEM_malloc_arrayN( | 
					
						
							|  |  |  |               bpoly->numverts, sizeof(*sdbind->vert_inds), "SDefCentVertInds"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           if (sdbind->vert_inds == NULL) { | 
					
						
							|  |  |  |             data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sortPolyVertsEdge(sdbind->vert_inds, | 
					
						
							|  |  |  |                             &data->mloop[bpoly->loopstart], | 
					
						
							|  |  |  |                             bpoly->edge_inds[bpoly->dominant_edge], | 
					
						
							|  |  |  |                             bpoly->numverts); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |           copy_v3_v3(v1, data->targetCos[sdbind->vert_inds[0]]); | 
					
						
							|  |  |  |           copy_v3_v3(v2, data->targetCos[sdbind->vert_inds[1]]); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           copy_v3_v3(v3, bpoly->centroid); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           mid_v3_v3v3v3(cent, v1, v2, v3); | 
					
						
							|  |  |  |           normal_tri_v3(norm, v1, v2, v3); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           add_v3_v3v3(tmp_vec, point_co, bpoly->normal); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           /* We are sure the line is not parallel to the plane.
 | 
					
						
							|  |  |  |            * Checking return value just to avoid warning... */ | 
					
						
							|  |  |  |           if (!isect_line_plane_v3(point_co_proj, point_co, tmp_vec, cent, norm)) { | 
					
						
							|  |  |  |             BLI_assert(false); | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           interp_weights_tri_v3(sdbind->vert_weights, v1, v2, v3, point_co_proj); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sdbind->normal_dist = computeNormalDisplacement(point_co, point_co_proj, bpoly->normal); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sdbind++; | 
					
						
							|  |  |  |           i++; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         if (bpoly->dominant_angle_weight >= FLT_EPSILON) { | 
					
						
							|  |  |  |           sdbind->influence = bpoly->weight * bpoly->dominant_angle_weight; | 
					
						
							|  |  |  |           sdbind->numverts = bpoly->numverts; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sdbind->mode = MOD_SDEF_MODE_LOOPTRI; | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |           sdbind->vert_weights = MEM_malloc_arrayN( | 
					
						
							|  |  |  |               3, sizeof(*sdbind->vert_weights), "SDefTriVertWeights"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           if (sdbind->vert_weights == NULL) { | 
					
						
							|  |  |  |             data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |           sdbind->vert_inds = MEM_malloc_arrayN( | 
					
						
							|  |  |  |               bpoly->numverts, sizeof(*sdbind->vert_inds), "SDefTriVertInds"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           if (sdbind->vert_inds == NULL) { | 
					
						
							|  |  |  |             data->success = MOD_SDEF_BIND_RESULT_MEM_ERR; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sortPolyVertsTri(sdbind->vert_inds, | 
					
						
							|  |  |  |                            &data->mloop[bpoly->loopstart], | 
					
						
							|  |  |  |                            bpoly->edge_vert_inds[0], | 
					
						
							|  |  |  |                            bpoly->numverts); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |           copy_v3_v3(v1, data->targetCos[sdbind->vert_inds[0]]); | 
					
						
							|  |  |  |           copy_v3_v3(v2, data->targetCos[sdbind->vert_inds[1]]); | 
					
						
							|  |  |  |           copy_v3_v3(v3, data->targetCos[sdbind->vert_inds[2]]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           mid_v3_v3v3v3(cent, v1, v2, v3); | 
					
						
							|  |  |  |           normal_tri_v3(norm, v1, v2, v3); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           add_v3_v3v3(tmp_vec, point_co, bpoly->normal); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           /* We are sure the line is not parallel to the plane.
 | 
					
						
							|  |  |  |            * Checking return value just to avoid warning... */ | 
					
						
							|  |  |  |           if (!isect_line_plane_v3(point_co_proj, point_co, tmp_vec, cent, norm)) { | 
					
						
							|  |  |  |             BLI_assert(false); | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           interp_weights_tri_v3(sdbind->vert_weights, v1, v2, v3, point_co_proj); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sdbind->normal_dist = computeNormalDisplacement(point_co, point_co_proj, bpoly->normal); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |           sdbind++; | 
					
						
							|  |  |  |           i++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   freeBindData(bwdata); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  | static bool surfacedeformBind(Object *ob, | 
					
						
							|  |  |  |                               SurfaceDeformModifierData *smd_orig, | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |                               SurfaceDeformModifierData *smd_eval, | 
					
						
							| 
									
										
										
										
											2018-05-12 08:04:56 +02:00
										 |  |  |                               float (*vertexCos)[3], | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |                               uint numverts, | 
					
						
							|  |  |  |                               uint tnumpoly, | 
					
						
							|  |  |  |                               uint tnumverts, | 
					
						
							| 
									
										
										
										
											2018-05-12 08:21:07 +02:00
										 |  |  |                               Mesh *target) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   BVHTreeFromMesh treeData = {NULL}; | 
					
						
							| 
									
										
										
										
											2018-05-09 11:33:32 +02:00
										 |  |  |   const MVert *mvert = target->mvert; | 
					
						
							|  |  |  |   const MPoly *mpoly = target->mpoly; | 
					
						
							|  |  |  |   const MEdge *medge = target->medge; | 
					
						
							|  |  |  |   const MLoop *mloop = target->mloop; | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint tnumedges = target->totedge; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   int adj_result; | 
					
						
							|  |  |  |   SDefAdjacencyArray *vert_edges; | 
					
						
							|  |  |  |   SDefAdjacency *adj_array; | 
					
						
							|  |  |  |   SDefEdgePolys *edge_polys; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |   vert_edges = MEM_calloc_arrayN(tnumverts, sizeof(*vert_edges), "SDefVertEdgeMap"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (vert_edges == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Out of memory"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |   adj_array = MEM_malloc_arrayN(tnumedges, 2 * sizeof(*adj_array), "SDefVertEdge"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (adj_array == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Out of memory"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     MEM_freeN(vert_edges); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |   edge_polys = MEM_calloc_arrayN(tnumedges, sizeof(*edge_polys), "SDefEdgeFaceMap"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (edge_polys == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Out of memory"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     MEM_freeN(vert_edges); | 
					
						
							|  |  |  |     MEM_freeN(adj_array); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |   smd_orig->verts = MEM_malloc_arrayN(numverts, sizeof(*smd_orig->verts), "SDefBindVerts"); | 
					
						
							|  |  |  |   if (smd_orig->verts == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Out of memory"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     freeAdjacencyMap(vert_edges, adj_array, edge_polys); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-09 11:33:32 +02:00
										 |  |  |   BKE_bvhtree_from_mesh_get(&treeData, target, BVHTREE_FROM_LOOPTRI, 2); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (treeData.tree == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Out of memory"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     freeAdjacencyMap(vert_edges, adj_array, edge_polys); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     MEM_freeN(smd_orig->verts); | 
					
						
							|  |  |  |     smd_orig->verts = NULL; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   adj_result = buildAdjacencyMap( | 
					
						
							|  |  |  |       mpoly, medge, mloop, tnumpoly, tnumedges, vert_edges, adj_array, edge_polys); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (adj_result == MOD_SDEF_BIND_RESULT_NONMANY_ERR) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error( | 
					
						
							|  |  |  |         ob, (ModifierData *)smd_eval, "Target has edges with more than two polygons"); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     freeAdjacencyMap(vert_edges, adj_array, edge_polys); | 
					
						
							|  |  |  |     free_bvhtree_from_mesh(&treeData); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     MEM_freeN(smd_orig->verts); | 
					
						
							|  |  |  |     smd_orig->verts = NULL; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |   smd_orig->numverts = numverts; | 
					
						
							|  |  |  |   smd_orig->numpoly = tnumpoly; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-07 00:06:58 +11:00
										 |  |  |   SDefBindCalcData data = { | 
					
						
							|  |  |  |       .treeData = &treeData, | 
					
						
							|  |  |  |       .vert_edges = vert_edges, | 
					
						
							|  |  |  |       .edge_polys = edge_polys, | 
					
						
							|  |  |  |       .mpoly = mpoly, | 
					
						
							|  |  |  |       .medge = medge, | 
					
						
							|  |  |  |       .mloop = mloop, | 
					
						
							|  |  |  |       .looptri = BKE_mesh_runtime_looptri_ensure(target), | 
					
						
							|  |  |  |       .targetCos = MEM_malloc_arrayN(tnumverts, sizeof(float[3]), "SDefTargetBindVertArray"), | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |       .bind_verts = smd_orig->verts, | 
					
						
							| 
									
										
										
										
											2019-01-07 00:06:58 +11:00
										 |  |  |       .vertexCos = vertexCos, | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |       .falloff = smd_orig->falloff, | 
					
						
							| 
									
										
										
										
											2019-01-07 00:06:58 +11:00
										 |  |  |       .success = MOD_SDEF_BIND_RESULT_SUCCESS, | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   if (data.targetCos == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Out of memory"); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     freeData((ModifierData *)smd_orig); | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |   invert_m4_m4(data.imat, smd_orig->mat); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   for (int i = 0; i < tnumverts; i++) { | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     mul_v3_m4v3(data.targetCos[i], smd_orig->mat, mvert[i].co); | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-30 14:56:47 +02:00
										 |  |  |   TaskParallelSettings settings; | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:48 +01:00
										 |  |  |   BLI_parallel_range_settings_defaults(&settings); | 
					
						
							|  |  |  |   settings.use_threading = (numverts > 10000); | 
					
						
							|  |  |  |   BLI_task_parallel_range(0, numverts, &data, bindVert, &settings); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   MEM_freeN(data.targetCos); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (data.success == MOD_SDEF_BIND_RESULT_MEM_ERR) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Out of memory"); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     freeData((ModifierData *)smd_orig); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							|  |  |  |   else if (data.success == MOD_SDEF_BIND_RESULT_NONMANY_ERR) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error( | 
					
						
							|  |  |  |         ob, (ModifierData *)smd_eval, "Target has edges with more than two polygons"); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     freeData((ModifierData *)smd_orig); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							|  |  |  |   else if (data.success == MOD_SDEF_BIND_RESULT_CONCAVE_ERR) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Target contains concave polygons"); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     freeData((ModifierData *)smd_orig); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							|  |  |  |   else if (data.success == MOD_SDEF_BIND_RESULT_OVERLAP_ERR) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Target contains overlapping vertices"); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     freeData((ModifierData *)smd_orig); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							|  |  |  |   else if (data.success == MOD_SDEF_BIND_RESULT_GENERIC_ERR) { | 
					
						
							|  |  |  |     /* I know this message is vague, but I could not think of a way
 | 
					
						
							| 
									
										
										
										
											2018-09-03 16:49:08 +02:00
										 |  |  |      * to explain this with a reasonably sized message. | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |      * Though it shouldn't really matter all that much, | 
					
						
							|  |  |  |      * because this is very unlikely to occur */ | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, (ModifierData *)smd_eval, "Target contains invalid polygons"); | 
					
						
							| 
									
										
										
										
											2020-01-03 22:10:15 +01:00
										 |  |  |     freeData((ModifierData *)smd_orig); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   freeAdjacencyMap(vert_edges, adj_array, edge_polys); | 
					
						
							|  |  |  |   free_bvhtree_from_mesh(&treeData); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   return data.success == 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-10 12:49:51 +01:00
										 |  |  | static void deformVert(void *__restrict userdata, | 
					
						
							|  |  |  |                        const int index, | 
					
						
							| 
									
										
										
										
											2019-07-30 14:56:47 +02:00
										 |  |  |                        const TaskParallelTLS *__restrict UNUSED(tls)) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   const SDefDeformData *const data = (SDefDeformData *)userdata; | 
					
						
							|  |  |  |   const SDefBind *sdbind = data->bind_verts[index].binds; | 
					
						
							| 
									
										
										
										
											2020-03-13 08:22:09 +01:00
										 |  |  |   const int num_binds = data->bind_verts[index].numbinds; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   float *const vertexCos = data->vertexCos[index]; | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   float norm[3], temp[3], offset[3]; | 
					
						
							| 
									
										
										
										
											2021-01-06 12:47:01 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Retrieve the value of the weight vertex group if specified. */ | 
					
						
							|  |  |  |   float weight = 1.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (data->dvert && data->defgrp_index != -1) { | 
					
						
							|  |  |  |     weight = BKE_defvert_find_weight(&data->dvert[index], data->defgrp_index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (data->invert_vgroup) { | 
					
						
							|  |  |  |       weight = 1.0f - weight; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   /* Check if this vertex will be deformed. If it is not deformed we return and avoid
 | 
					
						
							| 
									
										
										
										
											2020-03-29 16:33:51 +11:00
										 |  |  |    * unnecessary calculations. */ | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   if (weight == 0.0f) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   zero_v3(offset); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 08:22:09 +01:00
										 |  |  |   /* Allocate a `coords_buffer` that fits all the temp-data. */ | 
					
						
							|  |  |  |   int max_verts = 0; | 
					
						
							|  |  |  |   for (int j = 0; j < num_binds; j++) { | 
					
						
							|  |  |  |     max_verts = MAX2(max_verts, sdbind[j].numverts); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const bool big_buffer = max_verts > 256; | 
					
						
							|  |  |  |   float(*coords_buffer)[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (UNLIKELY(big_buffer)) { | 
					
						
							|  |  |  |     coords_buffer = MEM_malloc_arrayN(max_verts, sizeof(*coords_buffer), __func__); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     coords_buffer = BLI_array_alloca(coords_buffer, max_verts); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 08:22:09 +01:00
										 |  |  |   for (int j = 0; j < num_binds; j++, sdbind++) { | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     for (int k = 0; k < sdbind->numverts; k++) { | 
					
						
							| 
									
										
										
										
											2020-03-13 08:22:09 +01:00
										 |  |  |       copy_v3_v3(coords_buffer[k], data->targetCos[sdbind->vert_inds[k]]); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 08:22:09 +01:00
										 |  |  |     normal_poly_v3(norm, coords_buffer, sdbind->numverts); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     zero_v3(temp); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  |     switch (sdbind->mode) { | 
					
						
							|  |  |  |       /* ---------- looptri mode ---------- */ | 
					
						
							|  |  |  |       case MOD_SDEF_MODE_LOOPTRI: { | 
					
						
							|  |  |  |         madd_v3_v3fl(temp, data->targetCos[sdbind->vert_inds[0]], sdbind->vert_weights[0]); | 
					
						
							|  |  |  |         madd_v3_v3fl(temp, data->targetCos[sdbind->vert_inds[1]], sdbind->vert_weights[1]); | 
					
						
							|  |  |  |         madd_v3_v3fl(temp, data->targetCos[sdbind->vert_inds[2]], sdbind->vert_weights[2]); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* ---------- ngon mode ---------- */ | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  |       case MOD_SDEF_MODE_NGON: { | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         for (int k = 0; k < sdbind->numverts; k++) { | 
					
						
							| 
									
										
										
										
											2020-03-13 08:22:09 +01:00
										 |  |  |           madd_v3_v3fl(temp, coords_buffer[k], sdbind->vert_weights[k]); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       /* ---------- centroid mode ---------- */ | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  |       case MOD_SDEF_MODE_CENTROID: { | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         float cent[3]; | 
					
						
							| 
									
										
										
										
											2020-03-13 08:22:09 +01:00
										 |  |  |         mid_v3_v3_array(cent, coords_buffer, sdbind->numverts); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |         madd_v3_v3fl(temp, data->targetCos[sdbind->vert_inds[0]], sdbind->vert_weights[0]); | 
					
						
							|  |  |  |         madd_v3_v3fl(temp, data->targetCos[sdbind->vert_inds[1]], sdbind->vert_weights[1]); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |         madd_v3_v3fl(temp, cent, sdbind->vert_weights[2]); | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* Apply normal offset (generic for all modes) */ | 
					
						
							|  |  |  |     madd_v3_v3fl(temp, norm, sdbind->normal_dist); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |     madd_v3_v3fl(offset, temp, sdbind->influence); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   /* Subtract the vertex coord to get the deformation offset. */ | 
					
						
							|  |  |  |   sub_v3_v3(offset, vertexCos); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Add the offset to start coord multiplied by the strength and weight values. */ | 
					
						
							|  |  |  |   madd_v3_v3fl(vertexCos, offset, data->strength * weight); | 
					
						
							| 
									
										
										
										
											2021-01-07 20:31:06 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (UNLIKELY(big_buffer)) { | 
					
						
							|  |  |  |     MEM_freeN(coords_buffer); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  | static void surfacedeformModifier_do(ModifierData *md, | 
					
						
							| 
									
										
										
										
											2019-04-04 14:42:33 +02:00
										 |  |  |                                      const ModifierEvalContext *ctx, | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  |                                      float (*vertexCos)[3], | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |                                      uint numverts, | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |                                      Object *ob, | 
					
						
							|  |  |  |                                      Mesh *mesh) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							| 
									
										
										
										
											2018-05-09 11:33:32 +02:00
										 |  |  |   Mesh *target; | 
					
						
							| 
									
										
										
										
											2019-09-19 13:32:36 +10:00
										 |  |  |   uint tnumverts, tnumpoly; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  |   /* Exit function if bind flag is not set (free bind data if any). */ | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   if (!(smd->flags & MOD_SDEF_BIND)) { | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  |     if (smd->verts != NULL) { | 
					
						
							| 
									
										
										
										
											2019-04-04 14:42:33 +02:00
										 |  |  |       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-04 14:42:33 +02:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-05-08 10:14:02 +02:00
										 |  |  |       ModifierData *md_orig = BKE_modifier_get_original(md); | 
					
						
							| 
									
										
										
										
											2019-04-04 14:42:33 +02:00
										 |  |  |       freeData(md_orig); | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-26 11:25:07 +01:00
										 |  |  |   Object *ob_target = smd->target; | 
					
						
							| 
									
										
										
										
											2019-02-11 20:20:12 +01:00
										 |  |  |   target = BKE_modifier_get_evaluated_mesh_from_evaluated_object(ob_target, false); | 
					
						
							| 
									
										
										
										
											2018-05-09 11:33:32 +02:00
										 |  |  |   if (!target) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, md, "No valid target mesh"); | 
					
						
							| 
									
										
										
										
											2017-04-13 15:39:16 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 15:54:05 +10:00
										 |  |  |   tnumverts = BKE_mesh_wrapper_vert_len(target); | 
					
						
							|  |  |  |   tnumpoly = BKE_mesh_wrapper_poly_len(target); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  |   /* If not bound, execute bind. */ | 
					
						
							|  |  |  |   if (smd->verts == NULL) { | 
					
						
							| 
									
										
										
										
											2019-04-04 14:42:33 +02:00
										 |  |  |     if (!DEG_is_active(ctx->depsgraph)) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |       BKE_modifier_set_error(ob, md, "Attempt to unbind from inactive dependency graph"); | 
					
						
							| 
									
										
										
										
											2019-02-11 20:20:12 +01:00
										 |  |  |       return; | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-08 19:02:03 +10:00
										 |  |  |     SurfaceDeformModifierData *smd_orig = (SurfaceDeformModifierData *)BKE_modifier_get_original( | 
					
						
							|  |  |  |         md); | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |     float tmp_mat[4][4]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |     invert_m4_m4(tmp_mat, ob->obmat); | 
					
						
							| 
									
										
										
										
											2019-04-04 14:42:33 +02:00
										 |  |  |     mul_m4_m4m4(smd_orig->mat, tmp_mat, ob_target->obmat); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 15:54:05 +10:00
										 |  |  |     /* Avoid converting edit-mesh data, binding is an exception. */ | 
					
						
							|  |  |  |     BKE_mesh_wrapper_ensure_mdata(target); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     if (!surfacedeformBind(ob, smd_orig, smd, vertexCos, numverts, tnumpoly, tnumverts, target)) { | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |       smd->flags &= ~MOD_SDEF_BIND; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-05 18:33:31 +01:00
										 |  |  |     /* Early abort, this is binding 'call', no need to perform whole evaluation. */ | 
					
						
							| 
									
										
										
										
											2019-02-11 20:20:12 +01:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   /* Poly count checks */ | 
					
						
							|  |  |  |   if (smd->numverts != numverts) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error(ob, md, "Vertices changed from %u to %u", smd->numverts, numverts); | 
					
						
							| 
									
										
										
										
											2019-02-11 20:20:12 +01:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 12:40:29 +02:00
										 |  |  |   if (smd->numpoly != tnumpoly) { | 
					
						
							| 
									
										
										
										
											2020-10-26 17:07:58 +11:00
										 |  |  |     BKE_modifier_set_error( | 
					
						
							|  |  |  |         ob, md, "Target polygons changed from %u to %u", smd->numpoly, tnumpoly); | 
					
						
							| 
									
										
										
										
											2019-02-11 20:20:12 +01:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-04 00:01:19 +11:00
										 |  |  |   /* Early out if modifier would not affect input at all - still *after* the sanity checks
 | 
					
						
							|  |  |  |    * (and potential binding) above. */ | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   if (smd->strength == 0.0f) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int defgrp_index; | 
					
						
							|  |  |  |   MDeformVert *dvert; | 
					
						
							|  |  |  |   MOD_get_vgroup(ob, mesh, smd->defgrp_name, &dvert, &defgrp_index); | 
					
						
							| 
									
										
										
										
											2021-01-06 12:47:01 +03:00
										 |  |  |   const bool invert_vgroup = (smd->flags & MOD_SDEF_INVERT_VGROUP) != 0; | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |   /* Actual vertex location update starts here */ | 
					
						
							| 
									
										
										
										
											2017-04-24 21:58:28 +10:00
										 |  |  |   SDefDeformData data = { | 
					
						
							|  |  |  |       .bind_verts = smd->verts, | 
					
						
							| 
									
										
										
										
											2018-01-14 22:14:20 +01:00
										 |  |  |       .targetCos = MEM_malloc_arrayN(tnumverts, sizeof(float[3]), "SDefTargetVertArray"), | 
					
						
							| 
									
										
										
										
											2017-04-24 21:58:28 +10:00
										 |  |  |       .vertexCos = vertexCos, | 
					
						
							| 
									
										
										
										
											2021-01-06 12:47:01 +03:00
										 |  |  |       .dvert = dvert, | 
					
						
							|  |  |  |       .defgrp_index = defgrp_index, | 
					
						
							|  |  |  |       .invert_vgroup = invert_vgroup, | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |       .strength = smd->strength, | 
					
						
							| 
									
										
										
										
											2017-04-24 21:58:28 +10:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |   if (data.targetCos != NULL) { | 
					
						
							| 
									
										
										
										
											2020-06-10 15:54:05 +10:00
										 |  |  |     BKE_mesh_wrapper_vert_coords_copy_with_mat4(target, data.targetCos, tnumverts, smd->mat); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-30 14:56:47 +02:00
										 |  |  |     TaskParallelSettings settings; | 
					
						
							| 
									
										
										
										
											2018-01-08 11:35:48 +01:00
										 |  |  |     BLI_parallel_range_settings_defaults(&settings); | 
					
						
							|  |  |  |     settings.use_threading = (numverts > 10000); | 
					
						
							|  |  |  |     BLI_task_parallel_range(0, numverts, &data, deformVert, &settings); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-06 02:40:29 -03:00
										 |  |  |     MEM_freeN(data.targetCos); | 
					
						
							| 
									
										
										
										
											2017-03-04 03:16:50 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  | static void deformVerts(ModifierData *md, | 
					
						
							| 
									
										
										
										
											2018-05-01 17:33:04 +02:00
										 |  |  |                         const ModifierEvalContext *ctx, | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |                         Mesh *mesh, | 
					
						
							| 
									
										
										
										
											2018-05-01 17:33:04 +02:00
										 |  |  |                         float (*vertexCos)[3], | 
					
						
							|  |  |  |                         int numVerts) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  |   Mesh *mesh_src = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (smd->defgrp_name[0] != '\0') { | 
					
						
							|  |  |  |     /* Only need to use mesh_src when a vgroup is used. */ | 
					
						
							|  |  |  |     mesh_src = MOD_deform_mesh_eval_get(ctx->object, NULL, mesh, NULL, numVerts, false, false); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   surfacedeformModifier_do(md, ctx, vertexCos, numVerts, ctx->object, mesh_src); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!ELEM(mesh_src, NULL, mesh)) { | 
					
						
							|  |  |  |     BKE_id_free(NULL, mesh_src); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  | static void deformVertsEM(ModifierData *md, | 
					
						
							| 
									
										
										
										
											2018-05-01 17:33:04 +02:00
										 |  |  |                           const ModifierEvalContext *ctx, | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |                           struct BMEditMesh *em, | 
					
						
							|  |  |  |                           Mesh *mesh, | 
					
						
							| 
									
										
										
										
											2018-02-13 20:35:29 +11:00
										 |  |  |                           float (*vertexCos)[3], | 
					
						
							|  |  |  |                           int numVerts) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  |   Mesh *mesh_src = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (smd->defgrp_name[0] != '\0') { | 
					
						
							|  |  |  |     /* Only need to use mesh_src when a vgroup is used. */ | 
					
						
							|  |  |  |     mesh_src = MOD_deform_mesh_eval_get(ctx->object, em, mesh, NULL, numVerts, false, false); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   surfacedeformModifier_do(md, ctx, vertexCos, numVerts, ctx->object, mesh_src); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!ELEM(mesh_src, NULL, mesh)) { | 
					
						
							|  |  |  |     BKE_id_free(NULL, mesh_src); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-10 12:14:08 +02:00
										 |  |  | static bool isDisabled(const Scene *UNUSED(scene), ModifierData *md, bool UNUSED(useRenderParams)) | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 (smd->target == NULL || smd->target->type != OB_MESH) && | 
					
						
							|  |  |  |          !(smd->verts != NULL && !(smd->flags & MOD_SDEF_BIND)); | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03: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
										 |  |  |   PointerRNA target_ptr = RNA_pointer_get(ptr, "target"); | 
					
						
							| 
									
										
										
										
											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, false); | 
					
						
							|  |  |  |   uiLayoutSetActive(col, !is_bound); | 
					
						
							| 
									
										
										
										
											2020-09-02 14:13:26 -05:00
										 |  |  |   uiItemR(col, ptr, "target", 0, NULL, ICON_NONE); | 
					
						
							|  |  |  |   uiItemR(col, ptr, "falloff", 0, NULL, ICON_NONE); | 
					
						
							| 
									
										
										
										
											2021-01-08 12:22:11 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   uiItemR(layout, ptr, "strength", 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
										 |  |  | 
 | 
					
						
							|  |  |  |   uiItemS(layout); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   col = uiLayoutColumn(layout, false); | 
					
						
							|  |  |  |   if (is_bound) { | 
					
						
							|  |  |  |     uiItemO(col, IFACE_("Unbind"), ICON_NONE, "OBJECT_OT_surfacedeform_bind"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     uiLayoutSetActive(col, !RNA_pointer_is_null(&target_ptr)); | 
					
						
							|  |  |  |     uiItemO(col, IFACE_("Bind"), ICON_NONE, "OBJECT_OT_surfacedeform_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_SurfaceDeform, panel_draw); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-23 16:51:43 +02:00
										 |  |  | static void blendWrite(BlendWriter *writer, const ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const SurfaceDeformModifierData *smd = (const SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLO_write_struct_array(writer, SDefVert, smd->numverts, smd->verts); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (smd->verts) { | 
					
						
							|  |  |  |     for (int i = 0; i < smd->numverts; i++) { | 
					
						
							|  |  |  |       BLO_write_struct_array(writer, SDefBind, smd->verts[i].numbinds, smd->verts[i].binds); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (smd->verts[i].binds) { | 
					
						
							|  |  |  |         for (int j = 0; j < smd->verts[i].numbinds; j++) { | 
					
						
							|  |  |  |           BLO_write_uint32_array( | 
					
						
							|  |  |  |               writer, smd->verts[i].binds[j].numverts, smd->verts[i].binds[j].vert_inds); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 12:30:59 +11:00
										 |  |  |           if (ELEM(smd->verts[i].binds[j].mode, MOD_SDEF_MODE_CENTROID, MOD_SDEF_MODE_LOOPTRI)) { | 
					
						
							| 
									
										
										
										
											2020-06-23 16:51:43 +02:00
										 |  |  |             BLO_write_float3_array(writer, 1, smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           else { | 
					
						
							|  |  |  |             BLO_write_float_array( | 
					
						
							|  |  |  |                 writer, smd->verts[i].binds[j].numverts, smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void blendRead(BlendDataReader *reader, ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   SurfaceDeformModifierData *smd = (SurfaceDeformModifierData *)md; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLO_read_data_address(reader, &smd->verts); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (smd->verts) { | 
					
						
							|  |  |  |     for (int i = 0; i < smd->numverts; i++) { | 
					
						
							|  |  |  |       BLO_read_data_address(reader, &smd->verts[i].binds); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (smd->verts[i].binds) { | 
					
						
							|  |  |  |         for (int j = 0; j < smd->verts[i].numbinds; j++) { | 
					
						
							|  |  |  |           BLO_read_uint32_array( | 
					
						
							|  |  |  |               reader, smd->verts[i].binds[j].numverts, &smd->verts[i].binds[j].vert_inds); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 12:30:59 +11:00
										 |  |  |           if (ELEM(smd->verts[i].binds[j].mode, MOD_SDEF_MODE_CENTROID, MOD_SDEF_MODE_LOOPTRI)) { | 
					
						
							| 
									
										
										
										
											2020-06-23 16:51:43 +02:00
										 |  |  |             BLO_read_float3_array(reader, 1, &smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           else { | 
					
						
							|  |  |  |             BLO_read_float_array( | 
					
						
							|  |  |  |                 reader, smd->verts[i].binds[j].numverts, &smd->verts[i].binds[j].vert_weights); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | ModifierTypeInfo modifierType_SurfaceDeform = { | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  |     /* name */ "SurfaceDeform", | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* structName */ "SurfaceDeformModifierData", | 
					
						
							|  |  |  |     /* structSize */ sizeof(SurfaceDeformModifierData), | 
					
						
							| 
									
										
										
										
											2020-09-25 12:49:18 +02:00
										 |  |  |     /* srna */ &RNA_SurfaceDeformModifier, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* type */ eModifierTypeType_OnlyDeform, | 
					
						
							|  |  |  |     /* flags */ eModifierTypeFlag_AcceptsMesh | eModifierTypeFlag_SupportsEditmode, | 
					
						
							| 
									
										
										
										
											2020-09-25 12:45:30 +02:00
										 |  |  |     /* icon */ ICON_MOD_MESHDEFORM, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* copyData */ copyData, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-09 11:33:32 +02:00
										 |  |  |     /* deformVerts */ deformVerts, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* deformMatrices */ NULL, | 
					
						
							| 
									
										
										
										
											2018-05-09 11:33:32 +02:00
										 |  |  |     /* deformVertsEM */ deformVertsEM, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* initData */ initData, | 
					
						
							| 
									
										
										
										
											2020-03-27 12:20:31 +01:00
										 |  |  |     /* requiredDataMask */ requiredDataMask, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* freeData */ freeData, | 
					
						
							|  |  |  |     /* isDisabled */ isDisabled, | 
					
						
							|  |  |  |     /* updateDepsgraph */ updateDepsgraph, | 
					
						
							|  |  |  |     /* dependsOnTime */ NULL, | 
					
						
							|  |  |  |     /* dependsOnNormals */ NULL, | 
					
						
							| 
									
										
										
										
											2020-10-01 18:05:23 +02:00
										 |  |  |     /* foreachIDLink */ foreachIDLink, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  |     /* foreachTexLink */ NULL, | 
					
						
							| 
									
										
										
										
											2019-03-18 15:56:16 +01:00
										 |  |  |     /* freeRuntimeData */ NULL, | 
					
						
							| 
									
										
										
										
											2020-06-05 10:41:03 -04:00
										 |  |  |     /* panelRegister */ panelRegister, | 
					
						
							| 
									
										
										
										
											2020-06-23 16:51:43 +02:00
										 |  |  |     /* blendWrite */ blendWrite, | 
					
						
							|  |  |  |     /* blendRead */ blendRead, | 
					
						
							| 
									
										
										
										
											2017-02-27 12:39:14 -03:00
										 |  |  | }; |