2
0

theme_editor_preview.cpp 20 KB

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