| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2011-10-23 17:52:20 +00:00
										 |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software  Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2005 by the Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup modifiers | 
					
						
							| 
									
										
										
										
											2011-02-25 13:57:17 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-25 11:56:24 +01:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_math.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 15:39:06 +00:00
										 |  |  | #include "DNA_object_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-11 10:55:26 +02:00
										 |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							|  |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-24 03:03:42 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 15:39:06 +00:00
										 |  |  | #include "BKE_deform.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-11 10:55:26 +02:00
										 |  |  | #include "BKE_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 11:17:45 +01:00
										 |  |  | #include "DEG_depsgraph_query.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 02:39:26 +00:00
										 |  |  | #include "bmesh.h"
 | 
					
						
							| 
									
										
										
										
											2013-08-23 04:22:07 +00:00
										 |  |  | #include "bmesh_tools.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-19 10:40:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 03:25:53 +00:00
										 |  |  | // #define USE_TIMEIT
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 02:09:41 +00:00
										 |  |  | #ifdef USE_TIMEIT
 | 
					
						
							|  |  |  | #  include "PIL_time.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-03 21:22:43 +00:00
										 |  |  | #  include "PIL_time_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-22 02:09:41 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-13 14:16:36 +00:00
										 |  |  | #include "MOD_util.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void initData(ModifierData *md) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DecimateModifierData *dmd = (DecimateModifierData *)md; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   dmd->percent = 1.0; | 
					
						
							|  |  |  |   dmd->angle = DEG2RADF(5.0f); | 
					
						
							|  |  |  |   dmd->defgrp_factor = 1.0; | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void requiredDataMask(Object *UNUSED(ob), | 
					
						
							|  |  |  |                              ModifierData *md, | 
					
						
							|  |  |  |                              CustomData_MeshMasks *r_cddata_masks) | 
					
						
							| 
									
										
										
										
											2012-10-22 15:39:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DecimateModifierData *dmd = (DecimateModifierData *)md; | 
					
						
							| 
									
										
										
										
											2012-10-22 15:39:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* ask for vertexgroups if we need them */ | 
					
						
							|  |  |  |   if (dmd->defgrp_name[0] != '\0' && (dmd->defgrp_factor > 0.0f)) { | 
					
						
							|  |  |  |     r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-10-22 15:39:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static DecimateModifierData *getOriginalModifierData(const DecimateModifierData *dmd, | 
					
						
							|  |  |  |                                                      const ModifierEvalContext *ctx) | 
					
						
							| 
									
										
										
										
											2018-12-20 11:17:45 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Object *ob_orig = DEG_get_original_object(ctx->object); | 
					
						
							|  |  |  |   return (DecimateModifierData *)modifiers_findByName(ob_orig, dmd->modifier.name); | 
					
						
							| 
									
										
										
										
											2018-12-20 11:17:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void updateFaceCount(const ModifierEvalContext *ctx, | 
					
						
							|  |  |  |                             DecimateModifierData *dmd, | 
					
						
							|  |  |  |                             int face_count) | 
					
						
							| 
									
										
										
										
											2018-12-21 16:53:33 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   dmd->face_count = face_count; | 
					
						
							| 
									
										
										
										
											2019-01-22 19:49:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (DEG_is_active(ctx->depsgraph)) { | 
					
						
							|  |  |  |     /* update for display only */ | 
					
						
							|  |  |  |     DecimateModifierData *dmd_orig = getOriginalModifierData(dmd, ctx); | 
					
						
							|  |  |  |     dmd_orig->face_count = face_count; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-12-21 16:53:33 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static Mesh *applyModifier(ModifierData *md, const ModifierEvalContext *ctx, Mesh *meshData) | 
					
						
							| 
									
										
										
										
											2012-10-19 10:40:32 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DecimateModifierData *dmd = (DecimateModifierData *)md; | 
					
						
							|  |  |  |   Mesh *mesh = meshData, *result = NULL; | 
					
						
							|  |  |  |   BMesh *bm; | 
					
						
							|  |  |  |   bool calc_face_normal; | 
					
						
							|  |  |  |   float *vweights = NULL; | 
					
						
							| 
									
										
										
										
											2012-10-22 15:39:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 02:09:41 +00:00
										 |  |  | #ifdef USE_TIMEIT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   TIMEIT_START(decim); | 
					
						
							| 
									
										
										
										
											2012-10-22 02:09:41 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-10-20 17:31:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* set up front so we dont show invalid info in the UI */ | 
					
						
							|  |  |  |   updateFaceCount(ctx, dmd, mesh->totpoly); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (dmd->mode) { | 
					
						
							|  |  |  |     case MOD_DECIM_MODE_COLLAPSE: | 
					
						
							|  |  |  |       if (dmd->percent == 1.0f) { | 
					
						
							|  |  |  |         return mesh; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       calc_face_normal = true; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case MOD_DECIM_MODE_UNSUBDIV: | 
					
						
							|  |  |  |       if (dmd->iter == 0) { | 
					
						
							|  |  |  |         return mesh; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       calc_face_normal = false; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case MOD_DECIM_MODE_DISSOLVE: | 
					
						
							|  |  |  |       if (dmd->angle == 0.0f) { | 
					
						
							|  |  |  |         return mesh; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       calc_face_normal = true; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return mesh; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (dmd->face_count <= 3) { | 
					
						
							|  |  |  |     modifier_setError(md, "Modifier requires more than 3 input faces"); | 
					
						
							|  |  |  |     return mesh; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (dmd->mode == MOD_DECIM_MODE_COLLAPSE) { | 
					
						
							|  |  |  |     if (dmd->defgrp_name[0] && (dmd->defgrp_factor > 0.0f)) { | 
					
						
							|  |  |  |       MDeformVert *dvert; | 
					
						
							|  |  |  |       int defgrp_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       MOD_get_vgroup(ctx->object, mesh, dmd->defgrp_name, &dvert, &defgrp_index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (dvert) { | 
					
						
							|  |  |  |         const unsigned int vert_tot = mesh->totvert; | 
					
						
							|  |  |  |         unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vweights = MEM_malloc_arrayN(vert_tot, sizeof(float), __func__); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (dmd->flag & MOD_DECIM_FLAG_INVERT_VGROUP) { | 
					
						
							|  |  |  |           for (i = 0; i < vert_tot; i++) { | 
					
						
							|  |  |  |             vweights[i] = 1.0f - defvert_find_weight(&dvert[i], defgrp_index); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           for (i = 0; i < vert_tot; i++) { | 
					
						
							|  |  |  |             vweights[i] = defvert_find_weight(&dvert[i], defgrp_index); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bm = BKE_mesh_to_bmesh_ex(mesh, | 
					
						
							|  |  |  |                             &(struct BMeshCreateParams){0}, | 
					
						
							|  |  |  |                             &(struct BMeshFromMeshParams){ | 
					
						
							|  |  |  |                                 .calc_face_normal = calc_face_normal, | 
					
						
							|  |  |  |                                 .cd_mask_extra = {.vmask = CD_MASK_ORIGINDEX, | 
					
						
							|  |  |  |                                                   .emask = CD_MASK_ORIGINDEX, | 
					
						
							|  |  |  |                                                   .pmask = CD_MASK_ORIGINDEX}, | 
					
						
							|  |  |  |                             }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (dmd->mode) { | 
					
						
							|  |  |  |     case MOD_DECIM_MODE_COLLAPSE: { | 
					
						
							|  |  |  |       const bool do_triangulate = (dmd->flag & MOD_DECIM_FLAG_TRIANGULATE) != 0; | 
					
						
							|  |  |  |       const int symmetry_axis = (dmd->flag & MOD_DECIM_FLAG_SYMMETRY) ? dmd->symmetry_axis : -1; | 
					
						
							|  |  |  |       const float symmetry_eps = 0.00002f; | 
					
						
							|  |  |  |       BM_mesh_decimate_collapse(bm, | 
					
						
							|  |  |  |                                 dmd->percent, | 
					
						
							|  |  |  |                                 vweights, | 
					
						
							|  |  |  |                                 dmd->defgrp_factor, | 
					
						
							|  |  |  |                                 do_triangulate, | 
					
						
							|  |  |  |                                 symmetry_axis, | 
					
						
							|  |  |  |                                 symmetry_eps); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case MOD_DECIM_MODE_UNSUBDIV: { | 
					
						
							|  |  |  |       BM_mesh_decimate_unsubdivide(bm, dmd->iter); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case MOD_DECIM_MODE_DISSOLVE: { | 
					
						
							|  |  |  |       const bool do_dissolve_boundaries = (dmd->flag & MOD_DECIM_FLAG_ALL_BOUNDARY_VERTS) != 0; | 
					
						
							|  |  |  |       BM_mesh_decimate_dissolve(bm, dmd->angle, do_dissolve_boundaries, (BMO_Delimit)dmd->delimit); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (vweights) { | 
					
						
							|  |  |  |     MEM_freeN(vweights); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   updateFaceCount(ctx, dmd, bm->totface); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   result = BKE_mesh_from_bmesh_for_eval_nomain(bm, NULL); | 
					
						
							| 
									
										
										
										
											2019-07-10 14:41:19 +10:00
										 |  |  |   /* make sure we never alloc'd these */ | 
					
						
							|  |  |  |   BLI_assert(bm->vtoolflagpool == NULL && bm->etoolflagpool == NULL && bm->ftoolflagpool == NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(bm->vtable == NULL && bm->etable == NULL && bm->ftable == NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BM_mesh_free(bm); | 
					
						
							| 
									
										
										
										
											2012-10-19 10:40:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 02:09:41 +00:00
										 |  |  | #ifdef USE_TIMEIT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   TIMEIT_END(decim); | 
					
						
							| 
									
										
										
										
											2012-10-22 02:09:41 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-10-20 17:31:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   result->runtime.cd_dirty_vert |= CD_MASK_NORMAL; | 
					
						
							| 
									
										
										
										
											2013-06-02 04:09:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return result; | 
					
						
							| 
									
										
										
										
											2012-10-19 10:40:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | ModifierTypeInfo modifierType_Decimate = { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* name */ "Decimate", | 
					
						
							|  |  |  |     /* structName */ "DecimateModifierData", | 
					
						
							|  |  |  |     /* structSize */ sizeof(DecimateModifierData), | 
					
						
							|  |  |  |     /* type */ eModifierTypeType_Nonconstructive, | 
					
						
							|  |  |  |     /* flags */ eModifierTypeFlag_AcceptsMesh | eModifierTypeFlag_AcceptsCVs, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* copyData */ modifier_copyData_generic, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* deformVerts */ NULL, | 
					
						
							|  |  |  |     /* deformMatrices */ NULL, | 
					
						
							|  |  |  |     /* deformVertsEM */ NULL, | 
					
						
							|  |  |  |     /* deformMatricesEM */ NULL, | 
					
						
							|  |  |  |     /* applyModifier */ applyModifier, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* initData */ initData, | 
					
						
							|  |  |  |     /* requiredDataMask */ requiredDataMask, | 
					
						
							|  |  |  |     /* freeData */ NULL, | 
					
						
							|  |  |  |     /* isDisabled */ NULL, | 
					
						
							|  |  |  |     /* updateDepsgraph */ NULL, | 
					
						
							|  |  |  |     /* dependsOnTime */ NULL, | 
					
						
							|  |  |  |     /* dependsOnNormals */ NULL, | 
					
						
							|  |  |  |     /* foreachObjectLink */ NULL, | 
					
						
							|  |  |  |     /* foreachIDLink */ NULL, | 
					
						
							|  |  |  |     /* foreachTexLink */ NULL, | 
					
						
							|  |  |  |     /* freeRuntimeData */ NULL, | 
					
						
							| 
									
										
										
										
											2010-04-11 22:12:30 +00:00
										 |  |  | }; |