Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Android: Support unicode character mapping + dead keys #3004

Merged
merged 1 commit into from
Aug 7, 2023

Conversation

rib
Copy link
Contributor

@rib rib commented Aug 4, 2023

Up until now the Android backend has been directly mapping key codes which essentially just represent the "physical" cap of the key (quoted since this also related to virtual keyboards).

Since we didn't account for any meta keys either it meant the backend only supported a 1:1 mapping from key codes, which only covers a tiny subset of characters. For example you couldn't type a colon since there's no keycode for that and we didn't try and map Shift+Semicolon into a colon character.

This has been tricky to support because the NativeActivity class doesn't have direct access to the Java KeyEvent object which exposes a more convenient getUnicodeChar API.

It is now possible to query a KeyCharcterMap for the device associated with a KeyEvent via the AndroidApp::device_key_character_map API which provides a binding to the SDK KeyCharacterMap API in Java:

https://developer.android.com/reference/android/view/KeyCharacterMap

This is effectively what getUnicodeChar is implemented based on and is a bit more general purpose.

KeyCharacterMap lets us map a key_code + meta_state from a KeyEvent into either a unicode character or dead key accent that can be combined with the following key. This mapping is done based on the user's chosen layout for the keyboard.

To enable support for key character maps the
AndroidApp::input_events() API was replaced by
AndroidApp::input_events_iter() which returns a (lending) iterator for events. This was changed because the previous design made it difficult to allow other AndroidApp APIs to be used while iterating events (mainly because AndroidApp held a lock over the backend during iteration)

  • Tested on all platforms changed
  • Added an entry to CHANGELOG.md if knowledge of this change could be valuable to users
  • Updated documentation to reflect any user-facing changes, including notes of platform-specific behavior
  • Created or updated an example program if it would help users understand this functionality
  • Updated feature matrix, if new features were added or implemented

--

Note: this currently depends on a git branch of android-activity https://github.com/rust-mobile/android-activity/tree/rib/pr/input-api-rework-with-key-character-maps which I'm aiming to land soon in preparation for a 0.5 release of android-activity in time for Winit 0.29

@rib rib requested review from MarijnS95 and kchibisov August 4, 2023 21:37
@rib rib requested a review from msiglreith as a code owner August 4, 2023 21:37
@rib rib added this to the Version 0.29.0 milestone Aug 4, 2023
@rib rib force-pushed the rib/pr/android-unicode-key-mapping branch from 8a7b1df to 7730670 Compare August 4, 2023 21:48
@rib rib force-pushed the rib/pr/android-unicode-key-mapping branch 3 times, most recently from 3f4a37a to f6c4423 Compare August 5, 2023 00:31
@rib rib requested a review from kchibisov August 6, 2023 14:39
Copy link
Member

@kchibisov kchibisov left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should be good after you fix formatting.

Up until now the Android backend has been directly mapping key codes
which essentially just represent the "physical" cap of the key (quoted
since this also related to virtual keyboards).

Since we didn't account for any meta keys either it meant the backend
only supported a 1:1 mapping from key codes, which only covers a tiny
subset of characters. For example you couldn't type a colon since
there's no keycode for that and we didn't try and map Shift+Semicolon
into a colon character.

This has been tricky to support because the `NativeActivity` class doesn't
have direct access to the Java `KeyEvent` object which exposes a more
convenient `getUnicodeChar` API.

It is now possible to query a `KeyCharcterMap` for the device associated
with a `KeyEvent` via the `AndroidApp::device_key_character_map` API
which provides a binding to the SDK `KeyCharacterMap` API in Java:

 https://developer.android.com/reference/android/view/KeyCharacterMap

This is effectively what `getUnicodeChar` is implemented based on and is
a bit more general purpose.

`KeyCharacterMap` lets us map a key_code + meta_state from a `KeyEvent`
into either a unicode character or dead key accent that can be combined
with the following key. This mapping is done based on the user's chosen
layout for the keyboard.

To enable support for key character maps the
`AndroidApp::input_events()` API was replaced by
`AndroidApp::input_events_iter()` which returns a (lending) iterator for
events. This was changed because the previous design made it difficult
to allow other AndroidApp APIs to be used while iterating events (mainly
because AndroidApp held a lock over the backend during iteration)
@rib rib force-pushed the rib/pr/android-unicode-key-mapping branch from f6c4423 to b97181c Compare August 7, 2023 22:32
@rib
Copy link
Contributor Author

rib commented Aug 7, 2023

For reference here, I've landed the corresponding support in android-activity main (after also reverting back to a rust-version of 1.64) so the updated Cargo.toml now pulls android-activity from main.

I could potentially do a 0.5 beta release if needs be, to avoid the temporary git dependency.

One small tweak that was made before I merged the android-activity API was to rename get_dead_key to get_dead_char since that's what the corresponding Android SDK is called.

@kchibisov kchibisov merged commit bd2f1e8 into master Aug 7, 2023
@kchibisov kchibisov deleted the rib/pr/android-unicode-key-mapping branch August 7, 2023 22:56
@kchibisov kchibisov mentioned this pull request Aug 8, 2023
7 tasks
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Development

Successfully merging this pull request may close these issues.

2 participants