[gtk+] Refactor GMenuModel rendering for Mac OS.
- From: Ryan Lortie <ryanl src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gtk+] Refactor GMenuModel rendering for Mac OS.
- Date: Mon, 19 Dec 2011 18:03:53 +0000 (UTC)
commit 1a02fc036dc0423991840ba3949febe22f1f149f
Author: William Hua <william attente ca>
Date: Sun Dec 11 21:49:21 2011 -0500
Refactor GMenuModel rendering for Mac OS.
gtk/gtkquartz-menu.c | 191 ++++++++++++++++++++++++++++++++-----------------
1 files changed, 125 insertions(+), 66 deletions(-)
---
diff --git a/gtk/gtkquartz-menu.c b/gtk/gtkquartz-menu.c
index a98b179..a68b27b 100644
--- a/gtk/gtkquartz-menu.c
+++ b/gtk/gtkquartz-menu.c
@@ -1,5 +1,5 @@
/*
- * Copyright  2011 William Hua
+ * Copyright  2011 William Hua, Ryan Lortie
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
@@ -17,6 +17,7 @@
* Boston, MA 02111-1307, USA.
*
* Author: William Hua <william attente ca>
+ * Ryan Lortie <desrt desrt ca>
*/
#include "gtkquartz-menu.h"
@@ -177,6 +178,25 @@ gtk_quartz_menu_get_unichar (gint key)
typedef struct _GtkQuartzActionObserver GtkQuartzActionObserver;
+
+
+ interface GNSMenu : NSMenu
+{
+ GActionObservable *actions;
+ GMenuModel *model;
+ guint update_idle;
+ GSList *connected;
+ gboolean with_separators;
+}
+
+- (id)initWithTitle:(NSString *)title model:(GMenuModel *)aModel actions:(GActionObservable *)someActions hasSeparators:(BOOL)hasSeparators;
+
+- (void)model:(GMenuModel *)model didChangeAtPosition:(NSInteger)position removed:(NSInteger)removed added:(NSInteger)added;
+
+ end
+
+
+
@interface GNSMenuItem : NSMenuItem
{
gchar *action;
@@ -207,6 +227,7 @@ struct _GtkQuartzActionObserver
};
+
typedef GObjectClass GtkQuartzActionObserverClass;
static void gtk_quartz_action_observer_observer_iface_init (GActionObserverInterface *iface);
@@ -288,100 +309,131 @@ gtk_quartz_action_observer_new (GNSMenuItem *item)
return observer;
}
-static NSMenu *
-gtk_quartz_menu_create_menu (const gchar *title,
- GMenuModel *model,
- GActionObservable *observable)
+static void
+gtk_quartz_menu_items_changed (GMenuModel *model,
+ gint position,
+ gint removed,
+ gint added,
+ gpointer user_data)
{
- if (model == NULL)
- return nil;
-
- NSMenu *menu = [[[NSMenu alloc] initWithTitle:[NSString stringWithUTF8String:title ? : ""]] autorelease];
+ GNSMenu *menu = user_data;
- [menu setAutoenablesItems:NO];
+ [menu model:model didChangeAtPosition:position removed:removed added:added];
+}
- gint n = g_menu_model_get_n_items (model);
- gint i;
+void
+gtk_quartz_set_main_menu (GMenuModel *model,
+ GActionObservable *observable)
+{
+ [NSApp setMainMenu:[[[GNSMenu alloc] initWithTitle:@"Main Menu" model:model actions:observable hasSeparators:NO] autorelease]];
+}
- for (i = 0; i < n; i++)
- {
- gchar *label = NULL;
+ interface GNSMenu ()
- if (g_menu_model_get_item_attribute (model, i, G_MENU_ATTRIBUTE_LABEL, "s", &label))
- {
- gchar *from, *to;
+- (void)appendFromModel:(GMenuModel *)aModel withSeparators:(BOOL)withSeparators;
- to = from = label;
+ end
- while (*from)
- {
- if (*from == '_' && from[1])
- from++;
- *to++ = *from++;
- }
- *to = '\0';
- }
+ implementation GNSMenu
- NSString *text = [NSString stringWithUTF8String:label ? : ""];
+- (void)model:(GMenuModel *)model didChangeAtPosition:(NSInteger)position removed:(NSInteger)removed added:(NSInteger)added
+{
+}
- GMenuModel *section_model = g_menu_model_get_item_link (model, i, G_MENU_LINK_SECTION);
- GMenuModel *submenu_model = g_menu_model_get_item_link (model, i, G_MENU_LINK_SUBMENU);
+- (void)appendItemFromModel:(GMenuModel *)aModel atIndex:(gint)index withHeading:(gchar **)heading
+{
+ GMenuModel *section;
- NSMenu *section = gtk_quartz_menu_create_menu (label, section_model, observable);
- NSMenu *submenu = gtk_quartz_menu_create_menu (label, submenu_model, observable);
+ if ((section = g_menu_model_get_item_link (aModel, index, G_MENU_LINK_SECTION)))
+ {
+ g_menu_model_get_item_attribute (aModel, index, G_MENU_ATTRIBUTE_LABEL, "s", heading);
+ [self appendFromModel:section withSeparators:NO];
+ g_object_unref (section);
+ }
+ else
+ [self addItem:[[[GNSMenuItem alloc] initWithModel:aModel index:index observable:actions] autorelease]];
+}
- if (section_model != NULL)
- g_object_unref (section_model);
+- (void)appendFromModel:(GMenuModel *)aModel withSeparators:(BOOL)withSeparators
+{
+ gint n, i;
- if (submenu_model != NULL)
- g_object_unref (submenu_model);
+ g_signal_connect (aModel, "items-changed", G_CALLBACK (gtk_quartz_menu_items_changed), self);
+ connected = g_slist_prepend (connected, g_object_ref (aModel));
- if (section != nil)
- {
- if ([menu numberOfItems] > 0)
- [menu addItem:[NSMenuItem separatorItem]];
+ n = g_menu_model_get_n_items (aModel);
- if ([text length] > 0)
- {
- NSMenuItem *header = [[[NSMenuItem alloc] initWithTitle:text action:NULL keyEquivalent:@""] autorelease];
+ for (i = 0; i < n; i++)
+ {
+ NSInteger ourPosition = [self numberOfItems];
+ gchar *heading = NULL;
- [header setEnabled:NO];
+ [self appendItemFromModel:aModel atIndex:i withHeading:&heading];
- [menu addItem:header];
- }
+ if (withSeparators && ourPosition < [self numberOfItems])
+ {
+ NSMenuItem *separator = nil;
- for (NSMenuItem *item in [section itemArray])
+ if (heading)
{
- [item retain];
- [[item menu] removeItem:item];
- [menu addItem:item];
- [item release];
+ separator = [[[NSMenuItem alloc] initWithTitle:[NSString stringWithUTF8String:heading] action:NULL keyEquivalent:@""] autorelease];
+
+ [separator setEnabled:NO];
}
+ else if (ourPosition > 0)
+ separator = [NSMenuItem separatorItem];
+
+ if (separator != nil)
+ [self insertItem:separator atIndex:ourPosition];
}
- else if (submenu != nil)
- {
- NSMenuItem *item = [[[NSMenuItem alloc] initWithTitle:text action:NULL keyEquivalent:@""] autorelease];
- [item setSubmenu:submenu];
+ g_free (heading);
+ }
+}
+
+- (void)populate
+{
+ [self removeAllItems];
- [menu addItem:item];
- }
- else
- [menu addItem:[[[GNSMenuItem alloc] initWithModel:model index:i observable:observable] autorelease]];
+ [self appendFromModel:model withSeparators:with_separators];
+}
+
+- (id)initWithTitle:(NSString *)title model:(GMenuModel *)aModel actions:(GActionObservable *)someActions hasSeparators:(BOOL)hasSeparators
+{
+ if((self = [super initWithTitle:title]) != nil)
+ {
+ [self setAutoenablesItems:NO];
+
+ model = g_object_ref (aModel);
+ actions = g_object_ref (someActions);
+ with_separators = hasSeparators;
+
+ [self populate];
}
- return menu;
+ return self;
}
-void
-gtk_quartz_set_main_menu (GMenuModel *model,
- GActionObservable *observable)
+- (void)dealloc
{
- [NSApp setMainMenu:gtk_quartz_menu_create_menu ("Main Menu", model, observable)];
+ while (connected)
+ {
+ g_signal_handlers_disconnect_by_func (connected->data, gtk_quartz_menu_items_changed, self);
+ g_object_unref (connected->data);
+
+ connected = g_slist_delete_link (connected, connected);
+ }
+
+ g_object_unref (actions);
+ g_object_unref (model);
+
+ [super dealloc];
}
+ end
+
@implementation GNSMenuItem
@@ -409,13 +461,20 @@ gtk_quartz_set_main_menu (GMenuModel *model,
if ((self = [super initWithTitle:[NSString stringWithUTF8String:title ? : ""] action:@selector(didSelectItem:) keyEquivalent:@""]) != nil)
{
+ GMenuModel *submenu;
g_menu_model_get_item_attribute (model, index, G_MENU_ATTRIBUTE_ACTION, "s", &action);
target = g_menu_model_get_item_attribute_value (model, index, G_MENU_ATTRIBUTE_TARGET, NULL);
actions = g_object_ref (observable);
observer = gtk_quartz_action_observer_new (self);
- if (action != NULL)
+ if ((submenu = g_menu_model_get_item_link (model, index, G_MENU_LINK_SUBMENU)))
+ {
+ [self setSubmenu:[[[GNSMenu alloc] initWithTitle:[NSString stringWithUTF8String:title] model:submenu actions:observable hasSeparators:YES] autorelease]];
+ g_object_unref (submenu);
+ }
+
+ else if (action != NULL)
{
GtkAccelKey key;
gchar *path;
@@ -456,7 +515,7 @@ gtk_quartz_set_main_menu (GMenuModel *model,
const GVariantType *parameterType;
GVariant *state;
- if (g_action_group_query_action (G_ACTION_GROUP (actions), action, &enabled, ¶meterType, NULL, NULL, &state))
+ if (g_action_group_query_action (actions, action, &enabled, ¶meterType, NULL, NULL, &state))
[self observableActionAddedWithParameterType:parameterType enabled:enabled state:state];
else
[self setEnabled:NO];
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]