tile_data_editors.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. /*************************************************************************/
  2. /* tile_data_editors.cpp */
  3. /*************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /*************************************************************************/
  8. /* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
  9. /* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /*************************************************************************/
  30. #include "tile_data_editors.h"
  31. #include "tile_set_editor.h"
  32. TileData *TileDataEditor::_get_tile_data(TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile) {
  33. ERR_FAIL_COND_V(!p_tile_set, nullptr);
  34. ERR_FAIL_COND_V(!p_tile_set->has_source(p_atlas_source_id), nullptr);
  35. TileData *td = nullptr;
  36. TileSetSource *source = *p_tile_set->get_source(p_atlas_source_id);
  37. TileSetAtlasSource *atlas_source = Object::cast_to<TileSetAtlasSource>(source);
  38. if (atlas_source) {
  39. ERR_FAIL_COND_V(!atlas_source->has_tile(p_atlas_coords), nullptr);
  40. ERR_FAIL_COND_V(!atlas_source->has_alternative_tile(p_atlas_coords, p_alternative_tile), nullptr);
  41. td = Object::cast_to<TileData>(atlas_source->get_tile_data(p_atlas_coords, p_alternative_tile));
  42. }
  43. return td;
  44. }
  45. void TileDataEditor::edit(TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  46. }
  47. void TileDataTextureOffsetEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  48. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  49. ERR_FAIL_COND(!tile_data);
  50. bool valid;
  51. Variant value = tile_data->get(p_property, &valid);
  52. if (!valid) {
  53. return;
  54. }
  55. ERR_FAIL_COND(value.get_type() != Variant::VECTOR2I);
  56. Vector2i tile_set_tile_size = p_tile_set->get_tile_size();
  57. Rect2i rect = Rect2i(-tile_set_tile_size / 2, tile_set_tile_size);
  58. p_tile_set->draw_tile_shape(p_canvas_item, p_transform.xform(rect), Color(1.0, 0.0, 0.0));
  59. }
  60. void TileDataIntegerEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  61. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  62. ERR_FAIL_COND(!tile_data);
  63. bool valid;
  64. Variant value = tile_data->get(p_property, &valid);
  65. if (!valid) {
  66. return;
  67. }
  68. ERR_FAIL_COND(value.get_type() != Variant::INT);
  69. Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font("bold", "EditorFonts");
  70. int height = font->get_height();
  71. int width = 200;
  72. p_canvas_item->draw_string(font, p_transform.get_origin() + Vector2i(-width / 2, height / 2), vformat("%d", value), HALIGN_CENTER, width, -1, Color(1, 1, 1), 1, Color(0, 0, 0, 1));
  73. }
  74. void TileDataFloatEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  75. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  76. ERR_FAIL_COND(!tile_data);
  77. bool valid;
  78. Variant value = tile_data->get(p_property, &valid);
  79. if (!valid) {
  80. return;
  81. }
  82. ERR_FAIL_COND(value.get_type() != Variant::FLOAT);
  83. Ref<Font> font = TileSetEditor::get_singleton()->get_theme_font("bold", "EditorFonts");
  84. int height = font->get_height();
  85. int width = 200;
  86. p_canvas_item->draw_string(font, p_transform.get_origin() + Vector2i(-width / 2, height / 2), vformat("%.2f", value), HALIGN_CENTER, width, -1, Color(1, 1, 1), 1, Color(0, 0, 0, 1));
  87. }
  88. void TileDataPositionEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  89. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  90. ERR_FAIL_COND(!tile_data);
  91. bool valid;
  92. Variant value = tile_data->get(p_property, &valid);
  93. if (!valid) {
  94. return;
  95. }
  96. ERR_FAIL_COND(value.get_type() != Variant::VECTOR2I && value.get_type() != Variant::VECTOR2);
  97. Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_theme_icon("EditorPosition", "EditorIcons");
  98. p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2(value)) - position_icon->get_size() / 2);
  99. }
  100. void TileDataYSortEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  101. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  102. ERR_FAIL_COND(!tile_data);
  103. bool valid;
  104. Variant value = tile_data->get(p_property, &valid);
  105. if (!valid) {
  106. return;
  107. }
  108. ERR_FAIL_COND(value.get_type() != Variant::INT);
  109. Ref<Texture2D> position_icon = TileSetEditor::get_singleton()->get_theme_icon("EditorPosition", "EditorIcons");
  110. p_canvas_item->draw_texture(position_icon, p_transform.xform(Vector2(0, value)) - position_icon->get_size() / 2);
  111. }
  112. void TileDataOcclusionShapeEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  113. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  114. ERR_FAIL_COND(!tile_data);
  115. Vector<String> components = String(p_property).split("/", true);
  116. if (components[0].begins_with("occlusion_layer_") && components[0].trim_prefix("occlusion_layer_").is_valid_int()) {
  117. int occlusion_layer = components[0].trim_prefix("occlusion_layer_").to_int();
  118. if (occlusion_layer >= 0 && occlusion_layer < p_tile_set->get_occlusion_layers_count()) {
  119. // Draw all shapes.
  120. Vector<Color> debug_occlusion_color;
  121. debug_occlusion_color.push_back(Color(0.5, 0, 0, 0.6));
  122. RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform);
  123. Ref<OccluderPolygon2D> occluder = tile_data->get_occluder(occlusion_layer);
  124. if (occluder.is_valid() && occluder->get_polygon().size() >= 3) {
  125. p_canvas_item->draw_polygon(Variant(occluder->get_polygon()), debug_occlusion_color);
  126. }
  127. RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D());
  128. }
  129. }
  130. }
  131. void TileDataCollisionShapeEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  132. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  133. ERR_FAIL_COND(!tile_data);
  134. Vector<String> components = String(p_property).split("/", true);
  135. if (components[0].begins_with("physics_layer_") && components[0].trim_prefix("physics_layer_").is_valid_int()) {
  136. int physics_layer = components[0].trim_prefix("physics_layer_").to_int();
  137. if (physics_layer >= 0 && physics_layer < p_tile_set->get_physics_layers_count()) {
  138. // Draw all shapes.
  139. Color debug_collision_color = p_canvas_item->get_tree()->get_debug_collisions_color();
  140. RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform);
  141. for (int i = 0; i < tile_data->get_collision_shapes_count(physics_layer); i++) {
  142. Ref<Shape2D> shape = tile_data->get_collision_shape_shape(physics_layer, i);
  143. if (shape.is_valid()) {
  144. shape->draw(p_canvas_item->get_canvas_item(), debug_collision_color);
  145. }
  146. }
  147. RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D());
  148. }
  149. }
  150. }
  151. void TileDataTerrainsEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  152. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  153. ERR_FAIL_COND(!tile_data);
  154. Vector<String> components = String(p_property).split("/", true);
  155. if (components[0] == "terrain_mode" || components[0] == "terrain" || components[0] == "terrains_peering_bit") {
  156. TileSetPluginAtlasTerrain::draw_terrains(p_canvas_item, p_transform, p_tile_set, tile_data);
  157. }
  158. }
  159. void TileDataNavigationPolygonEditor::draw_over_tile(CanvasItem *p_canvas_item, Transform2D p_transform, TileSet *p_tile_set, int p_atlas_source_id, Vector2i p_atlas_coords, int p_alternative_tile, String p_property) {
  160. TileData *tile_data = _get_tile_data(p_tile_set, p_atlas_source_id, p_atlas_coords, p_alternative_tile);
  161. ERR_FAIL_COND(!tile_data);
  162. Vector<String> components = String(p_property).split("/", true);
  163. if (components[0].begins_with("navigation_layer_") && components[0].trim_prefix("navigation_layer_").is_valid_int()) {
  164. int navigation_layer = components[0].trim_prefix("navigation_layer_").to_int();
  165. if (navigation_layer >= 0 && navigation_layer < p_tile_set->get_navigation_layers_count()) {
  166. // Draw all shapes.
  167. RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), p_transform);
  168. Ref<NavigationPolygon> navigation_polygon = tile_data->get_navigation_polygon(navigation_layer);
  169. if (navigation_polygon.is_valid()) {
  170. Vector<Vector2> verts = navigation_polygon->get_vertices();
  171. if (verts.size() < 3) {
  172. return;
  173. }
  174. Color color = p_canvas_item->get_tree()->get_debug_navigation_color();
  175. RandomPCG rand;
  176. for (int i = 0; i < navigation_polygon->get_polygon_count(); i++) {
  177. // An array of vertices for this polygon.
  178. Vector<int> polygon = navigation_polygon->get_polygon(i);
  179. Vector<Vector2> vertices;
  180. vertices.resize(polygon.size());
  181. for (int j = 0; j < polygon.size(); j++) {
  182. ERR_FAIL_INDEX(polygon[j], verts.size());
  183. vertices.write[j] = verts[polygon[j]];
  184. }
  185. // Generate the polygon color, slightly randomly modified from the settings one.
  186. Color random_variation_color;
  187. 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);
  188. random_variation_color.a = color.a;
  189. Vector<Color> colors;
  190. colors.push_back(random_variation_color);
  191. RenderingServer::get_singleton()->canvas_item_add_polygon(p_canvas_item->get_canvas_item(), vertices, colors);
  192. }
  193. }
  194. RenderingServer::get_singleton()->canvas_item_add_set_transform(p_canvas_item->get_canvas_item(), Transform2D());
  195. }
  196. }
  197. }