theme_editor_preview.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524
  1. /*************************************************************************/
  2. /* theme_editor_preview.cpp */
  3. /*************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /*************************************************************************/
  8. /* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
  9. /* Copyright (c) 2014-2022 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 "theme_editor_preview.h"
  31. #include "core/config/project_settings.h"
  32. #include "core/input/input.h"
  33. #include "core/math/math_funcs.h"
  34. #include "editor/editor_node.h"
  35. #include "editor/editor_scale.h"
  36. #include "scene/gui/button.h"
  37. #include "scene/gui/check_button.h"
  38. #include "scene/gui/color_picker.h"
  39. #include "scene/gui/progress_bar.h"
  40. #include "scene/resources/packed_scene.h"
  41. #include "scene/theme/theme_db.h"
  42. constexpr double REFRESH_TIMER = 1.5;
  43. void ThemeEditorPreview::set_preview_theme(const Ref<Theme> &p_theme) {
  44. preview_content->set_theme(p_theme);
  45. }
  46. void ThemeEditorPreview::add_preview_overlay(Control *p_overlay) {
  47. preview_overlay->add_child(p_overlay);
  48. p_overlay->hide();
  49. }
  50. void ThemeEditorPreview::_propagate_redraw(Control *p_at) {
  51. p_at->notification(NOTIFICATION_THEME_CHANGED);
  52. p_at->update_minimum_size();
  53. p_at->queue_redraw();
  54. for (int i = 0; i < p_at->get_child_count(); i++) {
  55. Control *a = Object::cast_to<Control>(p_at->get_child(i));
  56. if (a) {
  57. _propagate_redraw(a);
  58. }
  59. }
  60. }
  61. void ThemeEditorPreview::_refresh_interval() {
  62. // In case the project settings have changed.
  63. preview_bg->set_color(GLOBAL_GET("rendering/environment/defaults/default_clear_color"));
  64. _propagate_redraw(preview_bg);
  65. _propagate_redraw(preview_content);
  66. }
  67. void ThemeEditorPreview::_preview_visibility_changed() {
  68. set_process(is_visible_in_tree());
  69. }
  70. void ThemeEditorPreview::_picker_button_cbk() {
  71. picker_overlay->set_visible(picker_button->is_pressed());
  72. if (picker_button->is_pressed()) {
  73. _reset_picker_overlay();
  74. }
  75. }
  76. Control *ThemeEditorPreview::_find_hovered_control(Control *p_parent, Vector2 p_mouse_position) {
  77. Control *found = nullptr;
  78. for (int i = p_parent->get_child_count() - 1; i >= 0; i--) {
  79. Control *cc = Object::cast_to<Control>(p_parent->get_child(i));
  80. if (!cc || !cc->is_visible()) {
  81. continue;
  82. }
  83. Rect2 crect = cc->get_rect();
  84. if (crect.has_point(p_mouse_position)) {
  85. // Check if there is a child control under mouse.
  86. if (cc->get_child_count() > 0) {
  87. found = _find_hovered_control(cc, p_mouse_position - cc->get_position());
  88. }
  89. // If there are no applicable children, use the control itself.
  90. if (!found) {
  91. found = cc;
  92. }
  93. break;
  94. }
  95. }
  96. return found;
  97. }
  98. void ThemeEditorPreview::_draw_picker_overlay() {
  99. if (!picker_button->is_pressed()) {
  100. return;
  101. }
  102. picker_overlay->draw_rect(Rect2(Vector2(0.0, 0.0), picker_overlay->get_size()), theme_cache.preview_picker_overlay_color);
  103. if (hovered_control) {
  104. Rect2 highlight_rect = hovered_control->get_global_rect();
  105. highlight_rect.position = picker_overlay->get_global_transform().affine_inverse().xform(highlight_rect.position);
  106. picker_overlay->draw_style_box(theme_cache.preview_picker_overlay, highlight_rect);
  107. String highlight_name = hovered_control->get_theme_type_variation();
  108. if (highlight_name == StringName()) {
  109. highlight_name = hovered_control->get_class_name();
  110. }
  111. Rect2 highlight_label_rect = highlight_rect;
  112. highlight_label_rect.size = theme_cache.preview_picker_font->get_string_size(highlight_name, HORIZONTAL_ALIGNMENT_LEFT, -1, theme_cache.font_size);
  113. int margin_top = theme_cache.preview_picker_label->get_margin(SIDE_TOP);
  114. int margin_left = theme_cache.preview_picker_label->get_margin(SIDE_LEFT);
  115. int margin_bottom = theme_cache.preview_picker_label->get_margin(SIDE_BOTTOM);
  116. int margin_right = theme_cache.preview_picker_label->get_margin(SIDE_RIGHT);
  117. highlight_label_rect.size.x += margin_left + margin_right;
  118. highlight_label_rect.size.y += margin_top + margin_bottom;
  119. highlight_label_rect.position = highlight_label_rect.position.clamp(Vector2(), picker_overlay->get_size());
  120. picker_overlay->draw_style_box(theme_cache.preview_picker_label, highlight_label_rect);
  121. Point2 label_pos = highlight_label_rect.position;
  122. label_pos.y += highlight_label_rect.size.y - margin_bottom;
  123. label_pos.x += margin_left;
  124. picker_overlay->draw_string(theme_cache.preview_picker_font, label_pos, highlight_name, HORIZONTAL_ALIGNMENT_LEFT, -1, theme_cache.font_size);
  125. }
  126. }
  127. void ThemeEditorPreview::_gui_input_picker_overlay(const Ref<InputEvent> &p_event) {
  128. if (!picker_button->is_pressed()) {
  129. return;
  130. }
  131. Ref<InputEventMouseButton> mb = p_event;
  132. if (mb.is_valid() && mb->is_pressed() && mb->get_button_index() == MouseButton::LEFT) {
  133. if (hovered_control) {
  134. StringName theme_type = hovered_control->get_theme_type_variation();
  135. if (theme_type == StringName()) {
  136. theme_type = hovered_control->get_class_name();
  137. }
  138. emit_signal(SNAME("control_picked"), theme_type);
  139. picker_button->set_pressed(false);
  140. picker_overlay->set_visible(false);
  141. return;
  142. }
  143. }
  144. Ref<InputEventMouseMotion> mm = p_event;
  145. if (mm.is_valid()) {
  146. Vector2 mp = preview_content->get_local_mouse_position();
  147. hovered_control = _find_hovered_control(preview_content, mp);
  148. picker_overlay->queue_redraw();
  149. }
  150. // Forward input to the scroll container underneath to allow scrolling.
  151. preview_container->gui_input(p_event);
  152. }
  153. void ThemeEditorPreview::_reset_picker_overlay() {
  154. hovered_control = nullptr;
  155. picker_overlay->queue_redraw();
  156. }
  157. void ThemeEditorPreview::_notification(int p_what) {
  158. switch (p_what) {
  159. case NOTIFICATION_ENTER_TREE: {
  160. if (is_visible_in_tree()) {
  161. set_process(true);
  162. }
  163. connect("visibility_changed", callable_mp(this, &ThemeEditorPreview::_preview_visibility_changed));
  164. [[fallthrough]];
  165. }
  166. case NOTIFICATION_THEME_CHANGED: {
  167. picker_button->set_icon(get_theme_icon(SNAME("ColorPick"), SNAME("EditorIcons")));
  168. theme_cache.preview_picker_overlay = get_theme_stylebox(SNAME("preview_picker_overlay"), SNAME("ThemeEditor"));
  169. theme_cache.preview_picker_overlay_color = get_theme_color(SNAME("preview_picker_overlay_color"), SNAME("ThemeEditor"));
  170. theme_cache.preview_picker_label = get_theme_stylebox(SNAME("preview_picker_label"), SNAME("ThemeEditor"));
  171. theme_cache.preview_picker_font = get_theme_font(SNAME("status_source"), SNAME("EditorFonts"));
  172. theme_cache.font_size = get_theme_font_size(SNAME("font_size"), SNAME("EditorFonts"));
  173. } break;
  174. case NOTIFICATION_PROCESS: {
  175. time_left -= get_process_delta_time();
  176. if (time_left < 0) {
  177. time_left = REFRESH_TIMER;
  178. _refresh_interval();
  179. }
  180. } break;
  181. }
  182. }
  183. void ThemeEditorPreview::_bind_methods() {
  184. ADD_SIGNAL(MethodInfo("control_picked", PropertyInfo(Variant::STRING, "class_name")));
  185. }
  186. ThemeEditorPreview::ThemeEditorPreview() {
  187. preview_toolbar = memnew(HBoxContainer);
  188. add_child(preview_toolbar);
  189. picker_button = memnew(Button);
  190. preview_toolbar->add_child(picker_button);
  191. picker_button->set_flat(true);
  192. picker_button->set_toggle_mode(true);
  193. picker_button->set_tooltip_text(TTR("Toggle the control picker, allowing to visually select control types for edit."));
  194. picker_button->connect("pressed", callable_mp(this, &ThemeEditorPreview::_picker_button_cbk));
  195. MarginContainer *preview_body = memnew(MarginContainer);
  196. preview_body->set_custom_minimum_size(Size2(480, 0) * EDSCALE);
  197. preview_body->set_v_size_flags(SIZE_EXPAND_FILL);
  198. add_child(preview_body);
  199. preview_container = memnew(ScrollContainer);
  200. preview_body->add_child(preview_container);
  201. MarginContainer *preview_root = memnew(MarginContainer);
  202. preview_container->add_child(preview_root);
  203. preview_root->set_theme(ThemeDB::get_singleton()->get_default_theme());
  204. preview_root->set_clip_contents(true);
  205. preview_root->set_custom_minimum_size(Size2(450, 0) * EDSCALE);
  206. preview_root->set_v_size_flags(SIZE_EXPAND_FILL);
  207. preview_root->set_h_size_flags(SIZE_EXPAND_FILL);
  208. preview_bg = memnew(ColorRect);
  209. preview_bg->set_anchors_and_offsets_preset(PRESET_FULL_RECT);
  210. preview_bg->set_color(GLOBAL_GET("rendering/environment/defaults/default_clear_color"));
  211. preview_root->add_child(preview_bg);
  212. preview_content = memnew(MarginContainer);
  213. preview_root->add_child(preview_content);
  214. preview_content->add_theme_constant_override("margin_right", 4 * EDSCALE);
  215. preview_content->add_theme_constant_override("margin_top", 4 * EDSCALE);
  216. preview_content->add_theme_constant_override("margin_left", 4 * EDSCALE);
  217. preview_content->add_theme_constant_override("margin_bottom", 4 * EDSCALE);
  218. preview_overlay = memnew(MarginContainer);
  219. preview_overlay->set_mouse_filter(MOUSE_FILTER_IGNORE);
  220. preview_overlay->set_clip_contents(true);
  221. preview_body->add_child(preview_overlay);
  222. picker_overlay = memnew(Control);
  223. add_preview_overlay(picker_overlay);
  224. picker_overlay->connect("draw", callable_mp(this, &ThemeEditorPreview::_draw_picker_overlay));
  225. picker_overlay->connect("gui_input", callable_mp(this, &ThemeEditorPreview::_gui_input_picker_overlay));
  226. picker_overlay->connect("mouse_exited", callable_mp(this, &ThemeEditorPreview::_reset_picker_overlay));
  227. }
  228. void DefaultThemeEditorPreview::_notification(int p_what) {
  229. switch (p_what) {
  230. case NOTIFICATION_ENTER_TREE:
  231. case NOTIFICATION_THEME_CHANGED: {
  232. test_color_picker_button->set_custom_minimum_size(Size2(0, get_theme_constant(SNAME("color_picker_button_height"), SNAME("Editor"))));
  233. } break;
  234. }
  235. }
  236. DefaultThemeEditorPreview::DefaultThemeEditorPreview() {
  237. Panel *main_panel = memnew(Panel);
  238. preview_content->add_child(main_panel);
  239. MarginContainer *main_mc = memnew(MarginContainer);
  240. main_mc->add_theme_constant_override("margin_right", 4 * EDSCALE);
  241. main_mc->add_theme_constant_override("margin_top", 4 * EDSCALE);
  242. main_mc->add_theme_constant_override("margin_left", 4 * EDSCALE);
  243. main_mc->add_theme_constant_override("margin_bottom", 4 * EDSCALE);
  244. preview_content->add_child(main_mc);
  245. HBoxContainer *main_hb = memnew(HBoxContainer);
  246. main_mc->add_child(main_hb);
  247. main_hb->add_theme_constant_override("separation", 20 * EDSCALE);
  248. VBoxContainer *first_vb = memnew(VBoxContainer);
  249. main_hb->add_child(first_vb);
  250. first_vb->set_h_size_flags(SIZE_EXPAND_FILL);
  251. first_vb->add_theme_constant_override("separation", 10 * EDSCALE);
  252. first_vb->add_child(memnew(Label("Label")));
  253. first_vb->add_child(memnew(Button("Button")));
  254. Button *bt = memnew(Button);
  255. bt->set_text(TTR("Toggle Button"));
  256. bt->set_toggle_mode(true);
  257. bt->set_pressed(true);
  258. first_vb->add_child(bt);
  259. bt = memnew(Button);
  260. bt->set_text(TTR("Disabled Button"));
  261. bt->set_disabled(true);
  262. first_vb->add_child(bt);
  263. Button *tb = memnew(Button);
  264. tb->set_flat(true);
  265. tb->set_text("Button");
  266. first_vb->add_child(tb);
  267. CheckButton *cb = memnew(CheckButton);
  268. cb->set_text("CheckButton");
  269. first_vb->add_child(cb);
  270. CheckBox *cbx = memnew(CheckBox);
  271. cbx->set_text("CheckBox");
  272. first_vb->add_child(cbx);
  273. MenuButton *test_menu_button = memnew(MenuButton);
  274. test_menu_button->set_text("MenuButton");
  275. test_menu_button->get_popup()->add_item(TTR("Item"));
  276. test_menu_button->get_popup()->add_item(TTR("Disabled Item"));
  277. test_menu_button->get_popup()->set_item_disabled(1, true);
  278. test_menu_button->get_popup()->add_separator();
  279. test_menu_button->get_popup()->add_check_item(TTR("Check Item"));
  280. test_menu_button->get_popup()->add_check_item(TTR("Checked Item"));
  281. test_menu_button->get_popup()->set_item_checked(4, true);
  282. test_menu_button->get_popup()->add_separator();
  283. test_menu_button->get_popup()->add_radio_check_item(TTR("Radio Item"));
  284. test_menu_button->get_popup()->add_radio_check_item(TTR("Checked Radio Item"));
  285. test_menu_button->get_popup()->set_item_checked(7, true);
  286. test_menu_button->get_popup()->add_separator(TTR("Named Separator"));
  287. PopupMenu *test_submenu = memnew(PopupMenu);
  288. test_menu_button->get_popup()->add_child(test_submenu);
  289. test_submenu->set_name("submenu");
  290. test_menu_button->get_popup()->add_submenu_item(TTR("Submenu"), "submenu");
  291. test_submenu->add_item(TTR("Subitem 1"));
  292. test_submenu->add_item(TTR("Subitem 2"));
  293. first_vb->add_child(test_menu_button);
  294. OptionButton *test_option_button = memnew(OptionButton);
  295. test_option_button->add_item("OptionButton");
  296. test_option_button->add_separator();
  297. test_option_button->add_item(TTR("Has"));
  298. test_option_button->add_item(TTR("Many"));
  299. test_option_button->add_item(TTR("Options"));
  300. first_vb->add_child(test_option_button);
  301. test_color_picker_button = memnew(ColorPickerButton);
  302. first_vb->add_child(test_color_picker_button);
  303. VBoxContainer *second_vb = memnew(VBoxContainer);
  304. second_vb->set_h_size_flags(SIZE_EXPAND_FILL);
  305. main_hb->add_child(second_vb);
  306. second_vb->add_theme_constant_override("separation", 10 * EDSCALE);
  307. LineEdit *le = memnew(LineEdit);
  308. le->set_text("LineEdit");
  309. second_vb->add_child(le);
  310. le = memnew(LineEdit);
  311. le->set_text(TTR("Disabled LineEdit"));
  312. le->set_editable(false);
  313. second_vb->add_child(le);
  314. TextEdit *te = memnew(TextEdit);
  315. te->set_text("TextEdit");
  316. te->set_custom_minimum_size(Size2(0, 100) * EDSCALE);
  317. second_vb->add_child(te);
  318. second_vb->add_child(memnew(SpinBox));
  319. HBoxContainer *vhb = memnew(HBoxContainer);
  320. second_vb->add_child(vhb);
  321. vhb->set_custom_minimum_size(Size2(0, 100) * EDSCALE);
  322. vhb->add_child(memnew(VSlider));
  323. VScrollBar *vsb = memnew(VScrollBar);
  324. vsb->set_page(25);
  325. vhb->add_child(vsb);
  326. vhb->add_child(memnew(VSeparator));
  327. VBoxContainer *hvb = memnew(VBoxContainer);
  328. vhb->add_child(hvb);
  329. hvb->set_alignment(BoxContainer::ALIGNMENT_CENTER);
  330. hvb->set_h_size_flags(SIZE_EXPAND_FILL);
  331. hvb->add_child(memnew(HSlider));
  332. HScrollBar *hsb = memnew(HScrollBar);
  333. hsb->set_page(25);
  334. hvb->add_child(hsb);
  335. HSlider *hs = memnew(HSlider);
  336. hs->set_editable(false);
  337. hvb->add_child(hs);
  338. hvb->add_child(memnew(HSeparator));
  339. ProgressBar *pb = memnew(ProgressBar);
  340. pb->set_value(50);
  341. hvb->add_child(pb);
  342. VBoxContainer *third_vb = memnew(VBoxContainer);
  343. third_vb->set_h_size_flags(SIZE_EXPAND_FILL);
  344. third_vb->add_theme_constant_override("separation", 10 * EDSCALE);
  345. main_hb->add_child(third_vb);
  346. TabContainer *tc = memnew(TabContainer);
  347. third_vb->add_child(tc);
  348. tc->set_custom_minimum_size(Size2(0, 135) * EDSCALE);
  349. Control *tcc = memnew(Control);
  350. tcc->set_name(TTR("Tab 1"));
  351. tc->add_child(tcc);
  352. tcc = memnew(Control);
  353. tcc->set_name(TTR("Tab 2"));
  354. tc->add_child(tcc);
  355. tcc = memnew(Control);
  356. tcc->set_name(TTR("Tab 3"));
  357. tc->add_child(tcc);
  358. tc->set_tab_disabled(2, true);
  359. Tree *test_tree = memnew(Tree);
  360. third_vb->add_child(test_tree);
  361. test_tree->set_custom_minimum_size(Size2(0, 175) * EDSCALE);
  362. TreeItem *item = test_tree->create_item();
  363. item->set_text(0, "Tree");
  364. item = test_tree->create_item(test_tree->get_root());
  365. item->set_text(0, "Item");
  366. item = test_tree->create_item(test_tree->get_root());
  367. item->set_editable(0, true);
  368. item->set_text(0, TTR("Editable Item"));
  369. TreeItem *sub_tree = test_tree->create_item(test_tree->get_root());
  370. sub_tree->set_text(0, TTR("Subtree"));
  371. item = test_tree->create_item(sub_tree);
  372. item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK);
  373. item->set_editable(0, true);
  374. item->set_text(0, "Check Item");
  375. item = test_tree->create_item(sub_tree);
  376. item->set_cell_mode(0, TreeItem::CELL_MODE_RANGE);
  377. item->set_editable(0, true);
  378. item->set_range_config(0, 0, 20, 0.1);
  379. item->set_range(0, 2);
  380. item = test_tree->create_item(sub_tree);
  381. item->set_cell_mode(0, TreeItem::CELL_MODE_RANGE);
  382. item->set_editable(0, true);
  383. item->set_text(0, TTR("Has,Many,Options"));
  384. item->set_range(0, 2);
  385. }
  386. void SceneThemeEditorPreview::_reload_scene() {
  387. if (loaded_scene.is_null()) {
  388. return;
  389. }
  390. if (loaded_scene->get_path().is_empty() || !ResourceLoader::exists(loaded_scene->get_path())) {
  391. EditorNode::get_singleton()->show_warning(TTR("Invalid path, the PackedScene resource was probably moved or removed."));
  392. emit_signal(SNAME("scene_invalidated"));
  393. return;
  394. }
  395. for (int i = preview_content->get_child_count() - 1; i >= 0; i--) {
  396. Node *node = preview_content->get_child(i);
  397. node->queue_delete();
  398. preview_content->remove_child(node);
  399. }
  400. Node *instance = loaded_scene->instantiate();
  401. if (!instance || !Object::cast_to<Control>(instance)) {
  402. EditorNode::get_singleton()->show_warning(TTR("Invalid PackedScene resource, must have a Control node at its root."));
  403. emit_signal(SNAME("scene_invalidated"));
  404. return;
  405. }
  406. preview_content->add_child(instance);
  407. emit_signal(SNAME("scene_reloaded"));
  408. }
  409. void SceneThemeEditorPreview::_notification(int p_what) {
  410. switch (p_what) {
  411. case NOTIFICATION_ENTER_TREE:
  412. case NOTIFICATION_THEME_CHANGED: {
  413. reload_scene_button->set_icon(get_theme_icon(SNAME("Reload"), SNAME("EditorIcons")));
  414. } break;
  415. }
  416. }
  417. void SceneThemeEditorPreview::_bind_methods() {
  418. ADD_SIGNAL(MethodInfo("scene_invalidated"));
  419. ADD_SIGNAL(MethodInfo("scene_reloaded"));
  420. }
  421. bool SceneThemeEditorPreview::set_preview_scene(const String &p_path) {
  422. loaded_scene = ResourceLoader::load(p_path);
  423. if (loaded_scene.is_null()) {
  424. EditorNode::get_singleton()->show_warning(TTR("Invalid file, not a PackedScene resource."));
  425. return false;
  426. }
  427. Node *instance = loaded_scene->instantiate();
  428. if (!instance || !Object::cast_to<Control>(instance)) {
  429. EditorNode::get_singleton()->show_warning(TTR("Invalid PackedScene resource, must have a Control node at its root."));
  430. return false;
  431. }
  432. preview_content->add_child(instance);
  433. return true;
  434. }
  435. String SceneThemeEditorPreview::get_preview_scene_path() const {
  436. if (loaded_scene.is_null()) {
  437. return "";
  438. }
  439. return loaded_scene->get_path();
  440. }
  441. SceneThemeEditorPreview::SceneThemeEditorPreview() {
  442. preview_toolbar->add_child(memnew(VSeparator));
  443. reload_scene_button = memnew(Button);
  444. reload_scene_button->set_flat(true);
  445. reload_scene_button->set_tooltip_text(TTR("Reload the scene to reflect its most actual state."));
  446. preview_toolbar->add_child(reload_scene_button);
  447. reload_scene_button->connect("pressed", callable_mp(this, &SceneThemeEditorPreview::_reload_scene));
  448. }