diff --git a/js/misc/util.js b/js/misc/util.js index f68907ec2..1bc1a7283 100644 --- a/js/misc/util.js +++ b/js/misc/util.js @@ -6,7 +6,6 @@ const GLib = imports.gi.GLib; const Shell = imports.gi.Shell; const Main = imports.ui.main; -const MessageTray = imports.ui.messageTray; const Gettext = imports.gettext.domain('gnome-shell'); const _ = Gettext.gettext; @@ -146,12 +145,7 @@ function trySpawnDesktop(id) { function _handleSpawnError(command, err) { let title = _("Execution of '%s' failed:").format(command); - - let source = new MessageTray.SystemNotificationSource(); - Main.messageTray.add(source); - let notification = new MessageTray.Notification(source, title, err.message); - notification.setTransient(true); - source.notify(notification); + Main.notifyProblem(title, err.message); } // killall: diff --git a/js/ui/main.js b/js/ui/main.js index 24dab3193..3c3763bfb 100644 --- a/js/ui/main.js +++ b/js/ui/main.js @@ -86,6 +86,9 @@ function start() { global.logError = _logError; global.log = _logDebug; + // Chain up async errors reported from C + global.connect('notify-error', function (global, msg, detail) { notifyError(msg, detail); }); + Gio.DesktopAppInfo.set_desktop_env('GNOME'); shellDBusService = new ShellDBus.GnomeShell(); @@ -395,6 +398,27 @@ function loadTheme() { themeContext.set_theme (theme); } +/** + * notifyError: + * @msg: An error message + * @details: Additional information + * + * See shell_global_notify_problem(). + */ +function notifyError(msg, details) { + // Also print to stderr so it's logged somewhere + if (details) + log("error: " + msg + ": " + details); + else + log("error: " + msg) + + let source = new MessageTray.SystemNotificationSource(); + messageTray.add(source); + let notification = new MessageTray.Notification(source, msg, details); + notification.setTransient(true); + source.notify(notification); +} + /** * _log: * @category: string message type ('info', 'error') diff --git a/src/shell-global.c b/src/shell-global.c index 88b8224b7..ab112cc0f 100644 --- a/src/shell-global.c +++ b/src/shell-global.c @@ -21,6 +21,7 @@ #include #include #include +#include #include #include #include @@ -102,6 +103,7 @@ enum XDND_POSITION_CHANGED, XDND_LEAVE, XDND_ENTER, + NOTIFY_ERROR, LAST_SIGNAL }; @@ -293,6 +295,17 @@ shell_global_class_init (ShellGlobalClass *klass) g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); + shell_global_signals[NOTIFY_ERROR] = + g_signal_new ("notify-error", + G_TYPE_FROM_CLASS (klass), + G_SIGNAL_RUN_LAST, + 0, + NULL, NULL, + gi_cclosure_marshal_generic, + G_TYPE_NONE, 2, + G_TYPE_STRING, + G_TYPE_STRING); + g_object_class_install_property (gobject_class, PROP_OVERLAY_GROUP, g_param_spec_object ("overlay-group", @@ -1238,6 +1251,25 @@ shell_global_maybe_gc (ShellGlobal *global) gjs_context_maybe_gc (global->js_context); } +/** + * shell_global_notify_error: + * @global: a #ShellGlobal + * @msg: Error message + * @details: Error details + * + * Show a system error notification. Use this function + * when a user-initiated action results in a non-fatal problem + * from causes that may not be under system control. For + * example, an application crash. + */ +void +shell_global_notify_error (ShellGlobal *global, + const char *msg, + const char *details) +{ + g_signal_emit_by_name (global, "notify-error", msg, details); +} + static void grab_notify (GtkWidget *widget, gboolean was_grabbed, gpointer user_data) { diff --git a/src/shell-global.h b/src/shell-global.h index 1ca247c50..f05391915 100644 --- a/src/shell-global.h +++ b/src/shell-global.h @@ -143,6 +143,11 @@ void shell_global_play_theme_sound (ShellGlobal *global, void shell_global_cancel_theme_sound (ShellGlobal *global, guint id); + +void shell_global_notify_error (ShellGlobal *global, + const char *msg, + const char *details); + void shell_global_init_xdnd (ShellGlobal *global); typedef void (*ShellGetTpContactCb) (TpConnection *connection,