Blender V5.0
editmesh_add_gizmo.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
10
11#include "MEM_guardedalloc.h"
12
13#include "DNA_object_types.h"
14#include "DNA_scene_types.h"
15
16#include "BKE_context.hh"
17#include "BKE_editmesh.hh"
18#include "BKE_object_types.hh"
19#include "BKE_scene.hh"
20
21#include "BLI_math_geom.h"
22#include "BLI_math_matrix.h"
23#include "BLI_math_vector.h"
24
25#include "ED_gizmo_library.hh"
26#include "ED_gizmo_utils.hh"
27#include "ED_mesh.hh"
28#include "ED_object.hh"
29#include "ED_screen.hh"
30#include "ED_undo.hh"
31#include "ED_view3d.hh"
32
33#include "RNA_access.hh"
34#include "RNA_define.hh"
35
36#include "WM_api.hh"
37#include "WM_types.hh"
38
39#include "UI_resources.hh"
40
41#include "mesh_intern.hh" /* own include */
42
43/* -------------------------------------------------------------------- */
46
55 const float mval[2],
56 float r_location[3],
57 float r_rotation[3][3])
58{
59
60 Scene *scene = CTX_data_scene(C);
61 ARegion *region = CTX_wm_region(C);
62 RegionView3D *rv3d = static_cast<RegionView3D *>(region->regiondata);
63
64 bool use_mouse_project = true; /* TODO: make optional */
65
66 const blender::float4x4 cursor_matrix = scene->cursor.matrix<blender::float4x4>();
67 float orient_matrix[3][3];
68
69 const float dots[3] = {
70 dot_v3v3(rv3d->viewinv[2], cursor_matrix[0]),
71 dot_v3v3(rv3d->viewinv[2], cursor_matrix[1]),
72 dot_v3v3(rv3d->viewinv[2], cursor_matrix[2]),
73 };
74 const int axis = axis_dominant_v3_single(dots);
75
76 copy_v3_v3(orient_matrix[0], cursor_matrix[(axis + 1) % 3]);
77 copy_v3_v3(orient_matrix[1], cursor_matrix[(axis + 2) % 3]);
78 copy_v3_v3(orient_matrix[2], cursor_matrix[axis]);
79
80 if (dot_v3v3(rv3d->viewinv[2], orient_matrix[2]) < 0.0f) {
81 negate_v3(orient_matrix[2]);
82 }
83 if (is_negative_m3(orient_matrix)) {
84 swap_v3_v3(orient_matrix[0], orient_matrix[1]);
85 }
86
87 if (use_mouse_project) {
88 float plane[4];
89 plane_from_point_normal_v3(plane, cursor_matrix[3], orient_matrix[2]);
90 if (ED_view3d_win_to_3d_on_plane(region, plane, mval, true, r_location)) {
91 copy_m3_m3(r_rotation, orient_matrix);
92 return;
93 }
94 }
95
96 /* fallback */
97 copy_v3_v3(r_location, cursor_matrix[3]);
98 copy_m3_m3(r_rotation, orient_matrix);
99}
100
102
103/* -------------------------------------------------------------------- */
106
115
122{
123 wmOperator *op = ggd->data.op;
124 if (op == WM_operator_last_redo(ggd->data.context)) {
126 }
127}
128
130{
131 wmOperator *op = ggd->data.op;
132 UNUSED_VARS(op);
133/* For now don't read back from the operator. */
134#if 0
136#endif
137}
138
139/* translate callbacks */
141 wmGizmoProperty *gz_prop,
142 void *value_p)
143{
145 wmOperator *op = ggd->data.op;
146 float *value = static_cast<float *>(value_p);
147 BLI_assert(gz_prop->type->array_length == 16);
148 UNUSED_VARS_NDEBUG(gz_prop);
149
150 if (value_p != ggd->cage->matrix_offset) {
152 static_cast<float (*)[4]>(value_p), ggd->cage->matrix_basis, ggd->cage->matrix_offset);
154 }
155}
156
158 wmGizmoProperty *gz_prop,
159 const void *value)
160{
162 wmOperator *op = ggd->data.op;
163
164 BLI_assert(gz_prop->type->array_length == 16);
165 UNUSED_VARS_NDEBUG(gz_prop);
166
167 float mat[4][4];
168 mul_m4_m4m4(mat, ggd->cage->matrix_basis, static_cast<const float (*)[4]>(value));
169
170 if (is_negative_m4(mat)) {
171 negate_mat3_m4(mat);
172 }
173
174 RNA_property_float_set_array(op->ptr, ggd->data.prop_matrix, &mat[0][0]);
175
177}
178
180{
182 C, gzgt, "MESH_OT_primitive_cube_add_gizmo");
183}
184
186{
187 GizmoPlacementGroup *ggd = static_cast<GizmoPlacementGroup *>(gzgroup->customdata);
188
189 /* Initial size. */
190 {
191 wmGizmo *gz = ggd->cage;
193
194 /* TODO: support zero scaled matrix in 'GIZMO_GT_cage_3d'. */
195 gz->matrix_offset[0][0] = 0.01;
196 gz->matrix_offset[1][1] = 0.01;
197 gz->matrix_offset[2][2] = 0.01;
198 gz->matrix_offset[3][3] = 1.0f;
199 }
200
201 /* Start off dragging. */
202 {
203 wmWindow *win = CTX_wm_window(C);
204 ARegion *region = CTX_wm_region(C);
205 wmGizmo *gz = ggd->cage;
206
207 {
208 float mat3[3][3];
209 float location[3];
210 float mval[2] = {
211 float(win->eventstate->xy[0] - region->winrct.xmin),
212 float(win->eventstate->xy[1] - region->winrct.ymin),
213 };
214 calc_initial_placement_point_from_view((bContext *)C, mval, location, mat3);
215 copy_m4_m3(gz->matrix_basis, mat3);
216 copy_v3_v3(gz->matrix_basis[3], location);
217 }
218
219 if (true) {
220 wmGizmoMap *gzmap = gzgroup->parent_gzmap;
222 (bContext *)C,
223 ggd->cage,
225 win->eventstate);
226 }
227 }
228}
229
231{
233
234 if (op == nullptr || !STREQ(op->type->idname, "MESH_OT_primitive_cube_add_gizmo")) {
235 return;
236 }
237
239 gzgroup->customdata = ggd;
240
241 const wmGizmoType *gzt_cage = WM_gizmotype_find("GIZMO_GT_cage_3d", true);
242
243 ggd->cage = WM_gizmo_new_ptr(gzt_cage, gzgroup, nullptr);
244
246
247 RNA_enum_set(ggd->cage->ptr,
248 "transform",
251
253
254 ggd->data.context = (bContext *)C;
255 ggd->data.op = op;
256 ggd->data.prop_matrix = RNA_struct_find_property(op->ptr, "matrix");
257
259
260 /* Setup property callbacks */
261 {
265 params.range_get_fn = nullptr;
266 params.user_data = nullptr;
268 }
269
271}
272
273static void gizmo_mesh_placement_draw_prepare(const bContext * /*C*/, wmGizmoGroup *gzgroup)
274{
275 GizmoPlacementGroup *ggd = static_cast<GizmoPlacementGroup *>(gzgroup->customdata);
276 if (ggd->data.op->next) {
278 }
280}
281
283{
284 gzgt->name = "Mesh Add Bounds";
285 gzgt->idname = "MESH_GGT_add_bounds";
286
288
291
295}
296
298
299/* -------------------------------------------------------------------- */
306
308{
309 Object *obedit = CTX_data_edit_object(C);
311 float matrix[4][4];
312
313 /* Get the matrix that defines the cube bounds (as set by the gizmo cage). */
314 {
315 PropertyRNA *prop_matrix = RNA_struct_find_property(op->ptr, "matrix");
316 if (RNA_property_is_set(op->ptr, prop_matrix)) {
317 RNA_property_float_get_array(op->ptr, prop_matrix, &matrix[0][0]);
318 invert_m4_m4(obedit->runtime->world_to_object.ptr(), obedit->object_to_world().ptr());
319 mul_m4_m4m4(matrix, obedit->world_to_object().ptr(), matrix);
320 }
321 else {
322 /* For the first update the widget may not set the matrix. */
323 return OPERATOR_FINISHED;
324 }
325 }
326
327 const bool calc_uvs = RNA_boolean_get(op->ptr, "calc_uvs");
328
329 if (calc_uvs) {
330 ED_mesh_uv_ensure(static_cast<Mesh *>(obedit->data), nullptr);
331 }
332
334 op,
335 "verts.out",
336 false,
337 "create_cube matrix=%m4 size=%f calc_uvs=%b",
338 matrix,
339 1.0f,
340 calc_uvs))
341 {
342 return OPERATOR_CANCELLED;
343 }
344
346 /* TODO(@ideasman42): maintain UV sync for newly created data. */
348
350 params.calc_looptris = true;
351 params.calc_normals = false;
352 params.is_destructive = true;
353 EDBM_update(static_cast<Mesh *>(obedit->data), &params);
354
355 return OPERATOR_FINISHED;
356}
357
359 wmOperator *op,
360 const wmEvent * /*event*/)
361{
362 View3D *v3d = CTX_wm_view3d(C);
363
365 if (ret & OPERATOR_FINISHED) {
366 /* Setup gizmos */
367 if (v3d && ((v3d->gizmo_flag & V3D_GIZMO_HIDE) == 0)) {
368 wmGizmoGroupType *gzgt = WM_gizmogrouptype_find("MESH_GGT_add_bounds", false);
370 Main *bmain = CTX_data_main(C);
372 }
373 }
374 }
375
376 return ret;
377}
378
380{
381 /* identifiers */
382 ot->name = "Add Cube";
383 ot->description = "Construct a cube mesh";
384 ot->idname = "MESH_OT_primitive_cube_add_gizmo";
385
386 /* API callbacks. */
390
391 /* flags */
393
396
397 /* hidden props */
399 ot->srna, "matrix", 4, 4, nullptr, 0.0f, 0.0f, "Matrix", "", 0.0f, 0.0f);
401
403}
404
wmWindow * CTX_wm_window(const bContext *C)
Scene * CTX_data_scene(const bContext *C)
Object * CTX_data_edit_object(const bContext *C)
Main * CTX_data_main(const bContext *C)
ARegion * CTX_wm_region(const bContext *C)
View3D * CTX_wm_view3d(const bContext *C)
BMEditMesh * BKE_editmesh_from_object(Object *ob)
Return the BMEditMesh for a given object.
Definition editmesh.cc:61
#define BLI_assert(a)
Definition BLI_assert.h:46
void plane_from_point_normal_v3(float r_plane[4], const float plane_co[3], const float plane_no[3])
Definition math_geom.cc:217
MINLINE int axis_dominant_v3_single(const float vec[3])
bool is_negative_m3(const float mat[3][3])
void zero_m4(float m[4][4])
void mul_m4_m4m4(float R[4][4], const float A[4][4], const float B[4][4])
void copy_m3_m3(float m1[3][3], const float m2[3][3])
void copy_m4_m3(float m1[4][4], const float m2[3][3])
bool is_negative_m4(const float mat[4][4])
void negate_mat3_m4(float R[4][4])
bool invert_m4_m4(float inverse[4][4], const float mat[4][4])
MINLINE void copy_v3_v3(float r[3], const float a[3])
MINLINE float dot_v3v3(const float a[3], const float b[3]) ATTR_WARN_UNUSED_RESULT
MINLINE void negate_v3(float r[3])
MINLINE void swap_v3_v3(float a[3], float b[3])
#define UNUSED_VARS(...)
#define UNUSED_VARS_NDEBUG(...)
#define STREQ(a, b)
Object is a sort of wrapper for general info.
@ SCE_SELECT_VERTEX
@ RGN_TYPE_WINDOW
@ SPACE_VIEW3D
@ V3D_GIZMO_HIDE
@ OPERATOR_CANCELLED
@ OPERATOR_FINISHED
@ ED_GIZMO_CAGE3D_PART_SCALE_MAX_X_MAX_Y_MAX_Z
@ ED_GIZMO_CAGE_XFORM_FLAG_SCALE
@ ED_GIZMO_CAGE_XFORM_FLAG_TRANSLATE
@ ED_GIZMO_CAGE_XFORM_FLAG_SCALE_SIGNED
bool ED_gizmo_poll_or_unlink_delayed_from_operator(const bContext *C, wmGizmoGroupType *gzgt, const char *idname)
void ED_mesh_uv_ensure(Mesh *mesh, const char *name)
Definition mesh_data.cc:322
void EDBM_update(Mesh *mesh, const EDBMUpdate_Params *params)
bool EDBM_uvselect_clear(BMEditMesh *em)
void EDBM_selectmode_flush_ex(BMEditMesh *em, short selectmode)
bool ED_operator_editmesh_view3d(bContext *C)
bool ED_undo_operator_repeat(bContext *C, wmOperator *op)
Definition ed_undo.cc:631
bool ED_view3d_win_to_3d_on_plane(const ARegion *region, const float plane[4], const float mval[2], bool do_clip, float r_out[3])
Read Guarded memory(de)allocation.
@ PROP_SKIP_SAVE
Definition RNA_types.hh:344
@ PROP_HIDDEN
Definition RNA_types.hh:338
#define C
Definition RandGen.cpp:29
void UI_GetThemeColor3fv(int colorid, float col[3])
@ TH_GIZMO_PRIMARY
@ WM_GIZMO_DRAW_VALUE
@ WM_GIZMOGROUPTYPE_3D
@ OPTYPE_UNDO
Definition WM_types.hh:182
@ OPTYPE_REGISTER
Definition WM_types.hh:180
BMesh const char void * data
nullptr float
static wmOperatorStatus add_primitive_cube_gizmo_invoke(bContext *C, wmOperator *op, const wmEvent *)
static void gizmo_placement_prop_matrix_set(const wmGizmo *gz, wmGizmoProperty *gz_prop, const void *value)
void MESH_OT_primitive_cube_add_gizmo(wmOperatorType *ot)
static void MESH_GGT_add_bounds(wmGizmoGroupType *gzgt)
static void gizmo_mesh_placement_update_from_op(GizmoPlacementGroup *ggd)
static wmOperatorStatus add_primitive_cube_gizmo_exec(bContext *C, wmOperator *op)
static void calc_initial_placement_point_from_view(bContext *C, const float mval[2], float r_location[3], float r_rotation[3][3])
static void gizmo_placement_exec(GizmoPlacementGroup *ggd)
static void gizmo_mesh_placement_modal_from_setup(const bContext *C, wmGizmoGroup *gzgroup)
static void gizmo_placement_prop_matrix_get(const wmGizmo *gz, wmGizmoProperty *gz_prop, void *value_p)
static bool gizmo_mesh_placement_poll(const bContext *C, wmGizmoGroupType *gzgt)
static void gizmo_mesh_placement_setup(const bContext *C, wmGizmoGroup *gzgroup)
static void gizmo_mesh_placement_draw_prepare(const bContext *, wmGizmoGroup *gzgroup)
bool EDBM_op_call_and_selectf(BMEditMesh *em, wmOperator *op, const char *select_slot_out, const bool select_extend, const char *fmt,...)
uiWidgetBaseParameters params[MAX_WIDGET_BASE_BATCH]
void * MEM_callocN(size_t len, const char *str)
Definition mallocn.cc:118
void add_generic_props(wmOperatorType *ot, bool do_editmode)
void add_mesh_props(wmOperatorType *ot)
MatBase< float, 4, 4 > float4x4
return ret
void RNA_property_float_get_array(PointerRNA *ptr, PropertyRNA *prop, float *values)
PropertyRNA * RNA_struct_find_property(PointerRNA *ptr, const char *identifier)
bool RNA_property_is_set(PointerRNA *ptr, PropertyRNA *prop)
void RNA_property_float_set_array(PointerRNA *ptr, PropertyRNA *prop, const float *values)
bool RNA_boolean_get(PointerRNA *ptr, const char *name)
void RNA_enum_set(PointerRNA *ptr, const char *name, int value)
PropertyRNA * RNA_def_float_matrix(StructOrFunctionRNA *cont_, const char *identifier, const int rows, const int columns, const float *default_value, const float hardmin, const float hardmax, const char *ui_name, const char *ui_description, const float softmin, const float softmax)
void RNA_def_property_flag(PropertyRNA *prop, PropertyFlag flag)
void * regiondata
struct GizmoPlacementGroup::@337135315033253013343055130044053026352067176050 data
ObjectRuntimeHandle * runtime
float viewinv[4][4]
View3DCursor cursor
int ymin
int xmin
int xy[2]
Definition WM_types.hh:761
wmGizmoGroupFnInit setup
const char * idname
wmGizmoMapType_Params gzmap_params
eWM_GizmoFlagGroupTypeFlag flag
wmGizmoGroupFnPoll poll
wmGizmoGroupFnDrawPrepare draw_prepare
wmGizmoMap * parent_gzmap
const wmGizmoPropertyType * type
wmGizmoGroup * parent_gzgroup
float matrix_basis[4][4]
float matrix_offset[4][4]
float color[4]
PointerRNA * ptr
const char * idname
Definition WM_types.hh:1035
struct wmOperator * next
struct wmOperatorType * type
struct PointerRNA * ptr
struct wmEvent * eventstate
wmOperatorType * ot
Definition wm_files.cc:4237
void WM_gizmo_modal_set_from_setup(wmGizmoMap *gzmap, bContext *C, wmGizmo *gz, int part_index, const wmEvent *event)
Definition wm_gizmo.cc:419
wmGizmo * WM_gizmo_new_ptr(const wmGizmoType *gzt, wmGizmoGroup *gzgroup, PointerRNA *properties)
Definition wm_gizmo.cc:85
void WM_gizmo_set_flag(wmGizmo *gz, const int flag, const bool enable)
Definition wm_gizmo.cc:307
bool WM_gizmo_group_type_ensure_ptr(wmGizmoGroupType *gzgt)
void WM_gizmo_group_type_reinit_ptr(Main *bmain, wmGizmoGroupType *gzgt)
wmGizmoGroupType * WM_gizmogrouptype_append(void(*wtfunc)(wmGizmoGroupType *))
wmGizmoGroupType * WM_gizmogrouptype_find(const StringRef idname, bool quiet)
void WM_gizmo_target_property_def_func(wmGizmo *gz, const char *idname, const wmGizmoPropertyFnParams *params)
const wmGizmoType * WM_gizmotype_find(const StringRef idname, bool quiet)
wmOperator * WM_operator_last_redo(const bContext *C)