Blender V4.3
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
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_customdata.hh"
27#include "BKE_deform.hh"
28#include "BKE_lib_query.hh"
29#include "BKE_mesh.hh"
30#include "BKE_modifier.hh"
31
32#include "UI_interface.hh"
33#include "UI_resources.hh"
34
35#include "RNA_access.hh"
36#include "RNA_prototypes.hh"
37
38#include "MOD_ui_common.hh"
39#include "MOD_util.hh"
40
41static void uv_warp_from_mat4_pair(float uv_dst[2],
42 const float uv_src[2],
43 const float warp_mat[4][4])
44{
45 float tuv[3] = {0.0f};
46
47 copy_v2_v2(tuv, uv_src);
48 mul_m4_v3(warp_mat, tuv);
49 copy_v2_v2(uv_dst, tuv);
50}
51
60
61static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
62{
64
65 /* Ask for vertex-groups if we need them. */
66 if (umd->vgroup_name[0] != '\0') {
67 r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT;
68 }
69}
70
71static void matrix_from_obj_pchan(float mat[4][4], Object *ob, const char *bonename)
72{
73 bPoseChannel *pchan = BKE_pose_channel_find_name(ob->pose, bonename);
74 if (pchan) {
75 mul_m4_m4m4(mat, ob->object_to_world().ptr(), pchan->pose_mat);
76 }
77 else {
78 copy_m4_m4(mat, ob->object_to_world().ptr());
79 }
80}
81
93
94static void uv_warp_compute(void *__restrict userdata,
95 const int i,
96 const TaskParallelTLS *__restrict /*tls*/)
97{
98 const UVWarpData *data = static_cast<const UVWarpData *>(userdata);
99 const blender::IndexRange face = data->faces[i];
100 const blender::Span<int> face_verts = data->corner_verts.slice(face);
101
102 float(*mluv)[2] = &data->mloopuv[face.start()];
103
104 const MDeformVert *dvert = data->dvert;
105 const int defgrp_index = data->defgrp_index;
106
107 float(*warp_mat)[4] = data->warp_mat;
108
109 int l;
110
111 if (dvert) {
112 for (l = 0; l < face.size(); l++, mluv++) {
113 const int vert_i = face_verts[l];
114 float uv[2];
115 const float weight = data->invert_vgroup ?
116 1.0f - BKE_defvert_find_weight(&dvert[vert_i], defgrp_index) :
117 BKE_defvert_find_weight(&dvert[vert_i], defgrp_index);
118
119 uv_warp_from_mat4_pair(uv, (*mluv), warp_mat);
120 interp_v2_v2v2((*mluv), (*mluv), uv, weight);
121 }
122 }
123 else {
124 for (l = 0; l < face.size(); l++, mluv++) {
125 uv_warp_from_mat4_pair(*mluv, *mluv, warp_mat);
126 }
127 }
128}
129
130static Mesh *modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
131{
133 const MDeformVert *dvert;
134 int defgrp_index;
135 char uvname[MAX_CUSTOMDATA_LAYER_NAME];
136 float warp_mat[4][4];
137 const int axis_u = umd->axis_u;
138 const int axis_v = umd->axis_v;
139 const bool invert_vgroup = (umd->flag & MOD_UVWARP_INVERT_VGROUP) != 0;
140
141 /* make sure there are UV Maps available */
142 if (!CustomData_has_layer(&mesh->corner_data, CD_PROP_FLOAT2)) {
143 return mesh;
144 }
145
146 if (!ELEM(nullptr, umd->object_src, umd->object_dst)) {
147 float mat_src[4][4];
148 float mat_dst[4][4];
149 float imat_dst[4][4];
150 float shuf_mat[4][4];
151
152 /* make sure anything moving UVs is available */
153 matrix_from_obj_pchan(mat_src, umd->object_src, umd->bone_src);
154 matrix_from_obj_pchan(mat_dst, umd->object_dst, umd->bone_dst);
155
156 invert_m4_m4(imat_dst, mat_dst);
157 mul_m4_m4m4(warp_mat, imat_dst, mat_src);
158
159 /* apply warp */
160 if (!is_zero_v2(umd->center)) {
161 float mat_cent[4][4];
162 float imat_cent[4][4];
163
164 unit_m4(mat_cent);
165 mat_cent[3][axis_u] = umd->center[0];
166 mat_cent[3][axis_v] = umd->center[1];
167
168 invert_m4_m4(imat_cent, mat_cent);
169
170 mul_m4_m4m4(warp_mat, warp_mat, imat_cent);
171 mul_m4_m4m4(warp_mat, mat_cent, warp_mat);
172 }
173
174 const int shuf_indices[4] = {axis_u, axis_v, -1, 3};
175 shuffle_m4(shuf_mat, shuf_indices);
176 mul_m4_m4m4(warp_mat, shuf_mat, warp_mat);
177 transpose_m4(shuf_mat);
178 mul_m4_m4m4(warp_mat, warp_mat, shuf_mat);
179 }
180 else {
181 unit_m4(warp_mat);
182 }
183
184 /* Apply direct 2d transform. */
185 translate_m4(warp_mat, umd->center[0], umd->center[1], 0.0f);
186 const float scale[3] = {umd->scale[0], umd->scale[1], 1.0f};
187 rescale_m4(warp_mat, scale);
188 rotate_m4(warp_mat, 'Z', umd->rotation);
189 translate_m4(warp_mat, umd->offset[0], umd->offset[1], 0.0f);
190 translate_m4(warp_mat, -umd->center[0], -umd->center[1], 0.0f);
191
192 /* make sure we're using an existing layer */
193 CustomData_validate_layer_name(&mesh->corner_data, CD_PROP_FLOAT2, umd->uvlayer_name, uvname);
194
195 const blender::OffsetIndices faces = mesh->faces();
196 const blender::Span<int> corner_verts = mesh->corner_verts();
197
198 float(*mloopuv)[2] = static_cast<float(*)[2]>(CustomData_get_layer_named_for_write(
199 &mesh->corner_data, CD_PROP_FLOAT2, uvname, corner_verts.size()));
200 MOD_get_vgroup(ctx->object, mesh, umd->vgroup_name, &dvert, &defgrp_index);
201
202 UVWarpData data{};
203 data.faces = faces;
204 data.corner_verts = corner_verts;
205 data.mloopuv = mloopuv;
206 data.dvert = dvert;
207 data.defgrp_index = defgrp_index;
208 data.warp_mat = warp_mat;
209 data.invert_vgroup = invert_vgroup;
210
211 TaskParallelSettings settings;
213 settings.use_threading = (faces.size() > 1000);
214 BLI_task_parallel_range(0, faces.size(), &data, uv_warp_compute, &settings);
215
216 mesh->runtime->is_original_bmesh = false;
217
218 return mesh;
219}
220
221static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
222{
224
225 walk(user_data, ob, (ID **)&umd->object_dst, IDWALK_CB_NOP);
226 walk(user_data, ob, (ID **)&umd->object_src, IDWALK_CB_NOP);
227}
228
230{
232
234 ctx->node, umd->object_src, umd->bone_src, "UVWarp Modifier");
236 ctx->node, umd->object_dst, umd->bone_dst, "UVWarp Modifier");
237
238 DEG_add_depends_on_transform_relation(ctx->node, "UVWarp Modifier");
239}
240
241static void panel_draw(const bContext * /*C*/, Panel *panel)
242{
243 uiLayout *col;
244 uiLayout *layout = panel->layout;
245
246 PointerRNA ob_ptr;
248
249 PointerRNA warp_obj_ptr;
250 PointerRNA obj_data_ptr = RNA_pointer_get(&ob_ptr, "data");
251
252 uiLayoutSetPropSep(layout, true);
253
254 uiItemPointerR(layout, ptr, "uv_layer", &obj_data_ptr, "uv_layers", nullptr, ICON_GROUP_UVS);
255
256 col = uiLayoutColumn(layout, false);
257 uiItemR(col, ptr, "center", UI_ITEM_NONE, nullptr, ICON_NONE);
258
259 col = uiLayoutColumn(layout, false);
260 uiItemR(col, ptr, "axis_u", UI_ITEM_NONE, IFACE_("Axis U"), ICON_NONE);
261 uiItemR(col, ptr, "axis_v", UI_ITEM_NONE, IFACE_("V"), ICON_NONE);
262
263 col = uiLayoutColumn(layout, false);
264 uiItemR(col, ptr, "object_from", UI_ITEM_NONE, nullptr, ICON_NONE);
265 warp_obj_ptr = RNA_pointer_get(ptr, "object_from");
266 if (!RNA_pointer_is_null(&warp_obj_ptr) && RNA_enum_get(&warp_obj_ptr, "type") == OB_ARMATURE) {
267 PointerRNA warp_obj_data_ptr = RNA_pointer_get(&warp_obj_ptr, "data");
268 uiItemPointerR(col, ptr, "bone_from", &warp_obj_data_ptr, "bones", nullptr, ICON_BONE_DATA);
269 }
270
271 uiItemR(col, ptr, "object_to", UI_ITEM_NONE, IFACE_("To"), ICON_NONE);
272 warp_obj_ptr = RNA_pointer_get(ptr, "object_to");
273 if (!RNA_pointer_is_null(&warp_obj_ptr) && RNA_enum_get(&warp_obj_ptr, "type") == OB_ARMATURE) {
274 PointerRNA warp_obj_data_ptr = RNA_pointer_get(&warp_obj_ptr, "data");
275 uiItemPointerR(col, ptr, "bone_to", &warp_obj_data_ptr, "bones", nullptr, ICON_BONE_DATA);
276 }
277
278 modifier_vgroup_ui(layout, ptr, &ob_ptr, "vertex_group", "invert_vertex_group", nullptr);
279
280 modifier_panel_end(layout, ptr);
281}
282
283static void transform_panel_draw(const bContext * /*C*/, Panel *panel)
284{
285 uiLayout *layout = panel->layout;
286
288
289 uiLayoutSetPropSep(layout, true);
290
291 uiItemR(layout, ptr, "offset", UI_ITEM_NONE, nullptr, ICON_NONE);
292 uiItemR(layout, ptr, "scale", UI_ITEM_NONE, nullptr, ICON_NONE);
293 uiItemR(layout, ptr, "rotation", UI_ITEM_NONE, nullptr, ICON_NONE);
294}
295
296static void panel_register(ARegionType *region_type)
297{
300 region_type, "offset", "Transform", nullptr, transform_panel_draw, panel_type);
301}
302
304 /*idname*/ "UVWarp",
305 /*name*/ N_("UVWarp"),
306 /*struct_name*/ "UVWarpModifierData",
307 /*struct_size*/ sizeof(UVWarpModifierData),
308 /*srna*/ &RNA_UVWarpModifier,
312 /*icon*/ ICON_MOD_UVPROJECT, /* TODO: Use correct icon. */
313
314 /*copy_data*/ BKE_modifier_copydata_generic,
315
316 /*deform_verts*/ nullptr,
317 /*deform_matrices*/ nullptr,
318 /*deform_verts_EM*/ nullptr,
319 /*deform_matrices_EM*/ nullptr,
320 /*modify_mesh*/ modify_mesh,
321 /*modify_geometry_set*/ nullptr,
322
323 /*init_data*/ init_data,
324 /*required_data_mask*/ required_data_mask,
325 /*free_data*/ nullptr,
326 /*is_disabled*/ nullptr,
327 /*update_depsgraph*/ update_depsgraph,
328 /*depends_on_time*/ nullptr,
329 /*depends_on_normals*/ nullptr,
330 /*foreach_ID_link*/ foreach_ID_link,
331 /*foreach_tex_link*/ nullptr,
332 /*free_runtime_data*/ nullptr,
333 /*panel_register*/ panel_register,
334 /*blend_write*/ nullptr,
335 /*blend_read*/ nullptr,
336 /*foreach_cache*/ nullptr,
337};
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_get_layer_named_for_write(CustomData *data, eCustomDataType type, blender::StringRef name, int totelem)
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:770
@ IDWALK_CB_NOP
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
@ eModifierTypeFlag_EnableInEditmode
@ eModifierTypeFlag_SupportsEditmode
@ eModifierTypeFlag_AcceptsMesh
void(*)(void *user_data, Object *ob, ID **idpoin, int cb_flag) IDWalkFunc
#define BLI_assert(a)
Definition BLI_assert.h:50
void mul_m4_m4m4(float R[4][4], const float A[4][4], const float B[4][4])
void unit_m4(float m[4][4])
Definition rct.c:1127
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 interp_v2_v2v2(float r[2], const float a[2], const float b[2], float t)
Definition math_vector.c:21
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:230
#define ELEM(...)
#define MEMCMP_STRUCT_AFTER_IS_ZERO(struct_var, member)
#define MEMCPY_STRUCT_AFTER(struct_dst, struct_src, member)
#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
struct UVWarpModifierData UVWarpModifierData
@ MOD_UVWARP_INVERT_VGROUP
Object is a sort of wrapper for general info.
@ OB_ARMATURE
PanelType * modifier_subpanel_register(ARegionType *region_type, const char *name, const char *label, PanelDrawFn draw_header, PanelDrawFn draw, PanelType *parent)
void modifier_panel_end(uiLayout *layout, PointerRNA *ptr)
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_vgroup_ui(uiLayout *layout, PointerRNA *ptr, PointerRNA *ob_ptr, const char *vgroup_prop, const char *invert_vgroup_prop, const char *text)
void MOD_depsgraph_update_object_bone_relation(DepsNodeHandle *node, Object *object, const char *bonename, const char *description)
Definition MOD_util.cc:189
void MOD_get_vgroup(const Object *ob, const Mesh *mesh, const char *name, const MDeformVert **dvert, int *defgrp_index)
Definition MOD_util.cc:159
static void init_data(ModifierData *md)
Definition MOD_uvwarp.cc:52
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:71
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:94
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:41
static void panel_draw(const bContext *, Panel *panel)
static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
Definition MOD_uvwarp.cc:61
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)
void uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
#define UI_ITEM_NONE
void uiItemPointerR(uiLayout *layout, PointerRNA *ptr, const char *propname, PointerRNA *searchptr, const char *searchpropname, const char *name, int icon)
uiLayout * uiLayoutColumn(uiLayout *layout, bool align)
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
ATTR_WARN_UNUSED_RESULT const BMLoop * l
constexpr Span slice(int64_t start, int64_t size) const
Definition BLI_span.hh:138
constexpr int64_t size() const
Definition BLI_span.hh:253
draw_view in_light_buf[] float
uint col
static char faces[256]
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:413
struct bPose * pose
struct uiLayout * layout
float(* mloopuv)[2]
Definition MOD_uvwarp.cc:85
blender::Span< int > corner_verts
Definition MOD_uvwarp.cc:84
blender::OffsetIndices< int > faces
Definition MOD_uvwarp.cc:83
bool invert_vgroup
Definition MOD_uvwarp.cc:91
int defgrp_index
Definition MOD_uvwarp.cc:88
float(* warp_mat)[4]
Definition MOD_uvwarp.cc:90
const MDeformVert * dvert
Definition MOD_uvwarp.cc:87
struct Object * object_dst
struct Object * object_src
float pose_mat[4][4]
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4126