2023-01-05 12:25:55 +00:00
|
|
|
/**************************************************************************/
|
|
|
|
/* editor_toaster.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. */
|
|
|
|
/**************************************************************************/
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2022-02-12 01:46:22 +00:00
|
|
|
#include "editor_toaster.h"
|
|
|
|
|
2022-02-14 13:00:03 +00:00
|
|
|
#include "editor/editor_settings.h"
|
2023-08-13 00:33:39 +00:00
|
|
|
#include "editor/editor_string_names.h"
|
2024-01-15 12:14:55 +00:00
|
|
|
#include "editor/themes/editor_scale.h"
|
2022-02-12 01:46:22 +00:00
|
|
|
#include "scene/gui/button.h"
|
2021-09-22 15:36:40 +00:00
|
|
|
#include "scene/gui/label.h"
|
|
|
|
#include "scene/gui/panel_container.h"
|
2023-07-14 20:35:39 +00:00
|
|
|
#include "scene/resources/style_box_flat.h"
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2021-10-19 15:18:45 +00:00
|
|
|
EditorToaster *EditorToaster::singleton = nullptr;
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
void EditorToaster::_notification(int p_what) {
|
|
|
|
switch (p_what) {
|
|
|
|
case NOTIFICATION_INTERNAL_PROCESS: {
|
|
|
|
double delta = get_process_delta_time();
|
|
|
|
|
|
|
|
// Check if one element is hovered, if so, don't elapse time.
|
|
|
|
bool hovered = false;
|
|
|
|
for (const KeyValue<Control *, Toast> &element : toasts) {
|
|
|
|
if (Rect2(Vector2(), element.key->get_size()).has_point(element.key->get_local_mouse_position())) {
|
|
|
|
hovered = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Elapses the time and remove toasts if needed.
|
|
|
|
if (!hovered) {
|
|
|
|
for (const KeyValue<Control *, Toast> &element : toasts) {
|
|
|
|
if (!element.value.popped || element.value.duration <= 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
toasts[element.key].remaining_time -= delta;
|
|
|
|
if (toasts[element.key].remaining_time < 0) {
|
|
|
|
close(element.key);
|
|
|
|
}
|
2022-08-13 21:21:24 +00:00
|
|
|
element.key->queue_redraw();
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Reset the timers when hovered.
|
|
|
|
for (const KeyValue<Control *, Toast> &element : toasts) {
|
|
|
|
if (!element.value.popped || element.value.duration <= 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
toasts[element.key].remaining_time = element.value.duration;
|
2022-08-13 21:21:24 +00:00
|
|
|
element.key->queue_redraw();
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change alpha over time.
|
|
|
|
bool needs_update = false;
|
|
|
|
for (const KeyValue<Control *, Toast> &element : toasts) {
|
2022-09-29 09:53:28 +00:00
|
|
|
Color modulate_fade = element.key->get_modulate();
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
// Change alpha over time.
|
2022-09-29 09:53:28 +00:00
|
|
|
if (element.value.popped && modulate_fade.a < 1.0) {
|
|
|
|
modulate_fade.a += delta * 3;
|
|
|
|
element.key->set_modulate(modulate_fade);
|
|
|
|
} else if (!element.value.popped && modulate_fade.a > 0.0) {
|
|
|
|
modulate_fade.a -= delta * 2;
|
|
|
|
element.key->set_modulate(modulate_fade);
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Hide element if it is not visible anymore.
|
2023-02-27 02:33:49 +00:00
|
|
|
if (modulate_fade.a <= 0.0 && element.key->is_visible()) {
|
2023-01-17 14:40:50 +00:00
|
|
|
element.key->hide();
|
|
|
|
needs_update = true;
|
2023-02-27 02:33:49 +00:00
|
|
|
} else if (modulate_fade.a > 0.0 && !element.key->is_visible()) {
|
2023-01-17 14:40:50 +00:00
|
|
|
element.key->show();
|
|
|
|
needs_update = true;
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (needs_update) {
|
|
|
|
_update_vbox_position();
|
|
|
|
_update_disable_notifications_button();
|
2022-08-13 21:21:24 +00:00
|
|
|
main_button->queue_redraw();
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
} break;
|
2022-02-15 23:52:32 +00:00
|
|
|
|
2022-08-29 09:04:31 +00:00
|
|
|
case NOTIFICATION_ENTER_TREE:
|
2021-09-22 15:36:40 +00:00
|
|
|
case NOTIFICATION_THEME_CHANGED: {
|
|
|
|
if (vbox_container->is_visible()) {
|
2023-08-13 00:33:39 +00:00
|
|
|
main_button->set_icon(get_editor_theme_icon(SNAME("Notification")));
|
2021-09-22 15:36:40 +00:00
|
|
|
} else {
|
2023-08-13 00:33:39 +00:00
|
|
|
main_button->set_icon(get_editor_theme_icon(SNAME("NotificationDisabled")));
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
2023-08-13 00:33:39 +00:00
|
|
|
disable_notifications_button->set_icon(get_editor_theme_icon(SNAME("NotificationDisabled")));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
// Styleboxes background.
|
2023-08-13 00:33:39 +00:00
|
|
|
info_panel_style_background->set_bg_color(get_theme_color(SNAME("base_color"), EditorStringName(Editor)));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2023-08-13 00:33:39 +00:00
|
|
|
warning_panel_style_background->set_bg_color(get_theme_color(SNAME("base_color"), EditorStringName(Editor)));
|
|
|
|
warning_panel_style_background->set_border_color(get_theme_color(SNAME("warning_color"), EditorStringName(Editor)));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2023-08-13 00:33:39 +00:00
|
|
|
error_panel_style_background->set_bg_color(get_theme_color(SNAME("base_color"), EditorStringName(Editor)));
|
|
|
|
error_panel_style_background->set_border_color(get_theme_color(SNAME("error_color"), EditorStringName(Editor)));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
// Styleboxes progress.
|
2023-08-13 00:33:39 +00:00
|
|
|
info_panel_style_progress->set_bg_color(get_theme_color(SNAME("base_color"), EditorStringName(Editor)).lightened(0.03));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2023-08-13 00:33:39 +00:00
|
|
|
warning_panel_style_progress->set_bg_color(get_theme_color(SNAME("base_color"), EditorStringName(Editor)).lightened(0.03));
|
|
|
|
warning_panel_style_progress->set_border_color(get_theme_color(SNAME("warning_color"), EditorStringName(Editor)));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2023-08-13 00:33:39 +00:00
|
|
|
error_panel_style_progress->set_bg_color(get_theme_color(SNAME("base_color"), EditorStringName(Editor)).lightened(0.03));
|
|
|
|
error_panel_style_progress->set_border_color(get_theme_color(SNAME("error_color"), EditorStringName(Editor)));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2022-08-13 21:21:24 +00:00
|
|
|
main_button->queue_redraw();
|
|
|
|
disable_notifications_button->queue_redraw();
|
2021-09-22 15:36:40 +00:00
|
|
|
} break;
|
2022-02-15 23:52:32 +00:00
|
|
|
|
2021-09-22 15:36:40 +00:00
|
|
|
case NOTIFICATION_TRANSFORM_CHANGED: {
|
|
|
|
_update_vbox_position();
|
|
|
|
_update_disable_notifications_button();
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_error_handler(void *p_self, const char *p_func, const char *p_file, int p_line, const char *p_error, const char *p_errorexp, bool p_editor_notify, ErrorHandlerType p_type) {
|
2023-01-19 12:07:19 +00:00
|
|
|
// This may be called from a thread. Since we will deal with non-thread-safe elements,
|
|
|
|
// we have to put it in the queue for safety.
|
2023-10-06 13:38:34 +00:00
|
|
|
callable_mp_static(&EditorToaster::_error_handler_impl).bind(String::utf8(p_file), p_line, String::utf8(p_error), String::utf8(p_errorexp), p_editor_notify, p_type).call_deferred();
|
2023-01-19 12:07:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_error_handler_impl(const String &p_file, int p_line, const String &p_error, const String &p_errorexp, bool p_editor_notify, int p_type) {
|
2021-10-19 15:18:45 +00:00
|
|
|
if (!EditorToaster::get_singleton() || !EditorToaster::get_singleton()->is_inside_tree()) {
|
2021-09-22 15:36:40 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEV_ENABLED
|
|
|
|
bool in_dev = true;
|
|
|
|
#else
|
|
|
|
bool in_dev = false;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int show_all_setting = EDITOR_GET("interface/editor/show_internal_errors_in_toast_notifications");
|
|
|
|
|
|
|
|
if (p_editor_notify || (show_all_setting == 0 && in_dev) || show_all_setting == 1) {
|
2023-01-19 12:07:19 +00:00
|
|
|
String err_str = !p_errorexp.is_empty() ? p_errorexp : p_error;
|
|
|
|
String tooltip_str = p_file + ":" + itos(p_line);
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
if (!p_editor_notify) {
|
|
|
|
if (p_type == ERR_HANDLER_WARNING) {
|
|
|
|
err_str = "INTERNAL WARNING: " + err_str;
|
|
|
|
} else {
|
|
|
|
err_str = "INTERNAL ERROR: " + err_str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-19 12:07:19 +00:00
|
|
|
Severity severity = ((ErrorHandlerType)p_type == ERR_HANDLER_WARNING) ? SEVERITY_WARNING : SEVERITY_ERROR;
|
2021-12-14 12:42:46 +00:00
|
|
|
EditorToaster::get_singleton()->popup_str(err_str, severity, tooltip_str);
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_update_vbox_position() {
|
|
|
|
// This is kind of a workaround because it's hard to keep the VBox anchroed to the bottom.
|
|
|
|
vbox_container->set_size(Vector2());
|
|
|
|
vbox_container->set_position(get_global_position() - vbox_container->get_size() + Vector2(get_size().x, -5 * EDSCALE));
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_update_disable_notifications_button() {
|
|
|
|
bool any_visible = false;
|
|
|
|
for (KeyValue<Control *, Toast> element : toasts) {
|
|
|
|
if (element.key->is_visible()) {
|
|
|
|
any_visible = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!any_visible || !vbox_container->is_visible()) {
|
|
|
|
disable_notifications_panel->hide();
|
|
|
|
} else {
|
|
|
|
disable_notifications_panel->show();
|
|
|
|
disable_notifications_panel->set_position(get_global_position() + Vector2(5 * EDSCALE, -disable_notifications_panel->get_minimum_size().y) + Vector2(get_size().x, -5 * EDSCALE));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_auto_hide_or_free_toasts() {
|
|
|
|
// Hide or free old temporary items.
|
|
|
|
int visible_temporary = 0;
|
|
|
|
int temporary = 0;
|
|
|
|
LocalVector<Control *> to_delete;
|
|
|
|
for (int i = vbox_container->get_child_count() - 1; i >= 0; i--) {
|
|
|
|
Control *control = Object::cast_to<Control>(vbox_container->get_child(i));
|
|
|
|
if (toasts[control].duration <= 0) {
|
|
|
|
continue; // Ignore non-temporary toasts.
|
|
|
|
}
|
|
|
|
|
|
|
|
temporary++;
|
|
|
|
if (control->is_visible()) {
|
|
|
|
visible_temporary++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hide
|
|
|
|
if (visible_temporary > max_temporary_count) {
|
|
|
|
close(control);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Free
|
|
|
|
if (temporary > max_temporary_count * 2) {
|
|
|
|
to_delete.push_back(control);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete the control right away (removed as child) as it might cause issues otherwise when iterative over the vbox_container children.
|
2022-12-29 00:24:45 +00:00
|
|
|
for (Control *c : to_delete) {
|
|
|
|
vbox_container->remove_child(c);
|
|
|
|
c->queue_free();
|
|
|
|
toasts.erase(c);
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
2022-06-14 00:35:25 +00:00
|
|
|
|
|
|
|
if (toasts.is_empty()) {
|
2022-08-25 10:42:17 +00:00
|
|
|
main_button->set_tooltip_text(TTR("No notifications."));
|
2022-06-14 00:35:25 +00:00
|
|
|
main_button->set_modulate(Color(0.5, 0.5, 0.5));
|
|
|
|
main_button->set_disabled(true);
|
|
|
|
} else {
|
2022-08-25 10:42:17 +00:00
|
|
|
main_button->set_tooltip_text(TTR("Show notifications."));
|
2022-06-14 00:35:25 +00:00
|
|
|
main_button->set_modulate(Color(1, 1, 1));
|
|
|
|
main_button->set_disabled(false);
|
|
|
|
}
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_draw_button() {
|
|
|
|
bool has_one = false;
|
|
|
|
Severity highest_severity = SEVERITY_INFO;
|
|
|
|
for (const KeyValue<Control *, Toast> &element : toasts) {
|
|
|
|
if (!element.key->is_visible()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
has_one = true;
|
|
|
|
if (element.value.severity > highest_severity) {
|
|
|
|
highest_severity = element.value.severity;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_one) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Color color;
|
|
|
|
real_t button_radius = main_button->get_size().x / 8;
|
|
|
|
switch (highest_severity) {
|
|
|
|
case SEVERITY_INFO:
|
2023-08-13 00:33:39 +00:00
|
|
|
color = get_theme_color(SNAME("accent_color"), EditorStringName(Editor));
|
2021-09-22 15:36:40 +00:00
|
|
|
break;
|
|
|
|
case SEVERITY_WARNING:
|
2023-08-13 00:33:39 +00:00
|
|
|
color = get_theme_color(SNAME("warning_color"), EditorStringName(Editor));
|
2021-09-22 15:36:40 +00:00
|
|
|
break;
|
|
|
|
case SEVERITY_ERROR:
|
2023-08-13 00:33:39 +00:00
|
|
|
color = get_theme_color(SNAME("error_color"), EditorStringName(Editor));
|
2021-09-22 15:36:40 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
main_button->draw_circle(Vector2(button_radius * 2, button_radius * 2), button_radius, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_draw_progress(Control *panel) {
|
|
|
|
if (toasts.has(panel) && toasts[panel].remaining_time > 0 && toasts[panel].duration > 0) {
|
|
|
|
Size2 size = panel->get_size();
|
2022-09-05 12:05:50 +00:00
|
|
|
size.x *= MIN(1, Math::remap(toasts[panel].remaining_time, 0, toasts[panel].duration, 0, 2));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
Ref<StyleBoxFlat> stylebox;
|
|
|
|
switch (toasts[panel].severity) {
|
|
|
|
case SEVERITY_INFO:
|
|
|
|
stylebox = info_panel_style_progress;
|
|
|
|
break;
|
|
|
|
case SEVERITY_WARNING:
|
|
|
|
stylebox = warning_panel_style_progress;
|
|
|
|
break;
|
|
|
|
case SEVERITY_ERROR:
|
|
|
|
stylebox = error_panel_style_progress;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
panel->draw_style_box(stylebox, Rect2(Vector2(), size));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_set_notifications_enabled(bool p_enabled) {
|
|
|
|
vbox_container->set_visible(p_enabled);
|
|
|
|
if (p_enabled) {
|
2023-08-13 00:33:39 +00:00
|
|
|
main_button->set_icon(get_editor_theme_icon(SNAME("Notification")));
|
2021-09-22 15:36:40 +00:00
|
|
|
} else {
|
2023-08-13 00:33:39 +00:00
|
|
|
main_button->set_icon(get_editor_theme_icon(SNAME("NotificationDisabled")));
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
_update_disable_notifications_button();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::_repop_old() {
|
|
|
|
// Repop olds, up to max_temporary_count
|
|
|
|
bool needs_update = false;
|
2022-09-29 09:53:28 +00:00
|
|
|
int visible_count = 0;
|
2021-09-22 15:36:40 +00:00
|
|
|
for (int i = vbox_container->get_child_count() - 1; i >= 0; i--) {
|
|
|
|
Control *control = Object::cast_to<Control>(vbox_container->get_child(i));
|
|
|
|
if (!control->is_visible()) {
|
|
|
|
control->show();
|
|
|
|
toasts[control].remaining_time = toasts[control].duration;
|
|
|
|
toasts[control].popped = true;
|
|
|
|
needs_update = true;
|
|
|
|
}
|
2022-09-29 09:53:28 +00:00
|
|
|
visible_count++;
|
|
|
|
if (visible_count >= max_temporary_count) {
|
2021-09-22 15:36:40 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (needs_update) {
|
|
|
|
_update_vbox_position();
|
|
|
|
_update_disable_notifications_button();
|
2022-08-13 21:21:24 +00:00
|
|
|
main_button->queue_redraw();
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-15 16:25:58 +00:00
|
|
|
Control *EditorToaster::popup(Control *p_control, Severity p_severity, double p_time, const String &p_tooltip) {
|
2021-09-22 15:36:40 +00:00
|
|
|
// Create the panel according to the severity.
|
|
|
|
PanelContainer *panel = memnew(PanelContainer);
|
2022-08-25 10:42:17 +00:00
|
|
|
panel->set_tooltip_text(p_tooltip);
|
2021-09-22 15:36:40 +00:00
|
|
|
switch (p_severity) {
|
|
|
|
case SEVERITY_INFO:
|
2022-02-08 09:14:58 +00:00
|
|
|
panel->add_theme_style_override("panel", info_panel_style_background);
|
2021-09-22 15:36:40 +00:00
|
|
|
break;
|
|
|
|
case SEVERITY_WARNING:
|
2022-02-08 09:14:58 +00:00
|
|
|
panel->add_theme_style_override("panel", warning_panel_style_background);
|
2021-09-22 15:36:40 +00:00
|
|
|
break;
|
|
|
|
case SEVERITY_ERROR:
|
2022-02-08 09:14:58 +00:00
|
|
|
panel->add_theme_style_override("panel", error_panel_style_background);
|
2021-09-22 15:36:40 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
panel->set_modulate(Color(1, 1, 1, 0));
|
2022-07-28 20:56:41 +00:00
|
|
|
panel->connect("draw", callable_mp(this, &EditorToaster::_draw_progress).bind(panel));
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
// Horizontal container.
|
|
|
|
HBoxContainer *hbox_container = memnew(HBoxContainer);
|
|
|
|
hbox_container->set_h_size_flags(SIZE_EXPAND_FILL);
|
|
|
|
panel->add_child(hbox_container);
|
|
|
|
|
|
|
|
// Content control.
|
|
|
|
p_control->set_h_size_flags(SIZE_EXPAND_FILL);
|
|
|
|
hbox_container->add_child(p_control);
|
|
|
|
|
|
|
|
// Close button.
|
|
|
|
if (p_time > 0.0) {
|
|
|
|
Button *close_button = memnew(Button);
|
|
|
|
close_button->set_flat(true);
|
2023-08-13 00:33:39 +00:00
|
|
|
close_button->set_icon(get_editor_theme_icon(SNAME("Close")));
|
2022-07-28 20:56:41 +00:00
|
|
|
close_button->connect("pressed", callable_mp(this, &EditorToaster::close).bind(panel));
|
|
|
|
close_button->connect("theme_changed", callable_mp(this, &EditorToaster::_close_button_theme_changed).bind(close_button));
|
2021-09-22 15:36:40 +00:00
|
|
|
hbox_container->add_child(close_button);
|
|
|
|
}
|
|
|
|
|
|
|
|
toasts[panel].severity = p_severity;
|
|
|
|
if (p_time > 0.0) {
|
|
|
|
toasts[panel].duration = p_time;
|
|
|
|
toasts[panel].remaining_time = p_time;
|
|
|
|
} else {
|
|
|
|
toasts[panel].duration = -1.0;
|
|
|
|
}
|
|
|
|
toasts[panel].popped = true;
|
|
|
|
vbox_container->add_child(panel);
|
|
|
|
_auto_hide_or_free_toasts();
|
|
|
|
_update_vbox_position();
|
|
|
|
_update_disable_notifications_button();
|
2022-08-13 21:21:24 +00:00
|
|
|
main_button->queue_redraw();
|
2021-09-22 15:36:40 +00:00
|
|
|
|
|
|
|
return panel;
|
|
|
|
}
|
|
|
|
|
2024-02-15 16:25:58 +00:00
|
|
|
void EditorToaster::popup_str(const String &p_message, Severity p_severity, const String &p_tooltip) {
|
2022-03-08 10:12:25 +00:00
|
|
|
if (is_processing_error) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-12-14 12:42:46 +00:00
|
|
|
// Since "_popup_str" adds nodes to the tree, and since the "add_child" method is not
|
|
|
|
// thread-safe, it's better to defer the call to the next cycle to be thread-safe.
|
2022-03-08 10:12:25 +00:00
|
|
|
is_processing_error = true;
|
2023-12-18 14:46:56 +00:00
|
|
|
callable_mp(this, &EditorToaster::_popup_str).call_deferred(p_message, p_severity, p_tooltip);
|
2022-03-08 10:12:25 +00:00
|
|
|
is_processing_error = false;
|
2021-12-14 12:42:46 +00:00
|
|
|
}
|
|
|
|
|
2024-02-15 16:25:58 +00:00
|
|
|
void EditorToaster::_popup_str(const String &p_message, Severity p_severity, const String &p_tooltip) {
|
2022-03-08 10:12:25 +00:00
|
|
|
is_processing_error = true;
|
2021-09-22 15:36:40 +00:00
|
|
|
// Check if we already have a popup with the given message.
|
|
|
|
Control *control = nullptr;
|
|
|
|
for (KeyValue<Control *, Toast> element : toasts) {
|
|
|
|
if (element.value.message == p_message && element.value.severity == p_severity && element.value.tooltip == p_tooltip) {
|
|
|
|
control = element.key;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new message if needed.
|
|
|
|
if (control == nullptr) {
|
2023-01-17 14:40:50 +00:00
|
|
|
HBoxContainer *hb = memnew(HBoxContainer);
|
|
|
|
hb->add_theme_constant_override("separation", 0);
|
|
|
|
|
2021-09-22 15:36:40 +00:00
|
|
|
Label *label = memnew(Label);
|
2023-01-17 14:40:50 +00:00
|
|
|
hb->add_child(label);
|
2021-09-22 15:36:40 +00:00
|
|
|
|
2023-01-17 14:40:50 +00:00
|
|
|
Label *count_label = memnew(Label);
|
|
|
|
hb->add_child(count_label);
|
|
|
|
|
|
|
|
control = popup(hb, p_severity, default_message_duration, p_tooltip);
|
2021-09-22 15:36:40 +00:00
|
|
|
toasts[control].message = p_message;
|
|
|
|
toasts[control].tooltip = p_tooltip;
|
|
|
|
toasts[control].count = 1;
|
2023-01-17 14:40:50 +00:00
|
|
|
toasts[control].message_label = label;
|
|
|
|
toasts[control].message_count_label = count_label;
|
2021-09-22 15:36:40 +00:00
|
|
|
} else {
|
|
|
|
if (toasts[control].popped) {
|
|
|
|
toasts[control].count += 1;
|
|
|
|
} else {
|
|
|
|
toasts[control].count = 1;
|
|
|
|
}
|
|
|
|
toasts[control].remaining_time = toasts[control].duration;
|
|
|
|
toasts[control].popped = true;
|
|
|
|
control->show();
|
|
|
|
vbox_container->move_child(control, vbox_container->get_child_count());
|
|
|
|
_auto_hide_or_free_toasts();
|
|
|
|
_update_vbox_position();
|
|
|
|
_update_disable_notifications_button();
|
2022-08-13 21:21:24 +00:00
|
|
|
main_button->queue_redraw();
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 14:40:50 +00:00
|
|
|
// Retrieve the label back, then update the text.
|
|
|
|
Label *message_label = toasts[control].message_label;
|
2023-09-09 15:24:40 +00:00
|
|
|
ERR_FAIL_NULL(message_label);
|
2023-01-17 14:40:50 +00:00
|
|
|
message_label->set_text(p_message);
|
|
|
|
message_label->set_text_overrun_behavior(TextServer::OVERRUN_NO_TRIMMING);
|
|
|
|
message_label->set_custom_minimum_size(Size2());
|
|
|
|
|
|
|
|
Size2i size = message_label->get_combined_minimum_size();
|
|
|
|
int limit_width = get_viewport_rect().size.x / 2; // Limit label size to half the viewport size.
|
|
|
|
if (size.x > limit_width) {
|
|
|
|
message_label->set_text_overrun_behavior(TextServer::OVERRUN_TRIM_ELLIPSIS);
|
|
|
|
message_label->set_custom_minimum_size(Size2(limit_width, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Retrieve the count label back, then update the text.
|
|
|
|
Label *message_count_label = toasts[control].message_count_label;
|
2021-09-22 15:36:40 +00:00
|
|
|
if (toasts[control].count == 1) {
|
2023-01-17 14:40:50 +00:00
|
|
|
message_count_label->hide();
|
2021-09-22 15:36:40 +00:00
|
|
|
} else {
|
2023-01-17 14:40:50 +00:00
|
|
|
message_count_label->set_text(vformat("(%d)", toasts[control].count));
|
|
|
|
message_count_label->show();
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
2023-01-17 14:40:50 +00:00
|
|
|
|
|
|
|
vbox_container->reset_size();
|
|
|
|
|
2022-03-08 10:12:25 +00:00
|
|
|
is_processing_error = false;
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void EditorToaster::close(Control *p_control) {
|
|
|
|
ERR_FAIL_COND(!toasts.has(p_control));
|
|
|
|
toasts[p_control].remaining_time = -1.0;
|
|
|
|
toasts[p_control].popped = false;
|
|
|
|
}
|
|
|
|
|
2022-01-18 10:41:07 +00:00
|
|
|
void EditorToaster::_close_button_theme_changed(Control *p_close_button) {
|
|
|
|
Button *close_button = Object::cast_to<Button>(p_close_button);
|
|
|
|
if (close_button) {
|
2023-08-13 00:33:39 +00:00
|
|
|
close_button->set_icon(get_editor_theme_icon(SNAME("Close")));
|
2022-01-18 10:41:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-22 15:36:40 +00:00
|
|
|
EditorToaster *EditorToaster::get_singleton() {
|
|
|
|
return singleton;
|
|
|
|
}
|
2021-12-14 12:42:46 +00:00
|
|
|
|
2021-09-22 15:36:40 +00:00
|
|
|
EditorToaster::EditorToaster() {
|
|
|
|
set_notify_transform(true);
|
|
|
|
set_process_internal(true);
|
|
|
|
|
|
|
|
// VBox.
|
|
|
|
vbox_container = memnew(VBoxContainer);
|
|
|
|
vbox_container->set_as_top_level(true);
|
|
|
|
vbox_container->connect("resized", callable_mp(this, &EditorToaster::_update_vbox_position));
|
|
|
|
add_child(vbox_container);
|
|
|
|
|
|
|
|
// Theming (background).
|
|
|
|
info_panel_style_background.instantiate();
|
|
|
|
info_panel_style_background->set_corner_radius_all(stylebox_radius * EDSCALE);
|
|
|
|
|
|
|
|
warning_panel_style_background.instantiate();
|
|
|
|
warning_panel_style_background->set_border_width(SIDE_LEFT, stylebox_radius * EDSCALE);
|
|
|
|
warning_panel_style_background->set_corner_radius_all(stylebox_radius * EDSCALE);
|
|
|
|
|
|
|
|
error_panel_style_background.instantiate();
|
|
|
|
error_panel_style_background->set_border_width(SIDE_LEFT, stylebox_radius * EDSCALE);
|
|
|
|
error_panel_style_background->set_corner_radius_all(stylebox_radius * EDSCALE);
|
|
|
|
|
|
|
|
Ref<StyleBoxFlat> boxes[] = { info_panel_style_background, warning_panel_style_background, error_panel_style_background };
|
|
|
|
for (int i = 0; i < 3; i++) {
|
2023-01-19 16:14:09 +00:00
|
|
|
boxes[i]->set_content_margin_individual(int(stylebox_radius * 2.5), 3, int(stylebox_radius * 2.5), 3);
|
2021-09-22 15:36:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Theming (progress).
|
|
|
|
info_panel_style_progress.instantiate();
|
|
|
|
info_panel_style_progress->set_corner_radius_all(stylebox_radius * EDSCALE);
|
|
|
|
|
|
|
|
warning_panel_style_progress.instantiate();
|
|
|
|
warning_panel_style_progress->set_border_width(SIDE_LEFT, stylebox_radius * EDSCALE);
|
|
|
|
warning_panel_style_progress->set_corner_radius_all(stylebox_radius * EDSCALE);
|
|
|
|
|
|
|
|
error_panel_style_progress.instantiate();
|
|
|
|
error_panel_style_progress->set_border_width(SIDE_LEFT, stylebox_radius * EDSCALE);
|
|
|
|
error_panel_style_progress->set_corner_radius_all(stylebox_radius * EDSCALE);
|
|
|
|
|
|
|
|
// Main button.
|
|
|
|
main_button = memnew(Button);
|
2022-08-25 10:42:17 +00:00
|
|
|
main_button->set_tooltip_text(TTR("No notifications."));
|
2022-06-14 00:35:25 +00:00
|
|
|
main_button->set_modulate(Color(0.5, 0.5, 0.5));
|
|
|
|
main_button->set_disabled(true);
|
2023-12-21 00:52:20 +00:00
|
|
|
main_button->set_theme_type_variation("FlatMenuButton");
|
2022-07-28 20:56:41 +00:00
|
|
|
main_button->connect("pressed", callable_mp(this, &EditorToaster::_set_notifications_enabled).bind(true));
|
2021-09-22 15:36:40 +00:00
|
|
|
main_button->connect("pressed", callable_mp(this, &EditorToaster::_repop_old));
|
|
|
|
main_button->connect("draw", callable_mp(this, &EditorToaster::_draw_button));
|
|
|
|
add_child(main_button);
|
|
|
|
|
|
|
|
// Disable notification button.
|
|
|
|
disable_notifications_panel = memnew(PanelContainer);
|
|
|
|
disable_notifications_panel->set_as_top_level(true);
|
2022-02-08 09:14:58 +00:00
|
|
|
disable_notifications_panel->add_theme_style_override("panel", info_panel_style_background);
|
2021-09-22 15:36:40 +00:00
|
|
|
add_child(disable_notifications_panel);
|
|
|
|
|
|
|
|
disable_notifications_button = memnew(Button);
|
2022-08-25 10:42:17 +00:00
|
|
|
disable_notifications_button->set_tooltip_text(TTR("Silence the notifications."));
|
2021-09-22 15:36:40 +00:00
|
|
|
disable_notifications_button->set_flat(true);
|
2022-07-28 20:56:41 +00:00
|
|
|
disable_notifications_button->connect("pressed", callable_mp(this, &EditorToaster::_set_notifications_enabled).bind(false));
|
2021-09-22 15:36:40 +00:00
|
|
|
disable_notifications_panel->add_child(disable_notifications_button);
|
|
|
|
|
|
|
|
// Other
|
|
|
|
singleton = this;
|
|
|
|
|
|
|
|
eh.errfunc = _error_handler;
|
|
|
|
add_error_handler(&eh);
|
|
|
|
};
|
|
|
|
|
|
|
|
EditorToaster::~EditorToaster() {
|
|
|
|
singleton = nullptr;
|
|
|
|
remove_error_handler(&eh);
|
|
|
|
}
|