blob: ece89bb9772f60bdf8b75287de8ec7965362d38e [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;
75
76 struct wl_list link;
77};
78
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010079struct text_backend {
80 struct weston_compositor *compositor;
81
82 struct {
83 char *path;
84 struct wl_resource *binding;
85 struct weston_process process;
86 struct wl_client *client;
87 } input_method;
88
89 struct wl_listener seat_created_listener;
90 struct wl_listener destroy_listener;
91};
92
Jan Arne Petersen620cd622012-09-09 23:08:32 +020093static void input_method_context_create(struct text_model *model,
94 struct input_method *input_method);
95
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +020096static void input_method_init_seat(struct weston_seat *seat);
97
Jan Arne Petersen1f17be42012-06-21 21:52:18 +020098static void
Jan Arne Petersene829adc2012-08-10 16:47:22 +020099deactivate_text_model(struct text_model *text_model,
100 struct input_method *input_method)
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200101{
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200102 struct weston_compositor *ec = text_model->ec;
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200103
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200104 if (input_method->model == text_model) {
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100105 if (input_method->context && input_method->input_method_binding)
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200106 input_method_send_deactivate(input_method->input_method_binding, &input_method->context->resource);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200107 wl_list_remove(&input_method->link);
108 input_method->model = NULL;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200109 input_method->context = NULL;
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400110 wl_signal_emit(&ec->hide_input_panel_signal, ec);
Jan Arne Petersen680275f2012-09-24 14:51:14 +0200111 text_model_send_leave(&text_model->resource);
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200112 }
113}
114
115static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200116destroy_text_model(struct wl_resource *resource)
117{
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400118 struct text_model *text_model =
119 container_of(resource, struct text_model, resource);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200120 struct input_method *input_method, *next;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200121
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200122 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link)
123 deactivate_text_model(text_model, input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200124
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200125 free(text_model);
126}
127
128static void
129text_model_set_surrounding_text(struct wl_client *client,
130 struct wl_resource *resource,
Jan Arne Petersencb08f4d2012-09-09 23:08:40 +0200131 const char *text,
132 uint32_t cursor,
133 uint32_t anchor)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200134{
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200135 struct text_model *text_model = resource->data;
136 struct input_method *input_method, *next;
137
138 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link) {
139 if (!input_method->context)
140 continue;
Jan Arne Petersencb08f4d2012-09-09 23:08:40 +0200141 input_method_context_send_surrounding_text(&input_method->context->resource,
142 text,
143 cursor,
144 anchor);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200145 }
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200146}
147
148static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200149text_model_activate(struct wl_client *client,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200150 struct wl_resource *resource,
151 struct wl_resource *seat,
152 struct wl_resource *surface)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200153{
154 struct text_model *text_model = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200155 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200156 struct input_method *input_method = weston_seat->input_method;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200157 struct text_model *old = weston_seat->input_method->model;
158 struct weston_compositor *ec = text_model->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200159
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200160 if (old == text_model)
161 return;
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200162
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200163 if (old) {
164 deactivate_text_model(old,
165 weston_seat->input_method);
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200166 }
167
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200168 input_method->model = text_model;
169 wl_list_insert(&text_model->input_methods, &input_method->link);
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200170 input_method_init_seat(weston_seat);
171
172 text_model->surface = surface->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200173
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200174 input_method_context_create(text_model, input_method);
175
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400176 wl_signal_emit(&ec->show_input_panel_signal, ec);
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200177
Jan Arne Petersen680275f2012-09-24 14:51:14 +0200178 text_model_send_enter(&text_model->resource, &text_model->surface->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200179}
180
181static void
182text_model_deactivate(struct wl_client *client,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200183 struct wl_resource *resource,
184 struct wl_resource *seat)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200185{
186 struct text_model *text_model = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200187 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200188
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200189 deactivate_text_model(text_model,
190 weston_seat->input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200191}
192
193static void
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200194text_model_reset(struct wl_client *client,
195 struct wl_resource *resource)
196{
197 struct text_model *text_model = resource->data;
198 struct input_method *input_method, *next;
199
200 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link) {
201 if (!input_method->context)
202 continue;
203 input_method_context_send_reset(&input_method->context->resource);
204 }
205}
206
207static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200208text_model_set_micro_focus(struct wl_client *client,
209 struct wl_resource *resource,
210 int32_t x,
211 int32_t y,
212 int32_t width,
213 int32_t height)
214{
215}
216
217static void
218text_model_set_preedit(struct wl_client *client,
219 struct wl_resource *resource)
220{
221}
222
223static void
224text_model_set_content_type(struct wl_client *client,
225 struct wl_resource *resource)
226{
227}
228
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200229static const struct text_model_interface text_model_implementation = {
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200230 text_model_set_surrounding_text,
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200231 text_model_activate,
232 text_model_deactivate,
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200233 text_model_reset,
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200234 text_model_set_micro_focus,
235 text_model_set_preedit,
236 text_model_set_content_type
237};
238
Jan Arne Petersen51963742012-08-10 16:47:20 +0200239static void text_model_factory_create_text_model(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200240 struct wl_resource *resource,
Jan Arne Petersen4c265182012-09-09 23:08:30 +0200241 uint32_t id)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200242{
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200243 struct text_model_factory *text_model_factory = resource->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200244 struct text_model *text_model;
245
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200246 text_model = calloc(1, sizeof *text_model);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200247
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200248 text_model->resource.object.id = id;
249 text_model->resource.object.interface = &text_model_interface;
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200250 text_model->resource.object.implementation =
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200251 (void (**)(void)) &text_model_implementation;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200252
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200253 text_model->resource.data = text_model;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200254 text_model->resource.destroy = destroy_text_model;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200255
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200256 text_model->ec = text_model_factory->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200257
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200258 wl_list_init(&text_model->input_methods);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200259
260 wl_client_add_resource(client, &text_model->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200261};
262
Jan Arne Petersen51963742012-08-10 16:47:20 +0200263static const struct text_model_factory_interface text_model_factory_implementation = {
264 text_model_factory_create_text_model
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200265};
266
267static void
Jan Arne Petersen51963742012-08-10 16:47:20 +0200268bind_text_model_factory(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200269 void *data,
270 uint32_t version,
271 uint32_t id)
272{
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200273 struct text_model_factory *text_model_factory = data;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200274
275 /* No checking for duplicate binding necessary.
276 * No events have to be sent, so we don't need the return value. */
Jan Arne Petersen51963742012-08-10 16:47:20 +0200277 wl_client_add_object(client, &text_model_factory_interface,
278 &text_model_factory_implementation,
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200279 id, text_model_factory);
280}
281
282static void
283text_model_factory_notifier_destroy(struct wl_listener *listener, void *data)
284{
285 struct text_model_factory *text_model_factory =
286 container_of(listener, struct text_model_factory, destroy_listener);
287
288 wl_display_remove_global(text_model_factory->ec->wl_display,
289 text_model_factory->text_model_factory_global);
290
291 free(text_model_factory);
292}
293
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100294static void
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200295text_model_factory_create(struct weston_compositor *ec)
296{
297 struct text_model_factory *text_model_factory;
298
299 text_model_factory = calloc(1, sizeof *text_model_factory);
300
301 text_model_factory->ec = ec;
302
303 text_model_factory->text_model_factory_global =
304 wl_display_add_global(ec->wl_display,
305 &text_model_factory_interface,
306 text_model_factory, bind_text_model_factory);
307
308 text_model_factory->destroy_listener.notify = text_model_factory_notifier_destroy;
309 wl_signal_add(&ec->destroy_signal, &text_model_factory->destroy_listener);
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200310}
311
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200312static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200313input_method_context_destroy(struct wl_client *client,
314 struct wl_resource *resource)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200315{
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200316 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200317}
318
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200319static void
320input_method_context_commit_string(struct wl_client *client,
321 struct wl_resource *resource,
322 const char *text,
323 uint32_t index)
324{
325 struct input_method_context *context = resource->data;
326
327 text_model_send_commit_string(&context->model->resource, text, index);
328}
329
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200330static void
331input_method_context_preedit_string(struct wl_client *client,
332 struct wl_resource *resource,
333 const char *text,
334 uint32_t index)
335{
336 struct input_method_context *context = resource->data;
337
338 text_model_send_preedit_string(&context->model->resource, text, index);
339}
340
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200341static void
342input_method_context_delete_surrounding_text(struct wl_client *client,
343 struct wl_resource *resource,
344 int32_t index,
345 uint32_t length)
346{
347 struct input_method_context *context = resource->data;
348
349 text_model_send_delete_surrounding_text(&context->model->resource, index, length);
350}
351
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200352static void
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100353input_method_context_modifiers_map(struct wl_client *client,
354 struct wl_resource *resource,
355 struct wl_array *map)
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200356{
357 struct input_method_context *context = resource->data;
358
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100359 text_model_send_modifiers_map(&context->model->resource, map);
360}
361
362static void
363input_method_context_keysym(struct wl_client *client,
364 struct wl_resource *resource,
365 uint32_t serial,
366 uint32_t time,
367 uint32_t sym,
368 uint32_t state,
369 uint32_t modifiers)
370{
371 struct input_method_context *context = resource->data;
372
373 text_model_send_keysym(&context->model->resource, serial, time,
374 sym, state, modifiers);
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200375}
376
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200377static const struct input_method_context_interface input_method_context_implementation = {
378 input_method_context_destroy,
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200379 input_method_context_commit_string,
380 input_method_context_preedit_string,
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200381 input_method_context_delete_surrounding_text,
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100382 input_method_context_modifiers_map,
383 input_method_context_keysym
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200384};
385
386static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200387destroy_input_method_context(struct wl_resource *resource)
388{
389 struct input_method_context *context = resource->data;
390
391 free(context);
392}
393
394static void
395input_method_context_create(struct text_model *model,
396 struct input_method *input_method)
397{
398 struct input_method_context *context;
399
400 if (!input_method->input_method_binding)
401 return;
402
403 context = malloc(sizeof *context);
404 if (context == NULL)
405 return;
406
407 context->resource.destroy = destroy_input_method_context;
408 context->resource.object.id = 0;
409 context->resource.object.interface = &input_method_context_interface;
410 context->resource.object.implementation =
411 (void (**)(void)) &input_method_context_implementation;
412 context->resource.data = context;
413 wl_signal_init(&context->resource.destroy_signal);
414
415 context->model = model;
416 input_method->context = context;
417
418 wl_client_add_resource(input_method->input_method_binding->client, &context->resource);
419
420 input_method_send_activate(input_method->input_method_binding, &context->resource);
421}
422
423static void
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200424unbind_input_method(struct wl_resource *resource)
425{
426 struct input_method *input_method = resource->data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100427 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200428
429 input_method->input_method_binding = NULL;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200430 input_method->context = NULL;
431
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100432 text_backend->input_method.binding = NULL;
433
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200434 free(resource);
435}
436
437static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200438bind_input_method(struct wl_client *client,
439 void *data,
440 uint32_t version,
441 uint32_t id)
442{
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200443 struct input_method *input_method = data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100444 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200445 struct wl_resource *resource;
446
447 resource = wl_client_add_object(client, &input_method_interface,
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200448 NULL,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200449 id, input_method);
450
451 if (input_method->input_method_binding == NULL) {
452 resource->destroy = unbind_input_method;
453 input_method->input_method_binding = resource;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100454
455 text_backend->input_method.binding = resource;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200456 return;
457 }
458
459 wl_resource_post_error(resource, WL_DISPLAY_ERROR_INVALID_OBJECT,
460 "interface object already bound");
461 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200462}
463
464static void
465input_method_notifier_destroy(struct wl_listener *listener, void *data)
466{
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400467 struct input_method *input_method =
468 container_of(listener, struct input_method, destroy_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200469
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200470 if (input_method->model)
471 deactivate_text_model(input_method->model, input_method);
472
473 wl_display_remove_global(input_method->seat->compositor->wl_display,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200474 input_method->input_method_global);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200475
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200476 free(input_method);
477}
478
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200479static void
480handle_keyboard_focus(struct wl_listener *listener, void *data)
481{
482 struct wl_keyboard *keyboard = data;
483 struct input_method *input_method =
484 container_of(listener, struct input_method, keyboard_focus_listener);
485 struct wl_surface *surface = keyboard->focus;
486
487 if (!input_method->model)
488 return;
489
490 if (!surface || input_method->model->surface != surface)
491 deactivate_text_model(input_method->model,
492 input_method);
493}
494
495static void
496input_method_init_seat(struct weston_seat *seat)
497{
498 if (seat->input_method->focus_listener_initialized)
499 return;
500
501 if (seat->has_keyboard) {
502 seat->input_method->keyboard_focus_listener.notify = handle_keyboard_focus;
503 wl_signal_add(&seat->seat.keyboard->focus_signal, &seat->input_method->keyboard_focus_listener);
504 }
505
506 seat->input_method->focus_listener_initialized = 1;
507}
508
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100509static void
510handle_input_method_sigchld(struct weston_process *process, int status)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200511{
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100512 struct text_backend *text_backend =
513 container_of(process, struct text_backend, input_method.process);
514
515 text_backend->input_method.process.pid = 0;
516 text_backend->input_method.client = NULL;
517}
518
519static void
520launch_input_method(struct text_backend *text_backend)
521{
522 if (text_backend->input_method.binding)
523 return;
524
525 if (!text_backend->input_method.path)
526 return;
527
528 if (text_backend->input_method.process.pid != 0)
529 return;
530
531 text_backend->input_method.client = weston_client_launch(text_backend->compositor,
532 &text_backend->input_method.process,
533 text_backend->input_method.path,
534 handle_input_method_sigchld);
535
536 if (!text_backend->input_method.client)
537 weston_log("not able to start %s\n", text_backend->input_method.path);
538}
539
540static void
541handle_seat_created(struct wl_listener *listener,
542 void *data)
543{
544 struct weston_seat *seat = data;
545 struct text_backend *text_backend =
546 container_of(listener, struct text_backend,
547 seat_created_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200548 struct input_method *input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100549 struct weston_compositor *ec = seat->compositor;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200550
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200551 input_method = calloc(1, sizeof *input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200552
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200553 input_method->seat = seat;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200554 input_method->model = NULL;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200555 input_method->focus_listener_initialized = 0;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200556 input_method->context = NULL;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100557 input_method->text_backend = text_backend;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200558
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200559 input_method->input_method_global =
560 wl_display_add_global(ec->wl_display,
561 &input_method_interface,
562 input_method, bind_input_method);
563
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200564 input_method->destroy_listener.notify = input_method_notifier_destroy;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200565 wl_signal_add(&seat->seat.destroy_signal, &input_method->destroy_listener);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200566
567 seat->input_method = input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100568
569 launch_input_method(text_backend);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200570}
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200571
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100572static void
573text_backend_configuration(struct text_backend *text_backend)
574{
575 char *config_file;
576 char *path = NULL;
577
578 struct config_key input_method_keys[] = {
579 { "path", CONFIG_KEY_STRING, &path }
580 };
581
582 struct config_section cs[] = {
583 { "input-method", input_method_keys, ARRAY_LENGTH(input_method_keys), NULL }
584 };
585
586 config_file = config_file_path("weston.ini");
587 parse_config_file(config_file, cs, ARRAY_LENGTH(cs), text_backend);
588 free(config_file);
589
590 if (path)
591 text_backend->input_method.path = path;
592 else
593 text_backend->input_method.path = strdup(LIBEXECDIR "/weston-keyboard");
594}
595
596static void
597text_backend_notifier_destroy(struct wl_listener *listener, void *data)
598{
599 struct text_backend *text_backend =
600 container_of(listener, struct text_backend, destroy_listener);
601
602 if (text_backend->input_method.client)
603 wl_client_destroy(text_backend->input_method.client);
604
605 free(text_backend->input_method.path);
606
607 free(text_backend);
608}
609
610
611WL_EXPORT int
612text_backend_init(struct weston_compositor *ec)
613{
614 struct text_backend *text_backend;
615
616 text_backend = calloc(1, sizeof(*text_backend));
617
618 text_backend->compositor = ec;
619
620 text_backend->seat_created_listener.notify = handle_seat_created;
621 wl_signal_add(&ec->seat_created_signal,
622 &text_backend->seat_created_listener);
623
624 text_backend->destroy_listener.notify = text_backend_notifier_destroy;
625 wl_signal_add(&ec->destroy_signal, &text_backend->destroy_listener);
626
627 text_backend_configuration(text_backend);
628
629 text_model_factory_create(ec);
630
631 return 0;
632}