forked from extern/Unexpected-Keyboard
234 lines
6.8 KiB
Java
234 lines
6.8 KiB
Java
package juloo.keyboard2;
|
||
|
||
import android.util.SparseArray;
|
||
import android.view.KeyCharacterMap;
|
||
import android.view.KeyEvent;
|
||
import java.util.HashMap;
|
||
|
||
class KeyModifier
|
||
{
|
||
/* Cache key is KeyValue's name */
|
||
private static HashMap<String, SparseArray<KeyValue>> _cache =
|
||
new HashMap<String, SparseArray<KeyValue>>();
|
||
|
||
/* Modify a key according to modifiers. */
|
||
public static KeyValue handleFlags(KeyValue k, int flags)
|
||
{
|
||
if (k == null || flags == 0) // No modifier
|
||
return k;
|
||
SparseArray<KeyValue> ks = cacheEntry(k);
|
||
KeyValue r = ks.get(flags);
|
||
if (r != null) // Found in cache
|
||
return r;
|
||
r = k;
|
||
r = handleFn(r, flags);
|
||
if (r != null)
|
||
r = handleShift(r, flags);
|
||
if (r != null)
|
||
r = handleAccents(r, flags);
|
||
ks.put(flags, r);
|
||
return r;
|
||
}
|
||
|
||
private static KeyValue handleAccents(KeyValue k, int flags)
|
||
{
|
||
if (k.char_ == KeyValue.CHAR_NONE || (flags & KeyValue.FLAGS_ACCENTS) == 0)
|
||
return k;
|
||
char c = handleAccentChar(k.char_, flags);
|
||
if (c == 0 || c == k.char_)
|
||
return k;
|
||
return k.withCharAndSymbol(c);
|
||
}
|
||
|
||
private static KeyValue handleShift(KeyValue k, int flags)
|
||
{
|
||
if ((flags & KeyValue.FLAG_SHIFT) == 0)
|
||
return k;
|
||
char c = k.char_;
|
||
if (k.char_ != KeyValue.CHAR_NONE)
|
||
c = Character.toUpperCase(c);
|
||
if (c == k.char_) // More rules if toUpperCase() did nothing
|
||
switch (k.symbol)
|
||
{
|
||
case "→": c = '⇒'; break;
|
||
case "←": c = '⇐'; break;
|
||
case "<": c = '«'; break;
|
||
case ">": c = '»'; break;
|
||
case "“": c = '”'; break;
|
||
default: return k;
|
||
}
|
||
return k.withCharAndSymbol(c);
|
||
}
|
||
|
||
private static char handleAccentChar(char c, int flags)
|
||
{
|
||
switch ((flags & KeyValue.FLAGS_ACCENTS))
|
||
{
|
||
case KeyValue.FLAG_ACCENT1:
|
||
return (char)KeyCharacterMap.getDeadChar('\u02CB', c);
|
||
case KeyValue.FLAG_ACCENT2:
|
||
switch (c)
|
||
{
|
||
case '`': return '´';
|
||
case '<': return '‘';
|
||
case '>': return '‘';
|
||
default: return (char)KeyCharacterMap.getDeadChar('\u00B4', c);
|
||
}
|
||
case KeyValue.FLAG_ACCENT3:
|
||
return (char)KeyCharacterMap.getDeadChar('\u02C6', c);
|
||
case KeyValue.FLAG_ACCENT4:
|
||
switch (c)
|
||
{
|
||
case '?': return '¿';
|
||
case '!': return '¡';
|
||
default: return (char)KeyCharacterMap.getDeadChar('\u02DC', c);
|
||
}
|
||
case KeyValue.FLAG_ACCENT5:
|
||
switch (c)
|
||
{
|
||
case 'u': return 'µ';
|
||
case '"': return '„';
|
||
case '\'': return '’';
|
||
case '-': return '¬';
|
||
case 'a': return 'æ';
|
||
case 'o': return 'œ';
|
||
default: return (char)KeyCharacterMap.getDeadChar('\u00B8', c);
|
||
}
|
||
case KeyValue.FLAG_ACCENT6:
|
||
switch (c)
|
||
{
|
||
case '-': return '÷';
|
||
default: return (char)KeyCharacterMap.getDeadChar('\u00A8', c);
|
||
}
|
||
case KeyValue.FLAG_ACCENT_CARON:
|
||
switch (c)
|
||
{
|
||
default: return (char)KeyCharacterMap.getDeadChar('\u02C7', c);
|
||
}
|
||
case KeyValue.FLAG_ACCENT_RING:
|
||
switch (c)
|
||
{
|
||
default: return (char)KeyCharacterMap.getDeadChar('\u02DA', c);
|
||
}
|
||
case KeyValue.FLAG_ACCENT_MACRON:
|
||
switch (c)
|
||
{
|
||
default: return (char)KeyCharacterMap.getDeadChar('\u00AF', c);
|
||
}
|
||
case KeyValue.FLAG_ACCENT_ORDINAL:
|
||
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;
|
||
}
|
||
case KeyValue.FLAG_ACCENT_SUPERSCRIPT:
|
||
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;
|
||
}
|
||
case KeyValue.FLAG_ACCENT_SUBSCRIPT:
|
||
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;
|
||
}
|
||
default: return c; // Can't happen
|
||
}
|
||
}
|
||
|
||
private static KeyValue handleFn(KeyValue k, int flags)
|
||
{
|
||
if ((flags & KeyValue.FLAG_FN) == 0)
|
||
return k;
|
||
String name;
|
||
switch (k.name)
|
||
{
|
||
case "1": name = "f1"; break;
|
||
case "2": name = "f2"; break;
|
||
case "3": name = "f3"; break;
|
||
case "4": name = "f4"; break;
|
||
case "5": name = "f5"; break;
|
||
case "6": name = "f6"; break;
|
||
case "7": name = "f7"; break;
|
||
case "8": name = "f8"; break;
|
||
case "9": name = "f9"; break;
|
||
case "0": name = "f10"; break;
|
||
case "up": name = "page_up"; break;
|
||
case "down": name = "page_down"; break;
|
||
case "left": name = "home"; break;
|
||
case "right": name = "end"; break;
|
||
case ">": name = "→"; break;
|
||
case "<": name = "←"; break;
|
||
case "\"": name = "“"; break;
|
||
case "-": name = "–"; break;
|
||
case "_": name = "—"; break;
|
||
case "esc": name = "insert"; break;
|
||
case "$": name = "€"; break;
|
||
case "#": name = "£"; break;
|
||
case "*": name = "°"; break;
|
||
case "tab": name = "\\t"; break;
|
||
case "€": case "£": return null; // Avoid showing these twice
|
||
default: return k;
|
||
}
|
||
return KeyValue.getKeyByName(name);
|
||
}
|
||
|
||
/* Lookup the cache entry for a key. Create it needed. */
|
||
private static SparseArray<KeyValue> cacheEntry(KeyValue k)
|
||
{
|
||
SparseArray<KeyValue> ks = _cache.get(k.name);
|
||
if (ks == null)
|
||
{
|
||
ks = new SparseArray<KeyValue>();
|
||
_cache.put(k.name, ks);
|
||
}
|
||
return ks;
|
||
}
|
||
}
|