Cleanup: split BKE_mesh_copy_settings into two functions
[blender.git] / source / blender / modifiers / intern / MOD_remesh.c
1 /*
2  * This program is free software; you can redistribute it and/or
3  * modify it under the terms of the GNU General Public License
4  * as published by the Free Software Foundation; either version 2
5  * of the License, or (at your option) any later version.
6  *
7  * This program is distributed in the hope that it will be useful,
8  * but WITHOUT ANY WARRANTY; without even the implied warranty of
9  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10  * GNU General Public License for more details.
11  *
12  * You should have received a copy of the GNU General Public License
13  * along with this program; if not, write to the Free Software Foundation,
14  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
15  *
16  * The Original Code is Copyright (C) 2011 by Nicholas Bishop.
17  */
18
19 /** \file
20  * \ingroup modifiers
21  */
22
23 #include "MEM_guardedalloc.h"
24
25 #include "BLI_utildefines.h"
26
27 #include "BLI_math_base.h"
28 #include "BLI_threads.h"
29
30 #include "BLT_translation.h"
31
32 #include "DNA_defaults.h"
33 #include "DNA_mesh_types.h"
34 #include "DNA_meshdata_types.h"
35 #include "DNA_modifier_types.h"
36 #include "DNA_object_types.h"
37 #include "DNA_screen_types.h"
38
39 #include "BKE_context.h"
40 #include "BKE_mesh.h"
41 #include "BKE_mesh_remesh_voxel.h"
42 #include "BKE_mesh_runtime.h"
43 #include "BKE_screen.h"
44
45 #include "UI_interface.h"
46 #include "UI_resources.h"
47
48 #include "RNA_access.h"
49
50 #include "MOD_modifiertypes.h"
51 #include "MOD_ui_common.h"
52
53 #include <stdlib.h>
54 #include <string.h>
55
56 #ifdef WITH_MOD_REMESH
57 #  include "BLI_math_vector.h"
58
59 #  include "dualcon.h"
60 #endif
61
62 static void initData(ModifierData *md)
63 {
64   RemeshModifierData *rmd = (RemeshModifierData *)md;
65
66   BLI_assert(MEMCMP_STRUCT_AFTER_IS_ZERO(rmd, modifier));
67
68   MEMCPY_STRUCT_AFTER(rmd, DNA_struct_default_get(RemeshModifierData), modifier);
69 }
70
71 #ifdef WITH_MOD_REMESH
72
73 static void init_dualcon_mesh(DualConInput *input, Mesh *mesh)
74 {
75   memset(input, 0, sizeof(DualConInput));
76
77   input->co = (void *)mesh->mvert;
78   input->co_stride = sizeof(MVert);
79   input->totco = mesh->totvert;
80
81   input->mloop = (void *)mesh->mloop;
82   input->loop_stride = sizeof(MLoop);
83
84   BKE_mesh_runtime_looptri_ensure(mesh);
85   input->looptri = (void *)mesh->runtime.looptris.array;
86   input->tri_stride = sizeof(MLoopTri);
87   input->tottri = mesh->runtime.looptris.len;
88
89   INIT_MINMAX(input->min, input->max);
90   BKE_mesh_minmax(mesh, input->min, input->max);
91 }
92
93 /* simple structure to hold the output: a CDDM and two counters to
94  * keep track of the current elements */
95 typedef struct {
96   Mesh *mesh;
97   int curvert, curface;
98 } DualConOutput;
99
100 /* allocate and initialize a DualConOutput */
101 static void *dualcon_alloc_output(int totvert, int totquad)
102 {
103   DualConOutput *output;
104
105   if (!(output = MEM_callocN(sizeof(DualConOutput), "DualConOutput"))) {
106     return NULL;
107   }
108
109   output->mesh = BKE_mesh_new_nomain(totvert, 0, 0, 4 * totquad, totquad);
110   return output;
111 }
112
113 static void dualcon_add_vert(void *output_v, const float co[3])
114 {
115   DualConOutput *output = output_v;
116   Mesh *mesh = output->mesh;
117
118   BLI_assert(output->curvert < mesh->totvert);
119
120   copy_v3_v3(mesh->mvert[output->curvert].co, co);
121   output->curvert++;
122 }
123
124 static void dualcon_add_quad(void *output_v, const int vert_indices[4])
125 {
126   DualConOutput *output = output_v;
127   Mesh *mesh = output->mesh;
128   MLoop *mloop;
129   MPoly *cur_poly;
130   int i;
131
132   BLI_assert(output->curface < mesh->totpoly);
133
134   mloop = mesh->mloop;
135   cur_poly = &mesh->mpoly[output->curface];
136
137   cur_poly->loopstart = output->curface * 4;
138   cur_poly->totloop = 4;
139   for (i = 0; i < 4; i++) {
140     mloop[output->curface * 4 + i].v = vert_indices[i];
141   }
142
143   output->curface++;
144 }
145
146 static Mesh *modifyMesh(ModifierData *md, const ModifierEvalContext *UNUSED(ctx), Mesh *mesh)
147 {
148   RemeshModifierData *rmd;
149   DualConOutput *output;
150   DualConInput input;
151   Mesh *result;
152   DualConFlags flags = 0;
153   DualConMode mode = 0;
154
155   rmd = (RemeshModifierData *)md;
156
157   if (rmd->mode == MOD_REMESH_VOXEL) {
158     /* OpenVDB modes. */
159     if (rmd->voxel_size == 0.0f) {
160       return NULL;
161     }
162     result = BKE_mesh_remesh_voxel_to_mesh_nomain(mesh, rmd->voxel_size, rmd->adaptivity, 0.0f);
163     if (result == NULL) {
164       return NULL;
165     }
166   }
167   else {
168     /* Dualcon modes. */
169     init_dualcon_mesh(&input, mesh);
170
171     if (rmd->flag & MOD_REMESH_FLOOD_FILL) {
172       flags |= DUALCON_FLOOD_FILL;
173     }
174
175     switch (rmd->mode) {
176       case MOD_REMESH_CENTROID:
177         mode = DUALCON_CENTROID;
178         break;
179       case MOD_REMESH_MASS_POINT:
180         mode = DUALCON_MASS_POINT;
181         break;
182       case MOD_REMESH_SHARP_FEATURES:
183         mode = DUALCON_SHARP_FEATURES;
184         break;
185       case MOD_REMESH_VOXEL:
186         /* Should have been processed before as an OpenVDB operation. */
187         BLI_assert(false);
188         break;
189     }
190     /* TODO(jbakker): Dualcon crashes when run in parallel. Could be related to incorrect
191      * input data or that the library isn't thread safe.
192      * This was identified when changing the task isolation's during T76553. */
193     static ThreadMutex dualcon_mutex = BLI_MUTEX_INITIALIZER;
194     BLI_mutex_lock(&dualcon_mutex);
195     output = dualcon(&input,
196                      dualcon_alloc_output,
197                      dualcon_add_vert,
198                      dualcon_add_quad,
199                      flags,
200                      mode,
201                      rmd->threshold,
202                      rmd->hermite_num,
203                      rmd->scale,
204                      rmd->depth);
205     BLI_mutex_unlock(&dualcon_mutex);
206
207     result = output->mesh;
208     MEM_freeN(output);
209   }
210
211   if (rmd->flag & MOD_REMESH_SMOOTH_SHADING) {
212     MPoly *mpoly = result->mpoly;
213     int i, totpoly = result->totpoly;
214
215     /* Apply smooth shading to output faces */
216     for (i = 0; i < totpoly; i++) {
217       mpoly[i].flag |= ME_SMOOTH;
218     }
219   }
220
221   BKE_mesh_copy_parameters_for_eval(result, mesh);
222   BKE_mesh_calc_edges(result, true, false);
223   result->runtime.cd_dirty_vert |= CD_MASK_NORMAL;
224   return result;
225 }
226
227 #else /* !WITH_MOD_REMESH */
228
229 static Mesh *modifyMesh(ModifierData *UNUSED(md),
230                         const ModifierEvalContext *UNUSED(ctx),
231                         Mesh *mesh)
232 {
233   return mesh;
234 }
235
236 #endif /* !WITH_MOD_REMESH */
237
238 static void panel_draw(const bContext *UNUSED(C), Panel *panel)
239 {
240   uiLayout *layout = panel->layout;
241 #ifdef WITH_MOD_REMESH
242   uiLayout *row, *col;
243
244   PointerRNA ob_ptr;
245   PointerRNA *ptr = modifier_panel_get_property_pointers(panel, &ob_ptr);
246
247   int mode = RNA_enum_get(ptr, "mode");
248
249   uiItemR(layout, ptr, "mode", UI_ITEM_R_EXPAND, NULL, ICON_NONE);
250
251   uiLayoutSetPropSep(layout, true);
252
253   col = uiLayoutColumn(layout, false);
254   if (mode == MOD_REMESH_VOXEL) {
255     uiItemR(col, ptr, "voxel_size", 0, NULL, ICON_NONE);
256     uiItemR(col, ptr, "adaptivity", 0, NULL, ICON_NONE);
257   }
258   else {
259     uiItemR(col, ptr, "octree_depth", 0, NULL, ICON_NONE);
260     uiItemR(col, ptr, "scale", 0, NULL, ICON_NONE);
261
262     if (mode == MOD_REMESH_SHARP_FEATURES) {
263       uiItemR(col, ptr, "sharpness", 0, NULL, ICON_NONE);
264     }
265
266     uiItemR(layout, ptr, "use_remove_disconnected", 0, NULL, ICON_NONE);
267     row = uiLayoutRow(layout, false);
268     uiLayoutSetActive(row, RNA_boolean_get(ptr, "use_remove_disconnected"));
269     uiItemR(layout, ptr, "threshold", 0, NULL, ICON_NONE);
270   }
271   uiItemR(layout, ptr, "use_smooth_shade", 0, NULL, ICON_NONE);
272
273   modifier_panel_end(layout, ptr);
274
275 #else  /* WITH_MOD_REMESH */
276   uiItemL(layout, IFACE_("Built without Remesh modifier"), ICON_NONE);
277 #endif /* WITH_MOD_REMESH */
278 }
279
280 static void panelRegister(ARegionType *region_type)
281 {
282   modifier_panel_register(region_type, eModifierType_Remesh, panel_draw);
283 }
284
285 ModifierTypeInfo modifierType_Remesh = {
286     /* name */ "Remesh",
287     /* structName */ "RemeshModifierData",
288     /* structSize */ sizeof(RemeshModifierData),
289     /* srna */ &RNA_RemeshModifier,
290     /* type */ eModifierTypeType_Nonconstructive,
291     /* flags */ eModifierTypeFlag_AcceptsMesh | eModifierTypeFlag_AcceptsCVs |
292         eModifierTypeFlag_SupportsEditmode,
293     /* icon */ ICON_MOD_REMESH,
294
295     /* copyData */ BKE_modifier_copydata_generic,
296
297     /* deformVerts */ NULL,
298     /* deformMatrices */ NULL,
299     /* deformVertsEM */ NULL,
300     /* deformMatricesEM */ NULL,
301     /* modifyMesh */ modifyMesh,
302     /* modifyHair */ NULL,
303     /* modifyGeometrySet */ NULL,
304
305     /* initData */ initData,
306     /* requiredDataMask */ NULL,
307     /* freeData */ NULL,
308     /* isDisabled */ NULL,
309     /* updateDepsgraph */ NULL,
310     /* dependsOnTime */ NULL,
311     /* dependsOnNormals */ NULL,
312     /* foreachIDLink */ NULL,
313     /* foreachTexLink */ NULL,
314     /* freeRuntimeData */ NULL,
315     /* panelRegister */ panelRegister,
316     /* blendWrite */ NULL,
317     /* blendRead */ NULL,
318 };