Flutter Engine
 
Loading...
Searching...
No Matches
fl_keyboard_manager.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
7#include <array>
8#include <cinttypes>
9#include <memory>
10#include <string>
11
16
17// Turn on this flag to print complete layout data when switching IMEs. The data
18// is used in unit tests.
19#define DEBUG_PRINT_LAYOUT
20
21namespace {
22
23static bool is_eascii(uint16_t character) {
24 return character < 256;
25}
26
27#ifdef DEBUG_PRINT_LAYOUT
28// Prints layout entries that will be parsed by `MockLayoutData`.
29void debug_format_layout_data(std::string& debug_layout_data,
30 uint16_t keycode,
31 uint16_t clue1,
32 uint16_t clue2) {
33 if (keycode % 4 == 0) {
34 debug_layout_data.append(" ");
35 }
36
37 constexpr int kBufferSize = 30;
38 char buffer[kBufferSize];
39 buffer[0] = 0;
40 buffer[kBufferSize - 1] = 0;
41
42 snprintf(buffer, kBufferSize, "0x%04x, 0x%04x, ", clue1, clue2);
43 debug_layout_data.append(buffer);
44
45 if (keycode % 4 == 3) {
46 snprintf(buffer, kBufferSize, " // 0x%02x", keycode);
47 debug_layout_data.append(buffer);
48 }
49}
50#endif
51
52} // namespace
53
54typedef struct {
55 // Event being handled.
56 FlKeyEvent* event;
57
58 // TRUE if the embedder has responded.
60
61 // TRUE if the channel has responded.
63
64 // TRUE if this event is to be redispatched;
65 gboolean redispatch;
66
67 // TRUE if either the embedder of channel handled this event (or both).
68 gboolean handled;
70
71static HandleEventData* handle_event_data_new(FlKeyEvent* event) {
73 static_cast<HandleEventData*>(g_new0(HandleEventData, 1));
74 data->event = FL_KEY_EVENT(g_object_ref(event));
75 return data;
76}
77
79 g_object_unref(data->event);
80 g_free(data);
81}
82
85
88
89 // Key events that have been redispatched.
91
92 FlKeyEmbedderResponder* key_embedder_responder;
93
94 FlKeyChannelResponder* key_channel_responder;
95
96 // Record the derived layout.
97 //
98 // It is cleared when the platform reports a layout switch. Each entry,
99 // which corresponds to a group, is only initialized on the arrival of the
100 // first event for that group that has a goal keycode.
101 FlKeyboardLayout* derived_layout;
102
103 // A static map from keycodes to all layout goals.
104 //
105 // It is set up when the manager is initialized and is not changed ever after.
106 std::unique_ptr<std::map<uint16_t, const LayoutGoal*>> keycode_to_goals;
107
108 // A static map from logical keys to all mandatory layout goals.
109 //
110 // It is set up when the manager is initialized and is not changed ever after.
111 std::unique_ptr<std::map<uint64_t, const LayoutGoal*>>
113
114 GdkKeymap* keymap;
115 gulong keymap_keys_changed_cb_id; // Signal connection ID for
116 // keymap-keys-changed
117
118 GCancellable* cancellable;
119};
120
121G_DEFINE_TYPE(FlKeyboardManager, fl_keyboard_manager, G_TYPE_OBJECT);
122
123static gboolean event_is_redispatched(FlKeyboardManager* self,
124 FlKeyEvent* event) {
125 guint32 time = fl_key_event_get_time(event);
126 gboolean is_press = !!fl_key_event_get_is_press(event);
127 guint16 keycode = fl_key_event_get_keycode(event);
128 for (guint i = 0; i < self->redispatched_key_events->len; i++) {
129 FlKeyEvent* e =
130 FL_KEY_EVENT(g_ptr_array_index(self->redispatched_key_events, i));
131 if (fl_key_event_get_time(e) == time &&
132 !!fl_key_event_get_is_press(e) == is_press &&
133 fl_key_event_get_keycode(e) == keycode) {
134 g_ptr_array_remove_index(self->redispatched_key_events, i);
135 return TRUE;
136 }
137 }
138
139 return FALSE;
140}
141
142static void keymap_keys_changed_cb(FlKeyboardManager* self) {
143 g_clear_object(&self->derived_layout);
144 self->derived_layout = fl_keyboard_layout_new();
145}
146
147static void complete_handle_event(FlKeyboardManager* self, GTask* task) {
149 static_cast<HandleEventData*>(g_task_get_task_data(task));
150
151 // Waiting for responses.
152 if (!data->embedder_responded || !data->channel_responded) {
153 return;
154 }
155
156 data->redispatch = !data->handled;
157 g_task_return_boolean(task, TRUE);
158}
159
160static void responder_handle_embedder_event_cb(GObject* object,
161 GAsyncResult* result,
162 gpointer user_data) {
163 g_autoptr(GTask) task = G_TASK(user_data);
164 FlKeyboardManager* self = FL_KEYBOARD_MANAGER(g_task_get_source_object(task));
165
167 static_cast<HandleEventData*>(g_task_get_task_data(G_TASK(task)));
169
170 g_autoptr(GError) error = nullptr;
171 gboolean handled;
173 FL_KEY_EMBEDDER_RESPONDER(object), result, &handled, &error)) {
174 if (!g_error_matches(error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) {
175 g_warning("Failed to handle key event in embedder: %s", error->message);
176 }
177 handled = FALSE;
178 }
179 if (handled) {
180 data->handled = TRUE;
181 }
182
184}
185
186static void responder_handle_channel_event_cb(GObject* object,
187 GAsyncResult* result,
188 gpointer user_data) {
189 g_autoptr(GTask) task = G_TASK(user_data);
190 FlKeyboardManager* self = FL_KEYBOARD_MANAGER(g_task_get_source_object(task));
191
193 static_cast<HandleEventData*>(g_task_get_task_data(G_TASK(task)));
195
196 g_autoptr(GError) error = nullptr;
197 gboolean handled;
199 FL_KEY_CHANNEL_RESPONDER(object), result, &handled, &error)) {
200 if (!g_error_matches(error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) {
201 g_warning("Failed to handle key event in platform: %s", error->message);
202 }
203 handled = FALSE;
204 }
205 if (handled) {
206 data->handled = TRUE;
207 }
208
210}
211
212static uint16_t convert_key_to_char(FlKeyboardManager* self,
213 guint keycode,
214 gint group,
215 gint level) {
216 GdkKeymapKey key = {keycode, group, level};
217 constexpr int kBmpMax = 0xD7FF;
218 guint origin;
219 if (self->lookup_key_handler != nullptr) {
220 origin = self->lookup_key_handler(&key, self->lookup_key_handler_user_data);
221 } else {
222 origin = gdk_keymap_lookup_key(self->keymap, &key);
223 }
224 return origin < kBmpMax ? origin : 0xFFFF;
225}
226
227// Make sure that Flutter has derived the layout for the group of the event,
228// if the event contains a goal keycode.
229static void guarantee_layout(FlKeyboardManager* self, FlKeyEvent* event) {
230 guint8 group = fl_key_event_get_group(event);
231 if (fl_keyboard_layout_has_group(self->derived_layout, group)) {
232 return;
233 }
234 if (self->keycode_to_goals->find(fl_key_event_get_keycode(event)) ==
235 self->keycode_to_goals->end()) {
236 return;
237 }
238
239 // Clone all mandatory goals. Each goal is removed from this cloned map when
240 // fulfilled, and the remaining ones will be assigned to a default position.
241 std::map<uint64_t, const LayoutGoal*> remaining_mandatory_goals =
242 *self->logical_to_mandatory_goals;
243
244#ifdef DEBUG_PRINT_LAYOUT
245 std::string debug_layout_data;
246 for (uint16_t keycode = 0; keycode < 128; keycode += 1) {
247 std::vector<uint16_t> this_key_clues = {
248 convert_key_to_char(self, keycode, group, 0),
249 convert_key_to_char(self, keycode, group, 1), // Shift
250 };
251 debug_format_layout_data(debug_layout_data, keycode, this_key_clues[0],
252 this_key_clues[1]);
253 }
254#endif
255
256 // It's important to only traverse layout goals instead of all keycodes.
257 // Some key codes outside of the standard keyboard also gives alpha-numeric
258 // letters, and will therefore take over mandatory goals from standard
259 // keyboard keys if they come first. Example: French keyboard digit 1.
260 for (const LayoutGoal& keycode_goal : layout_goals) {
261 uint16_t keycode = keycode_goal.keycode;
262 std::vector<uint16_t> this_key_clues = {
263 convert_key_to_char(self, keycode, group, 0),
264 convert_key_to_char(self, keycode, group, 1), // Shift
265 };
266
267 // The logical key should be the first available clue from below:
268 //
269 // - Mandatory goal, if it matches any clue. This ensures that all alnum
270 // keys can be found somewhere.
271 // - US layout, if neither clue of the key is EASCII. This ensures that
272 // there are no non-latin logical keys.
273 // - A value derived on the fly from keycode & keyval.
274 for (uint16_t clue : this_key_clues) {
275 auto matching_goal = remaining_mandatory_goals.find(clue);
276 if (matching_goal != remaining_mandatory_goals.end()) {
277 // Found a key that produces a mandatory char. Use it.
278 g_return_if_fail(fl_keyboard_layout_get_logical_key(
279 self->derived_layout, group, keycode) == 0);
280 fl_keyboard_layout_set_logical_key(self->derived_layout, group, keycode,
281 clue);
282 remaining_mandatory_goals.erase(matching_goal);
283 break;
284 }
285 }
286 bool has_any_eascii =
287 is_eascii(this_key_clues[0]) || is_eascii(this_key_clues[1]);
288 // See if any produced char meets the requirement as a logical key.
289 if (fl_keyboard_layout_get_logical_key(self->derived_layout, group,
290 keycode) == 0 &&
291 !has_any_eascii) {
292 auto found_us_layout = self->keycode_to_goals->find(keycode);
293 if (found_us_layout != self->keycode_to_goals->end()) {
295 self->derived_layout, group, keycode,
296 found_us_layout->second->logical_key);
297 }
298 }
299 }
300
301 // Ensure all mandatory goals are assigned.
302 for (const auto mandatory_goal_iter : remaining_mandatory_goals) {
303 const LayoutGoal* goal = mandatory_goal_iter.second;
304 fl_keyboard_layout_set_logical_key(self->derived_layout, group,
305 goal->keycode, goal->logical_key);
306 }
307}
308
309static void fl_keyboard_manager_dispose(GObject* object) {
310 FlKeyboardManager* self = FL_KEYBOARD_MANAGER(object);
311
312 g_cancellable_cancel(self->cancellable);
313
314 self->keycode_to_goals.reset();
315 self->logical_to_mandatory_goals.reset();
316
317 g_clear_pointer(&self->redispatched_key_events, g_ptr_array_unref);
318 g_clear_object(&self->key_embedder_responder);
319 g_clear_object(&self->key_channel_responder);
320 g_clear_object(&self->derived_layout);
321 if (self->keymap_keys_changed_cb_id != 0) {
322 g_signal_handler_disconnect(self->keymap, self->keymap_keys_changed_cb_id);
323 self->keymap_keys_changed_cb_id = 0;
324 }
325 g_clear_object(&self->cancellable);
326
327 G_OBJECT_CLASS(fl_keyboard_manager_parent_class)->dispose(object);
328}
329
330static void fl_keyboard_manager_class_init(FlKeyboardManagerClass* klass) {
331 G_OBJECT_CLASS(klass)->dispose = fl_keyboard_manager_dispose;
332}
333
334static void fl_keyboard_manager_init(FlKeyboardManager* self) {
335 self->redispatched_key_events =
336 g_ptr_array_new_with_free_func(g_object_unref);
337 self->derived_layout = fl_keyboard_layout_new();
338
339 self->keycode_to_goals =
340 std::make_unique<std::map<uint16_t, const LayoutGoal*>>();
341 self->logical_to_mandatory_goals =
342 std::make_unique<std::map<uint64_t, const LayoutGoal*>>();
343 for (const LayoutGoal& goal : layout_goals) {
344 (*self->keycode_to_goals)[goal.keycode] = &goal;
345 if (goal.mandatory) {
346 (*self->logical_to_mandatory_goals)[goal.logical_key] = &goal;
347 }
348 }
349
351 self->keymap_keys_changed_cb_id = g_signal_connect_swapped(
352 self->keymap, "keys-changed", G_CALLBACK(keymap_keys_changed_cb), self);
353 self->cancellable = g_cancellable_new();
354}
355
356FlKeyboardManager* fl_keyboard_manager_new(FlEngine* engine) {
357 FlKeyboardManager* self = FL_KEYBOARD_MANAGER(
358 g_object_new(fl_keyboard_manager_get_type(), nullptr));
359
360 self->key_embedder_responder = fl_key_embedder_responder_new(engine);
361 self->key_channel_responder =
363
364 return self;
365}
366
368 FlKeyEvent* event) {
369 g_return_if_fail(FL_IS_KEYBOARD_MANAGER(self));
370
371 g_ptr_array_add(self->redispatched_key_events, g_object_ref(event));
372}
373
375 FlKeyEvent* event,
376 GCancellable* cancellable,
377 GAsyncReadyCallback callback,
378 gpointer user_data) {
379 g_return_if_fail(FL_IS_KEYBOARD_MANAGER(self));
380 g_return_if_fail(event != nullptr);
381
382 g_autoptr(GTask) task = g_task_new(self, cancellable, callback, user_data);
383
384 guarantee_layout(self, event);
385
386 g_task_set_task_data(
387 task, handle_event_data_new(event),
388 reinterpret_cast<GDestroyNotify>(handle_event_data_free));
389
390 if (event_is_redispatched(self, event)) {
392 static_cast<HandleEventData*>(g_task_get_task_data(task));
393 data->handled = TRUE;
394 g_task_return_boolean(task, TRUE);
395 return;
396 }
397
398 uint64_t specified_logical_key = fl_keyboard_layout_get_logical_key(
399 self->derived_layout, fl_key_event_get_group(event),
402 self->key_embedder_responder, event, specified_logical_key,
404 g_object_ref(task));
406 self->key_channel_responder, event, specified_logical_key,
407 self->cancellable, responder_handle_channel_event_cb, g_object_ref(task));
408}
409
411 FlKeyboardManager* self,
412 GAsyncResult* result,
413 FlKeyEvent** redispatched_event,
414 GError** error) {
415 g_return_val_if_fail(FL_IS_KEYBOARD_MANAGER(self), FALSE);
416 g_return_val_if_fail(g_task_is_valid(result, self), FALSE);
417
419 static_cast<HandleEventData*>(g_task_get_task_data(G_TASK(result)));
420 if (redispatched_event != nullptr && data->redispatch) {
421 *redispatched_event = FL_KEY_EVENT(g_object_ref(data->event));
422 }
423
424 return g_task_propagate_boolean(G_TASK(result), error);
425}
426
428 guint state,
429 double event_time) {
430 g_return_if_fail(FL_IS_KEYBOARD_MANAGER(self));
432 self->key_embedder_responder, state, event_time);
433}
434
435GHashTable* fl_keyboard_manager_get_pressed_state(FlKeyboardManager* self) {
436 g_return_val_if_fail(FL_IS_KEYBOARD_MANAGER(self), nullptr);
438 self->key_embedder_responder);
439}
440
442 FlKeyboardManager* self,
443 FlKeyboardManagerLookupKeyHandler lookup_key_handler,
444 gpointer user_data) {
445 g_return_if_fail(FL_IS_KEYBOARD_MANAGER(self));
446 self->lookup_key_handler = lookup_key_handler;
447 self->lookup_key_handler_user_data = user_data;
448}
FlutterEngine engine
Definition main.cc:84
g_autoptr(GMutexLocker) locker
return TRUE
G_MODULE_EXPORT FlBinaryMessenger * fl_engine_get_binary_messenger(FlEngine *self)
void fl_key_channel_responder_handle_event(FlKeyChannelResponder *self, FlKeyEvent *event, uint64_t specified_logical_key, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data)
gboolean fl_key_channel_responder_handle_event_finish(FlKeyChannelResponder *self, GAsyncResult *result, gboolean *handled, GError **error)
FlKeyChannelResponder * fl_key_channel_responder_new(FlBinaryMessenger *messenger)
FlKeyEmbedderResponder * fl_key_embedder_responder_new(FlEngine *engine)
GHashTable * fl_key_embedder_responder_get_pressed_state(FlKeyEmbedderResponder *self)
void fl_key_embedder_responder_sync_modifiers_if_needed(FlKeyEmbedderResponder *self, guint state, double event_time)
void fl_key_embedder_responder_handle_event(FlKeyEmbedderResponder *self, FlKeyEvent *event, uint64_t specified_logical_key, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data)
gboolean fl_key_embedder_responder_handle_event_finish(FlKeyEmbedderResponder *self, GAsyncResult *result, gboolean *handled, GError **error)
guint16 fl_key_event_get_keycode(FlKeyEvent *self)
gboolean fl_key_event_get_is_press(FlKeyEvent *self)
guint8 fl_key_event_get_group(FlKeyEvent *self)
guint32 fl_key_event_get_time(FlKeyEvent *self)
FlKeyboardLayout * fl_keyboard_layout_new()
uint64_t fl_keyboard_layout_get_logical_key(FlKeyboardLayout *self, uint8_t group, uint16_t keycode)
void fl_keyboard_layout_set_logical_key(FlKeyboardLayout *self, uint8_t group, uint16_t keycode, uint64_t logical_key)
gboolean fl_keyboard_layout_has_group(FlKeyboardLayout *self, uint8_t group)
static void fl_keyboard_manager_class_init(FlKeyboardManagerClass *klass)
static void fl_keyboard_manager_dispose(GObject *object)
static void guarantee_layout(FlKeyboardManager *self, FlKeyEvent *event)
static void responder_handle_channel_event_cb(GObject *object, GAsyncResult *result, gpointer user_data)
G_DEFINE_TYPE(FlKeyboardManager, fl_keyboard_manager, G_TYPE_OBJECT)
void fl_keyboard_manager_handle_event(FlKeyboardManager *self, FlKeyEvent *event, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data)
static void handle_event_data_free(HandleEventData *data)
static uint16_t convert_key_to_char(FlKeyboardManager *self, guint keycode, gint group, gint level)
GHashTable * fl_keyboard_manager_get_pressed_state(FlKeyboardManager *self)
static void fl_keyboard_manager_init(FlKeyboardManager *self)
static void responder_handle_embedder_event_cb(GObject *object, GAsyncResult *result, gpointer user_data)
static gboolean event_is_redispatched(FlKeyboardManager *self, FlKeyEvent *event)
static void keymap_keys_changed_cb(FlKeyboardManager *self)
static HandleEventData * handle_event_data_new(FlKeyEvent *event)
gboolean fl_keyboard_manager_handle_event_finish(FlKeyboardManager *self, GAsyncResult *result, FlKeyEvent **redispatched_event, GError **error)
void fl_keyboard_manager_set_lookup_key_handler(FlKeyboardManager *self, FlKeyboardManagerLookupKeyHandler lookup_key_handler, gpointer user_data)
static void complete_handle_event(FlKeyboardManager *self, GTask *task)
FlKeyboardManager * fl_keyboard_manager_new(FlEngine *engine)
void fl_keyboard_manager_add_redispatched_event(FlKeyboardManager *self, FlKeyEvent *event)
void fl_keyboard_manager_sync_modifier_if_needed(FlKeyboardManager *self, guint state, double event_time)
guint(* FlKeyboardManagerLookupKeyHandler)(const GdkKeymapKey *key, gpointer user_data)
const uint8_t uint32_t uint32_t GError ** error
FlutterDesktopBinaryReply callback
const std::vector< LayoutGoal > layout_goals
GdkDisplay * gdk_display_get_default()
Definition mock_gtk.cc:65
guint gdk_keymap_lookup_key(GdkKeymap *keymap, const GdkKeymapKey *key)
Definition mock_gtk.cc:60
GdkKeymap * gdk_keymap_get_for_display(GdkDisplay *display)
Definition mock_gtk.cc:52
FlKeyboardManagerLookupKeyHandler lookup_key_handler
FlKeyChannelResponder * key_channel_responder
std::unique_ptr< std::map< uint64_t, const LayoutGoal * > > logical_to_mandatory_goals
FlKeyEmbedderResponder * key_embedder_responder
GPtrArray * redispatched_key_events
std::unique_ptr< std::map< uint16_t, const LayoutGoal * > > keycode_to_goals
FlKeyboardLayout * derived_layout
uint16_t keycode
Definition key_mapping.h:42
uint64_t logical_key
Definition key_mapping.h:45
std::shared_ptr< const fml::Mapping > data