forked from AuroraMiddleware/gtk
632 lines
18 KiB
C
632 lines
18 KiB
C
/*
|
|
* 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
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the licence, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* Author: William Hua <william@attente.ca>
|
|
* Ryan Lortie <desrt@desrt.ca>
|
|
*/
|
|
|
|
#include "gtkquartz-menu.h"
|
|
|
|
#include <gdk/gdkkeysyms.h>
|
|
#include "gtkaccelmapprivate.h"
|
|
|
|
#import <Cocoa/Cocoa.h>
|
|
|
|
/*
|
|
* Code for key code conversion
|
|
*
|
|
* Copyright (C) 2009 Paul Davis
|
|
*/
|
|
static unichar
|
|
gtk_quartz_menu_get_unichar (gint key)
|
|
{
|
|
if (key >= GDK_KEY_A && key <= GDK_KEY_Z)
|
|
return key + (GDK_KEY_a - GDK_KEY_A);
|
|
|
|
if (key >= GDK_KEY_space && key <= GDK_KEY_asciitilde)
|
|
return key;
|
|
|
|
switch (key)
|
|
{
|
|
case GDK_KEY_BackSpace:
|
|
return NSBackspaceCharacter;
|
|
case GDK_KEY_Delete:
|
|
return NSDeleteFunctionKey;
|
|
case GDK_KEY_Pause:
|
|
return NSPauseFunctionKey;
|
|
case GDK_KEY_Scroll_Lock:
|
|
return NSScrollLockFunctionKey;
|
|
case GDK_KEY_Sys_Req:
|
|
return NSSysReqFunctionKey;
|
|
case GDK_KEY_Home:
|
|
return NSHomeFunctionKey;
|
|
case GDK_KEY_Left:
|
|
case GDK_KEY_leftarrow:
|
|
return NSLeftArrowFunctionKey;
|
|
case GDK_KEY_Up:
|
|
case GDK_KEY_uparrow:
|
|
return NSUpArrowFunctionKey;
|
|
case GDK_KEY_Right:
|
|
case GDK_KEY_rightarrow:
|
|
return NSRightArrowFunctionKey;
|
|
case GDK_KEY_Down:
|
|
case GDK_KEY_downarrow:
|
|
return NSDownArrowFunctionKey;
|
|
case GDK_KEY_Page_Up:
|
|
return NSPageUpFunctionKey;
|
|
case GDK_KEY_Page_Down:
|
|
return NSPageDownFunctionKey;
|
|
case GDK_KEY_End:
|
|
return NSEndFunctionKey;
|
|
case GDK_KEY_Begin:
|
|
return NSBeginFunctionKey;
|
|
case GDK_KEY_Select:
|
|
return NSSelectFunctionKey;
|
|
case GDK_KEY_Print:
|
|
return NSPrintFunctionKey;
|
|
case GDK_KEY_Execute:
|
|
return NSExecuteFunctionKey;
|
|
case GDK_KEY_Insert:
|
|
return NSInsertFunctionKey;
|
|
case GDK_KEY_Undo:
|
|
return NSUndoFunctionKey;
|
|
case GDK_KEY_Redo:
|
|
return NSRedoFunctionKey;
|
|
case GDK_KEY_Menu:
|
|
return NSMenuFunctionKey;
|
|
case GDK_KEY_Find:
|
|
return NSFindFunctionKey;
|
|
case GDK_KEY_Help:
|
|
return NSHelpFunctionKey;
|
|
case GDK_KEY_Break:
|
|
return NSBreakFunctionKey;
|
|
case GDK_KEY_Mode_switch:
|
|
return NSModeSwitchFunctionKey;
|
|
case GDK_KEY_F1:
|
|
return NSF1FunctionKey;
|
|
case GDK_KEY_F2:
|
|
return NSF2FunctionKey;
|
|
case GDK_KEY_F3:
|
|
return NSF3FunctionKey;
|
|
case GDK_KEY_F4:
|
|
return NSF4FunctionKey;
|
|
case GDK_KEY_F5:
|
|
return NSF5FunctionKey;
|
|
case GDK_KEY_F6:
|
|
return NSF6FunctionKey;
|
|
case GDK_KEY_F7:
|
|
return NSF7FunctionKey;
|
|
case GDK_KEY_F8:
|
|
return NSF8FunctionKey;
|
|
case GDK_KEY_F9:
|
|
return NSF9FunctionKey;
|
|
case GDK_KEY_F10:
|
|
return NSF10FunctionKey;
|
|
case GDK_KEY_F11:
|
|
return NSF11FunctionKey;
|
|
case GDK_KEY_F12:
|
|
return NSF12FunctionKey;
|
|
case GDK_KEY_F13:
|
|
return NSF13FunctionKey;
|
|
case GDK_KEY_F14:
|
|
return NSF14FunctionKey;
|
|
case GDK_KEY_F15:
|
|
return NSF15FunctionKey;
|
|
case GDK_KEY_F16:
|
|
return NSF16FunctionKey;
|
|
case GDK_KEY_F17:
|
|
return NSF17FunctionKey;
|
|
case GDK_KEY_F18:
|
|
return NSF18FunctionKey;
|
|
case GDK_KEY_F19:
|
|
return NSF19FunctionKey;
|
|
case GDK_KEY_F20:
|
|
return NSF20FunctionKey;
|
|
case GDK_KEY_F21:
|
|
return NSF21FunctionKey;
|
|
case GDK_KEY_F22:
|
|
return NSF22FunctionKey;
|
|
case GDK_KEY_F23:
|
|
return NSF23FunctionKey;
|
|
case GDK_KEY_F24:
|
|
return NSF24FunctionKey;
|
|
case GDK_KEY_F25:
|
|
return NSF25FunctionKey;
|
|
case GDK_KEY_F26:
|
|
return NSF26FunctionKey;
|
|
case GDK_KEY_F27:
|
|
return NSF27FunctionKey;
|
|
case GDK_KEY_F28:
|
|
return NSF28FunctionKey;
|
|
case GDK_KEY_F29:
|
|
return NSF29FunctionKey;
|
|
case GDK_KEY_F30:
|
|
return NSF30FunctionKey;
|
|
case GDK_KEY_F31:
|
|
return NSF31FunctionKey;
|
|
case GDK_KEY_F32:
|
|
return NSF32FunctionKey;
|
|
case GDK_KEY_F33:
|
|
return NSF33FunctionKey;
|
|
case GDK_KEY_F34:
|
|
return NSF34FunctionKey;
|
|
case GDK_KEY_F35:
|
|
return NSF35FunctionKey;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return '\0';
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
- (gboolean)handleChanges;
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface GNSMenuItem : NSMenuItem
|
|
{
|
|
gchar *action;
|
|
GVariant *target;
|
|
BOOL canActivate;
|
|
GActionGroup *actions;
|
|
GtkQuartzActionObserver *observer;
|
|
}
|
|
|
|
- (id)initWithModel:(GMenuModel *)model index:(NSInteger)index observable:(GActionObservable *)observable;
|
|
|
|
- (void)observableActionAddedWithParameterType:(const GVariantType *)parameterType enabled:(BOOL)enabled state:(GVariant *)state;
|
|
- (void)observableActionEnabledChangedTo:(BOOL)enabled;
|
|
- (void)observableActionStateChangedTo:(GVariant *)state;
|
|
- (void)observableActionRemoved;
|
|
|
|
- (void)didSelectItem:(id)sender;
|
|
|
|
@end
|
|
|
|
|
|
|
|
struct _GtkQuartzActionObserver
|
|
{
|
|
GObject parent_instance;
|
|
|
|
GNSMenuItem *item;
|
|
};
|
|
|
|
|
|
|
|
typedef GObjectClass GtkQuartzActionObserverClass;
|
|
|
|
static void gtk_quartz_action_observer_observer_iface_init (GActionObserverInterface *iface);
|
|
G_DEFINE_TYPE_WITH_CODE (GtkQuartzActionObserver, gtk_quartz_action_observer, G_TYPE_OBJECT,
|
|
G_IMPLEMENT_INTERFACE (G_TYPE_ACTION_OBSERVER, gtk_quartz_action_observer_observer_iface_init))
|
|
|
|
static void
|
|
gtk_quartz_action_observer_action_added (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name,
|
|
const GVariantType *parameter_type,
|
|
gboolean enabled,
|
|
GVariant *state)
|
|
{
|
|
GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
|
|
|
|
[qao->item observableActionAddedWithParameterType:parameter_type enabled:enabled state:state];
|
|
}
|
|
|
|
static void
|
|
gtk_quartz_action_observer_action_enabled_changed (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name,
|
|
gboolean enabled)
|
|
{
|
|
GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
|
|
|
|
[qao->item observableActionEnabledChangedTo:enabled];
|
|
}
|
|
|
|
static void
|
|
gtk_quartz_action_observer_action_state_changed (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name,
|
|
GVariant *state)
|
|
{
|
|
GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
|
|
|
|
[qao->item observableActionStateChangedTo:state];
|
|
}
|
|
|
|
static void
|
|
gtk_quartz_action_observer_action_removed (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name)
|
|
{
|
|
GtkQuartzActionObserver *qao = (GtkQuartzActionObserver *) observer;
|
|
|
|
[qao->item observableActionRemoved];
|
|
}
|
|
|
|
static void
|
|
gtk_quartz_action_observer_init (GtkQuartzActionObserver *item)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gtk_quartz_action_observer_observer_iface_init (GActionObserverInterface *iface)
|
|
{
|
|
iface->action_added = gtk_quartz_action_observer_action_added;
|
|
iface->action_enabled_changed = gtk_quartz_action_observer_action_enabled_changed;
|
|
iface->action_state_changed = gtk_quartz_action_observer_action_state_changed;
|
|
iface->action_removed = gtk_quartz_action_observer_action_removed;
|
|
}
|
|
|
|
static void
|
|
gtk_quartz_action_observer_class_init (GtkQuartzActionObserverClass *class)
|
|
{
|
|
}
|
|
|
|
static GtkQuartzActionObserver *
|
|
gtk_quartz_action_observer_new (GNSMenuItem *item)
|
|
{
|
|
GtkQuartzActionObserver *observer;
|
|
|
|
observer = g_object_new (gtk_quartz_action_observer_get_type (), NULL);
|
|
observer->item = item;
|
|
|
|
return observer;
|
|
}
|
|
|
|
static gboolean
|
|
gtk_quartz_menu_handle_changes (gpointer user_data)
|
|
{
|
|
GNSMenu *menu = user_data;
|
|
|
|
return [menu handleChanges];
|
|
}
|
|
|
|
static void
|
|
gtk_quartz_menu_items_changed (GMenuModel *model,
|
|
gint position,
|
|
gint removed,
|
|
gint added,
|
|
gpointer user_data)
|
|
{
|
|
GNSMenu *menu = user_data;
|
|
|
|
[menu model:model didChangeAtPosition:position removed:removed added:added];
|
|
}
|
|
|
|
void
|
|
gtk_quartz_set_main_menu (GMenuModel *model,
|
|
GActionObservable *observable)
|
|
{
|
|
[NSApp setMainMenu:[[[GNSMenu alloc] initWithTitle:@"Main Menu" model:model actions:observable hasSeparators:NO] autorelease]];
|
|
}
|
|
|
|
@interface GNSMenu ()
|
|
|
|
- (void)appendFromModel:(GMenuModel *)aModel withSeparators:(BOOL)withSeparators;
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation GNSMenu
|
|
|
|
- (void)model:(GMenuModel *)model didChangeAtPosition:(NSInteger)position removed:(NSInteger)removed added:(NSInteger)added
|
|
{
|
|
if (update_idle == 0)
|
|
update_idle = gdk_threads_add_idle (gtk_quartz_menu_handle_changes, self);
|
|
}
|
|
|
|
- (void)appendItemFromModel:(GMenuModel *)aModel atIndex:(gint)index withHeading:(gchar **)heading
|
|
{
|
|
GMenuModel *section;
|
|
|
|
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]];
|
|
}
|
|
|
|
- (void)appendFromModel:(GMenuModel *)aModel withSeparators:(BOOL)withSeparators
|
|
{
|
|
gint n, i;
|
|
|
|
g_signal_connect (aModel, "items-changed", G_CALLBACK (gtk_quartz_menu_items_changed), self);
|
|
connected = g_slist_prepend (connected, g_object_ref (aModel));
|
|
|
|
n = g_menu_model_get_n_items (aModel);
|
|
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
NSInteger ourPosition = [self numberOfItems];
|
|
gchar *heading = NULL;
|
|
|
|
[self appendItemFromModel:aModel atIndex:i withHeading:&heading];
|
|
|
|
if (withSeparators && ourPosition < [self numberOfItems])
|
|
{
|
|
NSMenuItem *separator = nil;
|
|
|
|
if (heading)
|
|
{
|
|
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];
|
|
}
|
|
|
|
g_free (heading);
|
|
}
|
|
}
|
|
|
|
- (void)populate
|
|
{
|
|
[self removeAllItems];
|
|
|
|
[self appendFromModel:model withSeparators:with_separators];
|
|
}
|
|
|
|
- (gboolean)handleChanges
|
|
{
|
|
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);
|
|
}
|
|
|
|
[self populate];
|
|
|
|
update_idle = 0;
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
- (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 self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
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
|
|
|
|
- (id)initWithModel:(GMenuModel *)model index:(NSInteger)index observable:(GActionObservable *)observable
|
|
{
|
|
gchar *title = NULL;
|
|
|
|
if (g_menu_model_get_item_attribute (model, index, G_MENU_ATTRIBUTE_LABEL, "s", &title))
|
|
{
|
|
gchar *from, *to;
|
|
|
|
to = from = title;
|
|
|
|
while (*from)
|
|
{
|
|
if (*from == '_' && from[1])
|
|
from++;
|
|
|
|
*to++ = *from++;
|
|
}
|
|
|
|
*to = '\0';
|
|
}
|
|
|
|
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 ((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;
|
|
|
|
g_action_observable_register_observer (observable, action, G_ACTION_OBSERVER (observer));
|
|
|
|
path = _gtk_accel_path_for_action (action, target);
|
|
if (gtk_accel_map_lookup_entry (path, &key))
|
|
{
|
|
unichar character = gtk_quartz_menu_get_unichar (key.accel_key);
|
|
|
|
if (character)
|
|
{
|
|
NSUInteger modifiers = 0;
|
|
|
|
if (key.accel_mods & GDK_SHIFT_MASK)
|
|
modifiers |= NSShiftKeyMask;
|
|
|
|
if (key.accel_mods & GDK_MOD1_MASK)
|
|
modifiers |= NSAlternateKeyMask;
|
|
|
|
if (key.accel_mods & GDK_CONTROL_MASK)
|
|
modifiers |= NSControlKeyMask;
|
|
|
|
if (key.accel_mods & GDK_META_MASK)
|
|
modifiers |= NSCommandKeyMask;
|
|
|
|
[self setKeyEquivalent:[NSString stringWithCharacters:&character length:1]];
|
|
[self setKeyEquivalentModifierMask:modifiers];
|
|
}
|
|
}
|
|
|
|
g_free (path);
|
|
|
|
[self setTarget:self];
|
|
|
|
gboolean enabled;
|
|
const GVariantType *parameterType;
|
|
GVariant *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];
|
|
}
|
|
}
|
|
|
|
g_free (title);
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
if (observer != NULL)
|
|
g_object_unref (observer);
|
|
|
|
if (actions != NULL)
|
|
g_object_unref (actions);
|
|
|
|
if (target != NULL)
|
|
g_variant_unref (target);
|
|
|
|
g_free (action);
|
|
|
|
[super dealloc];
|
|
}
|
|
|
|
- (void)observableActionAddedWithParameterType:(const GVariantType *)parameterType enabled:(BOOL)enabled state:(GVariant *)state
|
|
{
|
|
canActivate = (target == NULL && parameterType == NULL) ||
|
|
(target != NULL && parameterType != NULL &&
|
|
g_variant_is_of_type (target, parameterType));
|
|
|
|
if (canActivate)
|
|
{
|
|
if (target != NULL && state != NULL)
|
|
{
|
|
[self setOnStateImage:[NSImage imageNamed:@"NSMenuRadio"]];
|
|
[self setState:g_variant_equal (state, target) ? NSOnState : NSOffState];
|
|
}
|
|
else if (state != NULL && g_variant_is_of_type (state, G_VARIANT_TYPE_BOOLEAN))
|
|
{
|
|
[self setOnStateImage:[NSImage imageNamed:@"NSMenuCheckmark"]];
|
|
[self setState:g_variant_get_boolean (state) ? NSOnState : NSOffState];
|
|
}
|
|
else
|
|
[self setState:NSOffState];
|
|
|
|
[self setEnabled:enabled];
|
|
}
|
|
else
|
|
[self setEnabled:NO];
|
|
}
|
|
|
|
- (void)observableActionEnabledChangedTo:(BOOL)enabled
|
|
{
|
|
if (canActivate)
|
|
[self setEnabled:enabled];
|
|
}
|
|
|
|
- (void)observableActionStateChangedTo:(GVariant *)state
|
|
{
|
|
if (canActivate)
|
|
{
|
|
if (target != NULL)
|
|
[self setState:g_variant_equal (state, target) ? NSOnState : NSOffState];
|
|
else if (g_variant_is_of_type (state, G_VARIANT_TYPE_BOOLEAN))
|
|
[self setState:g_variant_get_boolean (state) ? NSOnState : NSOffState];
|
|
}
|
|
}
|
|
|
|
- (void)observableActionRemoved
|
|
{
|
|
if (canActivate)
|
|
[self setEnabled:NO];
|
|
}
|
|
|
|
- (void)didSelectItem:(id)sender
|
|
{
|
|
if (canActivate)
|
|
g_action_group_activate_action (actions, action, target);
|
|
}
|
|
|
|
@end
|