blob: 7f8c16e03e1e545a290eb348d2cbe2785d4c1478 [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
35struct text_model {
36 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 Petersen1f17be42012-06-21 21:52:18 +020043};
44
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +020045struct text_model_factory {
Jan Arne Petersen51963742012-08-10 16:47:20 +020046 struct wl_global *text_model_factory_global;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020047 struct wl_listener destroy_listener;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020048
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +020049 struct weston_compositor *ec;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +020050};
51
52struct input_method {
53 struct wl_resource *input_method_binding;
54 struct wl_global *input_method_global;
55 struct wl_listener destroy_listener;
56
57 struct weston_seat *seat;
Jan Arne Petersene829adc2012-08-10 16:47:22 +020058 struct text_model *model;
59
60 struct wl_list link;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +020061
62 struct wl_listener keyboard_focus_listener;
63
64 int focus_listener_initialized;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020065
66 struct input_method_context *context;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010067
68 struct text_backend *text_backend;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020069};
70
Jan Arne Petersen620cd622012-09-09 23:08:32 +020071struct input_method_context {
72 struct wl_resource resource;
73
74 struct text_model *model;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +010075 struct input_method *input_method;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020076
77 struct wl_list link;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +010078
79 struct wl_resource *keyboard;
80 struct wl_keyboard_grab grab;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020081};
82
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010083struct text_backend {
84 struct weston_compositor *compositor;
85
86 struct {
87 char *path;
88 struct wl_resource *binding;
89 struct weston_process process;
90 struct wl_client *client;
91 } input_method;
92
93 struct wl_listener seat_created_listener;
94 struct wl_listener destroy_listener;
95};
96
Jan Arne Petersen620cd622012-09-09 23:08:32 +020097static void input_method_context_create(struct text_model *model,
98 struct input_method *input_method);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +010099static void input_method_context_end_keyboard_grab(struct input_method_context *context);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200100
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200101static void input_method_init_seat(struct weston_seat *seat);
102
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200103static void
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200104deactivate_text_model(struct text_model *text_model,
105 struct input_method *input_method)
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200106{
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200107 struct weston_compositor *ec = text_model->ec;
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200108
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200109 if (input_method->model == text_model) {
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100110 if (input_method->context && input_method->input_method_binding) {
111 input_method_context_end_keyboard_grab(input_method->context);
112 input_method_send_deactivate(input_method->input_method_binding,
113 &input_method->context->resource);
114 }
115
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200116 wl_list_remove(&input_method->link);
117 input_method->model = NULL;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200118 input_method->context = NULL;
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400119 wl_signal_emit(&ec->hide_input_panel_signal, ec);
Jan Arne Petersen680275f2012-09-24 14:51:14 +0200120 text_model_send_leave(&text_model->resource);
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200121 }
122}
123
124static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200125destroy_text_model(struct wl_resource *resource)
126{
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400127 struct text_model *text_model =
128 container_of(resource, struct text_model, resource);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200129 struct input_method *input_method, *next;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200130
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200131 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link)
132 deactivate_text_model(text_model, input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200133
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200134 free(text_model);
135}
136
137static void
138text_model_set_surrounding_text(struct wl_client *client,
139 struct wl_resource *resource,
Jan Arne Petersencb08f4d2012-09-09 23:08:40 +0200140 const char *text,
141 uint32_t cursor,
142 uint32_t anchor)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200143{
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200144 struct text_model *text_model = resource->data;
145 struct input_method *input_method, *next;
146
147 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link) {
148 if (!input_method->context)
149 continue;
Jan Arne Petersencb08f4d2012-09-09 23:08:40 +0200150 input_method_context_send_surrounding_text(&input_method->context->resource,
151 text,
152 cursor,
153 anchor);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200154 }
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200155}
156
157static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200158text_model_activate(struct wl_client *client,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200159 struct wl_resource *resource,
160 struct wl_resource *seat,
161 struct wl_resource *surface)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200162{
163 struct text_model *text_model = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200164 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200165 struct input_method *input_method = weston_seat->input_method;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200166 struct text_model *old = weston_seat->input_method->model;
167 struct weston_compositor *ec = text_model->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200168
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200169 if (old == text_model)
170 return;
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200171
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200172 if (old) {
173 deactivate_text_model(old,
174 weston_seat->input_method);
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200175 }
176
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200177 input_method->model = text_model;
178 wl_list_insert(&text_model->input_methods, &input_method->link);
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200179 input_method_init_seat(weston_seat);
180
181 text_model->surface = surface->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200182
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200183 input_method_context_create(text_model, input_method);
184
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400185 wl_signal_emit(&ec->show_input_panel_signal, ec);
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200186
Jan Arne Petersen680275f2012-09-24 14:51:14 +0200187 text_model_send_enter(&text_model->resource, &text_model->surface->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200188}
189
190static void
191text_model_deactivate(struct wl_client *client,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200192 struct wl_resource *resource,
193 struct wl_resource *seat)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200194{
195 struct text_model *text_model = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200196 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200197
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200198 deactivate_text_model(text_model,
199 weston_seat->input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200200}
201
202static void
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200203text_model_reset(struct wl_client *client,
204 struct wl_resource *resource)
205{
206 struct text_model *text_model = resource->data;
207 struct input_method *input_method, *next;
208
209 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link) {
210 if (!input_method->context)
211 continue;
212 input_method_context_send_reset(&input_method->context->resource);
213 }
214}
215
216static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200217text_model_set_micro_focus(struct wl_client *client,
218 struct wl_resource *resource,
219 int32_t x,
220 int32_t y,
221 int32_t width,
222 int32_t height)
223{
224}
225
226static void
227text_model_set_preedit(struct wl_client *client,
228 struct wl_resource *resource)
229{
230}
231
232static void
233text_model_set_content_type(struct wl_client *client,
234 struct wl_resource *resource)
235{
236}
237
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200238static const struct text_model_interface text_model_implementation = {
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200239 text_model_set_surrounding_text,
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200240 text_model_activate,
241 text_model_deactivate,
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200242 text_model_reset,
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200243 text_model_set_micro_focus,
244 text_model_set_preedit,
245 text_model_set_content_type
246};
247
Jan Arne Petersen51963742012-08-10 16:47:20 +0200248static void text_model_factory_create_text_model(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200249 struct wl_resource *resource,
Jan Arne Petersen4c265182012-09-09 23:08:30 +0200250 uint32_t id)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200251{
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200252 struct text_model_factory *text_model_factory = resource->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200253 struct text_model *text_model;
254
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200255 text_model = calloc(1, sizeof *text_model);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200256
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200257 text_model->resource.object.id = id;
258 text_model->resource.object.interface = &text_model_interface;
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200259 text_model->resource.object.implementation =
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200260 (void (**)(void)) &text_model_implementation;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200261
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200262 text_model->resource.data = text_model;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200263 text_model->resource.destroy = destroy_text_model;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200264
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200265 text_model->ec = text_model_factory->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200266
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200267 wl_list_init(&text_model->input_methods);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200268
269 wl_client_add_resource(client, &text_model->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200270};
271
Jan Arne Petersen51963742012-08-10 16:47:20 +0200272static const struct text_model_factory_interface text_model_factory_implementation = {
273 text_model_factory_create_text_model
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200274};
275
276static void
Jan Arne Petersen51963742012-08-10 16:47:20 +0200277bind_text_model_factory(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200278 void *data,
279 uint32_t version,
280 uint32_t id)
281{
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200282 struct text_model_factory *text_model_factory = data;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200283
284 /* No checking for duplicate binding necessary.
285 * No events have to be sent, so we don't need the return value. */
Jan Arne Petersen51963742012-08-10 16:47:20 +0200286 wl_client_add_object(client, &text_model_factory_interface,
287 &text_model_factory_implementation,
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200288 id, text_model_factory);
289}
290
291static void
292text_model_factory_notifier_destroy(struct wl_listener *listener, void *data)
293{
294 struct text_model_factory *text_model_factory =
295 container_of(listener, struct text_model_factory, destroy_listener);
296
297 wl_display_remove_global(text_model_factory->ec->wl_display,
298 text_model_factory->text_model_factory_global);
299
300 free(text_model_factory);
301}
302
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100303static void
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200304text_model_factory_create(struct weston_compositor *ec)
305{
306 struct text_model_factory *text_model_factory;
307
308 text_model_factory = calloc(1, sizeof *text_model_factory);
309
310 text_model_factory->ec = ec;
311
312 text_model_factory->text_model_factory_global =
313 wl_display_add_global(ec->wl_display,
314 &text_model_factory_interface,
315 text_model_factory, bind_text_model_factory);
316
317 text_model_factory->destroy_listener.notify = text_model_factory_notifier_destroy;
318 wl_signal_add(&ec->destroy_signal, &text_model_factory->destroy_listener);
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200319}
320
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200321static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200322input_method_context_destroy(struct wl_client *client,
323 struct wl_resource *resource)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200324{
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200325 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200326}
327
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200328static void
329input_method_context_commit_string(struct wl_client *client,
330 struct wl_resource *resource,
331 const char *text,
332 uint32_t index)
333{
334 struct input_method_context *context = resource->data;
335
336 text_model_send_commit_string(&context->model->resource, text, index);
337}
338
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200339static void
340input_method_context_preedit_string(struct wl_client *client,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100341 struct wl_resource *resource,
342 const char *text,
343 const char *commit)
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200344{
345 struct input_method_context *context = resource->data;
346
Jan Arne Petersen46535312013-01-16 21:26:38 +0100347 text_model_send_preedit_string(&context->model->resource, text, commit);
348}
349
350static void
351input_method_context_preedit_styling(struct wl_client *client,
352 struct wl_resource *resource,
353 uint32_t index,
354 uint32_t length,
355 uint32_t style)
356{
357 struct input_method_context *context = resource->data;
358
359 text_model_send_preedit_styling(&context->model->resource, index, length, style);
360}
361
362static void
363input_method_context_preedit_cursor(struct wl_client *client,
364 struct wl_resource *resource,
365 int32_t cursor)
366{
367 struct input_method_context *context = resource->data;
368
369 text_model_send_preedit_cursor(&context->model->resource, cursor);
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200370}
371
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200372static void
373input_method_context_delete_surrounding_text(struct wl_client *client,
374 struct wl_resource *resource,
375 int32_t index,
376 uint32_t length)
377{
378 struct input_method_context *context = resource->data;
379
380 text_model_send_delete_surrounding_text(&context->model->resource, index, length);
381}
382
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200383static void
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100384input_method_context_modifiers_map(struct wl_client *client,
385 struct wl_resource *resource,
386 struct wl_array *map)
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200387{
388 struct input_method_context *context = resource->data;
389
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100390 text_model_send_modifiers_map(&context->model->resource, map);
391}
392
393static void
394input_method_context_keysym(struct wl_client *client,
395 struct wl_resource *resource,
396 uint32_t serial,
397 uint32_t time,
398 uint32_t sym,
399 uint32_t state,
400 uint32_t modifiers)
401{
402 struct input_method_context *context = resource->data;
403
404 text_model_send_keysym(&context->model->resource, serial, time,
405 sym, state, modifiers);
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200406}
407
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100408static void
409unbind_keyboard(struct wl_resource *resource)
410{
411 struct input_method_context *context = resource->data;
412
413 input_method_context_end_keyboard_grab(context);
414 context->keyboard = NULL;
415
416 free(resource);
417}
418
419static void
420input_method_context_grab_key(struct wl_keyboard_grab *grab,
421 uint32_t time, uint32_t key, uint32_t state_w)
422{
423 struct input_method_context *input_method_context = container_of(grab, struct input_method_context, grab);
424 uint32_t serial;
425
426 if (input_method_context->keyboard) {
427 serial = wl_display_next_serial(input_method_context->model->ec->wl_display);
428 wl_keyboard_send_key(input_method_context->keyboard,
429 serial, time, key, state_w);
430 }
431}
432
433static void
434input_method_context_grab_modifier(struct wl_keyboard_grab *grab, uint32_t serial,
435 uint32_t mods_depressed, uint32_t mods_latched,
436 uint32_t mods_locked, uint32_t group)
437{
438 struct input_method_context *input_method_context = container_of(grab, struct input_method_context, grab);
439
440 if (!input_method_context->keyboard)
441 return;
442
443 wl_keyboard_send_modifiers(input_method_context->keyboard,
444 serial, mods_depressed, mods_latched,
445 mods_locked, group);
446}
447
448static const struct wl_keyboard_grab_interface input_method_context_grab = {
449 input_method_context_grab_key,
450 input_method_context_grab_modifier,
451};
452
453static void
454input_method_context_grab_keyboard(struct wl_client *client,
455 struct wl_resource *resource,
456 uint32_t id)
457{
458 struct input_method_context *context = resource->data;
459 struct wl_resource *cr;
460 struct weston_seat *seat = context->input_method->seat;
461 struct wl_keyboard *keyboard = seat->seat.keyboard;
462
463 cr = wl_client_add_object(client, &wl_keyboard_interface,
464 NULL, id, context);
465 cr->destroy = unbind_keyboard;
466
467 context->keyboard = cr;
468
469 wl_keyboard_send_keymap(cr, WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1,
470 seat->xkb_info.keymap_fd,
471 seat->xkb_info.keymap_size);
472
473 if (keyboard->grab != &keyboard->default_grab) {
474 wl_keyboard_end_grab(keyboard);
475 }
476 wl_keyboard_start_grab(keyboard, &context->grab);
477}
478
Jan Arne Petersen337df952012-11-18 19:06:46 +0100479static void
480input_method_context_key(struct wl_client *client,
481 struct wl_resource *resource,
482 uint32_t serial,
483 uint32_t time,
484 uint32_t key,
485 uint32_t state_w)
486{
487 struct input_method_context *context = resource->data;
488 struct weston_seat *seat = context->input_method->seat;
489 struct wl_keyboard *keyboard = seat->seat.keyboard;
490 struct wl_keyboard_grab *default_grab = &keyboard->default_grab;
491
492 default_grab->interface->key(default_grab, time, key, state_w);
493}
494
495static void
496input_method_context_modifiers(struct wl_client *client,
497 struct wl_resource *resource,
498 uint32_t serial,
499 uint32_t mods_depressed,
500 uint32_t mods_latched,
501 uint32_t mods_locked,
502 uint32_t group)
503{
504 struct input_method_context *context = resource->data;
505
506 struct weston_seat *seat = context->input_method->seat;
507 struct wl_keyboard *keyboard = seat->seat.keyboard;
508 struct wl_keyboard_grab *default_grab = &keyboard->default_grab;
509
510 default_grab->interface->modifiers(default_grab,
511 serial, mods_depressed,
512 mods_latched, mods_locked,
513 group);
514}
515
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200516static const struct input_method_context_interface input_method_context_implementation = {
517 input_method_context_destroy,
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200518 input_method_context_commit_string,
519 input_method_context_preedit_string,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100520 input_method_context_preedit_styling,
521 input_method_context_preedit_cursor,
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200522 input_method_context_delete_surrounding_text,
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100523 input_method_context_modifiers_map,
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100524 input_method_context_keysym,
Jan Arne Petersen337df952012-11-18 19:06:46 +0100525 input_method_context_grab_keyboard,
526 input_method_context_key,
527 input_method_context_modifiers
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200528};
529
530static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200531destroy_input_method_context(struct wl_resource *resource)
532{
533 struct input_method_context *context = resource->data;
534
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100535 if (context->keyboard) {
536 wl_resource_destroy(context->keyboard);
537 }
538
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200539 free(context);
540}
541
542static void
543input_method_context_create(struct text_model *model,
544 struct input_method *input_method)
545{
546 struct input_method_context *context;
547
548 if (!input_method->input_method_binding)
549 return;
550
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100551 context = calloc(1, sizeof *context);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200552 if (context == NULL)
553 return;
554
555 context->resource.destroy = destroy_input_method_context;
556 context->resource.object.id = 0;
557 context->resource.object.interface = &input_method_context_interface;
558 context->resource.object.implementation =
559 (void (**)(void)) &input_method_context_implementation;
560 context->resource.data = context;
561 wl_signal_init(&context->resource.destroy_signal);
562
563 context->model = model;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100564 context->input_method = input_method;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200565 input_method->context = context;
566
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100567 context->grab.interface = &input_method_context_grab;
568
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200569 wl_client_add_resource(input_method->input_method_binding->client, &context->resource);
570
571 input_method_send_activate(input_method->input_method_binding, &context->resource);
572}
573
574static void
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100575input_method_context_end_keyboard_grab(struct input_method_context *context)
576{
577 struct wl_keyboard_grab *grab = &context->grab;
578
579 if (grab->keyboard && (grab->keyboard->grab == grab)) {
580 wl_keyboard_end_grab(grab->keyboard);
581 }
582}
583
584
585static void
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200586unbind_input_method(struct wl_resource *resource)
587{
588 struct input_method *input_method = resource->data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100589 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200590
591 input_method->input_method_binding = NULL;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200592 input_method->context = NULL;
593
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100594 text_backend->input_method.binding = NULL;
595
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200596 free(resource);
597}
598
599static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200600bind_input_method(struct wl_client *client,
601 void *data,
602 uint32_t version,
603 uint32_t id)
604{
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200605 struct input_method *input_method = data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100606 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200607 struct wl_resource *resource;
608
609 resource = wl_client_add_object(client, &input_method_interface,
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200610 NULL,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200611 id, input_method);
612
613 if (input_method->input_method_binding == NULL) {
614 resource->destroy = unbind_input_method;
615 input_method->input_method_binding = resource;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100616
617 text_backend->input_method.binding = resource;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200618 return;
619 }
620
621 wl_resource_post_error(resource, WL_DISPLAY_ERROR_INVALID_OBJECT,
622 "interface object already bound");
623 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200624}
625
626static void
627input_method_notifier_destroy(struct wl_listener *listener, void *data)
628{
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400629 struct input_method *input_method =
630 container_of(listener, struct input_method, destroy_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200631
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200632 if (input_method->model)
633 deactivate_text_model(input_method->model, input_method);
634
635 wl_display_remove_global(input_method->seat->compositor->wl_display,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200636 input_method->input_method_global);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200637
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200638 free(input_method);
639}
640
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200641static void
642handle_keyboard_focus(struct wl_listener *listener, void *data)
643{
644 struct wl_keyboard *keyboard = data;
645 struct input_method *input_method =
646 container_of(listener, struct input_method, keyboard_focus_listener);
647 struct wl_surface *surface = keyboard->focus;
648
649 if (!input_method->model)
650 return;
651
652 if (!surface || input_method->model->surface != surface)
653 deactivate_text_model(input_method->model,
654 input_method);
655}
656
657static void
658input_method_init_seat(struct weston_seat *seat)
659{
660 if (seat->input_method->focus_listener_initialized)
661 return;
662
663 if (seat->has_keyboard) {
664 seat->input_method->keyboard_focus_listener.notify = handle_keyboard_focus;
665 wl_signal_add(&seat->seat.keyboard->focus_signal, &seat->input_method->keyboard_focus_listener);
666 }
667
668 seat->input_method->focus_listener_initialized = 1;
669}
670
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100671static void
672handle_input_method_sigchld(struct weston_process *process, int status)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200673{
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100674 struct text_backend *text_backend =
675 container_of(process, struct text_backend, input_method.process);
676
677 text_backend->input_method.process.pid = 0;
678 text_backend->input_method.client = NULL;
679}
680
681static void
682launch_input_method(struct text_backend *text_backend)
683{
684 if (text_backend->input_method.binding)
685 return;
686
687 if (!text_backend->input_method.path)
688 return;
689
690 if (text_backend->input_method.process.pid != 0)
691 return;
692
693 text_backend->input_method.client = weston_client_launch(text_backend->compositor,
694 &text_backend->input_method.process,
695 text_backend->input_method.path,
696 handle_input_method_sigchld);
697
698 if (!text_backend->input_method.client)
699 weston_log("not able to start %s\n", text_backend->input_method.path);
700}
701
702static void
703handle_seat_created(struct wl_listener *listener,
704 void *data)
705{
706 struct weston_seat *seat = data;
707 struct text_backend *text_backend =
708 container_of(listener, struct text_backend,
709 seat_created_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200710 struct input_method *input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100711 struct weston_compositor *ec = seat->compositor;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200712
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200713 input_method = calloc(1, sizeof *input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200714
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200715 input_method->seat = seat;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200716 input_method->model = NULL;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200717 input_method->focus_listener_initialized = 0;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200718 input_method->context = NULL;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100719 input_method->text_backend = text_backend;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200720
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200721 input_method->input_method_global =
722 wl_display_add_global(ec->wl_display,
723 &input_method_interface,
724 input_method, bind_input_method);
725
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200726 input_method->destroy_listener.notify = input_method_notifier_destroy;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200727 wl_signal_add(&seat->seat.destroy_signal, &input_method->destroy_listener);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200728
729 seat->input_method = input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100730
731 launch_input_method(text_backend);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200732}
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200733
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100734static void
735text_backend_configuration(struct text_backend *text_backend)
736{
737 char *config_file;
738 char *path = NULL;
739
740 struct config_key input_method_keys[] = {
741 { "path", CONFIG_KEY_STRING, &path }
742 };
743
744 struct config_section cs[] = {
745 { "input-method", input_method_keys, ARRAY_LENGTH(input_method_keys), NULL }
746 };
747
748 config_file = config_file_path("weston.ini");
749 parse_config_file(config_file, cs, ARRAY_LENGTH(cs), text_backend);
750 free(config_file);
751
752 if (path)
753 text_backend->input_method.path = path;
754 else
755 text_backend->input_method.path = strdup(LIBEXECDIR "/weston-keyboard");
756}
757
758static void
759text_backend_notifier_destroy(struct wl_listener *listener, void *data)
760{
761 struct text_backend *text_backend =
762 container_of(listener, struct text_backend, destroy_listener);
763
764 if (text_backend->input_method.client)
765 wl_client_destroy(text_backend->input_method.client);
766
767 free(text_backend->input_method.path);
768
769 free(text_backend);
770}
771
772
773WL_EXPORT int
774text_backend_init(struct weston_compositor *ec)
775{
776 struct text_backend *text_backend;
777
778 text_backend = calloc(1, sizeof(*text_backend));
779
780 text_backend->compositor = ec;
781
782 text_backend->seat_created_listener.notify = handle_seat_created;
783 wl_signal_add(&ec->seat_created_signal,
784 &text_backend->seat_created_listener);
785
786 text_backend->destroy_listener.notify = text_backend_notifier_destroy;
787 wl_signal_add(&ec->destroy_signal, &text_backend->destroy_listener);
788
789 text_backend_configuration(text_backend);
790
791 text_model_factory_create(ec);
792
793 return 0;
794}