blob: 07a066c99d89d438426bfd7f705e08dbd246a654 [file] [log] [blame]
Jan Arne Petersen1f17be42012-06-21 21:52:18 +02001/*
2 * Copyright © 2012 Openismus GmbH
Jan Arne Petersen4c265182012-09-09 23:08:30 +02003 * Copyright © 2012 Intel Corporation
Jan Arne Petersen1f17be42012-06-21 21:52:18 +02004 *
5 * Permission to use, copy, modify, distribute, and sell this software and
6 * its documentation for any purpose is hereby granted without fee, provided
7 * that the above copyright notice appear in all copies and that both that
8 * copyright notice and this permission notice appear in supporting
9 * documentation, and that the name of the copyright holders not be used in
10 * advertising or publicity pertaining to distribution of the software
11 * without specific, written prior permission. The copyright holders make
12 * no representations about the suitability of this software for any
13 * purpose. It is provided "as is" without express or implied warranty.
14 *
15 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
16 * SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
17 * FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
18 * SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
19 * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
20 * CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
21 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
22 */
23
24#include <stdlib.h>
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010025#include <string.h>
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020026
27#include "compositor.h"
28#include "text-server-protocol.h"
Jan Arne Petersen30b66ef2012-09-09 23:08:41 +020029#include "input-method-server-protocol.h"
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020030
31struct input_method;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020032struct input_method_context;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010033struct text_backend;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020034
Jan Arne Petersen78d00e42013-04-18 16:47:24 +020035struct text_input {
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020036 struct wl_resource resource;
37
Jan Arne Petersene829adc2012-08-10 16:47:22 +020038 struct weston_compositor *ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020039
Jan Arne Petersene829adc2012-08-10 16:47:22 +020040 struct wl_list input_methods;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +020041
42 struct wl_surface *surface;
Jan Arne Petersen61381972013-01-31 15:52:21 +010043
Jan Arne Petersen14da96b2013-04-18 16:47:28 +020044 pixman_box32_t cursor_rectangle;
45
Jan Arne Petersen61381972013-01-31 15:52:21 +010046 uint32_t input_panel_visible;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020047};
48
Jan Arne Petersen78d00e42013-04-18 16:47:24 +020049struct text_input_manager {
50 struct wl_global *text_input_manager_global;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020051 struct wl_listener destroy_listener;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020052
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +020053 struct weston_compositor *ec;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +020054};
55
56struct input_method {
57 struct wl_resource *input_method_binding;
58 struct wl_global *input_method_global;
59 struct wl_listener destroy_listener;
60
61 struct weston_seat *seat;
Jan Arne Petersen78d00e42013-04-18 16:47:24 +020062 struct text_input *model;
Jan Arne Petersene829adc2012-08-10 16:47:22 +020063
64 struct wl_list link;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +020065
66 struct wl_listener keyboard_focus_listener;
67
68 int focus_listener_initialized;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020069
70 struct input_method_context *context;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010071
72 struct text_backend *text_backend;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020073};
74
Jan Arne Petersen620cd622012-09-09 23:08:32 +020075struct input_method_context {
76 struct wl_resource resource;
77
Jan Arne Petersen78d00e42013-04-18 16:47:24 +020078 struct text_input *model;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +010079 struct input_method *input_method;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020080
81 struct wl_list link;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +010082
83 struct wl_resource *keyboard;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020084};
85
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010086struct text_backend {
87 struct weston_compositor *compositor;
88
89 struct {
90 char *path;
91 struct wl_resource *binding;
92 struct weston_process process;
93 struct wl_client *client;
94 } input_method;
95
96 struct wl_listener seat_created_listener;
97 struct wl_listener destroy_listener;
98};
99
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200100static void input_method_context_create(struct text_input *model,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100101 struct input_method *input_method,
102 uint32_t serial);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100103static void input_method_context_end_keyboard_grab(struct input_method_context *context);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200104
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200105static void input_method_init_seat(struct weston_seat *seat);
106
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200107static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200108deactivate_text_input(struct text_input *text_input,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200109 struct input_method *input_method)
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200110{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200111 struct weston_compositor *ec = text_input->ec;
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200112
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200113 if (input_method->model == text_input) {
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100114 if (input_method->context && input_method->input_method_binding) {
115 input_method_context_end_keyboard_grab(input_method->context);
116 input_method_send_deactivate(input_method->input_method_binding,
117 &input_method->context->resource);
118 }
119
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200120 wl_list_remove(&input_method->link);
121 input_method->model = NULL;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200122 input_method->context = NULL;
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400123 wl_signal_emit(&ec->hide_input_panel_signal, ec);
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200124 text_input_send_leave(&text_input->resource);
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200125 }
126}
127
128static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200129destroy_text_input(struct wl_resource *resource)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200130{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200131 struct text_input *text_input =
132 container_of(resource, struct text_input, resource);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200133 struct input_method *input_method, *next;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200134
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200135 wl_list_for_each_safe(input_method, next, &text_input->input_methods, link)
136 deactivate_text_input(text_input, input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200137
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200138 free(text_input);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200139}
140
141static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200142text_input_set_surrounding_text(struct wl_client *client,
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200143 struct wl_resource *resource,
Jan Arne Petersencb08f4d2012-09-09 23:08:40 +0200144 const char *text,
145 uint32_t cursor,
146 uint32_t anchor)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200147{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200148 struct text_input *text_input = resource->data;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200149 struct input_method *input_method, *next;
150
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200151 wl_list_for_each_safe(input_method, next, &text_input->input_methods, link) {
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200152 if (!input_method->context)
153 continue;
Jan Arne Petersencb08f4d2012-09-09 23:08:40 +0200154 input_method_context_send_surrounding_text(&input_method->context->resource,
155 text,
156 cursor,
157 anchor);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200158 }
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200159}
160
161static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200162text_input_activate(struct wl_client *client,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200163 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100164 uint32_t serial,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200165 struct wl_resource *seat,
166 struct wl_resource *surface)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200167{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200168 struct text_input *text_input = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200169 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200170 struct input_method *input_method = weston_seat->input_method;
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200171 struct text_input *old = weston_seat->input_method->model;
172 struct weston_compositor *ec = text_input->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200173
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200174 if (old == text_input)
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200175 return;
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200176
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200177 if (old) {
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200178 deactivate_text_input(old,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200179 weston_seat->input_method);
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200180 }
181
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200182 input_method->model = text_input;
183 wl_list_insert(&text_input->input_methods, &input_method->link);
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200184 input_method_init_seat(weston_seat);
185
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200186 text_input->surface = surface->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200187
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200188 input_method_context_create(text_input, input_method, serial);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200189
Jan Arne Petersen14da96b2013-04-18 16:47:28 +0200190 if (text_input->input_panel_visible) {
191 wl_signal_emit(&ec->show_input_panel_signal, text_input->surface);
192 wl_signal_emit(&ec->update_input_panel_signal, &text_input->cursor_rectangle);
193 }
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200194
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200195 text_input_send_enter(&text_input->resource, &text_input->surface->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200196}
197
198static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200199text_input_deactivate(struct wl_client *client,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200200 struct wl_resource *resource,
201 struct wl_resource *seat)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200202{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200203 struct text_input *text_input = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200204 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200205
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200206 deactivate_text_input(text_input,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200207 weston_seat->input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200208}
209
210static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200211text_input_reset(struct wl_client *client,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100212 struct wl_resource *resource,
213 uint32_t serial)
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200214{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200215 struct text_input *text_input = resource->data;
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200216 struct input_method *input_method, *next;
217
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200218 wl_list_for_each_safe(input_method, next, &text_input->input_methods, link) {
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200219 if (!input_method->context)
220 continue;
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100221 input_method_context_send_reset(&input_method->context->resource, serial);
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200222 }
223}
224
225static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200226text_input_set_cursor_rectangle(struct wl_client *client,
Jan Arne Petersen7ef8eff2013-04-18 16:47:23 +0200227 struct wl_resource *resource,
228 int32_t x,
229 int32_t y,
230 int32_t width,
231 int32_t height)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200232{
Jan Arne Petersen14da96b2013-04-18 16:47:28 +0200233 struct text_input *text_input = resource->data;
234 struct weston_compositor *ec = text_input->ec;
235
236 text_input->cursor_rectangle.x1 = x;
237 text_input->cursor_rectangle.y1 = y;
238 text_input->cursor_rectangle.x2 = x + width;
239 text_input->cursor_rectangle.y2 = y + height;
240
241 wl_signal_emit(&ec->update_input_panel_signal, &text_input->cursor_rectangle);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200242}
243
244static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200245text_input_set_content_type(struct wl_client *client,
Jan Arne Petersen26ffa812013-01-16 21:26:43 +0100246 struct wl_resource *resource,
247 uint32_t hint,
248 uint32_t purpose)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200249{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200250 struct text_input *text_input = resource->data;
Jan Arne Petersen26ffa812013-01-16 21:26:43 +0100251 struct input_method *input_method, *next;
252
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200253 wl_list_for_each_safe(input_method, next, &text_input->input_methods, link) {
Jan Arne Petersen26ffa812013-01-16 21:26:43 +0100254 if (!input_method->context)
255 continue;
256 input_method_context_send_content_type(&input_method->context->resource, hint, purpose);
257 }
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200258}
259
Jan Arne Petersenadfedc12013-01-16 21:26:46 +0100260static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200261text_input_invoke_action(struct wl_client *client,
Jan Arne Petersenadfedc12013-01-16 21:26:46 +0100262 struct wl_resource *resource,
263 uint32_t button,
264 uint32_t index)
265{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200266 struct text_input *text_input = resource->data;
Jan Arne Petersenadfedc12013-01-16 21:26:46 +0100267 struct input_method *input_method, *next;
268
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200269 wl_list_for_each_safe(input_method, next, &text_input->input_methods, link) {
Jan Arne Petersenadfedc12013-01-16 21:26:46 +0100270 if (!input_method->context)
271 continue;
272 input_method_context_send_invoke_action(&input_method->context->resource, button, index);
273 }
274}
275
Jan Arne Petersen0eabcaa2013-01-31 15:52:20 +0100276static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200277text_input_commit_state(struct wl_client *client,
Jan Arne Petersenfa357382013-04-18 16:47:22 +0200278 struct wl_resource *resource)
Jan Arne Petersen0eabcaa2013-01-31 15:52:20 +0100279{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200280 struct text_input *text_input = resource->data;
Jan Arne Petersen0eabcaa2013-01-31 15:52:20 +0100281 struct input_method *input_method, *next;
282
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200283 wl_list_for_each_safe(input_method, next, &text_input->input_methods, link) {
Jan Arne Petersen0eabcaa2013-01-31 15:52:20 +0100284 if (!input_method->context)
285 continue;
286 input_method_context_send_commit(&input_method->context->resource);
287 }
288}
289
Jan Arne Petersen61381972013-01-31 15:52:21 +0100290static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200291text_input_show_input_panel(struct wl_client *client,
Jan Arne Petersen61381972013-01-31 15:52:21 +0100292 struct wl_resource *resource)
293{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200294 struct text_input *text_input = resource->data;
295 struct weston_compositor *ec = text_input->ec;
Jan Arne Petersen61381972013-01-31 15:52:21 +0100296
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200297 text_input->input_panel_visible = 1;
Jan Arne Petersen61381972013-01-31 15:52:21 +0100298
Jan Arne Petersen14da96b2013-04-18 16:47:28 +0200299 if (!wl_list_empty(&text_input->input_methods)) {
300 wl_signal_emit(&ec->show_input_panel_signal, text_input->surface);
301 wl_signal_emit(&ec->update_input_panel_signal, &text_input->cursor_rectangle);
302 }
Jan Arne Petersen61381972013-01-31 15:52:21 +0100303}
304
305static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200306text_input_hide_input_panel(struct wl_client *client,
Jan Arne Petersen61381972013-01-31 15:52:21 +0100307 struct wl_resource *resource)
308{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200309 struct text_input *text_input = resource->data;
310 struct weston_compositor *ec = text_input->ec;
Jan Arne Petersen61381972013-01-31 15:52:21 +0100311
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200312 text_input->input_panel_visible = 0;
Jan Arne Petersen61381972013-01-31 15:52:21 +0100313
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200314 if (!wl_list_empty(&text_input->input_methods))
Jan Arne Petersen61381972013-01-31 15:52:21 +0100315 wl_signal_emit(&ec->hide_input_panel_signal, ec);
316}
317
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200318static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200319text_input_set_preferred_language(struct wl_client *client,
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200320 struct wl_resource *resource,
321 const char *language)
322{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200323 struct text_input *text_input = resource->data;
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200324 struct input_method *input_method, *next;
325
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200326 wl_list_for_each_safe(input_method, next, &text_input->input_methods, link) {
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200327 if (!input_method->context)
328 continue;
329 input_method_context_send_preferred_language(&input_method->context->resource,
330 language);
331 }
332}
333
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200334static const struct text_input_interface text_input_implementation = {
335 text_input_activate,
336 text_input_deactivate,
337 text_input_show_input_panel,
338 text_input_hide_input_panel,
339 text_input_reset,
340 text_input_set_surrounding_text,
341 text_input_set_content_type,
342 text_input_set_cursor_rectangle,
343 text_input_set_preferred_language,
344 text_input_commit_state,
345 text_input_invoke_action
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200346};
347
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200348static void text_input_manager_create_text_input(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200349 struct wl_resource *resource,
Jan Arne Petersen4c265182012-09-09 23:08:30 +0200350 uint32_t id)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200351{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200352 struct text_input_manager *text_input_manager = resource->data;
353 struct text_input *text_input;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200354
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200355 text_input = calloc(1, sizeof *text_input);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200356
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200357 text_input->resource.object.id = id;
358 text_input->resource.object.interface = &text_input_interface;
359 text_input->resource.object.implementation =
360 (void (**)(void)) &text_input_implementation;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200361
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200362 text_input->resource.data = text_input;
363 text_input->resource.destroy = destroy_text_input;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200364
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200365 text_input->ec = text_input_manager->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200366
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200367 wl_list_init(&text_input->input_methods);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200368
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200369 wl_client_add_resource(client, &text_input->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200370};
371
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200372static const struct text_input_manager_interface text_input_manager_implementation = {
373 text_input_manager_create_text_input
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200374};
375
376static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200377bind_text_input_manager(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200378 void *data,
379 uint32_t version,
380 uint32_t id)
381{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200382 struct text_input_manager *text_input_manager = data;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200383
384 /* No checking for duplicate binding necessary.
385 * No events have to be sent, so we don't need the return value. */
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200386 wl_client_add_object(client, &text_input_manager_interface,
387 &text_input_manager_implementation,
388 id, text_input_manager);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200389}
390
391static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200392text_input_manager_notifier_destroy(struct wl_listener *listener, void *data)
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200393{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200394 struct text_input_manager *text_input_manager =
395 container_of(listener, struct text_input_manager, destroy_listener);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200396
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200397 wl_display_remove_global(text_input_manager->ec->wl_display,
398 text_input_manager->text_input_manager_global);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200399
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200400 free(text_input_manager);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200401}
402
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100403static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200404text_input_manager_create(struct weston_compositor *ec)
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200405{
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200406 struct text_input_manager *text_input_manager;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200407
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200408 text_input_manager = calloc(1, sizeof *text_input_manager);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200409
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200410 text_input_manager->ec = ec;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200411
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200412 text_input_manager->text_input_manager_global =
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200413 wl_display_add_global(ec->wl_display,
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200414 &text_input_manager_interface,
415 text_input_manager, bind_text_input_manager);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200416
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200417 text_input_manager->destroy_listener.notify = text_input_manager_notifier_destroy;
418 wl_signal_add(&ec->destroy_signal, &text_input_manager->destroy_listener);
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200419}
420
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200421static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200422input_method_context_destroy(struct wl_client *client,
423 struct wl_resource *resource)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200424{
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200425 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200426}
427
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200428static void
429input_method_context_commit_string(struct wl_client *client,
430 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100431 uint32_t serial,
Jan Arne Petersen1cc9e082013-01-31 15:52:23 +0100432 const char *text)
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200433{
434 struct input_method_context *context = resource->data;
435
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200436 text_input_send_commit_string(&context->model->resource, serial, text);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200437}
438
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200439static void
440input_method_context_preedit_string(struct wl_client *client,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100441 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100442 uint32_t serial,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100443 const char *text,
444 const char *commit)
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200445{
446 struct input_method_context *context = resource->data;
447
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200448 text_input_send_preedit_string(&context->model->resource, serial, text, commit);
Jan Arne Petersen46535312013-01-16 21:26:38 +0100449}
450
451static void
452input_method_context_preedit_styling(struct wl_client *client,
453 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100454 uint32_t serial,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100455 uint32_t index,
456 uint32_t length,
457 uint32_t style)
458{
459 struct input_method_context *context = resource->data;
460
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200461 text_input_send_preedit_styling(&context->model->resource, serial, index, length, style);
Jan Arne Petersen46535312013-01-16 21:26:38 +0100462}
463
464static void
465input_method_context_preedit_cursor(struct wl_client *client,
466 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100467 uint32_t serial,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100468 int32_t cursor)
469{
470 struct input_method_context *context = resource->data;
471
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200472 text_input_send_preedit_cursor(&context->model->resource, serial, cursor);
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200473}
474
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200475static void
476input_method_context_delete_surrounding_text(struct wl_client *client,
477 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100478 uint32_t serial,
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200479 int32_t index,
480 uint32_t length)
481{
482 struct input_method_context *context = resource->data;
483
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200484 text_input_send_delete_surrounding_text(&context->model->resource, serial, index, length);
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200485}
486
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200487static void
Jan Arne Petersen1cc9e082013-01-31 15:52:23 +0100488input_method_context_cursor_position(struct wl_client *client,
489 struct wl_resource *resource,
490 uint32_t serial,
491 int32_t index,
492 int32_t anchor)
493{
494 struct input_method_context *context = resource->data;
495
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200496 text_input_send_cursor_position(&context->model->resource, serial, index, anchor);
Jan Arne Petersen1cc9e082013-01-31 15:52:23 +0100497}
498
499static void
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100500input_method_context_modifiers_map(struct wl_client *client,
501 struct wl_resource *resource,
502 struct wl_array *map)
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200503{
504 struct input_method_context *context = resource->data;
505
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200506 text_input_send_modifiers_map(&context->model->resource, map);
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100507}
508
509static void
510input_method_context_keysym(struct wl_client *client,
511 struct wl_resource *resource,
512 uint32_t serial,
513 uint32_t time,
514 uint32_t sym,
515 uint32_t state,
516 uint32_t modifiers)
517{
518 struct input_method_context *context = resource->data;
519
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200520 text_input_send_keysym(&context->model->resource, serial, time,
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100521 sym, state, modifiers);
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200522}
523
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100524static void
525unbind_keyboard(struct wl_resource *resource)
526{
527 struct input_method_context *context = resource->data;
528
529 input_method_context_end_keyboard_grab(context);
530 context->keyboard = NULL;
531
532 free(resource);
533}
534
535static void
536input_method_context_grab_key(struct wl_keyboard_grab *grab,
537 uint32_t time, uint32_t key, uint32_t state_w)
538{
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100539 struct weston_keyboard *keyboard = (struct weston_keyboard *)grab->keyboard;
540 struct wl_display *display;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100541 uint32_t serial;
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100542
543 if (!keyboard->input_method_resource)
544 return;
545
546 display = wl_client_get_display(keyboard->input_method_resource->client);
547 serial = wl_display_next_serial(display);
548 wl_keyboard_send_key(keyboard->input_method_resource,
549 serial, time, key, state_w);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100550}
551
552static void
553input_method_context_grab_modifier(struct wl_keyboard_grab *grab, uint32_t serial,
554 uint32_t mods_depressed, uint32_t mods_latched,
555 uint32_t mods_locked, uint32_t group)
556{
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100557 struct weston_keyboard *keyboard = (struct weston_keyboard *)grab->keyboard;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100558
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100559 if (!keyboard->input_method_resource)
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100560 return;
561
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100562 wl_keyboard_send_modifiers(keyboard->input_method_resource,
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100563 serial, mods_depressed, mods_latched,
564 mods_locked, group);
565}
566
567static const struct wl_keyboard_grab_interface input_method_context_grab = {
568 input_method_context_grab_key,
569 input_method_context_grab_modifier,
570};
571
572static void
573input_method_context_grab_keyboard(struct wl_client *client,
574 struct wl_resource *resource,
575 uint32_t id)
576{
577 struct input_method_context *context = resource->data;
578 struct wl_resource *cr;
579 struct weston_seat *seat = context->input_method->seat;
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100580 struct weston_keyboard *keyboard = &seat->keyboard;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100581
582 cr = wl_client_add_object(client, &wl_keyboard_interface,
583 NULL, id, context);
584 cr->destroy = unbind_keyboard;
585
586 context->keyboard = cr;
587
588 wl_keyboard_send_keymap(cr, WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1,
589 seat->xkb_info.keymap_fd,
590 seat->xkb_info.keymap_size);
591
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100592 if (keyboard->keyboard.grab != &keyboard->keyboard.default_grab) {
593 wl_keyboard_end_grab(&keyboard->keyboard);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100594 }
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100595 wl_keyboard_start_grab(&keyboard->keyboard, &keyboard->input_method_grab);
596 keyboard->input_method_resource = cr;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100597}
598
Jan Arne Petersen337df952012-11-18 19:06:46 +0100599static void
600input_method_context_key(struct wl_client *client,
601 struct wl_resource *resource,
602 uint32_t serial,
603 uint32_t time,
604 uint32_t key,
605 uint32_t state_w)
606{
607 struct input_method_context *context = resource->data;
608 struct weston_seat *seat = context->input_method->seat;
609 struct wl_keyboard *keyboard = seat->seat.keyboard;
610 struct wl_keyboard_grab *default_grab = &keyboard->default_grab;
611
612 default_grab->interface->key(default_grab, time, key, state_w);
613}
614
615static void
616input_method_context_modifiers(struct wl_client *client,
617 struct wl_resource *resource,
618 uint32_t serial,
619 uint32_t mods_depressed,
620 uint32_t mods_latched,
621 uint32_t mods_locked,
622 uint32_t group)
623{
624 struct input_method_context *context = resource->data;
625
626 struct weston_seat *seat = context->input_method->seat;
627 struct wl_keyboard *keyboard = seat->seat.keyboard;
628 struct wl_keyboard_grab *default_grab = &keyboard->default_grab;
629
630 default_grab->interface->modifiers(default_grab,
631 serial, mods_depressed,
632 mods_latched, mods_locked,
633 group);
634}
635
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200636static void
637input_method_context_language(struct wl_client *client,
638 struct wl_resource *resource,
639 uint32_t serial,
640 const char *language)
641{
642 struct input_method_context *context = resource->data;
643
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200644 text_input_send_language(&context->model->resource, serial, language);
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200645}
646
647static void
648input_method_context_text_direction(struct wl_client *client,
649 struct wl_resource *resource,
650 uint32_t serial,
651 uint32_t direction)
652{
653 struct input_method_context *context = resource->data;
654
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200655 text_input_send_text_direction(&context->model->resource, serial, direction);
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200656}
657
658
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200659static const struct input_method_context_interface input_method_context_implementation = {
660 input_method_context_destroy,
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200661 input_method_context_commit_string,
662 input_method_context_preedit_string,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100663 input_method_context_preedit_styling,
664 input_method_context_preedit_cursor,
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200665 input_method_context_delete_surrounding_text,
Jan Arne Petersen1cc9e082013-01-31 15:52:23 +0100666 input_method_context_cursor_position,
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100667 input_method_context_modifiers_map,
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100668 input_method_context_keysym,
Jan Arne Petersen337df952012-11-18 19:06:46 +0100669 input_method_context_grab_keyboard,
670 input_method_context_key,
Jan Arne Petersenece6b5a2013-04-18 16:47:15 +0200671 input_method_context_modifiers,
672 input_method_context_language,
673 input_method_context_text_direction
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200674};
675
676static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200677destroy_input_method_context(struct wl_resource *resource)
678{
679 struct input_method_context *context = resource->data;
680
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100681 if (context->keyboard) {
682 wl_resource_destroy(context->keyboard);
683 }
684
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200685 free(context);
686}
687
688static void
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200689input_method_context_create(struct text_input *model,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100690 struct input_method *input_method,
691 uint32_t serial)
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200692{
693 struct input_method_context *context;
694
695 if (!input_method->input_method_binding)
696 return;
697
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100698 context = calloc(1, sizeof *context);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200699 if (context == NULL)
700 return;
701
702 context->resource.destroy = destroy_input_method_context;
703 context->resource.object.id = 0;
704 context->resource.object.interface = &input_method_context_interface;
705 context->resource.object.implementation =
706 (void (**)(void)) &input_method_context_implementation;
707 context->resource.data = context;
708 wl_signal_init(&context->resource.destroy_signal);
709
710 context->model = model;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100711 context->input_method = input_method;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200712 input_method->context = context;
713
714 wl_client_add_resource(input_method->input_method_binding->client, &context->resource);
715
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100716 input_method_send_activate(input_method->input_method_binding, &context->resource, serial);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200717}
718
719static void
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100720input_method_context_end_keyboard_grab(struct input_method_context *context)
721{
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100722 struct wl_keyboard_grab *grab = &context->input_method->seat->keyboard.input_method_grab;
723 struct weston_keyboard *keyboard = (struct weston_keyboard *)grab->keyboard;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100724
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100725 if (!grab->keyboard)
726 return;
727
728 if (grab->keyboard->grab == grab)
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100729 wl_keyboard_end_grab(grab->keyboard);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100730
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100731 keyboard->input_method_resource = NULL;
732}
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100733
734static void
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200735unbind_input_method(struct wl_resource *resource)
736{
737 struct input_method *input_method = resource->data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100738 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200739
740 input_method->input_method_binding = NULL;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200741 input_method->context = NULL;
742
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100743 text_backend->input_method.binding = NULL;
744
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200745 free(resource);
746}
747
748static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200749bind_input_method(struct wl_client *client,
750 void *data,
751 uint32_t version,
752 uint32_t id)
753{
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200754 struct input_method *input_method = data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100755 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200756 struct wl_resource *resource;
757
758 resource = wl_client_add_object(client, &input_method_interface,
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200759 NULL,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200760 id, input_method);
761
762 if (input_method->input_method_binding == NULL) {
763 resource->destroy = unbind_input_method;
764 input_method->input_method_binding = resource;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100765
766 text_backend->input_method.binding = resource;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200767 return;
768 }
769
770 wl_resource_post_error(resource, WL_DISPLAY_ERROR_INVALID_OBJECT,
771 "interface object already bound");
772 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200773}
774
775static void
776input_method_notifier_destroy(struct wl_listener *listener, void *data)
777{
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400778 struct input_method *input_method =
779 container_of(listener, struct input_method, destroy_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200780
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200781 if (input_method->model)
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200782 deactivate_text_input(input_method->model, input_method);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200783
784 wl_display_remove_global(input_method->seat->compositor->wl_display,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200785 input_method->input_method_global);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200786
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200787 free(input_method);
788}
789
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200790static void
791handle_keyboard_focus(struct wl_listener *listener, void *data)
792{
793 struct wl_keyboard *keyboard = data;
794 struct input_method *input_method =
795 container_of(listener, struct input_method, keyboard_focus_listener);
796 struct wl_surface *surface = keyboard->focus;
797
798 if (!input_method->model)
799 return;
800
801 if (!surface || input_method->model->surface != surface)
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200802 deactivate_text_input(input_method->model,
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200803 input_method);
804}
805
806static void
807input_method_init_seat(struct weston_seat *seat)
808{
809 if (seat->input_method->focus_listener_initialized)
810 return;
811
812 if (seat->has_keyboard) {
813 seat->input_method->keyboard_focus_listener.notify = handle_keyboard_focus;
814 wl_signal_add(&seat->seat.keyboard->focus_signal, &seat->input_method->keyboard_focus_listener);
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100815 seat->keyboard.input_method_grab.interface = &input_method_context_grab;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200816 }
817
818 seat->input_method->focus_listener_initialized = 1;
819}
820
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100821static void
822handle_input_method_sigchld(struct weston_process *process, int status)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200823{
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100824 struct text_backend *text_backend =
825 container_of(process, struct text_backend, input_method.process);
826
827 text_backend->input_method.process.pid = 0;
828 text_backend->input_method.client = NULL;
829}
830
831static void
832launch_input_method(struct text_backend *text_backend)
833{
834 if (text_backend->input_method.binding)
835 return;
836
837 if (!text_backend->input_method.path)
838 return;
839
840 if (text_backend->input_method.process.pid != 0)
841 return;
842
843 text_backend->input_method.client = weston_client_launch(text_backend->compositor,
844 &text_backend->input_method.process,
845 text_backend->input_method.path,
846 handle_input_method_sigchld);
847
848 if (!text_backend->input_method.client)
849 weston_log("not able to start %s\n", text_backend->input_method.path);
850}
851
852static void
853handle_seat_created(struct wl_listener *listener,
854 void *data)
855{
856 struct weston_seat *seat = data;
857 struct text_backend *text_backend =
858 container_of(listener, struct text_backend,
859 seat_created_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200860 struct input_method *input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100861 struct weston_compositor *ec = seat->compositor;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200862
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200863 input_method = calloc(1, sizeof *input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200864
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200865 input_method->seat = seat;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200866 input_method->model = NULL;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200867 input_method->focus_listener_initialized = 0;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200868 input_method->context = NULL;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100869 input_method->text_backend = text_backend;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200870
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200871 input_method->input_method_global =
872 wl_display_add_global(ec->wl_display,
873 &input_method_interface,
874 input_method, bind_input_method);
875
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200876 input_method->destroy_listener.notify = input_method_notifier_destroy;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200877 wl_signal_add(&seat->seat.destroy_signal, &input_method->destroy_listener);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200878
879 seat->input_method = input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100880
881 launch_input_method(text_backend);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200882}
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200883
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100884static void
885text_backend_configuration(struct text_backend *text_backend)
886{
887 char *config_file;
888 char *path = NULL;
889
890 struct config_key input_method_keys[] = {
891 { "path", CONFIG_KEY_STRING, &path }
892 };
893
894 struct config_section cs[] = {
895 { "input-method", input_method_keys, ARRAY_LENGTH(input_method_keys), NULL }
896 };
897
898 config_file = config_file_path("weston.ini");
899 parse_config_file(config_file, cs, ARRAY_LENGTH(cs), text_backend);
900 free(config_file);
901
902 if (path)
903 text_backend->input_method.path = path;
904 else
905 text_backend->input_method.path = strdup(LIBEXECDIR "/weston-keyboard");
906}
907
908static void
909text_backend_notifier_destroy(struct wl_listener *listener, void *data)
910{
911 struct text_backend *text_backend =
912 container_of(listener, struct text_backend, destroy_listener);
913
914 if (text_backend->input_method.client)
915 wl_client_destroy(text_backend->input_method.client);
916
917 free(text_backend->input_method.path);
918
919 free(text_backend);
920}
921
922
923WL_EXPORT int
924text_backend_init(struct weston_compositor *ec)
925{
926 struct text_backend *text_backend;
927
928 text_backend = calloc(1, sizeof(*text_backend));
929
930 text_backend->compositor = ec;
931
932 text_backend->seat_created_listener.notify = handle_seat_created;
933 wl_signal_add(&ec->seat_created_signal,
934 &text_backend->seat_created_listener);
935
936 text_backend->destroy_listener.notify = text_backend_notifier_destroy;
937 wl_signal_add(&ec->destroy_signal, &text_backend->destroy_listener);
938
939 text_backend_configuration(text_backend);
940
Jan Arne Petersen78d00e42013-04-18 16:47:24 +0200941 text_input_manager_create(ec);
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100942
943 return 0;
944}