From c2ef7661a6d0173668e30bf43f2270ca897faddf Mon Sep 17 00:00:00 2001
From: Hugo Villeneuve <hugo@hugovil.com>
Date: Sun, 21 Dec 2014 18:32:35 -0500
Subject: [PATCH] Add Gtk+-3 support

---
 configure.ac  | 63 +++++++++++++++++++++++++++++++++++++++++++++++++--
 src/clock.c   | 41 +++++++++++++++++++++++++++++----
 src/dockapp.c | 49 ++++++++++++++++++++++++++++-----------
 3 files changed, 134 insertions(+), 19 deletions(-)

diff --git a/configure.ac b/configure.ac
index 28892cc..85b50b0 100644
--- a/configure.ac
+++ b/configure.ac
@@ -33,8 +33,67 @@ dnl Checking for X11 library.
 AC_CHECK_LIB(X11, XOpenDisplay, LIBS="${LIBS} -lX11",
              echo "Can't find X11 library" ; exit 1, "${X_LIBS}")
 
-dnl Checks for Gtk+-2.0
-PKG_CHECK_MODULES(GTK, gtk+-2.0 >= 2.8, CFLAGS="${CFLAGS} -DGDK_PIXBUF_DISABLE_DEPRECATED -DGDK_DISABLE_DEPRECATED -DGTK_DISABLE_DEPRECATED")
+dnl --with-gtk3 option
+with_gtk3=no
+
+dnl Checks for Gtk+ (version 3 or 2)
+GTK_TOOLKIT=
+check_gtk2=no
+gtk3_pkg_errors=
+
+dnl First test if Gtk+-3 is available
+GLIB_REQUIRED=2.28
+GTK_REQUIRED=3.10
+GTK_MODULES="gtk+-3.0 >= $GTK_REQUIRED glib-2.0 >= $GLIB_REQUIRED"
+PKG_CHECK_MODULES(GTK3, $GTK_MODULES, pkg_check_gtk=yes, pkg_check_gtk=no)
+
+if test "$pkg_check_gtk" = "no" && test "$with_gtk3" = "yes"; then
+    dnl Error if Gtk+-3 was explicitely requested, but is not found
+    AC_MSG_ERROR($GTK3_PKG_ERRORS)
+fi
+if test "$pkg_check_gtk" = "yes"; then
+    AC_DEFINE(HAVE_GTK3, 1, [Define to 1 if using GTK 3 or later.])
+    GTK_TOOLKIT="GTK3"
+    if test "x$ac_enable_gtk_deprecation_warnings" = x; then
+    AC_DEFINE([GDK_DISABLE_DEPRECATION_WARNINGS], [1],
+        [Define to 1 to disable GTK+/GDK deprecation warnings.])
+    AC_DEFINE([GLIB_DISABLE_DEPRECATION_WARNINGS], [1],
+        [Define to 1 to disable Glib deprecation warnings.])
+    fi
+else
+    dnl Gtk+-3 was not found, try to find Gtk+-2
+    check_gtk2=yes
+    gtk3_pkg_errors="$GTK3_PKG_ERRORS "
+fi
+
+if test "${with_gtk2}" = "yes" || test "$check_gtk2" = "yes"; then
+    dnl Test if Gtk+-2 is available
+    GLIB_REQUIRED=2.10
+    GTK_REQUIRED=2.8
+    GTK_MODULES="gtk+-2.0 >= $GTK_REQUIRED glib-2.0 >= $GLIB_REQUIRED"
+    PKG_CHECK_MODULES(GTK2, $GTK_MODULES, pkg_check_gtk=yes, pkg_check_gtk=no)
+    if test "$pkg_check_gtk" = "no" && test "$with_gtk2" = "yes"; then
+      AC_MSG_ERROR($gtk3_pkg_errors$GTK2_PKG_ERRORS)
+    fi
+    if test "$pkg_check_gtk" = "yes"; then
+        AC_DEFINE(HAVE_GTK2, 1, [Define to 1 if using GTK 2.])
+        GTK_TOOLKIT="GTK2"
+    fi
+fi
+
+if test "${GTK_TOOLKIT}" = "GTK3"; then
+    GTK_CFLAGS=$GTK3_CFLAGS
+    GTK_LIBS=$GTK3_LIBS
+else
+if test "${GTK_TOOLKIT}" = "GTK2"; then
+    GTK_CFLAGS=$GTK2_CFLAGS
+    GTK_LIBS=$GTK2_LIBS
+else
+    dnl Error, no Gtk+-3 or Gtk+-2 was found
+    AC_MSG_ERROR([No suitable Gtk+ version found])
+fi
+fi
+
 AC_SUBST(GTK_CFLAGS)
 AC_SUBST(GTK_LIBS)
 
diff --git a/src/clock.c b/src/clock.c
index 05964b5..3fa1d36 100644
--- a/src/clock.c
+++ b/src/clock.c
@@ -82,9 +82,11 @@
 /* Starting mode is clock mode */
 static int hvclock_mode = CLOCK_MODE;
 
+#if defined (HAVE_GTK2)
 /* In preparation for Gtk+-3 support. */
 #define gtk_widget_get_allocated_width(_widget_) _widget_->allocation.width
 #define gtk_widget_get_allocated_height(_widget_) _widget_->allocation.height
+#endif
 
 static double
 get_clock_face_radius(GtkWidget *clock)
@@ -377,16 +379,32 @@ draw_calendar(GtkWidget *clock, cairo_t *cr)
 }
 
 static gboolean
+#if defined (HAVE_GTK2)
 hvclock_expose(GtkWidget *clock, GdkEventExpose *event)
+#elif defined (HAVE_GTK3)
+draw_handler(GtkWidget *clock, cairo_t *cr, gpointer d G_GNUC_UNUSED)
+#endif
 {
+#if defined (HAVE_GTK2)
 	cairo_t *cr;
+#elif defined (HAVE_GTK3)
+	GdkRectangle rect;
+#endif
 
+#if defined (HAVE_GTK2)
 	/* Get a cairo_t */
 	cr = gdk_cairo_create(gtk_widget_get_window(clock));
 
 	cairo_rectangle(cr, event->area.x, event->area.y,
-			event->area.width, event->area.height);
+		event->area.width, event->area.height);
+#elif defined (HAVE_GTK3)
+	gdk_cairo_get_clip_rectangle(cr, &rect);
+	printf("Redrawing (%d,%d+%d+%d)\n", rect.x, rect.y,
+	       rect.width, rect.height);
+
+	cairo_rectangle(cr, rect.x, rect.y, rect.width, rect.height);
 	cairo_clip(cr);
+#endif
 
 	if (hvclock_mode == CLOCK_MODE) {
 		dockapp_set_mask(clock, clock_mask_xpm);
@@ -397,25 +415,36 @@ hvclock_expose(GtkWidget *clock, GdkEventExpose *event)
 		draw_calendar(clock, cr);
 	}
 
-	cairo_destroy(cr);
-
 	return TRUE;
 }
 
 static void
 hvclock_redraw_canvas(GtkWidget *widget)
 {
+#if defined (HAVE_GTK2)
 	GdkRegion *region;
+#elif defined (HAVE_GTK3)
+	cairo_region_t *region;
+#endif
 
 	if (!gtk_widget_get_window(widget))
 		return;
 
+#if defined (HAVE_GTK2)
 	region = gdk_drawable_get_clip_region(gtk_widget_get_window(widget));
+#elif defined (HAVE_GTK3)
+	region = gdk_window_get_clip_region(gtk_widget_get_window(widget));
+#endif
+
 	/* redraw the cairo canvas completely by exposing it */
 	gdk_window_invalidate_region(gtk_widget_get_window(widget), region, TRUE);
 	gdk_window_process_updates(gtk_widget_get_window(widget), TRUE);
 
+#if defined (HAVE_GTK2)
 	gdk_region_destroy(region);
+#elif defined (HAVE_GTK3)
+	cairo_region_destroy(region);
+#endif
 }
 
 static gboolean
@@ -481,9 +510,13 @@ hvclock_init(GtkWidget *win)
 			      GDK_BUTTON_RELEASE_MASK |
 			      GDK_POINTER_MOTION_MASK);
 
+#if defined (HAVE_GTK2)
 	g_signal_connect(G_OBJECT(win), "expose-event",
 			 G_CALLBACK(hvclock_expose), NULL);
-
+#elif defined (HAVE_GTK3)
+	g_signal_connect(G_OBJECT(win), "draw",
+			 G_CALLBACK(draw_handler), NULL);
+#endif
 	g_signal_connect(G_OBJECT(win), "button-press-event",
 			 G_CALLBACK(hvclock_button_release), NULL);
 	g_signal_connect(G_OBJECT(win), "button-release-event",
diff --git a/src/dockapp.c b/src/dockapp.c
index 8ded719..f01c9f9 100644
--- a/src/dockapp.c
+++ b/src/dockapp.c
@@ -38,14 +38,17 @@
 #ifdef GTK_WITHDRAWN_HACK
 
 static void
-dockapp_set_withdrawn_state(Display *d, Window w)
+dockapp_set_withdrawn_state(Display *d, Window mw, Window iw)
 {
 	XWMHints *hints;
 
-	hints = XGetWMHints(d, w);
-	hints->flags |= StateHint;
-	hints->initial_state = WithdrawnState;
-	XSetWMHints(d, w, hints);
+	hints = XGetWMHints(d, mw);
+	if (!hints) {
+		hints = XAllocWMHints();
+	}
+	hints->flags |= IconWindowHint;
+        hints->icon_window = iw;
+	XSetWMHints(d, mw, hints);
 	XFree(hints);
 }
 
@@ -68,22 +71,22 @@ dockapp_gtk_withdrawn_hack(GtkWidget *dockwin, GtkWidget *iconwin)
 	unsigned int nchildren;
 	Window win_temp;
 	Window win_orig;
-
-	(void) iconwin; /* Unused parameter. */
+        Window iw;
 
 	d = GDK_DISPLAY_XDISPLAY(gdk_display_get_default());
 	win_orig = GDK_WINDOW_XID(gtk_widget_get_window(dockwin));
+	iw = GDK_WINDOW_XID(gtk_widget_get_window(iconwin));
 
 	XQueryTree(d, win_orig, &root, &p, &children, &nchildren);
-
 	if (children)
 		XFree(children);
 
 	win_temp = XCreateSimpleWindow(d, p, 0, 0, 1, 1, 0, 0, 0);
 	XReparentWindow(d, win_orig, win_temp, 0, 0);
 	gtk_widget_show(dockwin);
+        gtk_widget_show(iconwin);
 
-	dockapp_set_withdrawn_state(d, win_orig);
+	dockapp_set_withdrawn_state(d, win_orig, iw);
 
 	XReparentWindow(d, win_orig, p, 0, 0);
 	XDestroyWindow(d, win_temp);
@@ -95,15 +98,22 @@ void
 dockapp_set_mask(GtkWidget *iconwin, char **xpm)
 {
 	GdkPixbuf *pixbuf;
+#if defined (HAVE_GTK2)
 	GdkPixmap *pixmap;
 	GdkBitmap *mask;
+#elif defined (HAVE_GTK3)
+	cairo_surface_t *surface;
+	cairo_region_t *shape;
+#endif
 
 	pixbuf = gdk_pixbuf_new_from_xpm_data((const char **)xpm);
+	if (!pixbuf) {
+		fprintf(stderr, "Could not load pixbuf\n");
+		exit(1);
+	}
 
+#if defined (HAVE_GTK2)
 	gdk_pixbuf_render_pixmap_and_mask(pixbuf, &pixmap, &mask, 127);
-
-	g_object_unref(pixbuf);
-
 	if (!pixmap) {
 		fprintf(stderr, "Could not load pixmap\n");
 		exit(1);
@@ -112,6 +122,16 @@ dockapp_set_mask(GtkWidget *iconwin, char **xpm)
 	gtk_widget_shape_combine_mask(iconwin, mask, 0, 0);
 	g_object_unref(mask);
 	g_object_unref(pixmap);
+#elif defined (HAVE_GTK3)
+	surface = gdk_cairo_surface_create_from_pixbuf(
+		pixbuf, 0,
+		gtk_widget_get_window(iconwin));
+	shape = gdk_cairo_region_create_from_surface(surface);
+	cairo_surface_destroy(surface);
+	gtk_widget_shape_combine_region(iconwin, shape);
+	cairo_region_destroy(shape);
+#endif
+	g_object_unref(pixbuf);
 }
 
 static GtkWidget *
@@ -146,7 +166,10 @@ dockapp_init(int argc, char *argv[])
 	g_signal_connect(dockwin, "destroy", G_CALLBACK(gtk_main_quit), NULL);
 
 	/* Sets the icon of dockwin as iconwin. */
-	gdk_window_set_icon(gtk_widget_get_window(dockwin), gtk_widget_get_window(iconwin), NULL, NULL);
+#if defined (HAVE_GTK2)
+	gdk_window_set_icon(gtk_widget_get_window(dockwin),
+			    gtk_widget_get_window(iconwin), NULL, NULL);
+#endif
 
 	hvclock_init(iconwin);
 	gtk_widget_show(iconwin);
-- 
2.20.1