Blender V4.5
MOD_weld.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2005 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
10
11/* TODOs:
12 * - Review weight and vertex color interpolation.;
13 */
14
15#include "BLI_utildefines.h"
16
17#include "BLI_array.hh"
18#include "BLI_index_range.hh"
19#include "BLI_span.hh"
20#include "BLI_vector.hh"
21
22#include "BLT_translation.hh"
23
24#include "DNA_defaults.h"
25#include "DNA_mesh_types.h"
26#include "DNA_meshdata_types.h"
27#include "DNA_modifier_types.h"
28#include "DNA_screen_types.h"
29
30#include "BKE_context.hh"
31#include "BKE_customdata.hh"
32#include "BKE_deform.hh"
33#include "BKE_modifier.hh"
34#include "BKE_screen.hh"
35
36#include "UI_interface.hh"
37#include "UI_resources.hh"
38
39#include "RNA_access.hh"
40#include "RNA_prototypes.hh"
41
42#include "MOD_modifiertypes.hh"
43#include "MOD_ui_common.hh"
44
46
47using blender::Array;
50using blender::Span;
51using blender::Vector;
52
53static Span<MDeformVert> get_vertex_group(const Mesh &mesh, const int defgrp_index)
54{
55 if (defgrp_index == -1) {
56 return {};
57 }
58 const MDeformVert *vertex_group = static_cast<const MDeformVert *>(
60 if (!vertex_group) {
61 return {};
62 }
63 return {vertex_group, mesh.verts_num};
64}
65
67 const int index,
68 const bool invert,
69 IndexMaskMemory &memory)
70{
72 vertex_group.index_range(), blender::GrainSize(512), memory, [&](const int i) {
73 return (BKE_defvert_find_weight(&vertex_group[i], index) > 0.0f) != invert;
74 });
75}
76
78 const int index,
79 const bool invert)
80{
81 Array<bool> selection(vertex_group.size());
82 for (const int i : vertex_group.index_range()) {
83 const bool found = BKE_defvert_find_weight(&vertex_group[i], index) > 0.0f;
84 selection[i] = (found != invert);
85 }
86 return selection;
87}
88
89static std::optional<Mesh *> calculate_weld(const Mesh &mesh, const WeldModifierData &wmd)
90{
91 const int defgrp_index = BKE_id_defgroup_name_index(&mesh.id, wmd.defgrp_name);
92 Span<MDeformVert> vertex_group = get_vertex_group(mesh, defgrp_index);
93 const bool invert = (wmd.flag & MOD_WELD_INVERT_VGROUP) != 0;
94
95 if (wmd.mode == MOD_WELD_MODE_ALL) {
96 if (!vertex_group.is_empty()) {
97 IndexMaskMemory memory;
98 const IndexMask selected_indices = selected_indices_from_vertex_group(
99 vertex_group, defgrp_index, invert, memory);
101 mesh, IndexMask(selected_indices), wmd.merge_dist);
102 }
104 mesh, IndexMask(mesh.verts_num), wmd.merge_dist);
105 }
106 if (wmd.mode == MOD_WELD_MODE_CONNECTED) {
107 const bool only_loose_edges = (wmd.flag & MOD_WELD_LOOSE_EDGES) != 0;
108 if (!vertex_group.is_empty()) {
110 vertex_group, defgrp_index, invert);
112 mesh, selection, wmd.merge_dist, only_loose_edges);
113 }
114 Array<bool> selection(mesh.verts_num, true);
116 mesh, selection, wmd.merge_dist, only_loose_edges);
117 }
118
120 return nullptr;
121}
122
123static Mesh *modify_mesh(ModifierData *md, const ModifierEvalContext * /*ctx*/, Mesh *mesh)
124{
125 const WeldModifierData &wmd = reinterpret_cast<WeldModifierData &>(*md);
126
127 std::optional<Mesh *> result = calculate_weld(*mesh, wmd);
128 if (!result) {
129 return mesh;
130 }
131 return *result;
132}
133
134static void init_data(ModifierData *md)
135{
137
139
141}
142
143static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
144{
146
147 /* Ask for vertex-groups if we need them. */
148 if (wmd->defgrp_name[0] != '\0') {
149 r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT;
150 }
151}
152
153static void panel_draw(const bContext * /*C*/, Panel *panel)
154{
155 uiLayout *layout = panel->layout;
156
157 PointerRNA ob_ptr;
159 int weld_mode = RNA_enum_get(ptr, "mode");
160
161 uiLayoutSetPropSep(layout, true);
162
163 layout->prop(ptr, "mode", UI_ITEM_NONE, std::nullopt, ICON_NONE);
164 layout->prop(ptr, "merge_threshold", UI_ITEM_NONE, IFACE_("Distance"), ICON_NONE);
165 if (weld_mode == MOD_WELD_MODE_CONNECTED) {
166 layout->prop(ptr, "loose_edges", UI_ITEM_NONE, std::nullopt, ICON_NONE);
167 }
168 modifier_vgroup_ui(layout, ptr, &ob_ptr, "vertex_group", "invert_vertex_group", std::nullopt);
169
171}
172
173static void panel_register(ARegionType *region_type)
174{
176}
177
179 /*idname*/ "Weld",
180 /*name*/ N_("Weld"),
181 /*struct_name*/ "WeldModifierData",
182 /*struct_size*/ sizeof(WeldModifierData),
183 /*srna*/ &RNA_WeldModifier,
185 /*flags*/
189 /*icon*/ ICON_AUTOMERGE_OFF, /* TODO: Use correct icon. */
190
191 /*copy_data*/ BKE_modifier_copydata_generic,
192
193 /*deform_verts*/ nullptr,
194 /*deform_matrices*/ nullptr,
195 /*deform_verts_EM*/ nullptr,
196 /*deform_matrices_EM*/ nullptr,
197 /*modify_mesh*/ modify_mesh,
198 /*modify_geometry_set*/ nullptr,
199
200 /*init_data*/ init_data,
201 /*required_data_mask*/ required_data_mask,
202 /*free_data*/ nullptr,
203 /*is_disabled*/ nullptr,
204 /*update_depsgraph*/ nullptr,
205 /*depends_on_time*/ nullptr,
206 /*depends_on_normals*/ nullptr,
207 /*foreach_ID_link*/ nullptr,
208 /*foreach_tex_link*/ nullptr,
209 /*free_runtime_data*/ nullptr,
210 /*panel_register*/ panel_register,
211 /*blend_write*/ nullptr,
212 /*blend_read*/ nullptr,
213 /*foreach_cache*/ nullptr,
214};
CustomData interface, see also DNA_customdata_types.h.
const void * CustomData_get_layer(const CustomData *data, eCustomDataType type)
support for deformation groups and hooks.
int BKE_id_defgroup_name_index(const ID *id, blender::StringRef name)
Definition deform.cc:538
float BKE_defvert_find_weight(const MDeformVert *dvert, int defgroup)
Definition deform.cc:763
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
@ eModifierTypeFlag_AcceptsCVs
@ eModifierTypeFlag_SupportsMapping
@ eModifierTypeFlag_EnableInEditmode
@ eModifierTypeFlag_SupportsEditmode
@ eModifierTypeFlag_AcceptsMesh
#define BLI_assert_unreachable()
Definition BLI_assert.h:93
#define BLI_assert(a)
Definition BLI_assert.h:46
#define MEMCMP_STRUCT_AFTER_IS_ZERO(struct_var, member)
#define MEMCPY_STRUCT_AFTER(struct_dst, struct_src, member)
#define IFACE_(msgid)
@ CD_MDEFORMVERT
#define DNA_struct_default_get(struct_name)
@ MOD_WELD_LOOSE_EDGES
@ MOD_WELD_INVERT_VGROUP
@ eModifierType_Weld
@ MOD_WELD_MODE_CONNECTED
@ MOD_WELD_MODE_ALL
static void init_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
static void required_data_mask(ModifierData *, CustomData_MeshMasks *r_cddata_masks)
static void panel_draw(const bContext *, Panel *panel)
static Mesh * modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
Definition MOD_array.cc:862
void modifier_vgroup_ui(uiLayout *layout, PointerRNA *ptr, PointerRNA *ob_ptr, const StringRefNull vgroup_prop, const std::optional< StringRefNull > invert_vgroup_prop, const std::optional< StringRefNull > text)
PanelType * modifier_panel_register(ARegionType *region_type, ModifierType type, PanelDrawFn draw)
PointerRNA * modifier_panel_get_property_pointers(Panel *panel, PointerRNA *r_ob_ptr)
void modifier_error_message_draw(uiLayout *layout, PointerRNA *ptr)
static void init_data(ModifierData *md)
Definition MOD_weld.cc:134
static void panel_register(ARegionType *region_type)
Definition MOD_weld.cc:173
static IndexMask selected_indices_from_vertex_group(Span< MDeformVert > vertex_group, const int index, const bool invert, IndexMaskMemory &memory)
Definition MOD_weld.cc:66
ModifierTypeInfo modifierType_Weld
Definition MOD_weld.cc:178
static Span< MDeformVert > get_vertex_group(const Mesh &mesh, const int defgrp_index)
Definition MOD_weld.cc:53
static Mesh * modify_mesh(ModifierData *md, const ModifierEvalContext *, Mesh *mesh)
Definition MOD_weld.cc:123
static void panel_draw(const bContext *, Panel *panel)
Definition MOD_weld.cc:153
static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
Definition MOD_weld.cc:143
static std::optional< Mesh * > calculate_weld(const Mesh &mesh, const WeldModifierData &wmd)
Definition MOD_weld.cc:89
static Array< bool > selection_array_from_vertex_group(Span< MDeformVert > vertex_group, const int index, const bool invert)
Definition MOD_weld.cc:77
void uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
#define UI_ITEM_NONE
static IndexMask from_predicate(const IndexMask &universe, GrainSize grain_size, IndexMaskMemory &memory, Fn &&predicate)
constexpr int64_t size() const
Definition BLI_span.hh:252
constexpr IndexRange index_range() const
Definition BLI_span.hh:401
constexpr bool is_empty() const
Definition BLI_span.hh:260
#define CD_MASK_MDEFORMVERT
CCL_NAMESPACE_BEGIN ccl_device float invert(const float color, const float factor)
Definition invert.h:11
std::optional< Mesh * > mesh_merge_by_distance_connected(const Mesh &mesh, Span< bool > selection, float merge_distance, bool only_loose_edges)
std::optional< Mesh * > mesh_merge_by_distance_all(const Mesh &mesh, const IndexMask &selection, float merge_distance)
int RNA_enum_get(PointerRNA *ptr, const char *name)
CustomData vert_data
int verts_num
struct uiLayout * layout
void prop(PointerRNA *ptr, PropertyRNA *prop, int index, int value, eUI_Item_Flag flag, std::optional< blender::StringRef > name_opt, int icon, std::optional< blender::StringRef > placeholder=std::nullopt)
i
Definition text_draw.cc:230
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4226