1187 lines
40 KiB
C++
1187 lines
40 KiB
C++
/* SPDX-FileCopyrightText: 2023 Blender Authors
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
#include "BLI_kdtree.h"
|
|
#include "BLI_rand.hh"
|
|
#include "BLI_task.hh"
|
|
#include "BLI_utildefines.h"
|
|
#include "BLI_vector_set.hh"
|
|
|
|
#include "BKE_brush.hh"
|
|
#include "BKE_bvhutils.hh"
|
|
#include "BKE_context.hh"
|
|
#include "BKE_curves.hh"
|
|
#include "BKE_modifier.hh"
|
|
#include "BKE_object.hh"
|
|
#include "BKE_paint.hh"
|
|
|
|
#include "WM_api.hh"
|
|
#include "WM_message.hh"
|
|
#include "WM_toolsystem.h"
|
|
|
|
#include "ED_curves.hh"
|
|
#include "ED_curves_sculpt.hh"
|
|
#include "ED_image.hh"
|
|
#include "ED_object.hh"
|
|
#include "ED_screen.hh"
|
|
#include "ED_space_api.hh"
|
|
#include "ED_view3d.hh"
|
|
|
|
#include "DEG_depsgraph.hh"
|
|
#include "DEG_depsgraph_query.hh"
|
|
|
|
#include "DNA_brush_types.h"
|
|
#include "DNA_curves_types.h"
|
|
#include "DNA_screen_types.h"
|
|
|
|
#include "RNA_access.hh"
|
|
#include "RNA_define.hh"
|
|
#include "RNA_enum_types.hh"
|
|
|
|
#include "curves_sculpt_intern.hh"
|
|
#include "paint_intern.hh"
|
|
|
|
#include "UI_interface.hh"
|
|
#include "UI_resources.hh"
|
|
|
|
#include "GPU_immediate.h"
|
|
#include "GPU_immediate_util.h"
|
|
#include "GPU_matrix.h"
|
|
#include "GPU_state.h"
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Poll Functions
|
|
* \{ */
|
|
|
|
bool CURVES_SCULPT_mode_poll(bContext *C)
|
|
{
|
|
const Object *ob = CTX_data_active_object(C);
|
|
return ob && ob->mode & OB_MODE_SCULPT_CURVES;
|
|
}
|
|
|
|
bool CURVES_SCULPT_mode_poll_view3d(bContext *C)
|
|
{
|
|
if (!CURVES_SCULPT_mode_poll(C)) {
|
|
return false;
|
|
}
|
|
if (CTX_wm_region_view3d(C) == nullptr) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
namespace blender::ed::sculpt_paint {
|
|
|
|
using blender::bke::CurvesGeometry;
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Brush Stroke Operator
|
|
* \{ */
|
|
|
|
float brush_radius_factor(const Brush &brush, const StrokeExtension &stroke_extension)
|
|
{
|
|
if (BKE_brush_use_size_pressure(&brush)) {
|
|
return stroke_extension.pressure;
|
|
}
|
|
return 1.0f;
|
|
}
|
|
|
|
float brush_radius_get(const Scene &scene,
|
|
const Brush &brush,
|
|
const StrokeExtension &stroke_extension)
|
|
{
|
|
return BKE_brush_size_get(&scene, &brush) * brush_radius_factor(brush, stroke_extension);
|
|
}
|
|
|
|
float brush_strength_factor(const Brush &brush, const StrokeExtension &stroke_extension)
|
|
{
|
|
if (BKE_brush_use_alpha_pressure(&brush)) {
|
|
return stroke_extension.pressure;
|
|
}
|
|
return 1.0f;
|
|
}
|
|
|
|
float brush_strength_get(const Scene &scene,
|
|
const Brush &brush,
|
|
const StrokeExtension &stroke_extension)
|
|
{
|
|
return BKE_brush_alpha_get(&scene, &brush) * brush_strength_factor(brush, stroke_extension);
|
|
}
|
|
|
|
static std::unique_ptr<CurvesSculptStrokeOperation> start_brush_operation(
|
|
bContext &C, wmOperator &op, const StrokeExtension &stroke_start)
|
|
{
|
|
const BrushStrokeMode mode = static_cast<BrushStrokeMode>(RNA_enum_get(op.ptr, "mode"));
|
|
|
|
const Scene &scene = *CTX_data_scene(&C);
|
|
const CurvesSculpt &curves_sculpt = *scene.toolsettings->curves_sculpt;
|
|
const Brush &brush = *BKE_paint_brush_for_read(&curves_sculpt.paint);
|
|
switch (brush.curves_sculpt_tool) {
|
|
case CURVES_SCULPT_TOOL_COMB:
|
|
return new_comb_operation();
|
|
case CURVES_SCULPT_TOOL_DELETE:
|
|
return new_delete_operation();
|
|
case CURVES_SCULPT_TOOL_SNAKE_HOOK:
|
|
return new_snake_hook_operation();
|
|
case CURVES_SCULPT_TOOL_ADD:
|
|
return new_add_operation();
|
|
case CURVES_SCULPT_TOOL_GROW_SHRINK:
|
|
return new_grow_shrink_operation(mode, C);
|
|
case CURVES_SCULPT_TOOL_SELECTION_PAINT:
|
|
return new_selection_paint_operation(mode, C);
|
|
case CURVES_SCULPT_TOOL_PINCH:
|
|
return new_pinch_operation(mode, C);
|
|
case CURVES_SCULPT_TOOL_SMOOTH:
|
|
return new_smooth_operation();
|
|
case CURVES_SCULPT_TOOL_PUFF:
|
|
return new_puff_operation();
|
|
case CURVES_SCULPT_TOOL_DENSITY:
|
|
return new_density_operation(mode, C, stroke_start);
|
|
case CURVES_SCULPT_TOOL_SLIDE:
|
|
return new_slide_operation();
|
|
}
|
|
BLI_assert_unreachable();
|
|
return {};
|
|
}
|
|
|
|
struct SculptCurvesBrushStrokeData {
|
|
std::unique_ptr<CurvesSculptStrokeOperation> operation;
|
|
PaintStroke *stroke;
|
|
};
|
|
|
|
static bool stroke_get_location(bContext *C,
|
|
float out[3],
|
|
const float mouse[2],
|
|
bool /*force_original*/)
|
|
{
|
|
out[0] = mouse[0];
|
|
out[1] = mouse[1];
|
|
out[2] = 0;
|
|
UNUSED_VARS(C);
|
|
return true;
|
|
}
|
|
|
|
static bool stroke_test_start(bContext *C, wmOperator *op, const float mouse[2])
|
|
{
|
|
UNUSED_VARS(C, op, mouse);
|
|
return true;
|
|
}
|
|
|
|
static void stroke_update_step(bContext *C,
|
|
wmOperator *op,
|
|
PaintStroke * /*stroke*/,
|
|
PointerRNA *stroke_element)
|
|
{
|
|
SculptCurvesBrushStrokeData *op_data = static_cast<SculptCurvesBrushStrokeData *>(
|
|
op->customdata);
|
|
|
|
StrokeExtension stroke_extension;
|
|
RNA_float_get_array(stroke_element, "mouse", stroke_extension.mouse_position);
|
|
stroke_extension.pressure = RNA_float_get(stroke_element, "pressure");
|
|
stroke_extension.reports = op->reports;
|
|
|
|
if (!op_data->operation) {
|
|
stroke_extension.is_first = true;
|
|
op_data->operation = start_brush_operation(*C, *op, stroke_extension);
|
|
}
|
|
else {
|
|
stroke_extension.is_first = false;
|
|
}
|
|
|
|
if (op_data->operation) {
|
|
op_data->operation->on_stroke_extended(*C, stroke_extension);
|
|
}
|
|
}
|
|
|
|
static void stroke_done(const bContext *C, PaintStroke *stroke)
|
|
{
|
|
UNUSED_VARS(C, stroke);
|
|
}
|
|
|
|
static int sculpt_curves_stroke_invoke(bContext *C, wmOperator *op, const wmEvent *event)
|
|
{
|
|
Scene *scene = CTX_data_scene(C);
|
|
Paint *paint = BKE_paint_get_active_from_paintmode(scene, PAINT_MODE_SCULPT_CURVES);
|
|
const Brush *brush = paint ? BKE_paint_brush_for_read(paint) : nullptr;
|
|
if (brush == nullptr) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
SculptCurvesBrushStrokeData *op_data = MEM_new<SculptCurvesBrushStrokeData>(__func__);
|
|
op_data->stroke = paint_stroke_new(C,
|
|
op,
|
|
stroke_get_location,
|
|
stroke_test_start,
|
|
stroke_update_step,
|
|
nullptr,
|
|
stroke_done,
|
|
event->type);
|
|
op->customdata = op_data;
|
|
|
|
int return_value = op->type->modal(C, op, event);
|
|
if (return_value == OPERATOR_FINISHED) {
|
|
if (op->customdata != nullptr) {
|
|
paint_stroke_free(C, op, op_data->stroke);
|
|
MEM_delete(op_data);
|
|
}
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
WM_event_add_modal_handler(C, op);
|
|
return OPERATOR_RUNNING_MODAL;
|
|
}
|
|
|
|
static int sculpt_curves_stroke_modal(bContext *C, wmOperator *op, const wmEvent *event)
|
|
{
|
|
SculptCurvesBrushStrokeData *op_data = static_cast<SculptCurvesBrushStrokeData *>(
|
|
op->customdata);
|
|
int return_value = paint_stroke_modal(C, op, event, &op_data->stroke);
|
|
if (ELEM(return_value, OPERATOR_FINISHED, OPERATOR_CANCELLED)) {
|
|
MEM_delete(op_data);
|
|
op->customdata = nullptr;
|
|
}
|
|
return return_value;
|
|
}
|
|
|
|
static void sculpt_curves_stroke_cancel(bContext *C, wmOperator *op)
|
|
{
|
|
if (op->customdata != nullptr) {
|
|
SculptCurvesBrushStrokeData *op_data = static_cast<SculptCurvesBrushStrokeData *>(
|
|
op->customdata);
|
|
paint_stroke_cancel(C, op, op_data->stroke);
|
|
MEM_delete(op_data);
|
|
}
|
|
}
|
|
|
|
static void SCULPT_CURVES_OT_brush_stroke(wmOperatorType *ot)
|
|
{
|
|
ot->name = "Stroke Curves Sculpt";
|
|
ot->idname = "SCULPT_CURVES_OT_brush_stroke";
|
|
ot->description = "Sculpt curves using a brush";
|
|
|
|
ot->invoke = sculpt_curves_stroke_invoke;
|
|
ot->modal = sculpt_curves_stroke_modal;
|
|
ot->cancel = sculpt_curves_stroke_cancel;
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
paint_stroke_operator_properties(ot);
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Toggle Sculpt Mode
|
|
* \{ */
|
|
|
|
static void curves_sculptmode_enter(bContext *C)
|
|
{
|
|
Scene *scene = CTX_data_scene(C);
|
|
wmMsgBus *mbus = CTX_wm_message_bus(C);
|
|
|
|
Object *ob = CTX_data_active_object(C);
|
|
BKE_paint_ensure(scene->toolsettings, (Paint **)&scene->toolsettings->curves_sculpt);
|
|
CurvesSculpt *curves_sculpt = scene->toolsettings->curves_sculpt;
|
|
|
|
ob->mode = OB_MODE_SCULPT_CURVES;
|
|
|
|
/* Setup cursor color. BKE_paint_init() could be used, but creates an additional brush. */
|
|
Paint *paint = BKE_paint_get_active_from_paintmode(scene, PAINT_MODE_SCULPT_CURVES);
|
|
copy_v3_v3_uchar(paint->paint_cursor_col, PAINT_CURSOR_SCULPT_CURVES);
|
|
paint->paint_cursor_col[3] = 128;
|
|
|
|
ED_paint_cursor_start(&curves_sculpt->paint, CURVES_SCULPT_mode_poll_view3d);
|
|
paint_init_pivot(ob, scene);
|
|
|
|
/* Necessary to change the object mode on the evaluated object. */
|
|
DEG_id_tag_update(&ob->id, ID_RECALC_COPY_ON_WRITE);
|
|
WM_msg_publish_rna_prop(mbus, &ob->id, ob, Object, mode);
|
|
WM_event_add_notifier(C, NC_SCENE | ND_MODE, nullptr);
|
|
}
|
|
|
|
static void curves_sculptmode_exit(bContext *C)
|
|
{
|
|
Object *ob = CTX_data_active_object(C);
|
|
ob->mode = OB_MODE_OBJECT;
|
|
}
|
|
|
|
static int curves_sculptmode_toggle_exec(bContext *C, wmOperator *op)
|
|
{
|
|
Object *ob = CTX_data_active_object(C);
|
|
wmMsgBus *mbus = CTX_wm_message_bus(C);
|
|
|
|
const bool is_mode_set = ob->mode == OB_MODE_SCULPT_CURVES;
|
|
|
|
if (is_mode_set) {
|
|
if (!ED_object_mode_compat_set(C, ob, OB_MODE_SCULPT_CURVES, op->reports)) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
|
|
if (is_mode_set) {
|
|
curves_sculptmode_exit(C);
|
|
}
|
|
else {
|
|
curves_sculptmode_enter(C);
|
|
}
|
|
|
|
WM_toolsystem_update_from_context_view3d(C);
|
|
|
|
/* Necessary to change the object mode on the evaluated object. */
|
|
DEG_id_tag_update(&ob->id, ID_RECALC_COPY_ON_WRITE);
|
|
WM_msg_publish_rna_prop(mbus, &ob->id, ob, Object, mode);
|
|
WM_event_add_notifier(C, NC_SCENE | ND_MODE, nullptr);
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
static void CURVES_OT_sculptmode_toggle(wmOperatorType *ot)
|
|
{
|
|
ot->name = "Curve Sculpt Mode Toggle";
|
|
ot->idname = "CURVES_OT_sculptmode_toggle";
|
|
ot->description = "Enter/Exit sculpt mode for curves";
|
|
|
|
ot->exec = curves_sculptmode_toggle_exec;
|
|
ot->poll = curves::curves_poll;
|
|
|
|
ot->flag = OPTYPE_UNDO | OPTYPE_REGISTER;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
namespace select_random {
|
|
|
|
static int select_random_exec(bContext *C, wmOperator *op)
|
|
{
|
|
VectorSet<Curves *> unique_curves = curves::get_unique_editable_curves(*C);
|
|
|
|
const int seed = RNA_int_get(op->ptr, "seed");
|
|
RandomNumberGenerator rng{uint32_t(seed)};
|
|
|
|
const bool partial = RNA_boolean_get(op->ptr, "partial");
|
|
const bool constant_per_curve = RNA_boolean_get(op->ptr, "constant_per_curve");
|
|
const float probability = RNA_float_get(op->ptr, "probability");
|
|
const float min_value = RNA_float_get(op->ptr, "min");
|
|
const auto next_partial_random_value = [&]() {
|
|
return rng.get_float() * (1.0f - min_value) + min_value;
|
|
};
|
|
const auto next_bool_random_value = [&]() { return rng.get_float() <= probability; };
|
|
|
|
for (Curves *curves_id : unique_curves) {
|
|
CurvesGeometry &curves = curves_id->geometry.wrap();
|
|
const bool was_anything_selected = curves::has_anything_selected(curves);
|
|
|
|
bke::SpanAttributeWriter<float> attribute = float_selection_ensure(*curves_id);
|
|
MutableSpan<float> selection = attribute.span;
|
|
if (!was_anything_selected) {
|
|
selection.fill(1.0f);
|
|
}
|
|
const OffsetIndices points_by_curve = curves.points_by_curve();
|
|
switch (curves_id->selection_domain) {
|
|
case ATTR_DOMAIN_POINT: {
|
|
if (partial) {
|
|
if (constant_per_curve) {
|
|
for (const int curve_i : curves.curves_range()) {
|
|
const float random_value = next_partial_random_value();
|
|
const IndexRange points = points_by_curve[curve_i];
|
|
for (const int point_i : points) {
|
|
selection[point_i] *= random_value;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
for (const int point_i : selection.index_range()) {
|
|
const float random_value = next_partial_random_value();
|
|
selection[point_i] *= random_value;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (constant_per_curve) {
|
|
for (const int curve_i : curves.curves_range()) {
|
|
const bool random_value = next_bool_random_value();
|
|
const IndexRange points = points_by_curve[curve_i];
|
|
if (!random_value) {
|
|
selection.slice(points).fill(0.0f);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
for (const int point_i : selection.index_range()) {
|
|
const bool random_value = next_bool_random_value();
|
|
if (!random_value) {
|
|
selection[point_i] = 0.0f;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case ATTR_DOMAIN_CURVE: {
|
|
if (partial) {
|
|
for (const int curve_i : curves.curves_range()) {
|
|
const float random_value = next_partial_random_value();
|
|
selection[curve_i] *= random_value;
|
|
}
|
|
}
|
|
else {
|
|
for (const int curve_i : curves.curves_range()) {
|
|
const bool random_value = next_bool_random_value();
|
|
if (!random_value) {
|
|
selection[curve_i] = 0.0f;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
const bool was_any_selected = std::any_of(
|
|
selection.begin(), selection.end(), [](const float v) { return v > 0.0f; });
|
|
if (was_any_selected) {
|
|
for (float &v : selection) {
|
|
v *= rng.get_float();
|
|
}
|
|
}
|
|
else {
|
|
for (float &v : selection) {
|
|
v = rng.get_float();
|
|
}
|
|
}
|
|
|
|
attribute.finish();
|
|
|
|
/* Use #ID_RECALC_GEOMETRY instead of #ID_RECALC_SELECT because it is handled as a generic
|
|
* attribute for now. */
|
|
DEG_id_tag_update(&curves_id->id, ID_RECALC_GEOMETRY);
|
|
WM_event_add_notifier(C, NC_GEOM | ND_DATA, curves_id);
|
|
}
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
static void select_random_ui(bContext * /*C*/, wmOperator *op)
|
|
{
|
|
uiLayout *layout = op->layout;
|
|
|
|
uiItemR(layout, op->ptr, "seed", UI_ITEM_NONE, nullptr, ICON_NONE);
|
|
uiItemR(layout, op->ptr, "constant_per_curve", UI_ITEM_NONE, nullptr, ICON_NONE);
|
|
uiItemR(layout, op->ptr, "partial", UI_ITEM_NONE, nullptr, ICON_NONE);
|
|
|
|
if (RNA_boolean_get(op->ptr, "partial")) {
|
|
uiItemR(layout, op->ptr, "min", UI_ITEM_R_SLIDER, "Min", ICON_NONE);
|
|
}
|
|
else {
|
|
uiItemR(layout, op->ptr, "probability", UI_ITEM_R_SLIDER, "Probability", ICON_NONE);
|
|
}
|
|
}
|
|
|
|
} // namespace select_random
|
|
|
|
static void SCULPT_CURVES_OT_select_random(wmOperatorType *ot)
|
|
{
|
|
ot->name = "Select Random";
|
|
ot->idname = __func__;
|
|
ot->description = "Randomizes existing selection or create new random selection";
|
|
|
|
ot->exec = select_random::select_random_exec;
|
|
ot->poll = curves::editable_curves_poll;
|
|
ot->ui = select_random::select_random_ui;
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
RNA_def_int(ot->srna,
|
|
"seed",
|
|
0,
|
|
INT32_MIN,
|
|
INT32_MAX,
|
|
"Seed",
|
|
"Source of randomness",
|
|
INT32_MIN,
|
|
INT32_MAX);
|
|
RNA_def_boolean(
|
|
ot->srna, "partial", false, "Partial", "Allow points or curves to be selected partially");
|
|
RNA_def_float(ot->srna,
|
|
"probability",
|
|
0.5f,
|
|
0.0f,
|
|
1.0f,
|
|
"Probability",
|
|
"Chance of every point or curve being included in the selection",
|
|
0.0f,
|
|
1.0f);
|
|
RNA_def_float(ot->srna,
|
|
"min",
|
|
0.0f,
|
|
0.0f,
|
|
1.0f,
|
|
"Min",
|
|
"Minimum value for the random selection",
|
|
0.0f,
|
|
1.0f);
|
|
RNA_def_boolean(ot->srna,
|
|
"constant_per_curve",
|
|
true,
|
|
"Constant per Curve",
|
|
"The generated random number is the same for every control point of a curve");
|
|
}
|
|
namespace select_grow {
|
|
|
|
struct GrowOperatorDataPerCurve : NonCopyable, NonMovable {
|
|
Curves *curves_id;
|
|
IndexMaskMemory selected_points_memory;
|
|
IndexMaskMemory unselected_points_memory;
|
|
IndexMask selected_points;
|
|
IndexMask unselected_points;
|
|
Array<float> distances_to_selected;
|
|
Array<float> distances_to_unselected;
|
|
|
|
GArray<> original_selection;
|
|
float pixel_to_distance_factor;
|
|
};
|
|
|
|
struct GrowOperatorData {
|
|
int initial_mouse_x;
|
|
Vector<std::unique_ptr<GrowOperatorDataPerCurve>> per_curve;
|
|
};
|
|
|
|
static void update_points_selection(const GrowOperatorDataPerCurve &data,
|
|
const float distance,
|
|
MutableSpan<float> points_selection)
|
|
{
|
|
if (distance > 0.0f) {
|
|
data.unselected_points.foreach_index(
|
|
GrainSize(256), [&](const int point_i, const int index_pos) {
|
|
const float distance_to_selected = data.distances_to_selected[index_pos];
|
|
const float selection = distance_to_selected <= distance ? 1.0f : 0.0f;
|
|
points_selection[point_i] = selection;
|
|
});
|
|
data.selected_points.foreach_index(
|
|
GrainSize(512), [&](const int point_i) { points_selection[point_i] = 1.0f; });
|
|
}
|
|
else {
|
|
data.selected_points.foreach_index(
|
|
GrainSize(256), [&](const int point_i, const int index_pos) {
|
|
const float distance_to_unselected = data.distances_to_unselected[index_pos];
|
|
const float selection = distance_to_unselected <= -distance ? 0.0f : 1.0f;
|
|
points_selection[point_i] = selection;
|
|
});
|
|
data.unselected_points.foreach_index(
|
|
GrainSize(512), [&](const int point_i) { points_selection[point_i] = 0.0f; });
|
|
}
|
|
}
|
|
|
|
static int select_grow_update(bContext *C, wmOperator *op, const float mouse_diff_x)
|
|
{
|
|
GrowOperatorData &op_data = *static_cast<GrowOperatorData *>(op->customdata);
|
|
|
|
for (std::unique_ptr<GrowOperatorDataPerCurve> &curve_op_data : op_data.per_curve) {
|
|
Curves &curves_id = *curve_op_data->curves_id;
|
|
CurvesGeometry &curves = curves_id.geometry.wrap();
|
|
const float distance = curve_op_data->pixel_to_distance_factor * mouse_diff_x;
|
|
|
|
bke::SpanAttributeWriter<float> selection = float_selection_ensure(curves_id);
|
|
const OffsetIndices points_by_curve = curves.points_by_curve();
|
|
|
|
/* Grow or shrink selection based on precomputed distances. */
|
|
switch (selection.domain) {
|
|
case ATTR_DOMAIN_POINT: {
|
|
update_points_selection(*curve_op_data, distance, selection.span);
|
|
break;
|
|
}
|
|
case ATTR_DOMAIN_CURVE: {
|
|
Array<float> new_points_selection(curves.points_num());
|
|
update_points_selection(*curve_op_data, distance, new_points_selection);
|
|
/* Propagate grown point selection to the curve selection. */
|
|
MutableSpan<float> curves_selection = selection.span;
|
|
for (const int curve_i : curves.curves_range()) {
|
|
const IndexRange points = points_by_curve[curve_i];
|
|
const Span<float> points_selection = new_points_selection.as_span().slice(points);
|
|
const float max_selection = *std::max_element(points_selection.begin(),
|
|
points_selection.end());
|
|
curves_selection[curve_i] = max_selection;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
BLI_assert_unreachable();
|
|
}
|
|
|
|
selection.finish();
|
|
|
|
/* Use #ID_RECALC_GEOMETRY instead of #ID_RECALC_SELECT because it is handled as a generic
|
|
* attribute for now. */
|
|
DEG_id_tag_update(&curves_id.id, ID_RECALC_GEOMETRY);
|
|
WM_event_add_notifier(C, NC_GEOM | ND_DATA, &curves_id);
|
|
}
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
static void select_grow_invoke_per_curve(const Curves &curves_id,
|
|
const Object &curves_ob,
|
|
const ARegion ®ion,
|
|
const View3D &v3d,
|
|
const RegionView3D &rv3d,
|
|
GrowOperatorDataPerCurve &curve_op_data)
|
|
{
|
|
const CurvesGeometry &curves = curves_id.geometry.wrap();
|
|
const Span<float3> positions = curves.positions();
|
|
|
|
if (const bke::GAttributeReader original_selection = curves.attributes().lookup(".selection")) {
|
|
curve_op_data.original_selection = GArray<>(original_selection.varray.type(),
|
|
original_selection.varray.size());
|
|
original_selection.varray.materialize(curve_op_data.original_selection.data());
|
|
}
|
|
|
|
/* Find indices of selected and unselected points. */
|
|
curve_op_data.selected_points = curves::retrieve_selected_points(
|
|
curves_id, curve_op_data.selected_points_memory);
|
|
curve_op_data.unselected_points = curve_op_data.selected_points.complement(
|
|
curves.points_range(), curve_op_data.unselected_points_memory);
|
|
|
|
threading::parallel_invoke(
|
|
1024 < curve_op_data.selected_points.size() + curve_op_data.unselected_points.size(),
|
|
[&]() {
|
|
/* Build KD-tree for the selected points. */
|
|
KDTree_3d *kdtree = BLI_kdtree_3d_new(curve_op_data.selected_points.size());
|
|
BLI_SCOPED_DEFER([&]() { BLI_kdtree_3d_free(kdtree); });
|
|
curve_op_data.selected_points.foreach_index([&](const int point_i) {
|
|
const float3 &position = positions[point_i];
|
|
BLI_kdtree_3d_insert(kdtree, point_i, position);
|
|
});
|
|
BLI_kdtree_3d_balance(kdtree);
|
|
|
|
/* For each unselected point, compute the distance to the closest selected point. */
|
|
curve_op_data.distances_to_selected.reinitialize(curve_op_data.unselected_points.size());
|
|
threading::parallel_for(
|
|
curve_op_data.unselected_points.index_range(), 256, [&](const IndexRange range) {
|
|
for (const int i : range) {
|
|
const int point_i = curve_op_data.unselected_points[i];
|
|
const float3 &position = positions[point_i];
|
|
KDTreeNearest_3d nearest;
|
|
BLI_kdtree_3d_find_nearest(kdtree, position, &nearest);
|
|
curve_op_data.distances_to_selected[i] = nearest.dist;
|
|
}
|
|
});
|
|
},
|
|
[&]() {
|
|
/* Build KD-tree for the unselected points. */
|
|
KDTree_3d *kdtree = BLI_kdtree_3d_new(curve_op_data.unselected_points.size());
|
|
BLI_SCOPED_DEFER([&]() { BLI_kdtree_3d_free(kdtree); });
|
|
curve_op_data.unselected_points.foreach_index([&](const int point_i) {
|
|
const float3 &position = positions[point_i];
|
|
BLI_kdtree_3d_insert(kdtree, point_i, position);
|
|
});
|
|
BLI_kdtree_3d_balance(kdtree);
|
|
|
|
/* For each selected point, compute the distance to the closest unselected point. */
|
|
curve_op_data.distances_to_unselected.reinitialize(curve_op_data.selected_points.size());
|
|
threading::parallel_for(
|
|
curve_op_data.selected_points.index_range(), 256, [&](const IndexRange range) {
|
|
for (const int i : range) {
|
|
const int point_i = curve_op_data.selected_points[i];
|
|
const float3 &position = positions[point_i];
|
|
KDTreeNearest_3d nearest;
|
|
BLI_kdtree_3d_find_nearest(kdtree, position, &nearest);
|
|
curve_op_data.distances_to_unselected[i] = nearest.dist;
|
|
}
|
|
});
|
|
});
|
|
|
|
float4x4 curves_to_world_mat = float4x4(curves_ob.object_to_world);
|
|
float4x4 world_to_curves_mat = math::invert(curves_to_world_mat);
|
|
|
|
float4x4 projection;
|
|
ED_view3d_ob_project_mat_get(&rv3d, &curves_ob, projection.ptr());
|
|
|
|
/* Compute how mouse movements in screen space are converted into grow/shrink distances in
|
|
* object space. */
|
|
curve_op_data.pixel_to_distance_factor = threading::parallel_reduce(
|
|
curve_op_data.selected_points.index_range(),
|
|
256,
|
|
FLT_MAX,
|
|
[&](const IndexRange range, float pixel_to_distance_factor) {
|
|
for (const int i : range) {
|
|
const int point_i = curve_op_data.selected_points[i];
|
|
const float3 &pos_cu = positions[point_i];
|
|
|
|
float2 pos_re;
|
|
ED_view3d_project_float_v2_m4(®ion, pos_cu, pos_re, projection.ptr());
|
|
if (pos_re.x < 0 || pos_re.y < 0 || pos_re.x > region.winx || pos_re.y > region.winy) {
|
|
continue;
|
|
}
|
|
/* Compute how far this point moves in curve space when it moves one unit in screen
|
|
* space. */
|
|
const float2 pos_offset_re = pos_re + float2(1, 0);
|
|
float3 pos_offset_wo;
|
|
ED_view3d_win_to_3d(&v3d,
|
|
®ion,
|
|
math::transform_point(curves_to_world_mat, pos_cu),
|
|
pos_offset_re,
|
|
pos_offset_wo);
|
|
const float3 pos_offset_cu = math::transform_point(world_to_curves_mat, pos_offset_wo);
|
|
const float dist_cu = math::distance(pos_cu, pos_offset_cu);
|
|
const float dist_re = math::distance(pos_re, pos_offset_re);
|
|
const float factor = dist_cu / dist_re;
|
|
math::min_inplace(pixel_to_distance_factor, factor);
|
|
}
|
|
return pixel_to_distance_factor;
|
|
},
|
|
[](const float a, const float b) { return std::min(a, b); });
|
|
}
|
|
|
|
static int select_grow_invoke(bContext *C, wmOperator *op, const wmEvent *event)
|
|
{
|
|
Object *active_ob = CTX_data_active_object(C);
|
|
ARegion *region = CTX_wm_region(C);
|
|
View3D *v3d = CTX_wm_view3d(C);
|
|
RegionView3D *rv3d = CTX_wm_region_view3d(C);
|
|
|
|
GrowOperatorData *op_data = MEM_new<GrowOperatorData>(__func__);
|
|
op->customdata = op_data;
|
|
|
|
op_data->initial_mouse_x = event->xy[0];
|
|
|
|
Curves &curves_id = *static_cast<Curves *>(active_ob->data);
|
|
auto curve_op_data = std::make_unique<GrowOperatorDataPerCurve>();
|
|
curve_op_data->curves_id = &curves_id;
|
|
select_grow_invoke_per_curve(curves_id, *active_ob, *region, *v3d, *rv3d, *curve_op_data);
|
|
op_data->per_curve.append(std::move(curve_op_data));
|
|
|
|
WM_event_add_modal_handler(C, op);
|
|
return OPERATOR_RUNNING_MODAL;
|
|
}
|
|
|
|
static int select_grow_modal(bContext *C, wmOperator *op, const wmEvent *event)
|
|
{
|
|
GrowOperatorData &op_data = *static_cast<GrowOperatorData *>(op->customdata);
|
|
const int mouse_x = event->xy[0];
|
|
const int mouse_diff_x = mouse_x - op_data.initial_mouse_x;
|
|
switch (event->type) {
|
|
case MOUSEMOVE: {
|
|
select_grow_update(C, op, mouse_diff_x);
|
|
break;
|
|
}
|
|
case LEFTMOUSE: {
|
|
MEM_delete(&op_data);
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
case EVT_ESCKEY:
|
|
case RIGHTMOUSE: {
|
|
/* Undo operator by resetting the selection to the original value. */
|
|
for (std::unique_ptr<GrowOperatorDataPerCurve> &curve_op_data : op_data.per_curve) {
|
|
Curves &curves_id = *curve_op_data->curves_id;
|
|
CurvesGeometry &curves = curves_id.geometry.wrap();
|
|
bke::MutableAttributeAccessor attributes = curves.attributes_for_write();
|
|
|
|
attributes.remove(".selection");
|
|
if (!curve_op_data->original_selection.is_empty()) {
|
|
attributes.add(
|
|
".selection",
|
|
eAttrDomain(curves_id.selection_domain),
|
|
bke::cpp_type_to_custom_data_type(curve_op_data->original_selection.type()),
|
|
bke::AttributeInitVArray(GVArray::ForSpan(curve_op_data->original_selection)));
|
|
}
|
|
|
|
/* Use #ID_RECALC_GEOMETRY instead of #ID_RECALC_SELECT because it is handled as a generic
|
|
* attribute for now. */
|
|
DEG_id_tag_update(&curves_id.id, ID_RECALC_GEOMETRY);
|
|
WM_event_add_notifier(C, NC_GEOM | ND_DATA, &curves_id);
|
|
}
|
|
MEM_delete(&op_data);
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
return OPERATOR_RUNNING_MODAL;
|
|
}
|
|
|
|
} // namespace select_grow
|
|
|
|
static void SCULPT_CURVES_OT_select_grow(wmOperatorType *ot)
|
|
{
|
|
ot->name = "Select Grow";
|
|
ot->idname = __func__;
|
|
ot->description = "Select curves which are close to curves that are selected already";
|
|
|
|
ot->invoke = select_grow::select_grow_invoke;
|
|
ot->modal = select_grow::select_grow_modal;
|
|
ot->poll = curves::editable_curves_poll;
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
PropertyRNA *prop;
|
|
prop = RNA_def_float(ot->srna,
|
|
"distance",
|
|
0.1f,
|
|
-FLT_MAX,
|
|
FLT_MAX,
|
|
"Distance",
|
|
"By how much to grow the selection",
|
|
-10.0f,
|
|
10.0f);
|
|
RNA_def_property_subtype(prop, PROP_DISTANCE);
|
|
}
|
|
|
|
namespace min_distance_edit {
|
|
|
|
static bool min_distance_edit_poll(bContext *C)
|
|
{
|
|
if (!curves::curves_with_surface_poll(C)) {
|
|
return false;
|
|
}
|
|
Scene *scene = CTX_data_scene(C);
|
|
const Brush *brush = BKE_paint_brush_for_read(&scene->toolsettings->curves_sculpt->paint);
|
|
if (brush == nullptr) {
|
|
return false;
|
|
}
|
|
if (brush->curves_sculpt_tool != CURVES_SCULPT_TOOL_DENSITY) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
struct MinDistanceEditData {
|
|
/** Brush whose minimum distance is modified. */
|
|
Brush *brush;
|
|
float4x4 curves_to_world_mat;
|
|
|
|
/** Where the preview is drawn. */
|
|
float3 pos_cu;
|
|
float3 normal_cu;
|
|
|
|
int2 initial_mouse;
|
|
float initial_minimum_distance;
|
|
|
|
/** The operator uses a new cursor, but the existing cursors should be restored afterwards. */
|
|
ListBase orig_paintcursors;
|
|
void *cursor;
|
|
|
|
/** Store the viewport region in case the operator was called from the header. */
|
|
ARegion *region;
|
|
RegionView3D *rv3d;
|
|
};
|
|
|
|
static int calculate_points_per_side(bContext *C, MinDistanceEditData &op_data)
|
|
{
|
|
Scene *scene = CTX_data_scene(C);
|
|
ARegion *region = op_data.region;
|
|
|
|
const float min_distance = op_data.brush->curves_sculpt_settings->minimum_distance;
|
|
const float brush_radius = BKE_brush_size_get(scene, op_data.brush);
|
|
|
|
float3 tangent_x_cu = math::cross(op_data.normal_cu, float3{0, 0, 1});
|
|
if (math::is_zero(tangent_x_cu)) {
|
|
tangent_x_cu = math::cross(op_data.normal_cu, float3{0, 1, 0});
|
|
}
|
|
tangent_x_cu = math::normalize(tangent_x_cu);
|
|
const float3 tangent_y_cu = math::normalize(math::cross(op_data.normal_cu, tangent_x_cu));
|
|
|
|
/* Sample a few points to get a good estimate of how large the grid has to be. */
|
|
Vector<float3> points_wo;
|
|
points_wo.append(op_data.pos_cu + min_distance * tangent_x_cu);
|
|
points_wo.append(op_data.pos_cu + min_distance * tangent_y_cu);
|
|
points_wo.append(op_data.pos_cu - min_distance * tangent_x_cu);
|
|
points_wo.append(op_data.pos_cu - min_distance * tangent_y_cu);
|
|
|
|
Vector<float2> points_re;
|
|
for (const float3 &pos_wo : points_wo) {
|
|
float2 pos_re;
|
|
ED_view3d_project_v2(region, pos_wo, pos_re);
|
|
points_re.append(pos_re);
|
|
}
|
|
|
|
float2 origin_re;
|
|
ED_view3d_project_v2(region, op_data.pos_cu, origin_re);
|
|
|
|
int needed_points = 0;
|
|
for (const float2 &pos_re : points_re) {
|
|
const float distance = math::length(pos_re - origin_re);
|
|
const int needed_points_iter = (brush_radius * 2.0f) / distance;
|
|
|
|
if (needed_points_iter > needed_points) {
|
|
needed_points = needed_points_iter;
|
|
}
|
|
}
|
|
|
|
/* Limit to a hard-coded number since it only adds noise at some point. */
|
|
return std::min(300, needed_points);
|
|
}
|
|
|
|
static void min_distance_edit_draw(bContext *C, int /*x*/, int /*y*/, void *customdata)
|
|
{
|
|
Scene *scene = CTX_data_scene(C);
|
|
MinDistanceEditData &op_data = *static_cast<MinDistanceEditData *>(customdata);
|
|
|
|
const float min_distance = op_data.brush->curves_sculpt_settings->minimum_distance;
|
|
|
|
float3 tangent_x_cu = math::cross(op_data.normal_cu, float3{0, 0, 1});
|
|
if (math::is_zero(tangent_x_cu)) {
|
|
tangent_x_cu = math::cross(op_data.normal_cu, float3{0, 1, 0});
|
|
}
|
|
tangent_x_cu = math::normalize(tangent_x_cu);
|
|
const float3 tangent_y_cu = math::normalize(math::cross(op_data.normal_cu, tangent_x_cu));
|
|
|
|
const int points_per_side = calculate_points_per_side(C, op_data);
|
|
const int points_per_axis_num = 2 * points_per_side + 1;
|
|
|
|
Vector<float3> points_wo;
|
|
for (const int x_i : IndexRange(points_per_axis_num)) {
|
|
for (const int y_i : IndexRange(points_per_axis_num)) {
|
|
const float x_iter = min_distance * (x_i - (points_per_axis_num - 1) / 2.0f);
|
|
const float y_iter = min_distance * (y_i - (points_per_axis_num - 1) / 2.0f);
|
|
|
|
const float3 point_pos_cu = op_data.pos_cu + op_data.normal_cu * 0.0001f +
|
|
x_iter * tangent_x_cu + y_iter * tangent_y_cu;
|
|
const float3 point_pos_wo = math::transform_point(op_data.curves_to_world_mat, point_pos_cu);
|
|
points_wo.append(point_pos_wo);
|
|
}
|
|
}
|
|
|
|
float4 circle_col = float4(op_data.brush->add_col);
|
|
float circle_alpha = op_data.brush->cursor_overlay_alpha;
|
|
float brush_radius_re = BKE_brush_size_get(scene, op_data.brush);
|
|
|
|
/* Draw the grid. */
|
|
GPU_matrix_push();
|
|
GPU_matrix_push_projection();
|
|
GPU_blend(GPU_BLEND_ALPHA);
|
|
|
|
ARegion *region = op_data.region;
|
|
RegionView3D *rv3d = op_data.rv3d;
|
|
wmWindow *win = CTX_wm_window(C);
|
|
|
|
/* It does the same as: `view3d_operator_needs_opengl(C);`. */
|
|
wmViewport(®ion->winrct);
|
|
GPU_matrix_projection_set(rv3d->winmat);
|
|
GPU_matrix_set(rv3d->viewmat);
|
|
|
|
GPUVertFormat *format3d = immVertexFormat();
|
|
|
|
const uint pos3d = GPU_vertformat_attr_add(format3d, "pos", GPU_COMP_F32, 3, GPU_FETCH_FLOAT);
|
|
const uint col3d = GPU_vertformat_attr_add(format3d, "color", GPU_COMP_F32, 4, GPU_FETCH_FLOAT);
|
|
const uint siz3d = GPU_vertformat_attr_add(format3d, "size", GPU_COMP_F32, 1, GPU_FETCH_FLOAT);
|
|
|
|
immBindBuiltinProgram(GPU_SHADER_3D_POINT_VARYING_SIZE_VARYING_COLOR);
|
|
GPU_program_point_size(true);
|
|
immBegin(GPU_PRIM_POINTS, points_wo.size());
|
|
|
|
float3 brush_origin_wo = math::transform_point(op_data.curves_to_world_mat, op_data.pos_cu);
|
|
float2 brush_origin_re;
|
|
ED_view3d_project_v2(region, brush_origin_wo, brush_origin_re);
|
|
|
|
/* Smooth alpha transition until the brush edge. */
|
|
const int alpha_border_re = 20;
|
|
const float dist_to_inner_border_re = brush_radius_re - alpha_border_re;
|
|
|
|
for (const float3 &pos_wo : points_wo) {
|
|
float2 pos_re;
|
|
ED_view3d_project_v2(region, pos_wo, pos_re);
|
|
|
|
const float dist_to_point_re = math::distance(pos_re, brush_origin_re);
|
|
const float alpha = 1.0f - ((dist_to_point_re - dist_to_inner_border_re) / alpha_border_re);
|
|
|
|
immAttr1f(siz3d, 3.0f);
|
|
immAttr4f(col3d, 0.9f, 0.9f, 0.9f, alpha);
|
|
immVertex3fv(pos3d, pos_wo);
|
|
}
|
|
immEnd();
|
|
immUnbindProgram();
|
|
|
|
/* Reset the drawing settings. */
|
|
GPU_point_size(1.0f);
|
|
GPU_matrix_pop_projection();
|
|
GPU_matrix_pop();
|
|
|
|
int4 scissor;
|
|
GPU_scissor_get(scissor);
|
|
wmWindowViewport(win);
|
|
GPU_scissor(scissor[0], scissor[1], scissor[2], scissor[3]);
|
|
|
|
/* Draw the brush circle. */
|
|
GPU_matrix_translate_2f(float(op_data.initial_mouse.x), float(op_data.initial_mouse.y));
|
|
|
|
GPUVertFormat *format = immVertexFormat();
|
|
uint pos2d = GPU_vertformat_attr_add(format, "pos", GPU_COMP_F32, 2, GPU_FETCH_FLOAT);
|
|
|
|
immBindBuiltinProgram(GPU_SHADER_3D_UNIFORM_COLOR);
|
|
|
|
immUniformColor3fvAlpha(circle_col, circle_alpha);
|
|
imm_draw_circle_wire_2d(pos2d, 0.0f, 0.0f, brush_radius_re, 80);
|
|
|
|
immUnbindProgram();
|
|
GPU_blend(GPU_BLEND_NONE);
|
|
}
|
|
|
|
static int min_distance_edit_invoke(bContext *C, wmOperator *op, const wmEvent *event)
|
|
{
|
|
Depsgraph *depsgraph = CTX_data_depsgraph_pointer(C);
|
|
ARegion *region = CTX_wm_region(C);
|
|
View3D *v3d = CTX_wm_view3d(C);
|
|
Scene *scene = CTX_data_scene(C);
|
|
|
|
Object &curves_ob_orig = *CTX_data_active_object(C);
|
|
Curves &curves_id_orig = *static_cast<Curves *>(curves_ob_orig.data);
|
|
Object &surface_ob_orig = *curves_id_orig.surface;
|
|
Object *surface_ob_eval = DEG_get_evaluated_object(depsgraph, &surface_ob_orig);
|
|
if (surface_ob_eval == nullptr) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
Mesh *surface_me_eval = BKE_object_get_evaluated_mesh(surface_ob_eval);
|
|
if (surface_me_eval == nullptr) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
BVHTreeFromMesh surface_bvh_eval;
|
|
BKE_bvhtree_from_mesh_get(&surface_bvh_eval, surface_me_eval, BVHTREE_FROM_LOOPTRI, 2);
|
|
BLI_SCOPED_DEFER([&]() { free_bvhtree_from_mesh(&surface_bvh_eval); });
|
|
|
|
const int2 mouse_pos_int_re{event->mval};
|
|
const float2 mouse_pos_re{mouse_pos_int_re};
|
|
|
|
float3 ray_start_wo, ray_end_wo;
|
|
ED_view3d_win_to_segment_clipped(
|
|
depsgraph, region, v3d, mouse_pos_re, ray_start_wo, ray_end_wo, true);
|
|
|
|
const CurvesSurfaceTransforms transforms{curves_ob_orig, &surface_ob_orig};
|
|
|
|
const float3 ray_start_su = math::transform_point(transforms.world_to_surface, ray_start_wo);
|
|
const float3 ray_end_su = math::transform_point(transforms.world_to_surface, ray_end_wo);
|
|
const float3 ray_direction_su = math::normalize(ray_end_su - ray_start_su);
|
|
|
|
BVHTreeRayHit ray_hit;
|
|
ray_hit.dist = FLT_MAX;
|
|
ray_hit.index = -1;
|
|
BLI_bvhtree_ray_cast(surface_bvh_eval.tree,
|
|
ray_start_su,
|
|
ray_direction_su,
|
|
0.0f,
|
|
&ray_hit,
|
|
surface_bvh_eval.raycast_callback,
|
|
&surface_bvh_eval);
|
|
if (ray_hit.index == -1) {
|
|
WM_report(RPT_ERROR, "Cursor must be over the surface mesh");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
const float3 hit_pos_su = ray_hit.co;
|
|
const float3 hit_normal_su = ray_hit.no;
|
|
|
|
const float3 hit_pos_cu = math::transform_point(transforms.surface_to_curves, hit_pos_su);
|
|
const float3 hit_normal_cu = math::normalize(
|
|
math::transform_direction(transforms.surface_to_curves_normal, hit_normal_su));
|
|
|
|
MinDistanceEditData *op_data = MEM_new<MinDistanceEditData>(__func__);
|
|
op_data->curves_to_world_mat = transforms.curves_to_world;
|
|
op_data->normal_cu = hit_normal_cu;
|
|
op_data->pos_cu = hit_pos_cu;
|
|
op_data->initial_mouse = event->xy;
|
|
op_data->brush = BKE_paint_brush(&scene->toolsettings->curves_sculpt->paint);
|
|
op_data->initial_minimum_distance = op_data->brush->curves_sculpt_settings->minimum_distance;
|
|
|
|
if (op_data->initial_minimum_distance <= 0.0f) {
|
|
op_data->initial_minimum_distance = 0.01f;
|
|
}
|
|
|
|
op->customdata = op_data;
|
|
|
|
/* Temporarily disable other paint cursors. */
|
|
wmWindowManager *wm = CTX_wm_manager(C);
|
|
op_data->orig_paintcursors = wm->paintcursors;
|
|
BLI_listbase_clear(&wm->paintcursors);
|
|
|
|
/* Add minimum distance paint cursor. */
|
|
op_data->cursor = WM_paint_cursor_activate(
|
|
SPACE_TYPE_ANY, RGN_TYPE_ANY, op->type->poll, min_distance_edit_draw, op_data);
|
|
|
|
op_data->region = CTX_wm_region(C);
|
|
op_data->rv3d = CTX_wm_region_view3d(C);
|
|
|
|
WM_event_add_modal_handler(C, op);
|
|
ED_region_tag_redraw(region);
|
|
return OPERATOR_RUNNING_MODAL;
|
|
}
|
|
|
|
static int min_distance_edit_modal(bContext *C, wmOperator *op, const wmEvent *event)
|
|
{
|
|
ARegion *region = CTX_wm_region(C);
|
|
MinDistanceEditData &op_data = *static_cast<MinDistanceEditData *>(op->customdata);
|
|
|
|
auto finish = [&]() {
|
|
wmWindowManager *wm = CTX_wm_manager(C);
|
|
|
|
/* Remove own cursor. */
|
|
WM_paint_cursor_end(static_cast<wmPaintCursor *>(op_data.cursor));
|
|
/* Restore original paint cursors. */
|
|
wm->paintcursors = op_data.orig_paintcursors;
|
|
|
|
ED_region_tag_redraw(region);
|
|
MEM_freeN(&op_data);
|
|
};
|
|
|
|
switch (event->type) {
|
|
case MOUSEMOVE: {
|
|
const int2 mouse_pos_int_re{event->xy};
|
|
const float2 mouse_pos_re{mouse_pos_int_re};
|
|
|
|
const float mouse_diff_x = mouse_pos_int_re.x - op_data.initial_mouse.x;
|
|
const float factor = powf(2, mouse_diff_x / UI_UNIT_X / 10.0f);
|
|
op_data.brush->curves_sculpt_settings->minimum_distance = op_data.initial_minimum_distance *
|
|
factor;
|
|
|
|
ED_region_tag_redraw(region);
|
|
WM_main_add_notifier(NC_SCENE | ND_TOOLSETTINGS, nullptr);
|
|
break;
|
|
}
|
|
case LEFTMOUSE: {
|
|
if (event->val == KM_PRESS) {
|
|
finish();
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
break;
|
|
}
|
|
case RIGHTMOUSE:
|
|
case EVT_ESCKEY: {
|
|
op_data.brush->curves_sculpt_settings->minimum_distance = op_data.initial_minimum_distance;
|
|
finish();
|
|
WM_main_add_notifier(NC_SCENE | ND_TOOLSETTINGS, nullptr);
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
|
|
return OPERATOR_RUNNING_MODAL;
|
|
}
|
|
|
|
} // namespace min_distance_edit
|
|
|
|
static void SCULPT_CURVES_OT_min_distance_edit(wmOperatorType *ot)
|
|
{
|
|
ot->name = "Edit Minimum Distance";
|
|
ot->idname = __func__;
|
|
ot->description = "Change the minimum distance used by the density brush";
|
|
|
|
ot->poll = min_distance_edit::min_distance_edit_poll;
|
|
ot->invoke = min_distance_edit::min_distance_edit_invoke;
|
|
ot->modal = min_distance_edit::min_distance_edit_modal;
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO | OPTYPE_DEPENDS_ON_CURSOR;
|
|
}
|
|
|
|
} // namespace blender::ed::sculpt_paint
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Registration
|
|
* \{ */
|
|
|
|
void ED_operatortypes_sculpt_curves()
|
|
{
|
|
using namespace blender::ed::sculpt_paint;
|
|
WM_operatortype_append(SCULPT_CURVES_OT_brush_stroke);
|
|
WM_operatortype_append(CURVES_OT_sculptmode_toggle);
|
|
WM_operatortype_append(SCULPT_CURVES_OT_select_random);
|
|
WM_operatortype_append(SCULPT_CURVES_OT_select_grow);
|
|
WM_operatortype_append(SCULPT_CURVES_OT_min_distance_edit);
|
|
}
|
|
|
|
/** \} */
|