Browse Source

Add a dedicated editor for Camera2D limits

kobewi 5 months ago
parent
commit
2bb3217819

+ 144 - 70
editor/plugins/camera_2d_editor_plugin.cpp

@@ -40,39 +40,157 @@
 #include "scene/gui/menu_button.h"
 
 void Camera2DEditor::edit(Camera2D *p_camera) {
-	if (p_camera == selected_camera) {
+if (p_camera == selected_camera) {
 		return;
 	}
+	const Callable update_overlays = callable_mp(plugin, &EditorPlugin::update_overlays);
+
+	if (selected_camera) {
+		selected_camera->disconnect(SceneStringName(draw), update_overlays);
+	}
 	selected_camera = p_camera;
+
+	if (selected_camera) {
+		selected_camera->connect(SceneStringName(draw), update_overlays);
+	}
+	plugin->update_overlays();
+}
+
+bool Camera2DEditor::forward_canvas_gui_input(const Ref<InputEvent> &p_event) {
+	if (!selected_camera || !selected_camera->is_limit_enabled()) {
+		return false;
+	}
+
+	Ref<InputEventMouseButton> mb = p_event;
+	if (mb.is_valid()) {
+		if (mb->get_button_index() == MouseButton::LEFT) {
+			if (mb->is_pressed()) {
+				const Rect2 limit_rect = selected_camera->get_limit_rect();
+				const Vector2 pos = CanvasItemEditor::get_singleton()->get_canvas_transform().affine_inverse().xform(mb->get_position());
+				drag_revert = limit_rect;
+
+				if (pos.y > limit_rect.position.y && pos.y < limit_rect.get_end().y) {
+					if (Math::abs(pos.x - limit_rect.position.x) < 8) {
+						drag_type = Drag::LEFT;
+						return true;
+					} else if (Math::abs(pos.x - limit_rect.get_end().x) < 8) {
+						drag_type = Drag::RIGHT;
+						return true;
+					}
+				} else if (pos.x > limit_rect.position.x && pos.x < limit_rect.get_end().x) {
+					if (Math::abs(pos.y - limit_rect.position.y) < 8) {
+						drag_type = Drag::TOP;
+						return true;
+					} else if (Math::abs(pos.y - limit_rect.get_end().y) < 8) {
+						drag_type = Drag::BOTTOM;
+						return true;
+					}
+				}
+
+				if (limit_rect.has_point(pos)) {
+					drag_type = Drag::CENTER;
+					center_drag_point = pos - limit_rect.position;
+					plugin->update_overlays();
+					return true;
+				}
+			} else if (drag_type != Drag::NONE) {
+				EditorUndoRedoManager *ur = EditorUndoRedoManager::get_singleton();
+				ur->create_action(TTR("Edit Camera2D Limits"));
+				ur->add_do_method(selected_camera, "_set_limit_rect", selected_camera->get_limit_rect());
+				ur->add_do_method(this, "_update_overlays_if_needed", selected_camera);
+				ur->add_undo_method(selected_camera, "_set_limit_rect", drag_revert);
+				ur->add_undo_method(this, "_update_overlays_if_needed", selected_camera);
+				ur->commit_action(false);
+
+				drag_type = Drag::NONE;
+				return true;
+			}
+		} else if (drag_type != Drag::NONE && mb->get_button_index() == MouseButton::RIGHT && mb->is_pressed()) {
+			selected_camera->set_limit_rect(drag_revert);
+			drag_type = Drag::NONE;
+			plugin->update_overlays();
+			return true;
+		}
+		return false;
+	}
+
+	if (drag_type == Drag::NONE) {
+		return false;
+	}
+
+	Ref<InputEventMouseMotion> mm = p_event;
+	if (mm.is_valid()) {
+		Vector2 pos = CanvasItemEditor::get_singleton()->get_canvas_transform().affine_inverse().xform(mm->get_position());
+		pos = CanvasItemEditor::get_singleton()->snap_point(pos);
+
+		switch (drag_type) {
+			case Drag::LEFT: {
+				selected_camera->set_limit(SIDE_LEFT, MIN(selected_camera->get_limit(SIDE_RIGHT), pos.x));
+				plugin->update_overlays();
+			} break;
+
+			case Drag::RIGHT: {
+				selected_camera->set_limit(SIDE_RIGHT, MAX(selected_camera->get_limit(SIDE_LEFT), pos.x));
+				plugin->update_overlays();
+			} break;
+
+			case Drag::TOP: {
+				selected_camera->set_limit(SIDE_TOP, MIN(selected_camera->get_limit(SIDE_BOTTOM), pos.y));
+				plugin->update_overlays();
+			} break;
+
+			case Drag::BOTTOM: {
+				selected_camera->set_limit(SIDE_BOTTOM, MAX(selected_camera->get_limit(SIDE_TOP), pos.y));
+				plugin->update_overlays();
+			} break;
+
+			case Drag::CENTER: {
+				Rect2 target_rect = selected_camera->get_limit_rect();
+				target_rect.position = pos - center_drag_point;
+				selected_camera->set_limit_rect(target_rect);
+				plugin->update_overlays();
+			} break;
+		}
+		return true;
+	}
+
+	return false;
+}
+
+void Camera2DEditor::forward_canvas_draw_over_viewport(Control *p_overlay) {
+	if (!selected_camera || !selected_camera->is_limit_enabled()) {
+		return;
+	}
+	Rect2 limit_rect = selected_camera->get_limit_rect();
+	limit_rect = CanvasItemEditor::get_singleton()->get_canvas_transform().xform(limit_rect);
+	p_overlay->draw_rect(limit_rect, Color(1, 1, 0.25, 0.63), false, 3);
 }
 
 void Camera2DEditor::_menu_option(int p_option) {
 	switch (p_option) {
 		case MENU_SNAP_LIMITS_TO_VIEWPORT: {
 			EditorUndoRedoManager *ur = EditorUndoRedoManager::get_singleton();
-			Rect2 prev_rect = selected_camera->get_limit_rect();
-			ur->create_action(TTR("Snap the Limits to the Viewport"), UndoRedo::MERGE_DISABLE, selected_camera);
-			ur->add_do_method(this, "_snap_limits_to_viewport");
-			ur->add_do_reference(selected_camera);
-			ur->add_undo_method(this, "_undo_snap_limits_to_viewport", prev_rect);
+			ur->create_action(TTR("Snap Camera2D Limits to the Viewport"), UndoRedo::MERGE_DISABLE, selected_camera);
+			ur->add_do_method(this, "_snap_limits_to_viewport", selected_camera);
+			ur->add_undo_method(selected_camera, "_set_limit_rect", selected_camera->get_limit_rect());
+			ur->add_undo_method(this, "_update_overlays_if_needed", selected_camera);
 			ur->commit_action();
 		} break;
 	}
 }
 
-void Camera2DEditor::_snap_limits_to_viewport() {
-	selected_camera->set_limit(SIDE_LEFT, 0);
-	selected_camera->set_limit(SIDE_TOP, 0);
-	selected_camera->set_limit(SIDE_RIGHT, GLOBAL_GET("display/window/size/viewport_width"));
-	selected_camera->set_limit(SIDE_BOTTOM, GLOBAL_GET("display/window/size/viewport_height"));
+void Camera2DEditor::_snap_limits_to_viewport(Camera2D *p_camera) {
+	p_camera->set_limit(SIDE_LEFT, 0);
+	p_camera->set_limit(SIDE_TOP, 0);
+	p_camera->set_limit(SIDE_RIGHT, GLOBAL_GET("display/window/size/viewport_width"));
+	p_camera->set_limit(SIDE_BOTTOM, GLOBAL_GET("display/window/size/viewport_height"));
+	_update_overlays_if_needed(p_camera);
 }
 
-void Camera2DEditor::_undo_snap_limits_to_viewport(const Rect2 &p_prev_rect) {
-	Point2 end = p_prev_rect.get_end();
-	selected_camera->set_limit(SIDE_LEFT, p_prev_rect.position.x);
-	selected_camera->set_limit(SIDE_TOP, p_prev_rect.position.y);
-	selected_camera->set_limit(SIDE_RIGHT, end.x);
-	selected_camera->set_limit(SIDE_BOTTOM, end.y);
+void Camera2DEditor::_update_overlays_if_needed(Camera2D *p_camera) {
+	if (p_camera == selected_camera) {
+		plugin->update_overlays();
+	}
 }
 
 void Camera2DEditor::_notification(int p_what) {
@@ -84,56 +202,24 @@ void Camera2DEditor::_notification(int p_what) {
 }
 
 void Camera2DEditor::_bind_methods() {
-	ClassDB::bind_method(D_METHOD("_snap_limits_to_viewport"), &Camera2DEditor::_snap_limits_to_viewport);
-	ClassDB::bind_method(D_METHOD("_undo_snap_limits_to_viewport", "prev_rect"), &Camera2DEditor::_undo_snap_limits_to_viewport);
+	ClassDB::bind_method(D_METHOD("_snap_limits_to_viewport", "camera"), &Camera2DEditor::_snap_limits_to_viewport);
+	ClassDB::bind_method(D_METHOD("_update_overlays_if_needed", "camera"), &Camera2DEditor::_update_overlays_if_needed);
 }
 
-Camera2DEditor::Camera2DEditor() {
-	options = memnew(MenuButton);
-
-	CanvasItemEditor::get_singleton()->add_control_to_menu_panel(options);
+Camera2DEditor::Camera2DEditor(EditorPlugin *p_plugin) {
+	plugin = p_plugin;
 
+	options = memnew(MenuButton);
 	options->set_text(TTRC("Camera2D"));
-
 	options->get_popup()->add_item(TTRC("Snap the Limits to the Viewport"), MENU_SNAP_LIMITS_TO_VIEWPORT);
 	options->set_switch_on_hover(true);
-
+	options->hide();
+	CanvasItemEditor::get_singleton()->add_control_to_menu_panel(options);
 	options->get_popup()->connect(SceneStringName(id_pressed), callable_mp(this, &Camera2DEditor::_menu_option));
-
-	add_user_signal(MethodInfo("_editor_theme_changed"));
-}
-
-void Camera2DEditorPlugin::_update_approach_text_visibility() {
-	if (camera_2d_editor->selected_camera == nullptr) {
-		return;
-	}
-	approach_to_move_rect->set_visible(camera_2d_editor->selected_camera->is_limit_enabled());
-}
-
-void Camera2DEditorPlugin::_editor_theme_changed() {
-	approach_to_move_rect->remove_theme_color_override(SceneStringName(font_color));
-	approach_to_move_rect->add_theme_color_override(SceneStringName(font_color), Color(0.6f, 0.6f, 0.6f, 1));
-	approach_to_move_rect->add_theme_color_override("font_shadow_color", Color(0.2f, 0.2f, 0.2f, 1));
-	approach_to_move_rect->add_theme_constant_override("shadow_outline_size", 1 * EDSCALE);
-	approach_to_move_rect->add_theme_constant_override("line_spacing", 0);
 }
 
 void Camera2DEditorPlugin::edit(Object *p_object) {
-	Callable update_text = callable_mp(this, &Camera2DEditorPlugin::_update_approach_text_visibility);
-	StringName update_signal = SNAME("_camera_limit_enabled_updated");
-
-	Camera2D *prev_cam = camera_2d_editor->selected_camera;
-	if (prev_cam != nullptr && prev_cam->is_connected(update_signal, update_text)) {
-		prev_cam->disconnect(update_signal, update_text);
-	}
-	Camera2D *cam = Object::cast_to<Camera2D>(p_object);
-	if (cam != nullptr) {
-		camera_2d_editor->edit(cam);
-		_update_approach_text_visibility();
-		if (!cam->is_connected(update_signal, update_text)) {
-			cam->connect(update_signal, update_text);
-		}
-	}
+	camera_2d_editor->edit(Object::cast_to<Camera2D>(p_object));
 }
 
 bool Camera2DEditorPlugin::handles(Object *p_object) const {
@@ -143,24 +229,12 @@ bool Camera2DEditorPlugin::handles(Object *p_object) const {
 void Camera2DEditorPlugin::make_visible(bool p_visible) {
 	if (p_visible) {
 		camera_2d_editor->options->show();
-		approach_to_move_rect->show();
 	} else {
 		camera_2d_editor->options->hide();
-		approach_to_move_rect->hide();
 	}
 }
 
 Camera2DEditorPlugin::Camera2DEditorPlugin() {
-	camera_2d_editor = memnew(Camera2DEditor);
+	camera_2d_editor = memnew(Camera2DEditor(this));
 	EditorNode::get_singleton()->get_gui_base()->add_child(camera_2d_editor);
-	camera_2d_editor->connect(SNAME("_editor_theme_changed"), callable_mp(this, &Camera2DEditorPlugin::_editor_theme_changed));
-
-	approach_to_move_rect = memnew(Label);
-	approach_to_move_rect->set_focus_mode(Control::FOCUS_ACCESSIBILITY);
-	approach_to_move_rect->set_text(TTRC("In Move Mode: \nHold Ctrl + left mouse button to move the limit rectangle.\nHold left mouse button to move the camera only."));
-	approach_to_move_rect->hide();
-	_editor_theme_changed();
-	CanvasItemEditor::get_singleton()->get_controls_container()->add_child(approach_to_move_rect);
-
-	make_visible(false);
 }

+ 23 - 8
editor/plugins/camera_2d_editor_plugin.h

@@ -39,18 +39,31 @@ class MenuButton;
 class Camera2DEditor : public Control {
 	GDCLASS(Camera2DEditor, Control);
 
+	EditorPlugin *plugin = nullptr;
+
 	enum Menu {
 		MENU_SNAP_LIMITS_TO_VIEWPORT,
 	};
 
+	enum class Drag {
+		NONE,
+		LEFT,
+		TOP,
+		RIGHT,
+		BOTTOM,
+		CENTER,
+	} drag_type;
+	Rect2 drag_revert;
+	Vector2 center_drag_point;
+
 	Camera2D *selected_camera = nullptr;
 
 	friend class Camera2DEditorPlugin;
 	MenuButton *options = nullptr;
 
 	void _menu_option(int p_option);
-	void _snap_limits_to_viewport();
-	void _undo_snap_limits_to_viewport(const Rect2 &p_prev_rect);
+	void _snap_limits_to_viewport(Camera2D *p_camera);
+	void _update_overlays_if_needed(Camera2D *p_camera);
 
 protected:
 	static void _bind_methods();
@@ -58,7 +71,11 @@ protected:
 
 public:
 	void edit(Camera2D *p_camera);
-	Camera2DEditor();
+
+	bool forward_canvas_gui_input(const Ref<InputEvent> &p_event);
+	void forward_canvas_draw_over_viewport(Control *p_overlay);
+
+	Camera2DEditor(EditorPlugin *p_plugin);
 };
 
 class Camera2DEditorPlugin : public EditorPlugin {
@@ -66,15 +83,13 @@ class Camera2DEditorPlugin : public EditorPlugin {
 
 	Camera2DEditor *camera_2d_editor = nullptr;
 
-	Label *approach_to_move_rect = nullptr;
-
-	void _editor_theme_changed();
-	void _update_approach_text_visibility();
-
 public:
 	virtual void edit(Object *p_object) override;
 	virtual bool handles(Object *p_object) const override;
 	virtual void make_visible(bool p_visible) override;
 
+	virtual bool forward_canvas_gui_input(const Ref<InputEvent> &p_event) override { return camera_2d_editor->forward_canvas_gui_input(p_event); }
+	virtual void forward_canvas_draw_over_viewport(Control *p_overlay) override { camera_2d_editor->forward_canvas_draw_over_viewport(p_overlay); }
+
 	Camera2DEditorPlugin();
 };

+ 6 - 77
scene/2d/camera_2d.cpp

@@ -34,58 +34,6 @@
 #include "core/input/input.h"
 #include "scene/main/viewport.h"
 
-#ifdef TOOLS_ENABLED
-Dictionary Camera2D::_edit_get_state() const {
-	Dictionary state = Node2D::_edit_get_state();
-	state["limit_rect"] = get_limit_rect();
-	return state;
-}
-
-void Camera2D::_edit_set_state(const Dictionary &p_state) {
-	if (p_state.has("limit_rect")) {
-		_set_limit_rect(p_state["limit_rect"]);
-	}
-	Node2D::_edit_set_state(p_state);
-}
-
-void Camera2D::_edit_set_position(const Point2 &p_position) {
-	if (_is_dragging_limit_rect()) {
-		Rect2 rect = get_limit_rect();
-		rect.position = p_position;
-		_set_limit_rect(rect);
-	} else {
-		Node2D::_edit_set_position(p_position);
-	}
-}
-
-Point2 Camera2D::_edit_get_position() const {
-	return _is_dragging_limit_rect() ? get_limit_rect().position : Node2D::_edit_get_position();
-}
-
-void Camera2D::_edit_set_rect(const Rect2 &p_rect) {
-	ERR_FAIL_COND(limit_enabled && !_edit_use_rect());
-	Rect2 rect = p_rect;
-	Vector2 scl = get_global_scale().abs();
-	rect.size *= scl;
-	rect.position = (rect.position + get_global_position()) * scl;
-	_set_limit_rect(rect);
-}
-#endif // TOOLS_ENABLED
-
-#ifdef DEBUG_ENABLED
-Rect2 Camera2D::_edit_get_rect() const {
-	Rect2 rect = get_limit_rect();
-	Vector2 scl = get_global_scale().abs();
-	rect.size /= scl;
-	rect.position = (rect.position - get_global_position()) / scl;
-	return rect;
-}
-
-bool Camera2D::_edit_use_rect() const {
-	return limit_enabled;
-}
-#endif // DEBUG_ENABLED
-
 void Camera2D::_update_scroll() {
 	if (!is_inside_tree() || !viewport) {
 		return;
@@ -120,10 +68,6 @@ void Camera2D::_update_scroll() {
 }
 
 #ifdef TOOLS_ENABLED
-bool Camera2D::_is_dragging_limit_rect() const {
-	return _edit_use_rect() && Input::get_singleton()->is_key_pressed(Key::CTRL);
-}
-
 void Camera2D::_project_settings_changed() {
 	if (screen_drawing_enabled) {
 		queue_redraw();
@@ -470,18 +414,9 @@ void Camera2D::_notification(int p_what) {
 					limit_drawing_width = 3;
 				}
 
-				Transform2D inv_transform = get_global_transform().affine_inverse();
-
-				Vector2 limit_points[4] = {
-					inv_transform.xform(Vector2(limit[SIDE_LEFT], limit[SIDE_TOP])),
-					inv_transform.xform(Vector2(limit[SIDE_RIGHT], limit[SIDE_TOP])),
-					inv_transform.xform(Vector2(limit[SIDE_RIGHT], limit[SIDE_BOTTOM])),
-					inv_transform.xform(Vector2(limit[SIDE_LEFT], limit[SIDE_BOTTOM]))
-				};
-
-				for (int i = 0; i < 4; i++) {
-					draw_line(limit_points[i], limit_points[(i + 1) % 4], Color(1, 1, 0.25, 0.63), limit_drawing_width);
-				}
+				draw_set_transform_matrix(get_global_transform().affine_inverse());
+				draw_rect(get_limit_rect(), Color(1, 1, 0.25, 0.63), false, limit_drawing_width);
+				draw_set_transform_matrix(Transform2D());
 			}
 
 			if (margin_drawing_enabled) {
@@ -564,9 +499,6 @@ void Camera2D::set_limit_enabled(bool p_limit_enabled) {
 	}
 	limit_enabled = p_limit_enabled;
 	_update_scroll();
-#ifdef TOOLS_ENABLED
-	emit_signal("_camera_limit_enabled_updated"); // Used for Camera2DEditorPlugin.
-#endif
 }
 
 bool Camera2D::is_limit_enabled() const {
@@ -632,8 +564,8 @@ void Camera2D::_make_current(Object *p_which) {
 	}
 }
 
-void Camera2D::_set_limit_rect(const Rect2 &p_limit_rect) {
-	Point2 limit_rect_end = p_limit_rect.get_end();
+void Camera2D::set_limit_rect(const Rect2 &p_limit_rect) {
+	const Point2 limit_rect_end = p_limit_rect.get_end();
 	set_limit(SIDE_LEFT, p_limit_rect.position.x);
 	set_limit(SIDE_TOP, p_limit_rect.position.y);
 	set_limit(SIDE_RIGHT, limit_rect_end.x);
@@ -964,6 +896,7 @@ void Camera2D::_bind_methods() {
 
 	ClassDB::bind_method(D_METHOD("set_limit", "margin", "limit"), &Camera2D::set_limit);
 	ClassDB::bind_method(D_METHOD("get_limit", "margin"), &Camera2D::get_limit);
+	ClassDB::bind_method(D_METHOD("_set_limit_rect", "rect"), &Camera2D::set_limit_rect);
 
 	ClassDB::bind_method(D_METHOD("set_limit_smoothing_enabled", "limit_smoothing_enabled"), &Camera2D::set_limit_smoothing_enabled);
 	ClassDB::bind_method(D_METHOD("is_limit_smoothing_enabled"), &Camera2D::is_limit_smoothing_enabled);
@@ -1066,8 +999,4 @@ void Camera2D::_bind_methods() {
 Camera2D::Camera2D() {
 	set_notify_transform(true);
 	set_hide_clip_children(true);
-
-#ifdef TOOLS_ENABLED
-	add_user_signal(MethodInfo("_camera_limit_enabled_updated")); // Camera2DEditorPlugin listens to this.
-#endif
 }

+ 1 - 19
scene/2d/camera_2d.h

@@ -90,7 +90,6 @@ protected:
 	void _update_scroll();
 
 #ifdef TOOLS_ENABLED
-	bool _is_dragging_limit_rect() const;
 	void _project_settings_changed();
 #endif
 
@@ -99,8 +98,6 @@ protected:
 
 	void _update_process_internal_for_smoothing();
 
-	void _set_limit_rect(const Rect2 &p_limit_rect);
-
 	bool screen_drawing_enabled = true;
 	bool limit_drawing_enabled = false;
 	bool margin_drawing_enabled = false;
@@ -124,22 +121,7 @@ protected:
 	static void _bind_methods();
 
 public:
-#ifdef TOOLS_ENABLED
-	virtual Dictionary _edit_get_state() const override;
-	virtual void _edit_set_state(const Dictionary &p_state) override;
-
-	virtual void _edit_set_position(const Point2 &p_position) override;
-	virtual Point2 _edit_get_position() const override;
-
-	virtual void _edit_set_rect(const Rect2 &p_rect) override;
-	virtual Size2 _edit_get_minimum_size() const override { return Size2(); }
-#endif // TOOLS_ENABLED
-
-#ifdef DEBUG_ENABLED
-	virtual Rect2 _edit_get_rect() const override;
-	virtual bool _edit_use_rect() const override;
-#endif // DEBUG_ENABLED
-
+	void set_limit_rect(const Rect2 &p_limit_rect);
 	Rect2 get_limit_rect() const;
 
 	void set_offset(const Vector2 &p_offset);