Blender V5.0
MOD_uvwarp.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2023 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
8
9#include <cstring>
10
11#include "BLI_utildefines.h"
12
13#include "BLI_math_matrix.h"
14#include "BLI_math_vector.h"
15#include "BLI_task.h"
16
17#include "BLT_translation.hh"
18
19#include "DNA_defaults.h"
20#include "DNA_mesh_types.h"
21#include "DNA_meshdata_types.h"
22#include "DNA_object_types.h"
23#include "DNA_screen_types.h"
24
25#include "BKE_action.hh" /* BKE_pose_channel_find_name */
26#include "BKE_attribute.hh"
27#include "BKE_customdata.hh"
28#include "BKE_deform.hh"
29#include "BKE_lib_query.hh"
30#include "BKE_mesh.hh"
31#include "BKE_modifier.hh"
32
34#include "UI_resources.hh"
35
36#include "RNA_access.hh"
37#include "RNA_prototypes.hh"
38
39#include "MOD_ui_common.hh"
40#include "MOD_util.hh"
41
42static void uv_warp_from_mat4_pair(float uv_dst[2],
43 const float uv_src[2],
44 const float warp_mat[4][4])
45{
46 float tuv[3] = {0.0f};
47
48 copy_v2_v2(tuv, uv_src);
49 mul_m4_v3(warp_mat, tuv);
50 copy_v2_v2(uv_dst, tuv);
51}
52
61
62static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
63{
65
66 /* Ask for vertex-groups if we need them. */
67 if (umd->vgroup_name[0] != '\0') {
68 r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT;
69 }
70}
71
72static void matrix_from_obj_pchan(float mat[4][4], Object *ob, const char *bonename)
73{
74 bPoseChannel *pchan = BKE_pose_channel_find_name(ob->pose, bonename);
75 if (pchan) {
76 mul_m4_m4m4(mat, ob->object_to_world().ptr(), pchan->pose_mat);
77 }
78 else {
79 copy_m4_m4(mat, ob->object_to_world().ptr());
80 }
81}
82
94
95static void uv_warp_compute(void *__restrict userdata,
96 const int i,
97 const TaskParallelTLS *__restrict /*tls*/)
98{
99 const UVWarpData *data = static_cast<const UVWarpData *>(userdata);
100 const blender::IndexRange face = data->faces[i];
101 const blender::Span<int> face_verts = data->corner_verts.slice(face);
102
103 blender::float2 *mluv = &data->uv_map[face.start()];
104
105 const MDeformVert *dvert = data->dvert;
106 const int defgrp_index = data->defgrp_index;
107
108 float (*warp_mat)[4] = data->warp_mat;
109
110 int l;
111
112 if (dvert) {
113 for (l = 0; l < face.size(); l++, mluv++) {
114 const int vert_i = face_verts[l];
115 float uv[2];
116 const float weight = data->invert_vgroup ?
117 1.0f - BKE_defvert_find_weight(&dvert[vert_i], defgrp_index) :
118 BKE_defvert_find_weight(&dvert[vert_i], defgrp_index);
119
120 uv_warp_from_mat4_pair(uv, (*mluv), warp_mat);
121 interp_v2_v2v2((*mluv), (*mluv), uv, weight);
122 }
123 }
124 else {
125 for (l = 0; l < face.size(); l++, mluv++) {
126 uv_warp_from_mat4_pair(*mluv, *mluv, warp_mat);
127 }
128 }
129}
130
131static Mesh *modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
132{
134 const MDeformVert *dvert;
135 int defgrp_index;
136 char uvname[MAX_CUSTOMDATA_LAYER_NAME];
137 float warp_mat[4][4];
138 const int axis_u = umd->axis_u;
139 const int axis_v = umd->axis_v;
140 const bool invert_vgroup = (umd->flag & MOD_UVWARP_INVERT_VGROUP) != 0;
141
142 /* make sure there are UV Maps available */
144 return mesh;
145 }
146
147 if (!ELEM(nullptr, umd->object_src, umd->object_dst)) {
148 float mat_src[4][4];
149 float mat_dst[4][4];
150 float imat_dst[4][4];
151 float shuf_mat[4][4];
152
153 /* make sure anything moving UVs is available */
154 matrix_from_obj_pchan(mat_src, umd->object_src, umd->bone_src);
155 matrix_from_obj_pchan(mat_dst, umd->object_dst, umd->bone_dst);
156
157 invert_m4_m4(imat_dst, mat_dst);
158 mul_m4_m4m4(warp_mat, imat_dst, mat_src);
159
160 /* apply warp */
161 if (!is_zero_v2(umd->center)) {
162 float mat_cent[4][4];
163 float imat_cent[4][4];
164
165 unit_m4(mat_cent);
166 mat_cent[3][axis_u] = umd->center[0];
167 mat_cent[3][axis_v] = umd->center[1];
168
169 invert_m4_m4(imat_cent, mat_cent);
170
171 mul_m4_m4m4(warp_mat, warp_mat, imat_cent);
172 mul_m4_m4m4(warp_mat, mat_cent, warp_mat);
173 }
174
175 const int shuf_indices[4] = {axis_u, axis_v, -1, 3};
176 shuffle_m4(shuf_mat, shuf_indices);
177 mul_m4_m4m4(warp_mat, shuf_mat, warp_mat);
178 transpose_m4(shuf_mat);
179 mul_m4_m4m4(warp_mat, warp_mat, shuf_mat);
180 }
181 else {
182 unit_m4(warp_mat);
183 }
184
185 /* Apply direct 2d transform. */
186 translate_m4(warp_mat, umd->center[0], umd->center[1], 0.0f);
187 const float scale[3] = {umd->scale[0], umd->scale[1], 1.0f};
188 rescale_m4(warp_mat, scale);
189 rotate_m4(warp_mat, 'Z', umd->rotation);
190 translate_m4(warp_mat, umd->offset[0], umd->offset[1], 0.0f);
191 translate_m4(warp_mat, -umd->center[0], -umd->center[1], 0.0f);
192
193 /* make sure we're using an existing layer */
195
196 const blender::OffsetIndices faces = mesh->faces();
197 const blender::Span<int> corner_verts = mesh->corner_verts();
198
199 blender::bke::MutableAttributeAccessor attributes = mesh->attributes_for_write();
203 MOD_get_vgroup(ctx->object, mesh, umd->vgroup_name, &dvert, &defgrp_index);
204
206 data.faces = faces;
207 data.corner_verts = corner_verts;
208 data.uv_map = uv_map.span;
209 data.dvert = dvert;
210 data.defgrp_index = defgrp_index;
211 data.warp_mat = warp_mat;
212 data.invert_vgroup = invert_vgroup;
213
214 TaskParallelSettings settings;
216 settings.use_threading = (faces.size() > 1000);
217 BLI_task_parallel_range(0, faces.size(), &data, uv_warp_compute, &settings);
218
219 mesh->runtime->is_original_bmesh = false;
220
221 uv_map.finish();
222
223 return mesh;
224}
225
226static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
227{
229
230 walk(user_data, ob, (ID **)&umd->object_dst, IDWALK_CB_NOP);
231 walk(user_data, ob, (ID **)&umd->object_src, IDWALK_CB_NOP);
232}
233
235{
237
239 ctx->node, umd->object_src, umd->bone_src, "UVWarp Modifier");
241 ctx->node, umd->object_dst, umd->bone_dst, "UVWarp Modifier");
242
243 DEG_add_depends_on_transform_relation(ctx->node, "UVWarp Modifier");
244}
245
246static void panel_draw(const bContext * /*C*/, Panel *panel)
247{
248 uiLayout *col;
249 uiLayout *layout = panel->layout;
250
251 PointerRNA ob_ptr;
253
254 PointerRNA warp_obj_ptr;
255 PointerRNA obj_data_ptr = RNA_pointer_get(&ob_ptr, "data");
256
257 layout->use_property_split_set(true);
258
259 layout->prop_search(ptr, "uv_layer", &obj_data_ptr, "uv_layers", std::nullopt, ICON_GROUP_UVS);
260
261 col = &layout->column(false);
262 col->prop(ptr, "center", UI_ITEM_NONE, std::nullopt, ICON_NONE);
263
264 col = &layout->column(false);
265 col->prop(ptr, "axis_u", UI_ITEM_NONE, IFACE_("Axis U"), ICON_NONE);
266 col->prop(ptr, "axis_v", UI_ITEM_NONE, IFACE_("V"), ICON_NONE);
267
268 col = &layout->column(false);
269 col->prop(ptr, "object_from", UI_ITEM_NONE, std::nullopt, ICON_NONE);
270 warp_obj_ptr = RNA_pointer_get(ptr, "object_from");
271 if (!RNA_pointer_is_null(&warp_obj_ptr) && RNA_enum_get(&warp_obj_ptr, "type") == OB_ARMATURE) {
272 PointerRNA warp_obj_data_ptr = RNA_pointer_get(&warp_obj_ptr, "data");
273 col->prop_search(ptr, "bone_from", &warp_obj_data_ptr, "bones", std::nullopt, ICON_BONE_DATA);
274 }
275
276 col->prop(ptr, "object_to", UI_ITEM_NONE, CTX_IFACE_(BLT_I18NCONTEXT_MODIFIER, "To"), ICON_NONE);
277 warp_obj_ptr = RNA_pointer_get(ptr, "object_to");
278 if (!RNA_pointer_is_null(&warp_obj_ptr) && RNA_enum_get(&warp_obj_ptr, "type") == OB_ARMATURE) {
279 PointerRNA warp_obj_data_ptr = RNA_pointer_get(&warp_obj_ptr, "data");
280 col->prop_search(ptr, "bone_to", &warp_obj_data_ptr, "bones", std::nullopt, ICON_BONE_DATA);
281 }
282
283 modifier_vgroup_ui(layout, ptr, &ob_ptr, "vertex_group", "invert_vertex_group", std::nullopt);
284
286}
287
288static void transform_panel_draw(const bContext * /*C*/, Panel *panel)
289{
290 uiLayout *layout = panel->layout;
291
293
294 layout->use_property_split_set(true);
295
296 layout->prop(ptr, "offset", UI_ITEM_NONE, std::nullopt, ICON_NONE);
297 layout->prop(ptr, "scale", UI_ITEM_NONE, std::nullopt, ICON_NONE);
298 layout->prop(ptr, "rotation", UI_ITEM_NONE, std::nullopt, ICON_NONE);
299}
300
301static void panel_register(ARegionType *region_type)
302{
305 region_type, "offset", "Transform", nullptr, transform_panel_draw, panel_type);
306}
307
309 /*idname*/ "UVWarp",
310 /*name*/ N_("UVWarp"),
311 /*struct_name*/ "UVWarpModifierData",
312 /*struct_size*/ sizeof(UVWarpModifierData),
313 /*srna*/ &RNA_UVWarpModifier,
317 /*icon*/ ICON_MOD_UVPROJECT, /* TODO: Use correct icon. */
318
319 /*copy_data*/ BKE_modifier_copydata_generic,
320
321 /*deform_verts*/ nullptr,
322 /*deform_matrices*/ nullptr,
323 /*deform_verts_EM*/ nullptr,
324 /*deform_matrices_EM*/ nullptr,
325 /*modify_mesh*/ modify_mesh,
326 /*modify_geometry_set*/ nullptr,
327
328 /*init_data*/ init_data,
329 /*required_data_mask*/ required_data_mask,
330 /*free_data*/ nullptr,
331 /*is_disabled*/ nullptr,
332 /*update_depsgraph*/ update_depsgraph,
333 /*depends_on_time*/ nullptr,
334 /*depends_on_normals*/ nullptr,
335 /*foreach_ID_link*/ foreach_ID_link,
336 /*foreach_tex_link*/ nullptr,
337 /*free_runtime_data*/ nullptr,
338 /*panel_register*/ panel_register,
339 /*blend_write*/ nullptr,
340 /*blend_read*/ nullptr,
341 /*foreach_cache*/ nullptr,
342 /*foreach_working_space_color*/ nullptr,
343};
Blender kernel action and pose functionality.
bPoseChannel * BKE_pose_channel_find_name(const bPose *pose, const char *name)
CustomData interface, see also DNA_customdata_types.h.
void CustomData_validate_layer_name(const CustomData *data, eCustomDataType type, blender::StringRef name, char *outname)
bool CustomData_has_layer(const CustomData *data, eCustomDataType type)
support for deformation groups and hooks.
float BKE_defvert_find_weight(const MDeformVert *dvert, int defgroup)
Definition deform.cc:774
@ IDWALK_CB_NOP
void(*)(void *user_data, Object *ob, ID **idpoin, LibraryForeachIDCallbackFlag cb_flag) IDWalkFunc
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
@ eModifierTypeFlag_EnableInEditmode
@ eModifierTypeFlag_SupportsEditmode
@ eModifierTypeFlag_AcceptsMesh
#define BLI_assert(a)
Definition BLI_assert.h:46
void mul_m4_m4m4(float R[4][4], const float A[4][4], const float B[4][4])
void translate_m4(float mat[4][4], float Tx, float Ty, float Tz)
void rescale_m4(float mat[4][4], const float scale[3])
void mul_m4_v3(const float M[4][4], float r[3])
void copy_m4_m4(float m1[4][4], const float m2[4][4])
void shuffle_m4(float R[4][4], const int index[4])
bool invert_m4_m4(float inverse[4][4], const float mat[4][4])
void rotate_m4(float mat[4][4], char axis, float angle)
void transpose_m4(float R[4][4])
void unit_m4(float m[4][4])
void interp_v2_v2v2(float r[2], const float a[2], const float b[2], float t)
MINLINE void copy_v2_v2(float r[2], const float a[2])
MINLINE bool is_zero_v2(const float v[2]) ATTR_WARN_UNUSED_RESULT
void BLI_task_parallel_range(int start, int stop, void *userdata, TaskParallelRangeFunc func, const TaskParallelSettings *settings)
Definition task_range.cc:99
BLI_INLINE void BLI_parallel_range_settings_defaults(TaskParallelSettings *settings)
Definition BLI_task.h:221
#define ELEM(...)
#define MEMCMP_STRUCT_AFTER_IS_ZERO(struct_var, member)
#define MEMCPY_STRUCT_AFTER(struct_dst, struct_src, member)
#define BLT_I18NCONTEXT_MODIFIER
#define CTX_IFACE_(context, msgid)
#define IFACE_(msgid)
void DEG_add_depends_on_transform_relation(DepsNodeHandle *node_handle, const char *description)
#define MAX_CUSTOMDATA_LAYER_NAME
#define CD_MASK_MDEFORMVERT
@ CD_PROP_FLOAT2
#define DNA_struct_default_get(struct_name)
@ eModifierType_UVWarp
@ MOD_UVWARP_INVERT_VGROUP
Object is a sort of wrapper for general info.
@ OB_ARMATURE
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 void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
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_subpanel_register(ARegionType *region_type, const char *name, const char *label, PanelDrawFn draw_header, PanelDrawFn draw, PanelType *parent)
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)
void MOD_depsgraph_update_object_bone_relation(DepsNodeHandle *node, Object *object, const char *bonename, const char *description)
Definition MOD_util.cc:186
void MOD_get_vgroup(const Object *ob, const Mesh *mesh, const char *name, const MDeformVert **dvert, int *defgrp_index)
Definition MOD_util.cc:156
static void init_data(ModifierData *md)
Definition MOD_uvwarp.cc:53
static void panel_register(ARegionType *region_type)
static void matrix_from_obj_pchan(float mat[4][4], Object *ob, const char *bonename)
Definition MOD_uvwarp.cc:72
static Mesh * modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
static void uv_warp_compute(void *__restrict userdata, const int i, const TaskParallelTLS *__restrict)
Definition MOD_uvwarp.cc:95
static void uv_warp_from_mat4_pair(float uv_dst[2], const float uv_src[2], const float warp_mat[4][4])
Definition MOD_uvwarp.cc:42
static void panel_draw(const bContext *, Panel *panel)
static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
Definition MOD_uvwarp.cc:62
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
ModifierTypeInfo modifierType_UVWarp
static void update_depsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
static void transform_panel_draw(const bContext *, Panel *panel)
#define UI_ITEM_NONE
BMesh const char void * data
ATTR_WARN_UNUSED_RESULT const BMLoop * l
constexpr int64_t size() const
constexpr int64_t start() const
GSpanAttributeWriter lookup_or_add_for_write_span(StringRef attribute_id, AttrDomain domain, AttrType data_type, const AttributeInit &initializer=AttributeInitDefaultValue())
nullptr float
uint col
static void update_depsgraph(tGraphSliderOp *gso)
static char faces[256]
VecBase< float, 2 > float2
PointerRNA RNA_pointer_get(PointerRNA *ptr, const char *name)
bool RNA_pointer_is_null(const PointerRNA *ptr)
int RNA_enum_get(PointerRNA *ptr, const char *name)
Definition DNA_ID.h:414
MeshRuntimeHandle * runtime
CustomData corner_data
struct bPose * pose
struct uiLayout * layout
blender::Span< int > corner_verts
Definition MOD_uvwarp.cc:85
blender::OffsetIndices< int > faces
Definition MOD_uvwarp.cc:84
blender::MutableSpan< blender::float2 > uv_map
Definition MOD_uvwarp.cc:86
bool invert_vgroup
Definition MOD_uvwarp.cc:92
int defgrp_index
Definition MOD_uvwarp.cc:89
float(* warp_mat)[4]
Definition MOD_uvwarp.cc:91
const MDeformVert * dvert
Definition MOD_uvwarp.cc:88
struct Object * object_dst
struct Object * object_src
float pose_mat[4][4]
uiLayout & column(bool align)
void prop_search(PointerRNA *ptr, PropertyRNA *prop, PointerRNA *searchptr, PropertyRNA *searchprop, PropertyRNA *item_searchpropname, std::optional< blender::StringRefNull > name, int icon, bool results_are_suggestions)
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