/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Copyright (C) 2002 CodeFactory AB * Copyright (C) 2002-2003 Richard Hult <richard@imendio.com> * * 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 of the * License, 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 <config.h> #include <string.h> #include <math.h> #include <glib/gi18n.h> #include <gtk/gtk.h> #include <gdk/gdkx.h> #include <gdk/gdkkeysyms.h> #include <gio/gio.h> #ifdef HAVE_CANBERRA_GTK #include <canberra-gtk.h> #endif #include "drwright.h" #include "drw-utils.h" #include "drw-break-window.h" #include "drw-timer.h" struct _DrwBreakWindowPrivate { GtkWidget *clock_label; GtkWidget *break_label; GtkWidget *image; GtkWidget *postpone_entry; GtkWidget *postpone_button; DrwTimer *timer; gint break_time; gchar *break_text; guint clock_timeout_id; guint postpone_timeout_id; guint postpone_sensitize_id; }; #define DRW_BREAK_WINDOW_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), DRW_TYPE_BREAK_WINDOW, DrwBreakWindowPrivate)) #define POSTPONE_CANCEL 30 /* Signals */ enum { DONE, POSTPONE, LAST_SIGNAL }; static void drw_break_window_class_init (DrwBreakWindowClass *klass); static void drw_break_window_init (DrwBreakWindow *window); static void drw_break_window_finalize (GObject *object); static void drw_break_window_dispose (GObject *object); static gboolean postpone_sensitize_cb (DrwBreakWindow *window); static gboolean clock_timeout_cb (DrwBreakWindow *window); static void postpone_clicked_cb (GtkWidget *button, GtkWidget *window); static gboolean label_draw_event_cb (GtkLabel *label, cairo_t *cr, gpointer user_data); static void label_size_request_cb (GtkLabel *label, GtkRequisition *requisition, gpointer user_data); G_DEFINE_TYPE (DrwBreakWindow, drw_break_window, GTK_TYPE_WINDOW) static guint signals[LAST_SIGNAL] = { 0 }; static void drw_break_window_class_init (DrwBreakWindowClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = drw_break_window_finalize; object_class->dispose = drw_break_window_dispose; signals[POSTPONE] = g_signal_new ("postpone", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, 0, NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[DONE] = g_signal_new ("done", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, 0, NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); g_type_class_add_private (klass, sizeof (DrwBreakWindowPrivate)); } static void drw_break_window_init (DrwBreakWindow *window) { DrwBreakWindowPrivate *priv; GtkWidget *vbox; GtkWidget *hbox; gchar *str; GtkWidget *outer_vbox; GtkWidget *button_box; gboolean allow_postpone; gint root_monitor = 0; GdkScreen *screen = NULL; #if GTK_CHECK_VERSION (3, 22, 0) GdkDisplay *display; #endif GdkRectangle monitor; gint right_padding; gint bottom_padding; GSettings *settings; priv = DRW_BREAK_WINDOW_GET_PRIVATE (window); window->priv = priv; settings = g_settings_new (TYPING_BREAK_SCHEMA); priv->break_time = 60 * g_settings_get_int (settings, "break-time"); allow_postpone = g_settings_get_boolean (settings, "allow-postpone"); g_object_unref (settings); g_object_set (window, "type", GTK_WINDOW_POPUP, NULL); gtk_window_set_keep_above (GTK_WINDOW (window), TRUE); gtk_window_fullscreen (GTK_WINDOW (window)); gtk_window_set_modal (GTK_WINDOW (window), TRUE); screen = gdk_screen_get_default (); #if GTK_CHECK_VERSION (3, 22, 0) display = gdk_screen_get_display (screen); gdk_monitor_get_geometry (gdk_display_get_monitor (display, root_monitor), &monitor); #else gdk_screen_get_monitor_geometry (screen, root_monitor, &monitor); #endif gtk_window_set_default_size (GTK_WINDOW (window), WidthOfScreen (gdk_x11_screen_get_xscreen (screen)), HeightOfScreen (gdk_x11_screen_get_xscreen (screen))); gtk_window_set_decorated (GTK_WINDOW (window), FALSE); gtk_widget_set_app_paintable (GTK_WIDGET (window), TRUE); drw_setup_background (GTK_WIDGET (window)); right_padding = WidthOfScreen (gdk_x11_screen_get_xscreen (screen)) - monitor.width - monitor.x; bottom_padding = HeightOfScreen (gdk_x11_screen_get_xscreen (screen)) - monitor.height - monitor.y; outer_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); gtk_widget_set_hexpand (outer_vbox, TRUE); gtk_widget_set_vexpand (outer_vbox, TRUE); gtk_widget_set_margin_top (outer_vbox, monitor.y); gtk_widget_set_margin_bottom (outer_vbox, bottom_padding); gtk_widget_set_margin_start (outer_vbox, monitor.x); gtk_widget_set_margin_end (outer_vbox, right_padding); gtk_widget_show (outer_vbox); gtk_container_add (GTK_CONTAINER (window), outer_vbox); if (allow_postpone) { button_box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0); gtk_widget_show (button_box); gtk_container_set_border_width (GTK_CONTAINER (button_box), 12); priv->postpone_button = gtk_button_new_with_mnemonic (_("_Postpone Break")); gtk_widget_show (priv->postpone_button); gtk_widget_set_sensitive (priv->postpone_button, FALSE); if (priv->postpone_sensitize_id) { g_source_remove (priv->postpone_sensitize_id); } priv->postpone_sensitize_id = g_timeout_add_seconds (5, (GSourceFunc) postpone_sensitize_cb, window); g_signal_connect (priv->postpone_button, "clicked", G_CALLBACK (postpone_clicked_cb), window); gtk_box_pack_end (GTK_BOX (button_box), priv->postpone_button, FALSE, TRUE, 0); priv->postpone_entry = gtk_entry_new (); gtk_entry_set_has_frame (GTK_ENTRY (priv->postpone_entry), FALSE); gtk_box_pack_end (GTK_BOX (button_box), priv->postpone_entry, FALSE, TRUE, 4); gtk_box_pack_end (GTK_BOX (outer_vbox), button_box, FALSE, TRUE, 0); } vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); gtk_widget_set_halign (vbox, GTK_ALIGN_CENTER); gtk_widget_set_valign (vbox, GTK_ALIGN_CENTER); gtk_widget_show (vbox); gtk_box_pack_start (GTK_BOX (outer_vbox), vbox, TRUE, TRUE, 0); hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0); gtk_widget_show (hbox); gtk_box_pack_start (GTK_BOX (vbox), hbox, TRUE, FALSE, 0); priv->image = gtk_image_new_from_icon_name ("process-stop", GTK_ICON_SIZE_DIALOG); gtk_widget_set_halign (priv->image, GTK_ALIGN_END); gtk_widget_set_valign (priv->image, GTK_ALIGN_CENTER); gtk_widget_show (priv->image); gtk_box_pack_start (GTK_BOX (hbox), priv->image, TRUE, TRUE, 8); priv->break_label = gtk_label_new (NULL); gtk_widget_show (priv->break_label); g_signal_connect (priv->break_label, "draw", G_CALLBACK (label_draw_event_cb), NULL); g_signal_connect_after (priv->break_label, "size_request", G_CALLBACK (label_size_request_cb), NULL); str = g_strdup_printf ("<span size=\"xx-large\" foreground=\"white\"><b>%s</b></span>", _("Take a break!")); gtk_label_set_markup (GTK_LABEL (priv->break_label), str); g_free (str); gtk_box_pack_start (GTK_BOX (hbox), priv->break_label, FALSE, FALSE, 12); priv->clock_label = gtk_label_new (NULL); gtk_widget_show (priv->clock_label); gtk_box_pack_start (GTK_BOX (vbox), priv->clock_label, TRUE, TRUE, 8); g_signal_connect (priv->clock_label, "draw", G_CALLBACK (label_draw_event_cb), NULL); g_signal_connect_after (priv->clock_label, "size_request", G_CALLBACK (label_size_request_cb), NULL); gtk_window_stick (GTK_WINDOW (window)); priv->timer = drw_timer_new (); /* Make sure we have a valid time label from the start. */ clock_timeout_cb (window); priv->clock_timeout_id = g_timeout_add (1000, (GSourceFunc) clock_timeout_cb, window); #ifdef HAVE_CANBERRA_GTK ca_context_play (ca_gtk_context_get (), 0, CA_PROP_EVENT_ID, "desktop-screen-lock", NULL); #endif } static void drw_break_window_finalize (GObject *object) { DrwBreakWindow *window = DRW_BREAK_WINDOW (object); DrwBreakWindowPrivate *priv; priv = window->priv; if (priv->clock_timeout_id != 0) { g_source_remove (priv->clock_timeout_id); } if (priv->postpone_timeout_id != 0) { g_source_remove (priv->postpone_timeout_id); } if (priv->postpone_sensitize_id != 0) { g_source_remove (priv->postpone_sensitize_id); } window->priv = NULL; if (G_OBJECT_CLASS (drw_break_window_parent_class)->finalize) { (* G_OBJECT_CLASS (drw_break_window_parent_class)->finalize) (object); } } static void drw_break_window_dispose (GObject *object) { DrwBreakWindow *window = DRW_BREAK_WINDOW (object); DrwBreakWindowPrivate *priv; priv = window->priv; if (priv->timer) { drw_timer_destroy (priv->timer); priv->timer = NULL; } if (priv->clock_timeout_id != 0) { g_source_remove (priv->clock_timeout_id); priv->clock_timeout_id = 0; } if (priv->postpone_timeout_id != 0) { g_source_remove (priv->postpone_timeout_id); priv->postpone_timeout_id = 0; } if (priv->postpone_sensitize_id != 0) { g_source_remove (priv->postpone_sensitize_id); } if (G_OBJECT_CLASS (drw_break_window_parent_class)->dispose) { (* G_OBJECT_CLASS (drw_break_window_parent_class)->dispose) (object); } } GtkWidget * drw_break_window_new (void) { GObject *object; object = g_object_new (DRW_TYPE_BREAK_WINDOW, "type", GTK_WINDOW_POPUP, "skip-taskbar-hint", TRUE, "skip-pager-hint", TRUE, "focus-on-map", TRUE, NULL); return GTK_WIDGET (object); } static gboolean postpone_sensitize_cb (DrwBreakWindow *window) { DrwBreakWindowPrivate *priv; priv = window->priv; gtk_widget_set_sensitive (priv->postpone_button, TRUE); priv->postpone_sensitize_id = 0; return FALSE; } static gboolean clock_timeout_cb (DrwBreakWindow *window) { DrwBreakWindowPrivate *priv; gchar *txt; gint minutes; gint seconds; g_return_val_if_fail (DRW_IS_BREAK_WINDOW (window), FALSE); priv = window->priv; seconds = 1 + priv->break_time - drw_timer_elapsed (priv->timer); seconds = MAX (0, seconds); if (seconds == 0) { /* Zero this out so the finalizer doesn't try to remove the * source, which would be done in the timeout callback == * no-no. */ priv->clock_timeout_id = 0; #ifdef HAVE_CANBERRA_GTK ca_context_play (ca_gtk_context_get (), 0, CA_PROP_EVENT_ID, "alarm-clock-elapsed", NULL); #endif g_signal_emit (window, signals[DONE], 0, NULL); return FALSE; } minutes = seconds / 60; seconds -= minutes * 60; txt = g_strdup_printf ("<span size=\"25000\" foreground=\"white\"><b>%d:%02d</b></span>", minutes, seconds); gtk_label_set_markup (GTK_LABEL (priv->clock_label), txt); g_free (txt); return TRUE; } static void postpone_entry_activate_cb (GtkWidget *entry, DrwBreakWindow *window) { const gchar *str; gchar *phrase; GSettings *settings = g_settings_new (TYPING_BREAK_SCHEMA); str = gtk_entry_get_text (GTK_ENTRY (entry)); phrase = g_settings_get_string (settings, "unlock-phrase"); g_object_unref (settings); if (!strcmp (str, phrase)) { g_signal_emit (window, signals[POSTPONE], 0, NULL); g_free (phrase); return; } g_free (phrase); gtk_entry_set_text (GTK_ENTRY (entry), ""); } static gboolean grab_on_window (GdkWindow *window, guint32 activate_time) { GdkDisplay *display; #if GTK_CHECK_VERSION (3, 20, 0) GdkSeat *seat; #else GdkDeviceManager *device_manager; GdkDevice *pointer; GdkDevice *keyboard; #endif display = gdk_window_get_display (window); #if GTK_CHECK_VERSION (3, 20, 0) seat = gdk_display_get_default_seat (display); return (gdk_seat_grab (seat, window, GDK_SEAT_CAPABILITY_ALL, TRUE, NULL, NULL, NULL, NULL) == GDK_GRAB_SUCCESS); #else device_manager = gdk_display_get_device_manager (display); pointer = gdk_device_manager_get_client_pointer (device_manager); keyboard = gdk_device_get_associated_device (pointer); if ((gdk_device_grab (pointer, window, GDK_OWNERSHIP_NONE, TRUE, GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | GDK_POINTER_MOTION_MASK, NULL, activate_time) == 0)) { if (gdk_device_grab (keyboard, window, GDK_OWNERSHIP_NONE, TRUE, GDK_KEY_PRESS_MASK, NULL, activate_time) == 0) return TRUE; else { gdk_device_ungrab (pointer, activate_time); return FALSE; } } return FALSE; #endif } static gboolean postpone_cancel_cb (DrwBreakWindow *window) { DrwBreakWindowPrivate *priv; priv = window->priv; gtk_entry_set_text (GTK_ENTRY (priv->postpone_entry), ""); gtk_widget_hide (priv->postpone_entry); priv->postpone_timeout_id = 0; return FALSE; } static gboolean postpone_entry_key_press_event_cb (GtkEntry *entry, GdkEventKey *event, DrwBreakWindow *window) { DrwBreakWindowPrivate *priv; priv = window->priv; if (event->keyval == GDK_KEY_Escape) { if (priv->postpone_timeout_id) { g_source_remove (priv->postpone_timeout_id); } postpone_cancel_cb (window); return TRUE; } g_source_remove (priv->postpone_timeout_id); priv->postpone_timeout_id = g_timeout_add_seconds (POSTPONE_CANCEL, (GSourceFunc) postpone_cancel_cb, window); return FALSE; } static void postpone_clicked_cb (GtkWidget *button, GtkWidget *window) { DrwBreakWindow *bw = DRW_BREAK_WINDOW (window); DrwBreakWindowPrivate *priv = bw->priv; gchar *phrase; /* Disable the phrase for now. */ phrase = NULL; /*g_settings_get_string (settings, "unlock-phrase");*/ if (!phrase || !phrase[0]) { g_signal_emit (window, signals[POSTPONE], 0, NULL); return; } if (gtk_widget_get_visible (priv->postpone_entry)) { gtk_widget_activate (priv->postpone_entry); return; } gtk_widget_show (priv->postpone_entry); priv->postpone_timeout_id = g_timeout_add_seconds (POSTPONE_CANCEL, (GSourceFunc) postpone_cancel_cb, bw); grab_on_window (gtk_widget_get_window (priv->postpone_entry), gtk_get_current_event_time ()); gtk_widget_grab_focus (priv->postpone_entry); g_signal_connect (priv->postpone_entry, "activate", G_CALLBACK (postpone_entry_activate_cb), bw); g_signal_connect (priv->postpone_entry, "key_press_event", G_CALLBACK (postpone_entry_key_press_event_cb), bw); } static void get_layout_location (GtkLabel *label, gint *xp, gint *yp) { #if !GTK_CHECK_VERSION (3, 16, 0) GtkMisc *misc; #endif GtkWidget *widget; GtkAllocation widget_allocation; GtkRequisition widget_requisition; gfloat xalign, yalign; gint x, y; gint xpad, ypad; #if GTK_CHECK_VERSION (3, 16, 0) gint margin_start, margin_end, margin_top, margin_bottom; widget = GTK_WIDGET (label); xalign = gtk_label_get_xalign (GTK_LABEL (label)); yalign = gtk_label_get_yalign (GTK_LABEL (label)); margin_start = gtk_widget_get_margin_start (widget); margin_end = gtk_widget_get_margin_end (widget); margin_top = gtk_widget_get_margin_top (widget); margin_bottom = gtk_widget_get_margin_bottom (widget); xpad = margin_start + margin_end; ypad = margin_top + margin_bottom; #else misc = GTK_MISC (label); widget = GTK_WIDGET (label); gtk_misc_get_alignment (misc, &xalign, &yalign); gtk_misc_get_padding (misc, &xpad, &ypad); #endif gtk_widget_get_allocation (widget, &widget_allocation); gtk_widget_get_requisition (widget, &widget_requisition); if (gtk_widget_get_direction (widget) != GTK_TEXT_DIR_LTR) xalign = 1.0 - xalign; x = floor (widget_allocation.x + (int)xpad + ((widget_allocation.width - widget_requisition.width - 1) * xalign) + 0.5); y = floor (widget_allocation.y + (int)ypad + ((widget_allocation.height - widget_requisition.height - 1) * yalign) + 0.5); if (xp) { *xp = x; } if (yp) { *yp = y; } } static gboolean label_draw_event_cb (GtkLabel *label, cairo_t *cr, gpointer user_data) { gint x, y; GtkWidget *widget; get_layout_location (label, &x, &y); widget = GTK_WIDGET (label); pango_cairo_show_layout (cr, gtk_label_get_layout (label)); gtk_paint_layout (gtk_widget_get_style (widget), cr, gtk_widget_get_state (widget), FALSE, widget, "label", x, y, gtk_label_get_layout (label)); return TRUE; } static void label_size_request_cb (GtkLabel *label, GtkRequisition *requisition, gpointer user_data) { requisition->width += 1; requisition->height += 1; }