/* Sticky Notes * Copyright (C) 2002-2003 Loban A Rahman * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301, USA. */ #include #include #include "stickynotes_applet_callbacks.h" #include "stickynotes.h" #include #include #include static void popup_add_note (StickyNotesApplet *applet, GtkWidget *item) { stickynotes_add (gtk_widget_get_screen (applet->w_applet)); } static void stickynote_show_notes (gboolean visible) { StickyNote *note; GList *l; if (stickynotes->visible == visible) return; if (g_settings_get_boolean (stickynotes->settings, "click-hide") && !visible) return; stickynotes->visible = visible; for (l = stickynotes->notes; l; l = l->next) { note = l->data; stickynote_set_visible (note, visible); } } static void stickynote_toggle_notes_visible (void) { stickynote_show_notes (!stickynotes->visible); } /* Applet Callback : Mouse button press on the applet. */ gboolean applet_button_cb (GtkWidget *widget, GdkEventButton *event, StickyNotesApplet *applet) { if (event->type == GDK_2BUTTON_PRESS) { popup_add_note (applet, NULL); return TRUE; } else if (event->button == 1) { stickynote_toggle_notes_visible (); return TRUE; } return FALSE; } /* Applet Callback : Keypress on the applet. */ gboolean applet_key_cb (GtkWidget *widget, GdkEventKey *event, StickyNotesApplet *applet) { switch (event->keyval) { case GDK_KEY_KP_Space: case GDK_KEY_space: case GDK_KEY_KP_Enter: case GDK_KEY_Return: stickynote_show_notes (TRUE); return TRUE; } return FALSE; } /* Applet Callback : Cross (enter or leave) the applet. */ gboolean applet_cross_cb (GtkWidget *widget, GdkEventCrossing *event, StickyNotesApplet *applet) { applet->prelighted = event->type == GDK_ENTER_NOTIFY || gtk_widget_has_focus (widget); stickynotes_applet_update_icon (applet); return FALSE; } /* Applet Callback : On focus (in or out) of the applet. */ gboolean applet_focus_cb (GtkWidget *widget, GdkEventFocus *event, StickyNotesApplet *applet) { stickynotes_applet_update_icon (applet); return FALSE; } #ifdef GDK_WINDOWING_X11 static gboolean get_desktop_window (GdkScreen *screen, Window *window) { Window *desktop_window; GdkWindow *root_window; GdkAtom type_returned; int format_returned; int length_returned; /*This is x11-only, so return FALSE in wayland or anything else*/ if (!(GDK_IS_X11_DISPLAY (gdk_screen_get_display (screen)))) return FALSE; root_window = gdk_screen_get_root_window (screen); if (gdk_property_get (root_window, gdk_atom_intern ("CAJA_DESKTOP_WINDOW_ID", FALSE), gdk_x11_xatom_to_atom (XA_WINDOW), 0, 4, FALSE, &type_returned, &format_returned, &length_returned, (guchar**) &desktop_window)) { *window = *desktop_window; g_free (desktop_window); return TRUE; } else { *window = 0; return FALSE; } } static GdkFilterReturn desktop_window_event_filter (GdkXEvent *xevent, GdkEvent *event, gpointer data) { gboolean desktop_hide; GdkScreen *screen = gdk_screen_get_default(); if (!(GDK_IS_X11_DISPLAY (gdk_screen_get_display (screen)))) { desktop_hide = FALSE; return GDK_FILTER_CONTINUE; } desktop_hide = g_settings_get_boolean (stickynotes->settings, "desktop-hide"); if (desktop_hide && (((XEvent*)xevent)->xany.type == PropertyNotify) && (((XEvent*)xevent)->xproperty.atom == gdk_x11_get_xatom_by_name ("_NET_WM_USER_TIME"))) { stickynote_show_notes (FALSE); } return GDK_FILTER_CONTINUE; } #endif /* GDK_WINDOWING_X11 */ void install_check_click_on_desktop (GdkScreen *screen) { #ifdef GDK_WINDOWING_X11 Window desktop_window; GdkWindow *window; Atom user_time_window; Atom user_time; GdkDisplay *display; display = gdk_screen_get_display (screen); /* All this uses X11-only APIs and features */ if (!GDK_IS_X11_DISPLAY (display)) return; if (!get_desktop_window (screen, &desktop_window)) { return; } /* Access the desktop window. desktop_window is the root window for the * default screen, so we know using gdk_display_get_default () is correct. * This code should not be reached if running in wayland */ window = gdk_x11_window_foreign_new_for_display (display, desktop_window); /* Avoid crash if the desktop window ID is set but invalid, e.g. if * Caja has set it but quit since then */ if (!window) { return; } /* It may contain an atom to tell us which other window to monitor */ user_time_window = gdk_x11_get_xatom_by_name ("_NET_WM_USER_TIME_WINDOW"); user_time = gdk_x11_get_xatom_by_name ("_NET_WM_USER_TIME"); if (user_time != None && user_time_window != None) { /* Looks like the atoms are there */ Atom actual_type; Window *data; int actual_format; gulong nitems; gulong bytes; /* We only use this extra property if the actual user-time property's missing */ XGetWindowProperty (GDK_DISPLAY_XDISPLAY (gdk_window_get_display (window)), desktop_window, user_time, 0, 4, False, AnyPropertyType, &actual_type, &actual_format, &nitems, &bytes, (unsigned char **)&data); if (actual_type == None) { /* No user-time property, so look for the user-time-window */ XGetWindowProperty (GDK_DISPLAY_XDISPLAY (gdk_window_get_display (window)), desktop_window, user_time_window, 0, 4, False, AnyPropertyType, &actual_type, &actual_format, &nitems, &bytes, (unsigned char **)&data); if (actual_type != None) { /* We have another window to monitor */ desktop_window = *data; window = gdk_x11_window_foreign_new_for_display (gdk_window_get_display (window), desktop_window); } } } gdk_window_set_events (window, GDK_PROPERTY_CHANGE_MASK); gdk_window_add_filter (window, desktop_window_event_filter, NULL); #endif } /* Applet Callback : Change the panel orientation. */ void applet_change_orient_cb (MatePanelApplet *mate_panel_applet, MatePanelAppletOrient orient, StickyNotesApplet *applet) { applet->panel_orient = orient; } /* Applet Callback : Resize the applet. */ void applet_size_allocate_cb (GtkWidget *widget, GtkAllocation *allocation, StickyNotesApplet *applet) { if ((applet->panel_orient == MATE_PANEL_APPLET_ORIENT_UP) || (applet->panel_orient == MATE_PANEL_APPLET_ORIENT_DOWN)) { if (applet->panel_size == allocation->height) return; applet->panel_size = allocation->height; } else { if (applet->panel_size == allocation->width) return; applet->panel_size = allocation->width; } stickynotes_applet_update_icon (applet); return; } /* Applet Callback : Deletes the applet. */ void applet_destroy_cb (MatePanelApplet *mate_panel_applet, StickyNotesApplet *applet) { GList *notes; stickynotes_save_now (); if (applet->destroy_all_dialog != NULL) gtk_widget_destroy (applet->destroy_all_dialog); if (applet->action_group) g_object_unref (applet->action_group); if (stickynotes->applets != NULL) stickynotes->applets = g_list_remove (stickynotes->applets, applet); if (stickynotes->applets == NULL) { notes = stickynotes->notes; while (notes) { StickyNote *note = notes->data; stickynote_free (note); notes = g_list_next (notes); } } } /* Destroy all response Callback: Callback for the destroy all dialog */ static void destroy_all_response_cb (GtkDialog *dialog, gint id, StickyNotesApplet *applet) { if (id == GTK_RESPONSE_OK) { while (g_list_length (stickynotes->notes) > 0) { StickyNote *note = g_list_nth_data (stickynotes->notes, 0); stickynote_free (note); stickynotes->notes = g_list_remove (stickynotes->notes, note); } } stickynotes_applet_update_tooltips (); stickynotes_save (); gtk_widget_destroy (GTK_WIDGET (dialog)); applet->destroy_all_dialog = NULL; } /* Menu Callback : New Note */ void menu_new_note_cb (GtkAction *action, StickyNotesApplet *applet) { popup_add_note (applet, NULL); } /* Menu Callback : Hide Notes */ void menu_hide_notes_cb (GtkAction *action, StickyNotesApplet *applet) { stickynote_show_notes (FALSE); } /* Menu Callback : Destroy all sticky notes */ void menu_destroy_all_cb (GtkAction *action, StickyNotesApplet *applet) { GtkBuilder *builder; builder = gtk_builder_new (); gtk_builder_add_from_resource (builder, GRESOURCE "sticky-notes-delete-all.ui", NULL); if (applet->destroy_all_dialog != NULL) { gtk_window_set_screen (GTK_WINDOW (applet->destroy_all_dialog), gtk_widget_get_screen (GTK_WIDGET (applet->w_applet))); gtk_window_present (GTK_WINDOW (applet->destroy_all_dialog)); return; } applet->destroy_all_dialog = GTK_WIDGET (gtk_builder_get_object (builder, "delete_all_dialog")); g_object_unref (builder); g_signal_connect (applet->destroy_all_dialog, "response", G_CALLBACK (destroy_all_response_cb), applet); gtk_window_set_screen (GTK_WINDOW (applet->destroy_all_dialog), gtk_widget_get_screen (applet->w_applet)); gtk_widget_show_all (applet->destroy_all_dialog); } /* Menu Callback: Lock/Unlock sticky notes */ void menu_toggle_lock_cb (GtkAction *action, StickyNotesApplet *applet) { gboolean locked = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); if (g_settings_is_writable (stickynotes->settings, "locked")) g_settings_set_boolean (stickynotes->settings, "locked", locked); } /* Menu Callback : Configure preferences */ void menu_preferences_cb (GtkAction *action, StickyNotesApplet *applet) { stickynotes_applet_update_prefs (); gtk_window_set_screen (GTK_WINDOW (stickynotes->w_prefs), gtk_widget_get_screen (applet->w_applet)); gtk_window_present (GTK_WINDOW (stickynotes->w_prefs)); } /* Menu Callback : Show help */ void menu_help_cb (GtkAction *action, StickyNotesApplet *applet) { GError *error = NULL; gtk_show_uri_on_window (NULL, "help:mate-stickynotes-applet", gtk_get_current_event_time (), &error); if (error) { GtkWidget *dialog = gtk_message_dialog_new (NULL, GTK_DIALOG_MODAL, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, _("There was an error displaying help: %s"), error->message); g_signal_connect (dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); gtk_window_set_resizable (GTK_WINDOW (dialog), FALSE); gtk_window_set_screen (GTK_WINDOW (dialog), gtk_widget_get_screen (applet->w_applet)); gtk_widget_show (dialog); g_error_free (error); } } /* Menu Callback : Display About window */ void menu_about_cb (GtkAction *action, StickyNotesApplet *applet) { static const gchar *authors[] = { "Loban A Rahman ", "Davyd Madeley ", NULL }; static const gchar *documenters[] = { "Loban A Rahman ", N_("Sun GNOME Documentation Team "), N_("MATE Documentation Team"), NULL }; #ifdef ENABLE_NLS const char **p; for (p = documenters; *p; ++p) *p = _(*p); #endif gtk_show_about_dialog (NULL, "title", _("About Sticky Notes"), "version", VERSION, "copyright", _("Copyright \xc2\xa9 2002-2003 Loban A Rahman\n" "Copyright \xc2\xa9 2005 Davyd Madeley\n" "Copyright \xc2\xa9 2012-2021 MATE developers"), "comments", _("Sticky Notes for the " "MATE Desktop Environment"), "authors", authors, "documenters", documenters, "translator-credits", _("translator-credits"), "logo-icon-name", "mate-sticky-notes-applet", NULL); } /* Preferences Callback : Save. */ void preferences_save_cb (gpointer data) { gint width; gint height; gboolean sys_color; gboolean sys_font; gboolean sticky; gboolean force_default; gboolean desktop_hide; gdouble adjustment_value; adjustment_value = gtk_adjustment_get_value (stickynotes->w_prefs_width); width = (gint) adjustment_value; adjustment_value = gtk_adjustment_get_value (stickynotes->w_prefs_height); height = (gint) adjustment_value; sys_color = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (stickynotes->w_prefs_sys_color)); sys_font = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (stickynotes->w_prefs_sys_font)); sticky = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (stickynotes->w_prefs_sticky)); force_default = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (stickynotes->w_prefs_force)); desktop_hide = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (stickynotes->w_prefs_desktop)); if (g_settings_is_writable (stickynotes->settings, "default-width")) g_settings_set_int (stickynotes->settings, "default-width", width); if (g_settings_is_writable (stickynotes->settings, "default-height")) g_settings_set_int (stickynotes->settings, "default-height", height); if (g_settings_is_writable (stickynotes->settings, "use-system-color")) g_settings_set_boolean (stickynotes->settings, "use-system-color", sys_color); if (g_settings_is_writable (stickynotes->settings, "use-system-font")) g_settings_set_boolean (stickynotes->settings, "use-system-font", sys_font); if (g_settings_is_writable (stickynotes->settings, "sticky")) g_settings_set_boolean (stickynotes->settings, "sticky", sticky); if (g_settings_is_writable (stickynotes->settings, "force-default")) g_settings_set_boolean (stickynotes->settings, "force-default", force_default); if (g_settings_is_writable (stickynotes->settings, "desktop-hide")) g_settings_set_boolean (stickynotes->settings, "desktop-hide", desktop_hide); } /* Preferences Callback : Change color. */ void preferences_color_cb (GtkWidget *button, gpointer data) { char *color_str, *font_color_str; GdkRGBA color, font_color; gtk_color_chooser_get_rgba (GTK_COLOR_CHOOSER (stickynotes->w_prefs_color), &color); gtk_color_chooser_get_rgba (GTK_COLOR_CHOOSER (stickynotes->w_prefs_font_color), &font_color); color_str = gdk_rgba_to_string (&color); font_color_str = gdk_rgba_to_string (&font_color); g_settings_set_string (stickynotes->settings, "default-color", color_str); g_settings_set_string (stickynotes->settings, "default-font-color", font_color_str); g_free (color_str); g_free (font_color_str); } /* Preferences Callback : Change font. */ void preferences_font_cb (GtkWidget *button, gpointer data) { const char *font_str; font_str = gtk_font_button_get_font_name (GTK_FONT_BUTTON (button)); g_settings_set_string (stickynotes->settings, "default-font", font_str); } /* Preferences Callback : Apply to existing notes. */ void preferences_apply_cb (GSettings *settings, gchar *key, gpointer data) { GList *l; StickyNote *note; if (!strcmp (key, "sticky")) { if (g_settings_get_boolean (settings, key)) for (l = stickynotes->notes; l; l = l->next) { note = l->data; gtk_window_stick (GTK_WINDOW (note->w_window)); } else for (l = stickynotes->notes; l; l = l->next) { note = l->data; gtk_window_unstick (GTK_WINDOW (note->w_window)); } } else if (!strcmp (key, "locked")) { for (l = stickynotes->notes; l; l = l->next) { note = l->data; stickynote_set_locked (note, g_settings_get_boolean (settings, key)); } stickynotes_save (); } else if (!strcmp (key, "use-system-color") || !strcmp (key, "default-color")) { for (l = stickynotes->notes; l; l = l->next) { note = l->data; stickynote_set_color (note, note->color, note->font_color, FALSE); } } else if (!strcmp (key, "use-system-font") || !strcmp (key, "default-font")) { for (l = stickynotes->notes; l; l = l->next) { note = l->data; stickynote_set_font (note, note->font, FALSE); } } else if (!strcmp (key, "force-default")) { for (l = stickynotes->notes; l; l = l->next) { note = l->data; stickynote_set_color (note, note->color, note->font_color, FALSE); stickynote_set_font (note, note->font, FALSE); } } stickynotes_applet_update_prefs (); stickynotes_applet_update_menus (); } /* Preferences Callback : Response. */ void preferences_response_cb (GtkWidget *dialog, gint response, gpointer data) { if (response == GTK_RESPONSE_HELP) { GError *error = NULL; gtk_show_uri_on_window (GTK_WINDOW (dialog), "help:mate-stickynotes-applet/stickynotes-advanced-settings", gtk_get_current_event_time (), &error); if (error) { dialog = gtk_message_dialog_new (NULL, GTK_DIALOG_MODAL, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, _("There was an error displaying help: %s"), error->message); g_signal_connect (dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); gtk_window_set_resizable (GTK_WINDOW (dialog), FALSE); gtk_window_set_screen (GTK_WINDOW (dialog), gtk_widget_get_screen (GTK_WIDGET (dialog))); gtk_widget_show (dialog); g_error_free (error); } } else if (response == GTK_RESPONSE_CLOSE) gtk_widget_hide (GTK_WIDGET (dialog)); } /* Preferences Callback : Delete */ gboolean preferences_delete_cb (GtkWidget *widget, GdkEvent *event, gpointer data) { gtk_widget_hide (widget); return TRUE; }