2 * Copyright © 2011 William Hua, Ryan Lortie
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the licence, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
19 * Author: William Hua <william@attente.ca>
20 * Ryan Lortie <desrt@desrt.ca>
23 #include "gtkquartz-menu.h"
25 #include <gdk/gdkkeysyms.h>
26 #include "gtkaccelmapprivate.h"
28 #import <Cocoa/Cocoa.h>
31 * Code for key code conversion
33 * Copyright (C) 2009 Paul Davis
36 gtk_quartz_menu_get_unichar (gint key)
38 if (key >= GDK_KEY_A && key <= GDK_KEY_Z)
39 return key + (GDK_KEY_a - GDK_KEY_A);
41 if (key >= GDK_KEY_space && key <= GDK_KEY_asciitilde)
46 case GDK_KEY_BackSpace:
47 return NSBackspaceCharacter;
49 return NSDeleteFunctionKey;
51 return NSPauseFunctionKey;
52 case GDK_KEY_Scroll_Lock:
53 return NSScrollLockFunctionKey;
55 return NSSysReqFunctionKey;
57 return NSHomeFunctionKey;
59 case GDK_KEY_leftarrow:
60 return NSLeftArrowFunctionKey;
63 return NSUpArrowFunctionKey;
65 case GDK_KEY_rightarrow:
66 return NSRightArrowFunctionKey;
68 case GDK_KEY_downarrow:
69 return NSDownArrowFunctionKey;
71 return NSPageUpFunctionKey;
72 case GDK_KEY_Page_Down:
73 return NSPageDownFunctionKey;
75 return NSEndFunctionKey;
77 return NSBeginFunctionKey;
79 return NSSelectFunctionKey;
81 return NSPrintFunctionKey;
83 return NSExecuteFunctionKey;
85 return NSInsertFunctionKey;
87 return NSUndoFunctionKey;
89 return NSRedoFunctionKey;
91 return NSMenuFunctionKey;
93 return NSFindFunctionKey;
95 return NSHelpFunctionKey;
97 return NSBreakFunctionKey;
98 case GDK_KEY_Mode_switch:
99 return NSModeSwitchFunctionKey;
101 return NSF1FunctionKey;
103 return NSF2FunctionKey;
105 return NSF3FunctionKey;
107 return NSF4FunctionKey;
109 return NSF5FunctionKey;
111 return NSF6FunctionKey;
113 return NSF7FunctionKey;
115 return NSF8FunctionKey;
117 return NSF9FunctionKey;
119 return NSF10FunctionKey;
121 return NSF11FunctionKey;
123 return NSF12FunctionKey;
125 return NSF13FunctionKey;
127 return NSF14FunctionKey;
129 return NSF15FunctionKey;
131 return NSF16FunctionKey;
133 return NSF17FunctionKey;
135 return NSF18FunctionKey;
137 return NSF19FunctionKey;
139 return NSF20FunctionKey;
141 return NSF21FunctionKey;
143 return NSF22FunctionKey;
145 return NSF23FunctionKey;
147 return NSF24FunctionKey;
149 return NSF25FunctionKey;
151 return NSF26FunctionKey;
153 return NSF27FunctionKey;
155 return NSF28FunctionKey;
157 return NSF29FunctionKey;
159 return NSF30FunctionKey;
161 return NSF31FunctionKey;
163 return NSF32FunctionKey;
165 return NSF33FunctionKey;
167 return NSF34FunctionKey;
169 return NSF35FunctionKey;
179 typedef struct _GtkQuartzActionObserver GtkQuartzActionObserver;
183 @interface GNSMenu : NSMenu
185 GActionObservable *actions;
189 gboolean with_separators;
192 - (id)initWithTitle:(NSString *)title model:(GMenuModel *)aModel actions:(GActionObservable *)someActions hasSeparators:(BOOL)hasSeparators;
194 - (void)model:(GMenuModel *)model didChangeAtPosition:(NSInteger)position removed:(NSInteger)removed added:(NSInteger)added;
196 - (gboolean)handleChanges;
202 @interface GNSMenuItem : NSMenuItem
207 GActionGroup *actions;
208 GtkQuartzActionObserver *observer;
211 - (id)initWithModel:(GMenuModel *)model index:(NSInteger)index observable:(GActionObservable *)observable;
213 - (void)observableActionAddedWithParameterType:(const GVariantType *)parameterType enabled:(BOOL)enabled state:(GVariant *)state;
214 - (void)observableActionEnabledChangedTo:(BOOL)enabled;
215 - (void)observableActionStateChangedTo:(GVariant *)state;
216 - (void)observableActionRemoved;
218 - (void)didSelectItem:(id)sender;
224 struct _GtkQuartzActionObserver
226 GObject parent_instance;
233 typedef GObjectClass GtkQuartzActionObserverClass;
235 static void gtk_quartz_action_observer_observer_iface_init (GActionObserverInterface *iface);
236 G_DEFINE_TYPE_WITH_CODE (GtkQuartzActionObserver, gtk_quartz_action_observer, G_TYPE_OBJECT,
237 G_IMPLEMENT_INTERFACE (G_TYPE_ACTION_OBSERVER, gtk_quartz_action_observer_observer_iface_init))
240 gtk_quartz_action_observer_action_added (GActionObserver *observer,
241 GActionObservable *observable,
242 const gchar *action_name,
243 const GVariantType *parameter_type,
247 GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
249 [qao->item observableActionAddedWithParameterType:parameter_type enabled:enabled state:state];
253 gtk_quartz_action_observer_action_enabled_changed (GActionObserver *observer,
254 GActionObservable *observable,
255 const gchar *action_name,
258 GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
260 [qao->item observableActionEnabledChangedTo:enabled];
264 gtk_quartz_action_observer_action_state_changed (GActionObserver *observer,
265 GActionObservable *observable,
266 const gchar *action_name,
269 GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
271 [qao->item observableActionStateChangedTo:state];
275 gtk_quartz_action_observer_action_removed (GActionObserver *observer,
276 GActionObservable *observable,
277 const gchar *action_name)
279 GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
281 [qao->item observableActionRemoved];
285 gtk_quartz_action_observer_init (GtkQuartzActionObserver *item)
290 gtk_quartz_action_observer_observer_iface_init (GActionObserverInterface *iface)
292 iface->action_added = gtk_quartz_action_observer_action_added;
293 iface->action_enabled_changed = gtk_quartz_action_observer_action_enabled_changed;
294 iface->action_state_changed = gtk_quartz_action_observer_action_state_changed;
295 iface->action_removed = gtk_quartz_action_observer_action_removed;
299 gtk_quartz_action_observer_class_init (GtkQuartzActionObserverClass *class)
303 static GtkQuartzActionObserver *
304 gtk_quartz_action_observer_new (GNSMenuItem *item)
306 GtkQuartzActionObserver *observer;
308 observer = g_object_new (gtk_quartz_action_observer_get_type (), NULL);
309 observer->item = item;
315 gtk_quartz_menu_handle_changes (gpointer user_data)
317 GNSMenu *menu = user_data;
319 return [menu handleChanges];
323 gtk_quartz_menu_items_changed (GMenuModel *model,
329 GNSMenu *menu = user_data;
331 [menu model:model didChangeAtPosition:position removed:removed added:added];
335 gtk_quartz_set_main_menu (GMenuModel *model,
336 GActionObservable *observable)
338 [NSApp setMainMenu:[[[GNSMenu alloc] initWithTitle:@"Main Menu" model:model actions:observable hasSeparators:NO] autorelease]];
341 @interface GNSMenu ()
343 - (void)appendFromModel:(GMenuModel *)aModel withSeparators:(BOOL)withSeparators;
349 @implementation GNSMenu
351 - (void)model:(GMenuModel *)model didChangeAtPosition:(NSInteger)position removed:(NSInteger)removed added:(NSInteger)added
353 if (update_idle == 0)
354 update_idle = gdk_threads_add_idle (gtk_quartz_menu_handle_changes, self);
357 - (void)appendItemFromModel:(GMenuModel *)aModel atIndex:(gint)index withHeading:(gchar **)heading
361 if ((section = g_menu_model_get_item_link (aModel, index, G_MENU_LINK_SECTION)))
363 g_menu_model_get_item_attribute (aModel, index, G_MENU_ATTRIBUTE_LABEL, "s", heading);
364 [self appendFromModel:section withSeparators:NO];
365 g_object_unref (section);
368 [self addItem:[[[GNSMenuItem alloc] initWithModel:aModel index:index observable:actions] autorelease]];
371 - (void)appendFromModel:(GMenuModel *)aModel withSeparators:(BOOL)withSeparators
375 g_signal_connect (aModel, "items-changed", G_CALLBACK (gtk_quartz_menu_items_changed), self);
376 connected = g_slist_prepend (connected, g_object_ref (aModel));
378 n = g_menu_model_get_n_items (aModel);
380 for (i = 0; i < n; i++)
382 NSInteger ourPosition = [self numberOfItems];
383 gchar *heading = NULL;
385 [self appendItemFromModel:aModel atIndex:i withHeading:&heading];
387 if (withSeparators && ourPosition < [self numberOfItems])
389 NSMenuItem *separator = nil;
393 separator = [[[NSMenuItem alloc] initWithTitle:[NSString stringWithUTF8String:heading] action:NULL keyEquivalent:@""] autorelease];
395 [separator setEnabled:NO];
397 else if (ourPosition > 0)
398 separator = [NSMenuItem separatorItem];
400 if (separator != nil)
401 [self insertItem:separator atIndex:ourPosition];
410 [self removeAllItems];
412 [self appendFromModel:model withSeparators:with_separators];
415 - (gboolean)handleChanges
419 g_signal_handlers_disconnect_by_func (connected->data, gtk_quartz_menu_items_changed, self);
420 g_object_unref (connected->data);
422 connected = g_slist_delete_link (connected, connected);
429 return G_SOURCE_REMOVE;
432 - (id)initWithTitle:(NSString *)title model:(GMenuModel *)aModel actions:(GActionObservable *)someActions hasSeparators:(BOOL)hasSeparators
434 if((self = [super initWithTitle:title]) != nil)
436 [self setAutoenablesItems:NO];
438 model = g_object_ref (aModel);
439 actions = g_object_ref (someActions);
440 with_separators = hasSeparators;
452 g_signal_handlers_disconnect_by_func (connected->data, gtk_quartz_menu_items_changed, self);
453 g_object_unref (connected->data);
455 connected = g_slist_delete_link (connected, connected);
458 g_object_unref (actions);
459 g_object_unref (model);
468 @implementation GNSMenuItem
470 - (id)initWithModel:(GMenuModel *)model index:(NSInteger)index observable:(GActionObservable *)observable
474 if (g_menu_model_get_item_attribute (model, index, G_MENU_ATTRIBUTE_LABEL, "s", &title))
482 if (*from == '_' && from[1])
491 if ((self = [super initWithTitle:[NSString stringWithUTF8String:title ? : ""] action:@selector(didSelectItem:) keyEquivalent:@""]) != nil)
495 g_menu_model_get_item_attribute (model, index, G_MENU_ATTRIBUTE_ACTION, "s", &action);
496 target = g_menu_model_get_item_attribute_value (model, index, G_MENU_ATTRIBUTE_TARGET, NULL);
497 actions = g_object_ref (observable);
498 observer = gtk_quartz_action_observer_new (self);
500 if ((submenu = g_menu_model_get_item_link (model, index, G_MENU_LINK_SUBMENU)))
502 [self setSubmenu:[[[GNSMenu alloc] initWithTitle:[NSString stringWithUTF8String:title] model:submenu actions:observable hasSeparators:YES] autorelease]];
503 g_object_unref (submenu);
506 else if (action != NULL)
511 g_action_observable_register_observer (observable, action, G_ACTION_OBSERVER (observer));
513 path = _gtk_accel_path_for_action (action, target);
514 if (gtk_accel_map_lookup_entry (path, &key))
516 unichar character = gtk_quartz_menu_get_unichar (key.accel_key);
520 NSUInteger modifiers = 0;
522 if (key.accel_mods & GDK_SHIFT_MASK)
523 modifiers |= NSShiftKeyMask;
525 if (key.accel_mods & GDK_MOD1_MASK)
526 modifiers |= NSAlternateKeyMask;
528 if (key.accel_mods & GDK_CONTROL_MASK)
529 modifiers |= NSControlKeyMask;
531 if (key.accel_mods & GDK_META_MASK)
532 modifiers |= NSCommandKeyMask;
534 [self setKeyEquivalent:[NSString stringWithCharacters:&character length:1]];
535 [self setKeyEquivalentModifierMask:modifiers];
541 [self setTarget:self];
544 const GVariantType *parameterType;
547 if (g_action_group_query_action (actions, action, &enabled, ¶meterType, NULL, NULL, &state))
548 [self observableActionAddedWithParameterType:parameterType enabled:enabled state:state];
550 [self setEnabled:NO];
561 if (observer != NULL)
562 g_object_unref (observer);
565 g_object_unref (actions);
568 g_variant_unref (target);
575 - (void)observableActionAddedWithParameterType:(const GVariantType *)parameterType enabled:(BOOL)enabled state:(GVariant *)state
577 canActivate = (target == NULL && parameterType == NULL) ||
578 (target != NULL && parameterType != NULL &&
579 g_variant_is_of_type (target, parameterType));
583 if (target != NULL && state != NULL)
585 [self setOnStateImage:[NSImage imageNamed:@"NSMenuRadio"]];
586 [self setState:g_variant_equal (state, target) ? NSOnState : NSOffState];
588 else if (state != NULL && g_variant_is_of_type (state, G_VARIANT_TYPE_BOOLEAN))
590 [self setOnStateImage:[NSImage imageNamed:@"NSMenuCheckmark"]];
591 [self setState:g_variant_get_boolean (state) ? NSOnState : NSOffState];
594 [self setState:NSOffState];
596 [self setEnabled:enabled];
599 [self setEnabled:NO];
602 - (void)observableActionEnabledChangedTo:(BOOL)enabled
605 [self setEnabled:enabled];
608 - (void)observableActionStateChangedTo:(GVariant *)state
613 [self setState:g_variant_equal (state, target) ? NSOnState : NSOffState];
614 else if (g_variant_is_of_type (state, G_VARIANT_TYPE_BOOLEAN))
615 [self setState:g_variant_get_boolean (state) ? NSOnState : NSOffState];
619 - (void)observableActionRemoved
622 [self setEnabled:NO];
625 - (void)didSelectItem:(id)sender
628 g_action_group_activate_action (actions, action, target);