Blender V5.0
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_deform.hh"
32#include "BKE_modifier.hh"
33#include "BKE_screen.hh"
34
36#include "UI_resources.hh"
37
38#include "RNA_access.hh"
39#include "RNA_prototypes.hh"
40
41#include "MOD_modifiertypes.hh"
42#include "MOD_ui_common.hh"
43
45
46using blender::Array;
49using blender::Span;
50using blender::Vector;
51
52static Span<MDeformVert> get_vertex_group(const Mesh &mesh, const int defgrp_index)
53{
54 if (defgrp_index == -1) {
55 return {};
56 }
57 return mesh.deform_verts();
58}
59
61 const int index,
62 const bool invert,
63 IndexMaskMemory &memory)
64{
66 vertex_group.index_range(), blender::GrainSize(512), memory, [&](const int i) {
67 return (BKE_defvert_find_weight(&vertex_group[i], index) > 0.0f) != invert;
68 });
69}
70
72 const int index,
73 const bool invert)
74{
75 Array<bool> selection(vertex_group.size());
76 for (const int i : vertex_group.index_range()) {
77 const bool found = BKE_defvert_find_weight(&vertex_group[i], index) > 0.0f;
78 selection[i] = (found != invert);
79 }
80 return selection;
81}
82
83static std::optional<Mesh *> calculate_weld(const Mesh &mesh, const WeldModifierData &wmd)
84{
85 const int defgrp_index = BKE_id_defgroup_name_index(&mesh.id, wmd.defgrp_name);
86 Span<MDeformVert> vertex_group = get_vertex_group(mesh, defgrp_index);
87 const bool invert = (wmd.flag & MOD_WELD_INVERT_VGROUP) != 0;
88
89 if (wmd.mode == MOD_WELD_MODE_ALL) {
90 if (!vertex_group.is_empty()) {
91 IndexMaskMemory memory;
92 const IndexMask selected_indices = selected_indices_from_vertex_group(
93 vertex_group, defgrp_index, invert, memory);
95 mesh, IndexMask(selected_indices), wmd.merge_dist);
96 }
98 mesh, IndexMask(mesh.verts_num), wmd.merge_dist);
99 }
100 if (wmd.mode == MOD_WELD_MODE_CONNECTED) {
101 const bool only_loose_edges = (wmd.flag & MOD_WELD_LOOSE_EDGES) != 0;
102 if (!vertex_group.is_empty()) {
104 vertex_group, defgrp_index, invert);
106 mesh, selection, wmd.merge_dist, only_loose_edges);
107 }
108 Array<bool> selection(mesh.verts_num, true);
110 mesh, selection, wmd.merge_dist, only_loose_edges);
111 }
112
114 return nullptr;
115}
116
117static Mesh *modify_mesh(ModifierData *md, const ModifierEvalContext * /*ctx*/, Mesh *mesh)
118{
119 const WeldModifierData &wmd = reinterpret_cast<WeldModifierData &>(*md);
120
121 std::optional<Mesh *> result = calculate_weld(*mesh, wmd);
122 if (!result) {
123 return mesh;
124 }
125 return *result;
126}
127
128static void init_data(ModifierData *md)
129{
131
133
135}
136
137static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
138{
140
141 /* Ask for vertex-groups if we need them. */
142 if (wmd->defgrp_name[0] != '\0') {
143 r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT;
144 }
145}
146
147static void panel_draw(const bContext * /*C*/, Panel *panel)
148{
149 uiLayout *layout = panel->layout;
150
151 PointerRNA ob_ptr;
153 int weld_mode = RNA_enum_get(ptr, "mode");
154
155 layout->use_property_split_set(true);
156
157 layout->prop(ptr, "mode", UI_ITEM_NONE, std::nullopt, ICON_NONE);
158 layout->prop(ptr, "merge_threshold", UI_ITEM_NONE, IFACE_("Distance"), ICON_NONE);
159 if (weld_mode == MOD_WELD_MODE_CONNECTED) {
160 layout->prop(ptr, "loose_edges", UI_ITEM_NONE, std::nullopt, ICON_NONE);
161 }
162 modifier_vgroup_ui(layout, ptr, &ob_ptr, "vertex_group", "invert_vertex_group", std::nullopt);
163
165}
166
167static void panel_register(ARegionType *region_type)
168{
170}
171
173 /*idname*/ "Weld",
174 /*name*/ N_("Weld"),
175 /*struct_name*/ "WeldModifierData",
176 /*struct_size*/ sizeof(WeldModifierData),
177 /*srna*/ &RNA_WeldModifier,
179 /*flags*/
183 /*icon*/ ICON_AUTOMERGE_OFF, /* TODO: Use correct icon. */
184
185 /*copy_data*/ BKE_modifier_copydata_generic,
186
187 /*deform_verts*/ nullptr,
188 /*deform_matrices*/ nullptr,
189 /*deform_verts_EM*/ nullptr,
190 /*deform_matrices_EM*/ nullptr,
191 /*modify_mesh*/ modify_mesh,
192 /*modify_geometry_set*/ nullptr,
193
194 /*init_data*/ init_data,
195 /*required_data_mask*/ required_data_mask,
196 /*free_data*/ nullptr,
197 /*is_disabled*/ nullptr,
198 /*update_depsgraph*/ nullptr,
199 /*depends_on_time*/ nullptr,
200 /*depends_on_normals*/ nullptr,
201 /*foreach_ID_link*/ nullptr,
202 /*foreach_tex_link*/ nullptr,
203 /*free_runtime_data*/ nullptr,
204 /*panel_register*/ panel_register,
205 /*blend_write*/ nullptr,
206 /*blend_read*/ nullptr,
207 /*foreach_cache*/ nullptr,
208 /*foreach_working_space_color*/ nullptr,
209};
support for deformation groups and hooks.
int BKE_id_defgroup_name_index(const ID *id, blender::StringRef name)
Definition deform.cc:549
float BKE_defvert_find_weight(const MDeformVert *dvert, int defgroup)
Definition deform.cc:774
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)
#define CD_MASK_MDEFORMVERT
#define DNA_struct_default_get(struct_name)
@ MOD_WELD_MODE_CONNECTED
@ MOD_WELD_MODE_ALL
@ eModifierType_Weld
@ MOD_WELD_LOOSE_EDGES
@ MOD_WELD_INVERT_VGROUP
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:128
static void panel_register(ARegionType *region_type)
Definition MOD_weld.cc:167
static IndexMask selected_indices_from_vertex_group(Span< MDeformVert > vertex_group, const int index, const bool invert, IndexMaskMemory &memory)
Definition MOD_weld.cc:60
ModifierTypeInfo modifierType_Weld
Definition MOD_weld.cc:172
static Span< MDeformVert > get_vertex_group(const Mesh &mesh, const int defgrp_index)
Definition MOD_weld.cc:52
static Mesh * modify_mesh(ModifierData *md, const ModifierEvalContext *, Mesh *mesh)
Definition MOD_weld.cc:117
static void panel_draw(const bContext *, Panel *panel)
Definition MOD_weld.cc:147
static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
Definition MOD_weld.cc:137
static std::optional< Mesh * > calculate_weld(const Mesh &mesh, const WeldModifierData &wmd)
Definition MOD_weld.cc:83
static Array< bool > selection_array_from_vertex_group(Span< MDeformVert > vertex_group, const int index, const bool invert)
Definition MOD_weld.cc:71
#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
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)
int verts_num
struct uiLayout * layout
void use_property_split_set(bool value)
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:4238