Make the calibrator window stay above other windows and cancel calibration if/when the calibrator loses input focus. https://bugzilla.gnome.org/show_bug.cgi?id=675354
472 lines
14 KiB
C
472 lines
14 KiB
C
/*
|
|
* Copyright (c) 2009 Tias Guns
|
|
* Copyright (c) 2009 Soren Hauberg
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <math.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <glib/gi18n.h>
|
|
#include <gtk/gtk.h>
|
|
#include <cairo.h>
|
|
|
|
#include "calibrator.h"
|
|
#include "gui_gtk.h"
|
|
|
|
struct CalibArea
|
|
{
|
|
struct Calib calibrator;
|
|
double X[4], Y[4];
|
|
int display_width, display_height;
|
|
int time_elapsed;
|
|
|
|
const char* message;
|
|
|
|
guint anim_id;
|
|
|
|
GtkWidget *window;
|
|
|
|
FinishCallback callback;
|
|
gpointer user_data;
|
|
};
|
|
|
|
|
|
/* Window parameters */
|
|
#define WINDOW_OPACITY 0.9
|
|
|
|
/* Timeout parameters */
|
|
#define TIME_STEP 100 /* in milliseconds */
|
|
#define MAX_TIME 15000 /* 5000 = 5 sec */
|
|
|
|
/* Clock appereance */
|
|
#define CROSS_LINES 47
|
|
#define CROSS_CIRCLE 7
|
|
#define CROSS_CIRCLE2 27
|
|
#define CLOCK_RADIUS 50
|
|
#define CLOCK_LINE_WIDTH 10
|
|
#define CLOCK_LINE_PADDING 10
|
|
|
|
/* Text printed on screen */
|
|
#define HELP_TEXT_TITLE N_("Screen Calibration")
|
|
#define HELP_TEXT_MAIN N_("Please tap the target markers as they appear on screen to calibrate the tablet.")
|
|
|
|
static void
|
|
set_display_size(CalibArea *calib_area,
|
|
int width,
|
|
int height)
|
|
{
|
|
int delta_x;
|
|
int delta_y;
|
|
|
|
calib_area->display_width = width;
|
|
calib_area->display_height = height;
|
|
|
|
/* Compute absolute circle centers */
|
|
delta_x = calib_area->display_width/NUM_BLOCKS;
|
|
delta_y = calib_area->display_height/NUM_BLOCKS;
|
|
|
|
calib_area->X[UL] = delta_x;
|
|
calib_area->Y[UL] = delta_y;
|
|
|
|
calib_area->X[UR] = calib_area->display_width - delta_x - 1;
|
|
calib_area->Y[UR] = delta_y;
|
|
|
|
calib_area->X[LL] = delta_x;
|
|
calib_area->Y[LL] = calib_area->display_height - delta_y - 1;
|
|
|
|
calib_area->X[LR] = calib_area->display_width - delta_x - 1;
|
|
calib_area->Y[LR] = calib_area->display_height - delta_y - 1;
|
|
|
|
/* reset calibration if already started */
|
|
reset(&calib_area->calibrator);
|
|
}
|
|
|
|
static void
|
|
resize_display(CalibArea *calib_area)
|
|
{
|
|
/* check that screensize did not change (if no manually specified geometry) */
|
|
GtkAllocation allocation;
|
|
gtk_widget_get_allocation(calib_area->window, &allocation);
|
|
if (calib_area->display_width != allocation.width ||
|
|
calib_area->display_height != allocation.height)
|
|
{
|
|
set_display_size(calib_area, allocation.width, allocation.height);
|
|
}
|
|
}
|
|
|
|
static void
|
|
draw(GtkWidget *widget, cairo_t *cr, gpointer data)
|
|
{
|
|
CalibArea *calib_area = (CalibArea*)data;
|
|
int i;
|
|
double x;
|
|
double y;
|
|
PangoLayout *layout;
|
|
PangoRectangle logical_rect;
|
|
GtkStyleContext *context;
|
|
char *markup;
|
|
|
|
resize_display(calib_area);
|
|
|
|
context = gtk_widget_get_style_context (widget);
|
|
|
|
/* Black background and reset the operator */
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_CLEAR);
|
|
cairo_paint (cr);
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
|
|
|
|
/* Print the help lines */
|
|
layout = pango_layout_new (gtk_widget_get_pango_context (widget));
|
|
pango_layout_set_alignment (layout, PANGO_ALIGN_CENTER);
|
|
markup = g_strdup_printf ("<span foreground=\"white\"><big><b>%s</b></big>\n<big>%s</big></span>",
|
|
_(HELP_TEXT_TITLE),
|
|
_(HELP_TEXT_MAIN));
|
|
pango_layout_set_markup (layout, markup, -1);
|
|
g_free (markup);
|
|
|
|
pango_layout_get_pixel_extents (layout, NULL, &logical_rect);
|
|
|
|
x = (calib_area->display_width - logical_rect.width) / 2 + logical_rect.x;
|
|
y = (calib_area->display_height - logical_rect.height) / 2 - logical_rect.height - 40 + logical_rect.y;
|
|
|
|
gtk_render_layout (context, cr,
|
|
x + logical_rect.x,
|
|
y + logical_rect.y,
|
|
layout);
|
|
g_object_unref (layout);
|
|
|
|
/* Draw the points */
|
|
cairo_set_source_rgb(cr, 1.0, 1.0, 1.0);
|
|
|
|
i = calib_area->calibrator.num_clicks;
|
|
|
|
cairo_set_line_width(cr, 1);
|
|
cairo_move_to(cr, calib_area->X[i] - CROSS_LINES, calib_area->Y[i] - 0.5);
|
|
cairo_rel_line_to(cr, CROSS_LINES*2, 0);
|
|
cairo_move_to(cr, calib_area->X[i] - 0.5, calib_area->Y[i] - CROSS_LINES);
|
|
cairo_rel_line_to(cr, 0, CROSS_LINES*2);
|
|
cairo_stroke(cr);
|
|
|
|
cairo_set_line_width(cr, 2);
|
|
cairo_arc(cr, calib_area->X[i] - 0.5, calib_area->Y[i] - 0.5, CROSS_CIRCLE, 0.0, 2.0 * M_PI);
|
|
cairo_stroke(cr);
|
|
|
|
cairo_set_line_width(cr, 5);
|
|
cairo_arc(cr, calib_area->X[i] - 0.5, calib_area->Y[i] - 0.5, CROSS_CIRCLE2, 0.0, 2.0 * M_PI);
|
|
cairo_stroke(cr);
|
|
|
|
/* Draw the clock background */
|
|
cairo_arc(cr, calib_area->display_width/2, calib_area->display_height/2, CLOCK_RADIUS/2, 0.0, 2.0 * M_PI);
|
|
cairo_set_source_rgb(cr, 0.5, 0.5, 0.5);
|
|
cairo_fill_preserve(cr);
|
|
cairo_stroke(cr);
|
|
|
|
cairo_set_line_width(cr, CLOCK_LINE_WIDTH);
|
|
cairo_arc(cr, calib_area->display_width/2, calib_area->display_height/2, (CLOCK_RADIUS - CLOCK_LINE_WIDTH - CLOCK_LINE_PADDING)/2,
|
|
3/2.0*M_PI, (3/2.0*M_PI) + ((double)calib_area->time_elapsed/(double)MAX_TIME) * 2*M_PI);
|
|
cairo_set_source_rgb(cr, 1.0, 1.0, 1.0);
|
|
cairo_stroke(cr);
|
|
|
|
/* Draw the message (if any) */
|
|
if (calib_area->message != NULL)
|
|
{
|
|
cairo_set_source_rgb(cr, 1.0, 1.0, 1.0);
|
|
|
|
/* Frame the message */
|
|
layout = pango_layout_new (gtk_widget_get_pango_context (widget));
|
|
pango_layout_set_alignment (layout, PANGO_ALIGN_CENTER);
|
|
markup = g_strdup_printf ("<span foreground=\"white\"><big>%s</big></span>",
|
|
_(calib_area->message));
|
|
pango_layout_set_markup (layout, markup, -1);
|
|
g_free (markup);
|
|
pango_layout_get_pixel_extents (layout, NULL, &logical_rect);
|
|
|
|
x = (calib_area->display_width - logical_rect.width) / 2 + logical_rect.x;
|
|
y = (calib_area->display_height - logical_rect.height + CLOCK_RADIUS) / 2 + 60 + logical_rect.y;
|
|
cairo_set_line_width(cr, 2);
|
|
cairo_rectangle(cr, x - 10 - 0.5 , y - 10 - 0.5,
|
|
logical_rect.width + 20 + 1, logical_rect.height + 20 + 1);
|
|
cairo_stroke (cr);
|
|
|
|
/* Print the message */
|
|
gtk_render_layout (context, cr,
|
|
x + logical_rect.x,
|
|
y + logical_rect.y,
|
|
layout);
|
|
g_object_unref (layout);
|
|
}
|
|
}
|
|
|
|
static void
|
|
draw_message(CalibArea *calib_area,
|
|
const char *msg)
|
|
{
|
|
calib_area->message = msg;
|
|
}
|
|
|
|
static void
|
|
redraw(CalibArea *calib_area)
|
|
{
|
|
GdkWindow *win = gtk_widget_get_window(calib_area->window);
|
|
if (win)
|
|
{
|
|
GdkRectangle rect;
|
|
rect.x = 0;
|
|
rect.y = 0;
|
|
rect.width = calib_area->display_width;
|
|
rect.height = calib_area->display_height;
|
|
gdk_window_invalidate_rect(win, &rect, FALSE);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
on_delete_event (GtkWidget *widget,
|
|
GdkEvent *event,
|
|
CalibArea *area)
|
|
{
|
|
if (area->anim_id > 0) {
|
|
g_source_remove (area->anim_id);
|
|
area->anim_id = 0;
|
|
}
|
|
|
|
gtk_widget_hide (area->window);
|
|
|
|
(*area->callback) (area, area->user_data);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
on_button_press_event(GtkWidget *widget,
|
|
GdkEventButton *event,
|
|
CalibArea *area)
|
|
{
|
|
gboolean success;
|
|
|
|
/* Handle click */
|
|
area->time_elapsed = 0;
|
|
success = add_click(&area->calibrator, (int)event->x_root, (int)event->y_root);
|
|
|
|
if (!success && area->calibrator.num_clicks == 0)
|
|
draw_message(area, N_("Mis-click detected, restarting..."));
|
|
else
|
|
draw_message(area, NULL);
|
|
|
|
/* Are we done yet? */
|
|
if (area->calibrator.num_clicks >= 4)
|
|
{
|
|
on_delete_event (widget, NULL, area);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Force a redraw */
|
|
redraw(area);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
on_key_release_event(GtkWidget *widget,
|
|
GdkEventKey *event,
|
|
CalibArea *area)
|
|
{
|
|
if (event->type != GDK_KEY_RELEASE)
|
|
return FALSE;
|
|
if (event->keyval != GDK_KEY_Escape)
|
|
return FALSE;
|
|
|
|
on_delete_event (widget, NULL, area);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
on_focus_out_event (GtkWidget *widget,
|
|
GdkEvent *event,
|
|
CalibArea *area)
|
|
{
|
|
/* If the calibrator window looses focus, simply bail out... */
|
|
on_delete_event (widget, NULL, area);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
on_timer_signal(CalibArea *area)
|
|
{
|
|
GdkWindow *win;
|
|
|
|
area->time_elapsed += TIME_STEP;
|
|
if (area->time_elapsed > MAX_TIME)
|
|
{
|
|
on_delete_event (NULL, NULL, area);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Update clock */
|
|
win = gtk_widget_get_window (area->window);
|
|
if (win)
|
|
{
|
|
GdkRectangle rect;
|
|
rect.x = area->display_width/2 - CLOCK_RADIUS - CLOCK_LINE_WIDTH;
|
|
rect.y = area->display_height/2 - CLOCK_RADIUS - CLOCK_LINE_WIDTH;
|
|
rect.width = 2 * CLOCK_RADIUS + 1 + 2 * CLOCK_LINE_WIDTH;
|
|
rect.height = 2 * CLOCK_RADIUS + 1 + 2 * CLOCK_LINE_WIDTH;
|
|
gdk_window_invalidate_rect(win, &rect, FALSE);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* Creates the windows and other objects required to do calibration
|
|
* under GTK. When the window is closed (timed out, calibration finished
|
|
* or user cancellation), callback will be called, where you should call
|
|
* calib_area_finish().
|
|
*/
|
|
CalibArea *
|
|
calib_area_new (GdkScreen *screen,
|
|
int monitor,
|
|
FinishCallback callback,
|
|
gpointer user_data,
|
|
XYinfo *old_axis,
|
|
int threshold_doubleclick,
|
|
int threshold_misclick)
|
|
{
|
|
CalibArea *calib_area;
|
|
GdkRectangle rect;
|
|
GdkWindow *window;
|
|
GdkRGBA black;
|
|
GdkCursor *cursor;
|
|
|
|
g_return_val_if_fail (old_axis, NULL);
|
|
g_return_val_if_fail (callback, NULL);
|
|
|
|
g_debug ("Current calibration: %d, %d, %d, %d\n",
|
|
old_axis->x_min,
|
|
old_axis->y_min,
|
|
old_axis->x_max,
|
|
old_axis->y_max);
|
|
|
|
calib_area = g_new0 (CalibArea, 1);
|
|
calib_area->callback = callback;
|
|
calib_area->user_data = user_data;
|
|
calib_area->calibrator.old_axis.x_min = old_axis->x_min;
|
|
calib_area->calibrator.old_axis.x_max = old_axis->x_max;
|
|
calib_area->calibrator.old_axis.y_min = old_axis->y_min;
|
|
calib_area->calibrator.old_axis.y_max = old_axis->y_max;
|
|
calib_area->calibrator.threshold_doubleclick = threshold_doubleclick;
|
|
calib_area->calibrator.threshold_misclick = threshold_misclick;
|
|
|
|
/* Set up the window */
|
|
calib_area->window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_app_paintable (GTK_WIDGET (calib_area->window), TRUE);
|
|
|
|
/* Black background */
|
|
gdk_rgba_parse (&black, "rgb(0,0,0)");
|
|
gtk_window_set_opacity (GTK_WINDOW (calib_area->window), WINDOW_OPACITY);
|
|
|
|
gtk_widget_realize (calib_area->window);
|
|
window = gtk_widget_get_window (calib_area->window);
|
|
gdk_window_set_background_rgba (window, &black);
|
|
|
|
/* No cursor */
|
|
cursor = gdk_cursor_new (GDK_BLANK_CURSOR);
|
|
gdk_window_set_cursor (window, cursor);
|
|
g_object_unref (cursor);
|
|
|
|
/* Listen for mouse events */
|
|
gtk_widget_add_events (calib_area->window, GDK_KEY_RELEASE_MASK | GDK_BUTTON_PRESS_MASK);
|
|
gtk_widget_set_can_focus (calib_area->window, TRUE);
|
|
gtk_window_fullscreen (GTK_WINDOW (calib_area->window));
|
|
gtk_window_set_keep_above (GTK_WINDOW (calib_area->window), TRUE);
|
|
|
|
/* Connect callbacks */
|
|
g_signal_connect (calib_area->window, "draw",
|
|
G_CALLBACK(draw), calib_area);
|
|
g_signal_connect (calib_area->window, "button-press-event",
|
|
G_CALLBACK(on_button_press_event), calib_area);
|
|
g_signal_connect (calib_area->window, "key-release-event",
|
|
G_CALLBACK(on_key_release_event), calib_area);
|
|
g_signal_connect (calib_area->window, "delete-event",
|
|
G_CALLBACK(on_delete_event), calib_area);
|
|
g_signal_connect (calib_area->window, "focus-out-event",
|
|
G_CALLBACK(on_focus_out_event), calib_area);
|
|
|
|
/* Setup timer for animation */
|
|
calib_area->anim_id = g_timeout_add(TIME_STEP, (GSourceFunc)on_timer_signal, calib_area);
|
|
|
|
/* Move to correct screen */
|
|
if (screen == NULL)
|
|
screen = gdk_screen_get_default ();
|
|
gdk_screen_get_monitor_geometry (screen, monitor, &rect);
|
|
gtk_window_move (GTK_WINDOW (calib_area->window), rect.x, rect.y);
|
|
gtk_window_set_default_size (GTK_WINDOW (calib_area->window), rect.width, rect.height);
|
|
|
|
calib_area->calibrator.geometry.x = rect.x;
|
|
calib_area->calibrator.geometry.y = rect.y;
|
|
calib_area->calibrator.geometry.width = rect.width;
|
|
calib_area->calibrator.geometry.height = rect.height;
|
|
|
|
gtk_widget_show_all (calib_area->window);
|
|
|
|
return calib_area;
|
|
}
|
|
|
|
/* Finishes the calibration. Note that CalibArea
|
|
* needs to be destroyed with calib_area_free() afterwards */
|
|
gboolean
|
|
calib_area_finish (CalibArea *area,
|
|
XYinfo *new_axis,
|
|
gboolean *swap_xy)
|
|
{
|
|
gboolean success;
|
|
|
|
g_return_val_if_fail (area != NULL, FALSE);
|
|
|
|
success = finish (&area->calibrator, new_axis, swap_xy);
|
|
|
|
if (success)
|
|
g_debug ("Final calibration: %d, %d, %d, %d\n",
|
|
new_axis->x_min,
|
|
new_axis->y_min,
|
|
new_axis->x_max,
|
|
new_axis->y_max);
|
|
else
|
|
g_debug ("Calibration was aborted or timed out");
|
|
|
|
return success;
|
|
}
|
|
|
|
void
|
|
calib_area_free (CalibArea *area)
|
|
{
|
|
g_return_if_fail (area != NULL);
|
|
|
|
if (area->anim_id > 0) {
|
|
g_source_remove (area->anim_id);
|
|
area->anim_id = 0;
|
|
}
|
|
gtk_widget_destroy (area->window);
|
|
g_free (area);
|
|
}
|