/*
* Copyright 2010 Canonical, Ltd.
*
* This program is free software: you can redistribute it and/or modify it
* under the terms of either or both of the following licenses:
*
* 1) the GNU Lesser General Public License version 3, as published by the
* Free Software Foundation; and/or
* 2) the GNU Lesser General Public License version 2.1, as published by
* the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranties of
* MERCHANTABILITY, SATISFACTORY QUALITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the applicable version of the GNU Lesser General Public
* License for more details.
*
* You should have received a copy of both the GNU Lesser General Public
* License version 3 and version 2.1 along with this program. If not, see
*
*
* Authors:
* Cody Russell
*/
#include
#include "idoentrymenuitem.h"
#include "config.h"
static void ido_entry_menu_item_finalize (GObject *item);
static void ido_entry_menu_item_select (GtkMenuItem *item);
static void ido_entry_menu_item_deselect (GtkMenuItem *item);
static gboolean ido_entry_menu_item_button_release (GtkWidget *widget,
GdkEventButton *event);
static gboolean ido_entry_menu_item_key_press (GtkWidget *widget,
GdkEventKey *event,
gpointer data);
static gboolean ido_entry_menu_item_button_press (GtkWidget *widget,
GdkEventButton *event);
static void ido_entry_menu_item_send_focus_change (GtkWidget *widget,
gboolean in);
static void entry_realized_cb (GtkWidget *widget,
IdoEntryMenuItem *item);
static void entry_move_focus_cb (GtkWidget *widget,
GtkDirectionType direction,
IdoEntryMenuItem *item);
typedef struct {
GtkWidget *box;
GtkWidget *entry;
gboolean selected;
} IdoEntryMenuItemPrivate;
G_DEFINE_TYPE_WITH_PRIVATE (IdoEntryMenuItem, ido_entry_menu_item, GTK_TYPE_MENU_ITEM)
static void
ido_entry_menu_item_class_init (IdoEntryMenuItemClass *klass)
{
GObjectClass *gobject_class;
GtkWidgetClass *widget_class;
GtkMenuItemClass *menu_item_class;
gobject_class = G_OBJECT_CLASS (klass);
widget_class = GTK_WIDGET_CLASS (klass);
menu_item_class = GTK_MENU_ITEM_CLASS (klass);
gobject_class->finalize = ido_entry_menu_item_finalize;
widget_class->button_release_event = ido_entry_menu_item_button_release;
widget_class->button_press_event = ido_entry_menu_item_button_press;
menu_item_class->select = ido_entry_menu_item_select;
menu_item_class->deselect = ido_entry_menu_item_deselect;
menu_item_class->hide_on_activate = TRUE;
}
static void
ido_entry_menu_item_init (IdoEntryMenuItem *item)
{
IdoEntryMenuItemPrivate *priv;
GtkBorder border;
border.left = 4;
border.right = 4;
border.top = 2;
border.bottom = 2;
priv = ido_entry_menu_item_get_instance_private(item);
priv->entry = g_object_new (gtk_entry_get_type (),
"inner-border", &border,
NULL);
g_signal_connect (priv->entry,
"realize",
G_CALLBACK (entry_realized_cb),
item);
g_signal_connect (priv->entry,
"move-focus",
G_CALLBACK (entry_move_focus_cb),
item);
priv->box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
gtk_box_pack_start (GTK_BOX (priv->box), priv->entry, FALSE, FALSE, 0);
gtk_container_add (GTK_CONTAINER (item), priv->box);
gtk_widget_show_all (priv->box);
}
static gboolean
is_key_press_valid (IdoEntryMenuItem *item,
gint key)
{
switch (key)
{
case GDK_KEY_Escape:
case GDK_KEY_Up:
case GDK_KEY_Down:
case GDK_KEY_KP_Up:
case GDK_KEY_KP_Down:
return FALSE;
default:
return TRUE;
}
}
static gboolean
ido_entry_menu_item_key_press (GtkWidget *widget,
GdkEventKey *event,
gpointer data)
{
IdoEntryMenuItem *menuitem = IDO_ENTRY_MENU_ITEM(data);
IdoEntryMenuItemPrivate *priv = ido_entry_menu_item_get_instance_private(menuitem);
if (priv->selected &&
is_key_press_valid (menuitem, event->keyval))
{
GtkWidget *entry = priv->entry;
gtk_widget_event (entry,
((GdkEvent *)(void*)(event)));
/* We've handled the event, but if the key was GDK_KEY_Return
* we still want to forward the event up to the menu shell
* to ensure that the menuitem receives the activate signal.
*/
return event->keyval != GDK_KEY_Return;
}
return FALSE;
}
static void
ido_entry_menu_item_send_focus_change (GtkWidget *widget,
gboolean in)
{
GdkEvent *event = gdk_event_new (GDK_FOCUS_CHANGE);
g_object_ref (widget);
event->focus_change.type = GDK_FOCUS_CHANGE;
event->focus_change.window = g_object_ref (gtk_widget_get_window (widget));
event->focus_change.in = in;
gtk_widget_event (widget, event);
g_object_notify (G_OBJECT (widget), "has-focus");
g_object_unref (widget);
gdk_event_free (event);
}
static gboolean
ido_entry_menu_item_button_press (GtkWidget *widget,
GdkEventButton *event)
{
IdoEntryMenuItem *menuitem = IDO_ENTRY_MENU_ITEM(widget);
IdoEntryMenuItemPrivate *priv = ido_entry_menu_item_get_instance_private(menuitem);
GtkWidget *entry = priv->entry;
if (event->button == 1)
{
if (gtk_widget_get_window (entry) != NULL)
{
gdk_window_raise (gtk_widget_get_window (entry));
}
if (!gtk_widget_has_focus (entry))
{
gtk_widget_grab_focus (entry);
}
gtk_widget_event (entry,
((GdkEvent *)(void*)(event)));
return TRUE;
}
return FALSE;
}
static gboolean
ido_entry_menu_item_button_release (GtkWidget *widget,
GdkEventButton *event)
{
IdoEntryMenuItem *menuitem = IDO_ENTRY_MENU_ITEM(widget);
IdoEntryMenuItemPrivate *priv = ido_entry_menu_item_get_instance_private(menuitem);
GtkWidget *entry = priv->entry;
gtk_widget_event (entry,
((GdkEvent *)(void*)(event)));
return TRUE;
}
static void
ido_entry_menu_item_select (GtkMenuItem *item)
{
IdoEntryMenuItem *menuitem = IDO_ENTRY_MENU_ITEM(item);
IdoEntryMenuItemPrivate *priv = ido_entry_menu_item_get_instance_private(menuitem);
priv->selected = TRUE;
ido_entry_menu_item_send_focus_change (GTK_WIDGET (priv->entry), TRUE);
}
static void
ido_entry_menu_item_deselect (GtkMenuItem *item)
{
IdoEntryMenuItem *menuitem = IDO_ENTRY_MENU_ITEM(item);
IdoEntryMenuItemPrivate *priv = ido_entry_menu_item_get_instance_private(menuitem);
priv->selected = FALSE;
ido_entry_menu_item_send_focus_change (GTK_WIDGET (priv->entry), FALSE);
}
static void
entry_realized_cb (GtkWidget *widget,
IdoEntryMenuItem *item)
{
if (gtk_widget_get_window (widget) != NULL)
{
gdk_window_raise (gtk_widget_get_window (widget));
}
g_signal_connect (gtk_widget_get_parent (GTK_WIDGET (item)),
"key-press-event",
G_CALLBACK (ido_entry_menu_item_key_press),
item);
ido_entry_menu_item_send_focus_change (widget, TRUE);
}
static void
entry_move_focus_cb (GtkWidget *widget,
GtkDirectionType direction,
IdoEntryMenuItem *item)
{
IdoEntryMenuItemPrivate *priv = ido_entry_menu_item_get_instance_private(item);
ido_entry_menu_item_send_focus_change (GTK_WIDGET (priv->entry), FALSE);
g_signal_emit_by_name (item,
"move-focus",
GTK_DIR_TAB_FORWARD);
}
/**
* ido_entry_menu_item_new:
*
* Creates a new #IdoEntryMenuItem.
*
* Return Value: the newly created #IdoEntryMenuItem.
*/
GtkWidget *
ido_entry_menu_item_new (void)
{
return g_object_new (IDO_TYPE_ENTRY_MENU_ITEM, NULL);
}
/**
* ido_entry_menu_item_get_entry:
* @menuitem: The #IdoEntryMenuItem.
*
* Get the #GtkEntry used in this menu item.
*
* Return Value: (transfer none): The #GtkEntry inside this menu item.
*/
GtkWidget *
ido_entry_menu_item_get_entry (IdoEntryMenuItem *menuitem)
{
g_return_val_if_fail (IDO_IS_ENTRY_MENU_ITEM (menuitem), NULL);
IdoEntryMenuItemPrivate *priv = ido_entry_menu_item_get_instance_private(menuitem);
return priv->entry;
}
static void
ido_entry_menu_item_finalize (GObject *gobject)
{
/* no-op */
}