2021-04-18 23:28:49 +02:00
|
|
|
|
package juloo.keyboard2;
|
|
|
|
|
|
|
|
|
|
import android.view.KeyCharacterMap;
|
2021-04-18 23:58:35 +02:00
|
|
|
|
import android.view.KeyEvent;
|
2021-04-18 23:28:49 +02:00
|
|
|
|
import java.util.HashMap;
|
|
|
|
|
|
|
|
|
|
class KeyModifier
|
|
|
|
|
{
|
2022-03-19 16:27:57 +01:00
|
|
|
|
/** Cache key is KeyValue's name */
|
2022-06-06 00:23:45 +02:00
|
|
|
|
private static HashMap<KeyValue, HashMap<Pointers.Modifiers, KeyValue>> _cache =
|
|
|
|
|
new HashMap<KeyValue, HashMap<Pointers.Modifiers, KeyValue>>();
|
2022-03-19 16:27:57 +01:00
|
|
|
|
|
|
|
|
|
/** Modify a key according to modifiers. */
|
2022-06-05 01:38:42 +02:00
|
|
|
|
public static KeyValue modify(KeyValue k, Pointers.Modifiers mods)
|
2021-04-18 23:28:49 +02:00
|
|
|
|
{
|
2022-03-19 16:27:57 +01:00
|
|
|
|
if (k == null)
|
|
|
|
|
return null;
|
2022-06-05 01:38:42 +02:00
|
|
|
|
int n_mods = mods.size();
|
|
|
|
|
HashMap<Pointers.Modifiers, KeyValue> ks = cacheEntry(k);
|
|
|
|
|
KeyValue r = ks.get(mods);
|
|
|
|
|
if (r == null)
|
2022-03-19 16:27:57 +01:00
|
|
|
|
{
|
|
|
|
|
r = k;
|
2022-06-05 01:38:42 +02:00
|
|
|
|
/* Order: Fn, Shift, accents */
|
|
|
|
|
for (int i = 0; i < n_mods; i++)
|
|
|
|
|
r = modify(r, mods.get(i));
|
|
|
|
|
ks.put(mods, r);
|
2022-03-19 16:27:57 +01:00
|
|
|
|
}
|
2022-06-06 00:23:45 +02:00
|
|
|
|
/* Keys with an empty string are placeholder keys. */
|
|
|
|
|
return (r.getString().length() == 0) ? null : r;
|
2021-04-18 23:28:49 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 19:30:53 +02:00
|
|
|
|
public static KeyValue modify(KeyValue k, KeyValue.Modifier mod)
|
2021-04-18 23:28:49 +02:00
|
|
|
|
{
|
2022-06-05 01:38:42 +02:00
|
|
|
|
switch (mod)
|
|
|
|
|
{
|
2022-06-05 19:30:53 +02:00
|
|
|
|
case CTRL:
|
|
|
|
|
case ALT:
|
|
|
|
|
case META: return turn_into_keyevent(k);
|
|
|
|
|
case FN: return apply_fn(k);
|
|
|
|
|
case SHIFT: return apply_shift(k);
|
|
|
|
|
case GRAVE: return apply_dead_char(k, '\u02CB');
|
|
|
|
|
case AIGU: return apply_dead_char(k, '\u00B4');
|
|
|
|
|
case CIRCONFLEXE: return apply_dead_char(k, '\u02C6');
|
2023-06-04 23:07:35 +02:00
|
|
|
|
case TILDE: return apply_map_or_dead_char(k, map_char_tilde, '\u02DC');
|
2022-06-05 19:30:53 +02:00
|
|
|
|
case CEDILLE: return apply_dead_char(k, '\u00B8');
|
|
|
|
|
case TREMA: return apply_dead_char(k, '\u00A8');
|
|
|
|
|
case CARON: return apply_dead_char(k, '\u02C7');
|
|
|
|
|
case RING: return apply_dead_char(k, '\u02DA');
|
|
|
|
|
case MACRON: return apply_dead_char(k, '\u00AF');
|
|
|
|
|
case OGONEK: return apply_dead_char(k, '\u02DB');
|
|
|
|
|
case DOT_ABOVE: return apply_dead_char(k, '\u02D9');
|
2022-10-15 23:28:36 +02:00
|
|
|
|
case BREVE: return apply_dead_char(k, '\u02D8');
|
2022-06-05 19:30:53 +02:00
|
|
|
|
case DOUBLE_AIGU: return apply_map_char(k, map_char_double_aigu);
|
|
|
|
|
case ORDINAL: return apply_map_char(k, map_char_ordinal);
|
|
|
|
|
case SUPERSCRIPT: return apply_map_char(k, map_char_superscript);
|
|
|
|
|
case SUBSCRIPT: return apply_map_char(k, map_char_subscript);
|
|
|
|
|
case ARROWS: return apply_map_char(k, map_char_arrows);
|
|
|
|
|
case BOX: return apply_map_char(k, map_char_box);
|
|
|
|
|
case SLASH: return apply_map_char(k, map_char_slash);
|
2022-11-05 19:26:49 +01:00
|
|
|
|
case BAR: return apply_map_char(k, map_char_bar);
|
2022-06-05 19:30:53 +02:00
|
|
|
|
case ARROW_RIGHT: return apply_combining(k, "\u20D7");
|
2023-02-09 18:40:48 +01:00
|
|
|
|
case DOT_BELOW: return apply_map_char(k, map_char_dot_below);
|
|
|
|
|
case HORN: return apply_map_char(k, map_char_horn);
|
|
|
|
|
case HOOK_ABOVE: return apply_map_char(k, map_char_hook_above);
|
2022-06-05 01:38:42 +02:00
|
|
|
|
default: return k;
|
|
|
|
|
}
|
2021-05-08 02:00:47 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-02-12 23:14:57 +01:00
|
|
|
|
/** Modify a key after a long press. */
|
|
|
|
|
public static KeyValue modify_long_press(KeyValue k)
|
|
|
|
|
{
|
2023-02-12 23:20:11 +01:00
|
|
|
|
switch (k.getKind())
|
|
|
|
|
{
|
|
|
|
|
case Event:
|
|
|
|
|
switch (k.getEvent())
|
|
|
|
|
{
|
|
|
|
|
case CHANGE_METHOD_PREV:
|
|
|
|
|
return KeyValue.getKeyByName("change_method");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2023-02-12 23:14:57 +01:00
|
|
|
|
return k;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static KeyValue apply_map_char(KeyValue k, Map_char map)
|
|
|
|
|
{
|
|
|
|
|
switch (k.getKind())
|
|
|
|
|
{
|
|
|
|
|
case Char:
|
|
|
|
|
char kc = k.getChar();
|
|
|
|
|
char c = map.apply(kc);
|
2022-06-05 18:14:50 +02:00
|
|
|
|
return (kc == c) ? k : k.withChar(c);
|
2022-06-05 17:26:34 +02:00
|
|
|
|
default: return k;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 01:38:42 +02:00
|
|
|
|
private static KeyValue apply_dead_char(KeyValue k, char dead_char)
|
2021-05-08 02:00:47 +02:00
|
|
|
|
{
|
2022-06-05 17:26:34 +02:00
|
|
|
|
switch (k.getKind())
|
|
|
|
|
{
|
|
|
|
|
case Char:
|
|
|
|
|
char kc = k.getChar();
|
|
|
|
|
char c = (char)KeyCharacterMap.getDeadChar(dead_char, kc);
|
2022-06-05 18:14:50 +02:00
|
|
|
|
return (c == 0 || kc == c) ? k : k.withChar(c);
|
2022-06-05 17:26:34 +02:00
|
|
|
|
default: return k;
|
|
|
|
|
}
|
2021-04-18 23:28:49 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-06-04 23:07:35 +02:00
|
|
|
|
/** Apply a [Map_char] or fallback to [apply_dead_char]. */
|
|
|
|
|
private static KeyValue apply_map_or_dead_char(KeyValue k, Map_char map, char dead_char)
|
|
|
|
|
{
|
|
|
|
|
switch (k.getKind())
|
|
|
|
|
{
|
|
|
|
|
case Char:
|
|
|
|
|
char kc = k.getChar();
|
|
|
|
|
char c = map.apply(kc);
|
|
|
|
|
if (kc == c)
|
|
|
|
|
return apply_dead_char(k, dead_char);
|
|
|
|
|
return k.withChar(c);
|
|
|
|
|
default: return k;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 13:51:09 +02:00
|
|
|
|
private static KeyValue apply_combining(KeyValue k, String combining)
|
|
|
|
|
{
|
2022-06-05 17:26:34 +02:00
|
|
|
|
switch (k.getKind())
|
|
|
|
|
{
|
|
|
|
|
case Char:
|
2022-06-05 18:14:50 +02:00
|
|
|
|
return k.withString(String.valueOf(k.getChar()) + combining);
|
2022-06-05 17:26:34 +02:00
|
|
|
|
default: return k;
|
|
|
|
|
}
|
2022-06-05 13:51:09 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 01:38:42 +02:00
|
|
|
|
private static KeyValue apply_shift(KeyValue k)
|
2021-04-18 23:28:49 +02:00
|
|
|
|
{
|
2022-06-05 17:26:34 +02:00
|
|
|
|
switch (k.getKind())
|
2022-06-05 13:51:09 +02:00
|
|
|
|
{
|
2022-06-05 17:26:34 +02:00
|
|
|
|
case Char:
|
|
|
|
|
char kc = k.getChar();
|
|
|
|
|
char c = map_char_shift(kc);
|
|
|
|
|
if (kc == c)
|
|
|
|
|
c = Character.toUpperCase(kc);
|
2022-06-05 18:14:50 +02:00
|
|
|
|
return (kc == c) ? k : k.withChar(c);
|
2022-06-05 17:26:34 +02:00
|
|
|
|
case String:
|
2022-06-05 18:14:50 +02:00
|
|
|
|
return k.withString(k.getString().toUpperCase());
|
2022-06-05 17:26:34 +02:00
|
|
|
|
default: return k;
|
2022-06-05 13:51:09 +02:00
|
|
|
|
}
|
2021-04-18 23:28:49 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 01:38:42 +02:00
|
|
|
|
private static KeyValue apply_fn(KeyValue k)
|
2021-04-18 23:58:35 +02:00
|
|
|
|
{
|
2022-06-06 00:23:45 +02:00
|
|
|
|
String name = null;
|
|
|
|
|
switch (k.getKind())
|
2021-04-18 23:58:35 +02:00
|
|
|
|
{
|
2022-06-06 00:23:45 +02:00
|
|
|
|
case Char: name = apply_fn_char(k.getChar()); break;
|
|
|
|
|
case Keyevent: name = apply_fn_keyevent(k.getKeyevent()); break;
|
2022-07-09 17:31:44 +02:00
|
|
|
|
case Event: name = apply_fn_event(k.getEvent()); break;
|
2023-01-30 21:29:59 +01:00
|
|
|
|
case Placeholder: name = apply_fn_placeholder(k.getPlaceholder()); break;
|
2022-06-06 00:23:45 +02:00
|
|
|
|
}
|
|
|
|
|
return (name == null) ? k : KeyValue.getKeyByName(name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static String apply_fn_keyevent(int code)
|
|
|
|
|
{
|
|
|
|
|
switch (code)
|
|
|
|
|
{
|
|
|
|
|
case KeyEvent.KEYCODE_DPAD_UP: return "page_up";
|
|
|
|
|
case KeyEvent.KEYCODE_DPAD_DOWN: return "page_down";
|
|
|
|
|
case KeyEvent.KEYCODE_DPAD_LEFT: return "home";
|
|
|
|
|
case KeyEvent.KEYCODE_DPAD_RIGHT: return "end";
|
|
|
|
|
case KeyEvent.KEYCODE_ESCAPE: return "insert";
|
2022-10-16 00:04:06 +02:00
|
|
|
|
case KeyEvent.KEYCODE_TAB: return "\\t";
|
2022-06-06 00:23:45 +02:00
|
|
|
|
default: return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-09 17:31:44 +02:00
|
|
|
|
private static String apply_fn_event(KeyValue.Event ev)
|
|
|
|
|
{
|
|
|
|
|
switch (ev)
|
|
|
|
|
{
|
|
|
|
|
case SWITCH_NUMERIC: return "switch_greekmath";
|
|
|
|
|
default: return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-30 21:29:59 +01:00
|
|
|
|
private static String apply_fn_placeholder(KeyValue.Placeholder p)
|
|
|
|
|
{
|
|
|
|
|
switch (p)
|
|
|
|
|
{
|
|
|
|
|
case F11: return "f11";
|
|
|
|
|
case F12: return "f12";
|
2023-01-20 03:58:40 +01:00
|
|
|
|
case SHINDOT: return "shindot";
|
|
|
|
|
case SINDOT: return "sindot";
|
|
|
|
|
case OLE: return "ole";
|
|
|
|
|
case METEG: return "meteg";
|
2023-01-30 21:29:59 +01:00
|
|
|
|
default: return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-06 00:23:45 +02:00
|
|
|
|
/** Return the name of modified key, or [null]. */
|
|
|
|
|
private static String apply_fn_char(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case '1': return "f1";
|
|
|
|
|
case '2': return "f2";
|
|
|
|
|
case '3': return "f3";
|
|
|
|
|
case '4': return "f4";
|
|
|
|
|
case '5': return "f5";
|
|
|
|
|
case '6': return "f6";
|
|
|
|
|
case '7': return "f7";
|
|
|
|
|
case '8': return "f8";
|
|
|
|
|
case '9': return "f9";
|
|
|
|
|
case '0': return "f10";
|
|
|
|
|
case '<': return "«";
|
|
|
|
|
case '>': return "»";
|
|
|
|
|
case '{': return "‹";
|
|
|
|
|
case '}': return "›";
|
|
|
|
|
case '[': return "‘";
|
|
|
|
|
case ']': return "’";
|
|
|
|
|
case '(': return "“";
|
|
|
|
|
case ')': return "”";
|
|
|
|
|
case '\'': return "‚";
|
|
|
|
|
case '"': return "„";
|
|
|
|
|
case '-': return "–";
|
|
|
|
|
case '_': return "—";
|
|
|
|
|
case '^': return "¬";
|
|
|
|
|
case '%': return "‰";
|
|
|
|
|
case '=': return "≈";
|
|
|
|
|
case 'u': return "µ";
|
|
|
|
|
case 'a': return "æ";
|
|
|
|
|
case 'o': return "œ";
|
|
|
|
|
case '*': return "°";
|
|
|
|
|
case '.': return "…";
|
|
|
|
|
case ',': return "·";
|
|
|
|
|
case '!': return "¡";
|
|
|
|
|
case '?': return "¿";
|
2023-01-20 03:58:40 +01:00
|
|
|
|
case '|': return "¦";
|
2022-07-09 17:40:20 +02:00
|
|
|
|
// arrows
|
2022-06-06 00:23:45 +02:00
|
|
|
|
case '↖': return "⇖";
|
|
|
|
|
case '↑': return "⇑";
|
|
|
|
|
case '↗': return "⇗";
|
|
|
|
|
case '←': return "⇐";
|
|
|
|
|
case '→': return "⇒";
|
|
|
|
|
case '↙': return "⇙";
|
|
|
|
|
case '↓': return "⇓";
|
|
|
|
|
case '↘': return "⇘";
|
2022-07-09 17:40:20 +02:00
|
|
|
|
case '↔': return "⇔";
|
|
|
|
|
case '↕': return "⇕";
|
2022-04-02 16:56:40 +02:00
|
|
|
|
// Currency symbols
|
2022-06-06 00:23:45 +02:00
|
|
|
|
case 'e': return "€";
|
|
|
|
|
case 'l': return "£";
|
|
|
|
|
case 'r': return "₹";
|
|
|
|
|
case 'y': return "¥";
|
|
|
|
|
case 'c': return "¢";
|
|
|
|
|
case 'p': return "₱";
|
|
|
|
|
case '€': case '£': return "removed"; // Avoid showing these twice
|
2022-07-09 17:40:20 +02:00
|
|
|
|
// alternating greek letters
|
|
|
|
|
case 'π': return "ϖ";
|
|
|
|
|
case 'θ': return "ϑ";
|
|
|
|
|
case 'Θ': return "ϴ";
|
|
|
|
|
case 'ε': return "ϵ";
|
|
|
|
|
case 'β': return "ϐ";
|
|
|
|
|
case 'ρ': return "ϱ";
|
|
|
|
|
case 'σ': return "ς";
|
|
|
|
|
case 'γ': return "ɣ";
|
|
|
|
|
case 'φ': return "ϕ";
|
|
|
|
|
case 'υ': return "ϒ";
|
|
|
|
|
case 'κ': return "ϰ";
|
|
|
|
|
// alternating math characters
|
|
|
|
|
case '∪': return "⋃";
|
|
|
|
|
case '∩': return "⋂";
|
|
|
|
|
case '∃': return "∄";
|
|
|
|
|
case '∈': return "∉";
|
|
|
|
|
case '∫': return "∮";
|
|
|
|
|
case 'Π': return "∏";
|
|
|
|
|
case 'Σ': return "∑";
|
|
|
|
|
case '∨': return "⋁";
|
|
|
|
|
case '∧': return "⋀";
|
|
|
|
|
case '⊷': return "⊶";
|
|
|
|
|
case '⊂': return "⊆";
|
|
|
|
|
case '⊃': return "⊇";
|
|
|
|
|
case '±': return "∓";
|
2023-01-20 03:58:40 +01:00
|
|
|
|
// hebrew niqqud
|
|
|
|
|
case 'ק': return "qamats"; // kamatz
|
|
|
|
|
case 'ר': return "hataf_qamats"; // reduced kamatz
|
|
|
|
|
case 'ו': return "holam";
|
|
|
|
|
case 'ם': return "rafe";
|
|
|
|
|
case 'פ': return "patah"; // patach
|
|
|
|
|
case 'ש': return "sheva";
|
|
|
|
|
case 'ד': return "dagesh"; // or mapiq
|
|
|
|
|
case 'ח': return "hiriq";
|
|
|
|
|
case 'ף': return "hataf_patah"; // reduced patach
|
|
|
|
|
case 'ז': return "qubuts"; // kubuts
|
|
|
|
|
case 'ס': return "segol";
|
|
|
|
|
case 'ב': return "hataf_segol"; // reduced segol
|
|
|
|
|
case 'צ': return "tsere";
|
2023-03-11 16:43:17 +01:00
|
|
|
|
// Devanagari symbols
|
|
|
|
|
case 'ए': return "ऍ";
|
|
|
|
|
case 'े': return "ॅ";
|
|
|
|
|
case 'ऐ': return "ऎ";
|
|
|
|
|
case 'ै': return "ॆ";
|
|
|
|
|
case 'ऋ': return "ॠ";
|
|
|
|
|
case 'ृ': return "ॄ";
|
|
|
|
|
case 'ळ': return "ऴ";
|
|
|
|
|
case 'र': return "ऱ";
|
|
|
|
|
case 'क': return "क़";
|
|
|
|
|
case 'ख': return "ख़";
|
|
|
|
|
case 'ग': return "ग़";
|
|
|
|
|
case 'घ': return "ॻ";
|
|
|
|
|
case 'ढ': return "ढ़";
|
|
|
|
|
case 'न': return "ऩ";
|
|
|
|
|
case 'ड': return "ड़";
|
|
|
|
|
case 'ट': return "ॸ";
|
|
|
|
|
case 'ण': return "ॾ";
|
|
|
|
|
case 'फ': return "फ़";
|
|
|
|
|
case 'ऌ': return "ॡ";
|
|
|
|
|
case 'ॢ': return "ॣ";
|
|
|
|
|
case 'औ': return "ॵ";
|
|
|
|
|
case 'ौ': return "ॏ";
|
|
|
|
|
case 'ओ': return "ऒ";
|
|
|
|
|
case 'ो': return "ॊ";
|
|
|
|
|
case 'च': return "ॼ";
|
|
|
|
|
case 'ज': return "ज़";
|
|
|
|
|
case 'ब': return "ॿ";
|
|
|
|
|
case 'व': return "ॺ";
|
|
|
|
|
case 'य': return "य़";
|
|
|
|
|
case 'अ': return "ॲ";
|
|
|
|
|
case 'आ': return "ऑ";
|
|
|
|
|
case 'ा': return "ॉ";
|
|
|
|
|
case 'झ': return "ॹ";
|
|
|
|
|
case 'ई': return "ॴ";
|
|
|
|
|
case 'ी': return "ऻ";
|
|
|
|
|
case 'इ': return "ॳ";
|
|
|
|
|
case 'ि': return "ऺ";
|
|
|
|
|
case 'उ': return "ॶ";
|
|
|
|
|
case 'ऊ': return "ॷ";
|
|
|
|
|
case 'ु': return "ऄ";
|
|
|
|
|
case 'ष': return "क्ष";
|
|
|
|
|
case 'थ': return "त्र";
|
|
|
|
|
case 'द': return "द्र";
|
|
|
|
|
case 'प': return "प्र";
|
|
|
|
|
case 'श': return "श्र";
|
|
|
|
|
case 'छ': return "श्च";
|
|
|
|
|
case 'ँ': return "ऀ";
|
|
|
|
|
case '₹': return "₨";
|
|
|
|
|
case 'ॖ': return "ॗ";
|
|
|
|
|
case '॓': return "॔";
|
|
|
|
|
case '॰': return "ॱ";
|
|
|
|
|
case '।': return "॥";
|
|
|
|
|
case 'ं': return "ॕ";
|
|
|
|
|
case '़': return "ॎ";
|
|
|
|
|
case 'ऽ': return "ॽ";
|
2022-07-09 17:40:20 +02:00
|
|
|
|
// other
|
2022-06-06 00:23:45 +02:00
|
|
|
|
case ' ': return "nbsp";
|
|
|
|
|
default: return null;
|
2021-04-18 23:58:35 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 17:44:00 +02:00
|
|
|
|
private static KeyValue turn_into_keyevent(KeyValue k)
|
|
|
|
|
{
|
|
|
|
|
if (k.getKind() != KeyValue.Kind.Char)
|
|
|
|
|
return k;
|
|
|
|
|
int e;
|
|
|
|
|
switch (k.getChar())
|
|
|
|
|
{
|
|
|
|
|
case 'a': e = KeyEvent.KEYCODE_A; break;
|
|
|
|
|
case 'b': e = KeyEvent.KEYCODE_B; break;
|
|
|
|
|
case 'c': e = KeyEvent.KEYCODE_C; break;
|
|
|
|
|
case 'd': e = KeyEvent.KEYCODE_D; break;
|
|
|
|
|
case 'e': e = KeyEvent.KEYCODE_E; break;
|
|
|
|
|
case 'f': e = KeyEvent.KEYCODE_F; break;
|
|
|
|
|
case 'g': e = KeyEvent.KEYCODE_G; break;
|
|
|
|
|
case 'h': e = KeyEvent.KEYCODE_H; break;
|
|
|
|
|
case 'i': e = KeyEvent.KEYCODE_I; break;
|
|
|
|
|
case 'j': e = KeyEvent.KEYCODE_J; break;
|
|
|
|
|
case 'k': e = KeyEvent.KEYCODE_K; break;
|
|
|
|
|
case 'l': e = KeyEvent.KEYCODE_L; break;
|
|
|
|
|
case 'm': e = KeyEvent.KEYCODE_M; break;
|
|
|
|
|
case 'n': e = KeyEvent.KEYCODE_N; break;
|
|
|
|
|
case 'o': e = KeyEvent.KEYCODE_O; break;
|
|
|
|
|
case 'p': e = KeyEvent.KEYCODE_P; break;
|
|
|
|
|
case 'q': e = KeyEvent.KEYCODE_Q; break;
|
|
|
|
|
case 'r': e = KeyEvent.KEYCODE_R; break;
|
|
|
|
|
case 's': e = KeyEvent.KEYCODE_S; break;
|
|
|
|
|
case 't': e = KeyEvent.KEYCODE_T; break;
|
|
|
|
|
case 'u': e = KeyEvent.KEYCODE_U; break;
|
|
|
|
|
case 'v': e = KeyEvent.KEYCODE_V; break;
|
|
|
|
|
case 'w': e = KeyEvent.KEYCODE_W; break;
|
|
|
|
|
case 'x': e = KeyEvent.KEYCODE_X; break;
|
|
|
|
|
case 'y': e = KeyEvent.KEYCODE_Y; break;
|
|
|
|
|
case 'z': e = KeyEvent.KEYCODE_Z; break;
|
|
|
|
|
case '0': e = KeyEvent.KEYCODE_0; break;
|
|
|
|
|
case '1': e = KeyEvent.KEYCODE_1; break;
|
|
|
|
|
case '2': e = KeyEvent.KEYCODE_2; break;
|
|
|
|
|
case '3': e = KeyEvent.KEYCODE_3; break;
|
|
|
|
|
case '4': e = KeyEvent.KEYCODE_4; break;
|
|
|
|
|
case '5': e = KeyEvent.KEYCODE_5; break;
|
|
|
|
|
case '6': e = KeyEvent.KEYCODE_6; break;
|
|
|
|
|
case '7': e = KeyEvent.KEYCODE_7; break;
|
|
|
|
|
case '8': e = KeyEvent.KEYCODE_8; break;
|
|
|
|
|
case '9': e = KeyEvent.KEYCODE_9; break;
|
|
|
|
|
case '`': e = KeyEvent.KEYCODE_GRAVE; break;
|
|
|
|
|
case '-': e = KeyEvent.KEYCODE_MINUS; break;
|
|
|
|
|
case '=': e = KeyEvent.KEYCODE_EQUALS; break;
|
|
|
|
|
case '[': e = KeyEvent.KEYCODE_LEFT_BRACKET; break;
|
|
|
|
|
case ']': e = KeyEvent.KEYCODE_RIGHT_BRACKET; break;
|
|
|
|
|
case '\\': e = KeyEvent.KEYCODE_BACKSLASH; break;
|
|
|
|
|
case ';': e = KeyEvent.KEYCODE_SEMICOLON; break;
|
|
|
|
|
case '\'': e = KeyEvent.KEYCODE_APOSTROPHE; break;
|
|
|
|
|
case '/': e = KeyEvent.KEYCODE_SLASH; break;
|
|
|
|
|
case '@': e = KeyEvent.KEYCODE_AT; break;
|
|
|
|
|
case '+': e = KeyEvent.KEYCODE_PLUS; break;
|
|
|
|
|
case ',': e = KeyEvent.KEYCODE_COMMA; break;
|
|
|
|
|
case '.': e = KeyEvent.KEYCODE_PERIOD; break;
|
|
|
|
|
case '*': e = KeyEvent.KEYCODE_STAR; break;
|
|
|
|
|
case '#': e = KeyEvent.KEYCODE_POUND; break;
|
|
|
|
|
case '(': e = KeyEvent.KEYCODE_NUMPAD_LEFT_PAREN; break;
|
|
|
|
|
case ')': e = KeyEvent.KEYCODE_NUMPAD_RIGHT_PAREN; break;
|
|
|
|
|
case ' ': e = KeyEvent.KEYCODE_SPACE; break;
|
|
|
|
|
default: return k;
|
|
|
|
|
}
|
2022-06-05 19:30:53 +02:00
|
|
|
|
return k.withKeyevent(e);
|
2022-06-05 17:44:00 +02:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-18 23:28:49 +02:00
|
|
|
|
/* Lookup the cache entry for a key. Create it needed. */
|
2022-06-05 01:38:42 +02:00
|
|
|
|
private static HashMap<Pointers.Modifiers, KeyValue> cacheEntry(KeyValue k)
|
2021-04-18 23:28:49 +02:00
|
|
|
|
{
|
2022-06-06 00:23:45 +02:00
|
|
|
|
HashMap<Pointers.Modifiers, KeyValue> ks = _cache.get(k);
|
2021-04-18 23:28:49 +02:00
|
|
|
|
if (ks == null)
|
|
|
|
|
{
|
2022-06-05 01:38:42 +02:00
|
|
|
|
ks = new HashMap<Pointers.Modifiers, KeyValue>();
|
2022-06-06 00:23:45 +02:00
|
|
|
|
_cache.put(k, ks);
|
2021-04-18 23:28:49 +02:00
|
|
|
|
}
|
|
|
|
|
return ks;
|
|
|
|
|
}
|
2022-06-05 01:38:42 +02:00
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static abstract class Map_char
|
|
|
|
|
{
|
|
|
|
|
public abstract char apply(char c);
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-05 01:38:42 +02:00
|
|
|
|
private static char map_char_shift(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case '↙': return '⇙';
|
|
|
|
|
case '↓': return '⇓';
|
|
|
|
|
case '↘': return '⇘';
|
|
|
|
|
case '←': return '⇐';
|
|
|
|
|
case '→': return '⇒';
|
|
|
|
|
case '↖': return '⇖';
|
|
|
|
|
case '↑': return '⇑';
|
|
|
|
|
case '↗': return '⇗';
|
|
|
|
|
case '└': return '╚';
|
|
|
|
|
case '┴': return '╩';
|
|
|
|
|
case '┘': return '╝';
|
|
|
|
|
case '├': return '╠';
|
|
|
|
|
case '┼': return '╬';
|
|
|
|
|
case '┤': return '╣';
|
|
|
|
|
case '┌': return '╔';
|
|
|
|
|
case '┬': return '╦';
|
|
|
|
|
case '┐': return '╗';
|
|
|
|
|
case '─': return '═';
|
|
|
|
|
case '│': return '║';
|
2022-07-08 03:04:10 +02:00
|
|
|
|
case 'ß': return 'ẞ';
|
2022-10-04 10:10:54 +02:00
|
|
|
|
/* In Turkish, upper case of 'iı' is 'İI' but Java's toUpperCase will
|
|
|
|
|
return 'II'. To make 'İ' accessible, make it the shift of 'ı'. This
|
|
|
|
|
has the inconvenient of swapping i and ı on the keyboard. */
|
|
|
|
|
case 'ı': return 'İ';
|
2022-11-01 21:44:49 +01:00
|
|
|
|
case '₹': return '₨';
|
2022-06-05 01:38:42 +02:00
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-04 23:07:35 +02:00
|
|
|
|
private static final Map_char map_char_tilde =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case 'n': return 'ñ';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static final Map_char map_char_double_aigu =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
// Composite characters: a̋ e̋ i̋ m̋ ӳ
|
|
|
|
|
case 'o': return 'ő';
|
|
|
|
|
case 'u': return 'ű';
|
|
|
|
|
case ' ': return '˝';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2022-06-05 01:38:42 +02:00
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static final Map_char map_char_ordinal =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case 'a': return 'ª';
|
|
|
|
|
case 'o': return 'º';
|
|
|
|
|
case '1': return 'ª';
|
|
|
|
|
case '2': return 'º';
|
|
|
|
|
case '3': return 'ⁿ';
|
|
|
|
|
case '4': return 'ᵈ';
|
|
|
|
|
case '5': return 'ᵉ';
|
|
|
|
|
case '6': return 'ʳ';
|
|
|
|
|
case '7': return 'ˢ';
|
|
|
|
|
case '8': return 'ᵗ';
|
|
|
|
|
case '9': return 'ʰ';
|
|
|
|
|
case '*': return '°';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2022-06-05 01:38:42 +02:00
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static final Map_char map_char_superscript =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case '1': return '¹';
|
|
|
|
|
case '2': return '²';
|
|
|
|
|
case '3': return '³';
|
|
|
|
|
case '4': return '⁴';
|
|
|
|
|
case '5': return '⁵';
|
|
|
|
|
case '6': return '⁶';
|
|
|
|
|
case '7': return '⁷';
|
|
|
|
|
case '8': return '⁸';
|
|
|
|
|
case '9': return '⁹';
|
|
|
|
|
case '0': return '⁰';
|
|
|
|
|
case 'i': return 'ⁱ';
|
|
|
|
|
case '+': return '⁺';
|
|
|
|
|
case '-': return '⁻';
|
|
|
|
|
case '=': return '⁼';
|
|
|
|
|
case '(': return '⁽';
|
|
|
|
|
case ')': return '⁾';
|
|
|
|
|
case 'n': return 'ⁿ';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2022-06-05 01:38:42 +02:00
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static final Map_char map_char_subscript =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case '1': return '₁';
|
|
|
|
|
case '2': return '₂';
|
|
|
|
|
case '3': return '₃';
|
|
|
|
|
case '4': return '₄';
|
|
|
|
|
case '5': return '₅';
|
|
|
|
|
case '6': return '₆';
|
|
|
|
|
case '7': return '₇';
|
|
|
|
|
case '8': return '₈';
|
|
|
|
|
case '9': return '₉';
|
|
|
|
|
case '0': return '₀';
|
|
|
|
|
case '+': return '₊';
|
|
|
|
|
case '-': return '₋';
|
|
|
|
|
case '=': return '₌';
|
|
|
|
|
case '(': return '₍';
|
|
|
|
|
case ')': return '₎';
|
|
|
|
|
case 'e': return 'ₑ';
|
|
|
|
|
case 'a': return 'ₐ';
|
|
|
|
|
case 'x': return 'ₓ';
|
|
|
|
|
case 'o': return 'ₒ';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2022-06-05 01:38:42 +02:00
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static final Map_char map_char_arrows =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case '1': return '↙';
|
|
|
|
|
case '2': return '↓';
|
|
|
|
|
case '3': return '↘';
|
|
|
|
|
case '4': return '←';
|
|
|
|
|
case '6': return '→';
|
|
|
|
|
case '7': return '↖';
|
|
|
|
|
case '8': return '↑';
|
|
|
|
|
case '9': return '↗';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2022-06-05 01:38:42 +02:00
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static final Map_char map_char_box =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case '1': return '└';
|
|
|
|
|
case '2': return '┴';
|
|
|
|
|
case '3': return '┘';
|
|
|
|
|
case '4': return '├';
|
|
|
|
|
case '5': return '┼';
|
|
|
|
|
case '6': return '┤';
|
|
|
|
|
case '7': return '┌';
|
|
|
|
|
case '8': return '┬';
|
|
|
|
|
case '9': return '┐';
|
|
|
|
|
case '0': return '─';
|
|
|
|
|
case '.': return '│';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2022-06-05 12:02:35 +02:00
|
|
|
|
|
2022-06-05 17:26:34 +02:00
|
|
|
|
private static final Map_char map_char_slash =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case 'a': return 'ⱥ';
|
2022-11-05 19:33:46 +01:00
|
|
|
|
case 'b': return '␢';
|
2022-06-05 17:26:34 +02:00
|
|
|
|
case 'c': return 'ȼ';
|
|
|
|
|
case 'e': return 'ɇ';
|
|
|
|
|
case 'g': return 'ꞡ';
|
2022-11-05 19:33:46 +01:00
|
|
|
|
case 'k': return 'ꝃ';
|
2022-06-05 17:26:34 +02:00
|
|
|
|
case 'l': return 'ł';
|
|
|
|
|
case 'n': return 'ꞥ';
|
|
|
|
|
case 'o': return 'ø';
|
2022-11-05 19:33:46 +01:00
|
|
|
|
case 'r': return 'ꞧ';
|
|
|
|
|
case 's': return 'ꞩ';
|
|
|
|
|
case 't': return 'ⱦ';
|
2022-06-05 17:26:34 +02:00
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2022-11-05 19:26:49 +01:00
|
|
|
|
|
|
|
|
|
private static final Map_char map_char_bar =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case 'b': return 'ƀ';
|
|
|
|
|
case 'c': return 'ꞓ';
|
|
|
|
|
case 'd': return 'đ';
|
|
|
|
|
case 'g': return 'ǥ';
|
|
|
|
|
case 'i': return 'ɨ';
|
|
|
|
|
case 'j': return 'ɉ';
|
|
|
|
|
case 'k': return 'ꝁ';
|
|
|
|
|
case 'l': return 'ƚ';
|
|
|
|
|
case 'o': return 'ɵ';
|
|
|
|
|
case 'p': return 'ᵽ';
|
|
|
|
|
case 'q': return 'ꝗ';
|
|
|
|
|
case 'r': return 'ɍ';
|
|
|
|
|
case 't': return 'ŧ';
|
|
|
|
|
case 'u': return 'ʉ';
|
|
|
|
|
case 'y': return 'ɏ';
|
|
|
|
|
case 'z': return 'ƶ';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2023-02-09 18:40:48 +01:00
|
|
|
|
|
|
|
|
|
private static final Map_char map_char_dot_below =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case 'a': return 'ạ';
|
|
|
|
|
case 'ă': return 'ặ';
|
|
|
|
|
case 'â': return 'ậ';
|
|
|
|
|
case 'e': return 'ẹ';
|
|
|
|
|
case 'ê': return 'ệ';
|
|
|
|
|
case 'i': return 'ị';
|
|
|
|
|
case 'o': return 'ọ';
|
|
|
|
|
case 'ô': return 'ộ';
|
|
|
|
|
case 'ơ': return 'ợ';
|
|
|
|
|
case 'u': return 'ụ';
|
|
|
|
|
case 'ư': return 'ự';
|
|
|
|
|
case 'y': return 'ỵ';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
private static final Map_char map_char_horn =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case 'o': return 'ơ';
|
2023-02-09 23:38:55 +01:00
|
|
|
|
case 'ó': return 'ớ';
|
|
|
|
|
case 'ò': return 'ờ';
|
|
|
|
|
case 'ỏ': return 'ở';
|
|
|
|
|
case 'õ': return 'ỡ';
|
|
|
|
|
case 'ọ': return 'ợ';
|
2023-02-09 18:40:48 +01:00
|
|
|
|
case 'u': return 'ư';
|
2023-02-09 23:38:55 +01:00
|
|
|
|
case 'ú': return 'ứ';
|
|
|
|
|
case 'ù': return 'ừ';
|
|
|
|
|
case 'ủ': return 'ử';
|
|
|
|
|
case 'ũ': return 'ữ';
|
|
|
|
|
case 'ụ': return 'ự';
|
2023-02-09 18:40:48 +01:00
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
private static final Map_char map_char_hook_above =
|
|
|
|
|
new Map_char() {
|
|
|
|
|
public char apply(char c)
|
|
|
|
|
{
|
|
|
|
|
switch (c)
|
|
|
|
|
{
|
|
|
|
|
case 'a': return 'ả';
|
|
|
|
|
case 'ă': return 'ẳ';
|
|
|
|
|
case 'â': return 'ẩ';
|
|
|
|
|
case 'e': return 'ẻ';
|
|
|
|
|
case 'ê': return 'ể';
|
|
|
|
|
case 'i': return 'ỉ';
|
|
|
|
|
case 'o': return 'ỏ';
|
|
|
|
|
case 'ô': return 'ổ';
|
|
|
|
|
case 'ơ': return 'ở';
|
|
|
|
|
case 'u': return 'ủ';
|
|
|
|
|
case 'ư': return 'ử';
|
|
|
|
|
case 'y': return 'ỷ';
|
|
|
|
|
default: return c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
2021-04-18 23:28:49 +02:00
|
|
|
|
}
|