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