| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*  tile_data_editors.cpp                                                 */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*                         This file is part of:                          */ | 
					
						
							|  |  |  | /*                             GODOT ENGINE                               */ | 
					
						
							|  |  |  | /*                        https://godotengine.org                         */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ | 
					
						
							|  |  |  | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* Permission is hereby granted, free of charge, to any person obtaining  */ | 
					
						
							|  |  |  | /* a copy of this software and associated documentation files (the        */ | 
					
						
							|  |  |  | /* "Software"), to deal in the Software without restriction, including    */ | 
					
						
							|  |  |  | /* without limitation the rights to use, copy, modify, merge, publish,    */ | 
					
						
							|  |  |  | /* distribute, sublicense, and/or sell copies of the Software, and to     */ | 
					
						
							|  |  |  | /* permit persons to whom the Software is furnished to do so, subject to  */ | 
					
						
							|  |  |  | /* the following conditions:                                              */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* The above copyright notice and this permission notice shall be         */ | 
					
						
							|  |  |  | /* included in all copies or substantial portions of the Software.        */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,        */ | 
					
						
							|  |  |  | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF     */ | 
					
						
							|  |  |  | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ | 
					
						
							|  |  |  | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY   */ | 
					
						
							|  |  |  | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,   */ | 
					
						
							|  |  |  | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE      */ | 
					
						
							|  |  |  | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                 */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "tile_data_editors.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "tile_set_editor.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | #include "core/math/geometry_2d.h"
 | 
					
						
							|  |  |  | #include "core/os/keyboard.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-12 02:46:22 +01:00
										 |  |  | #include "editor/editor_node.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | #include "editor/editor_properties.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-19 12:45:49 +01:00
										 |  |  | #include "editor/editor_settings.h"
 | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | #include "editor/editor_string_names.h"
 | 
					
						
							| 
									
										
										
										
											2022-03-25 18:06:46 +01:00
										 |  |  | #include "editor/editor_undo_redo_manager.h"
 | 
					
						
							| 
									
										
										
										
											2024-01-15 13:14:55 +01:00
										 |  |  | #include "editor/themes/editor_scale.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | #include "scene/gui/control.h"
 | 
					
						
							|  |  |  | #include "scene/gui/label.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-19 12:45:49 +01:00
										 |  |  | #include "scene/gui/menu_button.h"
 | 
					
						
							|  |  |  | #include "scene/gui/option_button.h"
 | 
					
						
							|  |  |  | #include "scene/gui/separator.h"
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | #include "scene/gui/spin_box.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-19 12:45:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-28 02:36:15 +02:00
										 |  |  | #include "servers/navigation_server_2d.h"
 | 
					
						
							| 
									
										
										
										
											2022-08-04 13:40:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-13 14:49:28 +02:00
										 |  |  | void TileDataEditor::_tile_set_changed_plan_update() { | 
					
						
							|  |  |  | 	_tile_set_changed_update_needed = true; | 
					
						
							| 
									
										
										
										
											2023-12-18 15:46:56 +01:00
										 |  |  | 	callable_mp(this, &TileDataEditor::_tile_set_changed_deferred_update).call_deferred(); | 
					
						
							| 
									
										
										
										
											2021-10-13 14:49:28 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataEditor::_tile_set_changed_deferred_update() { | 
					
						
							|  |  |  | 	if (_tile_set_changed_update_needed) { | 
					
						
							|  |  |  | 		_tile_set_changed(); | 
					
						
							|  |  |  | 		_tile_set_changed_update_needed = false; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileData *TileDataEditor::_get_tile_data(TileMapCell p_cell) { | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!tile_set.is_valid(), nullptr); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!tile_set->has_source(p_cell.source_id), nullptr); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	TileData *td = nullptr; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	TileSetSource *source = *tile_set->get_source(p_cell.source_id); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	TileSetAtlasSource *atlas_source = Object::cast_to<TileSetAtlasSource>(source); | 
					
						
							|  |  |  | 	if (atlas_source) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		ERR_FAIL_COND_V(!atlas_source->has_tile(p_cell.get_atlas_coords()), nullptr); | 
					
						
							|  |  |  | 		ERR_FAIL_COND_V(!atlas_source->has_alternative_tile(p_cell.get_atlas_coords(), p_cell.alternative_tile), nullptr); | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 		td = atlas_source->get_tile_data(p_cell.get_atlas_coords(), p_cell.alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return td; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataEditor::_bind_methods() { | 
					
						
							|  |  |  | 	ADD_SIGNAL(MethodInfo("needs_redraw")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataEditor::set_tile_set(Ref<TileSet> p_tile_set) { | 
					
						
							|  |  |  | 	if (tile_set.is_valid()) { | 
					
						
							| 
									
										
										
										
											2023-07-03 21:29:37 +02:00
										 |  |  | 		tile_set->disconnect_changed(callable_mp(this, &TileDataEditor::_tile_set_changed_plan_update)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	tile_set = p_tile_set; | 
					
						
							|  |  |  | 	if (tile_set.is_valid()) { | 
					
						
							| 
									
										
										
										
											2023-07-03 21:29:37 +02:00
										 |  |  | 		tile_set->connect_changed(callable_mp(this, &TileDataEditor::_tile_set_changed_plan_update)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-10-13 14:49:28 +02:00
										 |  |  | 	_tile_set_changed_plan_update(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DummyObject::_set(const StringName &p_name, const Variant &p_value) { | 
					
						
							|  |  |  | 	if (properties.has(p_name)) { | 
					
						
							|  |  |  | 		properties[p_name] = p_value; | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DummyObject::_get(const StringName &p_name, Variant &r_ret) const { | 
					
						
							|  |  |  | 	if (properties.has(p_name)) { | 
					
						
							|  |  |  | 		r_ret = properties[p_name]; | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-01 16:47:20 -03:00
										 |  |  | bool DummyObject::has_dummy_property(const StringName &p_name) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	return properties.has(p_name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-01 16:47:20 -03:00
										 |  |  | void DummyObject::add_dummy_property(const StringName &p_name) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	ERR_FAIL_COND(properties.has(p_name)); | 
					
						
							|  |  |  | 	properties[p_name] = Variant(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-01 16:47:20 -03:00
										 |  |  | void DummyObject::remove_dummy_property(const StringName &p_name) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	ERR_FAIL_COND(!properties.has(p_name)); | 
					
						
							|  |  |  | 	properties.erase(p_name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DummyObject::clear_dummy_properties() { | 
					
						
							|  |  |  | 	properties.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_base_control_draw() { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!tile_set.is_valid()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-23 17:59:19 +02:00
										 |  |  | 	real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 	Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 	const Ref<Texture2D> handle = get_editor_theme_icon(SNAME("EditorPathSharpHandle")); | 
					
						
							|  |  |  | 	const Ref<Texture2D> add_handle = get_editor_theme_icon(SNAME("EditorHandleAdd")); | 
					
						
							|  |  |  | 	const Ref<StyleBox> focus_stylebox = get_theme_stylebox(SNAME("Focus"), EditorStringName(EditorStyles)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	// Get the background data.
 | 
					
						
							| 
									
										
										
										
											2024-07-05 14:00:30 +02:00
										 |  |  | 	Rect2 background_region; | 
					
						
							|  |  |  | 	TileData *tile_data = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (background_atlas_source.is_valid()) { | 
					
						
							|  |  |  | 		tile_data = background_atlas_source->get_tile_data(background_atlas_coords, background_alternative_id); | 
					
						
							|  |  |  | 		ERR_FAIL_NULL(tile_data); | 
					
						
							|  |  |  | 		background_region = background_atlas_source->get_tile_texture_region(background_atlas_coords); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// If no tile was selected yet, use default size.
 | 
					
						
							|  |  |  | 		background_region.size = tile_set->get_tile_size(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	// Draw the focus rectangle.
 | 
					
						
							|  |  |  | 	if (base_control->has_focus()) { | 
					
						
							|  |  |  | 		base_control->draw_style_box(focus_stylebox, Rect2(Vector2(), base_control->get_size())); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw tile-related things.
 | 
					
						
							| 
									
										
										
										
											2024-05-20 20:13:12 +02:00
										 |  |  | 	const Size2 base_tile_size = tile_set->get_tile_size(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Transform2D xform; | 
					
						
							|  |  |  | 	xform.set_origin(base_control->get_size() / 2 + panning); | 
					
						
							|  |  |  | 	xform.set_scale(Vector2(editor_zoom_widget->get_zoom(), editor_zoom_widget->get_zoom())); | 
					
						
							|  |  |  | 	base_control->draw_set_transform_matrix(xform); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	// Draw fill rect under texture region.
 | 
					
						
							| 
									
										
										
										
											2024-08-09 19:26:09 +02:00
										 |  |  | 	Rect2 texture_rect(Vector2(), background_region.size); | 
					
						
							| 
									
										
										
										
											2024-07-05 14:00:30 +02:00
										 |  |  | 	if (tile_data) { | 
					
						
							|  |  |  | 		texture_rect.position -= tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2024-08-09 19:26:09 +02:00
										 |  |  | 		if (tile_data->get_transpose()) { | 
					
						
							|  |  |  | 			texture_rect.size = Size2(texture_rect.size.y, texture_rect.size.x); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-07-05 14:00:30 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-08-09 19:26:09 +02:00
										 |  |  | 	texture_rect.position -= texture_rect.size / 2; // Half-size offset must be applied after transposing.
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	base_control->draw_rect(texture_rect, Color(1, 1, 1, 0.3)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the background.
 | 
					
						
							| 
									
										
										
										
											2024-07-05 14:00:30 +02:00
										 |  |  | 	if (tile_data && background_atlas_source->get_texture().is_valid()) { | 
					
						
							| 
									
										
										
										
											2022-11-25 00:44:02 +08:00
										 |  |  | 		Size2 region_size = background_region.size; | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 		if (tile_data->get_flip_h()) { | 
					
						
							| 
									
										
										
										
											2022-11-25 00:44:02 +08:00
										 |  |  | 			region_size.x = -region_size.x; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 		if (tile_data->get_flip_v()) { | 
					
						
							| 
									
										
										
										
											2022-11-25 00:44:02 +08:00
										 |  |  | 			region_size.y = -region_size.y; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-08-09 19:26:09 +02:00
										 |  |  | 		// Destination rect position must account for transposing, size must not.
 | 
					
						
							|  |  |  | 		base_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()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	// Compute and draw the grid area.
 | 
					
						
							|  |  |  | 	Rect2 grid_area = Rect2(-base_tile_size / 2, base_tile_size); | 
					
						
							| 
									
										
										
										
											2024-08-09 19:26:09 +02:00
										 |  |  | 	grid_area.expand_to(texture_rect.position); | 
					
						
							|  |  |  | 	grid_area.expand_to(texture_rect.get_end()); | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	base_control->draw_rect(grid_area, Color(1, 1, 1, 0.3), false); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 	// Draw grid.
 | 
					
						
							|  |  |  | 	if (current_snap_option == SNAP_GRID) { | 
					
						
							| 
									
										
										
										
											2024-05-20 20:13:12 +02:00
										 |  |  | 		Vector2 spacing = base_tile_size / snap_subdivision->get_value(); | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 		Vector2 origin = -base_tile_size / 2; | 
					
						
							|  |  |  | 		for (real_t y = origin.y; y < grid_area.get_end().y; y += spacing.y) { | 
					
						
							|  |  |  | 			base_control->draw_line(Vector2(grid_area.get_position().x, y), Vector2(grid_area.get_end().x, y), Color(1, 1, 1, 0.33)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for (real_t y = origin.y - spacing.y; y > grid_area.get_position().y; y -= spacing.y) { | 
					
						
							|  |  |  | 			base_control->draw_line(Vector2(grid_area.get_position().x, y), Vector2(grid_area.get_end().x, y), Color(1, 1, 1, 0.33)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for (real_t x = origin.x; x < grid_area.get_end().x; x += spacing.x) { | 
					
						
							|  |  |  | 			base_control->draw_line(Vector2(x, grid_area.get_position().y), Vector2(x, grid_area.get_end().y), Color(1, 1, 1, 0.33)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for (real_t x = origin.x - spacing.x; x > grid_area.get_position().x; x -= spacing.x) { | 
					
						
							|  |  |  | 			base_control->draw_line(Vector2(x, grid_area.get_position().y), Vector2(x, grid_area.get_end().y), Color(1, 1, 1, 0.33)); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	// Draw the polygons.
 | 
					
						
							| 
									
										
										
										
											2022-12-29 01:24:45 +01:00
										 |  |  | 	for (const Vector<Vector2> &polygon : polygons) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		Color color = polygon_color; | 
					
						
							|  |  |  | 		if (!in_creation_polygon.is_empty()) { | 
					
						
							|  |  |  | 			color = color.darkened(0.3); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		color.a = 0.5; | 
					
						
							|  |  |  | 		Vector<Color> v_color; | 
					
						
							|  |  |  | 		v_color.push_back(color); | 
					
						
							|  |  |  | 		base_control->draw_polygon(polygon, v_color); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		color.a = 0.7; | 
					
						
							|  |  |  | 		for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 			base_control->draw_line(polygon[j], polygon[(j + 1) % polygon.size()], color); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the polygon in creation.
 | 
					
						
							|  |  |  | 	if (!in_creation_polygon.is_empty()) { | 
					
						
							|  |  |  | 		for (int i = 0; i < in_creation_polygon.size() - 1; i++) { | 
					
						
							|  |  |  | 			base_control->draw_line(in_creation_polygon[i], in_creation_polygon[i + 1], Color(1.0, 1.0, 1.0)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Point2 in_creation_point = xform.affine_inverse().xform(base_control->get_local_mouse_position()); | 
					
						
							|  |  |  | 	float in_creation_distance = grab_threshold * 2.0; | 
					
						
							|  |  |  | 	_snap_to_tile_shape(in_creation_point, in_creation_distance, grab_threshold / editor_zoom_widget->get_zoom()); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 	_snap_point(in_creation_point); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (drag_type == DRAG_TYPE_CREATE_POINT && !in_creation_polygon.is_empty()) { | 
					
						
							|  |  |  | 		base_control->draw_line(in_creation_polygon[in_creation_polygon.size() - 1], in_creation_point, Color(1.0, 1.0, 1.0)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the handles.
 | 
					
						
							|  |  |  | 	int tinted_polygon_index = -1; | 
					
						
							|  |  |  | 	int tinted_point_index = -1; | 
					
						
							|  |  |  | 	if (drag_type == DRAG_TYPE_DRAG_POINT) { | 
					
						
							|  |  |  | 		tinted_polygon_index = drag_polygon_index; | 
					
						
							|  |  |  | 		tinted_point_index = drag_point_index; | 
					
						
							|  |  |  | 	} else if (hovered_point_index >= 0) { | 
					
						
							|  |  |  | 		tinted_polygon_index = hovered_polygon_index; | 
					
						
							|  |  |  | 		tinted_point_index = hovered_point_index; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	base_control->draw_set_transform_matrix(Transform2D()); | 
					
						
							|  |  |  | 	if (!in_creation_polygon.is_empty()) { | 
					
						
							|  |  |  | 		for (int i = 0; i < in_creation_polygon.size(); i++) { | 
					
						
							|  |  |  | 			base_control->draw_texture(handle, xform.xform(in_creation_polygon[i]) - handle->get_size() / 2); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		for (int i = 0; i < (int)polygons.size(); i++) { | 
					
						
							|  |  |  | 			const Vector<Vector2> &polygon = polygons[i]; | 
					
						
							|  |  |  | 			for (int j = 0; j < polygon.size(); j++) { | 
					
						
							| 
									
										
										
										
											2024-06-28 23:53:30 +03:00
										 |  |  | 				const Color poly_modulate = (tinted_polygon_index == i && tinted_point_index == j) ? Color(0.4, 1, 1) : Color(1, 1, 1); | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 				base_control->draw_texture(handle, xform.xform(polygon[j]) - handle->get_size() / 2, poly_modulate); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the text on top of the selected point.
 | 
					
						
							|  |  |  | 	if (tinted_polygon_index >= 0) { | 
					
						
							| 
									
										
										
										
											2024-05-14 15:57:29 +02:00
										 |  |  | 		Ref<Font> font = get_theme_font(SceneStringName(font), SNAME("Label")); | 
					
						
							|  |  |  | 		int font_size = get_theme_font_size(SceneStringName(font_size), SNAME("Label")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		String text = multiple_polygon_mode ? vformat("%d:%d", tinted_polygon_index, tinted_point_index) : vformat("%d", tinted_point_index); | 
					
						
							| 
									
										
										
										
											2022-05-09 12:47:10 +03:00
										 |  |  | 		Size2 text_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size); | 
					
						
							| 
									
										
										
										
											2021-11-24 20:58:47 -06:00
										 |  |  | 		base_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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (drag_type == DRAG_TYPE_CREATE_POINT) { | 
					
						
							| 
									
										
										
										
											2024-06-28 23:53:30 +03:00
										 |  |  | 		base_control->draw_texture(handle, xform.xform(in_creation_point) - handle->get_size() / 2, Color(0.4, 1, 1)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the point creation preview in edit mode.
 | 
					
						
							|  |  |  | 	if (hovered_segment_index >= 0) { | 
					
						
							|  |  |  | 		base_control->draw_texture(add_handle, xform.xform(hovered_segment_point) - add_handle->get_size() / 2); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the tile shape line.
 | 
					
						
							|  |  |  | 	base_control->draw_set_transform_matrix(xform); | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	Transform2D tile_xform; | 
					
						
							|  |  |  | 	tile_xform.set_scale(base_tile_size); | 
					
						
							| 
									
										
										
										
											2021-09-14 12:02:13 +02:00
										 |  |  | 	tile_set->draw_tile_shape(base_control, tile_xform, grid_color, false); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	base_control->draw_set_transform_matrix(Transform2D()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_center_view() { | 
					
						
							|  |  |  | 	panning = Vector2(); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	button_center_view->set_disabled(true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_zoom_changed() { | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_advanced_menu_item_pressed(int p_item_pressed) { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 	EditorUndoRedoManager *undo_redo; | 
					
						
							| 
									
										
										
										
											2022-03-25 18:06:46 +01:00
										 |  |  | 	if (use_undo_redo) { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 		undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2022-03-25 18:06:46 +01:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		// This nice hack allows for discarding undo actions without making code too complex.
 | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 		undo_redo = memnew(EditorUndoRedoManager); | 
					
						
							| 
									
										
										
										
											2022-03-25 18:06:46 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	switch (p_item_pressed) { | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 		case RESET_TO_DEFAULT_TILE: { | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 			undo_redo->create_action(TTR("Reset Polygons")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			undo_redo->add_do_method(this, "clear_polygons"); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 			Vector<Vector2> polygon = tile_set->get_tile_shape_polygon(); | 
					
						
							|  |  |  | 			for (int i = 0; i < polygon.size(); i++) { | 
					
						
							|  |  |  | 				polygon.write[i] = polygon[i] * tile_set->get_tile_size(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			undo_redo->add_do_method(this, "add_polygon", polygon); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 			undo_redo->add_do_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			undo_redo->add_do_method(this, "emit_signal", "polygons_changed"); | 
					
						
							|  |  |  | 			undo_redo->add_undo_method(this, "clear_polygons"); | 
					
						
							| 
									
										
										
										
											2022-12-29 01:24:45 +01:00
										 |  |  | 			for (const PackedVector2Array &poly : polygons) { | 
					
						
							|  |  |  | 				undo_redo->add_undo_method(this, "add_polygon", poly); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 			undo_redo->add_undo_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			undo_redo->add_undo_method(this, "emit_signal", "polygons_changed"); | 
					
						
							|  |  |  | 			undo_redo->commit_action(true); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 		} break; | 
					
						
							|  |  |  | 		case CLEAR_TILE: { | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 			undo_redo->create_action(TTR("Clear Polygons")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			undo_redo->add_do_method(this, "clear_polygons"); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 			undo_redo->add_do_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			undo_redo->add_do_method(this, "emit_signal", "polygons_changed"); | 
					
						
							|  |  |  | 			undo_redo->add_undo_method(this, "clear_polygons"); | 
					
						
							| 
									
										
										
										
											2022-12-29 01:24:45 +01:00
										 |  |  | 			for (const PackedVector2Array &polygon : polygons) { | 
					
						
							|  |  |  | 				undo_redo->add_undo_method(this, "add_polygon", polygon); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 			undo_redo->add_undo_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			undo_redo->add_undo_method(this, "emit_signal", "polygons_changed"); | 
					
						
							|  |  |  | 			undo_redo->commit_action(true); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 		case ROTATE_RIGHT: | 
					
						
							|  |  |  | 		case ROTATE_LEFT: | 
					
						
							|  |  |  | 		case FLIP_HORIZONTALLY: | 
					
						
							|  |  |  | 		case FLIP_VERTICALLY: { | 
					
						
							| 
									
										
										
										
											2022-12-18 01:57:54 +01:00
										 |  |  | 			switch (p_item_pressed) { | 
					
						
							|  |  |  | 				case ROTATE_RIGHT: { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Rotate Polygons Right")); | 
					
						
							|  |  |  | 				} break; | 
					
						
							|  |  |  | 				case ROTATE_LEFT: { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Rotate Polygons Left")); | 
					
						
							|  |  |  | 				} break; | 
					
						
							|  |  |  | 				case FLIP_HORIZONTALLY: { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Flip Polygons Horizontally")); | 
					
						
							|  |  |  | 				} break; | 
					
						
							|  |  |  | 				case FLIP_VERTICALLY: { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Flip Polygons Vertically")); | 
					
						
							|  |  |  | 				} break; | 
					
						
							|  |  |  | 				default: | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 			for (unsigned int i = 0; i < polygons.size(); i++) { | 
					
						
							|  |  |  | 				Vector<Point2> new_polygon; | 
					
						
							| 
									
										
										
										
											2022-12-29 01:24:45 +01:00
										 |  |  | 				for (const Vector2 &vec : polygons[i]) { | 
					
						
							|  |  |  | 					Vector2 point = vec; | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 					switch (p_item_pressed) { | 
					
						
							|  |  |  | 						case ROTATE_RIGHT: { | 
					
						
							|  |  |  | 							point = Vector2(-point.y, point.x); | 
					
						
							|  |  |  | 						} break; | 
					
						
							|  |  |  | 						case ROTATE_LEFT: { | 
					
						
							|  |  |  | 							point = Vector2(point.y, -point.x); | 
					
						
							|  |  |  | 						} break; | 
					
						
							|  |  |  | 						case FLIP_HORIZONTALLY: { | 
					
						
							|  |  |  | 							point = Vector2(-point.x, point.y); | 
					
						
							|  |  |  | 						} break; | 
					
						
							|  |  |  | 						case FLIP_VERTICALLY: { | 
					
						
							|  |  |  | 							point = Vector2(point.x, -point.y); | 
					
						
							|  |  |  | 						} break; | 
					
						
							|  |  |  | 						default: | 
					
						
							|  |  |  | 							break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					new_polygon.push_back(point); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				undo_redo->add_do_method(this, "set_polygon", i, new_polygon); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 			undo_redo->add_do_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 			undo_redo->add_do_method(this, "emit_signal", "polygons_changed"); | 
					
						
							| 
									
										
										
										
											2023-10-10 10:33:05 +02:00
										 |  |  | 			for (unsigned int i = 0; i < polygons.size(); i++) { | 
					
						
							|  |  |  | 				undo_redo->add_undo_method(this, "set_polygon", i, polygons[i]); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 			undo_redo->add_undo_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 			undo_redo->add_undo_method(this, "emit_signal", "polygons_changed"); | 
					
						
							|  |  |  | 			undo_redo->commit_action(true); | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		default: | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!use_undo_redo) { | 
					
						
							|  |  |  | 		memdelete(undo_redo); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_grab_polygon_point(Vector2 p_pos, const Transform2D &p_polygon_xform, int &r_polygon_index, int &r_point_index) { | 
					
						
							| 
									
										
										
										
											2021-08-23 17:59:19 +02:00
										 |  |  | 	const real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	r_polygon_index = -1; | 
					
						
							|  |  |  | 	r_point_index = -1; | 
					
						
							|  |  |  | 	float closest_distance = grab_threshold + 1.0; | 
					
						
							|  |  |  | 	for (unsigned int i = 0; i < polygons.size(); i++) { | 
					
						
							|  |  |  | 		const Vector<Vector2> &polygon = polygons[i]; | 
					
						
							|  |  |  | 		for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 			float distance = p_pos.distance_to(p_polygon_xform.xform(polygon[j])); | 
					
						
							|  |  |  | 			if (distance < grab_threshold && distance < closest_distance) { | 
					
						
							|  |  |  | 				r_polygon_index = i; | 
					
						
							|  |  |  | 				r_point_index = j; | 
					
						
							|  |  |  | 				closest_distance = distance; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_grab_polygon_segment_point(Vector2 p_pos, const Transform2D &p_polygon_xform, int &r_polygon_index, int &r_segment_index, Vector2 &r_point) { | 
					
						
							| 
									
										
										
										
											2021-08-23 17:59:19 +02:00
										 |  |  | 	const real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Point2 point = p_polygon_xform.affine_inverse().xform(p_pos); | 
					
						
							|  |  |  | 	r_polygon_index = -1; | 
					
						
							|  |  |  | 	r_segment_index = -1; | 
					
						
							|  |  |  | 	float closest_distance = grab_threshold * 2.0; | 
					
						
							|  |  |  | 	for (unsigned int i = 0; i < polygons.size(); i++) { | 
					
						
							|  |  |  | 		const Vector<Vector2> &polygon = polygons[i]; | 
					
						
							|  |  |  | 		for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 			Vector2 segment[2] = { polygon[j], polygon[(j + 1) % polygon.size()] }; | 
					
						
							|  |  |  | 			Vector2 closest_point = Geometry2D::get_closest_point_to_segment(point, segment); | 
					
						
							|  |  |  | 			float distance = closest_point.distance_to(point); | 
					
						
							|  |  |  | 			if (distance < grab_threshold / editor_zoom_widget->get_zoom() && distance < closest_distance) { | 
					
						
							|  |  |  | 				r_polygon_index = i; | 
					
						
							|  |  |  | 				r_segment_index = j; | 
					
						
							|  |  |  | 				r_point = closest_point; | 
					
						
							|  |  |  | 				closest_distance = distance; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_snap_to_tile_shape(Point2 &r_point, float &r_current_snapped_dist, float p_snap_dist) { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!tile_set.is_valid()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Vector<Point2> polygon = tile_set->get_tile_shape_polygon(); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	for (int i = 0; i < polygon.size(); i++) { | 
					
						
							|  |  |  | 		polygon.write[i] = polygon[i] * tile_set->get_tile_size(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Point2 snapped_point = r_point; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Snap to polygon vertices.
 | 
					
						
							|  |  |  | 	bool snapped = false; | 
					
						
							|  |  |  | 	for (int i = 0; i < polygon.size(); i++) { | 
					
						
							|  |  |  | 		float distance = r_point.distance_to(polygon[i]); | 
					
						
							|  |  |  | 		if (distance < p_snap_dist && distance < r_current_snapped_dist) { | 
					
						
							|  |  |  | 			snapped_point = polygon[i]; | 
					
						
							|  |  |  | 			r_current_snapped_dist = distance; | 
					
						
							|  |  |  | 			snapped = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Snap to edges if we did not snap to vertices.
 | 
					
						
							|  |  |  | 	if (!snapped) { | 
					
						
							|  |  |  | 		for (int i = 0; i < polygon.size(); i++) { | 
					
						
							|  |  |  | 			Point2 segment[2] = { polygon[i], polygon[(i + 1) % polygon.size()] }; | 
					
						
							|  |  |  | 			Point2 point = Geometry2D::get_closest_point_to_segment(r_point, segment); | 
					
						
							|  |  |  | 			float distance = r_point.distance_to(point); | 
					
						
							|  |  |  | 			if (distance < p_snap_dist && distance < r_current_snapped_dist) { | 
					
						
							|  |  |  | 				snapped_point = point; | 
					
						
							|  |  |  | 				r_current_snapped_dist = distance; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	r_point = snapped_point; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | void GenericTilePolygonEditor::_snap_point(Point2 &r_point) { | 
					
						
							|  |  |  | 	switch (current_snap_option) { | 
					
						
							|  |  |  | 		case SNAP_NONE: | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case SNAP_HALF_PIXEL: | 
					
						
							| 
									
										
										
										
											2024-03-03 14:37:52 +01:00
										 |  |  | 			r_point = r_point.snappedf(0.5); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case SNAP_GRID: { | 
					
						
							|  |  |  | 			const Vector2 tile_size = tile_set->get_tile_size(); | 
					
						
							|  |  |  | 			r_point = (r_point + tile_size / 2).snapped(tile_size / snap_subdivision->get_value()) - tile_size / 2; | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_base_control_gui_input(Ref<InputEvent> p_event) { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 	EditorUndoRedoManager *undo_redo; | 
					
						
							| 
									
										
										
										
											2022-03-25 18:06:46 +01:00
										 |  |  | 	if (use_undo_redo) { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 		undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2022-03-25 18:06:46 +01:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		// This nice hack allows for discarding undo actions without making code too complex.
 | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 		undo_redo = memnew(EditorUndoRedoManager); | 
					
						
							| 
									
										
										
										
											2022-03-25 18:06:46 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-23 17:59:19 +02:00
										 |  |  | 	real_t grab_threshold = EDITOR_GET("editors/polygon_editor/point_grab_radius"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	hovered_polygon_index = -1; | 
					
						
							|  |  |  | 	hovered_point_index = -1; | 
					
						
							|  |  |  | 	hovered_segment_index = -1; | 
					
						
							|  |  |  | 	hovered_segment_point = Vector2(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Transform2D xform; | 
					
						
							|  |  |  | 	xform.set_origin(base_control->get_size() / 2 + panning); | 
					
						
							|  |  |  | 	xform.set_scale(Vector2(editor_zoom_widget->get_zoom(), editor_zoom_widget->get_zoom())); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-29 11:41:15 +03:00
										 |  |  | 	Ref<InputEventPanGesture> pan_gesture = p_event; | 
					
						
							|  |  |  | 	if (pan_gesture.is_valid()) { | 
					
						
							|  |  |  | 		panning += pan_gesture->get_delta() * 8; | 
					
						
							|  |  |  | 		drag_last_pos = Vector2(); | 
					
						
							|  |  |  | 		button_center_view->set_disabled(panning.is_zero_approx()); | 
					
						
							|  |  |  | 		accept_event(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ref<InputEventMagnifyGesture> magnify_gesture = p_event; | 
					
						
							|  |  |  | 	if (magnify_gesture.is_valid()) { | 
					
						
							|  |  |  | 		editor_zoom_widget->set_zoom(editor_zoom_widget->get_zoom() * magnify_gesture->get_factor()); | 
					
						
							|  |  |  | 		_zoom_changed(); | 
					
						
							|  |  |  | 		accept_event(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Ref<InputEventMouseMotion> mm = p_event; | 
					
						
							|  |  |  | 	if (mm.is_valid()) { | 
					
						
							|  |  |  | 		if (drag_type == DRAG_TYPE_DRAG_POINT) { | 
					
						
							|  |  |  | 			ERR_FAIL_INDEX(drag_polygon_index, (int)polygons.size()); | 
					
						
							|  |  |  | 			ERR_FAIL_INDEX(drag_point_index, polygons[drag_polygon_index].size()); | 
					
						
							|  |  |  | 			Point2 point = xform.affine_inverse().xform(mm->get_position()); | 
					
						
							|  |  |  | 			float distance = grab_threshold * 2.0; | 
					
						
							|  |  |  | 			_snap_to_tile_shape(point, distance, grab_threshold / editor_zoom_widget->get_zoom()); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 			_snap_point(point); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			polygons[drag_polygon_index].write[drag_point_index] = point; | 
					
						
							|  |  |  | 		} else if (drag_type == DRAG_TYPE_PAN) { | 
					
						
							|  |  |  | 			panning += mm->get_position() - drag_last_pos; | 
					
						
							|  |  |  | 			drag_last_pos = mm->get_position(); | 
					
						
							| 
									
										
										
										
											2022-09-01 20:32:33 -04:00
										 |  |  | 			button_center_view->set_disabled(panning.is_zero_approx()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			// Update hovered point.
 | 
					
						
							|  |  |  | 			_grab_polygon_point(mm->get_position(), xform, hovered_polygon_index, hovered_point_index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// If we have no hovered point, check if we hover a segment.
 | 
					
						
							|  |  |  | 			if (hovered_point_index == -1) { | 
					
						
							|  |  |  | 				_grab_polygon_segment_point(mm->get_position(), xform, hovered_polygon_index, hovered_segment_index, hovered_segment_point); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ref<InputEventMouseButton> mb = p_event; | 
					
						
							|  |  |  | 	if (mb.is_valid()) { | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 		if (mb->get_button_index() == MouseButton::WHEEL_UP && mb->is_command_or_control_pressed()) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			editor_zoom_widget->set_zoom_by_increments(1); | 
					
						
							|  |  |  | 			_zoom_changed(); | 
					
						
							|  |  |  | 			accept_event(); | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 		} else if (mb->get_button_index() == MouseButton::WHEEL_DOWN && mb->is_command_or_control_pressed()) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			editor_zoom_widget->set_zoom_by_increments(-1); | 
					
						
							|  |  |  | 			_zoom_changed(); | 
					
						
							|  |  |  | 			accept_event(); | 
					
						
							| 
									
										
										
										
											2021-08-13 16:31:57 -05:00
										 |  |  | 		} else if (mb->get_button_index() == MouseButton::LEFT) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (mb->is_pressed()) { | 
					
						
							|  |  |  | 				if (tools_button_group->get_pressed_button() != button_create) { | 
					
						
							|  |  |  | 					in_creation_polygon.clear(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				if (tools_button_group->get_pressed_button() == button_create) { | 
					
						
							|  |  |  | 					// Create points.
 | 
					
						
							|  |  |  | 					if (in_creation_polygon.size() >= 3 && mb->get_position().distance_to(xform.xform(in_creation_polygon[0])) < grab_threshold) { | 
					
						
							|  |  |  | 						// Closes and create polygon.
 | 
					
						
							|  |  |  | 						if (!multiple_polygon_mode) { | 
					
						
							|  |  |  | 							clear_polygons(); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						int added = add_polygon(in_creation_polygon); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						in_creation_polygon.clear(); | 
					
						
							|  |  |  | 						button_edit->set_pressed(true); | 
					
						
							|  |  |  | 						undo_redo->create_action(TTR("Edit Polygons")); | 
					
						
							|  |  |  | 						if (!multiple_polygon_mode) { | 
					
						
							|  |  |  | 							undo_redo->add_do_method(this, "clear_polygons"); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						undo_redo->add_do_method(this, "add_polygon", in_creation_polygon); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 						undo_redo->add_do_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						undo_redo->add_undo_method(this, "remove_polygon", added); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 						undo_redo->add_undo_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						undo_redo->commit_action(false); | 
					
						
							| 
									
										
										
										
											2021-07-17 18:22:52 -03:00
										 |  |  | 						emit_signal(SNAME("polygons_changed")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} else { | 
					
						
							|  |  |  | 						// Create a new point.
 | 
					
						
							|  |  |  | 						drag_type = DRAG_TYPE_CREATE_POINT; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} else if (tools_button_group->get_pressed_button() == button_edit) { | 
					
						
							|  |  |  | 					// Edit points.
 | 
					
						
							|  |  |  | 					int closest_polygon; | 
					
						
							|  |  |  | 					int closest_point; | 
					
						
							|  |  |  | 					_grab_polygon_point(mb->get_position(), xform, closest_polygon, closest_point); | 
					
						
							|  |  |  | 					if (closest_polygon >= 0) { | 
					
						
							|  |  |  | 						drag_type = DRAG_TYPE_DRAG_POINT; | 
					
						
							|  |  |  | 						drag_polygon_index = closest_polygon; | 
					
						
							|  |  |  | 						drag_point_index = closest_point; | 
					
						
							|  |  |  | 						drag_old_polygon = polygons[drag_polygon_index]; | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						// Create a point.
 | 
					
						
							|  |  |  | 						Vector2 point_to_create; | 
					
						
							|  |  |  | 						_grab_polygon_segment_point(mb->get_position(), xform, closest_polygon, closest_point, point_to_create); | 
					
						
							|  |  |  | 						if (closest_polygon >= 0) { | 
					
						
							|  |  |  | 							polygons[closest_polygon].insert(closest_point + 1, point_to_create); | 
					
						
							|  |  |  | 							drag_type = DRAG_TYPE_DRAG_POINT; | 
					
						
							|  |  |  | 							drag_polygon_index = closest_polygon; | 
					
						
							|  |  |  | 							drag_point_index = closest_point + 1; | 
					
						
							|  |  |  | 							drag_old_polygon = polygons[closest_polygon]; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} else if (tools_button_group->get_pressed_button() == button_delete) { | 
					
						
							|  |  |  | 					// Remove point.
 | 
					
						
							|  |  |  | 					int closest_polygon; | 
					
						
							|  |  |  | 					int closest_point; | 
					
						
							|  |  |  | 					_grab_polygon_point(mb->get_position(), xform, closest_polygon, closest_point); | 
					
						
							|  |  |  | 					if (closest_polygon >= 0) { | 
					
						
							|  |  |  | 						PackedVector2Array old_polygon = polygons[closest_polygon]; | 
					
						
							| 
									
										
										
										
											2021-07-03 16:17:03 -06:00
										 |  |  | 						polygons[closest_polygon].remove_at(closest_point); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						undo_redo->create_action(TTR("Edit Polygons")); | 
					
						
							|  |  |  | 						if (polygons[closest_polygon].size() < 3) { | 
					
						
							|  |  |  | 							remove_polygon(closest_polygon); | 
					
						
							|  |  |  | 							undo_redo->add_do_method(this, "remove_polygon", closest_polygon); | 
					
						
							|  |  |  | 							undo_redo->add_undo_method(this, "add_polygon", old_polygon, closest_polygon); | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							undo_redo->add_do_method(this, "set_polygon", closest_polygon, polygons[closest_polygon]); | 
					
						
							|  |  |  | 							undo_redo->add_undo_method(this, "set_polygon", closest_polygon, old_polygon); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 						undo_redo->add_do_method(base_control, "queue_redraw"); | 
					
						
							|  |  |  | 						undo_redo->add_undo_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						undo_redo->commit_action(false); | 
					
						
							| 
									
										
										
										
											2021-07-17 18:22:52 -03:00
										 |  |  | 						emit_signal(SNAME("polygons_changed")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				if (drag_type == DRAG_TYPE_DRAG_POINT) { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Edit Polygons")); | 
					
						
							|  |  |  | 					undo_redo->add_do_method(this, "set_polygon", drag_polygon_index, polygons[drag_polygon_index]); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 					undo_redo->add_do_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					undo_redo->add_undo_method(this, "set_polygon", drag_polygon_index, drag_old_polygon); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 					undo_redo->add_undo_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					undo_redo->commit_action(false); | 
					
						
							| 
									
										
										
										
											2021-07-17 18:22:52 -03:00
										 |  |  | 					emit_signal(SNAME("polygons_changed")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} else if (drag_type == DRAG_TYPE_CREATE_POINT) { | 
					
						
							|  |  |  | 					Point2 point = xform.affine_inverse().xform(mb->get_position()); | 
					
						
							|  |  |  | 					float distance = grab_threshold * 2; | 
					
						
							|  |  |  | 					_snap_to_tile_shape(point, distance, grab_threshold / editor_zoom_widget->get_zoom()); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 					_snap_point(point); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					in_creation_polygon.push_back(point); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				drag_type = DRAG_TYPE_NONE; | 
					
						
							|  |  |  | 				drag_point_index = -1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 16:31:57 -05:00
										 |  |  | 		} else if (mb->get_button_index() == MouseButton::RIGHT) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (mb->is_pressed()) { | 
					
						
							|  |  |  | 				if (tools_button_group->get_pressed_button() == button_edit) { | 
					
						
							|  |  |  | 					// Remove point or pan.
 | 
					
						
							|  |  |  | 					int closest_polygon; | 
					
						
							|  |  |  | 					int closest_point; | 
					
						
							|  |  |  | 					_grab_polygon_point(mb->get_position(), xform, closest_polygon, closest_point); | 
					
						
							|  |  |  | 					if (closest_polygon >= 0) { | 
					
						
							|  |  |  | 						PackedVector2Array old_polygon = polygons[closest_polygon]; | 
					
						
							| 
									
										
										
										
											2021-07-03 16:17:03 -06:00
										 |  |  | 						polygons[closest_polygon].remove_at(closest_point); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						undo_redo->create_action(TTR("Edit Polygons")); | 
					
						
							|  |  |  | 						if (polygons[closest_polygon].size() < 3) { | 
					
						
							|  |  |  | 							remove_polygon(closest_polygon); | 
					
						
							|  |  |  | 							undo_redo->add_do_method(this, "remove_polygon", closest_polygon); | 
					
						
							|  |  |  | 							undo_redo->add_undo_method(this, "add_polygon", old_polygon, closest_polygon); | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							undo_redo->add_do_method(this, "set_polygon", closest_polygon, polygons[closest_polygon]); | 
					
						
							|  |  |  | 							undo_redo->add_undo_method(this, "set_polygon", closest_polygon, old_polygon); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 						undo_redo->add_do_method(base_control, "queue_redraw"); | 
					
						
							|  |  |  | 						undo_redo->add_undo_method(base_control, "queue_redraw"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						undo_redo->commit_action(false); | 
					
						
							| 
									
										
										
										
											2021-07-17 18:22:52 -03:00
										 |  |  | 						emit_signal(SNAME("polygons_changed")); | 
					
						
							| 
									
										
										
										
											2024-02-27 15:16:27 +01:00
										 |  |  | 						drag_type = DRAG_TYPE_NONE; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} else { | 
					
						
							|  |  |  | 						drag_type = DRAG_TYPE_PAN; | 
					
						
							|  |  |  | 						drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_PAN; | 
					
						
							|  |  |  | 					drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				drag_type = DRAG_TYPE_NONE; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-08-13 16:31:57 -05:00
										 |  |  | 		} else if (mb->get_button_index() == MouseButton::MIDDLE) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (mb->is_pressed()) { | 
					
						
							|  |  |  | 				drag_type = DRAG_TYPE_PAN; | 
					
						
							|  |  |  | 				drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				drag_type = DRAG_TYPE_NONE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!use_undo_redo) { | 
					
						
							|  |  |  | 		memdelete(undo_redo); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | void GenericTilePolygonEditor::_set_snap_option(int p_index) { | 
					
						
							|  |  |  | 	current_snap_option = p_index; | 
					
						
							|  |  |  | 	button_pixel_snap->set_icon(button_pixel_snap->get_popup()->get_item_icon(p_index)); | 
					
						
							|  |  |  | 	snap_subdivision->set_visible(p_index == SNAP_GRID); | 
					
						
							| 
									
										
										
										
											2024-03-07 14:36:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (initializing) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							|  |  |  | 	_store_snap_options(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_store_snap_options() { | 
					
						
							|  |  |  | 	EditorSettings::get_singleton()->set_project_metadata("editor_metadata", "tile_snap_option", current_snap_option); | 
					
						
							|  |  |  | 	EditorSettings::get_singleton()->set_project_metadata("editor_metadata", "tile_snap_subdiv", snap_subdivision->get_value()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-15 20:58:52 +02:00
										 |  |  | void GenericTilePolygonEditor::_toggle_expand(bool p_expand) { | 
					
						
							|  |  |  | 	if (p_expand) { | 
					
						
							|  |  |  | 		TileSetEditor::get_singleton()->add_expanded_editor(this); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		TileSetEditor::get_singleton()->remove_expanded_editor(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | void GenericTilePolygonEditor::set_use_undo_redo(bool p_use_undo_redo) { | 
					
						
							|  |  |  | 	use_undo_redo = p_use_undo_redo; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::set_tile_set(Ref<TileSet> p_tile_set) { | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	ERR_FAIL_COND(!p_tile_set.is_valid()); | 
					
						
							|  |  |  | 	if (tile_set == p_tile_set) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Set the default tile shape
 | 
					
						
							|  |  |  | 	clear_polygons(); | 
					
						
							|  |  |  | 	if (p_tile_set.is_valid()) { | 
					
						
							|  |  |  | 		Vector<Vector2> polygon = p_tile_set->get_tile_shape_polygon(); | 
					
						
							|  |  |  | 		for (int i = 0; i < polygon.size(); i++) { | 
					
						
							|  |  |  | 			polygon.write[i] = polygon[i] * p_tile_set->get_tile_size(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 		add_polygon(polygon); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	// Trigger a redraw on tile_set change.
 | 
					
						
							|  |  |  | 	Callable callable = callable_mp((CanvasItem *)base_control, &CanvasItem::queue_redraw); | 
					
						
							|  |  |  | 	if (tile_set.is_valid()) { | 
					
						
							|  |  |  | 		tile_set->disconnect_changed(callable); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	tile_set = p_tile_set; | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	if (tile_set.is_valid()) { | 
					
						
							|  |  |  | 		tile_set->connect_changed(callable); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	// Set the default zoom value.
 | 
					
						
							|  |  |  | 	int default_control_y_size = 200 * EDSCALE; | 
					
						
							|  |  |  | 	Vector2 zoomed_tile = editor_zoom_widget->get_zoom() * tile_set->get_tile_size(); | 
					
						
							|  |  |  | 	while (zoomed_tile.y < default_control_y_size) { | 
					
						
							|  |  |  | 		editor_zoom_widget->set_zoom_by_increments(6, false); | 
					
						
							| 
									
										
										
										
											2023-12-28 10:42:37 +08:00
										 |  |  | 		float current_zoom = editor_zoom_widget->get_zoom(); | 
					
						
							|  |  |  | 		zoomed_tile = current_zoom * tile_set->get_tile_size(); | 
					
						
							|  |  |  | 		if (Math::is_equal_approx(current_zoom, editor_zoom_widget->get_max_zoom())) { | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	while (zoomed_tile.y > default_control_y_size) { | 
					
						
							|  |  |  | 		editor_zoom_widget->set_zoom_by_increments(-6, false); | 
					
						
							| 
									
										
										
										
											2023-12-28 10:42:37 +08:00
										 |  |  | 		float current_zoom = editor_zoom_widget->get_zoom(); | 
					
						
							|  |  |  | 		zoomed_tile = current_zoom * tile_set->get_tile_size(); | 
					
						
							|  |  |  | 		if (Math::is_equal_approx(current_zoom, editor_zoom_widget->get_min_zoom())) { | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	editor_zoom_widget->set_zoom_by_increments(-6, false); | 
					
						
							|  |  |  | 	_zoom_changed(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | void GenericTilePolygonEditor::set_background_tile(const TileSetAtlasSource *p_atlas_source, const Vector2 &p_atlas_coords, int p_alternative_id) { | 
					
						
							|  |  |  | 	ERR_FAIL_NULL(p_atlas_source); | 
					
						
							|  |  |  | 	background_atlas_source = p_atlas_source; | 
					
						
							|  |  |  | 	background_atlas_coords = p_atlas_coords; | 
					
						
							|  |  |  | 	background_alternative_id = p_alternative_id; | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GenericTilePolygonEditor::get_polygon_count() { | 
					
						
							|  |  |  | 	return polygons.size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | int GenericTilePolygonEditor::add_polygon(const Vector<Point2> &p_polygon, int p_index) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	ERR_FAIL_COND_V(p_polygon.size() < 3, -1); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(!multiple_polygon_mode && polygons.size() >= 1, -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_index < 0) { | 
					
						
							|  |  |  | 		polygons.push_back(p_polygon); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 		base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		button_edit->set_pressed(true); | 
					
						
							|  |  |  | 		return polygons.size() - 1; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		polygons.insert(p_index, p_polygon); | 
					
						
							|  |  |  | 		button_edit->set_pressed(true); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 		base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		return p_index; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::remove_polygon(int p_index) { | 
					
						
							|  |  |  | 	ERR_FAIL_INDEX(p_index, (int)polygons.size()); | 
					
						
							| 
									
										
										
										
											2021-07-03 16:17:03 -06:00
										 |  |  | 	polygons.remove_at(p_index); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (polygons.size() == 0) { | 
					
						
							|  |  |  | 		button_create->set_pressed(true); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::clear_polygons() { | 
					
						
							|  |  |  | 	polygons.clear(); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void GenericTilePolygonEditor::set_polygon(int p_polygon_index, const Vector<Point2> &p_polygon) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	ERR_FAIL_INDEX(p_polygon_index, (int)polygons.size()); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(p_polygon.size() < 3); | 
					
						
							|  |  |  | 	polygons[p_polygon_index] = p_polygon; | 
					
						
							|  |  |  | 	button_edit->set_pressed(true); | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Vector<Point2> GenericTilePolygonEditor::get_polygon(int p_polygon_index) { | 
					
						
							|  |  |  | 	ERR_FAIL_INDEX_V(p_polygon_index, (int)polygons.size(), Vector<Point2>()); | 
					
						
							|  |  |  | 	return polygons[p_polygon_index]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::set_polygons_color(Color p_color) { | 
					
						
							|  |  |  | 	polygon_color = p_color; | 
					
						
							| 
									
										
										
										
											2022-08-13 23:21:24 +02:00
										 |  |  | 	base_control->queue_redraw(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::set_multiple_polygon_mode(bool p_multiple_polygon_mode) { | 
					
						
							|  |  |  | 	multiple_polygon_mode = p_multiple_polygon_mode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_notification(int p_what) { | 
					
						
							|  |  |  | 	switch (p_what) { | 
					
						
							| 
									
										
										
										
											2023-07-15 20:58:52 +02:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: { | 
					
						
							|  |  |  | 			if (!get_meta("reparented", false)) { | 
					
						
							|  |  |  | 				button_expand->set_pressed_no_signal(false); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2022-01-28 20:21:50 +03:00
										 |  |  | 		case NOTIFICATION_THEME_CHANGED: { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 			button_expand->set_icon(get_editor_theme_icon(SNAME("DistractionFree"))); | 
					
						
							|  |  |  | 			button_create->set_icon(get_editor_theme_icon(SNAME("CurveCreate"))); | 
					
						
							|  |  |  | 			button_edit->set_icon(get_editor_theme_icon(SNAME("CurveEdit"))); | 
					
						
							|  |  |  | 			button_delete->set_icon(get_editor_theme_icon(SNAME("CurveDelete"))); | 
					
						
							|  |  |  | 			button_center_view->set_icon(get_editor_theme_icon(SNAME("CenterView"))); | 
					
						
							|  |  |  | 			button_advanced_menu->set_icon(get_editor_theme_icon(SNAME("GuiTabMenuHl"))); | 
					
						
							|  |  |  | 			button_pixel_snap->get_popup()->set_item_icon(0, get_editor_theme_icon(SNAME("SnapDisable"))); | 
					
						
							|  |  |  | 			button_pixel_snap->get_popup()->set_item_icon(1, get_editor_theme_icon(SNAME("Snap"))); | 
					
						
							|  |  |  | 			button_pixel_snap->get_popup()->set_item_icon(2, get_editor_theme_icon(SNAME("SnapGrid"))); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 			button_pixel_snap->set_icon(button_pixel_snap->get_popup()->get_item_icon(current_snap_option)); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			PopupMenu *p = button_advanced_menu->get_popup(); | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 			p->set_item_icon(p->get_item_index(ROTATE_RIGHT), get_editor_theme_icon(SNAME("RotateRight"))); | 
					
						
							|  |  |  | 			p->set_item_icon(p->get_item_index(ROTATE_LEFT), get_editor_theme_icon(SNAME("RotateLeft"))); | 
					
						
							|  |  |  | 			p->set_item_icon(p->get_item_index(FLIP_HORIZONTALLY), get_editor_theme_icon(SNAME("MirrorX"))); | 
					
						
							|  |  |  | 			p->set_item_icon(p->get_item_index(FLIP_VERTICALLY), get_editor_theme_icon(SNAME("MirrorY"))); | 
					
						
							| 
									
										
										
										
											2022-01-28 20:21:50 +03:00
										 |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GenericTilePolygonEditor::_bind_methods() { | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("get_polygon_count"), &GenericTilePolygonEditor::get_polygon_count); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("add_polygon", "polygon", "index"), &GenericTilePolygonEditor::add_polygon, DEFVAL(-1)); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("remove_polygon", "index"), &GenericTilePolygonEditor::remove_polygon); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("clear_polygons"), &GenericTilePolygonEditor::clear_polygons); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("set_polygon", "index", "polygon"), &GenericTilePolygonEditor::set_polygon); | 
					
						
							| 
									
										
										
										
											2023-02-25 00:57:34 +05:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("get_polygon", "index"), &GenericTilePolygonEditor::get_polygon); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ADD_SIGNAL(MethodInfo("polygons_changed")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GenericTilePolygonEditor::GenericTilePolygonEditor() { | 
					
						
							|  |  |  | 	toolbar = memnew(HBoxContainer); | 
					
						
							|  |  |  | 	add_child(toolbar); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tools_button_group.instantiate(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-15 20:58:52 +02:00
										 |  |  | 	button_expand = memnew(Button); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	button_expand->set_theme_type_variation("FlatButton"); | 
					
						
							| 
									
										
										
										
											2023-07-15 20:58:52 +02:00
										 |  |  | 	button_expand->set_toggle_mode(true); | 
					
						
							|  |  |  | 	button_expand->set_pressed(false); | 
					
						
							|  |  |  | 	button_expand->set_tooltip_text(TTR("Expand editor")); | 
					
						
							| 
									
										
										
										
											2024-06-01 13:15:13 +03:00
										 |  |  | 	button_expand->connect(SceneStringName(toggled), callable_mp(this, &GenericTilePolygonEditor::_toggle_expand)); | 
					
						
							| 
									
										
										
										
											2023-07-15 20:58:52 +02:00
										 |  |  | 	toolbar->add_child(button_expand); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	toolbar->add_child(memnew(VSeparator)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	button_create = memnew(Button); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	button_create->set_theme_type_variation("FlatButton"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	button_create->set_toggle_mode(true); | 
					
						
							|  |  |  | 	button_create->set_button_group(tools_button_group); | 
					
						
							|  |  |  | 	button_create->set_pressed(true); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 	button_create->set_tooltip_text(TTR("Add polygon tool")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	toolbar->add_child(button_create); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	button_edit = memnew(Button); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	button_edit->set_theme_type_variation("FlatButton"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	button_edit->set_toggle_mode(true); | 
					
						
							|  |  |  | 	button_edit->set_button_group(tools_button_group); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 	button_edit->set_tooltip_text(TTR("Edit points tool")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	toolbar->add_child(button_edit); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	button_delete = memnew(Button); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	button_delete->set_theme_type_variation("FlatButton"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	button_delete->set_toggle_mode(true); | 
					
						
							|  |  |  | 	button_delete->set_button_group(tools_button_group); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 	button_delete->set_tooltip_text(TTR("Delete points tool")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	toolbar->add_child(button_delete); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	button_advanced_menu = memnew(MenuButton); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	button_advanced_menu->set_flat(false); | 
					
						
							|  |  |  | 	button_advanced_menu->set_theme_type_variation("FlatMenuButton"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	button_advanced_menu->set_toggle_mode(true); | 
					
						
							| 
									
										
										
										
											2022-06-07 02:03:04 +02:00
										 |  |  | 	button_advanced_menu->get_popup()->add_item(TTR("Reset to default tile shape"), RESET_TO_DEFAULT_TILE, Key::F); | 
					
						
							|  |  |  | 	button_advanced_menu->get_popup()->add_item(TTR("Clear"), CLEAR_TILE, Key::C); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	button_advanced_menu->get_popup()->add_separator(); | 
					
						
							| 
									
										
										
										
											2023-09-13 13:14:07 +02:00
										 |  |  | 	button_advanced_menu->get_popup()->add_item(TTR("Rotate Right"), ROTATE_RIGHT, Key::R); | 
					
						
							|  |  |  | 	button_advanced_menu->get_popup()->add_item(TTR("Rotate Left"), ROTATE_LEFT, Key::E); | 
					
						
							|  |  |  | 	button_advanced_menu->get_popup()->add_item(TTR("Flip Horizontally"), FLIP_HORIZONTALLY, Key::H); | 
					
						
							|  |  |  | 	button_advanced_menu->get_popup()->add_item(TTR("Flip Vertically"), FLIP_VERTICALLY, Key::V); | 
					
						
							| 
									
										
										
										
											2024-05-14 14:13:31 +02:00
										 |  |  | 	button_advanced_menu->get_popup()->connect(SceneStringName(id_pressed), callable_mp(this, &GenericTilePolygonEditor::_advanced_menu_item_pressed)); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	button_advanced_menu->set_focus_mode(FOCUS_ALL); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	toolbar->add_child(button_advanced_menu); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	toolbar->add_child(memnew(VSeparator)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 	button_pixel_snap = memnew(MenuButton); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	toolbar->add_child(button_pixel_snap); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	button_pixel_snap->set_flat(false); | 
					
						
							|  |  |  | 	button_pixel_snap->set_theme_type_variation("FlatMenuButton"); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 	button_pixel_snap->set_tooltip_text(TTR("Toggle Grid Snap")); | 
					
						
							|  |  |  | 	button_pixel_snap->get_popup()->add_item(TTR("Disable Snap"), SNAP_NONE); | 
					
						
							|  |  |  | 	button_pixel_snap->get_popup()->add_item(TTR("Half-Pixel Snap"), SNAP_HALF_PIXEL); | 
					
						
							|  |  |  | 	button_pixel_snap->get_popup()->add_item(TTR("Grid Snap"), SNAP_GRID); | 
					
						
							|  |  |  | 	button_pixel_snap->get_popup()->connect("index_pressed", callable_mp(this, &GenericTilePolygonEditor::_set_snap_option)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	snap_subdivision = memnew(SpinBox); | 
					
						
							|  |  |  | 	toolbar->add_child(snap_subdivision); | 
					
						
							|  |  |  | 	snap_subdivision->get_line_edit()->add_theme_constant_override("minimum_character_width", 2); | 
					
						
							|  |  |  | 	snap_subdivision->set_min(1); | 
					
						
							|  |  |  | 	snap_subdivision->set_max(99); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Control *root = memnew(Control); | 
					
						
							| 
									
										
										
										
											2023-07-15 20:58:52 +02:00
										 |  |  | 	root->set_v_size_flags(Control::SIZE_EXPAND_FILL); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	root->set_custom_minimum_size(Size2(0, 200 * EDSCALE)); | 
					
						
							|  |  |  | 	root->set_mouse_filter(Control::MOUSE_FILTER_IGNORE); | 
					
						
							|  |  |  | 	add_child(root); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	panel = memnew(Panel); | 
					
						
							| 
									
										
										
										
											2022-03-18 19:02:57 -05:00
										 |  |  | 	panel->set_anchors_and_offsets_preset(Control::PRESET_FULL_RECT); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	panel->set_mouse_filter(Control::MOUSE_FILTER_IGNORE); | 
					
						
							|  |  |  | 	root->add_child(panel); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	base_control = memnew(Control); | 
					
						
							|  |  |  | 	base_control->set_texture_filter(CanvasItem::TEXTURE_FILTER_NEAREST); | 
					
						
							| 
									
										
										
										
											2022-03-18 19:02:57 -05:00
										 |  |  | 	base_control->set_anchors_and_offsets_preset(Control::PRESET_FULL_RECT); | 
					
						
							| 
									
										
										
										
											2024-05-13 16:56:03 +02:00
										 |  |  | 	base_control->connect(SceneStringName(draw), callable_mp(this, &GenericTilePolygonEditor::_base_control_draw)); | 
					
						
							|  |  |  | 	base_control->connect(SceneStringName(gui_input), callable_mp(this, &GenericTilePolygonEditor::_base_control_gui_input)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	base_control->set_clip_contents(true); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	base_control->set_focus_mode(Control::FOCUS_CLICK); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	root->add_child(base_control); | 
					
						
							| 
									
										
										
										
											2024-05-14 11:42:00 +02:00
										 |  |  | 	snap_subdivision->connect(SceneStringName(value_changed), callable_mp((CanvasItem *)base_control, &CanvasItem::queue_redraw).unbind(1)); | 
					
						
							|  |  |  | 	snap_subdivision->connect(SceneStringName(value_changed), callable_mp(this, &GenericTilePolygonEditor::_store_snap_options).unbind(1)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	editor_zoom_widget = memnew(EditorZoomWidget); | 
					
						
							| 
									
										
										
										
											2023-09-17 18:33:02 +03:00
										 |  |  | 	editor_zoom_widget->setup_zoom_limits(0.125, 128.0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	editor_zoom_widget->set_position(Vector2(5, 5)); | 
					
						
							|  |  |  | 	editor_zoom_widget->connect("zoom_changed", callable_mp(this, &GenericTilePolygonEditor::_zoom_changed).unbind(1)); | 
					
						
							| 
									
										
										
										
											2022-03-24 23:03:43 +10:00
										 |  |  | 	editor_zoom_widget->set_shortcut_context(this); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	root->add_child(editor_zoom_widget); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	button_center_view = memnew(Button); | 
					
						
							|  |  |  | 	button_center_view->set_anchors_and_offsets_preset(Control::PRESET_TOP_RIGHT, Control::PRESET_MODE_MINSIZE, 5); | 
					
						
							| 
									
										
										
										
											2024-03-13 12:46:14 +08:00
										 |  |  | 	button_center_view->set_grow_direction_preset(Control::PRESET_TOP_RIGHT); | 
					
						
							| 
									
										
										
										
											2024-05-14 09:40:21 +02:00
										 |  |  | 	button_center_view->connect(SceneStringName(pressed), callable_mp(this, &GenericTilePolygonEditor::_center_view)); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	button_center_view->set_theme_type_variation("FlatButton"); | 
					
						
							| 
									
										
										
										
											2024-03-13 12:46:14 +08:00
										 |  |  | 	button_center_view->set_tooltip_text(TTR("Center View")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	button_center_view->set_disabled(true); | 
					
						
							|  |  |  | 	root->add_child(button_center_view); | 
					
						
							| 
									
										
										
										
											2022-12-23 19:34:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	snap_subdivision->set_value_no_signal(EditorSettings::get_singleton()->get_project_metadata("editor_metadata", "tile_snap_subdiv", 4)); | 
					
						
							|  |  |  | 	_set_snap_option(EditorSettings::get_singleton()->get_project_metadata("editor_metadata", "tile_snap_option", SNAP_NONE)); | 
					
						
							| 
									
										
										
										
											2024-03-07 14:36:20 +01:00
										 |  |  | 	initializing = false; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataDefaultEditor::_property_value_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field) { | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(dummy_object); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	dummy_object->set(p_property, p_value); | 
					
						
							| 
									
										
										
										
											2022-11-27 18:35:47 +08:00
										 |  |  | 	emit_signal(SNAME("needs_redraw")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant TileDataDefaultEditor::_get_painted_value() { | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL_V(dummy_object, Variant()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	return dummy_object->get(property); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataDefaultEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Variant value = tile_data->get(property); | 
					
						
							|  |  |  | 	dummy_object->set(property, value); | 
					
						
							|  |  |  | 	if (property_editor) { | 
					
						
							|  |  |  | 		property_editor->update_property(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataDefaultEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	tile_data->set(property, p_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant TileDataDefaultEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL_V(tile_data, Variant()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	return tile_data->get(property); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataDefaultEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 	EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 	for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) { | 
					
						
							|  |  |  | 		Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							|  |  |  | 		undo_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); | 
					
						
							|  |  |  | 		undo_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataDefaultEditor::forward_draw_over_atlas(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform) { | 
					
						
							|  |  |  | 	if (drag_type == DRAG_TYPE_PAINT_RECT) { | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 		Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		Color 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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p_canvas_item->draw_set_transform_matrix(p_transform); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Rect2i rect; | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 		rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true)); | 
					
						
							|  |  |  | 		rect.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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		rect = rect.abs(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 15:04:37 +02:00
										 |  |  | 		RBSet<TileMapCell> edited; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		for (int x = rect.get_position().x; x <= rect.get_end().x; x++) { | 
					
						
							|  |  |  | 			for (int y = rect.get_position().y; y <= rect.get_end().y; y++) { | 
					
						
							|  |  |  | 				Vector2i coords = Vector2i(x, y); | 
					
						
							|  |  |  | 				coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 				if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 					TileMapCell cell; | 
					
						
							|  |  |  | 					cell.source_id = 0; | 
					
						
							|  |  |  | 					cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 					cell.alternative_tile = 0; | 
					
						
							|  |  |  | 					edited.insert(cell); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-18 17:43:40 -06:00
										 |  |  | 		for (const TileMapCell &E : edited) { | 
					
						
							|  |  |  | 			Vector2i coords = E.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			p_canvas_item->draw_rect(p_tile_set_atlas_source->get_tile_texture_region(coords), selection_color, false); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		p_canvas_item->draw_set_transform_matrix(Transform2D()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataDefaultEditor::forward_draw_over_alternatives(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform){ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataDefaultEditor::forward_painting_atlas_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) { | 
					
						
							|  |  |  | 	Ref<InputEventMouseMotion> mm = p_event; | 
					
						
							|  |  |  | 	if (mm.is_valid()) { | 
					
						
							|  |  |  | 		if (drag_type == DRAG_TYPE_PAINT) { | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 			Vector<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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			for (int i = 0; i < line.size(); i++) { | 
					
						
							|  |  |  | 				Vector2i coords = p_tile_set_atlas_source->get_tile_at_coords(line[i]); | 
					
						
							|  |  |  | 				if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 					TileMapCell cell; | 
					
						
							|  |  |  | 					cell.source_id = 0; | 
					
						
							|  |  |  | 					cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 					cell.alternative_tile = 0; | 
					
						
							|  |  |  | 					if (!drag_modified.has(cell)) { | 
					
						
							|  |  |  | 						drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, 0); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					_set_value(p_tile_set_atlas_source, coords, 0, drag_painted_value); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			drag_last_pos = mm->get_position(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 	EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Ref<InputEventMouseButton> mb = p_event; | 
					
						
							|  |  |  | 	if (mb.is_valid()) { | 
					
						
							| 
									
										
										
										
											2021-08-13 16:31:57 -05:00
										 |  |  | 		if (mb->get_button_index() == MouseButton::LEFT) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (mb->is_pressed()) { | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 				if (picker_button->is_pressed() || (mb->is_command_or_control_pressed() && !mb->is_shift_pressed())) { | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 					Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 					if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 						_set_painted_value(p_tile_set_atlas_source, coords, 0); | 
					
						
							|  |  |  | 						picker_button->set_pressed(false); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 				} else if (mb->is_command_or_control_pressed() && mb->is_shift_pressed()) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					drag_type = DRAG_TYPE_PAINT_RECT; | 
					
						
							|  |  |  | 					drag_modified.clear(); | 
					
						
							|  |  |  | 					drag_painted_value = _get_painted_value(); | 
					
						
							|  |  |  | 					drag_start_pos = mb->get_position(); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_PAINT; | 
					
						
							|  |  |  | 					drag_modified.clear(); | 
					
						
							|  |  |  | 					drag_painted_value = _get_painted_value(); | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 					Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 					if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 						TileMapCell cell; | 
					
						
							|  |  |  | 						cell.source_id = 0; | 
					
						
							|  |  |  | 						cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 						cell.alternative_tile = 0; | 
					
						
							|  |  |  | 						drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, 0); | 
					
						
							|  |  |  | 						_set_value(p_tile_set_atlas_source, coords, 0, drag_painted_value); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				if (drag_type == DRAG_TYPE_PAINT_RECT) { | 
					
						
							|  |  |  | 					Rect2i rect; | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 					rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true)); | 
					
						
							|  |  |  | 					rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					rect = rect.abs(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					drag_modified.clear(); | 
					
						
							|  |  |  | 					for (int x = rect.get_position().x; x <= rect.get_end().x; x++) { | 
					
						
							|  |  |  | 						for (int y = rect.get_position().y; y <= rect.get_end().y; y++) { | 
					
						
							|  |  |  | 							Vector2i coords = Vector2i(x, y); | 
					
						
							|  |  |  | 							coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 							if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 								TileMapCell cell; | 
					
						
							|  |  |  | 								cell.source_id = 0; | 
					
						
							|  |  |  | 								cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 								cell.alternative_tile = 0; | 
					
						
							|  |  |  | 								drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, 0); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Tiles Property")); | 
					
						
							|  |  |  | 					_setup_undo_redo_action(p_tile_set_atlas_source, drag_modified, drag_painted_value); | 
					
						
							|  |  |  | 					undo_redo->commit_action(true); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							|  |  |  | 				} else if (drag_type == DRAG_TYPE_PAINT) { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Tiles Property")); | 
					
						
							|  |  |  | 					_setup_undo_redo_action(p_tile_set_atlas_source, drag_modified, drag_painted_value); | 
					
						
							|  |  |  | 					undo_redo->commit_action(false); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataDefaultEditor::forward_painting_alternatives_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) { | 
					
						
							|  |  |  | 	Ref<InputEventMouseMotion> mm = p_event; | 
					
						
							|  |  |  | 	if (mm.is_valid()) { | 
					
						
							|  |  |  | 		if (drag_type == DRAG_TYPE_PAINT) { | 
					
						
							|  |  |  | 			Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mm->get_position()); | 
					
						
							|  |  |  | 			Vector2i coords = Vector2i(tile.x, tile.y); | 
					
						
							|  |  |  | 			int alternative_tile = tile.z; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 				TileMapCell cell; | 
					
						
							|  |  |  | 				cell.source_id = 0; | 
					
						
							|  |  |  | 				cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 				cell.alternative_tile = alternative_tile; | 
					
						
							|  |  |  | 				if (!drag_modified.has(cell)) { | 
					
						
							|  |  |  | 					drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, alternative_tile); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				_set_value(p_tile_set_atlas_source, coords, alternative_tile, drag_painted_value); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			drag_last_pos = mm->get_position(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ref<InputEventMouseButton> mb = p_event; | 
					
						
							|  |  |  | 	if (mb.is_valid()) { | 
					
						
							| 
									
										
										
										
											2021-08-13 16:31:57 -05:00
										 |  |  | 		if (mb->get_button_index() == MouseButton::LEFT) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (mb->is_pressed()) { | 
					
						
							|  |  |  | 				if (picker_button->is_pressed()) { | 
					
						
							|  |  |  | 					Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position()); | 
					
						
							|  |  |  | 					Vector2i coords = Vector2i(tile.x, tile.y); | 
					
						
							|  |  |  | 					int alternative_tile = tile.z; | 
					
						
							|  |  |  | 					if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 						_set_painted_value(p_tile_set_atlas_source, coords, alternative_tile); | 
					
						
							|  |  |  | 						picker_button->set_pressed(false); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_PAINT; | 
					
						
							|  |  |  | 					drag_modified.clear(); | 
					
						
							|  |  |  | 					drag_painted_value = _get_painted_value(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position()); | 
					
						
							|  |  |  | 					Vector2i coords = Vector2i(tile.x, tile.y); | 
					
						
							|  |  |  | 					int alternative_tile = tile.z; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 						TileMapCell cell; | 
					
						
							|  |  |  | 						cell.source_id = 0; | 
					
						
							|  |  |  | 						cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 						cell.alternative_tile = alternative_tile; | 
					
						
							|  |  |  | 						drag_modified[cell] = _get_value(p_tile_set_atlas_source, coords, alternative_tile); | 
					
						
							|  |  |  | 						_set_value(p_tile_set_atlas_source, coords, alternative_tile, drag_painted_value); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 				EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				undo_redo->create_action(TTR("Painting Tiles Property")); | 
					
						
							|  |  |  | 				_setup_undo_redo_action(p_tile_set_atlas_source, drag_modified, drag_painted_value); | 
					
						
							|  |  |  | 				undo_redo->commit_action(false); | 
					
						
							|  |  |  | 				drag_type = DRAG_TYPE_NONE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataDefaultEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							|  |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	bool valid; | 
					
						
							|  |  |  | 	Variant value = tile_data->get(property, &valid); | 
					
						
							|  |  |  | 	if (!valid) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 	Vector2 texture_origin = tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	if (value.get_type() == Variant::BOOL) { | 
					
						
							|  |  |  | 		Ref<Texture2D> texture = (bool)value ? tile_bool_checked : tile_bool_unchecked; | 
					
						
							|  |  |  | 		int size = MIN(tile_set->get_tile_size().x, tile_set->get_tile_size().y) / 3; | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 		Rect2 rect = p_transform.xform(Rect2(Vector2(-size / 2, -size / 2) - texture_origin, Vector2(size, size))); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		p_canvas_item->draw_texture_rect(texture, rect); | 
					
						
							|  |  |  | 	} else if (value.get_type() == Variant::COLOR) { | 
					
						
							|  |  |  | 		int size = MIN(tile_set->get_tile_size().x, tile_set->get_tile_size().y) / 3; | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 		Rect2 rect = p_transform.xform(Rect2(Vector2(-size / 2, -size / 2) - texture_origin, Vector2(size, size))); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		p_canvas_item->draw_rect(rect, value); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 		Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts)); | 
					
						
							|  |  |  | 		int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		String text; | 
					
						
							| 
									
										
										
										
											2023-06-21 17:43:45 +02:00
										 |  |  | 		// Round floating point precision to 2 digits, as tiles don't have that much space.
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		switch (value.get_type()) { | 
					
						
							|  |  |  | 			case Variant::FLOAT: | 
					
						
							|  |  |  | 				text = vformat("%.2f", value); | 
					
						
							|  |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2023-06-21 17:43:45 +02:00
										 |  |  | 			case Variant::VECTOR2: | 
					
						
							|  |  |  | 			case Variant::VECTOR3: | 
					
						
							|  |  |  | 			case Variant::VECTOR4: | 
					
						
							|  |  |  | 				text = vformat("%.2v", value); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			default: | 
					
						
							| 
									
										
										
										
											2023-06-21 17:43:45 +02:00
										 |  |  | 				text = value.stringify(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Color color = Color(1, 1, 1); | 
					
						
							|  |  |  | 		if (p_selected) { | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 			Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			Color 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); | 
					
						
							|  |  |  | 			selection_color.set_v(0.9); | 
					
						
							|  |  |  | 			color = selection_color; | 
					
						
							|  |  |  | 		} else if (is_visible_in_tree()) { | 
					
						
							|  |  |  | 			Variant painted_value = _get_painted_value(); | 
					
						
							|  |  |  | 			bool equal = (painted_value.get_type() == Variant::FLOAT && value.get_type() == Variant::FLOAT) ? Math::is_equal_approx(float(painted_value), float(value)) : painted_value == value; | 
					
						
							|  |  |  | 			if (equal) { | 
					
						
							|  |  |  | 				color = Color(0.7, 0.7, 0.7); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-09 12:47:10 +03:00
										 |  |  | 		Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size); | 
					
						
							| 
									
										
										
										
											2023-06-21 17:43:45 +02:00
										 |  |  | 		p_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)); | 
					
						
							|  |  |  | 		p_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataDefaultEditor::setup_property_editor(Variant::Type p_type, const String &p_property, const String &p_label, const Variant &p_default_value) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	ERR_FAIL_COND_MSG(!property.is_empty(), "Cannot setup TileDataDefaultEditor twice"); | 
					
						
							|  |  |  | 	property = p_property; | 
					
						
							| 
									
										
										
										
											2022-09-28 19:17:53 +02:00
										 |  |  | 	property_type = p_type; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Update everything.
 | 
					
						
							|  |  |  | 	if (property_editor) { | 
					
						
							| 
									
										
										
										
											2022-10-24 22:07:02 +01:00
										 |  |  | 		property_editor->queue_free(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Update the dummy object.
 | 
					
						
							|  |  |  | 	dummy_object->add_dummy_property(p_property); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Get the default value for the type.
 | 
					
						
							|  |  |  | 	if (p_default_value == Variant()) { | 
					
						
							|  |  |  | 		Callable::CallError error; | 
					
						
							|  |  |  | 		Variant painted_value; | 
					
						
							|  |  |  | 		Variant::construct(p_type, painted_value, nullptr, 0, error); | 
					
						
							|  |  |  | 		dummy_object->set(p_property, painted_value); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		dummy_object->set(p_property, p_default_value); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create and setup the property editor.
 | 
					
						
							|  |  |  | 	property_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, p_type, p_property, PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT); | 
					
						
							|  |  |  | 	property_editor->set_object_and_property(dummy_object, p_property); | 
					
						
							|  |  |  | 	if (p_label.is_empty()) { | 
					
						
							| 
									
										
										
										
											2023-10-05 21:12:10 +08:00
										 |  |  | 		property_editor->set_label(EditorPropertyNameProcessor::get_singleton()->process_name(p_property, EditorPropertyNameProcessor::get_default_inspector_style(), p_property)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		property_editor->set_label(p_label); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	property_editor->connect("property_changed", callable_mp(this, &TileDataDefaultEditor::_property_value_changed).unbind(1)); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 	property_editor->set_tooltip_text(p_property); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	property_editor->update_property(); | 
					
						
							|  |  |  | 	add_child(property_editor); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataDefaultEditor::_notification(int p_what) { | 
					
						
							|  |  |  | 	switch (p_what) { | 
					
						
							| 
									
										
										
										
											2022-08-29 11:04:31 +02:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		case NOTIFICATION_THEME_CHANGED: { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 			picker_button->set_icon(get_editor_theme_icon(SNAME("ColorPick"))); | 
					
						
							|  |  |  | 			tile_bool_checked = get_editor_theme_icon(SNAME("TileChecked")); | 
					
						
							|  |  |  | 			tile_bool_unchecked = get_editor_theme_icon(SNAME("TileUnchecked")); | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-28 19:17:53 +02:00
										 |  |  | Variant::Type TileDataDefaultEditor::get_property_type() { | 
					
						
							|  |  |  | 	return property_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | TileDataDefaultEditor::TileDataDefaultEditor() { | 
					
						
							|  |  |  | 	label = memnew(Label); | 
					
						
							| 
									
										
										
										
											2021-07-19 15:10:05 -04:00
										 |  |  | 	label->set_text(TTR("Painting:")); | 
					
						
							| 
									
										
										
										
											2022-06-19 21:55:52 -05:00
										 |  |  | 	label->set_theme_type_variation("HeaderSmall"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	add_child(label); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	picker_button = memnew(Button); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	picker_button->set_theme_type_variation("FlatButton"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	picker_button->set_toggle_mode(true); | 
					
						
							| 
									
										
										
										
											2024-05-21 22:11:50 +02:00
										 |  |  | 	picker_button->set_shortcut(ED_GET_SHORTCUT("tiles_editor/picker")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	toolbar->add_child(picker_button); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileDataDefaultEditor::~TileDataDefaultEditor() { | 
					
						
							| 
									
										
										
										
											2022-10-24 22:07:02 +01:00
										 |  |  | 	toolbar->queue_free(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	memdelete(dummy_object); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | void TileDataTextureOriginEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Vector2i tile_set_tile_size = tile_set->get_tile_size(); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 	Color color = Color(1.0, 1.0, 1.0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	if (p_selected) { | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 		Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		Color 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); | 
					
						
							|  |  |  | 		color = selection_color; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	TileSetSource *source = *(tile_set->get_source(p_cell.source_id)); | 
					
						
							|  |  |  | 	TileSetAtlasSource *atlas_source = Object::cast_to<TileSetAtlasSource>(source); | 
					
						
							| 
									
										
										
										
											2024-08-09 19:26:02 +02:00
										 |  |  | 	if (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))) { | 
					
						
							|  |  |  | 		tile_set->draw_tile_shape(p_canvas_item, p_transform.scaled_local(tile_set_tile_size), color); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (atlas_source->is_position_in_tile_texture_region(p_cell.get_atlas_coords(), p_cell.alternative_tile, Vector2())) { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 		Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_editor_theme_icon(SNAME("EditorPosition")); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 		p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2()) - (position_icon->get_size() / 2), color); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 		Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts)); | 
					
						
							|  |  |  | 		int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts)); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 		Vector2 texture_origin = tile_data->get_texture_origin(); | 
					
						
							|  |  |  | 		String text = vformat("%s", texture_origin); | 
					
						
							|  |  |  | 		Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size); | 
					
						
							|  |  |  | 		p_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)); | 
					
						
							|  |  |  | 		p_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); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataPositionEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							|  |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	bool valid; | 
					
						
							|  |  |  | 	Variant value = tile_data->get(property, &valid); | 
					
						
							|  |  |  | 	if (!valid) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ERR_FAIL_COND(value.get_type() != Variant::VECTOR2I && value.get_type() != Variant::VECTOR2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Color color = Color(1.0, 1.0, 1.0); | 
					
						
							|  |  |  | 	if (p_selected) { | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 		Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		Color 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); | 
					
						
							|  |  |  | 		color = selection_color; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 	Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_editor_theme_icon(SNAME("EditorPosition")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2(value)) - position_icon->get_size() / 2, color); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataYSortEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							|  |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Color color = Color(1.0, 1.0, 1.0); | 
					
						
							|  |  |  | 	if (p_selected) { | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 		Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		Color 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); | 
					
						
							|  |  |  | 		color = selection_color; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 	Vector2 texture_origin = tile_data->get_texture_origin(); | 
					
						
							|  |  |  | 	TileSetSource *source = *(tile_set->get_source(p_cell.source_id)); | 
					
						
							|  |  |  | 	TileSetAtlasSource *atlas_source = Object::cast_to<TileSetAtlasSource>(source); | 
					
						
							|  |  |  | 	if (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()))) { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 		Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_editor_theme_icon(SNAME("EditorPosition")); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 		p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2(0, tile_data->get_y_sort_origin())) - position_icon->get_size() / 2, color); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 		Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts)); | 
					
						
							|  |  |  | 		int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts)); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 		String text = vformat("%s", tile_data->get_y_sort_origin()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size); | 
					
						
							|  |  |  | 		p_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)); | 
					
						
							|  |  |  | 		p_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); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataOcclusionShapeEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							|  |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 	Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Color 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); | 
					
						
							|  |  |  | 	Color color = grid_color.darkened(0.2); | 
					
						
							|  |  |  | 	if (p_selected) { | 
					
						
							|  |  |  | 		color = selection_color.darkened(0.2); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	color.a *= 0.5; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Vector<Color> debug_occlusion_color; | 
					
						
							|  |  |  | 	debug_occlusion_color.push_back(color); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform); | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 	for (int i = 0; i < tile_data->get_occluder_polygons_count(occlusion_layer); i++) { | 
					
						
							|  |  |  | 		Ref<OccluderPolygon2D> occluder = tile_data->get_occluder_polygon(occlusion_layer, i); | 
					
						
							|  |  |  | 		if (occluder.is_valid() && occluder->get_polygon().size() >= 3) { | 
					
						
							|  |  |  | 			p_canvas_item->draw_polygon(Variant(occluder->get_polygon()), debug_occlusion_color); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant TileDataOcclusionShapeEditor::_get_painted_value() { | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 	Array polygons; | 
					
						
							|  |  |  | 	for (int i = 0; i < polygon_editor->get_polygon_count(); i++) { | 
					
						
							|  |  |  | 		Ref<OccluderPolygon2D> occluder_polygon; | 
					
						
							| 
									
										
										
										
											2023-11-09 12:37:21 +01:00
										 |  |  | 		occluder_polygon.instantiate(); | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 		occluder_polygon->set_polygon(polygon_editor->get_polygon(i)); | 
					
						
							|  |  |  | 		polygons.push_back(occluder_polygon); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 	return polygons; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataOcclusionShapeEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	polygon_editor->clear_polygons(); | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 	for (int i = 0; i < tile_data->get_occluder_polygons_count(occlusion_layer); i++) { | 
					
						
							|  |  |  | 		Ref<OccluderPolygon2D> occluder_polygon = tile_data->get_occluder_polygon(occlusion_layer, i); | 
					
						
							|  |  |  | 		if (occluder_polygon.is_valid()) { | 
					
						
							|  |  |  | 			polygon_editor->add_polygon(occluder_polygon->get_polygon()); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataOcclusionShapeEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Array polygons = p_value; | 
					
						
							|  |  |  | 	tile_data->set_occluder_polygons_count(occlusion_layer, polygons.size()); | 
					
						
							|  |  |  | 	for (int i = 0; i < polygons.size(); i++) { | 
					
						
							|  |  |  | 		Ref<OccluderPolygon2D> occluder_polygon = polygons[i]; | 
					
						
							|  |  |  | 		tile_data->set_occluder_polygon(occlusion_layer, i, occluder_polygon); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant TileDataOcclusionShapeEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL_V(tile_data, Variant()); | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 	Array polygons; | 
					
						
							|  |  |  | 	for (int i = 0; i < tile_data->get_occluder_polygons_count(occlusion_layer); i++) { | 
					
						
							|  |  |  | 		polygons.push_back(tile_data->get_occluder_polygon(occlusion_layer, i)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return polygons; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataOcclusionShapeEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 	EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 	for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) { | 
					
						
							|  |  |  | 		Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							|  |  |  | 		undo_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); | 
					
						
							|  |  |  | 		undo_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataOcclusionShapeEditor::_tile_set_changed() { | 
					
						
							|  |  |  | 	polygon_editor->set_tile_set(tile_set); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataOcclusionShapeEditor::_notification(int p_what) { | 
					
						
							|  |  |  | 	switch (p_what) { | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			polygon_editor->set_polygons_color(get_tree()->get_debug_collisions_color()); | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileDataOcclusionShapeEditor::TileDataOcclusionShapeEditor() { | 
					
						
							|  |  |  | 	polygon_editor = memnew(GenericTilePolygonEditor); | 
					
						
							| 
									
										
										
										
											2024-06-11 12:59:55 +02:00
										 |  |  | 	polygon_editor->set_multiple_polygon_mode(true); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	add_child(polygon_editor); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataCollisionEditor::_property_value_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	dummy_object->set(p_property, p_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-01 16:47:20 -03:00
										 |  |  | void TileDataCollisionEditor::_property_selected(const StringName &p_path, int p_focusable) { | 
					
						
							| 
									
										
										
										
											2021-12-26 16:26:58 +01:00
										 |  |  | 	// Deselect all other properties
 | 
					
						
							|  |  |  | 	for (KeyValue<StringName, EditorProperty *> &editor : property_editors) { | 
					
						
							|  |  |  | 		if (editor.key != p_path) { | 
					
						
							|  |  |  | 			editor.value->deselect(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataCollisionEditor::_polygons_changed() { | 
					
						
							|  |  |  | 	// Update the dummy object properties and their editors.
 | 
					
						
							|  |  |  | 	for (int i = 0; i < polygon_editor->get_polygon_count(); i++) { | 
					
						
							|  |  |  | 		StringName one_way_property = vformat("polygon_%d_one_way", i); | 
					
						
							|  |  |  | 		StringName one_way_margin_property = vformat("polygon_%d_one_way_margin", i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!dummy_object->has_dummy_property(one_way_property)) { | 
					
						
							|  |  |  | 			dummy_object->add_dummy_property(one_way_property); | 
					
						
							|  |  |  | 			dummy_object->set(one_way_property, false); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!dummy_object->has_dummy_property(one_way_margin_property)) { | 
					
						
							|  |  |  | 			dummy_object->add_dummy_property(one_way_margin_property); | 
					
						
							|  |  |  | 			dummy_object->set(one_way_margin_property, 1.0); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!property_editors.has(one_way_property)) { | 
					
						
							|  |  |  | 			EditorProperty *one_way_property_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::BOOL, one_way_property, PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT); | 
					
						
							|  |  |  | 			one_way_property_editor->set_object_and_property(dummy_object, one_way_property); | 
					
						
							|  |  |  | 			one_way_property_editor->set_label(one_way_property); | 
					
						
							|  |  |  | 			one_way_property_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1)); | 
					
						
							| 
									
										
										
										
											2021-12-26 16:26:58 +01:00
										 |  |  | 			one_way_property_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected)); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 			one_way_property_editor->set_tooltip_text(one_way_property_editor->get_edited_property()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			one_way_property_editor->update_property(); | 
					
						
							|  |  |  | 			add_child(one_way_property_editor); | 
					
						
							|  |  |  | 			property_editors[one_way_property] = one_way_property_editor; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!property_editors.has(one_way_margin_property)) { | 
					
						
							|  |  |  | 			EditorProperty *one_way_margin_property_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::FLOAT, one_way_margin_property, PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT); | 
					
						
							|  |  |  | 			one_way_margin_property_editor->set_object_and_property(dummy_object, one_way_margin_property); | 
					
						
							|  |  |  | 			one_way_margin_property_editor->set_label(one_way_margin_property); | 
					
						
							|  |  |  | 			one_way_margin_property_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1)); | 
					
						
							| 
									
										
										
										
											2021-12-26 16:26:58 +01:00
										 |  |  | 			one_way_margin_property_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected)); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 			one_way_margin_property_editor->set_tooltip_text(one_way_margin_property_editor->get_edited_property()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			one_way_margin_property_editor->update_property(); | 
					
						
							|  |  |  | 			add_child(one_way_margin_property_editor); | 
					
						
							|  |  |  | 			property_editors[one_way_margin_property] = one_way_margin_property_editor; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
											
												Fix various typos with codespell
Found via `codespell -q 3 -S ./thirdparty,*.po,./DONORS.md -L ackward,ang,ans,ba,beng,cas,childs,childrens,dof,doubleclick,fave,findn,hist,inout,leapyear,lod,nd,numer,ois,ony,paket,seeked,sinc,switchs,te,uint`
											
										 
											2021-07-07 11:17:32 -04:00
										 |  |  | 	// Remove unneeded properties and their editors.
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	for (int i = polygon_editor->get_polygon_count(); dummy_object->has_dummy_property(vformat("polygon_%d_one_way", i)); i++) { | 
					
						
							|  |  |  | 		dummy_object->remove_dummy_property(vformat("polygon_%d_one_way", i)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for (int i = polygon_editor->get_polygon_count(); dummy_object->has_dummy_property(vformat("polygon_%d_one_way_margin", i)); i++) { | 
					
						
							|  |  |  | 		dummy_object->remove_dummy_property(vformat("polygon_%d_one_way_margin", i)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for (int i = polygon_editor->get_polygon_count(); property_editors.has(vformat("polygon_%d_one_way", i)); i++) { | 
					
						
							| 
									
										
										
										
											2022-10-24 22:07:02 +01:00
										 |  |  | 		property_editors[vformat("polygon_%d_one_way", i)]->queue_free(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		property_editors.erase(vformat("polygon_%d_one_way", i)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for (int i = polygon_editor->get_polygon_count(); property_editors.has(vformat("polygon_%d_one_way_margin", i)); i++) { | 
					
						
							| 
									
										
										
										
											2022-10-24 22:07:02 +01:00
										 |  |  | 		property_editors[vformat("polygon_%d_one_way_margin", i)]->queue_free(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		property_editors.erase(vformat("polygon_%d_one_way_margin", i)); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | Variant TileDataCollisionEditor::_get_painted_value() { | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	Dictionary dict; | 
					
						
							|  |  |  | 	dict["linear_velocity"] = dummy_object->get("linear_velocity"); | 
					
						
							|  |  |  | 	dict["angular_velocity"] = dummy_object->get("angular_velocity"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Array array; | 
					
						
							|  |  |  | 	for (int i = 0; i < polygon_editor->get_polygon_count(); i++) { | 
					
						
							|  |  |  | 		ERR_FAIL_COND_V(polygon_editor->get_polygon(i).size() < 3, Variant()); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 		Dictionary polygon_dict; | 
					
						
							|  |  |  | 		polygon_dict["points"] = polygon_editor->get_polygon(i); | 
					
						
							|  |  |  | 		polygon_dict["one_way"] = dummy_object->get(vformat("polygon_%d_one_way", i)); | 
					
						
							|  |  |  | 		polygon_dict["one_way_margin"] = dummy_object->get(vformat("polygon_%d_one_way_margin", i)); | 
					
						
							|  |  |  | 		array.push_back(polygon_dict); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	dict["polygons"] = array; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	return dict; | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataCollisionEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	polygon_editor->clear_polygons(); | 
					
						
							|  |  |  | 	for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) { | 
					
						
							|  |  |  | 		Vector<Vector2> polygon = tile_data->get_collision_polygon_points(physics_layer, i); | 
					
						
							|  |  |  | 		if (polygon.size() >= 3) { | 
					
						
							|  |  |  | 			polygon_editor->add_polygon(polygon); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	_polygons_changed(); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	dummy_object->set("linear_velocity", tile_data->get_constant_linear_velocity(physics_layer)); | 
					
						
							|  |  |  | 	dummy_object->set("angular_velocity", tile_data->get_constant_angular_velocity(physics_layer)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) { | 
					
						
							|  |  |  | 		dummy_object->set(vformat("polygon_%d_one_way", i), tile_data->is_collision_polygon_one_way(physics_layer, i)); | 
					
						
							|  |  |  | 		dummy_object->set(vformat("polygon_%d_one_way_margin", i), tile_data->get_collision_polygon_one_way_margin(physics_layer, i)); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 	for (const KeyValue<StringName, EditorProperty *> &E : property_editors) { | 
					
						
							|  |  |  | 		E.value->update_property(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataCollisionEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	Dictionary dict = p_value; | 
					
						
							|  |  |  | 	tile_data->set_constant_linear_velocity(physics_layer, dict["linear_velocity"]); | 
					
						
							|  |  |  | 	tile_data->set_constant_angular_velocity(physics_layer, dict["angular_velocity"]); | 
					
						
							|  |  |  | 	Array array = dict["polygons"]; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	tile_data->set_collision_polygons_count(physics_layer, array.size()); | 
					
						
							|  |  |  | 	for (int i = 0; i < array.size(); i++) { | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 		Dictionary polygon_dict = array[i]; | 
					
						
							|  |  |  | 		tile_data->set_collision_polygon_points(physics_layer, i, polygon_dict["points"]); | 
					
						
							|  |  |  | 		tile_data->set_collision_polygon_one_way(physics_layer, i, polygon_dict["one_way"]); | 
					
						
							|  |  |  | 		tile_data->set_collision_polygon_one_way_margin(physics_layer, i, polygon_dict["one_way_margin"]); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | Variant TileDataCollisionEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL_V(tile_data, Variant()); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	Dictionary dict; | 
					
						
							|  |  |  | 	dict["linear_velocity"] = tile_data->get_constant_linear_velocity(physics_layer); | 
					
						
							|  |  |  | 	dict["angular_velocity"] = tile_data->get_constant_angular_velocity(physics_layer); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Array array; | 
					
						
							|  |  |  | 	for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) { | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 		Dictionary polygon_dict; | 
					
						
							|  |  |  | 		polygon_dict["points"] = tile_data->get_collision_polygon_points(physics_layer, i); | 
					
						
							|  |  |  | 		polygon_dict["one_way"] = tile_data->is_collision_polygon_one_way(physics_layer, i); | 
					
						
							|  |  |  | 		polygon_dict["one_way_margin"] = tile_data->get_collision_polygon_one_way_margin(physics_layer, i); | 
					
						
							|  |  |  | 		array.push_back(polygon_dict); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	dict["polygons"] = array; | 
					
						
							|  |  |  | 	return dict; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataCollisionEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) { | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 	Dictionary new_dict = p_new_value; | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 	EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-08-01 16:47:20 -03:00
										 |  |  | 	for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) { | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 		Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		Dictionary old_dict = E.value; | 
					
						
							|  |  |  | 		undo_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"]); | 
					
						
							|  |  |  | 		undo_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"]); | 
					
						
							|  |  |  | 		Array old_polygon_array = old_dict["polygons"]; | 
					
						
							|  |  |  | 		undo_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()); | 
					
						
							|  |  |  | 		for (int i = 0; i < old_polygon_array.size(); i++) { | 
					
						
							|  |  |  | 			Dictionary polygon_dict = old_polygon_array[i]; | 
					
						
							|  |  |  | 			undo_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"]); | 
					
						
							|  |  |  | 			undo_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"]); | 
					
						
							|  |  |  | 			undo_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"]); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 		undo_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"]); | 
					
						
							|  |  |  | 		undo_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"]); | 
					
						
							|  |  |  | 		Array new_polygon_array = new_dict["polygons"]; | 
					
						
							|  |  |  | 		undo_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()); | 
					
						
							|  |  |  | 		for (int i = 0; i < new_polygon_array.size(); i++) { | 
					
						
							|  |  |  | 			Dictionary polygon_dict = new_polygon_array[i]; | 
					
						
							|  |  |  | 			undo_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"]); | 
					
						
							|  |  |  | 			undo_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"]); | 
					
						
							|  |  |  | 			undo_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"]); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataCollisionEditor::_tile_set_changed() { | 
					
						
							|  |  |  | 	polygon_editor->set_tile_set(tile_set); | 
					
						
							|  |  |  | 	_polygons_changed(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataCollisionEditor::_notification(int p_what) { | 
					
						
							|  |  |  | 	switch (p_what) { | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			polygon_editor->set_polygons_color(get_tree()->get_debug_collisions_color()); | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | TileDataCollisionEditor::TileDataCollisionEditor() { | 
					
						
							|  |  |  | 	polygon_editor = memnew(GenericTilePolygonEditor); | 
					
						
							|  |  |  | 	polygon_editor->set_multiple_polygon_mode(true); | 
					
						
							|  |  |  | 	polygon_editor->connect("polygons_changed", callable_mp(this, &TileDataCollisionEditor::_polygons_changed)); | 
					
						
							|  |  |  | 	add_child(polygon_editor); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	dummy_object->add_dummy_property("linear_velocity"); | 
					
						
							|  |  |  | 	dummy_object->set("linear_velocity", Vector2()); | 
					
						
							|  |  |  | 	dummy_object->add_dummy_property("angular_velocity"); | 
					
						
							|  |  |  | 	dummy_object->set("angular_velocity", 0.0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	EditorProperty *linear_velocity_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::VECTOR2, "linear_velocity", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT); | 
					
						
							|  |  |  | 	linear_velocity_editor->set_object_and_property(dummy_object, "linear_velocity"); | 
					
						
							|  |  |  | 	linear_velocity_editor->set_label("linear_velocity"); | 
					
						
							|  |  |  | 	linear_velocity_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1)); | 
					
						
							| 
									
										
										
										
											2021-12-26 16:26:58 +01:00
										 |  |  | 	linear_velocity_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected)); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 	linear_velocity_editor->set_tooltip_text(linear_velocity_editor->get_edited_property()); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	linear_velocity_editor->update_property(); | 
					
						
							|  |  |  | 	add_child(linear_velocity_editor); | 
					
						
							|  |  |  | 	property_editors["linear_velocity"] = linear_velocity_editor; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	EditorProperty *angular_velocity_editor = EditorInspectorDefaultPlugin::get_editor_for_property(dummy_object, Variant::FLOAT, "angular_velocity", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_DEFAULT); | 
					
						
							|  |  |  | 	angular_velocity_editor->set_object_and_property(dummy_object, "angular_velocity"); | 
					
						
							|  |  |  | 	angular_velocity_editor->set_label("angular_velocity"); | 
					
						
							|  |  |  | 	angular_velocity_editor->connect("property_changed", callable_mp(this, &TileDataCollisionEditor::_property_value_changed).unbind(1)); | 
					
						
							| 
									
										
										
										
											2021-12-26 16:26:58 +01:00
										 |  |  | 	angular_velocity_editor->connect("selected", callable_mp(this, &TileDataCollisionEditor::_property_selected)); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 	angular_velocity_editor->set_tooltip_text(angular_velocity_editor->get_edited_property()); | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 	angular_velocity_editor->update_property(); | 
					
						
							|  |  |  | 	add_child(angular_velocity_editor); | 
					
						
							| 
									
										
										
										
											2021-10-19 11:40:46 +02:00
										 |  |  | 	property_editors["angular_velocity"] = angular_velocity_editor; | 
					
						
							| 
									
										
										
										
											2021-09-15 15:23:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	_polygons_changed(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileDataCollisionEditor::~TileDataCollisionEditor() { | 
					
						
							|  |  |  | 	memdelete(dummy_object); | 
					
						
							| 
									
										
										
										
											2021-05-21 10:08:15 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataCollisionEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							|  |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-05-21 10:08:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	// Draw all shapes.
 | 
					
						
							|  |  |  | 	Vector<Color> color; | 
					
						
							|  |  |  | 	if (p_selected) { | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 		Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		Color 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); | 
					
						
							|  |  |  | 		selection_color.a = 0.7; | 
					
						
							|  |  |  | 		color.push_back(selection_color); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		Color debug_collision_color = p_canvas_item->get_tree()->get_debug_collisions_color(); | 
					
						
							|  |  |  | 		color.push_back(debug_collision_color); | 
					
						
							| 
									
										
										
										
											2021-05-21 10:08:15 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform); | 
					
						
							| 
									
										
										
										
											2022-10-12 00:53:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 	Ref<Texture2D> one_way_icon = get_editor_theme_icon(SNAME("OneWayTile")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	for (int i = 0; i < tile_data->get_collision_polygons_count(physics_layer); i++) { | 
					
						
							|  |  |  | 		Vector<Vector2> polygon = tile_data->get_collision_polygon_points(physics_layer, i); | 
					
						
							| 
									
										
										
										
											2022-10-12 00:53:53 +02:00
										 |  |  | 		if (polygon.size() < 3) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p_canvas_item->draw_polygon(polygon, color); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (tile_data->is_collision_polygon_one_way(physics_layer, i)) { | 
					
						
							|  |  |  | 			PackedVector2Array uvs; | 
					
						
							|  |  |  | 			uvs.resize(polygon.size()); | 
					
						
							|  |  |  | 			Vector2 size_1 = Vector2(1, 1) / tile_set->get_tile_size(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 				uvs.write[j] = polygon[j] * size_1 + Vector2(0.5, 0.5); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Vector<Color> color2; | 
					
						
							|  |  |  | 			color2.push_back(Color(1, 1, 1, 0.4)); | 
					
						
							|  |  |  | 			p_canvas_item->draw_polygon(polygon, color2, uvs, one_way_icon); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-10-12 00:53:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D()); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataTerrainsEditor::_update_terrain_selector() { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!tile_set.is_valid()); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	// Update the terrain set selector.
 | 
					
						
							|  |  |  | 	Vector<String> options; | 
					
						
							|  |  |  | 	options.push_back(String(TTR("No terrains")) + String(":-1")); | 
					
						
							|  |  |  | 	for (int i = 0; i < tile_set->get_terrain_sets_count(); i++) { | 
					
						
							|  |  |  | 		options.push_back(vformat("Terrain Set %d", i)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	terrain_set_property_editor->setup(options); | 
					
						
							|  |  |  | 	terrain_set_property_editor->update_property(); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	// Update the terrain selector.
 | 
					
						
							|  |  |  | 	int terrain_set = int(dummy_object->get("terrain_set")); | 
					
						
							|  |  |  | 	if (terrain_set == -1) { | 
					
						
							|  |  |  | 		terrain_property_editor->hide(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		options.clear(); | 
					
						
							|  |  |  | 		Vector<Vector<Ref<Texture2D>>> icons = tile_set->generate_terrains_icons(Size2(16, 16) * EDSCALE); | 
					
						
							|  |  |  | 		options.push_back(String(TTR("No terrain")) + String(":-1")); | 
					
						
							|  |  |  | 		for (int i = 0; i < tile_set->get_terrains_count(terrain_set); i++) { | 
					
						
							|  |  |  | 			String name = tile_set->get_terrain_name(terrain_set, i); | 
					
						
							|  |  |  | 			if (name.is_empty()) { | 
					
						
							|  |  |  | 				options.push_back(vformat("Terrain %d", i)); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				options.push_back(name); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		terrain_property_editor->setup(options); | 
					
						
							|  |  |  | 		terrain_property_editor->update_property(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Kind of a hack to set icons.
 | 
					
						
							|  |  |  | 		// We could provide a way to modify that in the EditorProperty.
 | 
					
						
							|  |  |  | 		OptionButton *option_button = Object::cast_to<OptionButton>(terrain_property_editor->get_child(0)); | 
					
						
							|  |  |  | 		for (int terrain = 0; terrain < tile_set->get_terrains_count(terrain_set); terrain++) { | 
					
						
							|  |  |  | 			option_button->set_item_icon(terrain + 1, icons[terrain_set][terrain]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		terrain_property_editor->show(); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataTerrainsEditor::_property_value_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	Variant old_value = dummy_object->get(p_property); | 
					
						
							|  |  |  | 	dummy_object->set(p_property, p_value); | 
					
						
							|  |  |  | 	if (p_property == "terrain_set") { | 
					
						
							|  |  |  | 		if (p_value != old_value) { | 
					
						
							|  |  |  | 			dummy_object->set("terrain", -1); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		_update_terrain_selector(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-07-17 18:22:52 -03:00
										 |  |  | 	emit_signal(SNAME("needs_redraw")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataTerrainsEditor::_tile_set_changed() { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!tile_set.is_valid()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Fix if wrong values are selected.
 | 
					
						
							| 
									
										
										
										
											2021-09-07 14:59:58 +02:00
										 |  |  | 	int terrain_set = int(dummy_object->get("terrain_set")); | 
					
						
							|  |  |  | 	if (terrain_set >= tile_set->get_terrain_sets_count()) { | 
					
						
							|  |  |  | 		terrain_set = -1; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		dummy_object->set("terrain_set", -1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (terrain_set >= 0) { | 
					
						
							| 
									
										
										
										
											2021-09-07 14:59:58 +02:00
										 |  |  | 		if (int(dummy_object->get("terrain")) >= tile_set->get_terrains_count(terrain_set)) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			dummy_object->set("terrain", -1); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_update_terrain_selector(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataTerrainsEditor::forward_draw_over_atlas(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform) { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!tile_set.is_valid()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the hovered terrain bit, or the whole tile if it has the wrong terrain set.
 | 
					
						
							|  |  |  | 	Vector2i hovered_coords = TileSetSource::INVALID_ATLAS_COORDS; | 
					
						
							|  |  |  | 	if (drag_type == DRAG_TYPE_NONE) { | 
					
						
							|  |  |  | 		Vector2i mouse_pos = p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position()); | 
					
						
							|  |  |  | 		hovered_coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mouse_pos); | 
					
						
							|  |  |  | 		hovered_coords = p_tile_set_atlas_source->get_tile_at_coords(hovered_coords); | 
					
						
							|  |  |  | 		if (hovered_coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 			TileData *tile_data = p_tile_set_atlas_source->get_tile_data(hovered_coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			int terrain_set = tile_data->get_terrain_set(); | 
					
						
							|  |  |  | 			Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(hovered_coords); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 			Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (terrain_set >= 0 && terrain_set == int(dummy_object->get("terrain_set"))) { | 
					
						
							|  |  |  | 				// Draw hovered bit.
 | 
					
						
							|  |  |  | 				Transform2D xform; | 
					
						
							|  |  |  | 				xform.set_origin(position); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Vector<Color> color; | 
					
						
							|  |  |  | 				color.push_back(Color(1.0, 1.0, 1.0, 0.5)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 				Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 				if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) { | 
					
						
							|  |  |  | 					p_canvas_item->draw_set_transform_matrix(p_transform * xform); | 
					
						
							|  |  |  | 					p_canvas_item->draw_polygon(polygon, color); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 					TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 					if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 						polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) { | 
					
						
							|  |  |  | 							p_canvas_item->draw_set_transform_matrix(p_transform * xform); | 
					
						
							|  |  |  | 							p_canvas_item->draw_polygon(polygon, color); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				// Draw hovered tile.
 | 
					
						
							| 
									
										
										
										
											2021-09-14 12:02:13 +02:00
										 |  |  | 				Transform2D tile_xform; | 
					
						
							|  |  |  | 				tile_xform.set_origin(position); | 
					
						
							|  |  |  | 				tile_xform.set_scale(tile_set->get_tile_size()); | 
					
						
							|  |  |  | 				tile_set->draw_tile_shape(p_canvas_item, p_transform * tile_xform, Color(1.0, 1.0, 1.0, 0.5), true); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Dim terrains with wrong terrain set.
 | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 	Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts)); | 
					
						
							|  |  |  | 	int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	for (int i = 0; i < p_tile_set_atlas_source->get_tiles_count(); i++) { | 
					
						
							|  |  |  | 		Vector2i coords = p_tile_set_atlas_source->get_tile_id(i); | 
					
						
							|  |  |  | 		if (coords != hovered_coords) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 			TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (tile_data->get_terrain_set() != int(dummy_object->get("terrain_set"))) { | 
					
						
							|  |  |  | 				// Dimming
 | 
					
						
							|  |  |  | 				p_canvas_item->draw_set_transform_matrix(p_transform); | 
					
						
							|  |  |  | 				Rect2i rect = p_tile_set_atlas_source->get_tile_texture_region(coords); | 
					
						
							|  |  |  | 				p_canvas_item->draw_rect(rect, Color(0.0, 0.0, 0.0, 0.3)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// Text
 | 
					
						
							|  |  |  | 				p_canvas_item->draw_set_transform_matrix(Transform2D()); | 
					
						
							|  |  |  | 				Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 				Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				Color color = Color(1, 1, 1); | 
					
						
							|  |  |  | 				String text; | 
					
						
							|  |  |  | 				if (tile_data->get_terrain_set() >= 0) { | 
					
						
							|  |  |  | 					text = vformat("%d", tile_data->get_terrain_set()); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					text = "-"; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2022-05-09 12:47:10 +03:00
										 |  |  | 				Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size); | 
					
						
							|  |  |  | 				p_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)); | 
					
						
							|  |  |  | 				p_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	p_canvas_item->draw_set_transform_matrix(Transform2D()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET_RECT) { | 
					
						
							|  |  |  | 		// Draw selection rectangle.
 | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 		Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		Color 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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p_canvas_item->draw_set_transform_matrix(p_transform); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Rect2i rect; | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 		rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true)); | 
					
						
							|  |  |  | 		rect.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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		rect = rect.abs(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 15:04:37 +02:00
										 |  |  | 		RBSet<TileMapCell> edited; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		for (int x = rect.get_position().x; x <= rect.get_end().x; x++) { | 
					
						
							|  |  |  | 			for (int y = rect.get_position().y; y <= rect.get_end().y; y++) { | 
					
						
							|  |  |  | 				Vector2i coords = Vector2i(x, y); | 
					
						
							|  |  |  | 				coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 				if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 					TileMapCell cell; | 
					
						
							|  |  |  | 					cell.source_id = 0; | 
					
						
							|  |  |  | 					cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 					cell.alternative_tile = 0; | 
					
						
							|  |  |  | 					edited.insert(cell); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-18 17:43:40 -06:00
										 |  |  | 		for (const TileMapCell &E : edited) { | 
					
						
							|  |  |  | 			Vector2i coords = E.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			p_canvas_item->draw_rect(p_tile_set_atlas_source->get_tile_texture_region(coords), selection_color, false); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		p_canvas_item->draw_set_transform_matrix(Transform2D()); | 
					
						
							|  |  |  | 	} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS_RECT) { | 
					
						
							|  |  |  | 		// Highlight selected peering bits.
 | 
					
						
							|  |  |  | 		Dictionary painted = Dictionary(drag_painted_value); | 
					
						
							|  |  |  | 		int terrain_set = int(painted["terrain_set"]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Rect2i rect; | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 		rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true)); | 
					
						
							|  |  |  | 		rect.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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		rect = rect.abs(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 15:04:37 +02:00
										 |  |  | 		RBSet<TileMapCell> edited; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		for (int x = rect.get_position().x; x <= rect.get_end().x; x++) { | 
					
						
							|  |  |  | 			for (int y = rect.get_position().y; y <= rect.get_end().y; y++) { | 
					
						
							|  |  |  | 				Vector2i coords = Vector2i(x, y); | 
					
						
							|  |  |  | 				coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 				if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 					TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					if (tile_data->get_terrain_set() == terrain_set) { | 
					
						
							|  |  |  | 						TileMapCell cell; | 
					
						
							|  |  |  | 						cell.source_id = 0; | 
					
						
							|  |  |  | 						cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 						cell.alternative_tile = 0; | 
					
						
							|  |  |  | 						edited.insert(cell); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector2 end = p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position()); | 
					
						
							|  |  |  | 		Vector<Point2> mouse_pos_rect_polygon; | 
					
						
							|  |  |  | 		mouse_pos_rect_polygon.push_back(drag_start_pos); | 
					
						
							|  |  |  | 		mouse_pos_rect_polygon.push_back(Vector2(end.x, drag_start_pos.y)); | 
					
						
							|  |  |  | 		mouse_pos_rect_polygon.push_back(end); | 
					
						
							|  |  |  | 		mouse_pos_rect_polygon.push_back(Vector2(drag_start_pos.x, end.y)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector<Color> color; | 
					
						
							|  |  |  | 		color.push_back(Color(1.0, 1.0, 1.0, 0.5)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p_canvas_item->draw_set_transform_matrix(p_transform); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-18 17:43:40 -06:00
										 |  |  | 		for (const TileMapCell &E : edited) { | 
					
						
							|  |  |  | 			Vector2i coords = E.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 			Vector2i position = texture_region.get_center() + p_tile_set_atlas_source->get_tile_data(coords, 0)->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 			Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 			for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 				polygon.write[j] += position; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) { | 
					
						
							|  |  |  | 				// Draw terrain.
 | 
					
						
							|  |  |  | 				p_canvas_item->draw_polygon(polygon, color); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 				TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 				if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 					polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 						polygon.write[j] += position; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) { | 
					
						
							|  |  |  | 						// Draw bit.
 | 
					
						
							|  |  |  | 						p_canvas_item->draw_polygon(polygon, color); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		p_canvas_item->draw_set_transform_matrix(Transform2D()); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataTerrainsEditor::forward_draw_over_alternatives(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, CanvasItem *p_canvas_item, Transform2D p_transform) { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!tile_set.is_valid()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Draw the hovered terrain bit, or the whole tile if it has the wrong terrain set.
 | 
					
						
							|  |  |  | 	Vector2i hovered_coords = TileSetSource::INVALID_ATLAS_COORDS; | 
					
						
							|  |  |  | 	int hovered_alternative = TileSetSource::INVALID_TILE_ALTERNATIVE; | 
					
						
							|  |  |  | 	if (drag_type == DRAG_TYPE_NONE) { | 
					
						
							|  |  |  | 		Vector2i mouse_pos = p_transform.affine_inverse().xform(p_canvas_item->get_local_mouse_position()); | 
					
						
							|  |  |  | 		Vector3i hovered = p_tile_atlas_view->get_alternative_tile_at_pos(mouse_pos); | 
					
						
							|  |  |  | 		hovered_coords = Vector2i(hovered.x, hovered.y); | 
					
						
							|  |  |  | 		hovered_alternative = hovered.z; | 
					
						
							|  |  |  | 		if (hovered_coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 			TileData *tile_data = p_tile_set_atlas_source->get_tile_data(hovered_coords, hovered_alternative); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			int terrain_set = tile_data->get_terrain_set(); | 
					
						
							|  |  |  | 			Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(hovered_coords, hovered_alternative); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 			Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (terrain_set == int(dummy_object->get("terrain_set"))) { | 
					
						
							|  |  |  | 				// Draw hovered bit.
 | 
					
						
							|  |  |  | 				Transform2D xform; | 
					
						
							|  |  |  | 				xform.set_origin(position); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Vector<Color> color; | 
					
						
							|  |  |  | 				color.push_back(Color(1.0, 1.0, 1.0, 0.5)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 				Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 				if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) { | 
					
						
							|  |  |  | 					p_canvas_item->draw_set_transform_matrix(p_transform * xform); | 
					
						
							|  |  |  | 					p_canvas_item->draw_polygon(polygon, color); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 					TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 					if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 						polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						if (Geometry2D::is_point_in_polygon(xform.affine_inverse().xform(mouse_pos), polygon)) { | 
					
						
							|  |  |  | 							p_canvas_item->draw_set_transform_matrix(p_transform * xform); | 
					
						
							|  |  |  | 							p_canvas_item->draw_polygon(polygon, color); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				// Draw hovered tile.
 | 
					
						
							| 
									
										
										
										
											2021-09-14 12:02:13 +02:00
										 |  |  | 				Transform2D tile_xform; | 
					
						
							|  |  |  | 				tile_xform.set_origin(position); | 
					
						
							|  |  |  | 				tile_xform.set_scale(tile_set->get_tile_size()); | 
					
						
							|  |  |  | 				tile_set->draw_tile_shape(p_canvas_item, p_transform * tile_xform, Color(1.0, 1.0, 1.0, 0.5), true); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Dim terrains with wrong terrain set.
 | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 	Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font(SNAME("bold"), EditorStringName(EditorFonts)); | 
					
						
							|  |  |  | 	int font_size = TileSetEditor::get_singleton()->get_theme_font_size(SNAME("bold_size"), EditorStringName(EditorFonts)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	for (int i = 0; i < p_tile_set_atlas_source->get_tiles_count(); i++) { | 
					
						
							|  |  |  | 		Vector2i coords = p_tile_set_atlas_source->get_tile_id(i); | 
					
						
							|  |  |  | 		for (int j = 1; j < p_tile_set_atlas_source->get_alternative_tiles_count(coords); j++) { | 
					
						
							|  |  |  | 			int alternative_tile = p_tile_set_atlas_source->get_alternative_tile_id(coords, j); | 
					
						
							|  |  |  | 			if (coords != hovered_coords || alternative_tile != hovered_alternative) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 				TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				if (tile_data->get_terrain_set() != int(dummy_object->get("terrain_set"))) { | 
					
						
							|  |  |  | 					// Dimming
 | 
					
						
							|  |  |  | 					p_canvas_item->draw_set_transform_matrix(p_transform); | 
					
						
							|  |  |  | 					Rect2i rect = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile); | 
					
						
							|  |  |  | 					p_canvas_item->draw_rect(rect, Color(0.0, 0.0, 0.0, 0.3)); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					// Text
 | 
					
						
							|  |  |  | 					p_canvas_item->draw_set_transform_matrix(Transform2D()); | 
					
						
							|  |  |  | 					Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 					Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					Color color = Color(1, 1, 1); | 
					
						
							|  |  |  | 					String text; | 
					
						
							|  |  |  | 					if (tile_data->get_terrain_set() >= 0) { | 
					
						
							|  |  |  | 						text = vformat("%d", tile_data->get_terrain_set()); | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						text = "-"; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2022-05-09 12:47:10 +03:00
										 |  |  | 					Vector2 string_size = font->get_string_size(text, HORIZONTAL_ALIGNMENT_LEFT, -1, font_size); | 
					
						
							|  |  |  | 					p_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)); | 
					
						
							|  |  |  | 					p_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	p_canvas_item->draw_set_transform_matrix(Transform2D()); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | void TileDataTerrainsEditor::forward_painting_atlas_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) { | 
					
						
							|  |  |  | 	Ref<InputEventMouseMotion> mm = p_event; | 
					
						
							|  |  |  | 	if (mm.is_valid()) { | 
					
						
							|  |  |  | 		if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) { | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 			Vector<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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			for (int i = 0; i < line.size(); i++) { | 
					
						
							|  |  |  | 				Vector2i coords = p_tile_set_atlas_source->get_tile_at_coords(line[i]); | 
					
						
							|  |  |  | 				if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 					int terrain_set = drag_painted_value; | 
					
						
							|  |  |  | 					TileMapCell cell; | 
					
						
							|  |  |  | 					cell.source_id = 0; | 
					
						
							|  |  |  | 					cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 					cell.alternative_tile = 0; | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					// Save the old terrain_set and terrains bits.
 | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 					TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					if (!drag_modified.has(cell)) { | 
					
						
							|  |  |  | 						Dictionary dict; | 
					
						
							|  |  |  | 						dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 						dict["terrain"] = tile_data->get_terrain(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						Array array; | 
					
						
							|  |  |  | 						for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(j); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						} | 
					
						
							|  |  |  | 						dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 						drag_modified[cell] = dict; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					// Set the terrain_set.
 | 
					
						
							|  |  |  | 					tile_data->set_terrain_set(terrain_set); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			drag_last_pos = mm->get_position(); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 			accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) { | 
					
						
							|  |  |  | 			int terrain_set = Dictionary(drag_painted_value)["terrain_set"]; | 
					
						
							|  |  |  | 			int terrain = Dictionary(drag_painted_value)["terrain"]; | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 			Vector<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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			for (int i = 0; i < line.size(); i++) { | 
					
						
							|  |  |  | 				Vector2i coords = p_tile_set_atlas_source->get_tile_at_coords(line[i]); | 
					
						
							|  |  |  | 				if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 					TileMapCell cell; | 
					
						
							|  |  |  | 					cell.source_id = 0; | 
					
						
							|  |  |  | 					cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 					cell.alternative_tile = 0; | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 					TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					if (tile_data->get_terrain_set() == terrain_set) { | 
					
						
							|  |  |  | 						// Save the old terrain_set and terrains bits.
 | 
					
						
							|  |  |  | 						if (!drag_modified.has(cell)) { | 
					
						
							|  |  |  | 							Dictionary dict; | 
					
						
							|  |  |  | 							dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							dict["terrain"] = tile_data->get_terrain(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							Array array; | 
					
						
							|  |  |  | 							for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) { | 
					
						
							|  |  |  | 								TileSet::CellNeighbor bit = TileSet::CellNeighbor(j); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 								array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							|  |  |  | 							dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 							drag_modified[cell] = dict; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						// Set the terrains bits.
 | 
					
						
							|  |  |  | 						Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 						Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						Vector<Vector2> polygon = tile_set->get_terrain_polygon(tile_data->get_terrain_set()); | 
					
						
							|  |  |  | 						if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) { | 
					
						
							|  |  |  | 							tile_data->set_terrain(terrain); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(j); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_data->is_valid_terrain_peering_bit(bit)) { | 
					
						
							|  |  |  | 								polygon = tile_set->get_terrain_peering_bit_polygon(tile_data->get_terrain_set(), bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) { | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 									tile_data->set_terrain_peering_bit(bit, terrain); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			drag_last_pos = mm->get_position(); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 			accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ref<InputEventMouseButton> mb = p_event; | 
					
						
							|  |  |  | 	if (mb.is_valid()) { | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 		if (mb->get_button_index() == MouseButton::LEFT || mb->get_button_index() == MouseButton::RIGHT) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (mb->is_pressed()) { | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 				if (picker_button->is_pressed() || (mb->is_command_or_control_pressed() && !mb->is_shift_pressed())) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position()); | 
					
						
							|  |  |  | 					coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 					if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 						TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						int terrain_set = tile_data->get_terrain_set(); | 
					
						
							|  |  |  | 						Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 						Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						dummy_object->set("terrain_set", terrain_set); | 
					
						
							|  |  |  | 						dummy_object->set("terrain", -1); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 						if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							|  |  |  | 							dummy_object->set("terrain", tile_data->get_terrain()); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 								polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 									dummy_object->set("terrain", tile_data->get_terrain_peering_bit(bit)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						terrain_set_property_editor->update_property(); | 
					
						
							|  |  |  | 						_update_terrain_selector(); | 
					
						
							|  |  |  | 						picker_button->set_pressed(false); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					Vector2i coords = p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position()); | 
					
						
							|  |  |  | 					coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 					TileData *tile_data = nullptr; | 
					
						
							|  |  |  | 					if (coords != TileSetAtlasSource::INVALID_ATLAS_COORDS) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 						tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					int terrain_set = int(dummy_object->get("terrain_set")); | 
					
						
							|  |  |  | 					int terrain = int(dummy_object->get("terrain")); | 
					
						
							|  |  |  | 					if (terrain_set == -1 || !tile_data || tile_data->get_terrain_set() != terrain_set) { | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						// Paint terrain sets.
 | 
					
						
							|  |  |  | 						if (mb->get_button_index() == MouseButton::RIGHT) { | 
					
						
							|  |  |  | 							terrain_set = -1; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 						if (mb->is_command_or_control_pressed() && mb->is_shift_pressed()) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							// Paint terrain set with rect.
 | 
					
						
							|  |  |  | 							drag_type = DRAG_TYPE_PAINT_TERRAIN_SET_RECT; | 
					
						
							|  |  |  | 							drag_modified.clear(); | 
					
						
							|  |  |  | 							drag_painted_value = terrain_set; | 
					
						
							|  |  |  | 							drag_start_pos = mb->get_position(); | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							// Paint terrain set.
 | 
					
						
							|  |  |  | 							drag_type = DRAG_TYPE_PAINT_TERRAIN_SET; | 
					
						
							|  |  |  | 							drag_modified.clear(); | 
					
						
							|  |  |  | 							drag_painted_value = terrain_set; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 								TileMapCell cell; | 
					
						
							|  |  |  | 								cell.source_id = 0; | 
					
						
							|  |  |  | 								cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 								cell.alternative_tile = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								// Save the old terrain_set and terrains bits.
 | 
					
						
							|  |  |  | 								Dictionary dict; | 
					
						
							|  |  |  | 								dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 								dict["terrain"] = tile_data->get_terrain(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								Array array; | 
					
						
							|  |  |  | 								for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 									TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 									array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								} | 
					
						
							|  |  |  | 								dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 								drag_modified[cell] = dict; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								// Set the terrain_set.
 | 
					
						
							|  |  |  | 								tile_data->set_terrain_set(terrain_set); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						accept_event(); | 
					
						
							| 
									
										
										
										
											2022-04-07 13:23:40 +03:00
										 |  |  | 					} else if (tile_data->get_terrain_set() == terrain_set) { | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						// Paint terrain bits.
 | 
					
						
							|  |  |  | 						if (mb->get_button_index() == MouseButton::RIGHT) { | 
					
						
							|  |  |  | 							terrain = -1; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2023-01-16 16:10:33 +01:00
										 |  |  | 						if (mb->is_command_or_control_pressed() && mb->is_shift_pressed()) { | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 							// Paint terrain bits with rect.
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							drag_type = DRAG_TYPE_PAINT_TERRAIN_BITS_RECT; | 
					
						
							|  |  |  | 							drag_modified.clear(); | 
					
						
							|  |  |  | 							Dictionary painted_dict; | 
					
						
							|  |  |  | 							painted_dict["terrain_set"] = terrain_set; | 
					
						
							|  |  |  | 							painted_dict["terrain"] = terrain; | 
					
						
							|  |  |  | 							drag_painted_value = painted_dict; | 
					
						
							|  |  |  | 							drag_start_pos = mb->get_position(); | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							// Paint terrain bits.
 | 
					
						
							|  |  |  | 							drag_type = DRAG_TYPE_PAINT_TERRAIN_BITS; | 
					
						
							|  |  |  | 							drag_modified.clear(); | 
					
						
							|  |  |  | 							Dictionary painted_dict; | 
					
						
							|  |  |  | 							painted_dict["terrain_set"] = terrain_set; | 
					
						
							|  |  |  | 							painted_dict["terrain"] = terrain; | 
					
						
							|  |  |  | 							drag_painted_value = painted_dict; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 								TileMapCell cell; | 
					
						
							|  |  |  | 								cell.source_id = 0; | 
					
						
							|  |  |  | 								cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 								cell.alternative_tile = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								// Save the old terrain_set and terrains bits.
 | 
					
						
							|  |  |  | 								Dictionary dict; | 
					
						
							|  |  |  | 								dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 								dict["terrain"] = tile_data->get_terrain(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								Array array; | 
					
						
							|  |  |  | 								for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 									TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 									array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								} | 
					
						
							|  |  |  | 								dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 								drag_modified[cell] = dict; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								// Set the terrain bit.
 | 
					
						
							|  |  |  | 								Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 								Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 								Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 								if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							|  |  |  | 									tile_data->set_terrain(terrain); | 
					
						
							|  |  |  | 								} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 									TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 									if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 										polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 										if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 											tile_data->set_terrain_peering_bit(bit, terrain); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 										} | 
					
						
							|  |  |  | 									} | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 				EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET_RECT) { | 
					
						
							|  |  |  | 					Rect2i rect; | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 					rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true)); | 
					
						
							|  |  |  | 					rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					rect = rect.abs(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 15:04:37 +02:00
										 |  |  | 					RBSet<TileMapCell> edited; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					for (int x = rect.get_position().x; x <= rect.get_end().x; x++) { | 
					
						
							|  |  |  | 						for (int y = rect.get_position().y; y <= rect.get_end().y; y++) { | 
					
						
							|  |  |  | 							Vector2i coords = Vector2i(x, y); | 
					
						
							|  |  |  | 							coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 							if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 								TileMapCell cell; | 
					
						
							|  |  |  | 								cell.source_id = 0; | 
					
						
							|  |  |  | 								cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 								cell.alternative_tile = 0; | 
					
						
							|  |  |  | 								edited.insert(cell); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Terrain Set")); | 
					
						
							| 
									
										
										
										
											2022-05-18 17:43:40 -06:00
										 |  |  | 					for (const TileMapCell &E : edited) { | 
					
						
							|  |  |  | 						Vector2i coords = E.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 						TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2022-05-18 17:43:40 -06:00
										 |  |  | 						undo_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); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						undo_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()); | 
					
						
							|  |  |  | 						if (tile_data->get_terrain_set() >= 0) { | 
					
						
							|  |  |  | 							undo_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()); | 
					
						
							|  |  |  | 							for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 								TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							|  |  |  | 								if (tile_data->is_valid_terrain_peering_bit(bit)) { | 
					
						
							|  |  |  | 									undo_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)); | 
					
						
							|  |  |  | 								} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->commit_action(true); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 					accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Terrain Set")); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 					for (KeyValue<TileMapCell, Variant> &E : drag_modified) { | 
					
						
							|  |  |  | 						Dictionary dict = E.value; | 
					
						
							|  |  |  | 						Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							|  |  |  | 						undo_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); | 
					
						
							|  |  |  | 						undo_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"]); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						if (int(dict["terrain_set"]) >= 0) { | 
					
						
							|  |  |  | 							undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]); | 
					
						
							|  |  |  | 							Array array = dict["terrain_peering_bits"]; | 
					
						
							|  |  |  | 							for (int i = 0; i < array.size(); i++) { | 
					
						
							|  |  |  | 								TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							|  |  |  | 								if (tile_set->is_valid_terrain_peering_bit(dict["terrain_set"], bit)) { | 
					
						
							|  |  |  | 									undo_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]); | 
					
						
							|  |  |  | 								} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->commit_action(false); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 					accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) { | 
					
						
							|  |  |  | 					Dictionary painted = Dictionary(drag_painted_value); | 
					
						
							|  |  |  | 					int terrain_set = int(painted["terrain_set"]); | 
					
						
							|  |  |  | 					int terrain = int(painted["terrain"]); | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Terrain")); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 					for (KeyValue<TileMapCell, Variant> &E : drag_modified) { | 
					
						
							|  |  |  | 						Dictionary dict = E.value; | 
					
						
							|  |  |  | 						Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 						undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), terrain); | 
					
						
							|  |  |  | 						undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						Array array = dict["terrain_peering_bits"]; | 
					
						
							|  |  |  | 						for (int i = 0; i < array.size(); i++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 								undo_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_set->is_valid_terrain_peering_bit(dict["terrain_set"], bit)) { | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 								undo_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]); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->commit_action(false); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 					accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS_RECT) { | 
					
						
							|  |  |  | 					Dictionary painted = Dictionary(drag_painted_value); | 
					
						
							|  |  |  | 					int terrain_set = int(painted["terrain_set"]); | 
					
						
							|  |  |  | 					int terrain = int(painted["terrain"]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					Rect2i rect; | 
					
						
							| 
									
										
										
										
											2023-01-17 10:26:10 +01:00
										 |  |  | 					rect.set_position(p_tile_atlas_view->get_atlas_tile_coords_at_pos(drag_start_pos, true)); | 
					
						
							|  |  |  | 					rect.set_end(p_tile_atlas_view->get_atlas_tile_coords_at_pos(mb->get_position(), true)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					rect = rect.abs(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 15:04:37 +02:00
										 |  |  | 					RBSet<TileMapCell> edited; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					for (int x = rect.get_position().x; x <= rect.get_end().x; x++) { | 
					
						
							|  |  |  | 						for (int y = rect.get_position().y; y <= rect.get_end().y; y++) { | 
					
						
							|  |  |  | 							Vector2i coords = Vector2i(x, y); | 
					
						
							|  |  |  | 							coords = p_tile_set_atlas_source->get_tile_at_coords(coords); | 
					
						
							|  |  |  | 							if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 								TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								if (tile_data->get_terrain_set() == terrain_set) { | 
					
						
							|  |  |  | 									TileMapCell cell; | 
					
						
							|  |  |  | 									cell.source_id = 0; | 
					
						
							|  |  |  | 									cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 									cell.alternative_tile = 0; | 
					
						
							|  |  |  | 									edited.insert(cell); | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					Vector<Point2> mouse_pos_rect_polygon; | 
					
						
							|  |  |  | 					mouse_pos_rect_polygon.push_back(drag_start_pos); | 
					
						
							|  |  |  | 					mouse_pos_rect_polygon.push_back(Vector2(mb->get_position().x, drag_start_pos.y)); | 
					
						
							|  |  |  | 					mouse_pos_rect_polygon.push_back(mb->get_position()); | 
					
						
							|  |  |  | 					mouse_pos_rect_polygon.push_back(Vector2(drag_start_pos.x, mb->get_position().y)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Terrain")); | 
					
						
							| 
									
										
										
										
											2022-05-18 17:43:40 -06:00
										 |  |  | 					for (const TileMapCell &E : edited) { | 
					
						
							|  |  |  | 						Vector2i coords = E.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 						TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, 0); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 						Rect2i texture_region = p_tile_set_atlas_source->get_tile_texture_region(coords); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 						Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 						for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 							polygon.write[j] += position; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) { | 
					
						
							|  |  |  | 							// Draw terrain.
 | 
					
						
							|  |  |  | 							undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.alternative_tile), terrain); | 
					
						
							|  |  |  | 							undo_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()); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 								polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 									polygon.write[j] += position; | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 								if (!Geometry2D::intersect_polygons(polygon, mouse_pos_rect_polygon).is_empty()) { | 
					
						
							|  |  |  | 									// Draw bit.
 | 
					
						
							| 
									
										
										
										
											2022-05-18 17:43:40 -06:00
										 |  |  | 									undo_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); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 									undo_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)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->commit_action(true); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 					accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataTerrainsEditor::forward_painting_alternatives_gui_input(TileAtlasView *p_tile_atlas_view, TileSetAtlasSource *p_tile_set_atlas_source, const Ref<InputEvent> &p_event) { | 
					
						
							|  |  |  | 	Ref<InputEventMouseMotion> mm = p_event; | 
					
						
							|  |  |  | 	if (mm.is_valid()) { | 
					
						
							|  |  |  | 		if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) { | 
					
						
							|  |  |  | 			Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mm->get_position()); | 
					
						
							|  |  |  | 			Vector2i coords = Vector2i(tile.x, tile.y); | 
					
						
							|  |  |  | 			int alternative_tile = tile.z; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 				TileMapCell cell; | 
					
						
							|  |  |  | 				cell.source_id = 0; | 
					
						
							|  |  |  | 				cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 				cell.alternative_tile = alternative_tile; | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 				TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				if (!drag_modified.has(cell)) { | 
					
						
							|  |  |  | 					Dictionary dict; | 
					
						
							|  |  |  | 					dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 					dict["terrain"] = tile_data->get_terrain(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					Array array; | 
					
						
							|  |  |  | 					for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) { | 
					
						
							|  |  |  | 						TileSet::CellNeighbor bit = TileSet::CellNeighbor(j); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 						array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 					drag_modified[cell] = dict; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				tile_data->set_terrain_set(drag_painted_value); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			drag_last_pos = mm->get_position(); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 			accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) { | 
					
						
							|  |  |  | 			Dictionary painted = Dictionary(drag_painted_value); | 
					
						
							|  |  |  | 			int terrain_set = int(painted["terrain_set"]); | 
					
						
							|  |  |  | 			int terrain = int(painted["terrain"]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mm->get_position()); | 
					
						
							|  |  |  | 			Vector2i coords = Vector2i(tile.x, tile.y); | 
					
						
							|  |  |  | 			int alternative_tile = tile.z; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 				TileMapCell cell; | 
					
						
							|  |  |  | 				cell.source_id = 0; | 
					
						
							|  |  |  | 				cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 				cell.alternative_tile = alternative_tile; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// Save the old terrain_set and terrains bits.
 | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 				TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				if (tile_data->get_terrain_set() == terrain_set) { | 
					
						
							|  |  |  | 					if (!drag_modified.has(cell)) { | 
					
						
							|  |  |  | 						Dictionary dict; | 
					
						
							|  |  |  | 						dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 						dict["terrain"] = tile_data->get_terrain(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						Array array; | 
					
						
							|  |  |  | 						for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(j); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						} | 
					
						
							|  |  |  | 						dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 						drag_modified[cell] = dict; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// Set the terrains bits.
 | 
					
						
							|  |  |  | 					Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 					Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					Vector<Vector2> polygon = tile_set->get_terrain_polygon(tile_data->get_terrain_set()); | 
					
						
							|  |  |  | 					if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) { | 
					
						
							|  |  |  | 						tile_data->set_terrain(terrain); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					for (int j = 0; j < TileSet::CELL_NEIGHBOR_MAX; j++) { | 
					
						
							|  |  |  | 						TileSet::CellNeighbor bit = TileSet::CellNeighbor(j); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 						if (tile_data->is_valid_terrain_peering_bit(bit)) { | 
					
						
							|  |  |  | 							polygon = tile_set->get_terrain_peering_bit_polygon(tile_data->get_terrain_set(), bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							if (Geometry2D::is_segment_intersecting_polygon(mm->get_position() - position, drag_last_pos - position, polygon)) { | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 								tile_data->set_terrain_peering_bit(bit, terrain); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			drag_last_pos = mm->get_position(); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 			accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ref<InputEventMouseButton> mb = p_event; | 
					
						
							|  |  |  | 	if (mb.is_valid()) { | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 		if (mb->get_button_index() == MouseButton::LEFT || mb->get_button_index() == MouseButton::RIGHT) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			if (mb->is_pressed()) { | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 				if (mb->get_button_index() == MouseButton::LEFT && picker_button->is_pressed()) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position()); | 
					
						
							|  |  |  | 					Vector2i coords = Vector2i(tile.x, tile.y); | 
					
						
							|  |  |  | 					int alternative_tile = tile.z; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 						TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						int terrain_set = tile_data->get_terrain_set(); | 
					
						
							|  |  |  | 						Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 						Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						dummy_object->set("terrain_set", terrain_set); | 
					
						
							|  |  |  | 						dummy_object->set("terrain", -1); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 						if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							|  |  |  | 							dummy_object->set("terrain", tile_data->get_terrain()); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 								polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 									dummy_object->set("terrain", tile_data->get_terrain_peering_bit(bit)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						terrain_set_property_editor->update_property(); | 
					
						
							|  |  |  | 						_update_terrain_selector(); | 
					
						
							|  |  |  | 						picker_button->set_pressed(false); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					int terrain_set = int(dummy_object->get("terrain_set")); | 
					
						
							|  |  |  | 					int terrain = int(dummy_object->get("terrain")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					Vector3i tile = p_tile_atlas_view->get_alternative_tile_at_pos(mb->get_position()); | 
					
						
							|  |  |  | 					Vector2i coords = Vector2i(tile.x, tile.y); | 
					
						
							|  |  |  | 					int alternative_tile = tile.z; | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 					if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 						TileData *tile_data = p_tile_set_atlas_source->get_tile_data(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						if (terrain_set == -1 || !tile_data || tile_data->get_terrain_set() != terrain_set) { | 
					
						
							|  |  |  | 							// Paint terrain sets.
 | 
					
						
							|  |  |  | 							drag_type = DRAG_TYPE_PAINT_TERRAIN_SET; | 
					
						
							|  |  |  | 							drag_modified.clear(); | 
					
						
							|  |  |  | 							drag_painted_value = int(dummy_object->get("terrain_set")); | 
					
						
							|  |  |  | 							if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 								TileMapCell cell; | 
					
						
							|  |  |  | 								cell.source_id = 0; | 
					
						
							|  |  |  | 								cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 								cell.alternative_tile = alternative_tile; | 
					
						
							|  |  |  | 								Dictionary dict; | 
					
						
							|  |  |  | 								dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							|  |  |  | 								Array array; | 
					
						
							|  |  |  | 								for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 									TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							|  |  |  | 									array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 								dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 								drag_modified[cell] = dict; | 
					
						
							|  |  |  | 								tile_data->set_terrain_set(drag_painted_value); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 							drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 							accept_event(); | 
					
						
							|  |  |  | 						} else if (tile_data->get_terrain_set() == terrain_set) { | 
					
						
							|  |  |  | 							// Paint terrain bits.
 | 
					
						
							|  |  |  | 							if (mb->get_button_index() == MouseButton::RIGHT) { | 
					
						
							|  |  |  | 								terrain = -1; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							// Paint terrain bits.
 | 
					
						
							|  |  |  | 							drag_type = DRAG_TYPE_PAINT_TERRAIN_BITS; | 
					
						
							|  |  |  | 							drag_modified.clear(); | 
					
						
							|  |  |  | 							Dictionary painted_dict; | 
					
						
							|  |  |  | 							painted_dict["terrain_set"] = terrain_set; | 
					
						
							|  |  |  | 							painted_dict["terrain"] = terrain; | 
					
						
							|  |  |  | 							drag_painted_value = painted_dict; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 							if (coords != TileSetSource::INVALID_ATLAS_COORDS) { | 
					
						
							|  |  |  | 								TileMapCell cell; | 
					
						
							|  |  |  | 								cell.source_id = 0; | 
					
						
							|  |  |  | 								cell.set_atlas_coords(coords); | 
					
						
							|  |  |  | 								cell.alternative_tile = alternative_tile; | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 								// Save the old terrain_set and terrains bits.
 | 
					
						
							|  |  |  | 								Dictionary dict; | 
					
						
							|  |  |  | 								dict["terrain_set"] = tile_data->get_terrain_set(); | 
					
						
							|  |  |  | 								dict["terrain"] = tile_data->get_terrain(); | 
					
						
							|  |  |  | 								Array array; | 
					
						
							|  |  |  | 								for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 									TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							|  |  |  | 									array.push_back(tile_data->is_valid_terrain_peering_bit(bit) ? tile_data->get_terrain_peering_bit(bit) : -1); | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 								dict["terrain_peering_bits"] = array; | 
					
						
							|  |  |  | 								drag_modified[cell] = dict; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								// Set the terrain bit.
 | 
					
						
							|  |  |  | 								Rect2i texture_region = p_tile_atlas_view->get_alternative_tile_rect(coords, alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-01-26 18:47:54 +01:00
										 |  |  | 								Vector2i position = texture_region.get_center() + tile_data->get_texture_origin(); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 								Vector<Vector2> polygon = tile_set->get_terrain_polygon(terrain_set); | 
					
						
							|  |  |  | 								if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							|  |  |  | 									tile_data->set_terrain(terrain); | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 								for (int i = 0; i < TileSet::CELL_NEIGHBOR_MAX; i++) { | 
					
						
							|  |  |  | 									TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							|  |  |  | 									if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							|  |  |  | 										polygon = tile_set->get_terrain_peering_bit_polygon(terrain_set, bit); | 
					
						
							|  |  |  | 										if (Geometry2D::is_point_in_polygon(mb->get_position() - position, polygon)) { | 
					
						
							|  |  |  | 											tile_data->set_terrain_peering_bit(bit, terrain); | 
					
						
							|  |  |  | 										} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 									} | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 							drag_last_pos = mb->get_position(); | 
					
						
							|  |  |  | 							accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 				EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				if (drag_type == DRAG_TYPE_PAINT_TERRAIN_SET) { | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Tiles Property")); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 					for (KeyValue<TileMapCell, Variant> &E : drag_modified) { | 
					
						
							|  |  |  | 						Dictionary dict = E.value; | 
					
						
							|  |  |  | 						Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							|  |  |  | 						undo_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); | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 						undo_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"]); | 
					
						
							|  |  |  | 						if (int(dict["terrain_set"]) >= 0) { | 
					
						
							|  |  |  | 							undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]); | 
					
						
							|  |  |  | 							Array array = dict["terrain_peering_bits"]; | 
					
						
							|  |  |  | 							for (int i = 0; i < array.size(); i++) { | 
					
						
							|  |  |  | 								undo_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]); | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->commit_action(false); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 					accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} else if (drag_type == DRAG_TYPE_PAINT_TERRAIN_BITS) { | 
					
						
							|  |  |  | 					Dictionary painted = Dictionary(drag_painted_value); | 
					
						
							|  |  |  | 					int terrain_set = int(painted["terrain_set"]); | 
					
						
							|  |  |  | 					int terrain = int(painted["terrain"]); | 
					
						
							|  |  |  | 					undo_redo->create_action(TTR("Painting Terrain")); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 					for (KeyValue<TileMapCell, Variant> &E : drag_modified) { | 
					
						
							|  |  |  | 						Dictionary dict = E.value; | 
					
						
							|  |  |  | 						Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 						undo_redo->add_do_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), terrain); | 
					
						
							|  |  |  | 						undo_redo->add_undo_property(p_tile_set_atlas_source, vformat("%d:%d/%d/terrain", coords.x, coords.y, E.key.alternative_tile), dict["terrain"]); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 						Array array = dict["terrain_peering_bits"]; | 
					
						
							|  |  |  | 						for (int i = 0; i < array.size(); i++) { | 
					
						
							|  |  |  | 							TileSet::CellNeighbor bit = TileSet::CellNeighbor(i); | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_set->is_valid_terrain_peering_bit(terrain_set, bit)) { | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 								undo_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							| 
									
										
										
										
											2022-02-23 17:25:50 +01:00
										 |  |  | 							if (tile_set->is_valid_terrain_peering_bit(dict["terrain_set"], bit)) { | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 								undo_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]); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					undo_redo->commit_action(false); | 
					
						
							|  |  |  | 					drag_type = DRAG_TYPE_NONE; | 
					
						
							| 
									
										
										
										
											2022-09-16 15:22:50 +02:00
										 |  |  | 					accept_event(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataTerrainsEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							|  |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	tile_set->draw_terrains(p_canvas_item, p_transform, tile_data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataTerrainsEditor::_notification(int p_what) { | 
					
						
							|  |  |  | 	switch (p_what) { | 
					
						
							| 
									
										
										
										
											2022-08-29 11:04:31 +02:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		case NOTIFICATION_THEME_CHANGED: { | 
					
						
							| 
									
										
										
										
											2023-08-13 02:33:39 +02:00
										 |  |  | 			picker_button->set_icon(get_editor_theme_icon(SNAME("ColorPick"))); | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileDataTerrainsEditor::TileDataTerrainsEditor() { | 
					
						
							|  |  |  | 	label = memnew(Label); | 
					
						
							| 
									
										
										
										
											2022-06-19 21:55:52 -05:00
										 |  |  | 	label->set_text(TTR("Painting:")); | 
					
						
							|  |  |  | 	label->set_theme_type_variation("HeaderSmall"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	add_child(label); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Toolbar
 | 
					
						
							|  |  |  | 	picker_button = memnew(Button); | 
					
						
							| 
									
										
										
										
											2023-09-19 18:03:10 +02:00
										 |  |  | 	picker_button->set_theme_type_variation("FlatButton"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	picker_button->set_toggle_mode(true); | 
					
						
							| 
									
										
										
										
											2024-05-21 22:11:50 +02:00
										 |  |  | 	picker_button->set_shortcut(ED_GET_SHORTCUT("tiles_editor/picker")); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	toolbar->add_child(picker_button); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Setup
 | 
					
						
							|  |  |  | 	dummy_object->add_dummy_property("terrain_set"); | 
					
						
							|  |  |  | 	dummy_object->set("terrain_set", -1); | 
					
						
							|  |  |  | 	dummy_object->add_dummy_property("terrain"); | 
					
						
							|  |  |  | 	dummy_object->set("terrain", -1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Get the default value for the type.
 | 
					
						
							|  |  |  | 	terrain_set_property_editor = memnew(EditorPropertyEnum); | 
					
						
							|  |  |  | 	terrain_set_property_editor->set_object_and_property(dummy_object, "terrain_set"); | 
					
						
							|  |  |  | 	terrain_set_property_editor->set_label("Terrain Set"); | 
					
						
							|  |  |  | 	terrain_set_property_editor->connect("property_changed", callable_mp(this, &TileDataTerrainsEditor::_property_value_changed).unbind(1)); | 
					
						
							| 
									
										
										
										
											2022-08-25 12:42:17 +02:00
										 |  |  | 	terrain_set_property_editor->set_tooltip_text(terrain_set_property_editor->get_edited_property()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	add_child(terrain_set_property_editor); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	terrain_property_editor = memnew(EditorPropertyEnum); | 
					
						
							|  |  |  | 	terrain_property_editor->set_object_and_property(dummy_object, "terrain"); | 
					
						
							|  |  |  | 	terrain_property_editor->set_label("Terrain"); | 
					
						
							|  |  |  | 	terrain_property_editor->connect("property_changed", callable_mp(this, &TileDataTerrainsEditor::_property_value_changed).unbind(1)); | 
					
						
							|  |  |  | 	add_child(terrain_property_editor); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileDataTerrainsEditor::~TileDataTerrainsEditor() { | 
					
						
							| 
									
										
										
										
											2022-10-24 22:07:02 +01:00
										 |  |  | 	toolbar->queue_free(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	memdelete(dummy_object); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant TileDataNavigationEditor::_get_painted_value() { | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 	Ref<NavigationPolygon> nav_polygon; | 
					
						
							|  |  |  | 	nav_polygon.instantiate(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-28 02:36:15 +02:00
										 |  |  | 	if (polygon_editor->get_polygon_count() > 0) { | 
					
						
							|  |  |  | 		Ref<NavigationMeshSourceGeometryData2D> source_geometry_data; | 
					
						
							|  |  |  | 		source_geometry_data.instantiate(); | 
					
						
							|  |  |  | 		for (int i = 0; i < polygon_editor->get_polygon_count(); i++) { | 
					
						
							|  |  |  | 			Vector<Vector2> polygon = polygon_editor->get_polygon(i); | 
					
						
							|  |  |  | 			nav_polygon->add_outline(polygon); | 
					
						
							|  |  |  | 			source_geometry_data->add_traversable_outline(polygon); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		nav_polygon->set_agent_radius(0.0); | 
					
						
							|  |  |  | 		NavigationServer2D::get_singleton()->bake_from_source_geometry_data(nav_polygon, source_geometry_data); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		nav_polygon->clear(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 	return nav_polygon; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataNavigationEditor::_set_painted_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 	Ref<NavigationPolygon> nav_polygon = tile_data->get_navigation_polygon(navigation_layer); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	polygon_editor->clear_polygons(); | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 	if (nav_polygon.is_valid()) { | 
					
						
							|  |  |  | 		for (int i = 0; i < nav_polygon->get_outline_count(); i++) { | 
					
						
							|  |  |  | 			polygon_editor->add_polygon(nav_polygon->get_outline(i)); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataNavigationEditor::_set_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile, const Variant &p_value) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 	Ref<NavigationPolygon> nav_polygon = p_value; | 
					
						
							|  |  |  | 	tile_data->set_navigation_polygon(navigation_layer, nav_polygon); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-11 19:21:34 +02:00
										 |  |  | 	polygon_editor->set_background_tile(p_tile_set_atlas_source, p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Variant TileDataNavigationEditor::_get_value(TileSetAtlasSource *p_tile_set_atlas_source, Vector2 p_coords, int p_alternative_tile) { | 
					
						
							| 
									
										
										
										
											2022-01-28 14:26:35 +01:00
										 |  |  | 	TileData *tile_data = p_tile_set_atlas_source->get_tile_data(p_coords, p_alternative_tile); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL_V(tile_data, Variant()); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	return tile_data->get_navigation_polygon(navigation_layer); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 13:25:58 -03:00
										 |  |  | void TileDataNavigationEditor::_setup_undo_redo_action(TileSetAtlasSource *p_tile_set_atlas_source, const HashMap<TileMapCell, Variant, TileMapCell> &p_previous_values, const Variant &p_new_value) { | 
					
						
							| 
									
										
										
										
											2022-12-23 23:53:16 +01:00
										 |  |  | 	EditorUndoRedoManager *undo_redo = EditorUndoRedoManager::get_singleton(); | 
					
						
							| 
									
										
										
										
											2021-08-09 14:13:42 -06:00
										 |  |  | 	for (const KeyValue<TileMapCell, Variant> &E : p_previous_values) { | 
					
						
							|  |  |  | 		Vector2i coords = E.key.get_atlas_coords(); | 
					
						
							|  |  |  | 		undo_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); | 
					
						
							|  |  |  | 		undo_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); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataNavigationEditor::_tile_set_changed() { | 
					
						
							|  |  |  | 	polygon_editor->set_tile_set(tile_set); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataNavigationEditor::_notification(int p_what) { | 
					
						
							|  |  |  | 	switch (p_what) { | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: { | 
					
						
							| 
									
										
										
										
											2022-08-04 13:40:05 +02:00
										 |  |  | #ifdef DEBUG_ENABLED
 | 
					
						
							| 
									
										
										
										
											2023-09-28 02:36:15 +02:00
										 |  |  | 			polygon_editor->set_polygons_color(NavigationServer2D::get_singleton()->get_debug_navigation_geometry_face_color()); | 
					
						
							| 
									
										
										
										
											2022-08-04 13:40:05 +02:00
										 |  |  | #endif // DEBUG_ENABLED
 | 
					
						
							| 
									
										
										
										
											2022-02-16 09:17:55 -05:00
										 |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileDataNavigationEditor::TileDataNavigationEditor() { | 
					
						
							|  |  |  | 	polygon_editor = memnew(GenericTilePolygonEditor); | 
					
						
							|  |  |  | 	polygon_editor->set_multiple_polygon_mode(true); | 
					
						
							|  |  |  | 	add_child(polygon_editor); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileDataNavigationEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileMapCell p_cell, bool p_selected) { | 
					
						
							|  |  |  | 	TileData *tile_data = _get_tile_data(p_cell); | 
					
						
							| 
									
										
										
										
											2023-09-09 17:24:40 +02:00
										 |  |  | 	ERR_FAIL_NULL(tile_data); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Draw all shapes.
 | 
					
						
							|  |  |  | 	RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 	Ref<NavigationPolygon> nav_polygon = tile_data->get_navigation_polygon(navigation_layer); | 
					
						
							|  |  |  | 	if (nav_polygon.is_valid()) { | 
					
						
							|  |  |  | 		Vector<Vector2> verts = nav_polygon->get_vertices(); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		if (verts.size() < 3) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 13:40:05 +02:00
										 |  |  | 		Color color = Color(0.5, 1.0, 1.0, 1.0); | 
					
						
							|  |  |  | #ifdef DEBUG_ENABLED
 | 
					
						
							| 
									
										
										
										
											2023-09-28 02:36:15 +02:00
										 |  |  | 		color = NavigationServer2D::get_singleton()->get_debug_navigation_geometry_face_color(); | 
					
						
							| 
									
										
										
										
											2022-08-04 13:40:05 +02:00
										 |  |  | #endif // DEBUG_ENABLED
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 		if (p_selected) { | 
					
						
							| 
									
										
										
										
											2022-10-18 16:43:37 +02:00
										 |  |  | 			Color grid_color = EDITOR_GET("editors/tiles_editor/grid_color"); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			Color 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); | 
					
						
							|  |  |  | 			selection_color.a = 0.7; | 
					
						
							|  |  |  | 			color = selection_color; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		RandomPCG rand; | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 		for (int i = 0; i < nav_polygon->get_polygon_count(); i++) { | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			// An array of vertices for this polygon.
 | 
					
						
							| 
									
										
										
										
											2022-09-29 12:53:28 +03:00
										 |  |  | 			Vector<int> polygon = nav_polygon->get_polygon(i); | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			Vector<Vector2> vertices; | 
					
						
							|  |  |  | 			vertices.resize(polygon.size()); | 
					
						
							|  |  |  | 			for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 				ERR_FAIL_INDEX(polygon[j], verts.size()); | 
					
						
							|  |  |  | 				vertices.write[j] = verts[polygon[j]]; | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 			// Generate the polygon color, slightly randomly modified from the settings one.
 | 
					
						
							|  |  |  | 			Color random_variation_color; | 
					
						
							|  |  |  | 			random_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); | 
					
						
							|  |  |  | 			random_variation_color.a = color.a; | 
					
						
							|  |  |  | 			Vector<Color> colors; | 
					
						
							|  |  |  | 			colors.push_back(random_variation_color); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			RenderingServer::get_singleton()->canvas_item_add_polygon(p_canvas_item->get_canvas_item(), vertices, colors); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-06-09 20:01:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D()); | 
					
						
							| 
									
										
										
										
											2021-05-07 15:41:39 +02:00
										 |  |  | } |