Path: blob/master/editor/scene/particle_process_material_editor_plugin.cpp
9896 views
/**************************************************************************/1/* particle_process_material_editor_plugin.cpp */2/**************************************************************************/3/* This file is part of: */4/* GODOT ENGINE */5/* https://godotengine.org */6/**************************************************************************/7/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */8/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */9/* */10/* Permission is hereby granted, free of charge, to any person obtaining */11/* a copy of this software and associated documentation files (the */12/* "Software"), to deal in the Software without restriction, including */13/* without limitation the rights to use, copy, modify, merge, publish, */14/* distribute, sublicense, and/or sell copies of the Software, and to */15/* permit persons to whom the Software is furnished to do so, subject to */16/* the following conditions: */17/* */18/* The above copyright notice and this permission notice shall be */19/* included in all copies or substantial portions of the Software. */20/* */21/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */22/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */23/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */24/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */25/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */26/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */27/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */28/**************************************************************************/2930#include "particle_process_material_editor_plugin.h"3132#include "editor/editor_string_names.h"33#include "editor/gui/editor_spin_slider.h"34#include "editor/settings/editor_settings.h"35#include "editor/themes/editor_theme_manager.h"36#include "scene/gui/box_container.h"37#include "scene/gui/button.h"38#include "scene/resources/particle_process_material.h"3940void ParticleProcessMaterialMinMaxPropertyEditor::_update_sizing() {41edit_size = range_edit_widget->get_size();42margin = Vector2(range_slider_left_icon->get_width(), (edit_size.y - range_slider_left_icon->get_height()) * 0.5);43usable_area = edit_size - margin * 2;44}4546void ParticleProcessMaterialMinMaxPropertyEditor::_range_edit_draw() {47ERR_FAIL_COND(range_slider_left_icon.is_null());48ERR_FAIL_COND(range_slider_right_icon.is_null());49_update_sizing();5051bool widget_active = mouse_inside || drag != Drag::NONE;5253// FIXME: Need to offset by 1 due to some outline bug.54range_edit_widget->draw_rect(Rect2(margin + Vector2(1, 1), usable_area - Vector2(1, 1)), widget_active ? background_color.lerp(normal_color, 0.3) : background_color, false, 1.0);5556Color draw_color;5758if (widget_active) {59float icon_offset = _get_left_offset() - range_slider_left_icon->get_width() - 1;6061if (drag == Drag::LEFT || drag == Drag::SCALE) {62draw_color = drag_color;63} else if (hover == Hover::LEFT) {64draw_color = hovered_color;65} else {66draw_color = normal_color;67}68range_edit_widget->draw_texture(range_slider_left_icon, Vector2(icon_offset, margin.y), draw_color);6970icon_offset = _get_right_offset();7172if (drag == Drag::RIGHT || drag == Drag::SCALE) {73draw_color = drag_color;74} else if (hover == Hover::RIGHT) {75draw_color = hovered_color;76} else {77draw_color = normal_color;78}79range_edit_widget->draw_texture(range_slider_right_icon, Vector2(icon_offset, margin.y), draw_color);80}8182if (drag == Drag::MIDDLE || drag == Drag::SCALE) {83draw_color = drag_color;84} else if (hover == Hover::MIDDLE) {85draw_color = hovered_color;86} else {87draw_color = normal_color;88}89range_edit_widget->draw_rect(_get_middle_rect(), draw_color);9091Rect2 midpoint_rect(Vector2(margin.x + usable_area.x * (_get_min_ratio() + _get_max_ratio()) * 0.5 - 1, margin.y + 2),92Vector2(2, usable_area.y - 4));9394range_edit_widget->draw_rect(midpoint_rect, midpoint_color);95}9697void ParticleProcessMaterialMinMaxPropertyEditor::_range_edit_gui_input(const Ref<InputEvent> &p_event) {98Ref<InputEventMouseButton> mb = p_event;99Ref<InputEventMouseMotion> mm = p_event;100101// Prevent unnecessary computations.102if ((mb.is_null() || mb->get_button_index() != MouseButton::LEFT) && (mm.is_null())) {103return;104}105106ERR_FAIL_COND(range_slider_left_icon.is_null());107ERR_FAIL_COND(range_slider_right_icon.is_null());108_update_sizing();109110if (mb.is_valid()) {111const Drag prev_drag = drag;112113if (mb->is_pressed()) {114if (mb->is_shift_pressed()) {115drag = Drag::SCALE;116drag_from_value = (max_range->get_value() - min_range->get_value()) * 0.5;117drag_midpoint = (max_range->get_value() + min_range->get_value()) * 0.5;118} else if (hover == Hover::LEFT) {119drag = Drag::LEFT;120drag_from_value = min_range->get_value();121} else if (hover == Hover::RIGHT) {122drag = Drag::RIGHT;123drag_from_value = max_range->get_value();124} else {125drag = Drag::MIDDLE;126drag_from_value = min_range->get_value();127}128drag_origin = mb->get_position().x;129} else {130drag = Drag::NONE;131}132133if (drag != prev_drag) {134range_edit_widget->queue_redraw();135}136}137138float property_length = property_range.y - property_range.x;139if (mm.is_valid()) {140switch (drag) {141case Drag::NONE: {142const Hover prev_hover = hover;143float left_icon_offset = _get_left_offset() - range_slider_left_icon->get_width() - 1;144145if (Rect2(Vector2(left_icon_offset, 0), range_slider_left_icon->get_size()).has_point(mm->get_position())) {146hover = Hover::LEFT;147} else if (Rect2(Vector2(_get_right_offset(), 0), range_slider_right_icon->get_size()).has_point(mm->get_position())) {148hover = Hover::RIGHT;149} else if (_get_middle_rect().has_point(mm->get_position())) {150hover = Hover::MIDDLE;151} else {152hover = Hover::NONE;153}154155if (hover != prev_hover) {156range_edit_widget->queue_redraw();157}158} break;159160case Drag::LEFT:161case Drag::RIGHT: {162float new_value = drag_from_value + (mm->get_position().x - drag_origin) / usable_area.x * property_length;163if (drag == Drag::LEFT) {164new_value = MIN(new_value, max_range->get_value());165_set_clamped_values(new_value, max_range->get_value());166} else {167new_value = MAX(new_value, min_range->get_value());168_set_clamped_values(min_range->get_value(), new_value);169}170} break;171172case Drag::MIDDLE: {173float delta = (mm->get_position().x - drag_origin) / usable_area.x * property_length;174float diff = max_range->get_value() - min_range->get_value();175delta = CLAMP(drag_from_value + delta, property_range.x, property_range.y - diff) - drag_from_value;176_set_clamped_values(drag_from_value + delta, drag_from_value + delta + diff);177} break;178179case Drag::SCALE: {180float delta = (mm->get_position().x - drag_origin) / usable_area.x * property_length + drag_from_value;181_set_clamped_values(MIN(drag_midpoint, drag_midpoint - delta), MAX(drag_midpoint, drag_midpoint + delta));182} break;183}184}185}186187void ParticleProcessMaterialMinMaxPropertyEditor::_set_mouse_inside(bool p_inside) {188mouse_inside = p_inside;189if (!p_inside) {190hover = Hover::NONE;191}192range_edit_widget->queue_redraw();193}194195float ParticleProcessMaterialMinMaxPropertyEditor::_get_min_ratio() const {196return (min_range->get_value() - property_range.x) / (property_range.y - property_range.x);197}198199float ParticleProcessMaterialMinMaxPropertyEditor::_get_max_ratio() const {200return (max_range->get_value() - property_range.x) / (property_range.y - property_range.x);201}202203float ParticleProcessMaterialMinMaxPropertyEditor::_get_left_offset() const {204return margin.x + usable_area.x * _get_min_ratio();205}206207float ParticleProcessMaterialMinMaxPropertyEditor::_get_right_offset() const {208return margin.x + usable_area.x * _get_max_ratio();209}210211Rect2 ParticleProcessMaterialMinMaxPropertyEditor::_get_middle_rect() const {212if (Math::is_equal_approx(min_range->get_value(), max_range->get_value())) {213return Rect2();214}215216return Rect2(217Vector2(_get_left_offset() - 1, margin.y),218Vector2(usable_area.x * (_get_max_ratio() - _get_min_ratio()) + 1, usable_area.y));219}220221void ParticleProcessMaterialMinMaxPropertyEditor::_set_clamped_values(float p_min, float p_max) {222// This is required for editing widget in case the properties have or_less or or_greater hint.223min_range->set_value(MAX(p_min, property_range.x));224max_range->set_value(MIN(p_max, property_range.y));225_update_slider_values();226_sync_property();227}228229void ParticleProcessMaterialMinMaxPropertyEditor::_sync_property() {230const Vector2 value = Vector2(min_range->get_value(), max_range->get_value());231emit_changed(get_edited_property(), value, "", true);232range_edit_widget->queue_redraw();233}234235void ParticleProcessMaterialMinMaxPropertyEditor::_update_mode() {236max_edit->set_read_only(false);237238switch (slider_mode) {239case Mode::RANGE: {240min_edit->set_label("min");241max_edit->set_label("max");242max_edit->set_block_signals(true);243max_edit->set_min(max_range->get_min());244max_edit->set_max(max_range->get_max());245max_edit->set_block_signals(false);246247min_edit->set_allow_lesser(min_range->is_lesser_allowed());248min_edit->set_allow_greater(min_range->is_greater_allowed());249max_edit->set_allow_lesser(max_range->is_lesser_allowed());250max_edit->set_allow_greater(max_range->is_greater_allowed());251} break;252253case Mode::MIDPOINT: {254min_edit->set_label("val");255max_edit->set_label(U"±");256max_edit->set_block_signals(true);257max_edit->set_min(0);258max_edit->set_block_signals(false);259260min_edit->set_allow_lesser(min_range->is_lesser_allowed());261min_edit->set_allow_greater(max_range->is_greater_allowed());262max_edit->set_allow_lesser(false);263max_edit->set_allow_greater(min_range->is_lesser_allowed() && max_range->is_greater_allowed());264} break;265}266_update_slider_values();267}268269void ParticleProcessMaterialMinMaxPropertyEditor::_toggle_mode(bool p_edit_mode) {270slider_mode = p_edit_mode ? Mode::MIDPOINT : Mode::RANGE;271EditorSettings::get_singleton()->set_project_metadata("editor_metadata", "particle_spin_mode", int(slider_mode));272_update_mode();273}274275void ParticleProcessMaterialMinMaxPropertyEditor::_update_slider_values() {276switch (slider_mode) {277case Mode::RANGE: {278min_edit->set_value_no_signal(min_range->get_value());279max_edit->set_value_no_signal(max_range->get_value());280} break;281282case Mode::MIDPOINT: {283min_edit->set_value_no_signal((min_range->get_value() + max_range->get_value()) * 0.5);284max_edit->set_value_no_signal((max_range->get_value() - min_range->get_value()) * 0.5);285286max_edit->set_block_signals(true);287max_edit->set_max(_get_max_spread());288max_edit->set_read_only(max_edit->get_max() == 0);289max_edit->set_block_signals(false);290} break;291}292}293294void ParticleProcessMaterialMinMaxPropertyEditor::_sync_sliders(float, const EditorSpinSlider *p_changed_slider) {295switch (slider_mode) {296case Mode::RANGE: {297if (p_changed_slider == max_edit) {298min_edit->set_value_no_signal(MIN(min_edit->get_value(), max_edit->get_value()));299}300min_range->set_value(min_edit->get_value());301if (p_changed_slider == min_edit) {302max_edit->set_value_no_signal(MAX(min_edit->get_value(), max_edit->get_value()));303}304max_range->set_value(max_edit->get_value());305_sync_property();306} break;307308case Mode::MIDPOINT: {309if (p_changed_slider == min_edit) {310max_edit->set_block_signals(true); // If max changes, value may change.311max_edit->set_max(_get_max_spread());312max_edit->set_read_only(max_edit->get_max() == 0);313max_edit->set_block_signals(false);314}315min_range->set_value(min_edit->get_value() - max_edit->get_value());316max_range->set_value(min_edit->get_value() + max_edit->get_value());317_sync_property();318} break;319}320321property_range.x = MIN(min_range->get_value(), min_range->get_min());322property_range.y = MAX(max_range->get_value(), max_range->get_max());323}324325float ParticleProcessMaterialMinMaxPropertyEditor::_get_max_spread() const {326float max_spread = max_range->get_max() - min_range->get_min();327328if (max_edit->is_greater_allowed()) {329return max_spread;330}331332if (!min_edit->is_lesser_allowed()) {333max_spread = MIN(max_spread, min_edit->get_value() - min_edit->get_min());334}335336if (!min_edit->is_greater_allowed()) {337max_spread = MIN(max_spread, min_edit->get_max() - min_edit->get_value());338}339340return max_spread;341}342343void ParticleProcessMaterialMinMaxPropertyEditor::_notification(int p_what) {344switch (p_what) {345case NOTIFICATION_THEME_CHANGED: {346toggle_mode_button->set_button_icon(get_editor_theme_icon(SNAME("Anchor")));347range_slider_left_icon = get_editor_theme_icon(SNAME("RangeSliderLeft"));348range_slider_right_icon = get_editor_theme_icon(SNAME("RangeSliderRight"));349350min_edit->add_theme_color_override(SNAME("label_color"), get_theme_color(SNAME("property_color_x"), EditorStringName(Editor)));351max_edit->add_theme_color_override(SNAME("label_color"), get_theme_color(SNAME("property_color_y"), EditorStringName(Editor)));352353const bool dark_theme = EditorThemeManager::is_dark_theme();354const Color accent_color = get_theme_color(SNAME("accent_color"), EditorStringName(Editor));355background_color = dark_theme ? Color(0.3, 0.3, 0.3) : Color(0.7, 0.7, 0.7);356normal_color = dark_theme ? Color(0.5, 0.5, 0.5) : Color(0.8, 0.8, 0.8);357hovered_color = dark_theme ? Color(0.8, 0.8, 0.8) : Color(0.6, 0.6, 0.6);358drag_color = hovered_color.lerp(accent_color, 0.8);359midpoint_color = dark_theme ? Color(1, 1, 1) : Color(0, 0, 0);360361range_edit_widget->set_custom_minimum_size(Vector2(0, range_slider_left_icon->get_height() + 8));362} break;363}364}365366void ParticleProcessMaterialMinMaxPropertyEditor::setup(float p_min, float p_max, float p_step, bool p_allow_less, bool p_allow_greater, bool p_degrees) {367property_range = Vector2(p_min, p_max);368369// Initially all Ranges share properties.370for (Range *range : Vector<Range *>{ min_range, min_edit, max_range, max_edit }) {371range->set_min(p_min);372range->set_max(p_max);373range->set_step(p_step);374range->set_allow_lesser(p_allow_less);375range->set_allow_greater(p_allow_greater);376}377378if (p_degrees) {379min_edit->set_suffix(U" \u00B0");380max_edit->set_suffix(U" \u00B0");381}382_update_mode();383}384385void ParticleProcessMaterialMinMaxPropertyEditor::update_property() {386const Vector2 value = get_edited_property_value();387min_range->set_value(value.x);388max_range->set_value(value.y);389_update_slider_values();390range_edit_widget->queue_redraw();391}392393ParticleProcessMaterialMinMaxPropertyEditor::ParticleProcessMaterialMinMaxPropertyEditor() {394VBoxContainer *content_vb = memnew(VBoxContainer);395content_vb->add_theme_constant_override(SNAME("separation"), 0);396add_child(content_vb);397398// Helper Range objects to keep absolute min and max values.399min_range = memnew(Range);400min_range->hide();401add_child(min_range);402403max_range = memnew(Range);404max_range->hide();405add_child(max_range);406407// Range edit widget.408HBoxContainer *hb = memnew(HBoxContainer);409content_vb->add_child(hb);410411range_edit_widget = memnew(Control);412range_edit_widget->set_h_size_flags(SIZE_EXPAND_FILL);413range_edit_widget->set_tooltip_text(TTR("Hold Shift to scale around midpoint instead of moving."));414hb->add_child(range_edit_widget);415range_edit_widget->connect(SceneStringName(draw), callable_mp(this, &ParticleProcessMaterialMinMaxPropertyEditor::_range_edit_draw));416range_edit_widget->connect(SceneStringName(gui_input), callable_mp(this, &ParticleProcessMaterialMinMaxPropertyEditor::_range_edit_gui_input));417range_edit_widget->connect(SceneStringName(mouse_entered), callable_mp(this, &ParticleProcessMaterialMinMaxPropertyEditor::_set_mouse_inside).bind(true));418range_edit_widget->connect(SceneStringName(mouse_exited), callable_mp(this, &ParticleProcessMaterialMinMaxPropertyEditor::_set_mouse_inside).bind(false));419420// Range controls for actual editing. Their min/max may depend on editing mode.421hb = memnew(HBoxContainer);422content_vb->add_child(hb);423424min_edit = memnew(EditorSpinSlider);425min_edit->set_h_size_flags(SIZE_EXPAND_FILL);426hb->add_child(min_edit);427min_edit->connect(SceneStringName(value_changed), callable_mp(this, &ParticleProcessMaterialMinMaxPropertyEditor::_sync_sliders).bind(min_edit));428429max_edit = memnew(EditorSpinSlider);430max_edit->set_h_size_flags(SIZE_EXPAND_FILL);431hb->add_child(max_edit);432max_edit->connect(SceneStringName(value_changed), callable_mp(this, &ParticleProcessMaterialMinMaxPropertyEditor::_sync_sliders).bind(max_edit));433434toggle_mode_button = memnew(Button);435toggle_mode_button->set_toggle_mode(true);436toggle_mode_button->set_tooltip_text(TTR("Toggle between minimum/maximum and base value/spread modes."));437hb->add_child(toggle_mode_button);438toggle_mode_button->connect(SceneStringName(toggled), callable_mp(this, &ParticleProcessMaterialMinMaxPropertyEditor::_toggle_mode));439440set_bottom_editor(content_vb);441}442443bool EditorInspectorParticleProcessMaterialPlugin::can_handle(Object *p_object) {444return Object::cast_to<ParticleProcessMaterial>(p_object);445}446447bool EditorInspectorParticleProcessMaterialPlugin::parse_property(Object *p_object, const Variant::Type p_type, const String &p_path, const PropertyHint p_hint, const String &p_hint_text, const BitField<PropertyUsageFlags> p_usage, const bool p_wide) {448if (!ParticleProcessMaterial::has_min_max_property(p_path)) {449return false;450}451ERR_FAIL_COND_V(p_hint != PROPERTY_HINT_RANGE, false);452453Ref<ParticleProcessMaterial> mat = Ref<ParticleProcessMaterial>(p_object);454ERR_FAIL_COND_V(mat.is_null(), false);455456PackedStringArray range_hint = p_hint_text.split(",");457float min = range_hint[0].to_float();458float max = range_hint[1].to_float();459float step = range_hint[2].to_float();460bool allow_less = range_hint.find("or_less", 3) > -1;461bool allow_greater = range_hint.find("or_greater", 3) > -1;462bool degrees = range_hint.find("degrees", 3) > -1;463464ParticleProcessMaterialMinMaxPropertyEditor *ed = memnew(ParticleProcessMaterialMinMaxPropertyEditor);465ed->setup(min, max, step, allow_less, allow_greater, degrees);466add_property_editor(p_path, ed);467468return true;469}470471472