Implement text cursor position protocol.

Here we create a new client/compositor interface in weston to allow
clients to report their x/y cursor position to the compositor. These
values are then used to center the zoom area on this point. This
is useful for everyone, especially people who are visually impaired.
dev
Scott Moreau 13 years ago committed by Kristian Høgsberg
parent d64bdf4755
commit 7a1b32a198
  1. 2
      clients/.gitignore
  2. 6
      clients/Makefile.am
  3. 14
      clients/terminal.c
  4. 26
      clients/window.c
  5. 3
      clients/window.h
  6. 11
      protocol/text-cursor-position.xml
  7. 2
      src/.gitignore
  8. 5
      src/Makefile.am
  9. 53
      src/compositor.c
  10. 13
      src/compositor.h
  11. 3
      src/shell.c
  12. 87
      src/text-cursor-position.c

@ -10,6 +10,8 @@ libtoytoolkit.a
resizor resizor
screenshooter-client-protocol.h screenshooter-client-protocol.h
screenshooter-protocol.c screenshooter-protocol.c
text-cursor-position-client-protocol.h
text-cursor-position-protocol.c
simple-egl simple-egl
simple-shm simple-shm
simple-touch simple-touch

@ -57,7 +57,9 @@ noinst_LIBRARIES = libtoytoolkit.a
libtoytoolkit_a_SOURCES = \ libtoytoolkit_a_SOURCES = \
window.c \ window.c \
window.h window.h \
text-cursor-position-protocol.c \
text-cursor-position-client-protocol.h
toolkit_libs = \ toolkit_libs = \
libtoytoolkit.a \ libtoytoolkit.a \
@ -106,6 +108,8 @@ weston_tablet_shell_LDADD = $(toolkit_libs)
BUILT_SOURCES = \ BUILT_SOURCES = \
screenshooter-client-protocol.h \ screenshooter-client-protocol.h \
screenshooter-protocol.c \ screenshooter-protocol.c \
text-cursor-position-client-protocol.h \
text-cursor-position-protocol.c \
desktop-shell-client-protocol.h \ desktop-shell-client-protocol.h \
desktop-shell-protocol.c \ desktop-shell-protocol.c \
tablet-shell-client-protocol.h \ tablet-shell-client-protocol.h \

@ -372,6 +372,7 @@ struct terminal {
int data_pitch, attr_pitch; /* The width in bytes of a line */ int data_pitch, attr_pitch; /* The width in bytes of a line */
int width, height, start, row, column; int width, height, start, row, column;
int saved_row, saved_column; int saved_row, saved_column;
int send_cursor_position;
int fd, master; int fd, master;
uint32_t modifiers; uint32_t modifiers;
char escape[MAX_ESCAPE+1]; char escape[MAX_ESCAPE+1];
@ -926,7 +927,7 @@ redraw_handler(struct widget *widget, void *data)
struct rectangle allocation; struct rectangle allocation;
cairo_t *cr; cairo_t *cr;
int top_margin, side_margin; int top_margin, side_margin;
int row, col; int row, col, cursor_x, cursor_y;
union utf8_char *p_row; union utf8_char *p_row;
union decoded_attr attr; union decoded_attr attr;
int text_x, text_y; int text_x, text_y;
@ -1022,6 +1023,16 @@ redraw_handler(struct widget *widget, void *data)
cairo_paint(cr); cairo_paint(cr);
cairo_destroy(cr); cairo_destroy(cr);
cairo_surface_destroy(surface); cairo_surface_destroy(surface);
if (terminal->send_cursor_position) {
cursor_x = side_margin + allocation.x +
terminal->column * extents.max_x_advance;
cursor_y = top_margin + allocation.y +
terminal->row * extents.height;
window_set_text_cursor_position(terminal->window,
cursor_x, cursor_y);
terminal->send_cursor_position = 0;
}
} }
static void static void
@ -1029,6 +1040,7 @@ terminal_write(struct terminal *terminal, const char *data, size_t length)
{ {
if (write(terminal->master, data, length) < 0) if (write(terminal->master, data, length) < 0)
abort(); abort();
terminal->send_cursor_position = 1;
} }
static void static void

@ -61,6 +61,7 @@
#include <linux/input.h> #include <linux/input.h>
#include <wayland-client.h> #include <wayland-client.h>
#include "../shared/cairo-util.h" #include "../shared/cairo-util.h"
#include "text-cursor-position-client-protocol.h"
#include "window.h" #include "window.h"
@ -72,6 +73,7 @@ struct display {
struct wl_shell *shell; struct wl_shell *shell;
struct wl_shm *shm; struct wl_shm *shm;
struct wl_data_device_manager *data_device_manager; struct wl_data_device_manager *data_device_manager;
struct text_cursor_position *text_cursor_position;
EGLDisplay dpy; EGLDisplay dpy;
EGLConfig argb_config; EGLConfig argb_config;
EGLContext argb_ctx; EGLContext argb_ctx;
@ -149,6 +151,7 @@ struct window {
int resize_needed; int resize_needed;
int type; int type;
int transparent; int transparent;
int send_cursor_position;
struct input *keyboard_device; struct input *keyboard_device;
enum window_buffer_type buffer_type; enum window_buffer_type buffer_type;
@ -1823,6 +1826,9 @@ keyboard_handle_key(void *data, struct wl_keyboard *keyboard,
if (!window || window->keyboard_device != input) if (!window || window->keyboard_device != input)
return; return;
if (state)
window->send_cursor_position = 1;
num_syms = xkb_key_get_syms(d->xkb.state, code, &syms); num_syms = xkb_key_get_syms(d->xkb.state, code, &syms);
xkb_state_update_key(d->xkb.state, code, xkb_state_update_key(d->xkb.state, code,
state ? XKB_KEY_DOWN : XKB_KEY_UP); state ? XKB_KEY_DOWN : XKB_KEY_UP);
@ -2609,6 +2615,21 @@ window_get_title(struct window *window)
return window->title; return window->title;
} }
void
window_set_text_cursor_position(struct window *window, int32_t x, int32_t y)
{
struct text_cursor_position *text_cursor_position =
window->display->text_cursor_position;
if (!window->send_cursor_position || !text_cursor_position)
return;
text_cursor_position_notify(text_cursor_position,
window->surface, x, y);
window->send_cursor_position = 0;
}
void void
window_damage(struct window *window, int32_t x, int32_t y, window_damage(struct window *window, int32_t x, int32_t y,
int32_t width, int32_t height) int32_t width, int32_t height)
@ -2695,6 +2716,7 @@ window_create_internal(struct display *display, struct window *parent)
window->allocation.height = 0; window->allocation.height = 0;
window->saved_allocation = window->allocation; window->saved_allocation = window->allocation;
window->transparent = 1; window->transparent = 1;
window->send_cursor_position = 0;
window->type = TYPE_NONE; window->type = TYPE_NONE;
window->input_region = NULL; window->input_region = NULL;
window->opaque_region = NULL; window->opaque_region = NULL;
@ -3097,6 +3119,10 @@ display_handle_global(struct wl_display *display, uint32_t id,
d->data_device_manager = d->data_device_manager =
wl_display_bind(display, id, wl_display_bind(display, id,
&wl_data_device_manager_interface); &wl_data_device_manager_interface);
} else if (strcmp(interface, "text_cursor_position") == 0) {
d->text_cursor_position =
wl_display_bind(display, id,
&text_cursor_position_interface);
} }
} }

@ -307,6 +307,9 @@ window_set_title(struct window *window, const char *title);
const char * const char *
window_get_title(struct window *window); window_get_title(struct window *window);
void
window_set_text_cursor_position(struct window *window, int32_t x, int32_t y);
int int
widget_set_tooltip(struct widget *parent, char *entry, float x, float y); widget_set_tooltip(struct widget *parent, char *entry, float x, float y);

@ -0,0 +1,11 @@
<protocol name="text_cursor_position">
<interface name="text_cursor_position" version="1">
<request name="notify">
<arg name="surface" type="object" interface="wl_surface"/>
<arg name="x" type="uint"/>
<arg name="y" type="uint"/>
</request>
</interface>
</protocol>

2
src/.gitignore vendored

@ -2,6 +2,8 @@ weston
weston-launch weston-launch
screenshooter-protocol.c screenshooter-protocol.c
screenshooter-server-protocol.h screenshooter-server-protocol.h
text-cursor-position-protocol.c
text-cursor-position-server-protocol.h
tablet-shell-protocol.c tablet-shell-protocol.c
tablet-shell-server-protocol.h tablet-shell-server-protocol.h
xserver-protocol.c xserver-protocol.c

@ -18,6 +18,9 @@ weston_SOURCES = \
screenshooter.c \ screenshooter.c \
screenshooter-protocol.c \ screenshooter-protocol.c \
screenshooter-server-protocol.h \ screenshooter-server-protocol.h \
text-cursor-position.c \
text-cursor-position-protocol.c \
text-cursor-position-server-protocol.h \
util.c \ util.c \
matrix.c \ matrix.c \
matrix.h \ matrix.h \
@ -144,6 +147,8 @@ endif
BUILT_SOURCES = \ BUILT_SOURCES = \
screenshooter-server-protocol.h \ screenshooter-server-protocol.h \
screenshooter-protocol.c \ screenshooter-protocol.c \
text-cursor-position-server-protocol.h \
text-cursor-position-protocol.c \
tablet-shell-protocol.c \ tablet-shell-protocol.c \
tablet-shell-server-protocol.h \ tablet-shell-server-protocol.h \
desktop-shell-protocol.c \ desktop-shell-protocol.c \

@ -1632,7 +1632,7 @@ notify_motion(struct wl_seat *seat, uint32_t time, wl_fixed_t x, wl_fixed_t y)
if (output->zoom.active && if (output->zoom.active &&
pixman_region32_contains_point(&output->region, pixman_region32_contains_point(&output->region,
ix, iy, NULL)) ix, iy, NULL))
weston_output_update_zoom(output, x, y); weston_output_update_zoom(output, x, y, ZOOM_POINTER);
weston_device_repick(seat); weston_device_repick(seat);
interface = seat->pointer->grab->interface; interface = seat->pointer->grab->interface;
@ -2445,9 +2445,34 @@ weston_output_destroy(struct weston_output *output)
} }
WL_EXPORT void WL_EXPORT void
weston_output_update_zoom(struct weston_output *output, wl_fixed_t fx, wl_fixed_t fy) weston_text_cursor_position_notify(struct weston_surface *surface,
int32_t cur_pos_x,
int32_t cur_pos_y)
{
struct weston_output *output;
int32_t global_x, global_y;
weston_surface_to_global(surface, cur_pos_x, cur_pos_y,
&global_x, &global_y);
wl_list_for_each(output, &surface->compositor->output_list, link)
if (output->zoom.active &&
pixman_region32_contains_point(&output->region,
global_x, global_y, NULL))
weston_output_update_zoom(output,
wl_fixed_from_int(global_x),
wl_fixed_from_int(global_y),
ZOOM_TEXT_CURSOR);
}
WL_EXPORT void
weston_output_update_zoom(struct weston_output *output,
wl_fixed_t fx,
wl_fixed_t fy,
uint32_t type)
{ {
int32_t x, y; int32_t x, y;
float trans_min, trans_max;
if (output->zoom.level >= 1.0) if (output->zoom.level >= 1.0)
return; return;
@ -2455,11 +2480,30 @@ weston_output_update_zoom(struct weston_output *output, wl_fixed_t fx, wl_fixed_
x = wl_fixed_to_int(fx); x = wl_fixed_to_int(fx);
y = wl_fixed_to_int(fy); y = wl_fixed_to_int(fy);
output->zoom.trans_x = (((float)(x - output->x) / output->current->width) * output->zoom.trans_x =
(((float)(x - output->x) / output->current->width) *
(output->zoom.level * 2)) - output->zoom.level; (output->zoom.level * 2)) - output->zoom.level;
output->zoom.trans_y = (((float)(y - output->y) / output->current->height) * output->zoom.trans_y =
(((float)(y - output->y) / output->current->height) *
(output->zoom.level * 2)) - output->zoom.level; (output->zoom.level * 2)) - output->zoom.level;
if (type == ZOOM_TEXT_CURSOR) {
output->zoom.trans_x *= 1 / output->zoom.level;
output->zoom.trans_y *= 1 / output->zoom.level;
trans_max = output->zoom.level * 2 - output->zoom.level;
trans_min = -trans_max;
if (output->zoom.trans_x > trans_max)
output->zoom.trans_x = trans_max;
else if (output->zoom.trans_x < trans_min)
output->zoom.trans_x = trans_min;
if (output->zoom.trans_y > trans_max)
output->zoom.trans_y = trans_max;
else if (output->zoom.trans_y < trans_min)
output->zoom.trans_y = trans_min;
}
output->dirty = 1; output->dirty = 1;
weston_output_damage(output); weston_output_damage(output);
} }
@ -2647,6 +2691,7 @@ weston_compositor_init(struct weston_compositor *ec, struct wl_display *display)
weston_layer_init(&ec->cursor_layer, &ec->fade_layer.link); weston_layer_init(&ec->cursor_layer, &ec->fade_layer.link);
screenshooter_create(ec); screenshooter_create(ec);
text_cursor_position_notifier_create(ec);
ec->ping_handler = NULL; ec->ping_handler = NULL;

@ -82,6 +82,11 @@ struct weston_border {
int32_t left, right, top, bottom; int32_t left, right, top, bottom;
}; };
enum {
ZOOM_POINTER,
ZOOM_TEXT_CURSOR
};
struct weston_output_zoom { struct weston_output_zoom {
int active; int active;
float increment; float increment;
@ -561,7 +566,10 @@ weston_compositor_init(struct weston_compositor *ec, struct wl_display *display)
void void
weston_compositor_shutdown(struct weston_compositor *ec); weston_compositor_shutdown(struct weston_compositor *ec);
void void
weston_output_update_zoom(struct weston_output *output, int x, int y); weston_output_update_zoom(struct weston_output *output,
int x, int y, uint32_t type);
void
weston_text_cursor_position_notify(struct weston_surface *surface, int x, int y);
void void
weston_output_update_matrix(struct weston_output *output); weston_output_update_matrix(struct weston_output *output);
void void
@ -598,6 +606,9 @@ tty_activate_vt(struct tty *tty, int vt);
void void
screenshooter_create(struct weston_compositor *ec); screenshooter_create(struct weston_compositor *ec);
void
text_cursor_position_notifier_create(struct weston_compositor *ec);
struct weston_process; struct weston_process;
typedef void (*weston_process_cleanup_func_t)(struct weston_process *process, typedef void (*weston_process_cleanup_func_t)(struct weston_process *process,
int status); int status);

@ -1689,7 +1689,8 @@ zoom_binding(struct wl_seat *seat, uint32_t time,
weston_output_update_zoom(output, weston_output_update_zoom(output,
seat->pointer->x, seat->pointer->x,
seat->pointer->y); seat->pointer->y,
ZOOM_POINTER);
} }
} }
} }

@ -0,0 +1,87 @@
/*
* Copyright © 2011 Intel Corporation
* Copyright © 2012 Scott Moreau
*
* Permission to use, copy, modify, distribute, and sell this software and
* its documentation for any purpose is hereby granted without fee, provided
* that the above copyright notice appear in all copies and that both that
* copyright notice and this permission notice appear in supporting
* documentation, and that the name of the copyright holders not be used in
* advertising or publicity pertaining to distribution of the software
* without specific, written prior permission. The copyright holders make
* no representations about the suitability of this software for any
* purpose. It is provided "as is" without express or implied warranty.
*
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
* SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
* CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include <stdlib.h>
#include "compositor.h"
#include "text-cursor-position-server-protocol.h"
struct text_cursor_position {
struct wl_object base;
struct weston_compositor *ec;
struct wl_global *global;
struct wl_listener destroy_listener;
};
static void
text_cursor_position_notify(struct wl_client *client,
struct wl_resource *resource,
struct wl_resource *surface_resource,
uint32_t x, uint32_t y)
{
weston_text_cursor_position_notify((struct weston_surface *) surface_resource, x, y);
}
struct text_cursor_position_interface text_cursor_position_implementation = {
text_cursor_position_notify
};
static void
bind_text_cursor_position(struct wl_client *client,
void *data, uint32_t version, uint32_t id)
{
wl_client_add_object(client, &text_cursor_position_interface,
&text_cursor_position_implementation, id, data);
}
static void
text_cursor_position_notifier_destroy(struct wl_listener *listener, void *data)
{
struct text_cursor_position *text_cursor_position =
container_of(listener, struct text_cursor_position, destroy_listener);
wl_display_remove_global(text_cursor_position->ec->wl_display, text_cursor_position->global);
free(text_cursor_position);
}
void
text_cursor_position_notifier_create(struct weston_compositor *ec)
{
struct text_cursor_position *text_cursor_position;
text_cursor_position = malloc(sizeof *text_cursor_position);
if (text_cursor_position == NULL)
return;
text_cursor_position->base.interface = &text_cursor_position_interface;
text_cursor_position->base.implementation =
(void(**)(void)) &text_cursor_position_implementation;
text_cursor_position->ec = ec;
text_cursor_position->global = wl_display_add_global(ec->wl_display,
&text_cursor_position_interface,
text_cursor_position, bind_text_cursor_position);
text_cursor_position->destroy_listener.notify = text_cursor_position_notifier_destroy;
wl_signal_add(&ec->destroy_signal, &text_cursor_position->destroy_listener);
}
Loading…
Cancel
Save