The check against fitting inside the Layout was removed: as an optimization it is unneeded, as the actual search must be optimized to be quick. In addition, the view bounds don't correspond to anything physical as long as negative offsets are allowed.
538 lines
18 KiB
C
538 lines
18 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
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#ifdef HAVE_LIBCANBERRA
|
|
#include <canberra-gtk.h>
|
|
#endif
|
|
|
|
#include <math.h>
|
|
#include <string.h>
|
|
|
|
#include "eek-renderer.h"
|
|
#include "eek-keyboard.h"
|
|
#include "src/symbol.h"
|
|
|
|
#include "eek-gtk-keyboard.h"
|
|
|
|
enum {
|
|
PROP_0,
|
|
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;
|
|
LevelKeyboard *keyboard;
|
|
GtkCssProvider *css_provider;
|
|
GtkStyleContext *scontext;
|
|
|
|
GdkEventSequence *sequence; // unowned reference
|
|
} EekGtkKeyboardPrivate;
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (EekGtkKeyboard, eek_gtk_keyboard, GTK_TYPE_DRAWING_AREA)
|
|
|
|
static void on_button_pressed (struct squeek_button *button, struct squeek_view *view,
|
|
EekGtkKeyboard *self);
|
|
static void on_button_released (struct squeek_button *button,
|
|
struct squeek_view *view,
|
|
EekGtkKeyboard *self);
|
|
static void render_pressed_button (GtkWidget *widget, struct button_place *place);
|
|
static void render_locked_button (GtkWidget *widget,
|
|
struct button_place *place);
|
|
static void render_released_button (GtkWidget *widget,
|
|
struct squeek_button *button);
|
|
|
|
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;
|
|
gtk_widget_get_allocation (self, &allocation);
|
|
|
|
if (!priv->renderer) {
|
|
PangoContext *pcontext = gtk_widget_get_pango_context (self);
|
|
|
|
priv->renderer = eek_renderer_new (priv->keyboard, pcontext, priv->scontext);
|
|
|
|
eek_renderer_set_allocation_size (priv->renderer,
|
|
allocation.width,
|
|
allocation.height);
|
|
eek_renderer_set_scale_factor (priv->renderer,
|
|
gtk_widget_get_scale_factor (self));
|
|
}
|
|
|
|
eek_renderer_render_keyboard (priv->renderer, cr);
|
|
|
|
struct squeek_view *view = priv->keyboard->views[priv->keyboard->level];
|
|
|
|
/* redraw pressed key */
|
|
const GList *list = priv->keyboard->pressed_buttons;
|
|
for (const GList *head = list; head; head = g_list_next (head)) {
|
|
struct button_place place = squeek_view_find_key(
|
|
view, squeek_button_get_key(head->data)
|
|
);
|
|
if (place.button)
|
|
render_pressed_button (self, &place);
|
|
}
|
|
|
|
/* redraw locked key */
|
|
list = priv->keyboard->locked_buttons;
|
|
for (const GList *head = list; head; head = g_list_next (head)) {
|
|
struct button_place place = squeek_view_find_key(
|
|
view, squeek_button_get_key(
|
|
((EekModifierKey *)head->data)->button
|
|
)
|
|
);
|
|
if (place.button)
|
|
render_locked_button (self, &place);
|
|
}
|
|
|
|
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);
|
|
struct squeek_view *view = level_keyboard_current(priv->keyboard);
|
|
struct squeek_button *button = eek_renderer_find_button_by_position (priv->renderer, view, x, y);
|
|
|
|
if (button) {
|
|
eek_keyboard_press_button(priv->keyboard, button, time);
|
|
on_button_pressed(button, view, self);
|
|
}
|
|
}
|
|
|
|
static void drag(EekGtkKeyboard *self,
|
|
gdouble x, gdouble y, guint32 time) {
|
|
EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
|
|
struct squeek_view *view = level_keyboard_current(priv->keyboard);
|
|
struct squeek_button *button = eek_renderer_find_button_by_position (priv->renderer, view, x, y);
|
|
GList *list, *head;
|
|
|
|
list = g_list_copy(priv->keyboard->pressed_buttons);
|
|
|
|
if (button) {
|
|
gboolean found = FALSE;
|
|
|
|
for (head = list; head; head = g_list_next (head)) {
|
|
if (head->data == button) {
|
|
found = TRUE;
|
|
} else {
|
|
eek_keyboard_release_button(priv->keyboard, head->data, time);
|
|
on_button_released(button, view, self);
|
|
}
|
|
}
|
|
g_list_free (list);
|
|
|
|
if (!found) {
|
|
eek_keyboard_press_button(priv->keyboard, button, time);
|
|
on_button_pressed(button, view, self);
|
|
}
|
|
} else {
|
|
for (head = list; head; head = g_list_next (head)) {
|
|
eek_keyboard_release_button(priv->keyboard, head->data, time);
|
|
on_button_released(head->data, view, self);
|
|
}
|
|
g_list_free (list);
|
|
}
|
|
}
|
|
|
|
static void release(EekGtkKeyboard *self, guint32 time) {
|
|
EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
|
|
|
|
struct squeek_view *view = level_keyboard_current(priv->keyboard);
|
|
|
|
GList *list = g_list_copy(priv->keyboard->pressed_buttons);
|
|
for (GList *head = list; head; head = g_list_next (head)) {
|
|
struct squeek_button *button = head->data;
|
|
eek_keyboard_release_button(priv->keyboard, button, time);
|
|
on_button_released(button, view, 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 *head;
|
|
|
|
for (head = priv->keyboard->pressed_buttons; head; head = g_list_next (head)) {
|
|
/* Unlike other places where we call this, we don't call
|
|
on_button_released afterwards since we don't want to queue a
|
|
redraw. */
|
|
eek_keyboard_release_button(priv->keyboard, head->data,
|
|
gdk_event_get_time(NULL));
|
|
}
|
|
}
|
|
|
|
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);
|
|
struct squeek_view *view = level_keyboard_current(priv->keyboard);
|
|
|
|
struct squeek_button *button = eek_renderer_find_button_by_position (priv->renderer,
|
|
view,
|
|
(gdouble)x,
|
|
(gdouble)y);
|
|
if (button) {
|
|
//struct squeek_symbol *symbol = eek_key_get_symbol_at_index(key, 0, priv->keyboard->level);
|
|
const gchar *text = NULL; // FIXME
|
|
if (text) {
|
|
gtk_tooltip_set_text (tooltip, text);
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
eek_gtk_keyboard_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (prop_id) {
|
|
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) {
|
|
GList *head;
|
|
|
|
for (head = priv->keyboard->pressed_buttons; head; head = g_list_next (head)) {
|
|
eek_keyboard_release_button(priv->keyboard, head->data,
|
|
gdk_event_get_time(NULL));
|
|
}
|
|
|
|
priv->keyboard = 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);
|
|
|
|
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;
|
|
}
|
|
|
|
static void
|
|
eek_gtk_keyboard_init (EekGtkKeyboard *self)
|
|
{
|
|
EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
|
|
|
|
/* Create a default CSS provider and load a style sheet */
|
|
priv->css_provider = gtk_css_provider_new ();
|
|
gtk_css_provider_load_from_resource (priv->css_provider,
|
|
"/sm/puri/squeekboard/style.css");
|
|
|
|
/* Apply the style to the widget */
|
|
priv->scontext = gtk_widget_get_style_context (GTK_WIDGET(self));
|
|
gtk_style_context_add_class (priv->scontext, "keyboard");
|
|
gtk_style_context_add_provider (priv->scontext,
|
|
GTK_STYLE_PROVIDER(priv->css_provider),
|
|
GTK_STYLE_PROVIDER_PRIORITY_USER);
|
|
gtk_style_context_set_state (priv->scontext, GTK_STATE_FLAG_NORMAL);
|
|
}
|
|
|
|
/**
|
|
* eek_gtk_keyboard_new:
|
|
* @keyboard: an #EekKeyboard
|
|
*
|
|
* Create a new #GtkWidget displaying @keyboard.
|
|
* Returns: a #GtkWidget
|
|
*/
|
|
GtkWidget *
|
|
eek_gtk_keyboard_new (LevelKeyboard *keyboard)
|
|
{
|
|
EekGtkKeyboard *ret = EEK_GTK_KEYBOARD(g_object_new (EEK_TYPE_GTK_KEYBOARD, NULL));
|
|
EekGtkKeyboardPrivate *priv = (EekGtkKeyboardPrivate*)eek_gtk_keyboard_get_instance_private (ret);
|
|
priv->keyboard = keyboard;
|
|
return GTK_WIDGET(ret);
|
|
}
|
|
|
|
static void
|
|
render_pressed_button (GtkWidget *widget,
|
|
struct button_place *place)
|
|
{
|
|
EekGtkKeyboard *self = EEK_GTK_KEYBOARD (widget);
|
|
EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
|
|
|
|
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_render_button (priv->renderer, cr, place, 1.0, TRUE);
|
|
/*
|
|
eek_renderer_render_key (priv->renderer, cr, key, 1.5, TRUE);
|
|
*/
|
|
gdk_window_end_draw_frame (window, context);
|
|
|
|
cairo_region_destroy (region);
|
|
}
|
|
|
|
static void
|
|
render_locked_button (GtkWidget *widget, struct button_place *place)
|
|
{
|
|
EekGtkKeyboard *self = EEK_GTK_KEYBOARD (widget);
|
|
EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
|
|
|
|
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_render_button (priv->renderer, cr, place, 1.0, TRUE);
|
|
|
|
gdk_window_end_draw_frame (window, context);
|
|
|
|
cairo_region_destroy (region);
|
|
}
|
|
|
|
static void
|
|
render_released_button (GtkWidget *widget,
|
|
struct squeek_button *button)
|
|
{
|
|
(void)button;
|
|
EekGtkKeyboard *self = EEK_GTK_KEYBOARD (widget);
|
|
EekGtkKeyboardPrivate *priv = eek_gtk_keyboard_get_instance_private (self);
|
|
|
|
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_render_keyboard (priv->renderer, cr);
|
|
|
|
gdk_window_end_draw_frame (window, context);
|
|
|
|
cairo_region_destroy (region);
|
|
}
|
|
|
|
static void
|
|
on_button_pressed (struct squeek_button *button,
|
|
struct squeek_view *view,
|
|
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;
|
|
|
|
struct button_place place = {
|
|
.button = button,
|
|
.row = squeek_view_get_row(view, button),
|
|
};
|
|
if (!place.row) {
|
|
return;
|
|
}
|
|
render_pressed_button (GTK_WIDGET(self), &place);
|
|
gtk_widget_queue_draw (GTK_WIDGET(self));
|
|
|
|
#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_button_released (struct squeek_button *button,
|
|
struct squeek_view *view,
|
|
EekGtkKeyboard *self)
|
|
{
|
|
(void)view;
|
|
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_button (GTK_WIDGET(self), button);
|
|
gtk_widget_queue_draw (GTK_WIDGET(self));
|
|
|
|
#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
|
|
}
|