diff gtk/gtkmenutray.c @ 14191:009db0b357b5

This is a hand-crafted commit to migrate across subversion revisions 16854:16861, due to some vagaries of the way the original renames were done. Witness that monotone can do in one revision what svn had to spread across several.
author Ethan Blanton <elb@pidgin.im>
date Sat, 16 Dec 2006 04:59:55 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/gtk/gtkmenutray.c	Sat Dec 16 04:59:55 2006 +0000
@@ -0,0 +1,254 @@
+/*
+ * Gaim is the legal property of its developers, whose names are too numerous
+ * to list here.  Please refer to the COPYRIGHT file distributed with this
+ * source distribution.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * 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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+#include "debug.h"
+
+#include "gtkmenutray.h"
+
+#include <gtk/gtkeventbox.h>
+#include <gtk/gtkiconfactory.h>
+#include <gtk/gtkversion.h>
+
+/******************************************************************************
+ * Enums
+ *****************************************************************************/
+enum {
+	PROP_ZERO = 0,
+	PROP_BOX
+};
+
+/******************************************************************************
+ * Globals
+ *****************************************************************************/
+static GObjectClass *parent_class = NULL;
+
+/******************************************************************************
+ * Internal Stuff
+ *****************************************************************************/
+
+/******************************************************************************
+ * Item Stuff
+ *****************************************************************************/
+static void
+gaim_gtk_menu_tray_select(GtkItem *item) {
+	/* this may look like nothing, but it's really overriding the
+	 * GtkMenuItem's select function so that it doesn't get highlighted like
+	 * a normal menu item would.
+	 */
+}
+
+static void
+gaim_gtk_menu_tray_deselect(GtkItem *item) {
+	/* Probably not necessary, but I'd rather be safe than sorry.  We're
+	 * overridding the select, so it makes sense to override deselect as well.
+	 */
+}
+
+/******************************************************************************
+ * Widget Stuff
+ *****************************************************************************/
+
+/******************************************************************************
+ * Object Stuff
+ *****************************************************************************/
+static void
+gaim_gtk_menu_tray_get_property(GObject *obj, guint param_id, GValue *value,
+								GParamSpec *pspec)
+{
+	GaimGtkMenuTray *menu_tray = GAIM_GTK_MENU_TRAY(obj);
+
+	switch(param_id) {
+		case PROP_BOX:
+			g_value_set_object(value, gaim_gtk_menu_tray_get_box(menu_tray));
+			break;
+		default:
+			G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec);
+			break;
+	}
+}
+
+static void
+gaim_gtk_menu_tray_finalize(GObject *obj) {
+#if 0
+	/* This _might_ be leaking, but I have a sneaking suspicion that the widget is
+	 * getting destroyed in GtkContainer's finalize function.  But if were are
+	 * leaking here, be sure to figure out why this causes a crash.
+	 *	-- Gary
+	 */
+	GaimGtkMenuTray *tray = GAIM_GTK_MENU_TRAY(obj);
+
+	if(GTK_IS_WIDGET(tray->tray))
+		gtk_widget_destroy(GTK_WIDGET(tray->tray));
+#endif
+
+	G_OBJECT_CLASS(parent_class)->finalize(obj);
+}
+
+static void
+gaim_gtk_menu_tray_class_init(GaimGtkMenuTrayClass *klass) {
+	GObjectClass *object_class = G_OBJECT_CLASS(klass);
+	GtkItemClass *item_class = GTK_ITEM_CLASS(klass);
+	GParamSpec *pspec;
+
+	parent_class = g_type_class_peek_parent(klass);
+
+	object_class->finalize = gaim_gtk_menu_tray_finalize;
+	object_class->get_property = gaim_gtk_menu_tray_get_property;
+
+	item_class->select = gaim_gtk_menu_tray_select;
+	item_class->deselect = gaim_gtk_menu_tray_deselect;
+
+	pspec = g_param_spec_object("box", "The box",
+								"The box",
+								GTK_TYPE_BOX,
+								G_PARAM_READABLE);
+	g_object_class_install_property(object_class, PROP_BOX, pspec);
+}
+
+static void
+gaim_gtk_menu_tray_init(GaimGtkMenuTray *menu_tray) {
+	GtkWidget *widget = GTK_WIDGET(menu_tray);
+#if GTK_CHECK_VERSION(2,2,0)
+	GtkSettings *settings;
+#endif
+	gint height = -1;
+
+	gtk_menu_item_set_right_justified(GTK_MENU_ITEM(menu_tray), TRUE);
+
+	if(!GTK_IS_WIDGET(menu_tray->tray))
+		menu_tray->tray = gtk_hbox_new(FALSE, 0);
+
+	menu_tray->tooltips = gtk_tooltips_new();
+
+#if GTK_CHECK_VERSION(2,2,0)
+	settings =
+		gtk_settings_get_for_screen(gtk_widget_get_screen(widget));
+
+	if(gtk_icon_size_lookup_for_settings(settings, GTK_ICON_SIZE_MENU,
+										 NULL, &height))
+#else
+	if(gtk_icon_size_lookup(GTK_ICON_SIZE_MENU, NULL, &height))
+#endif
+	{
+		gtk_widget_set_size_request(widget, -1, height);
+	}
+
+	gtk_container_add(GTK_CONTAINER(menu_tray), menu_tray->tray);
+
+	gtk_widget_show(menu_tray->tray);
+}
+
+/******************************************************************************
+ * API
+ *****************************************************************************/
+GType
+gaim_gtk_menu_tray_get_gtype(void) {
+	static GType type = 0;
+
+	if(type == 0) {
+		static const GTypeInfo info = {
+			sizeof(GaimGtkMenuTrayClass),
+			NULL,
+			NULL,
+			(GClassInitFunc)gaim_gtk_menu_tray_class_init,
+			NULL,
+			NULL,
+			sizeof(GaimGtkMenuTray),
+			0,
+			(GInstanceInitFunc)gaim_gtk_menu_tray_init,
+			NULL
+		};
+
+		type = g_type_register_static(GTK_TYPE_MENU_ITEM,
+									  "GaimGtkMenuTray",
+									  &info, 0);
+	}
+
+	return type;
+}
+
+GtkWidget *
+gaim_gtk_menu_tray_new() {
+	return g_object_new(GAIM_GTK_TYPE_MENU_TRAY, NULL);
+}
+
+GtkWidget *
+gaim_gtk_menu_tray_get_box(GaimGtkMenuTray *menu_tray) {
+	g_return_val_if_fail(GAIM_GTK_IS_MENU_TRAY(menu_tray), NULL);
+	return menu_tray->tray;
+}
+
+static void
+gaim_gtk_menu_tray_add(GaimGtkMenuTray *menu_tray, GtkWidget *widget,
+					   const char *tooltip, gboolean prepend)
+{
+	g_return_if_fail(GAIM_GTK_IS_MENU_TRAY(menu_tray));
+	g_return_if_fail(GTK_IS_WIDGET(widget));
+
+	if (GTK_WIDGET_NO_WINDOW(widget))
+	{
+		GtkWidget *event;
+
+		event = gtk_event_box_new();
+		gtk_container_add(GTK_CONTAINER(event), widget);
+		gtk_widget_show(event);
+		widget = event;
+	}
+
+	gaim_gtk_menu_tray_set_tooltip(menu_tray, widget, tooltip);
+
+	if (prepend)
+		gtk_box_pack_start(GTK_BOX(menu_tray->tray), widget, FALSE, FALSE, 0);
+	else
+		gtk_box_pack_end(GTK_BOX(menu_tray->tray), widget, FALSE, FALSE, 0);
+}
+
+void
+gaim_gtk_menu_tray_append(GaimGtkMenuTray *menu_tray, GtkWidget *widget, const char *tooltip)
+{
+	gaim_gtk_menu_tray_add(menu_tray, widget, tooltip, FALSE);
+}
+
+void
+gaim_gtk_menu_tray_prepend(GaimGtkMenuTray *menu_tray, GtkWidget *widget, const char *tooltip)
+{
+	gaim_gtk_menu_tray_add(menu_tray, widget, tooltip, TRUE);
+}
+
+void
+gaim_gtk_menu_tray_set_tooltip(GaimGtkMenuTray *menu_tray, GtkWidget *widget, const char *tooltip)
+{
+	if (!menu_tray->tooltips)
+		return;
+
+	/* Should we check whether widget is a child of menu_tray? */
+
+	/*
+	 * If the widget does not have it's own window, then it
+	 * must have automatically been added to an event box
+	 * when it was added to the menu tray.  If this is the
+	 * case, we want to set the tooltip on the widget's parent,
+	 * not on the widget itself.
+	 */
+	if (GTK_WIDGET_NO_WINDOW(widget))
+		widget = widget->parent;
+
+	gtk_tooltips_set_tip(menu_tray->tooltips, widget, tooltip, NULL);
+}
+