Path: blob/master/editor/scene/2d/tiles/tile_data_editors.cpp
9906 views
/**************************************************************************/1/* tile_data_editors.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 "tile_data_editors.h"3132#include "tile_set_editor.h"3334#include "core/math/geometry_2d.h"35#include "core/math/random_pcg.h"36#include "core/os/keyboard.h"3738#include "editor/editor_node.h"39#include "editor/editor_string_names.h"40#include "editor/editor_undo_redo_manager.h"41#include "editor/inspector/editor_properties.h"42#include "editor/settings/editor_settings.h"43#include "editor/themes/editor_scale.h"4445#include "scene/gui/control.h"46#include "scene/gui/label.h"47#include "scene/gui/menu_button.h"48#include "scene/gui/option_button.h"49#include "scene/gui/separator.h"50#include "scene/gui/spin_box.h"5152#include "servers/navigation_server_2d.h"5354void TileDataEditor::_tile_set_changed_plan_update() {55_tile_set_changed_update_needed = true;56callable_mp(this, &TileDataEditor::_tile_set_changed_deferred_update).call_deferred();57}5859void TileDataEditor::_tile_set_changed_deferred_update() {60if (_tile_set_changed_update_needed) {61_tile_set_changed();62_tile_set_changed_update_needed = false;63}64}6566TileData *TileDataEditor::_get_tile_data(TileMapCell p_cell) {67ERR_FAIL_COND_V(tile_set.is_null(), nullptr);68ERR_FAIL_COND_V(!tile_set->has_source(p_cell.source_id), nullptr);6970TileData *td = nullptr;71TileSetSource *source = *tile_set->get_source(p_cell.source_id);72TileSetAtlasSource *atlas_source = Object::cast_to<TileSetAtlasSource>(source);73if (atlas_source) {74ERR_FAIL_COND_V(!atlas_source->has_tile(p_cell.get_atlas_coords()), nullptr);75ERR_FAIL_COND_V(!atlas_source->has_alternative_tile(p_cell.get_atlas_coords(), p_cell.alternative_tile), nullptr);76td = atlas_source->get_tile_data(p_cell.get_atlas_coords(), p_cell.alternative_tile);77}7879return td;80}8182void TileDataEditor::_bind_methods() {83ADD_SIGNAL(MethodInfo("needs_redraw"));84}8586void TileDataEditor::set_tile_set(Ref<TileSet> p_tile_set) {87if (tile_set.is_valid()) {88tile_set->disconnect_changed(callable_mp(this, &TileDataEditor::_tile_set_changed_plan_update));89}90tile_set = p_tile_set;91if (tile_set.is_valid()) {92tile_set->connect_changed(callable_mp(this, &TileDataEditor::_tile_set_changed_plan_update));93}94_tile_set_changed_plan_update();95}9697bool DummyObject::_set(const StringName &p_name, const Variant &p_value) {98if (properties.has(p_name)) {99properties[p_name] = p_value;100return true;101}102return false;103}104105bool DummyObject::_get(const StringName &p_name, Variant &r_ret) const {106if (properties.has(p_name)) {107r_ret = properties[p_name];108return true;109}110return false;111}112113bool DummyObject::has_dummy_property(const StringName &p_name) {114return properties.has(p_name);115}116117void DummyObject::add_dummy_property(const StringName &p_name) {118ERR_FAIL_COND(properties.has(p_name));119properties[p_name] = Variant();120}121122void DummyObject::remove_dummy_property(const StringName &p_name) {123ERR_FAIL_COND(!properties.has(p_name));124properties.erase(p_name);125}126127void DummyObject::clear_dummy_properties() {128properties.clear();129}130131void GenericTilePolygonEditor::_base_control_draw() {132ERR_FAIL_COND(tile_set.is_null());133134real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius");135136Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");137const Ref<Texture2D> handle = get_editor_theme_icon(SNAME("EditorPathSharpHandle"));138const Ref<Texture2D> add_handle = get_editor_theme_icon(SNAME("EditorHandleAdd"));139const Ref<StyleBox> focus_stylebox = get_theme_stylebox(SNAME("Focus"), EditorStringName(EditorStyles));140141// Get the background data.142Rect2 background_region;143TileData *tile_data = nullptr;144145if (background_atlas_source.is_valid()) {146tile_data = background_atlas_source->get_tile_data(background_atlas_coords, background_alternative_id);147ERR_FAIL_NULL(tile_data);148background_region = background_atlas_source->get_tile_texture_region(background_atlas_coords);149} else {150// If no tile was selected yet, use default size.151background_region.size = tile_set->get_tile_size();152}153154// Draw the focus rectangle.155if (base_control->has_focus()) {156base_control->draw_style_box(focus_stylebox, Rect2(Vector2(), base_control->get_size()));157}158159// Draw tile-related things.160const Size2 base_tile_size = tile_set->get_tile_size();161162Transform2D xform;163xform.set_origin(base_control->get_size() / 2 + panning);164xform.set_scale(Vector2(editor_zoom_widget->get_zoom(), editor_zoom_widget->get_zoom()));165base_control->draw_set_transform_matrix(xform);166167// Draw fill rect under texture region.168Rect2 texture_rect(Vector2(), background_region.size);169if (tile_data) {170texture_rect.position -= tile_data->get_texture_origin();171if (tile_data->get_transpose()) {172texture_rect.size = Size2(texture_rect.size.y, texture_rect.size.x);173}174}175texture_rect.position -= texture_rect.size / 2; // Half-size offset must be applied after transposing.176base_control->draw_rect(texture_rect, Color(1, 1, 1, 0.3));177178// Draw the background.179if (tile_data && background_atlas_source->get_texture().is_valid()) {180Size2 region_size = background_region.size;181if (tile_data->get_flip_h()) {182region_size.x = -region_size.x;183}184if (tile_data->get_flip_v()) {185region_size.y = -region_size.y;186}187// Destination rect position must account for transposing, size must not.188base_control->draw_texture_rect_region(background_atlas_source->get_texture(), Rect2(texture_rect.position, region_size), background_region, tile_data->get_modulate(), tile_data->get_transpose());189}190191// Compute and draw the grid area.192Rect2 grid_area = Rect2(-base_tile_size / 2, base_tile_size);193grid_area.expand_to(texture_rect.position);194grid_area.expand_to(texture_rect.get_end());195base_control->draw_rect(grid_area, Color(1, 1, 1, 0.3), false);196197// Draw grid.198if (current_snap_option == SNAP_GRID) {199Vector2 spacing = base_tile_size / snap_subdivision->get_value();200Vector2 origin = -base_tile_size / 2;201for (real_t y = origin.y; y < grid_area.get_end().y; y += spacing.y) {202base_control->draw_line(Vector2(grid_area.get_position().x, y), Vector2(grid_area.get_end().x, y), Color(1, 1, 1, 0.33));203}204for (real_t y = origin.y - spacing.y; y > grid_area.get_position().y; y -= spacing.y) {205base_control->draw_line(Vector2(grid_area.get_position().x, y), Vector2(grid_area.get_end().x, y), Color(1, 1, 1, 0.33));206}207for (real_t x = origin.x; x < grid_area.get_end().x; x += spacing.x) {208base_control->draw_line(Vector2(x, grid_area.get_position().y), Vector2(x, grid_area.get_end().y), Color(1, 1, 1, 0.33));209}210for (real_t x = origin.x - spacing.x; x > grid_area.get_position().x; x -= spacing.x) {211base_control->draw_line(Vector2(x, grid_area.get_position().y), Vector2(x, grid_area.get_end().y), Color(1, 1, 1, 0.33));212}213}214215// Draw the polygons.216for (const Vector<Vector2> &polygon : polygons) {217Color color = polygon_color;218if (!in_creation_polygon.is_empty()) {219color = color.darkened(0.3);220}221color.a = 0.5;222Vector<Color> v_color = { color };223base_control->draw_polygon(polygon, v_color);224225color.a = 0.7;226for (int j = 0; j < polygon.size(); j++) {227base_control->draw_line(polygon[j], polygon[(j + 1) % polygon.size()], color);228}229}230231// Draw the polygon in creation.232if (!in_creation_polygon.is_empty()) {233for (int i = 0; i < in_creation_polygon.size() - 1; i++) {234base_control->draw_line(in_creation_polygon[i], in_creation_polygon[i + 1], Color(1.0, 1.0, 1.0));235}236}237238Point2 in_creation_point = xform.affine_inverse().xform(base_control->get_local_mouse_position());239float in_creation_distance = grab_threshold * 2.0;240_snap_to_tile_shape(in_creation_point, in_creation_distance, grab_threshold / editor_zoom_widget->get_zoom());241_snap_point(in_creation_point);242243if (drag_type == DRAG_TYPE_CREATE_POINT && !in_creation_polygon.is_empty()) {244base_control->draw_line(in_creation_polygon[in_creation_polygon.size() - 1], in_creation_point, Color(1.0, 1.0, 1.0));245}246247// Draw the handles.248int tinted_polygon_index = -1;249int tinted_point_index = -1;250if (drag_type == DRAG_TYPE_DRAG_POINT) {251tinted_polygon_index = drag_polygon_index;252tinted_point_index = drag_point_index;253} else if (hovered_point_index >= 0) {254tinted_polygon_index = hovered_polygon_index;255tinted_point_index = hovered_point_index;256}257258base_control->draw_set_transform_matrix(Transform2D());259if (!in_creation_polygon.is_empty()) {260for (int i = 0; i < in_creation_polygon.size(); i++) {261base_control->draw_texture(handle, xform.xform(in_creation_polygon[i]) - handle->get_size() / 2);262}263} else {264for (int i = 0; i < (int)polygons.size(); i++) {265const Vector<Vector2> &polygon = polygons[i];266for (int j = 0; j < polygon.size(); j++) {267const Color poly_modulate = (tinted_polygon_index == i && tinted_point_index == j) ? Color(0.4, 1, 1) : Color(1, 1, 1);268base_control->draw_texture(handle, xform.xform(polygon[j]) - handle->get_size() / 2, poly_modulate);269}270}271}272273// Draw the text on top of the selected point.274if (tinted_polygon_index >= 0) {275Ref<Font> font = get_theme_font(SceneStringName(font), SNAME("Label"));276int font_size = get_theme_font_size(SceneStringName(font_size), SNAME("Label"));277String text = multiple_polygon_mode ? vformat("%d:%d", tinted_polygon_index, tinted_point_index) : vformat("%d", tinted_point_index);278Size2 text_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size);279base_control->draw_string(font, xform.xform(polygons[tinted_polygon_index][tinted_point_index]) - text_size * 0.5, text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size, Color(1.0, 1.0, 1.0, 0.5));280}281282if (drag_type == DRAG_TYPE_CREATE_POINT) {283base_control->draw_texture(handle, xform.xform(in_creation_point) - handle->get_size() / 2, Color(0.4, 1, 1));284}285286// Draw the point creation preview in edit mode.287if (hovered_segment_index >= 0) {288base_control->draw_texture(add_handle, xform.xform(hovered_segment_point) - add_handle->get_size() / 2);289}290291// Draw the tile shape line.292base_control->draw_set_transform_matrix(xform);293Transform2D tile_xform;294tile_xform.set_scale(base_tile_size);295tile_set->draw_tile_shape(base_control, tile_xform, grid_color, false);296base_control->draw_set_transform_matrix(Transform2D());297}298299void GenericTilePolygonEditor::_center_view() {300panning = Vector2();301base_control->queue_redraw();302button_center_view->set_disabled(true);303}304305void GenericTilePolygonEditor::_zoom_changed() {306base_control->queue_redraw();307}308309void GenericTilePolygonEditor::_advanced_menu_item_pressed(int p_item_pressed) {310EditorUndoRedoManager *undo_redo;311if (use_undo_redo) {312undo_redo = EditorUndoRedoManager::get_singleton();313} else {314// This nice hack allows for discarding undo actions without making code too complex.315undo_redo = memnew(EditorUndoRedoManager);316}317318switch (p_item_pressed) {319case RESET_TO_DEFAULT_TILE: {320undo_redo->create_action(TTR("Reset Polygons"));321undo_redo->add_do_method(this, "clear_polygons");322Vector<Vector2> polygon = tile_set->get_tile_shape_polygon();323for (int i = 0; i < polygon.size(); i++) {324polygon.write[i] = polygon[i] * tile_set->get_tile_size();325}326undo_redo->add_do_method(this, "add_polygon", polygon);327undo_redo->add_do_method(base_control, "queue_redraw");328undo_redo->add_do_method(this, "emit_signal", "polygons_changed");329undo_redo->add_undo_method(this, "clear_polygons");330for (const PackedVector2Array &poly : polygons) {331undo_redo->add_undo_method(this, "add_polygon", poly);332}333undo_redo->add_undo_method(base_control, "queue_redraw");334undo_redo->add_undo_method(this, "emit_signal", "polygons_changed");335undo_redo->commit_action(true);336} break;337case CLEAR_TILE: {338undo_redo->create_action(TTR("Clear Polygons"));339undo_redo->add_do_method(this, "clear_polygons");340undo_redo->add_do_method(base_control, "queue_redraw");341undo_redo->add_do_method(this, "emit_signal", "polygons_changed");342undo_redo->add_undo_method(this, "clear_polygons");343for (const PackedVector2Array &polygon : polygons) {344undo_redo->add_undo_method(this, "add_polygon", polygon);345}346undo_redo->add_undo_method(base_control, "queue_redraw");347undo_redo->add_undo_method(this, "emit_signal", "polygons_changed");348undo_redo->commit_action(true);349} break;350case ROTATE_RIGHT:351case ROTATE_LEFT:352case FLIP_HORIZONTALLY:353case FLIP_VERTICALLY: {354switch (p_item_pressed) {355case ROTATE_RIGHT: {356undo_redo->create_action(TTR("Rotate Polygons Right"));357} break;358case ROTATE_LEFT: {359undo_redo->create_action(TTR("Rotate Polygons Left"));360} break;361case FLIP_HORIZONTALLY: {362undo_redo->create_action(TTR("Flip Polygons Horizontally"));363} break;364case FLIP_VERTICALLY: {365undo_redo->create_action(TTR("Flip Polygons Vertically"));366} break;367default:368break;369}370for (unsigned int i = 0; i < polygons.size(); i++) {371Vector<Point2> new_polygon;372for (const Vector2 &vec : polygons[i]) {373Vector2 point = vec;374switch (p_item_pressed) {375case ROTATE_RIGHT: {376point = Vector2(-point.y, point.x);377} break;378case ROTATE_LEFT: {379point = Vector2(point.y, -point.x);380} break;381case FLIP_HORIZONTALLY: {382point = Vector2(-point.x, point.y);383} break;384case FLIP_VERTICALLY: {385point = Vector2(point.x, -point.y);386} break;387default:388break;389}390new_polygon.push_back(point);391}392undo_redo->add_do_method(this, "set_polygon", i, new_polygon);393}394undo_redo->add_do_method(base_control, "queue_redraw");395undo_redo->add_do_method(this, "emit_signal", "polygons_changed");396for (unsigned int i = 0; i < polygons.size(); i++) {397undo_redo->add_undo_method(this, "set_polygon", i, polygons[i]);398}399undo_redo->add_undo_method(base_control, "queue_redraw");400undo_redo->add_undo_method(this, "emit_signal", "polygons_changed");401undo_redo->commit_action(true);402} break;403default:404break;405}406407if (!use_undo_redo) {408memdelete(undo_redo);409}410}411412void GenericTilePolygonEditor::_grab_polygon_point(Vector2 p_pos, const Transform2D &p_polygon_xform, int &r_polygon_index, int &r_point_index) {413const real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius");414r_polygon_index = -1;415r_point_index = -1;416float closest_distance = grab_threshold + 1.0;417for (unsigned int i = 0; i < polygons.size(); i++) {418const Vector<Vector2> &polygon = polygons[i];419for (int j = 0; j < polygon.size(); j++) {420float distance = p_pos.distance_to(p_polygon_xform.xform(polygon[j]));421if (distance < grab_threshold && distance < closest_distance) {422r_polygon_index = i;423r_point_index = j;424closest_distance = distance;425}426}427}428}429430void GenericTilePolygonEditor::_grab_polygon_segment_point(Vector2 p_pos, const Transform2D &p_polygon_xform, int &r_polygon_index, int &r_segment_index, Vector2 &r_point) {431const real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius");432433Point2 point = p_polygon_xform.affine_inverse().xform(p_pos);434r_polygon_index = -1;435r_segment_index = -1;436float closest_distance = grab_threshold * 2.0;437for (unsigned int i = 0; i < polygons.size(); i++) {438const Vector<Vector2> &polygon = polygons[i];439for (int j = 0; j < polygon.size(); j++) {440const Vector2 segment_a = polygon[j];441const Vector2 segment_b = polygon[(j + 1) % polygon.size()];442Vector2 closest_point = Geometry2D::get_closest_point_to_segment(point, segment_a, segment_b);443float distance = closest_point.distance_to(point);444if (distance < grab_threshold / editor_zoom_widget->get_zoom() && distance < closest_distance) {445r_polygon_index = i;446r_segment_index = j;447r_point = closest_point;448closest_distance = distance;449}450}451}452}453454void GenericTilePolygonEditor::_snap_to_tile_shape(Point2 &r_point, float &r_current_snapped_dist, float p_snap_dist) {455ERR_FAIL_COND(tile_set.is_null());456457Vector<Point2> polygon = tile_set->get_tile_shape_polygon();458for (int i = 0; i < polygon.size(); i++) {459polygon.write[i] = polygon[i] * tile_set->get_tile_size();460}461Point2 snapped_point = r_point;462463// Snap to polygon vertices.464bool snapped = false;465for (int i = 0; i < polygon.size(); i++) {466float distance = r_point.distance_to(polygon[i]);467if (distance < p_snap_dist && distance < r_current_snapped_dist) {468snapped_point = polygon[i];469r_current_snapped_dist = distance;470snapped = true;471}472}473474// Snap to edges if we did not snap to vertices.475if (!snapped) {476for (int i = 0; i < polygon.size(); i++) {477const Vector2 segment_a = polygon[i];478const Vector2 segment_b = polygon[(i + 1) % polygon.size()];479Point2 point = Geometry2D::get_closest_point_to_segment(r_point, segment_a, segment_b);480float distance = r_point.distance_to(point);481if (distance < p_snap_dist && distance < r_current_snapped_dist) {482snapped_point = point;483r_current_snapped_dist = distance;484}485}486}487488r_point = snapped_point;489}490491void GenericTilePolygonEditor::_snap_point(Point2 &r_point) {492switch (current_snap_option) {493case SNAP_NONE:494break;495496case SNAP_HALF_PIXEL:497r_point = r_point.snappedf(0.5);498break;499500case SNAP_GRID: {501const Vector2 tile_size = tile_set->get_tile_size();502r_point = (r_point + tile_size / 2).snapped(tile_size / snap_subdivision->get_value()) - tile_size / 2;503} break;504}505}506507void GenericTilePolygonEditor::_base_control_gui_input(Ref<InputEvent> p_event) {508EditorUndoRedoManager *undo_redo;509if (use_undo_redo) {510undo_redo = EditorUndoRedoManager::get_singleton();511} else {512// This nice hack allows for discarding undo actions without making code too complex.513undo_redo = memnew(EditorUndoRedoManager);514}515516real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius");517518hovered_polygon_index = -1;519hovered_point_index = -1;520hovered_segment_index = -1;521hovered_segment_point = Vector2();522523Transform2D xform;524xform.set_origin(base_control->get_size() / 2 + panning);525xform.set_scale(Vector2(editor_zoom_widget->get_zoom(), editor_zoom_widget->get_zoom()));526527Ref<InputEventPanGesture> pan_gesture = p_event;528if (pan_gesture.is_valid()) {529panning += pan_gesture->get_delta() * 8;530drag_last_pos = Vector2();531button_center_view->set_disabled(panning.is_zero_approx());532accept_event();533}534535Ref<InputEventMagnifyGesture> magnify_gesture = p_event;536if (magnify_gesture.is_valid()) {537editor_zoom_widget->set_zoom(editor_zoom_widget->get_zoom() * magnify_gesture->get_factor());538_zoom_changed();539accept_event();540}541542Ref<InputEventMouseMotion> mm = p_event;543if (mm.is_valid()) {544if (drag_type == DRAG_TYPE_DRAG_POINT) {545ERR_FAIL_INDEX(drag_polygon_index, (int)polygons.size());546ERR_FAIL_INDEX(drag_point_index, polygons[drag_polygon_index].size());547Point2 point = xform.affine_inverse().xform(mm->get_position());548float distance = grab_threshold * 2.0;549_snap_to_tile_shape(point, distance, grab_threshold / editor_zoom_widget->get_zoom());550_snap_point(point);551polygons[drag_polygon_index].write[drag_point_index] = point;552} else if (drag_type == DRAG_TYPE_PAN) {553panning += mm->get_position() - drag_last_pos;554drag_last_pos = mm->get_position();555button_center_view->set_disabled(panning.is_zero_approx());556} else {557// Update hovered point.558_grab_polygon_point(mm->get_position(), xform, hovered_polygon_index, hovered_point_index);559560// If we have no hovered point, check if we hover a segment.561if (hovered_point_index == -1) {562_grab_polygon_segment_point(mm->get_position(), xform, hovered_polygon_index, hovered_segment_index, hovered_segment_point);563}564}565}566567Ref<InputEventMouseButton> mb = p_event;568if (mb.is_valid()) {569if (mb->get_button_index() == MouseButton::WHEEL_UP && mb->is_command_or_control_pressed()) {570editor_zoom_widget->set_zoom_by_increments(1);571_zoom_changed();572accept_event();573} else if (mb->get_button_index() == MouseButton::WHEEL_DOWN && mb->is_command_or_control_pressed()) {574editor_zoom_widget->set_zoom_by_increments(-1);575_zoom_changed();576accept_event();577} else if (mb->get_button_index() == MouseButton::LEFT) {578if (mb->is_pressed()) {579if (tools_button_group->get_pressed_button() != button_create) {580in_creation_polygon.clear();581}582if (tools_button_group->get_pressed_button() == button_create) {583// Create points.584if (in_creation_polygon.size() >= 3 && mb->get_position().distance_to(xform.xform(in_creation_polygon[0])) < grab_threshold) {585// Closes and create polygon.586if (!multiple_polygon_mode) {587clear_polygons();588}589int added = add_polygon(in_creation_polygon);590591in_creation_polygon.clear();592button_edit->set_pressed(true);593undo_redo->create_action(TTR("Edit Polygons"));594if (!multiple_polygon_mode) {595undo_redo->add_do_method(this, "clear_polygons");596}597undo_redo->add_do_method(this, "add_polygon", in_creation_polygon);598undo_redo->add_do_method(base_control, "queue_redraw");599undo_redo->add_undo_method(this, "remove_polygon", added);600undo_redo->add_undo_method(base_control, "queue_redraw");601undo_redo->commit_action(false);602emit_signal(SNAME("polygons_changed"));603} else {604// Create a new point.605drag_type = DRAG_TYPE_CREATE_POINT;606}607} else if (tools_button_group->get_pressed_button() == button_edit) {608// Edit points.609int closest_polygon;610int closest_point;611_grab_polygon_point(mb->get_position(), xform, closest_polygon, closest_point);612if (closest_polygon >= 0) {613drag_type = DRAG_TYPE_DRAG_POINT;614drag_polygon_index = closest_polygon;615drag_point_index = closest_point;616drag_old_polygon = polygons[drag_polygon_index];617} else {618// Create a point.619Vector2 point_to_create;620_grab_polygon_segment_point(mb->get_position(), xform, closest_polygon, closest_point, point_to_create);621if (closest_polygon >= 0) {622polygons[closest_polygon].insert(closest_point + 1, point_to_create);623drag_type = DRAG_TYPE_DRAG_POINT;624drag_polygon_index = closest_polygon;625drag_point_index = closest_point + 1;626drag_old_polygon = polygons[closest_polygon];627}628}629} else if (tools_button_group->get_pressed_button() == button_delete) {630// Remove point.631int closest_polygon;632int closest_point;633_grab_polygon_point(mb->get_position(), xform, closest_polygon, closest_point);634if (closest_polygon >= 0) {635PackedVector2Array old_polygon = polygons[closest_polygon];636polygons[closest_polygon].remove_at(closest_point);637undo_redo->create_action(TTR("Edit Polygons"));638if (polygons[closest_polygon].size() < 3) {639remove_polygon(closest_polygon);640undo_redo->add_do_method(this, "remove_polygon", closest_polygon);641undo_redo->add_undo_method(this, "add_polygon", old_polygon, closest_polygon);642} else {643undo_redo->add_do_method(this, "set_polygon", closest_polygon, polygons[closest_polygon]);644undo_redo->add_undo_method(this, "set_polygon", closest_polygon, old_polygon);645}646undo_redo->add_do_method(base_control, "queue_redraw");647undo_redo->add_undo_method(base_control, "queue_redraw");648undo_redo->commit_action(false);649emit_signal(SNAME("polygons_changed"));650}651}652} else {653if (drag_type == DRAG_TYPE_DRAG_POINT) {654undo_redo->create_action(TTR("Edit Polygons"));655undo_redo->add_do_method(this, "set_polygon", drag_polygon_index, polygons[drag_polygon_index]);656undo_redo->add_do_method(base_control, "queue_redraw");657undo_redo->add_undo_method(this, "set_polygon", drag_polygon_index, drag_old_polygon);658undo_redo->add_undo_method(base_control, "queue_redraw");659undo_redo->commit_action(false);660emit_signal(SNAME("polygons_changed"));661} else if (drag_type == DRAG_TYPE_CREATE_POINT) {662Point2 point = xform.affine_inverse().xform(mb->get_position());663float distance = grab_threshold * 2;664_snap_to_tile_shape(point, distance, grab_threshold / editor_zoom_widget->get_zoom());665_snap_point(point);666in_creation_polygon.push_back(point);667}668drag_type = DRAG_TYPE_NONE;669drag_point_index = -1;670}671672} else if (mb->get_button_index() == MouseButton::RIGHT) {673if (mb->is_pressed()) {674if (tools_button_group->get_pressed_button() == button_edit) {675// Remove point or pan.676int closest_polygon;677int closest_point;678_grab_polygon_point(mb->get_position(), xform, closest_polygon, closest_point);679if (closest_polygon >= 0) {680PackedVector2Array old_polygon = polygons[closest_polygon];681polygons[closest_polygon].remove_at(closest_point);682undo_redo->create_action(TTR("Edit Polygons"));683if (polygons[closest_polygon].size() < 3) {684remove_polygon(closest_polygon);685undo_redo->add_do_method(this, "remove_polygon", closest_polygon);686undo_redo->add_undo_method(this, "add_polygon", old_polygon, closest_polygon);687} else {688undo_redo->add_do_method(this, "set_polygon", closest_polygon, polygons[closest_polygon]);689undo_redo->add_undo_method(this, "set_polygon", closest_polygon, old_polygon);690}691undo_redo->add_do_method(base_control, "queue_redraw");692undo_redo->add_undo_method(base_control, "queue_redraw");693undo_redo->commit_action(false);694emit_signal(SNAME("polygons_changed"));695drag_type = DRAG_TYPE_NONE;696} else {697drag_type = DRAG_TYPE_PAN;698drag_last_pos = mb->get_position();699}700} else {701drag_type = DRAG_TYPE_PAN;702drag_last_pos = mb->get_position();703}704} else {705drag_type = DRAG_TYPE_NONE;706}707} else if (mb->get_button_index() == MouseButton::MIDDLE) {708if (mb->is_pressed()) {709drag_type = DRAG_TYPE_PAN;710drag_last_pos = mb->get_position();711} else {712drag_type = DRAG_TYPE_NONE;713}714}715}716717base_control->queue_redraw();718719if (!use_undo_redo) {720memdelete(undo_redo);721}722}723724void GenericTilePolygonEditor::_set_snap_option(int p_index) {725current_snap_option = p_index;726button_pixel_snap->set_button_icon(button_pixel_snap->get_popup()->get_item_icon(p_index));727snap_subdivision->set_visible(p_index == SNAP_GRID);728729if (initializing) {730return;731}732733base_control->queue_redraw();734_store_snap_options();735}736737void GenericTilePolygonEditor::_store_snap_options() {738EditorSettings::get_singleton()->set_project_metadata("editor_metadata", "tile_snap_option", current_snap_option);739EditorSettings::get_singleton()->set_project_metadata("editor_metadata", "tile_snap_subdiv", snap_subdivision->get_value());740}741742void GenericTilePolygonEditor::_toggle_expand(bool p_expand) {743if (p_expand) {744TileSetEditor::get_singleton()->add_expanded_editor(this);745} else {746TileSetEditor::get_singleton()->remove_expanded_editor();747}748}749750void GenericTilePolygonEditor::set_use_undo_redo(bool p_use_undo_redo) {751use_undo_redo = p_use_undo_redo;752}753754void GenericTilePolygonEditor::set_tile_set(Ref<TileSet> p_tile_set) {755ERR_FAIL_COND(p_tile_set.is_null());756if (tile_set == p_tile_set) {757return;758}759760// Set the default tile shape761clear_polygons();762if (p_tile_set.is_valid()) {763Vector<Vector2> polygon = p_tile_set->get_tile_shape_polygon();764for (int i = 0; i < polygon.size(); i++) {765polygon.write[i] = polygon[i] * p_tile_set->get_tile_size();766}767add_polygon(polygon);768}769770// Trigger a redraw on tile_set change.771Callable callable = callable_mp((CanvasItem *)base_control, &CanvasItem::queue_redraw);772if (tile_set.is_valid()) {773tile_set->disconnect_changed(callable);774}775776tile_set = p_tile_set;777778if (tile_set.is_valid()) {779tile_set->connect_changed(callable);780}781782// Set the default zoom value.783int default_control_y_size = 200 * EDSCALE;784Vector2 zoomed_tile = editor_zoom_widget->get_zoom() * tile_set->get_tile_size();785while (zoomed_tile.y < default_control_y_size) {786editor_zoom_widget->set_zoom_by_increments(6, false);787float current_zoom = editor_zoom_widget->get_zoom();788zoomed_tile = current_zoom * tile_set->get_tile_size();789if (Math::is_equal_approx(current_zoom, editor_zoom_widget->get_max_zoom())) {790break;791}792}793while (zoomed_tile.y > default_control_y_size) {794editor_zoom_widget->set_zoom_by_increments(-6, false);795float current_zoom = editor_zoom_widget->get_zoom();796zoomed_tile = current_zoom * tile_set->get_tile_size();797if (Math::is_equal_approx(current_zoom, editor_zoom_widget->get_min_zoom())) {798break;799}800}801editor_zoom_widget->set_zoom_by_increments(-6, false);802_zoom_changed();803}804805void GenericTilePolygonEditor::set_background_tile(const TileSetAtlasSource *p_atlas_source, const Vector2 &p_atlas_coords, int p_alternative_id) {806ERR_FAIL_NULL(p_atlas_source);807background_atlas_source = p_atlas_source;808background_atlas_coords = p_atlas_coords;809background_alternative_id = p_alternative_id;810base_control->queue_redraw();811}812813int GenericTilePolygonEditor::get_polygon_count() {814return polygons.size();815}816817int GenericTilePolygonEditor::add_polygon(const Vector<Point2> &p_polygon, int p_index) {818ERR_FAIL_COND_V(p_polygon.size() < 3, -1);819ERR_FAIL_COND_V(!multiple_polygon_mode && polygons.size() >= 1, -1);820821if (p_index < 0) {822polygons.push_back(p_polygon);823base_control->queue_redraw();824button_edit->set_pressed(true);825return polygons.size() - 1;826} else {827polygons.insert(p_index, p_polygon);828button_edit->set_pressed(true);829base_control->queue_redraw();830return p_index;831}832}833834void GenericTilePolygonEditor::remove_polygon(int p_index) {835ERR_FAIL_INDEX(p_index, (int)polygons.size());836polygons.remove_at(p_index);837838if (polygons.is_empty()) {839button_create->set_pressed(true);840}841base_control->queue_redraw();842}843844void GenericTilePolygonEditor::clear_polygons() {845polygons.clear();846base_control->queue_redraw();847}848849void GenericTilePolygonEditor::set_polygon(int p_polygon_index, const Vector<Point2> &p_polygon) {850ERR_FAIL_INDEX(p_polygon_index, (int)polygons.size());851ERR_FAIL_COND(p_polygon.size() < 3);852polygons[p_polygon_index] = p_polygon;853button_edit->set_pressed(true);854base_control->queue_redraw();855}856857Vector<Point2> GenericTilePolygonEditor::get_polygon(int p_polygon_index) {858ERR_FAIL_INDEX_V(p_polygon_index, (int)polygons.size(), Vector<Point2>());859return polygons[p_polygon_index];860}861862void GenericTilePolygonEditor::set_polygons_color(Color p_color) {863polygon_color = p_color;864base_control->queue_redraw();865}866867void GenericTilePolygonEditor::set_multiple_polygon_mode(bool p_multiple_polygon_mode) {868multiple_polygon_mode = p_multiple_polygon_mode;869}870871void GenericTilePolygonEditor::_notification(int p_what) {872switch (p_what) {873case NOTIFICATION_ENTER_TREE: {874if (!get_meta("reparented", false)) {875button_expand->set_pressed_no_signal(false);876}877} break;878879case NOTIFICATION_READY: {880get_parent()->connect(SceneStringName(tree_exited), callable_mp(TileSetEditor::get_singleton(), &TileSetEditor::remove_expanded_editor));881} break;882883case NOTIFICATION_THEME_CHANGED: {884button_expand->set_button_icon(get_editor_theme_icon(SNAME("DistractionFree")));885button_create->set_button_icon(get_editor_theme_icon(SNAME("CurveCreate")));886button_edit->set_button_icon(get_editor_theme_icon(SNAME("CurveEdit")));887button_delete->set_button_icon(get_editor_theme_icon(SNAME("CurveDelete")));888button_center_view->set_button_icon(get_editor_theme_icon(SNAME("CenterView")));889button_advanced_menu->set_button_icon(get_editor_theme_icon(SNAME("GuiTabMenuHl")));890button_pixel_snap->get_popup()->set_item_icon(0, get_editor_theme_icon(SNAME("SnapDisable")));891button_pixel_snap->get_popup()->set_item_icon(1, get_editor_theme_icon(SNAME("Snap")));892button_pixel_snap->get_popup()->set_item_icon(2, get_editor_theme_icon(SNAME("SnapGrid")));893button_pixel_snap->set_button_icon(button_pixel_snap->get_popup()->get_item_icon(current_snap_option));894895PopupMenu *p = button_advanced_menu->get_popup();896p->set_item_icon(p->get_item_index(ROTATE_RIGHT), get_editor_theme_icon(SNAME("RotateRight")));897p->set_item_icon(p->get_item_index(ROTATE_LEFT), get_editor_theme_icon(SNAME("RotateLeft")));898p->set_item_icon(p->get_item_index(FLIP_HORIZONTALLY), get_editor_theme_icon(SNAME("MirrorX")));899p->set_item_icon(p->get_item_index(FLIP_VERTICALLY), get_editor_theme_icon(SNAME("MirrorY")));900} break;901}902}903904void GenericTilePolygonEditor::_bind_methods() {905ClassDB::bind_method(D_METHOD("get_polygon_count"), &GenericTilePolygonEditor::get_polygon_count);906ClassDB::bind_method(D_METHOD("add_polygon", "polygon", "index"), &GenericTilePolygonEditor::add_polygon, DEFVAL(-1));907ClassDB::bind_method(D_METHOD("remove_polygon", "index"), &GenericTilePolygonEditor::remove_polygon);908ClassDB::bind_method(D_METHOD("clear_polygons"), &GenericTilePolygonEditor::clear_polygons);909ClassDB::bind_method(D_METHOD("set_polygon", "index", "polygon"), &GenericTilePolygonEditor::set_polygon);910ClassDB::bind_method(D_METHOD("get_polygon", "index"), &GenericTilePolygonEditor::get_polygon);911912ADD_SIGNAL(MethodInfo("polygons_changed"));913}914915GenericTilePolygonEditor::GenericTilePolygonEditor() {916toolbar = memnew(HBoxContainer);917add_child(toolbar);918919tools_button_group.instantiate();920921button_expand = memnew(Button);922button_expand->set_theme_type_variation(SceneStringName(FlatButton));923button_expand->set_toggle_mode(true);924button_expand->set_pressed(false);925button_expand->set_tooltip_text(TTR("Expand editor"));926button_expand->connect(SceneStringName(toggled), callable_mp(this, &GenericTilePolygonEditor::_toggle_expand));927toolbar->add_child(button_expand);928929toolbar->add_child(memnew(VSeparator));930931button_create = memnew(Button);932button_create->set_theme_type_variation(SceneStringName(FlatButton));933button_create->set_toggle_mode(true);934button_create->set_button_group(tools_button_group);935button_create->set_pressed(true);936button_create->set_tooltip_text(TTR("Add polygon tool"));937toolbar->add_child(button_create);938939button_edit = memnew(Button);940button_edit->set_theme_type_variation(SceneStringName(FlatButton));941button_edit->set_toggle_mode(true);942button_edit->set_button_group(tools_button_group);943button_edit->set_tooltip_text(TTR("Edit points tool"));944toolbar->add_child(button_edit);945946button_delete = memnew(Button);947button_delete->set_theme_type_variation(SceneStringName(FlatButton));948button_delete->set_toggle_mode(true);949button_delete->set_button_group(tools_button_group);950button_delete->set_tooltip_text(TTR("Delete points tool"));951toolbar->add_child(button_delete);952953button_advanced_menu = memnew(MenuButton);954button_advanced_menu->set_flat(false);955button_advanced_menu->set_accessibility_name(TTRC("Advanced"));956button_advanced_menu->set_theme_type_variation("FlatMenuButton");957button_advanced_menu->set_toggle_mode(true);958button_advanced_menu->get_popup()->add_item(TTR("Reset to default tile shape"), RESET_TO_DEFAULT_TILE, Key::F);959button_advanced_menu->get_popup()->add_item(TTR("Clear"), CLEAR_TILE, Key::C);960button_advanced_menu->get_popup()->add_separator();961button_advanced_menu->get_popup()->add_item(TTR("Rotate Right"), ROTATE_RIGHT, Key::R);962button_advanced_menu->get_popup()->add_item(TTR("Rotate Left"), ROTATE_LEFT, Key::E);963button_advanced_menu->get_popup()->add_item(TTR("Flip Horizontally"), FLIP_HORIZONTALLY, Key::H);964button_advanced_menu->get_popup()->add_item(TTR("Flip Vertically"), FLIP_VERTICALLY, Key::V);965button_advanced_menu->get_popup()->connect(SceneStringName(id_pressed), callable_mp(this, &GenericTilePolygonEditor::_advanced_menu_item_pressed));966button_advanced_menu->set_focus_mode(FOCUS_ALL);967toolbar->add_child(button_advanced_menu);968969toolbar->add_child(memnew(VSeparator));970971button_pixel_snap = memnew(MenuButton);972toolbar->add_child(button_pixel_snap);973button_pixel_snap->set_flat(false);974button_pixel_snap->set_accessibility_name(TTRC("Snap"));975button_pixel_snap->set_theme_type_variation("FlatMenuButton");976button_pixel_snap->set_tooltip_text(TTR("Toggle Grid Snap"));977button_pixel_snap->get_popup()->add_item(TTR("Disable Snap"), SNAP_NONE);978button_pixel_snap->get_popup()->add_item(TTR("Half-Pixel Snap"), SNAP_HALF_PIXEL);979button_pixel_snap->get_popup()->add_item(TTR("Grid Snap"), SNAP_GRID);980button_pixel_snap->get_popup()->connect("index_pressed", callable_mp(this, &GenericTilePolygonEditor::_set_snap_option));981982snap_subdivision = memnew(SpinBox);983toolbar->add_child(snap_subdivision);984snap_subdivision->set_accessibility_name(TTRC("Subdivision"));985snap_subdivision->get_line_edit()->add_theme_constant_override("minimum_character_width", 2);986snap_subdivision->set_min(1);987snap_subdivision->set_max(99);988989Control *root = memnew(Control);990root->set_v_size_flags(Control::SIZE_EXPAND_FILL);991root->set_custom_minimum_size(Size2(0, 200 * EDSCALE));992root->set_mouse_filter(Control::MOUSE_FILTER_IGNORE);993add_child(root);994995panel = memnew(Panel);996panel->set_anchors_and_offsets_preset(Control::PRESET_FULL_RECT);997panel->set_mouse_filter(Control::MOUSE_FILTER_IGNORE);998root->add_child(panel);9991000base_control = memnew(Control);1001base_control->set_texture_filter(CanvasItem::TEXTURE_FILTER_NEAREST);1002base_control->set_anchors_and_offsets_preset(Control::PRESET_FULL_RECT);1003base_control->connect(SceneStringName(draw), callable_mp(this, &GenericTilePolygonEditor::_base_control_draw));1004base_control->connect(SceneStringName(gui_input), callable_mp(this, &GenericTilePolygonEditor::_base_control_gui_input));1005base_control->set_clip_contents(true);1006base_control->set_focus_mode(Control::FOCUS_CLICK);1007root->add_child(base_control);1008snap_subdivision->connect(SceneStringName(value_changed), callable_mp((CanvasItem *)base_control, &CanvasItem::queue_redraw).unbind(1));1009snap_subdivision->connect(SceneStringName(value_changed), callable_mp(this, &GenericTilePolygonEditor::_store_snap_options).unbind(1));10101011editor_zoom_widget = memnew(EditorZoomWidget);1012editor_zoom_widget->setup_zoom_limits(0.125, 128.0);1013editor_zoom_widget->set_position(Vector2(5, 5));1014editor_zoom_widget->connect("zoom_changed", callable_mp(this, &GenericTilePolygonEditor::_zoom_changed).unbind(1));1015editor_zoom_widget->set_shortcut_context(this);1016root->add_child(editor_zoom_widget);10171018button_center_view = memnew(Button);1019button_center_view->set_anchors_and_offsets_preset(Control::PRESET_TOP_RIGHT, Control::PRESET_MODE_MINSIZE, 5);1020button_center_view->set_grow_direction_preset(Control::PRESET_TOP_RIGHT);1021button_center_view->connect(SceneStringName(pressed), callable_mp(this, &GenericTilePolygonEditor::_center_view));1022button_center_view->set_theme_type_variation(SceneStringName(FlatButton));1023button_center_view->set_tooltip_text(TTR("Center View"));1024button_center_view->set_disabled(true);1025root->add_child(button_center_view);10261027snap_subdivision->set_value_no_signal(EditorSettings::get_singleton()->get_project_metadata("editor_metadata", "tile_snap_subdiv", 4));1028_set_snap_option(EditorSettings::get_singleton()->get_project_metadata("editor_metadata", "tile_snap_option", SNAP_NONE));1029initializing = false;1030}10311032void TileDataDefaultEditor::_property_value_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field) {1033ERR_FAIL_NULL(dummy_object);1034dummy_object->set(p_property, p_value);1035emit_signal(SNAME("needs_redraw"));1036}10371038Variant TileDataDefaultEditor::_get_painted_value() {1039ERR_FAIL_NULL_V(dummy_object, Variant());1040return dummy_object->get(property);1041}10421043void TileDataDefaultEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {1044TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1045ERR_FAIL_NULL(tile_data);1046Variant value = tile_data->get(property);1047dummy_object->set(property, value);1048if (property_editor) {1049property_editor->update_property();1050}1051}10521053void TileDataDefaultEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) {1054TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1055ERR_FAIL_NULL(tile_data);1056tile_data->set(property, p_value);1057}10581059Variant TileDataDefaultEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {1060TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1061ERR_FAIL_NULL_V(tile_data, Variant());1062return tile_data->get(property);1063}10641065void TileDataDefaultEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) {1066EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();1067for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) {1068Vector2i coords = E.key.get_atlas_coords();1069undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/%s", coords.x, coords.y, E.key.alternative_tile, property), E.value);1070undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/%s", coords.x, coords.y, E.key.alternative_tile, property), p_new_value);1071}1072}10731074void TileDataDefaultEditor::forward_draw_over_atlas(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform) {1075if (drag_type == DRAG_TYPE_PAINT_RECT) {1076Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");1077Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);10781079p_canvas_item->draw_set_transform_matrix(p_transform);10801081Rect2i rect;1082rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true));1083rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position()), true));1084rect = rect.abs();10851086RBSet<TileMapCell> edited;1087for (int x = rect.get_position().x; x <= rect.get_end().x; x++) {1088for (int y = rect.get_position().y; y <= rect.get_end().y; y++) {1089Vector2i coords = Vector2i(x, y);1090coords = p_tile_set_atlas_source->get_tile_at_coords(coords);1091if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1092TileMapCell cell;1093cell.source_id = 0;1094cell.set_atlas_coords(coords);1095cell.alternative_tile = 0;1096edited.insert(cell);1097}1098}1099}11001101for (const TileMapCell &E : edited) {1102Vector2i coords = E.get_atlas_coords();1103p_canvas_item->draw_rect(p_tile_set_atlas_source->get_tile_texture_region(coords), selection_color, false);1104}1105p_canvas_item->draw_set_transform_matrix(Transform2D());1106}1107}11081109void TileDataDefaultEditor::forward_draw_over_alternatives(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform) {1110}11111112void TileDataDefaultEditor::forward_painting_atlas_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) {1113Ref<InputEventMouseMotion> mm = p_event;1114if (mm.is_valid()) {1115if (drag_type == DRAG_TYPE_PAINT) {1116Vector<Vector2i> line = Geometry2D::bresenham_line(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_last_pos, true), p_tile_atlas_view->get_atlas_tile_coords_at_pos(mm->get_position(), true));1117for (int i = 0; i < line.size(); i++) {1118Vector2i coords = p_tile_set_atlas_source->get_tile_at_coords(line[i]);1119if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1120TileMapCell cell;1121cell.source_id = 0;1122cell.set_atlas_coords(coords);1123cell.alternative_tile = 0;1124if (!drag_modified.has(cell)) {1125drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, 0);1126}1127_set_value(p_tile_set_atlas_source, coords, 0, drag_painted_value);1128}1129}1130drag_last_pos = mm->get_position();1131}1132}11331134EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();1135Ref<InputEventMouseButton> mb = p_event;1136if (mb.is_valid()) {1137if (mb->get_button_index() == MouseButton::LEFT) {1138if (mb->is_pressed()) {1139if (picker_button->is_pressed() || (mb->is_command_or_control_pressed() && !mb->is_shift_pressed())) {1140Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true);1141coords = p_tile_set_atlas_source->get_tile_at_coords(coords);1142if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1143_set_painted_value(p_tile_set_atlas_source, coords, 0);1144picker_button->set_pressed(false);1145}1146} else if (mb->is_command_or_control_pressed() && mb->is_shift_pressed()) {1147drag_type = DRAG_TYPE_PAINT_RECT;1148drag_modified.clear();1149drag_painted_value = _get_painted_value();1150drag_start_pos = mb->get_position();1151} else {1152drag_type = DRAG_TYPE_PAINT;1153drag_modified.clear();1154drag_painted_value = _get_painted_value();1155Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true);1156coords = p_tile_set_atlas_source->get_tile_at_coords(coords);1157if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1158TileMapCell cell;1159cell.source_id = 0;1160cell.set_atlas_coords(coords);1161cell.alternative_tile = 0;1162drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, 0);1163_set_value(p_tile_set_atlas_source, coords, 0, drag_painted_value);1164}1165drag_last_pos = mb->get_position();1166}1167} else {1168if (drag_type == DRAG_TYPE_PAINT_RECT) {1169Rect2i rect;1170rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true));1171rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true));1172rect = rect.abs();11731174drag_modified.clear();1175for (int x = rect.get_position().x; x <= rect.get_end().x; x++) {1176for (int y = rect.get_position().y; y <= rect.get_end().y; y++) {1177Vector2i coords = Vector2i(x, y);1178coords = p_tile_set_atlas_source->get_tile_at_coords(coords);1179if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1180TileMapCell cell;1181cell.source_id = 0;1182cell.set_atlas_coords(coords);1183cell.alternative_tile = 0;1184drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, 0);1185}1186}1187}1188undo_redo->create_action(TTR("Painting Tiles Property"));1189_setup_undo_redo_action(p_tile_set_atlas_source, drag_modified, drag_painted_value);1190undo_redo->commit_action(true);1191drag_type = DRAG_TYPE_NONE;1192} else if (drag_type == DRAG_TYPE_PAINT) {1193undo_redo->create_action(TTR("Painting Tiles Property"));1194_setup_undo_redo_action(p_tile_set_atlas_source, drag_modified, drag_painted_value);1195undo_redo->commit_action(false);1196drag_type = DRAG_TYPE_NONE;1197}1198}1199}1200}1201}12021203void TileDataDefaultEditor::forward_painting_alternatives_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) {1204Ref<InputEventMouseMotion> mm = p_event;1205if (mm.is_valid()) {1206if (drag_type == DRAG_TYPE_PAINT) {1207Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mm->get_position());1208Vector2i coords = Vector2i(tile.x, tile.y);1209int alternative_tile = tile.z;12101211if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1212TileMapCell cell;1213cell.source_id = 0;1214cell.set_atlas_coords(coords);1215cell.alternative_tile = alternative_tile;1216if (!drag_modified.has(cell)) {1217drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, alternative_tile);1218}1219_set_value(p_tile_set_atlas_source, coords, alternative_tile, drag_painted_value);1220}12211222drag_last_pos = mm->get_position();1223}1224}12251226Ref<InputEventMouseButton> mb = p_event;1227if (mb.is_valid()) {1228if (mb->get_button_index() == MouseButton::LEFT) {1229if (mb->is_pressed()) {1230if (picker_button->is_pressed()) {1231Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position());1232Vector2i coords = Vector2i(tile.x, tile.y);1233int alternative_tile = tile.z;1234if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1235_set_painted_value(p_tile_set_atlas_source, coords, alternative_tile);1236picker_button->set_pressed(false);1237}1238} else {1239drag_type = DRAG_TYPE_PAINT;1240drag_modified.clear();1241drag_painted_value = _get_painted_value();12421243Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position());1244Vector2i coords = Vector2i(tile.x, tile.y);1245int alternative_tile = tile.z;12461247if (coords != TileSetSource::INVALID_ATLAS_COORDS) {1248TileMapCell cell;1249cell.source_id = 0;1250cell.set_atlas_coords(coords);1251cell.alternative_tile = alternative_tile;1252drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, alternative_tile);1253_set_value(p_tile_set_atlas_source, coords, alternative_tile, drag_painted_value);1254}1255drag_last_pos = mb->get_position();1256}1257} else {1258EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();1259undo_redo->create_action(TTR("Painting Tiles Property"));1260_setup_undo_redo_action(p_tile_set_atlas_source, drag_modified, drag_painted_value);1261undo_redo->commit_action(false);1262drag_type = DRAG_TYPE_NONE;1263}1264}1265}1266}12671268void TileDataDefaultEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {1269TileData *tile_data = _get_tile_data(p_cell);1270ERR_FAIL_NULL(tile_data);12711272bool valid;1273Variant value = tile_data->get(property, &valid);1274if (!valid) {1275return;1276}12771278Vector2 texture_origin = tile_data->get_texture_origin();1279if (value.get_type() == Variant::BOOL) {1280Ref<Texture2D> texture = (bool)value ? tile_bool_checked : tile_bool_unchecked;1281int size = MIN(tile_set->get_tile_size().x, tile_set->get_tile_size().y) / 3;1282Rect2 rect = p_transform.xform(Rect2(Vector2(-size / 2, -size / 2) - texture_origin, Vector2(size, size)));1283p_canvas_item->draw_texture_rect(texture, rect);1284} else if (value.get_type() == Variant::COLOR) {1285int size = MIN(tile_set->get_tile_size().x, tile_set->get_tile_size().y) / 3;1286Rect2 rect = p_transform.xform(Rect2(Vector2(-size / 2, -size / 2) - texture_origin, Vector2(size, size)));1287p_canvas_item->draw_rect(rect, value);1288} else {1289Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts));1290int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts));1291String text;1292// Round floating point precision to 2 digits, as tiles don't have that much space.1293switch (value.get_type()) {1294case Variant::FLOAT:1295text = vformat("%.2f", value);1296break;1297case Variant::VECTOR2:1298case Variant::VECTOR3:1299case Variant::VECTOR4:1300text = vformat("%.2v", value);1301break;1302default:1303text = value.stringify();1304break;1305}13061307Color color = Color(1, 1, 1);1308if (p_selected) {1309Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");1310Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);1311selection_color.set_v(0.9);1312color = selection_color;1313} else if (is_visible_in_tree()) {1314Variant painted_value = _get_painted_value();1315bool equal = (painted_value.get_type() == Variant::FLOAT && value.get_type() == Variant::FLOAT) ? Math::is_equal_approx(float(painted_value), float(value)) : painted_value == value;1316if (equal) {1317color = Color(0.7, 0.7, 0.7);1318}1319}13201321Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size);1322p_canvas_item->draw_string_outline(font, p_transform.xform(-texture_origin) + Vector2i(-string_size.x / 2, string_size.y / 4), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, 3, Color(0, 0, 0));1323p_canvas_item->draw_string(font, p_transform.xform(-texture_origin) + Vector2i(-string_size.x / 2, string_size.y / 4), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, color);1324}1325}13261327void TileDataDefaultEditor::setup_property_editor(Variant::Type p_type, const String &p_property, const String &p_label, const Variant &p_default_value) {1328ERR_FAIL_COND_MSG(!property.is_empty(), "Cannot setup TileDataDefaultEditor twice");1329property = p_property;1330property_type = p_type;13311332// Update everything.1333if (property_editor) {1334property_editor->queue_free();1335}13361337// Update the dummy object.1338dummy_object->add_dummy_property(p_property);13391340// Get the default value for the type.1341if (p_default_value == Variant()) {1342Callable::CallError error;1343Variant painted_value;1344Variant::construct(p_type, painted_value, nullptr, 0, error);1345dummy_object->set(p_property, painted_value);1346} else {1347dummy_object->set(p_property, p_default_value);1348}13491350// Create and setup the property editor.1351property_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, p_type, p_property, PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT);1352property_editor->set_object_and_property(dummy_object, p_property);1353if (p_label.is_empty()) {1354property_editor->set_label(EditorPropertyNameProcessor::get_singleton()->process_name(p_property, EditorPropertyNameProcessor::get_default_inspector_style(), p_property));1355} else {1356property_editor->set_label(p_label);1357}1358property_editor->connect("property_changed", callable_mp(this, &TileDataDefaultEditor::_property_value_changed).unbind(1));1359property_editor->set_tooltip_text(p_property);1360property_editor->update_property();1361add_child(property_editor);1362}13631364void TileDataDefaultEditor::_notification(int p_what) {1365switch (p_what) {1366case NOTIFICATION_THEME_CHANGED: {1367picker_button->set_button_icon(get_editor_theme_icon(SNAME("ColorPick")));1368tile_bool_checked = get_editor_theme_icon(SNAME("TileChecked"));1369tile_bool_unchecked = get_editor_theme_icon(SNAME("TileUnchecked"));1370} break;1371}1372}13731374Variant::Type TileDataDefaultEditor::get_property_type() {1375return property_type;1376}13771378TileDataDefaultEditor::TileDataDefaultEditor() {1379label = memnew(Label);1380label->set_text(TTR("Painting:"));1381label->set_theme_type_variation("HeaderSmall");1382add_child(label);13831384picker_button = memnew(Button);1385picker_button->set_theme_type_variation(SceneStringName(FlatButton));1386picker_button->set_toggle_mode(true);1387picker_button->set_shortcut(ED_GET_SHORTCUT("tiles_editor/picker"));1388toolbar->add_child(picker_button);1389}13901391TileDataDefaultEditor::~TileDataDefaultEditor() {1392toolbar->queue_free();1393memdelete(dummy_object);1394}13951396void TileDataTextureOriginEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {1397TileData *tile_data = _get_tile_data(p_cell);1398ERR_FAIL_NULL(tile_data);13991400Vector2i tile_set_tile_size = tile_set->get_tile_size();1401Color color = Color(1.0, 1.0, 1.0);1402if (p_selected) {1403Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");1404Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);1405color = selection_color;1406}14071408TileSetSource *source = *(tile_set->get_source(p_cell.source_id));1409TileSetAtlasSource *atlas_source = Object::cast_to<TileSetAtlasSource>(source);1410if (atlas_source->is_rect_in_tile_texture_region(p_cell.get_atlas_coords(), p_cell.alternative_tile, Rect2(Vector2(-tile_set_tile_size) / 2, tile_set_tile_size))) {1411tile_set->draw_tile_shape(p_canvas_item, p_transform.scaled_local(tile_set_tile_size), color);1412}14131414if (atlas_source->is_position_in_tile_texture_region(p_cell.get_atlas_coords(), p_cell.alternative_tile, Vector2())) {1415Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_editor_theme_icon(SNAME("EditorPosition"));1416p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2()) - (position_icon->get_size() / 2), color);1417} else {1418Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts));1419int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts));1420Vector2 texture_origin = tile_data->get_texture_origin();1421String text = vformat("%s", texture_origin);1422Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size);1423p_canvas_item->draw_string_outline(font, p_transform.xform(-texture_origin) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, 1, Color(0, 0, 0, 1));1424p_canvas_item->draw_string(font, p_transform.xform(-texture_origin) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, color);1425}1426}14271428void TileDataPositionEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {1429TileData *tile_data = _get_tile_data(p_cell);1430ERR_FAIL_NULL(tile_data);14311432bool valid;1433Variant value = tile_data->get(property, &valid);1434if (!valid) {1435return;1436}1437ERR_FAIL_COND(value.get_type() != Variant::VECTOR2I && value.get_type() != Variant::VECTOR2);14381439Color color = Color(1.0, 1.0, 1.0);1440if (p_selected) {1441Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");1442Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);1443color = selection_color;1444}1445Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_editor_theme_icon(SNAME("EditorPosition"));1446p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2(value)) - position_icon->get_size() / 2, color);1447}14481449void TileDataYSortEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {1450TileData *tile_data = _get_tile_data(p_cell);1451ERR_FAIL_NULL(tile_data);14521453Color color = Color(1.0, 1.0, 1.0);1454if (p_selected) {1455Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");1456Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);1457color = selection_color;1458}1459Vector2 texture_origin = tile_data->get_texture_origin();1460TileSetSource *source = *(tile_set->get_source(p_cell.source_id));1461TileSetAtlasSource *atlas_source = Object::cast_to<TileSetAtlasSource>(source);1462if (atlas_source->is_position_in_tile_texture_region(p_cell.get_atlas_coords(), p_cell.alternative_tile, Vector2(0, tile_data->get_y_sort_origin()))) {1463Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_editor_theme_icon(SNAME("EditorPosition"));1464p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2(0, tile_data->get_y_sort_origin())) - position_icon->get_size() / 2, color);1465} else {1466Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts));1467int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts));1468String text = vformat("%s", tile_data->get_y_sort_origin());14691470Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size);1471p_canvas_item->draw_string_outline(font, p_transform.xform(-texture_origin) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, 1, Color(0, 0, 0, 1));1472p_canvas_item->draw_string(font, p_transform.xform(-texture_origin) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, color);1473}1474}14751476void TileDataOcclusionShapeEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {1477TileData *tile_data = _get_tile_data(p_cell);1478ERR_FAIL_NULL(tile_data);14791480Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");1481Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);1482Color color = grid_color.darkened(0.2);1483if (p_selected) {1484color = selection_color.darkened(0.2);1485}1486color.a *= 0.5;14871488Vector<Color> debug_occlusion_color = { color };14891490RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform);1491for (int i = 0; i < tile_data->get_occluder_polygons_count(occlusion_layer); i++) {1492Ref<OccluderPolygon2D> occluder = tile_data->get_occluder_polygon(occlusion_layer, i);1493if (occluder.is_valid() && occluder->get_polygon().size() >= 3) {1494p_canvas_item->draw_polygon(Variant(occluder->get_polygon()), debug_occlusion_color);1495}1496}1497RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D());1498}14991500Variant TileDataOcclusionShapeEditor::_get_painted_value() {1501Array polygons;1502for (int i = 0; i < polygon_editor->get_polygon_count(); i++) {1503Ref<OccluderPolygon2D> occluder_polygon;1504occluder_polygon.instantiate();1505occluder_polygon->set_polygon(polygon_editor->get_polygon(i));1506polygons.push_back(occluder_polygon);1507}1508return polygons;1509}15101511void TileDataOcclusionShapeEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {1512TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1513ERR_FAIL_NULL(tile_data);15141515polygon_editor->clear_polygons();1516for (int i = 0; i < tile_data->get_occluder_polygons_count(occlusion_layer); i++) {1517Ref<OccluderPolygon2D> occluder_polygon = tile_data->get_occluder_polygon(occlusion_layer, i);1518if (occluder_polygon.is_valid()) {1519polygon_editor->add_polygon(occluder_polygon->get_polygon());1520}1521}1522polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile);1523}15241525void TileDataOcclusionShapeEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) {1526TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1527ERR_FAIL_NULL(tile_data);15281529Array polygons = p_value;1530tile_data->set_occluder_polygons_count(occlusion_layer, polygons.size());1531for (int i = 0; i < polygons.size(); i++) {1532Ref<OccluderPolygon2D> occluder_polygon = polygons[i];1533tile_data->set_occluder_polygon(occlusion_layer, i, occluder_polygon);1534}15351536polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile);1537}15381539Variant TileDataOcclusionShapeEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {1540TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1541ERR_FAIL_NULL_V(tile_data, Variant());1542Array polygons;1543for (int i = 0; i < tile_data->get_occluder_polygons_count(occlusion_layer); i++) {1544polygons.push_back(tile_data->get_occluder_polygon(occlusion_layer, i));1545}1546return polygons;1547}15481549void TileDataOcclusionShapeEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) {1550EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();1551for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) {1552Vector2i coords = E.key.get_atlas_coords();1553undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/occlusion_layer_%d/polygon", coords.x, coords.y, E.key.alternative_tile, occlusion_layer), E.value);1554undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/occlusion_layer_%d/polygon", coords.x, coords.y, E.key.alternative_tile, occlusion_layer), p_new_value);1555}1556}15571558void TileDataOcclusionShapeEditor::_tile_set_changed() {1559polygon_editor->set_tile_set(tile_set);1560}15611562void TileDataOcclusionShapeEditor::_notification(int p_what) {1563switch (p_what) {1564case NOTIFICATION_ENTER_TREE: {1565polygon_editor->set_polygons_color(get_tree()->get_debug_collisions_color());1566} break;1567}1568}15691570TileDataOcclusionShapeEditor::TileDataOcclusionShapeEditor() {1571polygon_editor = memnew(GenericTilePolygonEditor);1572polygon_editor->set_multiple_polygon_mode(true);1573add_child(polygon_editor);1574}15751576void TileDataCollisionEditor::_property_value_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field) {1577dummy_object->set(p_property, p_value);1578}15791580void TileDataCollisionEditor::_property_selected(const StringName &p_path, int p_focusable) {1581// Deselect all other properties1582for (KeyValue<StringName, EditorProperty *> &editor : property_editors) {1583if (editor.key != p_path) {1584editor.value->deselect();1585}1586}1587}15881589void TileDataCollisionEditor::_polygons_changed() {1590// Update the dummy object properties and their editors.1591for (int i = 0; i < polygon_editor->get_polygon_count(); i++) {1592StringName one_way_property = vformat("polygon_%d_one_way", i);1593StringName one_way_margin_property = vformat("polygon_%d_one_way_margin", i);15941595if (!dummy_object->has_dummy_property(one_way_property)) {1596dummy_object->add_dummy_property(one_way_property);1597dummy_object->set(one_way_property, false);1598}15991600if (!dummy_object->has_dummy_property(one_way_margin_property)) {1601dummy_object->add_dummy_property(one_way_margin_property);1602dummy_object->set(one_way_margin_property, 1.0);1603}16041605if (!property_editors.has(one_way_property)) {1606EditorProperty *one_way_property_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::BOOL, one_way_property, PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT);1607one_way_property_editor->set_object_and_property(dummy_object, one_way_property);1608one_way_property_editor->set_label(one_way_property);1609one_way_property_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1));1610one_way_property_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected));1611one_way_property_editor->set_tooltip_text(one_way_property_editor->get_edited_property());1612one_way_property_editor->update_property();1613add_child(one_way_property_editor);1614property_editors[one_way_property] = one_way_property_editor;1615}16161617if (!property_editors.has(one_way_margin_property)) {1618EditorProperty *one_way_margin_property_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::FLOAT, one_way_margin_property, PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT);1619one_way_margin_property_editor->set_object_and_property(dummy_object, one_way_margin_property);1620one_way_margin_property_editor->set_label(one_way_margin_property);1621one_way_margin_property_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1));1622one_way_margin_property_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected));1623one_way_margin_property_editor->set_tooltip_text(one_way_margin_property_editor->get_edited_property());1624one_way_margin_property_editor->update_property();1625add_child(one_way_margin_property_editor);1626property_editors[one_way_margin_property] = one_way_margin_property_editor;1627}1628}16291630// Remove unneeded properties and their editors.1631for (int i = polygon_editor->get_polygon_count(); dummy_object->has_dummy_property(vformat("polygon_%d_one_way", i)); i++) {1632dummy_object->remove_dummy_property(vformat("polygon_%d_one_way", i));1633}1634for (int i = polygon_editor->get_polygon_count(); dummy_object->has_dummy_property(vformat("polygon_%d_one_way_margin", i)); i++) {1635dummy_object->remove_dummy_property(vformat("polygon_%d_one_way_margin", i));1636}1637for (int i = polygon_editor->get_polygon_count(); property_editors.has(vformat("polygon_%d_one_way", i)); i++) {1638property_editors[vformat("polygon_%d_one_way", i)]->queue_free();1639property_editors.erase(vformat("polygon_%d_one_way", i));1640}1641for (int i = polygon_editor->get_polygon_count(); property_editors.has(vformat("polygon_%d_one_way_margin", i)); i++) {1642property_editors[vformat("polygon_%d_one_way_margin", i)]->queue_free();1643property_editors.erase(vformat("polygon_%d_one_way_margin", i));1644}1645}16461647Variant TileDataCollisionEditor::_get_painted_value() {1648Dictionary dict;1649dict["linear_velocity"] = dummy_object->get("linear_velocity");1650dict["angular_velocity"] = dummy_object->get("angular_velocity");1651Array array;1652for (int i = 0; i < polygon_editor->get_polygon_count(); i++) {1653ERR_FAIL_COND_V(polygon_editor->get_polygon(i).size() < 3, Variant());1654Dictionary polygon_dict;1655polygon_dict["points"] = polygon_editor->get_polygon(i);1656polygon_dict["one_way"] = dummy_object->get(vformat("polygon_%d_one_way", i));1657polygon_dict["one_way_margin"] = dummy_object->get(vformat("polygon_%d_one_way_margin", i));1658array.push_back(polygon_dict);1659}1660dict["polygons"] = array;16611662return dict;1663}16641665void TileDataCollisionEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {1666TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1667ERR_FAIL_NULL(tile_data);16681669polygon_editor->clear_polygons();1670for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) {1671Vector<Vector2> polygon = tile_data->get_collision_polygon_points(physics_layer, i);1672if (polygon.size() >= 3) {1673polygon_editor->add_polygon(polygon);1674}1675}16761677_polygons_changed();1678dummy_object->set("linear_velocity", tile_data->get_constant_linear_velocity(physics_layer));1679dummy_object->set("angular_velocity", tile_data->get_constant_angular_velocity(physics_layer));1680for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) {1681dummy_object->set(vformat("polygon_%d_one_way", i), tile_data->is_collision_polygon_one_way(physics_layer, i));1682dummy_object->set(vformat("polygon_%d_one_way_margin", i), tile_data->get_collision_polygon_one_way_margin(physics_layer, i));1683}1684for (const KeyValue<StringName, EditorProperty *> &E : property_editors) {1685E.value->update_property();1686}16871688polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile);1689}16901691void TileDataCollisionEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) {1692TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1693ERR_FAIL_NULL(tile_data);16941695Dictionary dict = p_value;1696tile_data->set_constant_linear_velocity(physics_layer, dict["linear_velocity"]);1697tile_data->set_constant_angular_velocity(physics_layer, dict["angular_velocity"]);1698Array array = dict["polygons"];1699tile_data->set_collision_polygons_count(physics_layer, array.size());1700for (int i = 0; i < array.size(); i++) {1701Dictionary polygon_dict = array[i];1702tile_data->set_collision_polygon_points(physics_layer, i, polygon_dict["points"]);1703tile_data->set_collision_polygon_one_way(physics_layer, i, polygon_dict["one_way"]);1704tile_data->set_collision_polygon_one_way_margin(physics_layer, i, polygon_dict["one_way_margin"]);1705}17061707polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile);1708}17091710Variant TileDataCollisionEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {1711TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);1712ERR_FAIL_NULL_V(tile_data, Variant());17131714Dictionary dict;1715dict["linear_velocity"] = tile_data->get_constant_linear_velocity(physics_layer);1716dict["angular_velocity"] = tile_data->get_constant_angular_velocity(physics_layer);1717Array array;1718for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) {1719Dictionary polygon_dict;1720polygon_dict["points"] = tile_data->get_collision_polygon_points(physics_layer, i);1721polygon_dict["one_way"] = tile_data->is_collision_polygon_one_way(physics_layer, i);1722polygon_dict["one_way_margin"] = tile_data->get_collision_polygon_one_way_margin(physics_layer, i);1723array.push_back(polygon_dict);1724}1725dict["polygons"] = array;1726return dict;1727}17281729void TileDataCollisionEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) {1730Dictionary new_dict = p_new_value;1731EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();1732for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) {1733Vector2i coords = E.key.get_atlas_coords();17341735Dictionary old_dict = E.value;1736undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/linear_velocity", coords.x, coords.y, E.key.alternative_tile, physics_layer), old_dict["linear_velocity"]);1737undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/angular_velocity", coords.x, coords.y, E.key.alternative_tile, physics_layer), old_dict["angular_velocity"]);1738Array old_polygon_array = old_dict["polygons"];1739undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygons_count", coords.x, coords.y, E.key.alternative_tile, physics_layer), old_polygon_array.size());1740for (int i = 0; i < old_polygon_array.size(); i++) {1741Dictionary polygon_dict = old_polygon_array[i];1742undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygon_%d/points", coords.x, coords.y, E.key.alternative_tile, physics_layer, i), polygon_dict["points"]);1743undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygon_%d/one_way", coords.x, coords.y, E.key.alternative_tile, physics_layer, i), polygon_dict["one_way"]);1744undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygon_%d/one_way_margin", coords.x, coords.y, E.key.alternative_tile, physics_layer, i), polygon_dict["one_way_margin"]);1745}17461747undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/linear_velocity", coords.x, coords.y, E.key.alternative_tile, physics_layer), new_dict["linear_velocity"]);1748undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/angular_velocity", coords.x, coords.y, E.key.alternative_tile, physics_layer), new_dict["angular_velocity"]);1749Array new_polygon_array = new_dict["polygons"];1750undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygons_count", coords.x, coords.y, E.key.alternative_tile, physics_layer), new_polygon_array.size());1751for (int i = 0; i < new_polygon_array.size(); i++) {1752Dictionary polygon_dict = new_polygon_array[i];1753undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygon_%d/points", coords.x, coords.y, E.key.alternative_tile, physics_layer, i), polygon_dict["points"]);1754undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygon_%d/one_way", coords.x, coords.y, E.key.alternative_tile, physics_layer, i), polygon_dict["one_way"]);1755undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/physics_layer_%d/polygon_%d/one_way_margin", coords.x, coords.y, E.key.alternative_tile, physics_layer, i), polygon_dict["one_way_margin"]);1756}1757}1758}17591760void TileDataCollisionEditor::_tile_set_changed() {1761polygon_editor->set_tile_set(tile_set);1762_polygons_changed();1763}17641765void TileDataCollisionEditor::_notification(int p_what) {1766switch (p_what) {1767case NOTIFICATION_ENTER_TREE: {1768polygon_editor->set_polygons_color(get_tree()->get_debug_collisions_color());1769} break;1770}1771}17721773TileDataCollisionEditor::TileDataCollisionEditor() {1774polygon_editor = memnew(GenericTilePolygonEditor);1775polygon_editor->set_multiple_polygon_mode(true);1776polygon_editor->connect("polygons_changed", callable_mp(this, &TileDataCollisionEditor::_polygons_changed));1777add_child(polygon_editor);17781779dummy_object->add_dummy_property("linear_velocity");1780dummy_object->set("linear_velocity", Vector2());1781dummy_object->add_dummy_property("angular_velocity");1782dummy_object->set("angular_velocity", 0.0);17831784EditorProperty *linear_velocity_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::VECTOR2, "linear_velocity", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT);1785linear_velocity_editor->set_object_and_property(dummy_object, "linear_velocity");1786linear_velocity_editor->set_label("linear_velocity");1787linear_velocity_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1));1788linear_velocity_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected));1789linear_velocity_editor->set_tooltip_text(linear_velocity_editor->get_edited_property());1790linear_velocity_editor->update_property();1791add_child(linear_velocity_editor);1792property_editors["linear_velocity"] = linear_velocity_editor;17931794EditorProperty *angular_velocity_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::FLOAT, "angular_velocity", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT);1795angular_velocity_editor->set_object_and_property(dummy_object, "angular_velocity");1796angular_velocity_editor->set_label("angular_velocity");1797angular_velocity_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1));1798angular_velocity_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected));1799angular_velocity_editor->set_tooltip_text(angular_velocity_editor->get_edited_property());1800angular_velocity_editor->update_property();1801add_child(angular_velocity_editor);1802property_editors["angular_velocity"] = angular_velocity_editor;18031804_polygons_changed();1805}18061807TileDataCollisionEditor::~TileDataCollisionEditor() {1808memdelete(dummy_object);1809}18101811void TileDataCollisionEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {1812TileData *tile_data = _get_tile_data(p_cell);1813ERR_FAIL_NULL(tile_data);18141815// Draw all shapes.1816Vector<Color> color;1817if (p_selected) {1818Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");1819Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);1820selection_color.a = 0.7;1821color.push_back(selection_color);1822} else {1823Color debug_collision_color = p_canvas_item->get_tree()->get_debug_collisions_color();1824color.push_back(debug_collision_color);1825}18261827RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform);18281829Ref<Texture2D> one_way_icon = get_editor_theme_icon(SNAME("OneWayTile"));1830for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) {1831Vector<Vector2> polygon = tile_data->get_collision_polygon_points(physics_layer, i);1832if (polygon.size() < 3) {1833continue;1834}18351836p_canvas_item->draw_polygon(polygon, color);18371838if (tile_data->is_collision_polygon_one_way(physics_layer, i)) {1839PackedVector2Array uvs;1840uvs.resize(polygon.size());1841Vector2 size_1 = Vector2(1, 1) / tile_set->get_tile_size();18421843for (int j = 0; j < polygon.size(); j++) {1844uvs.write[j] = polygon[j] * size_1 + Vector2(0.5, 0.5);1845}18461847Vector<Color> color2;1848color2.push_back(Color(1, 1, 1, 0.4));1849p_canvas_item->draw_polygon(polygon, color2, uvs, one_way_icon);1850}1851}18521853RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D());1854}18551856void TileDataTerrainsEditor::_update_terrain_selector() {1857ERR_FAIL_COND(tile_set.is_null());18581859// Update the terrain set selector.1860Vector<String> options;1861options.push_back(String(TTR("No terrains")) + String(":-1"));1862for (int i = 0; i < tile_set->get_terrain_sets_count(); i++) {1863options.push_back(vformat("Terrain Set %d", i));1864}1865terrain_set_property_editor->setup(options);1866terrain_set_property_editor->update_property();18671868// Update the terrain selector.1869int terrain_set = int(dummy_object->get("terrain_set"));1870if (terrain_set == -1) {1871terrain_property_editor->hide();1872} else {1873options.clear();1874Vector<Vector<Ref<Texture2D>>> icons = tile_set->generate_terrains_icons(Size2(16, 16) * EDSCALE);1875options.push_back(String(TTR("No terrain")) + String(":-1"));1876for (int i = 0; i < tile_set->get_terrains_count(terrain_set); i++) {1877String name = tile_set->get_terrain_name(terrain_set, i);1878if (name.is_empty()) {1879options.push_back(vformat("Terrain %d", i));1880} else {1881options.push_back(name);1882}1883}1884terrain_property_editor->setup(options);1885terrain_property_editor->update_property();18861887// Kind of a hack to set icons.1888// We could provide a way to modify that in the EditorProperty.1889OptionButton *option_button = Object::cast_to<OptionButton>(terrain_property_editor->get_child(0));1890for (int terrain = 0; terrain < tile_set->get_terrains_count(terrain_set); terrain++) {1891option_button->set_item_icon(terrain + 1, icons[terrain_set][terrain]);1892}1893terrain_property_editor->show();1894}1895}18961897void TileDataTerrainsEditor::_property_value_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field) {1898Variant old_value = dummy_object->get(p_property);1899dummy_object->set(p_property, p_value);1900if (p_property == "terrain_set") {1901if (p_value != old_value) {1902dummy_object->set("terrain", -1);1903}1904_update_terrain_selector();1905}1906emit_signal(SNAME("needs_redraw"));1907}19081909void TileDataTerrainsEditor::_tile_set_changed() {1910ERR_FAIL_COND(tile_set.is_null());19111912// Fix if wrong values are selected.1913int terrain_set = int(dummy_object->get("terrain_set"));1914if (terrain_set >= tile_set->get_terrain_sets_count()) {1915terrain_set = -1;1916dummy_object->set("terrain_set", -1);1917}1918if (terrain_set >= 0) {1919if (int(dummy_object->get("terrain")) >= tile_set->get_terrains_count(terrain_set)) {1920dummy_object->set("terrain", -1);1921}1922}19231924_update_terrain_selector();1925}19261927void TileDataTerrainsEditor::forward_draw_over_atlas(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform) {1928ERR_FAIL_COND(tile_set.is_null());19291930// Draw the hovered terrain bit, or the whole tile if it has the wrong terrain set.1931Vector2i hovered_coords = TileSetSource::INVALID_ATLAS_COORDS;1932if (drag_type == DRAG_TYPE_NONE) {1933Vector2i mouse_pos = p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position());1934hovered_coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mouse_pos);1935hovered_coords = p_tile_set_atlas_source->get_tile_at_coords(hovered_coords);1936if (hovered_coords != TileSetSource::INVALID_ATLAS_COORDS) {1937TileData *tile_data = p_tile_set_atlas_source->get_tile_data(hovered_coords, 0);1938int terrain_set = tile_data->get_terrain_set();1939Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(hovered_coords);1940Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();19411942if (terrain_set >= 0 && terrain_set == int(dummy_object->get("terrain_set"))) {1943// Draw hovered bit.1944Transform2D xform;1945xform.set_origin(position);19461947Vector<Color> color;1948color.push_back(Color(1.0, 1.0, 1.0, 0.5));19491950Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);1951if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) {1952p_canvas_item->draw_set_transform_matrix(p_transform * xform);1953p_canvas_item->draw_polygon(polygon, color);1954}1955for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {1956TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);1957if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {1958polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);1959if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) {1960p_canvas_item->draw_set_transform_matrix(p_transform * xform);1961p_canvas_item->draw_polygon(polygon, color);1962}1963}1964}1965} else {1966// Draw hovered tile.1967Transform2D tile_xform;1968tile_xform.set_origin(position);1969tile_xform.set_scale(tile_set->get_tile_size());1970tile_set->draw_tile_shape(p_canvas_item, p_transform * tile_xform, Color(1.0, 1.0, 1.0, 0.5), true);1971}1972}1973}19741975// Dim terrains with wrong terrain set.1976Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts));1977int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts));1978for (int i = 0; i < p_tile_set_atlas_source->get_tiles_count(); i++) {1979Vector2i coords = p_tile_set_atlas_source->get_tile_id(i);1980if (coords != hovered_coords) {1981TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);1982if (tile_data->get_terrain_set() != int(dummy_object->get("terrain_set"))) {1983// Dimming1984p_canvas_item->draw_set_transform_matrix(p_transform);1985Rect2i rect = p_tile_set_atlas_source->get_tile_texture_region(coords);1986p_canvas_item->draw_rect(rect, Color(0.0, 0.0, 0.0, 0.3));19871988// Text1989p_canvas_item->draw_set_transform_matrix(Transform2D());1990Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords);1991Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();19921993Color color = Color(1, 1, 1);1994String text;1995if (tile_data->get_terrain_set() >= 0) {1996text = vformat("%d", tile_data->get_terrain_set());1997} else {1998text = "-";1999}2000Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size);2001p_canvas_item->draw_string_outline(font, p_transform.xform(position) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, 1, Color(0, 0, 0, 1));2002p_canvas_item->draw_string(font, p_transform.xform(position) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, color);2003}2004}2005}2006p_canvas_item->draw_set_transform_matrix(Transform2D());20072008if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET_RECT) {2009// Draw selection rectangle.2010Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");2011Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);20122013p_canvas_item->draw_set_transform_matrix(p_transform);20142015Rect2i rect;2016rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true));2017rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position()), true));2018rect = rect.abs();20192020RBSet<TileMapCell> edited;2021for (int x = rect.get_position().x; x <= rect.get_end().x; x++) {2022for (int y = rect.get_position().y; y <= rect.get_end().y; y++) {2023Vector2i coords = Vector2i(x, y);2024coords = p_tile_set_atlas_source->get_tile_at_coords(coords);2025if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2026TileMapCell cell;2027cell.source_id = 0;2028cell.set_atlas_coords(coords);2029cell.alternative_tile = 0;2030edited.insert(cell);2031}2032}2033}20342035for (const TileMapCell &E : edited) {2036Vector2i coords = E.get_atlas_coords();2037p_canvas_item->draw_rect(p_tile_set_atlas_source->get_tile_texture_region(coords), selection_color, false);2038}2039p_canvas_item->draw_set_transform_matrix(Transform2D());2040} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS_RECT) {2041// Highlight selected peering bits.2042Dictionary painted = Dictionary(drag_painted_value);2043int terrain_set = int(painted["terrain_set"]);20442045Rect2i rect;2046rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true));2047rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position()), true));2048rect = rect.abs();20492050RBSet<TileMapCell> edited;2051for (int x = rect.get_position().x; x <= rect.get_end().x; x++) {2052for (int y = rect.get_position().y; y <= rect.get_end().y; y++) {2053Vector2i coords = Vector2i(x, y);2054coords = p_tile_set_atlas_source->get_tile_at_coords(coords);2055if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2056TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);2057if (tile_data->get_terrain_set() == terrain_set) {2058TileMapCell cell;2059cell.source_id = 0;2060cell.set_atlas_coords(coords);2061cell.alternative_tile = 0;2062edited.insert(cell);2063}2064}2065}2066}20672068Vector2 end = p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position());2069Vector<Point2> mouse_pos_rect_polygon = {2070drag_start_pos, Vector2(end.x, drag_start_pos.y),2071end, Vector2(drag_start_pos.x, end.y)2072};20732074Vector<Color> color = { Color(1.0, 1.0, 1.0, 0.5) };20752076p_canvas_item->draw_set_transform_matrix(p_transform);20772078for (const TileMapCell &E : edited) {2079Vector2i coords = E.get_atlas_coords();20802081Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords);2082Vector2i position = texture_region.get_center() + p_tile_set_atlas_source->get_tile_data(coords, 0)->get_texture_origin();20832084Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);2085for (int j = 0; j < polygon.size(); j++) {2086polygon.write[j] += position;2087}2088if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) {2089// Draw terrain.2090p_canvas_item->draw_polygon(polygon, color);2091}20922093for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2094TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2095if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2096polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);2097for (int j = 0; j < polygon.size(); j++) {2098polygon.write[j] += position;2099}2100if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) {2101// Draw bit.2102p_canvas_item->draw_polygon(polygon, color);2103}2104}2105}2106}21072108p_canvas_item->draw_set_transform_matrix(Transform2D());2109}2110}21112112void TileDataTerrainsEditor::forward_draw_over_alternatives(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform) {2113ERR_FAIL_COND(tile_set.is_null());21142115// Draw the hovered terrain bit, or the whole tile if it has the wrong terrain set.2116Vector2i hovered_coords = TileSetSource::INVALID_ATLAS_COORDS;2117int hovered_alternative = TileSetSource::INVALID_TILE_ALTERNATIVE;2118if (drag_type == DRAG_TYPE_NONE) {2119Vector2i mouse_pos = p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position());2120Vector3i hovered = p_tile_atlas_view->get_alternative_tile_at_pos(mouse_pos);2121hovered_coords = Vector2i(hovered.x, hovered.y);2122hovered_alternative = hovered.z;2123if (hovered_coords != TileSetSource::INVALID_ATLAS_COORDS) {2124TileData *tile_data = p_tile_set_atlas_source->get_tile_data(hovered_coords, hovered_alternative);2125int terrain_set = tile_data->get_terrain_set();2126Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(hovered_coords, hovered_alternative);2127Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();21282129if (terrain_set == int(dummy_object->get("terrain_set"))) {2130// Draw hovered bit.2131Transform2D xform;2132xform.set_origin(position);21332134Vector<Color> color = { Color(1.0, 1.0, 1.0, 0.5) };21352136Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);2137if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) {2138p_canvas_item->draw_set_transform_matrix(p_transform * xform);2139p_canvas_item->draw_polygon(polygon, color);2140}21412142for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2143TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2144if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2145polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);2146if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) {2147p_canvas_item->draw_set_transform_matrix(p_transform * xform);2148p_canvas_item->draw_polygon(polygon, color);2149}2150}2151}2152} else {2153// Draw hovered tile.2154Transform2D tile_xform;2155tile_xform.set_origin(position);2156tile_xform.set_scale(tile_set->get_tile_size());2157tile_set->draw_tile_shape(p_canvas_item, p_transform * tile_xform, Color(1.0, 1.0, 1.0, 0.5), true);2158}2159}2160}21612162// Dim terrains with wrong terrain set.2163Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts));2164int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts));2165for (int i = 0; i < p_tile_set_atlas_source->get_tiles_count(); i++) {2166Vector2i coords = p_tile_set_atlas_source->get_tile_id(i);2167for (int j = 1; j < p_tile_set_atlas_source->get_alternative_tiles_count(coords); j++) {2168int alternative_tile = p_tile_set_atlas_source->get_alternative_tile_id(coords, j);2169if (coords != hovered_coords || alternative_tile != hovered_alternative) {2170TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile);2171if (tile_data->get_terrain_set() != int(dummy_object->get("terrain_set"))) {2172// Dimming2173p_canvas_item->draw_set_transform_matrix(p_transform);2174Rect2i rect = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile);2175p_canvas_item->draw_rect(rect, Color(0.0, 0.0, 0.0, 0.3));21762177// Text2178p_canvas_item->draw_set_transform_matrix(Transform2D());2179Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile);2180Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();21812182Color color = Color(1, 1, 1);2183String text;2184if (tile_data->get_terrain_set() >= 0) {2185text = vformat("%d", tile_data->get_terrain_set());2186} else {2187text = "-";2188}2189Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size);2190p_canvas_item->draw_string_outline(font, p_transform.xform(position) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, 1, Color(0, 0, 0, 1));2191p_canvas_item->draw_string(font, p_transform.xform(position) + Vector2i(-string_size.x / 2, string_size.y / 2), text, HORIZONTAL_ALIGNMENT_CENTER, string_size.x, font_size, color);2192}2193}2194}2195}21962197p_canvas_item->draw_set_transform_matrix(Transform2D());2198}21992200void TileDataTerrainsEditor::forward_painting_atlas_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) {2201Ref<InputEventMouseMotion> mm = p_event;2202if (mm.is_valid()) {2203if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) {2204Vector<Vector2i> line = Geometry2D::bresenham_line(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_last_pos, true), p_tile_atlas_view->get_atlas_tile_coords_at_pos(mm->get_position(), true));2205for (int i = 0; i < line.size(); i++) {2206Vector2i coords = p_tile_set_atlas_source->get_tile_at_coords(line[i]);2207if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2208int terrain_set = drag_painted_value;2209TileMapCell cell;2210cell.source_id = 0;2211cell.set_atlas_coords(coords);2212cell.alternative_tile = 0;22132214// Save the old terrain_set and terrains bits.2215TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);2216if (!drag_modified.has(cell)) {2217Dictionary dict;2218dict["terrain_set"] = tile_data->get_terrain_set();2219dict["terrain"] = tile_data->get_terrain();2220Array array;2221for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) {2222TileSet::CellNeighbor bit = TileSet::CellNeighbor(j);2223array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2224}2225dict["terrain_peering_bits"] = array;2226drag_modified[cell] = dict;2227}22282229// Set the terrain_set.2230tile_data->set_terrain_set(terrain_set);2231}2232}2233drag_last_pos = mm->get_position();2234accept_event();2235} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) {2236int terrain_set = Dictionary(drag_painted_value)["terrain_set"];2237int terrain = Dictionary(drag_painted_value)["terrain"];2238Vector<Vector2i> line = Geometry2D::bresenham_line(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_last_pos, true), p_tile_atlas_view->get_atlas_tile_coords_at_pos(mm->get_position(), true));2239for (int i = 0; i < line.size(); i++) {2240Vector2i coords = p_tile_set_atlas_source->get_tile_at_coords(line[i]);2241if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2242TileMapCell cell;2243cell.source_id = 0;2244cell.set_atlas_coords(coords);2245cell.alternative_tile = 0;22462247TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);2248if (tile_data->get_terrain_set() == terrain_set) {2249// Save the old terrain_set and terrains bits.2250if (!drag_modified.has(cell)) {2251Dictionary dict;2252dict["terrain_set"] = tile_data->get_terrain_set();2253dict["terrain"] = tile_data->get_terrain();2254Array array;2255for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) {2256TileSet::CellNeighbor bit = TileSet::CellNeighbor(j);2257array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2258}2259dict["terrain_peering_bits"] = array;2260drag_modified[cell] = dict;2261}22622263// Set the terrains bits.2264Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords);2265Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();22662267Vector<Vector2> polygon = tile_set->get_terrain_polygon(tile_data->get_terrain_set());2268if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) {2269tile_data->set_terrain(terrain);2270}2271for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) {2272TileSet::CellNeighbor bit = TileSet::CellNeighbor(j);2273if (tile_data->is_valid_terrain_peering_bit(bit)) {2274polygon = tile_set->get_terrain_peering_bit_polygon(tile_data->get_terrain_set(), bit);2275if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) {2276tile_data->set_terrain_peering_bit(bit, terrain);2277}2278}2279}2280}2281}2282}2283drag_last_pos = mm->get_position();2284accept_event();2285}2286}22872288Ref<InputEventMouseButton> mb = p_event;2289if (mb.is_valid()) {2290if (mb->get_button_index() == MouseButton::LEFT || mb->get_button_index() == MouseButton::RIGHT) {2291if (mb->is_pressed()) {2292if (picker_button->is_pressed() || (mb->is_command_or_control_pressed() && !mb->is_shift_pressed())) {2293Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position());2294coords = p_tile_set_atlas_source->get_tile_at_coords(coords);2295if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2296TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);2297int terrain_set = tile_data->get_terrain_set();2298Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords);2299Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();2300dummy_object->set("terrain_set", terrain_set);2301dummy_object->set("terrain", -1);23022303Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);2304if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2305dummy_object->set("terrain", tile_data->get_terrain());2306}2307for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2308TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2309if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2310polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);2311if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2312dummy_object->set("terrain", tile_data->get_terrain_peering_bit(bit));2313}2314}2315}2316terrain_set_property_editor->update_property();2317_update_terrain_selector();2318picker_button->set_pressed(false);2319accept_event();2320}2321} else {2322Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position());2323coords = p_tile_set_atlas_source->get_tile_at_coords(coords);2324TileData *tile_data = nullptr;2325if (coords != TileSetAtlasSource::INVALID_ATLAS_COORDS) {2326tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);2327}2328int terrain_set = int(dummy_object->get("terrain_set"));2329int terrain = int(dummy_object->get("terrain"));2330if (terrain_set == -1 || !tile_data || tile_data->get_terrain_set() != terrain_set) {2331// Paint terrain sets.2332if (mb->get_button_index() == MouseButton::RIGHT) {2333terrain_set = -1;2334}2335if (mb->is_command_or_control_pressed() && mb->is_shift_pressed()) {2336// Paint terrain set with rect.2337drag_type = DRAG_TYPE_PAINT_TERRAIN_SET_RECT;2338drag_modified.clear();2339drag_painted_value = terrain_set;2340drag_start_pos = mb->get_position();2341} else {2342// Paint terrain set.2343drag_type = DRAG_TYPE_PAINT_TERRAIN_SET;2344drag_modified.clear();2345drag_painted_value = terrain_set;23462347if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2348TileMapCell cell;2349cell.source_id = 0;2350cell.set_atlas_coords(coords);2351cell.alternative_tile = 0;23522353// Save the old terrain_set and terrains bits.2354Dictionary dict;2355dict["terrain_set"] = tile_data->get_terrain_set();2356dict["terrain"] = tile_data->get_terrain();2357Array array;2358for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2359TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2360array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2361}2362dict["terrain_peering_bits"] = array;2363drag_modified[cell] = dict;23642365// Set the terrain_set.2366tile_data->set_terrain_set(terrain_set);2367}2368drag_last_pos = mb->get_position();2369}2370accept_event();2371} else if (tile_data->get_terrain_set() == terrain_set) {2372// Paint terrain bits.2373if (mb->get_button_index() == MouseButton::RIGHT) {2374terrain = -1;2375}2376if (mb->is_command_or_control_pressed() && mb->is_shift_pressed()) {2377// Paint terrain bits with rect.2378drag_type = DRAG_TYPE_PAINT_TERRAIN_BITS_RECT;2379drag_modified.clear();2380Dictionary painted_dict;2381painted_dict["terrain_set"] = terrain_set;2382painted_dict["terrain"] = terrain;2383drag_painted_value = painted_dict;2384drag_start_pos = mb->get_position();2385} else {2386// Paint terrain bits.2387drag_type = DRAG_TYPE_PAINT_TERRAIN_BITS;2388drag_modified.clear();2389Dictionary painted_dict;2390painted_dict["terrain_set"] = terrain_set;2391painted_dict["terrain"] = terrain;2392drag_painted_value = painted_dict;23932394if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2395TileMapCell cell;2396cell.source_id = 0;2397cell.set_atlas_coords(coords);2398cell.alternative_tile = 0;23992400// Save the old terrain_set and terrains bits.2401Dictionary dict;2402dict["terrain_set"] = tile_data->get_terrain_set();2403dict["terrain"] = tile_data->get_terrain();2404Array array;2405for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2406TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2407array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2408}2409dict["terrain_peering_bits"] = array;2410drag_modified[cell] = dict;24112412// Set the terrain bit.2413Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords);2414Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();24152416Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);2417if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2418tile_data->set_terrain(terrain);2419}2420for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2421TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2422if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2423polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);2424if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2425tile_data->set_terrain_peering_bit(bit, terrain);2426}2427}2428}2429}2430drag_last_pos = mb->get_position();2431}2432accept_event();2433}2434}2435} else {2436EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();2437if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET_RECT) {2438Rect2i rect;2439rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true));2440rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true));2441rect = rect.abs();24422443RBSet<TileMapCell> edited;2444for (int x = rect.get_position().x; x <= rect.get_end().x; x++) {2445for (int y = rect.get_position().y; y <= rect.get_end().y; y++) {2446Vector2i coords = Vector2i(x, y);2447coords = p_tile_set_atlas_source->get_tile_at_coords(coords);2448if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2449TileMapCell cell;2450cell.source_id = 0;2451cell.set_atlas_coords(coords);2452cell.alternative_tile = 0;2453edited.insert(cell);2454}2455}2456}2457undo_redo->create_action(TTR("Painting Terrain Set"));2458for (const TileMapCell &E : edited) {2459Vector2i coords = E.get_atlas_coords();2460TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);2461undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain_set", coords.x, coords.y, E.alternative_tile), drag_painted_value);2462undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain_set", coords.x, coords.y, E.alternative_tile), tile_data->get_terrain_set());2463if (tile_data->get_terrain_set() >= 0) {2464undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.alternative_tile), tile_data->get_terrain());2465for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2466TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2467if (tile_data->is_valid_terrain_peering_bit(bit)) {2468undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.alternative_tile), tile_data->get_terrain_peering_bit(bit));2469}2470}2471}2472}2473undo_redo->commit_action(true);2474drag_type = DRAG_TYPE_NONE;2475accept_event();2476} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) {2477undo_redo->create_action(TTR("Painting Terrain Set"));2478for (KeyValue<TileMapCell, Variant> &E : drag_modified) {2479Dictionary dict = E.value;2480Vector2i coords = E.key.get_atlas_coords();2481undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain_set", coords.x, coords.y, E.key.alternative_tile), drag_painted_value);2482undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain_set", coords.x, coords.y, E.key.alternative_tile), dict["terrain_set"]);2483if (int(dict["terrain_set"]) >= 0) {2484undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]);2485Array array = dict["terrain_peering_bits"];2486for (int i = 0; i < array.size(); i++) {2487TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2488if (tile_set->is_valid_terrain_peering_bit(dict["terrain_set"], bit)) {2489undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.key.alternative_tile), array[i]);2490}2491}2492}2493}2494undo_redo->commit_action(false);2495drag_type = DRAG_TYPE_NONE;2496accept_event();2497} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) {2498Dictionary painted = Dictionary(drag_painted_value);2499int terrain_set = int(painted["terrain_set"]);2500int terrain = int(painted["terrain"]);2501undo_redo->create_action(TTR("Painting Terrain"));2502for (KeyValue<TileMapCell, Variant> &E : drag_modified) {2503Dictionary dict = E.value;2504Vector2i coords = E.key.get_atlas_coords();2505undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), terrain);2506undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]);2507Array array = dict["terrain_peering_bits"];2508for (int i = 0; i < array.size(); i++) {2509TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2510if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2511undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.key.alternative_tile), terrain);2512}2513if (tile_set->is_valid_terrain_peering_bit(dict["terrain_set"], bit)) {2514undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.key.alternative_tile), array[i]);2515}2516}2517}2518undo_redo->commit_action(false);2519drag_type = DRAG_TYPE_NONE;2520accept_event();2521} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS_RECT) {2522Dictionary painted = Dictionary(drag_painted_value);2523int terrain_set = int(painted["terrain_set"]);2524int terrain = int(painted["terrain"]);25252526Rect2i rect;2527rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true));2528rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true));2529rect = rect.abs();25302531RBSet<TileMapCell> edited;2532for (int x = rect.get_position().x; x <= rect.get_end().x; x++) {2533for (int y = rect.get_position().y; y <= rect.get_end().y; y++) {2534Vector2i coords = Vector2i(x, y);2535coords = p_tile_set_atlas_source->get_tile_at_coords(coords);2536if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2537TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);2538if (tile_data->get_terrain_set() == terrain_set) {2539TileMapCell cell;2540cell.source_id = 0;2541cell.set_atlas_coords(coords);2542cell.alternative_tile = 0;2543edited.insert(cell);2544}2545}2546}2547}25482549Vector<Point2> mouse_pos_rect_polygon = {2550drag_start_pos, Vector2(mb->get_position().x, drag_start_pos.y),2551mb->get_position(), Vector2(drag_start_pos.x, mb->get_position().y)2552};25532554undo_redo->create_action(TTR("Painting Terrain"));2555for (const TileMapCell &E : edited) {2556Vector2i coords = E.get_atlas_coords();2557TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0);25582559Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords);2560Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();25612562Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);2563for (int j = 0; j < polygon.size(); j++) {2564polygon.write[j] += position;2565}2566if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) {2567// Draw terrain.2568undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.alternative_tile), terrain);2569undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.alternative_tile), tile_data->get_terrain());2570}25712572for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2573TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2574if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2575polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);2576for (int j = 0; j < polygon.size(); j++) {2577polygon.write[j] += position;2578}2579if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) {2580// Draw bit.2581undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.alternative_tile), terrain);2582undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.alternative_tile), tile_data->get_terrain_peering_bit(bit));2583}2584}2585}2586}2587undo_redo->commit_action(true);2588drag_type = DRAG_TYPE_NONE;2589accept_event();2590}2591}2592}2593}2594}25952596void TileDataTerrainsEditor::forward_painting_alternatives_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) {2597Ref<InputEventMouseMotion> mm = p_event;2598if (mm.is_valid()) {2599if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) {2600Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mm->get_position());2601Vector2i coords = Vector2i(tile.x, tile.y);2602int alternative_tile = tile.z;26032604if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2605TileMapCell cell;2606cell.source_id = 0;2607cell.set_atlas_coords(coords);2608cell.alternative_tile = alternative_tile;2609TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile);2610if (!drag_modified.has(cell)) {2611Dictionary dict;2612dict["terrain_set"] = tile_data->get_terrain_set();2613dict["terrain"] = tile_data->get_terrain();2614Array array;2615for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) {2616TileSet::CellNeighbor bit = TileSet::CellNeighbor(j);2617array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2618}2619dict["terrain_peering_bits"] = array;2620drag_modified[cell] = dict;2621}2622tile_data->set_terrain_set(drag_painted_value);2623}26242625drag_last_pos = mm->get_position();2626accept_event();2627} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) {2628Dictionary painted = Dictionary(drag_painted_value);2629int terrain_set = int(painted["terrain_set"]);2630int terrain = int(painted["terrain"]);26312632Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mm->get_position());2633Vector2i coords = Vector2i(tile.x, tile.y);2634int alternative_tile = tile.z;26352636if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2637TileMapCell cell;2638cell.source_id = 0;2639cell.set_atlas_coords(coords);2640cell.alternative_tile = alternative_tile;26412642// Save the old terrain_set and terrains bits.2643TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile);2644if (tile_data->get_terrain_set() == terrain_set) {2645if (!drag_modified.has(cell)) {2646Dictionary dict;2647dict["terrain_set"] = tile_data->get_terrain_set();2648dict["terrain"] = tile_data->get_terrain();2649Array array;2650for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) {2651TileSet::CellNeighbor bit = TileSet::CellNeighbor(j);2652array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2653}2654dict["terrain_peering_bits"] = array;2655drag_modified[cell] = dict;2656}26572658// Set the terrains bits.2659Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile);2660Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();26612662Vector<Vector2> polygon = tile_set->get_terrain_polygon(tile_data->get_terrain_set());2663if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) {2664tile_data->set_terrain(terrain);2665}26662667for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) {2668TileSet::CellNeighbor bit = TileSet::CellNeighbor(j);2669if (tile_data->is_valid_terrain_peering_bit(bit)) {2670polygon = tile_set->get_terrain_peering_bit_polygon(tile_data->get_terrain_set(), bit);2671if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) {2672tile_data->set_terrain_peering_bit(bit, terrain);2673}2674}2675}2676}2677}2678drag_last_pos = mm->get_position();2679accept_event();2680}2681}26822683Ref<InputEventMouseButton> mb = p_event;2684if (mb.is_valid()) {2685if (mb->get_button_index() == MouseButton::LEFT || mb->get_button_index() == MouseButton::RIGHT) {2686if (mb->is_pressed()) {2687if (mb->get_button_index() == MouseButton::LEFT && picker_button->is_pressed()) {2688Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position());2689Vector2i coords = Vector2i(tile.x, tile.y);2690int alternative_tile = tile.z;26912692if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2693TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile);2694int terrain_set = tile_data->get_terrain_set();2695Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile);2696Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();2697dummy_object->set("terrain_set", terrain_set);2698dummy_object->set("terrain", -1);26992700Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);2701if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2702dummy_object->set("terrain", tile_data->get_terrain());2703}27042705for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2706TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2707if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2708polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);2709if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2710dummy_object->set("terrain", tile_data->get_terrain_peering_bit(bit));2711}2712}2713}2714terrain_set_property_editor->update_property();2715_update_terrain_selector();2716picker_button->set_pressed(false);2717accept_event();2718}2719} else {2720int terrain_set = int(dummy_object->get("terrain_set"));2721int terrain = int(dummy_object->get("terrain"));27222723Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position());2724Vector2i coords = Vector2i(tile.x, tile.y);2725int alternative_tile = tile.z;27262727if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2728TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile);27292730if (terrain_set == -1 || !tile_data || tile_data->get_terrain_set() != terrain_set) {2731// Paint terrain sets.2732drag_type = DRAG_TYPE_PAINT_TERRAIN_SET;2733drag_modified.clear();2734drag_painted_value = int(dummy_object->get("terrain_set"));2735if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2736TileMapCell cell;2737cell.source_id = 0;2738cell.set_atlas_coords(coords);2739cell.alternative_tile = alternative_tile;2740Dictionary dict;2741dict["terrain_set"] = tile_data->get_terrain_set();2742Array array;2743for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2744TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2745array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2746}2747dict["terrain_peering_bits"] = array;2748drag_modified[cell] = dict;2749tile_data->set_terrain_set(drag_painted_value);2750}2751drag_last_pos = mb->get_position();2752accept_event();2753} else if (tile_data->get_terrain_set() == terrain_set) {2754// Paint terrain bits.2755if (mb->get_button_index() == MouseButton::RIGHT) {2756terrain = -1;2757}2758// Paint terrain bits.2759drag_type = DRAG_TYPE_PAINT_TERRAIN_BITS;2760drag_modified.clear();2761Dictionary painted_dict;2762painted_dict["terrain_set"] = terrain_set;2763painted_dict["terrain"] = terrain;2764drag_painted_value = painted_dict;27652766if (coords != TileSetSource::INVALID_ATLAS_COORDS) {2767TileMapCell cell;2768cell.source_id = 0;2769cell.set_atlas_coords(coords);2770cell.alternative_tile = alternative_tile;27712772// Save the old terrain_set and terrains bits.2773Dictionary dict;2774dict["terrain_set"] = tile_data->get_terrain_set();2775dict["terrain"] = tile_data->get_terrain();2776Array array;2777for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2778TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2779array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1);2780}2781dict["terrain_peering_bits"] = array;2782drag_modified[cell] = dict;27832784// Set the terrain bit.2785Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile);2786Vector2i position = texture_region.get_center() + tile_data->get_texture_origin();27872788Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set);2789if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2790tile_data->set_terrain(terrain);2791}2792for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) {2793TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2794if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2795polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit);2796if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) {2797tile_data->set_terrain_peering_bit(bit, terrain);2798}2799}2800}2801}2802drag_last_pos = mb->get_position();2803accept_event();2804}2805}2806}2807} else {2808EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();2809if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) {2810undo_redo->create_action(TTR("Painting Tiles Property"));2811for (KeyValue<TileMapCell, Variant> &E : drag_modified) {2812Dictionary dict = E.value;2813Vector2i coords = E.key.get_atlas_coords();2814undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain_set", coords.x, coords.y, E.key.alternative_tile), drag_painted_value);2815undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain_set", coords.x, coords.y, E.key.alternative_tile), dict["terrain_set"]);2816if (int(dict["terrain_set"]) >= 0) {2817undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]);2818Array array = dict["terrain_peering_bits"];2819for (int i = 0; i < array.size(); i++) {2820undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.key.alternative_tile), array[i]);2821}2822}2823}2824undo_redo->commit_action(false);2825drag_type = DRAG_TYPE_NONE;2826accept_event();2827} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) {2828Dictionary painted = Dictionary(drag_painted_value);2829int terrain_set = int(painted["terrain_set"]);2830int terrain = int(painted["terrain"]);2831undo_redo->create_action(TTR("Painting Terrain"));2832for (KeyValue<TileMapCell, Variant> &E : drag_modified) {2833Dictionary dict = E.value;2834Vector2i coords = E.key.get_atlas_coords();2835undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), terrain);2836undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]);2837Array array = dict["terrain_peering_bits"];2838for (int i = 0; i < array.size(); i++) {2839TileSet::CellNeighbor bit = TileSet::CellNeighbor(i);2840if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) {2841undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.key.alternative_tile), terrain);2842}2843if (tile_set->is_valid_terrain_peering_bit(dict["terrain_set"], bit)) {2844undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrains_peering_bit/" + String(TileSet::CELL_NEIGHBOR_ENUM_TO_TEXT[i]), coords.x, coords.y, E.key.alternative_tile), array[i]);2845}2846}2847}2848undo_redo->commit_action(false);2849drag_type = DRAG_TYPE_NONE;2850accept_event();2851}2852}2853}2854}2855}28562857void TileDataTerrainsEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {2858TileData *tile_data = _get_tile_data(p_cell);2859ERR_FAIL_NULL(tile_data);28602861tile_set->draw_terrains(p_canvas_item, p_transform, tile_data);2862}28632864void TileDataTerrainsEditor::_notification(int p_what) {2865switch (p_what) {2866case NOTIFICATION_THEME_CHANGED: {2867picker_button->set_button_icon(get_editor_theme_icon(SNAME("ColorPick")));2868} break;2869}2870}28712872TileDataTerrainsEditor::TileDataTerrainsEditor() {2873label = memnew(Label);2874label->set_text(TTR("Painting:"));2875label->set_theme_type_variation("HeaderSmall");2876add_child(label);28772878// Toolbar2879picker_button = memnew(Button);2880picker_button->set_theme_type_variation(SceneStringName(FlatButton));2881picker_button->set_toggle_mode(true);2882picker_button->set_shortcut(ED_GET_SHORTCUT("tiles_editor/picker"));2883picker_button->set_accessibility_name(TTRC("Pick"));2884toolbar->add_child(picker_button);28852886// Setup2887dummy_object->add_dummy_property("terrain_set");2888dummy_object->set("terrain_set", -1);2889dummy_object->add_dummy_property("terrain");2890dummy_object->set("terrain", -1);28912892// Get the default value for the type.2893terrain_set_property_editor = memnew(EditorPropertyEnum);2894terrain_set_property_editor->set_object_and_property(dummy_object, "terrain_set");2895terrain_set_property_editor->set_label("Terrain Set");2896terrain_set_property_editor->connect("property_changed", callable_mp(this, &TileDataTerrainsEditor::_property_value_changed).unbind(1));2897terrain_set_property_editor->set_tooltip_text(terrain_set_property_editor->get_edited_property());2898add_child(terrain_set_property_editor);28992900terrain_property_editor = memnew(EditorPropertyEnum);2901terrain_property_editor->set_object_and_property(dummy_object, "terrain");2902terrain_property_editor->set_label("Terrain");2903terrain_property_editor->connect("property_changed", callable_mp(this, &TileDataTerrainsEditor::_property_value_changed).unbind(1));2904add_child(terrain_property_editor);2905}29062907TileDataTerrainsEditor::~TileDataTerrainsEditor() {2908toolbar->queue_free();2909memdelete(dummy_object);2910}29112912Variant TileDataNavigationEditor::_get_painted_value() {2913Ref<NavigationPolygon> nav_polygon;2914nav_polygon.instantiate();29152916if (polygon_editor->get_polygon_count() > 0) {2917Ref<NavigationMeshSourceGeometryData2D> source_geometry_data;2918source_geometry_data.instantiate();2919for (int i = 0; i < polygon_editor->get_polygon_count(); i++) {2920Vector<Vector2> polygon = polygon_editor->get_polygon(i);2921nav_polygon->add_outline(polygon);2922source_geometry_data->add_traversable_outline(polygon);2923}2924nav_polygon->set_agent_radius(0.0);2925NavigationServer2D::get_singleton()->bake_from_source_geometry_data(nav_polygon, source_geometry_data);2926} else {2927nav_polygon->clear();2928}29292930return nav_polygon;2931}29322933void TileDataNavigationEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {2934TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);2935ERR_FAIL_NULL(tile_data);29362937Ref<NavigationPolygon> nav_polygon = tile_data->get_navigation_polygon(navigation_layer);2938polygon_editor->clear_polygons();2939if (nav_polygon.is_valid()) {2940for (int i = 0; i < nav_polygon->get_outline_count(); i++) {2941polygon_editor->add_polygon(nav_polygon->get_outline(i));2942}2943}2944polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile);2945}29462947void TileDataNavigationEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) {2948TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);2949ERR_FAIL_NULL(tile_data);2950Ref<NavigationPolygon> nav_polygon = p_value;2951tile_data->set_navigation_polygon(navigation_layer, nav_polygon);29522953polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile);2954}29552956Variant TileDataNavigationEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) {2957TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile);2958ERR_FAIL_NULL_V(tile_data, Variant());2959return tile_data->get_navigation_polygon(navigation_layer);2960}29612962void TileDataNavigationEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) {2963EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton();2964for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) {2965Vector2i coords = E.key.get_atlas_coords();2966undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/navigation_layer_%d/polygon", coords.x, coords.y, E.key.alternative_tile, navigation_layer), E.value);2967undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/navigation_layer_%d/polygon", coords.x, coords.y, E.key.alternative_tile, navigation_layer), p_new_value);2968}2969}29702971void TileDataNavigationEditor::_tile_set_changed() {2972polygon_editor->set_tile_set(tile_set);2973}29742975void TileDataNavigationEditor::_notification(int p_what) {2976switch (p_what) {2977case NOTIFICATION_ENTER_TREE: {2978#ifdef DEBUG_ENABLED2979polygon_editor->set_polygons_color(NavigationServer2D::get_singleton()->get_debug_navigation_geometry_face_color());2980#endif // DEBUG_ENABLED2981} break;2982}2983}29842985TileDataNavigationEditor::TileDataNavigationEditor() {2986polygon_editor = memnew(GenericTilePolygonEditor);2987polygon_editor->set_multiple_polygon_mode(true);2988add_child(polygon_editor);2989}29902991void TileDataNavigationEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) {2992TileData *tile_data = _get_tile_data(p_cell);2993ERR_FAIL_NULL(tile_data);29942995// Draw all shapes.2996RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform);29972998Ref<NavigationPolygon> nav_polygon = tile_data->get_navigation_polygon(navigation_layer);2999if (nav_polygon.is_valid()) {3000Vector<Vector2> verts = nav_polygon->get_vertices();3001if (verts.size() < 3) {3002return;3003}30043005Color color = Color(0.5, 1.0, 1.0, 1.0);3006#ifdef DEBUG_ENABLED3007color = NavigationServer2D::get_singleton()->get_debug_navigation_geometry_face_color();3008#endif // DEBUG_ENABLED3009if (p_selected) {3010Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color");3011Color selection_color = Color().from_hsv(Math::fposmod(grid_color.get_h() + 0.5, 1.0), grid_color.get_s(), grid_color.get_v(), 1.0);3012selection_color.a = 0.7;3013color = selection_color;3014}30153016RandomPCG rand;3017for (int i = 0; i < nav_polygon->get_polygon_count(); i++) {3018// An array of vertices for this polygon.3019Vector<int> polygon = nav_polygon->get_polygon(i);3020Vector<Vector2> vertices;3021vertices.resize(polygon.size());3022for (int j = 0; j < polygon.size(); j++) {3023ERR_FAIL_INDEX(polygon[j], verts.size());3024vertices.write[j] = verts[polygon[j]];3025}30263027// Generate the polygon color, slightly randomly modified from the settings one.3028Color random_variation_color;3029random_variation_color.set_hsv(color.get_h() + rand.random(-1.0, 1.0) * 0.05, color.get_s(), color.get_v() + rand.random(-1.0, 1.0) * 0.1);3030random_variation_color.a = color.a;3031Vector<Color> colors = { random_variation_color };30323033RenderingServer::get_singleton()->canvas_item_add_polygon(p_canvas_item->get_canvas_item(), vertices, colors);3034}3035}30363037RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D());3038}303930403041