blob: c0cddfe2972601c732de0689efe8e33a6cf7ee96 [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;
Jan Arne Petersen620cd622012-09-09 23:08:32 +020080};
81
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +010082struct text_backend {
83 struct weston_compositor *compositor;
84
85 struct {
86 char *path;
87 struct wl_resource *binding;
88 struct weston_process process;
89 struct wl_client *client;
90 } input_method;
91
92 struct wl_listener seat_created_listener;
93 struct wl_listener destroy_listener;
94};
95
Jan Arne Petersen620cd622012-09-09 23:08:32 +020096static void input_method_context_create(struct text_model *model,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +010097 struct input_method *input_method,
98 uint32_t serial);
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,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100160 uint32_t serial,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200161 struct wl_resource *seat,
162 struct wl_resource *surface)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200163{
164 struct text_model *text_model = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200165 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200166 struct input_method *input_method = weston_seat->input_method;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200167 struct text_model *old = weston_seat->input_method->model;
168 struct weston_compositor *ec = text_model->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200169
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200170 if (old == text_model)
171 return;
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200172
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200173 if (old) {
174 deactivate_text_model(old,
175 weston_seat->input_method);
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200176 }
177
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200178 input_method->model = text_model;
179 wl_list_insert(&text_model->input_methods, &input_method->link);
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200180 input_method_init_seat(weston_seat);
181
182 text_model->surface = surface->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200183
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100184 input_method_context_create(text_model, input_method, serial);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200185
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400186 wl_signal_emit(&ec->show_input_panel_signal, ec);
Jan Arne Petersende3b6a12012-08-10 16:47:21 +0200187
Jan Arne Petersen680275f2012-09-24 14:51:14 +0200188 text_model_send_enter(&text_model->resource, &text_model->surface->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200189}
190
191static void
192text_model_deactivate(struct wl_client *client,
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200193 struct wl_resource *resource,
194 struct wl_resource *seat)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200195{
196 struct text_model *text_model = resource->data;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200197 struct weston_seat *weston_seat = seat->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200198
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200199 deactivate_text_model(text_model,
200 weston_seat->input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200201}
202
203static void
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200204text_model_reset(struct wl_client *client,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100205 struct wl_resource *resource,
206 uint32_t serial)
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200207{
208 struct text_model *text_model = resource->data;
209 struct input_method *input_method, *next;
210
211 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link) {
212 if (!input_method->context)
213 continue;
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100214 input_method_context_send_reset(&input_method->context->resource, serial);
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200215 }
216}
217
218static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200219text_model_set_micro_focus(struct wl_client *client,
220 struct wl_resource *resource,
221 int32_t x,
222 int32_t y,
223 int32_t width,
224 int32_t height)
225{
226}
227
228static void
229text_model_set_preedit(struct wl_client *client,
230 struct wl_resource *resource)
231{
232}
233
234static void
235text_model_set_content_type(struct wl_client *client,
Jan Arne Petersen26ffa812013-01-16 21:26:43 +0100236 struct wl_resource *resource,
237 uint32_t hint,
238 uint32_t purpose)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200239{
Jan Arne Petersen26ffa812013-01-16 21:26:43 +0100240 struct text_model *text_model = resource->data;
241 struct input_method *input_method, *next;
242
243 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link) {
244 if (!input_method->context)
245 continue;
246 input_method_context_send_content_type(&input_method->context->resource, hint, purpose);
247 }
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200248}
249
Jan Arne Petersenadfedc12013-01-16 21:26:46 +0100250static void
251text_model_invoke_action(struct wl_client *client,
252 struct wl_resource *resource,
253 uint32_t button,
254 uint32_t index)
255{
256 struct text_model *text_model = resource->data;
257 struct input_method *input_method, *next;
258
259 wl_list_for_each_safe(input_method, next, &text_model->input_methods, link) {
260 if (!input_method->context)
261 continue;
262 input_method_context_send_invoke_action(&input_method->context->resource, button, index);
263 }
264}
265
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200266static const struct text_model_interface text_model_implementation = {
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200267 text_model_set_surrounding_text,
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200268 text_model_activate,
269 text_model_deactivate,
Jan Arne Petersenc1e481e2012-09-09 23:08:46 +0200270 text_model_reset,
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200271 text_model_set_micro_focus,
272 text_model_set_preedit,
Jan Arne Petersenadfedc12013-01-16 21:26:46 +0100273 text_model_set_content_type,
274 text_model_invoke_action
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200275};
276
Jan Arne Petersen51963742012-08-10 16:47:20 +0200277static void text_model_factory_create_text_model(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200278 struct wl_resource *resource,
Jan Arne Petersen4c265182012-09-09 23:08:30 +0200279 uint32_t id)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200280{
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200281 struct text_model_factory *text_model_factory = resource->data;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200282 struct text_model *text_model;
283
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200284 text_model = calloc(1, sizeof *text_model);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200285
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200286 text_model->resource.object.id = id;
287 text_model->resource.object.interface = &text_model_interface;
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200288 text_model->resource.object.implementation =
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200289 (void (**)(void)) &text_model_implementation;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200290
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200291 text_model->resource.data = text_model;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200292 text_model->resource.destroy = destroy_text_model;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200293
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200294 text_model->ec = text_model_factory->ec;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200295
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200296 wl_list_init(&text_model->input_methods);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200297
298 wl_client_add_resource(client, &text_model->resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200299};
300
Jan Arne Petersen51963742012-08-10 16:47:20 +0200301static const struct text_model_factory_interface text_model_factory_implementation = {
302 text_model_factory_create_text_model
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200303};
304
305static void
Jan Arne Petersen51963742012-08-10 16:47:20 +0200306bind_text_model_factory(struct wl_client *client,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200307 void *data,
308 uint32_t version,
309 uint32_t id)
310{
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200311 struct text_model_factory *text_model_factory = data;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200312
313 /* No checking for duplicate binding necessary.
314 * No events have to be sent, so we don't need the return value. */
Jan Arne Petersen51963742012-08-10 16:47:20 +0200315 wl_client_add_object(client, &text_model_factory_interface,
316 &text_model_factory_implementation,
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200317 id, text_model_factory);
318}
319
320static void
321text_model_factory_notifier_destroy(struct wl_listener *listener, void *data)
322{
323 struct text_model_factory *text_model_factory =
324 container_of(listener, struct text_model_factory, destroy_listener);
325
326 wl_display_remove_global(text_model_factory->ec->wl_display,
327 text_model_factory->text_model_factory_global);
328
329 free(text_model_factory);
330}
331
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100332static void
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200333text_model_factory_create(struct weston_compositor *ec)
334{
335 struct text_model_factory *text_model_factory;
336
337 text_model_factory = calloc(1, sizeof *text_model_factory);
338
339 text_model_factory->ec = ec;
340
341 text_model_factory->text_model_factory_global =
342 wl_display_add_global(ec->wl_display,
343 &text_model_factory_interface,
344 text_model_factory, bind_text_model_factory);
345
346 text_model_factory->destroy_listener.notify = text_model_factory_notifier_destroy;
347 wl_signal_add(&ec->destroy_signal, &text_model_factory->destroy_listener);
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200348}
349
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200350static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200351input_method_context_destroy(struct wl_client *client,
352 struct wl_resource *resource)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200353{
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200354 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200355}
356
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200357static void
358input_method_context_commit_string(struct wl_client *client,
359 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100360 uint32_t serial,
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200361 const char *text,
362 uint32_t index)
363{
364 struct input_method_context *context = resource->data;
365
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100366 text_model_send_commit_string(&context->model->resource, serial, text, index);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200367}
368
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200369static void
370input_method_context_preedit_string(struct wl_client *client,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100371 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100372 uint32_t serial,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100373 const char *text,
374 const char *commit)
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200375{
376 struct input_method_context *context = resource->data;
377
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100378 text_model_send_preedit_string(&context->model->resource, serial, text, commit);
Jan Arne Petersen46535312013-01-16 21:26:38 +0100379}
380
381static void
382input_method_context_preedit_styling(struct wl_client *client,
383 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100384 uint32_t serial,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100385 uint32_t index,
386 uint32_t length,
387 uint32_t style)
388{
389 struct input_method_context *context = resource->data;
390
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100391 text_model_send_preedit_styling(&context->model->resource, serial, index, length, style);
Jan Arne Petersen46535312013-01-16 21:26:38 +0100392}
393
394static void
395input_method_context_preedit_cursor(struct wl_client *client,
396 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100397 uint32_t serial,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100398 int32_t cursor)
399{
400 struct input_method_context *context = resource->data;
401
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100402 text_model_send_preedit_cursor(&context->model->resource, serial, cursor);
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200403}
404
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200405static void
406input_method_context_delete_surrounding_text(struct wl_client *client,
407 struct wl_resource *resource,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100408 uint32_t serial,
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200409 int32_t index,
410 uint32_t length)
411{
412 struct input_method_context *context = resource->data;
413
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100414 text_model_send_delete_surrounding_text(&context->model->resource, serial, index, length);
Jan Arne Petersene202bae2012-09-09 23:08:44 +0200415}
416
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200417static void
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100418input_method_context_modifiers_map(struct wl_client *client,
419 struct wl_resource *resource,
420 struct wl_array *map)
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200421{
422 struct input_method_context *context = resource->data;
423
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100424 text_model_send_modifiers_map(&context->model->resource, map);
425}
426
427static void
428input_method_context_keysym(struct wl_client *client,
429 struct wl_resource *resource,
430 uint32_t serial,
431 uint32_t time,
432 uint32_t sym,
433 uint32_t state,
434 uint32_t modifiers)
435{
436 struct input_method_context *context = resource->data;
437
438 text_model_send_keysym(&context->model->resource, serial, time,
439 sym, state, modifiers);
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200440}
441
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100442static void
443unbind_keyboard(struct wl_resource *resource)
444{
445 struct input_method_context *context = resource->data;
446
447 input_method_context_end_keyboard_grab(context);
448 context->keyboard = NULL;
449
450 free(resource);
451}
452
453static void
454input_method_context_grab_key(struct wl_keyboard_grab *grab,
455 uint32_t time, uint32_t key, uint32_t state_w)
456{
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100457 struct weston_keyboard *keyboard = (struct weston_keyboard *)grab->keyboard;
458 struct wl_display *display;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100459 uint32_t serial;
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100460
461 if (!keyboard->input_method_resource)
462 return;
463
464 display = wl_client_get_display(keyboard->input_method_resource->client);
465 serial = wl_display_next_serial(display);
466 wl_keyboard_send_key(keyboard->input_method_resource,
467 serial, time, key, state_w);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100468}
469
470static void
471input_method_context_grab_modifier(struct wl_keyboard_grab *grab, uint32_t serial,
472 uint32_t mods_depressed, uint32_t mods_latched,
473 uint32_t mods_locked, uint32_t group)
474{
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100475 struct weston_keyboard *keyboard = (struct weston_keyboard *)grab->keyboard;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100476
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100477 if (!keyboard->input_method_resource)
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100478 return;
479
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100480 wl_keyboard_send_modifiers(keyboard->input_method_resource,
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100481 serial, mods_depressed, mods_latched,
482 mods_locked, group);
483}
484
485static const struct wl_keyboard_grab_interface input_method_context_grab = {
486 input_method_context_grab_key,
487 input_method_context_grab_modifier,
488};
489
490static void
491input_method_context_grab_keyboard(struct wl_client *client,
492 struct wl_resource *resource,
493 uint32_t id)
494{
495 struct input_method_context *context = resource->data;
496 struct wl_resource *cr;
497 struct weston_seat *seat = context->input_method->seat;
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100498 struct weston_keyboard *keyboard = &seat->keyboard;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100499
500 cr = wl_client_add_object(client, &wl_keyboard_interface,
501 NULL, id, context);
502 cr->destroy = unbind_keyboard;
503
504 context->keyboard = cr;
505
506 wl_keyboard_send_keymap(cr, WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1,
507 seat->xkb_info.keymap_fd,
508 seat->xkb_info.keymap_size);
509
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100510 if (keyboard->keyboard.grab != &keyboard->keyboard.default_grab) {
511 wl_keyboard_end_grab(&keyboard->keyboard);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100512 }
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100513 wl_keyboard_start_grab(&keyboard->keyboard, &keyboard->input_method_grab);
514 keyboard->input_method_resource = cr;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100515}
516
Jan Arne Petersen337df952012-11-18 19:06:46 +0100517static void
518input_method_context_key(struct wl_client *client,
519 struct wl_resource *resource,
520 uint32_t serial,
521 uint32_t time,
522 uint32_t key,
523 uint32_t state_w)
524{
525 struct input_method_context *context = resource->data;
526 struct weston_seat *seat = context->input_method->seat;
527 struct wl_keyboard *keyboard = seat->seat.keyboard;
528 struct wl_keyboard_grab *default_grab = &keyboard->default_grab;
529
530 default_grab->interface->key(default_grab, time, key, state_w);
531}
532
533static void
534input_method_context_modifiers(struct wl_client *client,
535 struct wl_resource *resource,
536 uint32_t serial,
537 uint32_t mods_depressed,
538 uint32_t mods_latched,
539 uint32_t mods_locked,
540 uint32_t group)
541{
542 struct input_method_context *context = resource->data;
543
544 struct weston_seat *seat = context->input_method->seat;
545 struct wl_keyboard *keyboard = seat->seat.keyboard;
546 struct wl_keyboard_grab *default_grab = &keyboard->default_grab;
547
548 default_grab->interface->modifiers(default_grab,
549 serial, mods_depressed,
550 mods_latched, mods_locked,
551 group);
552}
553
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200554static const struct input_method_context_interface input_method_context_implementation = {
555 input_method_context_destroy,
Jan Arne Petersen43f4aa82012-09-09 23:08:43 +0200556 input_method_context_commit_string,
557 input_method_context_preedit_string,
Jan Arne Petersen46535312013-01-16 21:26:38 +0100558 input_method_context_preedit_styling,
559 input_method_context_preedit_cursor,
Jan Arne Petersence8a4432012-09-09 23:08:45 +0200560 input_method_context_delete_surrounding_text,
Jan Arne Petersend9be93b2012-11-18 19:06:43 +0100561 input_method_context_modifiers_map,
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100562 input_method_context_keysym,
Jan Arne Petersen337df952012-11-18 19:06:46 +0100563 input_method_context_grab_keyboard,
564 input_method_context_key,
565 input_method_context_modifiers
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200566};
567
568static void
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200569destroy_input_method_context(struct wl_resource *resource)
570{
571 struct input_method_context *context = resource->data;
572
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100573 if (context->keyboard) {
574 wl_resource_destroy(context->keyboard);
575 }
576
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200577 free(context);
578}
579
580static void
581input_method_context_create(struct text_model *model,
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100582 struct input_method *input_method,
583 uint32_t serial)
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200584{
585 struct input_method_context *context;
586
587 if (!input_method->input_method_binding)
588 return;
589
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100590 context = calloc(1, sizeof *context);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200591 if (context == NULL)
592 return;
593
594 context->resource.destroy = destroy_input_method_context;
595 context->resource.object.id = 0;
596 context->resource.object.interface = &input_method_context_interface;
597 context->resource.object.implementation =
598 (void (**)(void)) &input_method_context_implementation;
599 context->resource.data = context;
600 wl_signal_init(&context->resource.destroy_signal);
601
602 context->model = model;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100603 context->input_method = input_method;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200604 input_method->context = context;
605
606 wl_client_add_resource(input_method->input_method_binding->client, &context->resource);
607
Jan Arne Petersenc7d2a982013-01-16 21:26:39 +0100608 input_method_send_activate(input_method->input_method_binding, &context->resource, serial);
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200609}
610
611static void
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100612input_method_context_end_keyboard_grab(struct input_method_context *context)
613{
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100614 struct wl_keyboard_grab *grab = &context->input_method->seat->keyboard.input_method_grab;
615 struct weston_keyboard *keyboard = (struct weston_keyboard *)grab->keyboard;
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100616
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100617 if (!grab->keyboard)
618 return;
619
620 if (grab->keyboard->grab == grab)
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100621 wl_keyboard_end_grab(grab->keyboard);
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100622
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100623 keyboard->input_method_resource = NULL;
624}
Jan Arne Petersen466b9c12012-11-18 19:06:45 +0100625
626static void
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200627unbind_input_method(struct wl_resource *resource)
628{
629 struct input_method *input_method = resource->data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100630 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200631
632 input_method->input_method_binding = NULL;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200633 input_method->context = NULL;
634
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100635 text_backend->input_method.binding = NULL;
636
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200637 free(resource);
638}
639
640static void
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200641bind_input_method(struct wl_client *client,
642 void *data,
643 uint32_t version,
644 uint32_t id)
645{
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200646 struct input_method *input_method = data;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100647 struct text_backend *text_backend = input_method->text_backend;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200648 struct wl_resource *resource;
649
650 resource = wl_client_add_object(client, &input_method_interface,
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200651 NULL,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200652 id, input_method);
653
654 if (input_method->input_method_binding == NULL) {
655 resource->destroy = unbind_input_method;
656 input_method->input_method_binding = resource;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100657
658 text_backend->input_method.binding = resource;
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200659 return;
660 }
661
662 wl_resource_post_error(resource, WL_DISPLAY_ERROR_INVALID_OBJECT,
663 "interface object already bound");
664 wl_resource_destroy(resource);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200665}
666
667static void
668input_method_notifier_destroy(struct wl_listener *listener, void *data)
669{
Kristian Høgsbergf97f3792012-07-22 11:51:42 -0400670 struct input_method *input_method =
671 container_of(listener, struct input_method, destroy_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200672
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200673 if (input_method->model)
674 deactivate_text_model(input_method->model, input_method);
675
676 wl_display_remove_global(input_method->seat->compositor->wl_display,
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200677 input_method->input_method_global);
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200678
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200679 free(input_method);
680}
681
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200682static void
683handle_keyboard_focus(struct wl_listener *listener, void *data)
684{
685 struct wl_keyboard *keyboard = data;
686 struct input_method *input_method =
687 container_of(listener, struct input_method, keyboard_focus_listener);
688 struct wl_surface *surface = keyboard->focus;
689
690 if (!input_method->model)
691 return;
692
693 if (!surface || input_method->model->surface != surface)
694 deactivate_text_model(input_method->model,
695 input_method);
696}
697
698static void
699input_method_init_seat(struct weston_seat *seat)
700{
701 if (seat->input_method->focus_listener_initialized)
702 return;
703
704 if (seat->has_keyboard) {
705 seat->input_method->keyboard_focus_listener.notify = handle_keyboard_focus;
706 wl_signal_add(&seat->seat.keyboard->focus_signal, &seat->input_method->keyboard_focus_listener);
Jan Arne Petersena75a7892013-01-16 21:26:50 +0100707 seat->keyboard.input_method_grab.interface = &input_method_context_grab;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200708 }
709
710 seat->input_method->focus_listener_initialized = 1;
711}
712
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100713static void
714handle_input_method_sigchld(struct weston_process *process, int status)
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200715{
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100716 struct text_backend *text_backend =
717 container_of(process, struct text_backend, input_method.process);
718
719 text_backend->input_method.process.pid = 0;
720 text_backend->input_method.client = NULL;
721}
722
723static void
724launch_input_method(struct text_backend *text_backend)
725{
726 if (text_backend->input_method.binding)
727 return;
728
729 if (!text_backend->input_method.path)
730 return;
731
732 if (text_backend->input_method.process.pid != 0)
733 return;
734
735 text_backend->input_method.client = weston_client_launch(text_backend->compositor,
736 &text_backend->input_method.process,
737 text_backend->input_method.path,
738 handle_input_method_sigchld);
739
740 if (!text_backend->input_method.client)
741 weston_log("not able to start %s\n", text_backend->input_method.path);
742}
743
744static void
745handle_seat_created(struct wl_listener *listener,
746 void *data)
747{
748 struct weston_seat *seat = data;
749 struct text_backend *text_backend =
750 container_of(listener, struct text_backend,
751 seat_created_listener);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200752 struct input_method *input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100753 struct weston_compositor *ec = seat->compositor;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200754
Philipp Brüschweiler17467812012-07-11 22:25:30 +0200755 input_method = calloc(1, sizeof *input_method);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200756
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200757 input_method->seat = seat;
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200758 input_method->model = NULL;
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200759 input_method->focus_listener_initialized = 0;
Jan Arne Petersen620cd622012-09-09 23:08:32 +0200760 input_method->context = NULL;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100761 input_method->text_backend = text_backend;
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200762
Philipp Brüschweilerf25602b2012-07-11 22:25:31 +0200763 input_method->input_method_global =
764 wl_display_add_global(ec->wl_display,
765 &input_method_interface,
766 input_method, bind_input_method);
767
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200768 input_method->destroy_listener.notify = input_method_notifier_destroy;
Philipp Brüschweilerb13b9ff2012-09-09 23:08:31 +0200769 wl_signal_add(&seat->seat.destroy_signal, &input_method->destroy_listener);
Jan Arne Petersene829adc2012-08-10 16:47:22 +0200770
771 seat->input_method = input_method;
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100772
773 launch_input_method(text_backend);
Jan Arne Petersen1f17be42012-06-21 21:52:18 +0200774}
Jan Arne Petersencd8cdcc2012-08-10 16:47:23 +0200775
Jan Arne Petersen674fd1d2012-11-18 19:06:42 +0100776static void
777text_backend_configuration(struct text_backend *text_backend)
778{
779 char *config_file;
780 char *path = NULL;
781
782 struct config_key input_method_keys[] = {
783 { "path", CONFIG_KEY_STRING, &path }
784 };
785
786 struct config_section cs[] = {
787 { "input-method", input_method_keys, ARRAY_LENGTH(input_method_keys), NULL }
788 };
789
790 config_file = config_file_path("weston.ini");
791 parse_config_file(config_file, cs, ARRAY_LENGTH(cs), text_backend);
792 free(config_file);
793
794 if (path)
795 text_backend->input_method.path = path;
796 else
797 text_backend->input_method.path = strdup(LIBEXECDIR "/weston-keyboard");
798}
799
800static void
801text_backend_notifier_destroy(struct wl_listener *listener, void *data)
802{
803 struct text_backend *text_backend =
804 container_of(listener, struct text_backend, destroy_listener);
805
806 if (text_backend->input_method.client)
807 wl_client_destroy(text_backend->input_method.client);
808
809 free(text_backend->input_method.path);
810
811 free(text_backend);
812}
813
814
815WL_EXPORT int
816text_backend_init(struct weston_compositor *ec)
817{
818 struct text_backend *text_backend;
819
820 text_backend = calloc(1, sizeof(*text_backend));
821
822 text_backend->compositor = ec;
823
824 text_backend->seat_created_listener.notify = handle_seat_created;
825 wl_signal_add(&ec->seat_created_signal,
826 &text_backend->seat_created_listener);
827
828 text_backend->destroy_listener.notify = text_backend_notifier_destroy;
829 wl_signal_add(&ec->destroy_signal, &text_backend->destroy_listener);
830
831 text_backend_configuration(text_backend);
832
833 text_model_factory_create(ec);
834
835 return 0;
836}