2010-12-14 06:31:47 +00:00
|
|
|
/*
|
2011-05-20 03:09:57 +00:00
|
|
|
* Copyright (C) 2010 The Android Open Source Project
|
2010-12-14 06:31:47 +00:00
|
|
|
*
|
2013-01-21 12:52:57 +00:00
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
2010-12-14 06:31:47 +00:00
|
|
|
*
|
2013-01-21 12:52:57 +00:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2010-12-14 06:31:47 +00:00
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
2013-01-21 12:52:57 +00:00
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
2010-12-14 06:31:47 +00:00
|
|
|
*/
|
|
|
|
|
2011-06-22 02:53:02 +00:00
|
|
|
package com.android.inputmethod.keyboard.internal;
|
2010-12-14 06:31:47 +00:00
|
|
|
|
2012-10-29 05:46:34 +00:00
|
|
|
import static com.android.inputmethod.latin.Constants.CODE_OUTPUT_TEXT;
|
|
|
|
import static com.android.inputmethod.latin.Constants.CODE_UNSPECIFIED;
|
2012-05-26 04:23:34 +00:00
|
|
|
|
2014-01-31 06:58:14 +00:00
|
|
|
import com.android.inputmethod.latin.Constants;
|
2013-06-23 16:11:32 +00:00
|
|
|
import com.android.inputmethod.latin.utils.StringUtils;
|
2010-12-14 06:31:47 +00:00
|
|
|
|
|
|
|
/**
|
2014-01-31 06:58:14 +00:00
|
|
|
* The string parser of the key specification.
|
|
|
|
*
|
|
|
|
* Each key specification is one of the following:
|
|
|
|
* - Label optionally followed by keyOutputText (keyLabel|keyOutputText).
|
|
|
|
* - Label optionally followed by code point (keyLabel|!code/code_name).
|
|
|
|
* - Icon followed by keyOutputText (!icon/icon_name|keyOutputText).
|
|
|
|
* - Icon followed by code point (!icon/icon_name|!code/code_name).
|
|
|
|
* Label and keyOutputText are one of the following:
|
|
|
|
* - Literal string.
|
|
|
|
* - Label reference represented by (!text/label_name), see {@link KeyboardTextsSet}.
|
|
|
|
* - String resource reference represented by (!text/resource_name), see {@link KeyboardTextsSet}.
|
|
|
|
* Icon is represented by (!icon/icon_name), see {@link KeyboardIconsSet}.
|
|
|
|
* Code is one of the following:
|
|
|
|
* - Code point presented by hexadecimal string prefixed with "0x"
|
|
|
|
* - Code reference represented by (!code/code_name), see {@link KeyboardCodesSet}.
|
2012-02-02 06:56:38 +00:00
|
|
|
* Special character, comma ',' backslash '\', and bar '|' can be escaped by '\' character.
|
2014-01-31 06:58:14 +00:00
|
|
|
* Note that the '\' is also parsed by XML parser and {@link MoreKeySpec#splitKeySpecs(String)}
|
|
|
|
* as well.
|
2010-12-14 06:31:47 +00:00
|
|
|
*/
|
2014-02-06 06:11:05 +00:00
|
|
|
// TODO: Rename to KeySpec and make this class to the key specification object.
|
2012-09-27 09:16:16 +00:00
|
|
|
public final class KeySpecParser {
|
2012-02-02 06:56:38 +00:00
|
|
|
// Constants for parsing.
|
2014-01-31 06:58:14 +00:00
|
|
|
private static final char BACKSLASH = Constants.CODE_BACKSLASH;
|
|
|
|
private static final char VERTICAL_BAR = Constants.CODE_VERTICAL_BAR;
|
2012-04-09 08:48:32 +00:00
|
|
|
private static final String PREFIX_HEX = "0x";
|
2010-12-14 06:31:47 +00:00
|
|
|
|
2012-02-02 06:41:11 +00:00
|
|
|
private KeySpecParser() {
|
2010-12-14 06:31:47 +00:00
|
|
|
// Intentional empty constructor for utility class.
|
|
|
|
}
|
|
|
|
|
2014-01-31 06:58:14 +00:00
|
|
|
private static boolean hasIcon(final String keySpec) {
|
|
|
|
return keySpec.startsWith(KeyboardIconsSet.PREFIX_ICON);
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|
|
|
|
|
2014-02-05 08:39:07 +00:00
|
|
|
private static boolean hasCode(final String keySpec, final int labelEnd) {
|
2014-02-06 05:39:10 +00:00
|
|
|
if (labelEnd <= 0 || labelEnd + 1 >= keySpec.length()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (keySpec.startsWith(KeyboardCodesSet.PREFIX_CODE, labelEnd + 1)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// This is a workaround to have a key that has a supplementary code point. We can't put a
|
|
|
|
// string in resource as a XML entity of a supplementary code point or a surrogate pair.
|
|
|
|
if (keySpec.startsWith(PREFIX_HEX, labelEnd + 1)) {
|
2010-12-14 06:31:47 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-08-30 05:22:40 +00:00
|
|
|
private static String parseEscape(final String text) {
|
2013-05-28 04:31:56 +00:00
|
|
|
if (text.indexOf(BACKSLASH) < 0) {
|
2010-12-14 06:31:47 +00:00
|
|
|
return text;
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2010-12-14 06:31:47 +00:00
|
|
|
final int length = text.length();
|
|
|
|
final StringBuilder sb = new StringBuilder();
|
|
|
|
for (int pos = 0; pos < length; pos++) {
|
|
|
|
final char c = text.charAt(pos);
|
2013-05-28 04:31:56 +00:00
|
|
|
if (c == BACKSLASH && pos + 1 < length) {
|
2012-02-02 06:41:11 +00:00
|
|
|
// Skip escape char
|
|
|
|
pos++;
|
|
|
|
sb.append(text.charAt(pos));
|
2010-12-14 06:31:47 +00:00
|
|
|
} else {
|
|
|
|
sb.append(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return sb.toString();
|
|
|
|
}
|
|
|
|
|
2014-02-05 08:39:07 +00:00
|
|
|
private static int indexOfLabelEnd(final String keySpec) {
|
2014-02-05 06:17:51 +00:00
|
|
|
final int length = keySpec.length();
|
2014-02-05 08:39:07 +00:00
|
|
|
if (keySpec.indexOf(BACKSLASH) < 0) {
|
|
|
|
final int labelEnd = keySpec.indexOf(VERTICAL_BAR);
|
|
|
|
if (labelEnd == 0) {
|
2014-02-05 06:17:51 +00:00
|
|
|
if (length == 1) {
|
|
|
|
// Treat a sole vertical bar as a special case of key label.
|
|
|
|
return -1;
|
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
throw new KeySpecParserError("Empty label");
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
return labelEnd;
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
for (int pos = 0; pos < length; pos++) {
|
2014-01-31 06:58:14 +00:00
|
|
|
final char c = keySpec.charAt(pos);
|
2013-05-28 04:31:56 +00:00
|
|
|
if (c == BACKSLASH && pos + 1 < length) {
|
2012-02-02 06:41:11 +00:00
|
|
|
// Skip escape char
|
2010-12-14 06:31:47 +00:00
|
|
|
pos++;
|
2013-05-28 04:31:56 +00:00
|
|
|
} else if (c == VERTICAL_BAR) {
|
2010-12-14 06:31:47 +00:00
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-02-05 08:39:07 +00:00
|
|
|
private static String getBeforeLabelEnd(final String keySpec, final int labelEnd) {
|
|
|
|
return (labelEnd < 0) ? keySpec : keySpec.substring(0, labelEnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static String getAfterLabelEnd(final String keySpec, final int labelEnd) {
|
|
|
|
return keySpec.substring(labelEnd + /* VERTICAL_BAR */1);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void checkDoubleLabelEnd(final String keySpec, final int labelEnd) {
|
|
|
|
if (indexOfLabelEnd(getAfterLabelEnd(keySpec, labelEnd)) < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
throw new KeySpecParserError("Multiple " + VERTICAL_BAR + ": " + keySpec);
|
|
|
|
}
|
|
|
|
|
2014-01-31 06:58:14 +00:00
|
|
|
public static String getLabel(final String keySpec) {
|
2014-01-31 10:18:15 +00:00
|
|
|
if (keySpec == null) {
|
|
|
|
// TODO: Throw {@link KeySpecParserError} once Key.keyLabel attribute becomes mandatory.
|
|
|
|
return null;
|
|
|
|
}
|
2014-01-31 06:58:14 +00:00
|
|
|
if (hasIcon(keySpec)) {
|
2010-12-14 06:31:47 +00:00
|
|
|
return null;
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
final int labelEnd = indexOfLabelEnd(keySpec);
|
|
|
|
final String label = parseEscape(getBeforeLabelEnd(keySpec, labelEnd));
|
2014-01-31 06:58:14 +00:00
|
|
|
if (label.isEmpty()) {
|
|
|
|
throw new KeySpecParserError("Empty label: " + keySpec);
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2010-12-14 06:31:47 +00:00
|
|
|
return label;
|
|
|
|
}
|
|
|
|
|
2014-02-05 08:39:07 +00:00
|
|
|
private static String getOutputTextInternal(final String keySpec, final int labelEnd) {
|
|
|
|
if (labelEnd <= 0) {
|
2012-02-07 11:42:07 +00:00
|
|
|
return null;
|
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
checkDoubleLabelEnd(keySpec, labelEnd);
|
|
|
|
return parseEscape(getAfterLabelEnd(keySpec, labelEnd));
|
2012-02-07 11:42:07 +00:00
|
|
|
}
|
|
|
|
|
2014-01-31 06:58:14 +00:00
|
|
|
public static String getOutputText(final String keySpec) {
|
2014-01-31 10:18:15 +00:00
|
|
|
if (keySpec == null) {
|
|
|
|
// TODO: Throw {@link KeySpecParserError} once Key.keyLabel attribute becomes mandatory.
|
|
|
|
return null;
|
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
final int labelEnd = indexOfLabelEnd(keySpec);
|
|
|
|
if (hasCode(keySpec, labelEnd)) {
|
2010-12-14 06:31:47 +00:00
|
|
|
return null;
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
final String outputText = getOutputTextInternal(keySpec, labelEnd);
|
2012-02-07 11:42:07 +00:00
|
|
|
if (outputText != null) {
|
2012-03-08 08:07:02 +00:00
|
|
|
if (StringUtils.codePointCount(outputText) == 1) {
|
2012-02-07 11:42:07 +00:00
|
|
|
// If output text is one code point, it should be treated as a code.
|
|
|
|
// See {@link #getCode(Resources, String)}.
|
|
|
|
return null;
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2014-01-31 06:58:14 +00:00
|
|
|
if (outputText.isEmpty()) {
|
|
|
|
throw new KeySpecParserError("Empty outputText: " + keySpec);
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2014-01-31 06:58:14 +00:00
|
|
|
return outputText;
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|
2014-01-31 06:58:14 +00:00
|
|
|
final String label = getLabel(keySpec);
|
2012-01-19 03:42:15 +00:00
|
|
|
if (label == null) {
|
2014-01-31 06:58:14 +00:00
|
|
|
throw new KeySpecParserError("Empty label: " + keySpec);
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2010-12-20 11:30:26 +00:00
|
|
|
// Code is automatically generated for one letter label. See {@link getCode()}.
|
2012-03-08 08:07:02 +00:00
|
|
|
return (StringUtils.codePointCount(label) == 1) ? null : label;
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|
|
|
|
|
2014-02-10 08:48:35 +00:00
|
|
|
public static int getCode(final String keySpec) {
|
2014-01-31 10:18:15 +00:00
|
|
|
if (keySpec == null) {
|
|
|
|
// TODO: Throw {@link KeySpecParserError} once Key.keyLabel attribute becomes mandatory.
|
|
|
|
return CODE_UNSPECIFIED;
|
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
final int labelEnd = indexOfLabelEnd(keySpec);
|
|
|
|
if (hasCode(keySpec, labelEnd)) {
|
|
|
|
checkDoubleLabelEnd(keySpec, labelEnd);
|
2014-02-10 08:48:35 +00:00
|
|
|
return parseCode(getAfterLabelEnd(keySpec, labelEnd), CODE_UNSPECIFIED);
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
final String outputText = getOutputTextInternal(keySpec, labelEnd);
|
2012-02-07 11:42:07 +00:00
|
|
|
if (outputText != null) {
|
|
|
|
// If output text is one code point, it should be treated as a code.
|
|
|
|
// See {@link #getOutputText(String)}.
|
2012-03-08 08:07:02 +00:00
|
|
|
if (StringUtils.codePointCount(outputText) == 1) {
|
2012-02-07 11:42:07 +00:00
|
|
|
return outputText.codePointAt(0);
|
|
|
|
}
|
2012-10-29 05:46:34 +00:00
|
|
|
return CODE_OUTPUT_TEXT;
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2014-01-31 06:58:14 +00:00
|
|
|
final String label = getLabel(keySpec);
|
2014-02-05 08:39:07 +00:00
|
|
|
if (label == null) {
|
|
|
|
throw new KeySpecParserError("Empty label: " + keySpec);
|
2012-01-19 03:42:15 +00:00
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
// Code is automatically generated for one letter label.
|
|
|
|
return (StringUtils.codePointCount(label) == 1) ? label.codePointAt(0) : CODE_OUTPUT_TEXT;
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|
|
|
|
|
2014-02-10 08:48:35 +00:00
|
|
|
public static int parseCode(final String text, final int defaultCode) {
|
2014-02-05 08:39:07 +00:00
|
|
|
if (text == null) {
|
2014-02-06 05:39:10 +00:00
|
|
|
return defaultCode;
|
2014-02-05 08:39:07 +00:00
|
|
|
}
|
2014-01-31 07:27:49 +00:00
|
|
|
if (text.startsWith(KeyboardCodesSet.PREFIX_CODE)) {
|
2014-02-10 08:48:35 +00:00
|
|
|
return KeyboardCodesSet.getCode(text.substring(KeyboardCodesSet.PREFIX_CODE.length()));
|
2014-02-05 08:39:07 +00:00
|
|
|
}
|
2014-02-06 05:39:10 +00:00
|
|
|
// This is a workaround to have a key that has a supplementary code point. We can't put a
|
|
|
|
// string in resource as a XML entity of a supplementary code point or a surrogate pair.
|
2014-02-05 08:39:07 +00:00
|
|
|
if (text.startsWith(PREFIX_HEX)) {
|
2012-04-09 08:48:32 +00:00
|
|
|
return Integer.parseInt(text.substring(PREFIX_HEX.length()), 16);
|
|
|
|
}
|
2014-02-06 05:39:10 +00:00
|
|
|
return defaultCode;
|
2012-04-09 08:48:32 +00:00
|
|
|
}
|
|
|
|
|
2014-01-31 06:58:14 +00:00
|
|
|
public static int getIconId(final String keySpec) {
|
2014-01-31 10:18:15 +00:00
|
|
|
if (keySpec == null) {
|
|
|
|
// TODO: Throw {@link KeySpecParserError} once Key.keyLabel attribute becomes mandatory.
|
|
|
|
return KeyboardIconsSet.ICON_UNDEFINED;
|
|
|
|
}
|
2014-02-05 08:39:07 +00:00
|
|
|
if (!hasIcon(keySpec)) {
|
|
|
|
return KeyboardIconsSet.ICON_UNDEFINED;
|
|
|
|
}
|
|
|
|
final int labelEnd = indexOfLabelEnd(keySpec);
|
|
|
|
final String iconName = getBeforeLabelEnd(keySpec, labelEnd)
|
|
|
|
.substring(KeyboardIconsSet.PREFIX_ICON.length());
|
|
|
|
return KeyboardIconsSet.getIconId(iconName);
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|
|
|
|
|
2012-01-30 02:14:45 +00:00
|
|
|
@SuppressWarnings("serial")
|
2012-09-27 09:16:16 +00:00
|
|
|
public static final class KeySpecParserError extends RuntimeException {
|
2012-08-30 05:22:40 +00:00
|
|
|
public KeySpecParserError(final String message) {
|
2012-01-30 02:14:45 +00:00
|
|
|
super(message);
|
2011-08-01 03:43:14 +00:00
|
|
|
}
|
|
|
|
}
|
2010-12-14 06:31:47 +00:00
|
|
|
}
|