- use G_DECLARE_ and G_DEFINE_ macros - move all data into ClassNamePrivate - use _get_instance_private() This should not introduce any functional changes or breakage. Skipped two classes (EekKeyboard and EekboardContextService) for now in order not to break the build. These two classes are used in some very funky WIP code that tries to circumvent encapsulation. (Funky code is in eekboard/key-emitter.c and eekboard/eekboard-context-service.c)
		
			
				
	
	
		
			723 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			723 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (C) 2010-2011 Daiki Ueno <ueno@unixuser.org>
 | 
						|
 * Copyright (C) 2010-2011 Red Hat, Inc.
 | 
						|
 *
 | 
						|
 * 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 License, 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., 51 Franklin Street, Fifth Floor, Boston, MA
 | 
						|
 * 02110-1301 USA
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * SECTION:eek-gtk-keyboard
 | 
						|
 * @short_description: a #GtkWidget displaying #EekKeyboard
 | 
						|
 */
 | 
						|
 | 
						|
#ifdef HAVE_CONFIG_H
 | 
						|
#include "config.h"
 | 
						|
#endif  /* HAVE_CONFIG_H */
 | 
						|
 | 
						|
#ifdef HAVE_LIBCANBERRA
 | 
						|
#include <canberra-gtk.h>
 | 
						|
#endif
 | 
						|
 | 
						|
#include <math.h>
 | 
						|
#include <string.h>
 | 
						|
 | 
						|
#include "eek-gtk-keyboard.h"
 | 
						|
#include "eek-gtk-renderer.h"
 | 
						|
#include "eek-keyboard.h"
 | 
						|
#include "eek-section.h"
 | 
						|
#include "eek-key.h"
 | 
						|
#include "eek-symbol.h"
 | 
						|
 | 
						|
enum {
 | 
						|
    PROP_0,
 | 
						|
    PROP_KEYBOARD,
 | 
						|
    PROP_LAST
 | 
						|
};
 | 
						|
 | 
						|
/* since 2.91.5 GDK_DRAWABLE was removed and gdk_cairo_create takes
 | 
						|
   GdkWindow as the argument */
 | 
						|
#ifndef GDK_DRAWABLE
 | 
						|
#define GDK_DRAWABLE(x) (x)
 | 
						|
#endif
 | 
						|
 | 
						|
typedef struct _EekGtkKeyboardPrivate
 | 
						|
{
 | 
						|
    EekRenderer *renderer;
 | 
						|
    EekKeyboard *keyboard;
 | 
						|
    gulong key_locked_handler;
 | 
						|
    gulong key_unlocked_handler;
 | 
						|
    gulong symbol_index_changed_handler;
 | 
						|
    EekTheme *theme;
 | 
						|
 | 
						|
    GdkEventSequence *sequence; // unowned reference
 | 
						|
} EekGtkKeyboardPrivate;
 | 
						|
 | 
						|
G_DEFINE_TYPE_WITH_PRIVATE (EekGtkKeyboard, eek_gtk_keyboard, GTK_TYPE_DRAWING_AREA)
 | 
						|
 | 
						|
static void       on_key_pressed          (EekKey      *key,
 | 
						|
                                           EekGtkKeyboard *self);
 | 
						|
static void       on_key_released         (EekKey      *key,
 | 
						|
                                           EekGtkKeyboard *self);
 | 
						|
static void       on_key_locked          (EekKeyboard *keyboard,
 | 
						|
                                           EekKey      *key,
 | 
						|
                                           gpointer     user_data);
 | 
						|
static void       on_key_unlocked         (EekKeyboard *keyboard,
 | 
						|
                                           EekKey      *key,
 | 
						|
                                           gpointer     user_data);
 | 
						|
static void       on_symbol_index_changed (EekKeyboard *keyboard,
 | 
						|
                                           gint         group,
 | 
						|
                                           gint         level,
 | 
						|
                                           gpointer     user_data);
 | 
						|
static void       render_pressed_key      (GtkWidget   *widget,
 | 
						|
                                           EekKey      *key);
 | 
						|
static void       render_locked_key       (GtkWidget   *widget,
 | 
						|
                                           EekKey      *key);
 | 
						|
static void       render_released_key     (GtkWidget   *widget,
 | 
						|
                                           EekKey      *key);
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_real_realize (GtkWidget      *self)
 | 
						|
{
 | 
						|
    gtk_widget_set_events (self,
 | 
						|
                           GDK_EXPOSURE_MASK |
 | 
						|
                           GDK_KEY_PRESS_MASK |
 | 
						|
                           GDK_KEY_RELEASE_MASK |
 | 
						|
                           GDK_BUTTON_PRESS_MASK |
 | 
						|
                           GDK_BUTTON_RELEASE_MASK |
 | 
						|
                           GDK_BUTTON_MOTION_MASK);
 | 
						|
 | 
						|
    GTK_WIDGET_CLASS (eek_gtk_keyboard_parent_class)->realize (self);
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
eek_gtk_keyboard_real_draw (GtkWidget *self,
 | 
						|
                            cairo_t   *cr)
 | 
						|
{
 | 
						|
    EekGtkKeyboardPrivate *priv =
 | 
						|
	    eek_gtk_keyboard_get_instance_private (EEK_GTK_KEYBOARD (self));
 | 
						|
    GtkAllocation allocation;
 | 
						|
    GList *list, *head;
 | 
						|
 | 
						|
    gtk_widget_get_allocation (self, &allocation);
 | 
						|
 | 
						|
    if (!priv->renderer) {
 | 
						|
        PangoContext *pcontext;
 | 
						|
 | 
						|
        pcontext = gtk_widget_get_pango_context (self);
 | 
						|
        priv->renderer = eek_gtk_renderer_new (priv->keyboard, pcontext, self);
 | 
						|
        if (priv->theme)
 | 
						|
            eek_renderer_set_theme (priv->renderer, priv->theme);
 | 
						|
 | 
						|
        eek_renderer_set_allocation_size (priv->renderer,
 | 
						|
                                          allocation.width,
 | 
						|
                                          allocation.height);
 | 
						|
    }
 | 
						|
 | 
						|
    eek_renderer_render_keyboard (priv->renderer, cr);
 | 
						|
 | 
						|
    /* redraw pressed key */
 | 
						|
    list = eek_keyboard_get_pressed_keys (priv->keyboard);
 | 
						|
    for (head = list; head; head = g_list_next (head)) {
 | 
						|
        render_pressed_key (self, head->data);
 | 
						|
    }
 | 
						|
    g_list_free (list);
 | 
						|
 | 
						|
    /* redraw locked key */
 | 
						|
    list = eek_keyboard_get_locked_keys (priv->keyboard);
 | 
						|
    for (head = list; head; head = g_list_next (head)) {
 | 
						|
        render_locked_key (self, ((EekModifierKey *)head->data)->key);
 | 
						|
    }
 | 
						|
    g_list_free (list);
 | 
						|
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_real_size_allocate (GtkWidget     *self,
 | 
						|
                                     GtkAllocation *allocation)
 | 
						|
{
 | 
						|
    EekGtkKeyboardPrivate *priv =
 | 
						|
	    eek_gtk_keyboard_get_instance_private (EEK_GTK_KEYBOARD (self));
 | 
						|
 | 
						|
    if (priv->renderer)
 | 
						|
        eek_renderer_set_allocation_size (priv->renderer,
 | 
						|
                                          allocation->width,
 | 
						|
                                          allocation->height);
 | 
						|
 | 
						|
    GTK_WIDGET_CLASS (eek_gtk_keyboard_parent_class)->
 | 
						|
        size_allocate (self, allocation);
 | 
						|
}
 | 
						|
 | 
						|
static void depress(EekGtkKeyboard *self,
 | 
						|
                    gdouble x, gdouble y, guint32 time) {
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
    EekKey *key = eek_renderer_find_key_by_position (priv->renderer, x, y);
 | 
						|
    if (key) {
 | 
						|
        eek_keyboard_press_key(priv->keyboard, key, time);
 | 
						|
        on_key_pressed(key, self);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void drag(EekGtkKeyboard *self,
 | 
						|
                 gdouble x, gdouble y, guint32 time) {
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
    EekKey *key = eek_renderer_find_key_by_position (priv->renderer, x, y);
 | 
						|
 | 
						|
    if (key) {
 | 
						|
        GList *list, *head;
 | 
						|
        gboolean found = FALSE;
 | 
						|
 | 
						|
        list = eek_keyboard_get_pressed_keys (priv->keyboard);
 | 
						|
        for (head = list; head; head = g_list_next (head)) {
 | 
						|
            if (head->data == key)
 | 
						|
                found = TRUE;
 | 
						|
            else {
 | 
						|
                eek_keyboard_release_key(priv->keyboard, EEK_KEY(head->data), time);
 | 
						|
                on_key_released(key, self);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        g_list_free (list);
 | 
						|
 | 
						|
        if (!found) {
 | 
						|
            eek_keyboard_press_key(priv->keyboard, key, time);
 | 
						|
            on_key_pressed(key, self);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void release(EekGtkKeyboard *self, guint32 time) {
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
 | 
						|
    GList *list = eek_keyboard_get_pressed_keys (priv->keyboard);
 | 
						|
    for (GList *head = list; head; head = g_list_next (head)) {
 | 
						|
        EekKey *key = EEK_KEY(head->data);
 | 
						|
        eek_keyboard_release_key(priv->keyboard, key, time);
 | 
						|
        on_key_released(key, self);
 | 
						|
    }
 | 
						|
    g_list_free (list);
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
eek_gtk_keyboard_real_button_press_event (GtkWidget      *self,
 | 
						|
                                          GdkEventButton *event)
 | 
						|
{
 | 
						|
    if (event->type == GDK_BUTTON_PRESS && event->button == 1) {
 | 
						|
        depress(EEK_GTK_KEYBOARD(self), event->x, event->y, event->time);
 | 
						|
    }
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
// TODO: this belongs more in gtk_keyboard, with a way to find out which key to re-render
 | 
						|
static gboolean
 | 
						|
eek_gtk_keyboard_real_button_release_event (GtkWidget      *self,
 | 
						|
                                            GdkEventButton *event)
 | 
						|
{
 | 
						|
    if (event->type == GDK_BUTTON_RELEASE && event->button == 1) {
 | 
						|
        // TODO: can the event have different coords than the previous move event?
 | 
						|
        release(EEK_GTK_KEYBOARD(self), event->time);
 | 
						|
    }
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
eek_gtk_keyboard_real_motion_notify_event (GtkWidget      *self,
 | 
						|
                                           GdkEventMotion *event)
 | 
						|
{
 | 
						|
    if (event->state & GDK_BUTTON1_MASK) {
 | 
						|
        drag(EEK_GTK_KEYBOARD(self), event->x, event->y, event->time);
 | 
						|
    }
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
// Only one touch stream at a time allowed. Others will be completely ignored.
 | 
						|
static gboolean
 | 
						|
handle_touch_event (GtkWidget     *widget,
 | 
						|
                    GdkEventTouch *event)
 | 
						|
{
 | 
						|
    EekGtkKeyboard        *self = EEK_GTK_KEYBOARD (widget);
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
 | 
						|
    if (event->type == GDK_TOUCH_BEGIN) {
 | 
						|
        if (priv->sequence) {
 | 
						|
            // Ignore second and following touch points
 | 
						|
            return FALSE;
 | 
						|
        }
 | 
						|
        priv->sequence = event->sequence;
 | 
						|
        depress(self, event->x, event->y, event->time);
 | 
						|
        return TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
    if (priv->sequence != event->sequence) {
 | 
						|
        return FALSE;
 | 
						|
    }
 | 
						|
 | 
						|
    if (event->type == GDK_TOUCH_UPDATE) {
 | 
						|
        drag(self, event->x, event->y, event->time);
 | 
						|
    }
 | 
						|
    if (event->type == GDK_TOUCH_END || event->type == GDK_TOUCH_CANCEL) {
 | 
						|
        // TODO: can the event have different coords than the previous update event?
 | 
						|
        release(self, event->time);
 | 
						|
        priv->sequence = NULL;
 | 
						|
    }
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_real_unmap (GtkWidget *self)
 | 
						|
{
 | 
						|
    EekGtkKeyboardPrivate *priv =
 | 
						|
	    eek_gtk_keyboard_get_instance_private (EEK_GTK_KEYBOARD (self));
 | 
						|
 | 
						|
    if (priv->keyboard) {
 | 
						|
        GList *list, *head;
 | 
						|
 | 
						|
        /* Make a copy of HEAD before sending "released" signal on
 | 
						|
           elements, so that the default handler of
 | 
						|
           EekKeyboard::key-released signal can remove elements from its
 | 
						|
           internal copy */
 | 
						|
        list = eek_keyboard_get_pressed_keys (priv->keyboard);
 | 
						|
        for (head = list; head; head = g_list_next (head)) {
 | 
						|
            g_log("squeek", G_LOG_LEVEL_DEBUG, "emit EekKey released");
 | 
						|
            g_signal_emit_by_name (head->data, "released");
 | 
						|
        }
 | 
						|
        g_list_free (list);
 | 
						|
    }
 | 
						|
 | 
						|
    GTK_WIDGET_CLASS (eek_gtk_keyboard_parent_class)->unmap (self);
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
eek_gtk_keyboard_real_query_tooltip (GtkWidget  *widget,
 | 
						|
                                     gint        x,
 | 
						|
                                     gint        y,
 | 
						|
                                     gboolean    keyboard_tooltip,
 | 
						|
                                     GtkTooltip *tooltip)
 | 
						|
{
 | 
						|
    EekGtkKeyboard        *self = EEK_GTK_KEYBOARD (widget);
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
 | 
						|
    EekKey *key;
 | 
						|
 | 
						|
    key = eek_renderer_find_key_by_position (priv->renderer,
 | 
						|
                                             (gdouble)x,
 | 
						|
                                             (gdouble)y);
 | 
						|
    if (key) {
 | 
						|
        EekSymbol *symbol = eek_key_get_symbol (key);
 | 
						|
        const gchar *text = eek_symbol_get_tooltip (symbol);
 | 
						|
        if (text) {
 | 
						|
            gtk_tooltip_set_text (tooltip, text);
 | 
						|
            return TRUE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_set_keyboard (EekGtkKeyboard *self,
 | 
						|
                               EekKeyboard    *keyboard)
 | 
						|
{
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
    priv->keyboard = g_object_ref (keyboard);
 | 
						|
 | 
						|
    priv->key_locked_handler =
 | 
						|
        g_signal_connect (priv->keyboard, "key-locked",
 | 
						|
                          G_CALLBACK(on_key_locked), self);
 | 
						|
    priv->key_unlocked_handler =
 | 
						|
        g_signal_connect (priv->keyboard, "key-unlocked",
 | 
						|
                          G_CALLBACK(on_key_unlocked), self);
 | 
						|
    priv->symbol_index_changed_handler =
 | 
						|
        g_signal_connect (priv->keyboard, "symbol-index-changed",
 | 
						|
                          G_CALLBACK(on_symbol_index_changed), self);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_set_property (GObject      *object,
 | 
						|
                               guint         prop_id,
 | 
						|
                               const GValue *value,
 | 
						|
                               GParamSpec   *pspec)
 | 
						|
{
 | 
						|
    EekKeyboard *keyboard;
 | 
						|
 | 
						|
    switch (prop_id) {
 | 
						|
    case PROP_KEYBOARD:
 | 
						|
        keyboard = g_value_get_object (value);
 | 
						|
        eek_gtk_keyboard_set_keyboard (EEK_GTK_KEYBOARD(object), keyboard);
 | 
						|
        break;
 | 
						|
    default:
 | 
						|
        G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_dispose (GObject *object)
 | 
						|
{
 | 
						|
    EekGtkKeyboard        *self = EEK_GTK_KEYBOARD (object);
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
 | 
						|
    if (priv->renderer) {
 | 
						|
        g_object_unref (priv->renderer);
 | 
						|
        priv->renderer = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    if (priv->keyboard) {
 | 
						|
        if (g_signal_handler_is_connected (priv->keyboard,
 | 
						|
                                           priv->key_locked_handler))
 | 
						|
            g_signal_handler_disconnect (priv->keyboard,
 | 
						|
                                         priv->key_locked_handler);
 | 
						|
        if (g_signal_handler_is_connected (priv->keyboard,
 | 
						|
                                           priv->key_unlocked_handler))
 | 
						|
            g_signal_handler_disconnect (priv->keyboard,
 | 
						|
                                         priv->key_unlocked_handler);
 | 
						|
        if (g_signal_handler_is_connected (priv->keyboard,
 | 
						|
                                           priv->symbol_index_changed_handler))
 | 
						|
            g_signal_handler_disconnect (priv->keyboard,
 | 
						|
                                         priv->symbol_index_changed_handler);
 | 
						|
 | 
						|
        GList *list, *head;
 | 
						|
 | 
						|
        list = eek_keyboard_get_pressed_keys (priv->keyboard);
 | 
						|
        for (head = list; head; head = g_list_next (head)) {
 | 
						|
            g_log("squeek", G_LOG_LEVEL_DEBUG, "emit EekKey pressed");
 | 
						|
            g_signal_emit_by_name (head->data, "released", priv->keyboard);
 | 
						|
        }
 | 
						|
        g_list_free (list);
 | 
						|
 | 
						|
        g_object_unref (priv->keyboard);
 | 
						|
        priv->keyboard = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    if (priv->theme) {
 | 
						|
        g_object_unref (priv->theme);
 | 
						|
        priv->theme = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    G_OBJECT_CLASS (eek_gtk_keyboard_parent_class)->dispose (object);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_class_init (EekGtkKeyboardClass *klass)
 | 
						|
{
 | 
						|
    GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
 | 
						|
    GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
 | 
						|
    GParamSpec *pspec;
 | 
						|
 | 
						|
    widget_class->realize = eek_gtk_keyboard_real_realize;
 | 
						|
    widget_class->unmap = eek_gtk_keyboard_real_unmap;
 | 
						|
    widget_class->draw = eek_gtk_keyboard_real_draw;
 | 
						|
    widget_class->size_allocate = eek_gtk_keyboard_real_size_allocate;
 | 
						|
    widget_class->button_press_event =
 | 
						|
        eek_gtk_keyboard_real_button_press_event;
 | 
						|
    widget_class->button_release_event =
 | 
						|
        eek_gtk_keyboard_real_button_release_event;
 | 
						|
    widget_class->motion_notify_event =
 | 
						|
        eek_gtk_keyboard_real_motion_notify_event;
 | 
						|
    widget_class->query_tooltip =
 | 
						|
        eek_gtk_keyboard_real_query_tooltip;
 | 
						|
    widget_class->touch_event = handle_touch_event;
 | 
						|
 | 
						|
    gobject_class->set_property = eek_gtk_keyboard_set_property;
 | 
						|
    gobject_class->dispose = eek_gtk_keyboard_dispose;
 | 
						|
 | 
						|
    pspec = g_param_spec_object ("keyboard",
 | 
						|
                                 "Keyboard",
 | 
						|
                                 "Keyboard",
 | 
						|
                                 EEK_TYPE_KEYBOARD,
 | 
						|
                                 G_PARAM_CONSTRUCT_ONLY | G_PARAM_WRITABLE);
 | 
						|
    g_object_class_install_property (gobject_class,
 | 
						|
                                     PROP_KEYBOARD,
 | 
						|
                                     pspec);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eek_gtk_keyboard_init (EekGtkKeyboard *self)
 | 
						|
{
 | 
						|
    /* void */
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * eek_gtk_keyboard_new:
 | 
						|
 * @keyboard: an #EekKeyboard
 | 
						|
 *
 | 
						|
 * Create a new #GtkWidget displaying @keyboard.
 | 
						|
 * Returns: a #GtkWidget
 | 
						|
 */
 | 
						|
GtkWidget *
 | 
						|
eek_gtk_keyboard_new (EekKeyboard *keyboard)
 | 
						|
{
 | 
						|
    return g_object_new (EEK_TYPE_GTK_KEYBOARD, "keyboard", keyboard, NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
magnify_bounds (GtkWidget *self,
 | 
						|
                EekBounds *bounds,
 | 
						|
                EekBounds *large_bounds,
 | 
						|
                gdouble    scale)
 | 
						|
{
 | 
						|
    GtkAllocation allocation;
 | 
						|
    gdouble x, y;
 | 
						|
 | 
						|
    g_assert (scale >= 1.0);
 | 
						|
 | 
						|
    gtk_widget_get_allocation (self, &allocation);
 | 
						|
 | 
						|
    large_bounds->width = bounds->width * scale;
 | 
						|
    large_bounds->height = bounds->height * scale;
 | 
						|
 | 
						|
    x = bounds->x - (large_bounds->width - bounds->width) / 2;
 | 
						|
    y = bounds->y - large_bounds->height;
 | 
						|
 | 
						|
    large_bounds->x = CLAMP(x, 0, allocation.width - large_bounds->width);
 | 
						|
    large_bounds->y = CLAMP(y, 0, allocation.height - large_bounds->height);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Alleviate the asymmetry between drawing a pressed key and a released key,
 | 
						|
 * and consistently draw to the exact same area.
 | 
						|
 *
 | 
						|
 * By saving the dirty rectangle we can limit drawing of the backbuffer to
 | 
						|
 * the screen as well, eg gdk_window_invalidate_rect() instead of
 | 
						|
 * gtk_widget_queue_draw() which redraws the entire widget.
 | 
						|
 *
 | 
						|
 * b1 is mandatory, b2 is optional
 | 
						|
 */
 | 
						|
static GdkRectangle
 | 
						|
clip_bounds_to_dirty_rectangle (cairo_t *cr, EekBounds *b1, EekBounds *b2)
 | 
						|
{
 | 
						|
    if (b2)
 | 
						|
        cairo_rectangle (cr, b2->x, b2->y, b2->width, b2->height);
 | 
						|
 | 
						|
    cairo_rectangle (cr, b1->x, b1->y, b1->width, b1->height);
 | 
						|
    cairo_clip (cr);
 | 
						|
 | 
						|
    /*
 | 
						|
     * save the clipped region to a bounding box so we can limit
 | 
						|
     * the drawing of the backbuffer to the screen to the same area
 | 
						|
     */
 | 
						|
    cairo_rectangle_t bbox;
 | 
						|
 | 
						|
    cairo_clip_extents (cr, &bbox.x, &bbox.y, &bbox.width, &bbox.height);
 | 
						|
 | 
						|
    /* convert double to int, making sure r strictly covers bbox to avoid
 | 
						|
     * artefacts. floor() is unnecessary, ceil() is not */
 | 
						|
    GdkRectangle r = {
 | 
						|
        floor (bbox.x),
 | 
						|
        floor (bbox.y),
 | 
						|
        ceil  (bbox.width),
 | 
						|
        ceil  (bbox.height)
 | 
						|
    };
 | 
						|
 | 
						|
    return r;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
render_pressed_key (GtkWidget *widget,
 | 
						|
                    EekKey    *key)
 | 
						|
{
 | 
						|
    EekGtkKeyboard        *self = EEK_GTK_KEYBOARD (widget);
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
    EekBounds bounds, large_bounds;
 | 
						|
 | 
						|
    GdkWindow         *window  = gtk_widget_get_window (widget);
 | 
						|
    cairo_region_t    *region  = gdk_window_get_clip_region (window);
 | 
						|
    GdkDrawingContext *context = gdk_window_begin_draw_frame (window, region);
 | 
						|
    cairo_t           *cr      = gdk_drawing_context_get_cairo_context (context);
 | 
						|
 | 
						|
    eek_renderer_get_key_bounds (priv->renderer, key, &bounds, TRUE);
 | 
						|
    magnify_bounds (widget, &bounds, &large_bounds, 1.5);
 | 
						|
 | 
						|
    /*
 | 
						|
     * clip to limit drawing to backbuffer and save clip region to dirty_rect
 | 
						|
     * to limit redrawing of the backbuffer to the same area
 | 
						|
     */
 | 
						|
    GdkRectangle dirty_rect = clip_bounds_to_dirty_rectangle (cr, &bounds, &large_bounds);
 | 
						|
 | 
						|
    cairo_save (cr);
 | 
						|
    cairo_translate (cr, bounds.x, bounds.y);
 | 
						|
    eek_renderer_render_key (priv->renderer, cr, key, 1.0, TRUE);
 | 
						|
    cairo_restore (cr);
 | 
						|
 | 
						|
    cairo_save (cr);
 | 
						|
    cairo_translate (cr, large_bounds.x, large_bounds.y);
 | 
						|
    eek_renderer_render_key (priv->renderer, cr, key, 1.5, TRUE);
 | 
						|
    cairo_restore (cr);
 | 
						|
 | 
						|
    gdk_window_end_draw_frame (window, context);
 | 
						|
 | 
						|
    cairo_region_destroy (region);
 | 
						|
 | 
						|
    /* force immediate drawing of the backbuffer to the screen */
 | 
						|
    gdk_window_invalidate_rect (window, &dirty_rect, FALSE);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
render_locked_key (GtkWidget *widget,
 | 
						|
                   EekKey    *key)
 | 
						|
{
 | 
						|
    EekGtkKeyboard        *self = EEK_GTK_KEYBOARD (widget);
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
    EekBounds bounds;
 | 
						|
 | 
						|
    GdkWindow         *window  = gtk_widget_get_window (widget);
 | 
						|
    cairo_region_t    *region  = gdk_window_get_clip_region (window);
 | 
						|
    GdkDrawingContext *context = gdk_window_begin_draw_frame (window, region);
 | 
						|
    cairo_t           *cr      = gdk_drawing_context_get_cairo_context (context);
 | 
						|
 | 
						|
    eek_renderer_get_key_bounds (priv->renderer, key, &bounds, TRUE);
 | 
						|
 | 
						|
    /*
 | 
						|
     * clip to limit drawing to backbuffer and save clip region to dirty_rect
 | 
						|
     * to limit redrawing of the backbuffer to the same area
 | 
						|
     */
 | 
						|
    GdkRectangle dirty_rect = clip_bounds_to_dirty_rectangle (cr, &bounds, NULL);
 | 
						|
 | 
						|
    cairo_translate (cr, bounds.x, bounds.y);
 | 
						|
    eek_renderer_render_key (priv->renderer, cr, key, 1.0, TRUE);
 | 
						|
 | 
						|
    gdk_window_end_draw_frame (window, context);
 | 
						|
 | 
						|
    cairo_region_destroy (region);
 | 
						|
 | 
						|
    /* force immediate drawing of the backbuffer to the screen */
 | 
						|
    gdk_window_invalidate_rect (window, &dirty_rect, FALSE);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
render_released_key (GtkWidget *widget,
 | 
						|
                     EekKey    *key)
 | 
						|
{
 | 
						|
    EekGtkKeyboard        *self = EEK_GTK_KEYBOARD (widget);
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
    EekBounds bounds, large_bounds;
 | 
						|
 | 
						|
    GdkWindow         *window  = gtk_widget_get_window (widget);
 | 
						|
    cairo_region_t    *region  = gdk_window_get_clip_region (window);
 | 
						|
    GdkDrawingContext *context = gdk_window_begin_draw_frame (window, region);
 | 
						|
    cairo_t           *cr      = gdk_drawing_context_get_cairo_context (context);
 | 
						|
 | 
						|
    eek_renderer_get_key_bounds (priv->renderer, key, &bounds, TRUE);
 | 
						|
    magnify_bounds (widget, &bounds, &large_bounds, 1.5);
 | 
						|
 | 
						|
    /*
 | 
						|
     * clip to limit drawing to backbuffer and save clip region to dirty_rect
 | 
						|
     * to limit redrawing of the backbuffer to the same area
 | 
						|
     */
 | 
						|
    GdkRectangle dirty_rect = clip_bounds_to_dirty_rectangle(cr, &bounds, &large_bounds);
 | 
						|
 | 
						|
    eek_renderer_render_keyboard (priv->renderer, cr);
 | 
						|
 | 
						|
    gdk_window_end_draw_frame (window, context);
 | 
						|
 | 
						|
    cairo_region_destroy (region);
 | 
						|
 | 
						|
    /* force immediate drawing of the backbuffer to the screen */
 | 
						|
    gdk_window_invalidate_rect (window, &dirty_rect, FALSE);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_key_pressed (EekKey      *key,
 | 
						|
                EekGtkKeyboard *self)
 | 
						|
{
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
 | 
						|
    /* renderer may have not been set yet if the widget is a popup */
 | 
						|
    if (!priv->renderer)
 | 
						|
        return;
 | 
						|
 | 
						|
    render_pressed_key (GTK_WIDGET(self), key);
 | 
						|
 | 
						|
#if HAVE_LIBCANBERRA
 | 
						|
    ca_gtk_play_for_widget (widget, 0,
 | 
						|
                            CA_PROP_EVENT_ID, "button-pressed",
 | 
						|
                            CA_PROP_EVENT_DESCRIPTION, "virtual key pressed",
 | 
						|
                            CA_PROP_APPLICATION_ID, "org.fedorahosted.Eekboard",
 | 
						|
                            NULL);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_key_released (EekKey      *key,
 | 
						|
                 EekGtkKeyboard *self)
 | 
						|
{
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
 | 
						|
 | 
						|
    /* renderer may have not been set yet if the widget is a popup */
 | 
						|
    if (!priv->renderer)
 | 
						|
        return;
 | 
						|
 | 
						|
    render_released_key (GTK_WIDGET(self), key);
 | 
						|
 | 
						|
#if HAVE_LIBCANBERRA
 | 
						|
    ca_gtk_play_for_widget (widget, 0,
 | 
						|
                            CA_PROP_EVENT_ID, "button-released",
 | 
						|
                            CA_PROP_EVENT_DESCRIPTION, "virtual key pressed",
 | 
						|
                            CA_PROP_APPLICATION_ID, "org.fedorahosted.Eekboard",
 | 
						|
                            NULL);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_key_locked (EekKeyboard *keyboard,
 | 
						|
               EekKey      *key,
 | 
						|
               gpointer     user_data)
 | 
						|
{
 | 
						|
    GtkWidget *widget = user_data;
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (user_data);
 | 
						|
 | 
						|
    /* renderer may have not been set yet if the widget is a popup */
 | 
						|
    if (!priv->renderer)
 | 
						|
        return;
 | 
						|
 | 
						|
    render_locked_key (widget, key);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_key_unlocked (EekKeyboard *keyboard,
 | 
						|
                 EekKey      *key,
 | 
						|
                 gpointer     user_data)
 | 
						|
{
 | 
						|
    GtkWidget *widget = user_data;
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (user_data);
 | 
						|
 | 
						|
    /* renderer may have not been set yet if the widget is a popup */
 | 
						|
    if (!priv->renderer)
 | 
						|
        return;
 | 
						|
 | 
						|
    render_released_key (widget, key);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_symbol_index_changed (EekKeyboard *keyboard,
 | 
						|
                         gint         group,
 | 
						|
                         gint         level,
 | 
						|
                         gpointer     user_data)
 | 
						|
{
 | 
						|
    GtkWidget *widget = user_data;
 | 
						|
 | 
						|
    gtk_widget_queue_draw (widget);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
eek_gtk_keyboard_set_theme (EekGtkKeyboard *keyboard,
 | 
						|
                            EekTheme       *theme)
 | 
						|
{
 | 
						|
    g_return_if_fail (EEK_IS_GTK_KEYBOARD(keyboard));
 | 
						|
    g_return_if_fail (EEK_IS_THEME(theme));
 | 
						|
 | 
						|
    EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (keyboard);
 | 
						|
    priv->theme = g_object_ref (theme);
 | 
						|
}
 |