2010-10-21 10:05:53 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2010 Google Inc.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package com.android.inputmethod.latin;
|
|
|
|
|
|
|
|
import com.android.inputmethod.latin.BaseKeyboard.Key;
|
|
|
|
import com.android.inputmethod.latin.BaseKeyboard.Row;
|
2010-11-11 23:28:14 +00:00
|
|
|
import com.android.inputmethod.latin.KeyboardSwitcher.KeyboardId;
|
2010-10-21 10:05:53 +00:00
|
|
|
|
|
|
|
import org.xmlpull.v1.XmlPullParserException;
|
|
|
|
|
|
|
|
import android.content.res.Resources;
|
|
|
|
import android.content.res.TypedArray;
|
|
|
|
import android.content.res.XmlResourceParser;
|
2010-10-21 16:24:23 +00:00
|
|
|
import android.util.Log;
|
2010-10-21 10:05:53 +00:00
|
|
|
import android.util.TypedValue;
|
|
|
|
import android.util.Xml;
|
|
|
|
import android.view.InflateException;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
2010-10-21 16:24:23 +00:00
|
|
|
import java.util.List;
|
2010-10-21 10:05:53 +00:00
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
/**
|
|
|
|
* Parser for BaseKeyboard.
|
|
|
|
*
|
|
|
|
* This class parses Keyboard XML file and fill out keys in BaseKeyboard.
|
|
|
|
* The Keyboard XML file looks like:
|
|
|
|
* <pre>
|
|
|
|
* >!-- xml/keyboard.xml --<
|
|
|
|
* >Keyboard keyboard_attributes*<
|
|
|
|
* >!-- Keyboard Content --<
|
|
|
|
* >Row row_attributes*<
|
|
|
|
* >!-- Row Content --<
|
|
|
|
* >Key key_attributes* /<
|
|
|
|
* >Spacer horizontalGap="0.2in" /<
|
|
|
|
* >include keyboardLayout="@xml/other_keys"<
|
|
|
|
* ...
|
|
|
|
* >/Row<
|
|
|
|
* >include keyboardLayout="@xml/other_rows"<
|
|
|
|
* ...
|
|
|
|
* >/Keyboard<
|
|
|
|
* </pre>
|
2010-11-11 23:28:14 +00:00
|
|
|
* The XML file which is included in other file must have >merge< as root element, such as:
|
2010-10-21 16:24:23 +00:00
|
|
|
* <pre>
|
|
|
|
* >!-- xml/other_keys.xml --<
|
|
|
|
* >merge<
|
|
|
|
* >Key key_attributes* /<
|
|
|
|
* ...
|
|
|
|
* >/merge<
|
|
|
|
* </pre>
|
|
|
|
* and
|
|
|
|
* <pre>
|
|
|
|
* >!-- xml/other_rows.xml --<
|
|
|
|
* >merge<
|
|
|
|
* >Row row_attributes*<
|
|
|
|
* >Key key_attributes* /<
|
|
|
|
* >/Row<
|
|
|
|
* ...
|
|
|
|
* >/merge<
|
|
|
|
* </pre>
|
2010-11-11 23:28:14 +00:00
|
|
|
* You can also use switch-case-default tags to select Rows and Keys.
|
|
|
|
* <pre>
|
|
|
|
* >switch<
|
|
|
|
* >case case_attribute*<
|
|
|
|
* >!-- Any valid tags at switch position --<
|
|
|
|
* >/case<
|
|
|
|
* ...
|
|
|
|
* >default<
|
|
|
|
* >!-- Any valid tags at switch position --<
|
|
|
|
* >/default<
|
|
|
|
* >/switch<
|
|
|
|
* </pre>
|
|
|
|
*
|
|
|
|
* TODO: These are some random ideas to improve this parser.
|
|
|
|
* - can specify keyWidth attribute by multiplication of default keyWidth
|
|
|
|
* for example: keyWidth="200%b" ("b" stands for "base")
|
|
|
|
* - can declare style and specify styles within tags.
|
|
|
|
* for example:
|
|
|
|
* >switch<
|
|
|
|
* >case colorScheme="white"<
|
|
|
|
* >declare-style name="shift-key" parentStyle="modifier-key"<
|
|
|
|
* >item name="keyIcon"<@drawable/sym_keyboard_shift">/item<
|
|
|
|
* >/declare-style<
|
|
|
|
* >/case<
|
|
|
|
* >case colorScheme="black"<
|
|
|
|
* >declare-style name="shift-key" parentStyle="modifier-key"<
|
|
|
|
* >item name="keyIcon"<@drawable/sym_bkeyboard_shift">/item<
|
|
|
|
* >/declare-style<
|
|
|
|
* >/case<
|
|
|
|
* >/switch<
|
|
|
|
* ...
|
|
|
|
* >Key include-style="shift-key" ... /<
|
2010-10-21 16:24:23 +00:00
|
|
|
*/
|
2010-10-21 10:05:53 +00:00
|
|
|
public class BaseKeyboardParser {
|
2010-10-21 16:24:23 +00:00
|
|
|
private static final String TAG = "BaseKeyboardParser";
|
|
|
|
private static final boolean DEBUG_TAG = false;
|
|
|
|
private static final boolean DEBUG_PARSER = false;
|
|
|
|
|
2010-10-21 10:05:53 +00:00
|
|
|
// Keyboard XML Tags
|
|
|
|
private static final String TAG_KEYBOARD = "Keyboard";
|
|
|
|
private static final String TAG_ROW = "Row";
|
|
|
|
private static final String TAG_KEY = "Key";
|
|
|
|
private static final String TAG_SPACER = "Spacer";
|
|
|
|
private static final String TAG_INCLUDE = "include";
|
|
|
|
private static final String TAG_MERGE = "merge";
|
2010-11-11 23:28:14 +00:00
|
|
|
private static final String TAG_SWITCH = "switch";
|
|
|
|
private static final String TAG_CASE = "case";
|
|
|
|
private static final String TAG_DEFAULT = "default";
|
|
|
|
|
2010-10-21 10:05:53 +00:00
|
|
|
private final BaseKeyboard mKeyboard;
|
|
|
|
private final Resources mResources;
|
|
|
|
|
|
|
|
private int mCurrentX = 0;
|
|
|
|
private int mCurrentY = 0;
|
|
|
|
private int mMaxRowWidth = 0;
|
|
|
|
private int mTotalHeight = 0;
|
|
|
|
private Row mCurrentRow = null;
|
|
|
|
|
|
|
|
public BaseKeyboardParser(BaseKeyboard keyboard, Resources res) {
|
|
|
|
mKeyboard = keyboard;
|
|
|
|
mResources = res;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getMaxRowWidth() {
|
|
|
|
return mMaxRowWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getTotalHeight() {
|
|
|
|
return mTotalHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void parseKeyboard(XmlResourceParser parser)
|
|
|
|
throws XmlPullParserException, IOException {
|
2010-10-21 16:24:23 +00:00
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseKeyboard", false);
|
|
|
|
int event;
|
|
|
|
while ((event = parser.next()) != XmlResourceParser.END_DOCUMENT) {
|
|
|
|
if (event == XmlResourceParser.START_TAG) {
|
|
|
|
final String tag = parser.getName();
|
|
|
|
if (DEBUG_TAG) debugStartTag("parseKeyboard", tag, false);
|
|
|
|
if (TAG_KEYBOARD.equals(tag)) {
|
|
|
|
parseKeyboardAttributes(parser);
|
|
|
|
parseKeyboardContent(parser, mKeyboard.getKeys());
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
throw new IllegalStartTag(parser, TAG_KEYBOARD);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseKeyboard", false);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void parseKeyboardAttributes(XmlResourceParser parser) {
|
2010-10-21 10:05:53 +00:00
|
|
|
final BaseKeyboard keyboard = mKeyboard;
|
2010-10-21 16:24:23 +00:00
|
|
|
final TypedArray a = mResources.obtainAttributes(Xml.asAttributeSet(parser),
|
|
|
|
R.styleable.BaseKeyboard);
|
|
|
|
final int width = keyboard.getKeyboardWidth();
|
|
|
|
final int height = keyboard.getKeyboardHeight();
|
|
|
|
keyboard.setKeyWidth(getDimensionOrFraction(a,
|
|
|
|
R.styleable.BaseKeyboard_keyWidth, width, width / 10));
|
|
|
|
keyboard.setKeyHeight(getDimensionOrFraction(a,
|
|
|
|
R.styleable.BaseKeyboard_keyHeight, height, 50));
|
|
|
|
keyboard.setHorizontalGap(getDimensionOrFraction(a,
|
|
|
|
R.styleable.BaseKeyboard_horizontalGap, width, 0));
|
|
|
|
keyboard.setVerticalGap(getDimensionOrFraction(a,
|
|
|
|
R.styleable.BaseKeyboard_verticalGap, height, 0));
|
|
|
|
a.recycle();
|
|
|
|
}
|
2010-10-21 10:05:53 +00:00
|
|
|
|
2010-11-11 23:28:14 +00:00
|
|
|
private void parseKeyboardContent(XmlResourceParser parser, List<Key> keys)
|
2010-10-21 16:24:23 +00:00
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseKeyboardContent", keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
int event;
|
|
|
|
while ((event = parser.next()) != XmlResourceParser.END_DOCUMENT) {
|
|
|
|
if (event == XmlResourceParser.START_TAG) {
|
2010-10-21 16:24:23 +00:00
|
|
|
final String tag = parser.getName();
|
|
|
|
if (DEBUG_TAG) debugStartTag("parseKeyboardContent", tag, keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
if (TAG_ROW.equals(tag)) {
|
2010-10-21 16:24:23 +00:00
|
|
|
Row row = mKeyboard.createRowFromXml(mResources, parser);
|
2010-11-11 23:28:14 +00:00
|
|
|
if (keys != null)
|
|
|
|
startRow(row);
|
|
|
|
parseRowContent(parser, row, keys);
|
2010-10-21 10:05:53 +00:00
|
|
|
} else if (TAG_INCLUDE.equals(tag)) {
|
2010-10-21 16:24:23 +00:00
|
|
|
parseIncludeKeyboardContent(parser, keys);
|
2010-11-11 23:28:14 +00:00
|
|
|
} else if (TAG_SWITCH.equals(tag)) {
|
|
|
|
parseSwitchKeyboardContent(parser, keys);
|
2010-10-21 10:05:53 +00:00
|
|
|
} else {
|
2010-10-21 16:24:23 +00:00
|
|
|
throw new IllegalStartTag(parser, TAG_ROW);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
} else if (event == XmlResourceParser.END_TAG) {
|
2010-10-21 16:24:23 +00:00
|
|
|
final String tag = parser.getName();
|
|
|
|
if (DEBUG_TAG) debugEndTag("parseKeyboardContent", tag, keys == null);
|
|
|
|
if (TAG_KEYBOARD.equals(tag)) {
|
2010-10-21 10:05:53 +00:00
|
|
|
endKeyboard(mKeyboard.getVerticalGap());
|
2010-10-21 16:24:23 +00:00
|
|
|
break;
|
2010-11-11 23:28:14 +00:00
|
|
|
} else if (TAG_CASE.equals(tag) || TAG_DEFAULT.equals(tag)) {
|
|
|
|
break;
|
2010-10-21 10:05:53 +00:00
|
|
|
} else if (TAG_MERGE.equals(tag)) {
|
2010-10-21 16:24:23 +00:00
|
|
|
break;
|
2010-10-21 10:05:53 +00:00
|
|
|
} else {
|
2010-10-21 16:24:23 +00:00
|
|
|
throw new IllegalEndTag(parser, TAG_ROW);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-10-21 16:24:23 +00:00
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseKeyboardContent", keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void parseRowContent(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseRowContent", keys == null);
|
|
|
|
int event;
|
|
|
|
while ((event = parser.next()) != XmlResourceParser.END_DOCUMENT) {
|
|
|
|
if (event == XmlResourceParser.START_TAG) {
|
|
|
|
final String tag = parser.getName();
|
|
|
|
if (DEBUG_TAG) debugStartTag("parseRowContent", tag, keys == null);
|
|
|
|
if (TAG_KEY.equals(tag)) {
|
|
|
|
parseKey(parser, row, keys);
|
|
|
|
} else if (TAG_SPACER.equals(tag)) {
|
|
|
|
parseSpacer(parser, keys);
|
|
|
|
} else if (TAG_INCLUDE.equals(tag)) {
|
|
|
|
parseIncludeRowContent(parser, row, keys);
|
2010-11-11 23:28:14 +00:00
|
|
|
} else if (TAG_SWITCH.equals(tag)) {
|
|
|
|
parseSwitchRowContent(parser, row, keys);
|
2010-10-21 16:24:23 +00:00
|
|
|
} else {
|
|
|
|
throw new IllegalStartTag(parser, TAG_KEY);
|
|
|
|
}
|
|
|
|
} else if (event == XmlResourceParser.END_TAG) {
|
|
|
|
final String tag = parser.getName();
|
|
|
|
if (DEBUG_TAG) debugEndTag("parseRowContent", tag, keys == null);
|
|
|
|
if (TAG_ROW.equals(tag)) {
|
|
|
|
if (keys != null)
|
|
|
|
endRow();
|
|
|
|
break;
|
2010-11-11 23:28:14 +00:00
|
|
|
} else if (TAG_CASE.equals(tag) || TAG_DEFAULT.equals(tag)) {
|
|
|
|
break;
|
2010-10-21 16:24:23 +00:00
|
|
|
} else if (TAG_MERGE.equals(tag)) {
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
throw new IllegalEndTag(parser, TAG_KEY);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseRowContent", keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void parseKey(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseKey", keys == null);
|
|
|
|
if (keys == null) {
|
|
|
|
checkEndTag(TAG_KEY, parser);
|
|
|
|
} else {
|
|
|
|
Key key = mKeyboard.createKeyFromXml(mResources, row, mCurrentX, mCurrentY, parser);
|
|
|
|
checkEndTag(TAG_KEY, parser);
|
|
|
|
keys.add(key);
|
|
|
|
if (key.codes[0] == BaseKeyboard.KEYCODE_SHIFT)
|
|
|
|
mKeyboard.getShiftKeys().add(key);
|
|
|
|
endKey(key);
|
|
|
|
}
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void parseSpacer(XmlResourceParser parser, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseSpacer", keys == null);
|
|
|
|
if (keys == null) {
|
|
|
|
checkEndTag(TAG_SPACER, parser);
|
|
|
|
} else {
|
|
|
|
final TypedArray a = mResources.obtainAttributes(Xml.asAttributeSet(parser),
|
|
|
|
R.styleable.BaseKeyboard);
|
2010-11-11 23:28:14 +00:00
|
|
|
final int gap = getDimensionOrFraction(a, R.styleable.BaseKeyboard_horizontalGap,
|
2010-10-21 16:24:23 +00:00
|
|
|
mKeyboard.getKeyboardWidth(), 0);
|
|
|
|
a.recycle();
|
|
|
|
checkEndTag(TAG_SPACER, parser);
|
|
|
|
setSpacer(gap);
|
|
|
|
}
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void parseIncludeKeyboardContent(XmlResourceParser parser, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
2010-11-11 23:28:14 +00:00
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseIncludeKeyboardContent", keys == null);
|
2010-10-21 16:24:23 +00:00
|
|
|
parseIncludeInternal(parser, null, keys);
|
2010-11-11 23:28:14 +00:00
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseIncludeKeyboardContent", keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void parseIncludeRowContent(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
2010-11-11 23:28:14 +00:00
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseIncludeRowContent", keys == null);
|
2010-10-21 16:24:23 +00:00
|
|
|
parseIncludeInternal(parser, row, keys);
|
2010-11-11 23:28:14 +00:00
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseIncludeRowContent", keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void parseIncludeInternal(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (keys == null) {
|
|
|
|
checkEndTag(TAG_INCLUDE, parser);
|
|
|
|
} else {
|
|
|
|
final TypedArray a = mResources.obtainAttributes(Xml.asAttributeSet(parser),
|
|
|
|
R.styleable.BaseKeyboard_Include);
|
|
|
|
final int keyboardLayout = a.getResourceId(
|
|
|
|
R.styleable.BaseKeyboard_Include_keyboardLayout, 0);
|
|
|
|
a.recycle();
|
2010-10-21 10:05:53 +00:00
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
checkEndTag(TAG_INCLUDE, parser);
|
|
|
|
if (keyboardLayout == 0)
|
|
|
|
throw new ParseException("No keyboardLayout attribute in <include/>", parser);
|
|
|
|
parseMerge(mResources.getLayout(keyboardLayout), row, keys);
|
|
|
|
}
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void parseMerge(XmlResourceParser parser, Row row, List<Key> keys)
|
2010-10-21 10:05:53 +00:00
|
|
|
throws XmlPullParserException, IOException {
|
2010-10-21 16:24:23 +00:00
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseMerge", keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
int event;
|
|
|
|
while ((event = parser.next()) != XmlResourceParser.END_DOCUMENT) {
|
|
|
|
if (event == XmlResourceParser.START_TAG) {
|
2010-11-11 23:28:14 +00:00
|
|
|
final String tag = parser.getName();
|
2010-10-21 16:24:23 +00:00
|
|
|
if (DEBUG_TAG) debugStartTag("parseMerge", tag, keys == null);
|
|
|
|
if (TAG_MERGE.equals(tag)) {
|
|
|
|
if (row == null) {
|
|
|
|
parseKeyboardContent(parser, keys);
|
|
|
|
} else {
|
|
|
|
parseRowContent(parser, row, keys);
|
|
|
|
}
|
|
|
|
break;
|
2010-10-21 10:05:53 +00:00
|
|
|
} else {
|
2010-10-21 16:24:23 +00:00
|
|
|
throw new ParseException(
|
|
|
|
"Included keyboard layout must have <merge> root element", parser);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-10-21 16:24:23 +00:00
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseMerge", keys == null);
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-11-11 23:28:14 +00:00
|
|
|
private void parseSwitchKeyboardContent(XmlResourceParser parser, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseSwitchKeyboardContent", keys == null);
|
|
|
|
parseSwitchInternal(parser, null, keys);
|
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseSwitchKeyboardContent", keys == null);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void parseSwitchRowContent(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseSwitchRowContent", keys == null);
|
|
|
|
parseSwitchInternal(parser, row, keys);
|
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseSwitchRowContent", keys == null);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void parseSwitchInternal(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
boolean selected = false;
|
|
|
|
int event;
|
|
|
|
if (DEBUG_TAG) Log.d(TAG, "parseSwitchInternal: id=" + mKeyboard.mId);
|
|
|
|
while ((event = parser.next()) != XmlResourceParser.END_DOCUMENT) {
|
|
|
|
if (event == XmlResourceParser.START_TAG) {
|
|
|
|
final String tag = parser.getName();
|
|
|
|
if (DEBUG_TAG) debugStartTag("parseSwitchInternal", tag, keys == null);
|
|
|
|
if (TAG_CASE.equals(tag)) {
|
|
|
|
selected |= parseCase(parser, row, selected ? null : keys);
|
|
|
|
} else if (TAG_DEFAULT.equals(tag)) {
|
|
|
|
selected |= parseDefault(parser, row, selected ? null : keys);
|
|
|
|
} else {
|
|
|
|
throw new IllegalStartTag(parser, TAG_KEY);
|
|
|
|
}
|
|
|
|
} else if (event == XmlResourceParser.END_TAG) {
|
|
|
|
final String tag = parser.getName();
|
|
|
|
if (DEBUG_TAG) debugEndTag("parseRowContent", tag, keys == null);
|
|
|
|
if (TAG_SWITCH.equals(tag)) {
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
throw new IllegalEndTag(parser, TAG_KEY);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean parseCase(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseCase", keys == null);
|
|
|
|
final boolean selected = parseCaseCondition(parser);
|
|
|
|
if (row == null) {
|
|
|
|
// Processing Rows.
|
|
|
|
parseKeyboardContent(parser, selected ? keys : null);
|
|
|
|
} else {
|
|
|
|
// Processing Keys.
|
|
|
|
parseRowContent(parser, row, selected ? keys : null);
|
|
|
|
}
|
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseCase", keys == null || !selected);
|
|
|
|
return selected;
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean parseCaseCondition(XmlResourceParser parser) {
|
|
|
|
final BaseKeyboard keyboard = mKeyboard;
|
|
|
|
final TypedArray a = mResources.obtainAttributes(Xml.asAttributeSet(parser),
|
|
|
|
R.styleable.BaseKeyboard_Case);
|
2010-11-19 22:57:24 +00:00
|
|
|
final int mode = a.getInt(R.styleable.BaseKeyboard_Case_mode, -1);
|
2010-11-11 23:28:14 +00:00
|
|
|
final String settingsKey = a.getString(R.styleable.BaseKeyboard_Case_settingsKey);
|
|
|
|
final String voiceKey = a.getString(R.styleable.BaseKeyboard_Case_voiceKey);
|
|
|
|
a.recycle();
|
|
|
|
|
|
|
|
final KeyboardId id = keyboard.mId;
|
|
|
|
if (id == null)
|
|
|
|
return true;
|
2010-11-19 22:57:24 +00:00
|
|
|
final boolean modeMatched = (mode == -1 || id.mMode == mode);
|
2010-11-11 23:28:14 +00:00
|
|
|
final boolean settingsKeyMatched = (settingsKey == null
|
|
|
|
|| id.mHasSettingsKey == Boolean.parseBoolean(settingsKey));
|
|
|
|
final boolean voiceKeyMatched = (voiceKey == null
|
|
|
|
|| id.mHasVoiceKey == Boolean.parseBoolean(voiceKey));
|
|
|
|
final boolean selected = modeMatched && settingsKeyMatched && voiceKeyMatched;
|
|
|
|
if (DEBUG_TAG) {
|
|
|
|
Log.d(TAG, "parseCaseCondition: " + Boolean.toString(selected).toUpperCase()
|
2010-11-19 22:57:24 +00:00
|
|
|
+ (mode != -1 ? " mode=" + mode : "")
|
2010-11-11 23:28:14 +00:00
|
|
|
+ (settingsKey != null ? " settingsKey="+settingsKey : "")
|
|
|
|
+ (voiceKey != null ? " voiceKey=" + voiceKey : ""));
|
|
|
|
}
|
|
|
|
return selected;
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean parseDefault(XmlResourceParser parser, Row row, List<Key> keys)
|
|
|
|
throws XmlPullParserException, IOException {
|
|
|
|
if (DEBUG_PARSER) debugEnterMethod("parseDefault", keys == null);
|
|
|
|
if (row == null) {
|
|
|
|
parseKeyboardContent(parser, keys);
|
|
|
|
} else {
|
|
|
|
parseRowContent(parser, row, keys);
|
|
|
|
}
|
|
|
|
if (DEBUG_PARSER) debugLeaveMethod("parseDefault", keys == null);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private static void checkEndTag(String tag, XmlResourceParser parser)
|
2010-10-21 10:05:53 +00:00
|
|
|
throws XmlPullParserException, IOException {
|
2010-10-21 16:24:23 +00:00
|
|
|
if (parser.next() == XmlResourceParser.END_TAG && tag.equals(parser.getName()))
|
|
|
|
return;
|
|
|
|
throw new NonEmptyTag(tag, parser);
|
|
|
|
}
|
|
|
|
|
2010-11-11 23:28:14 +00:00
|
|
|
private void startRow(Row row) {
|
|
|
|
mCurrentX = 0;
|
|
|
|
mCurrentRow = row;
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 16:24:23 +00:00
|
|
|
private void endRow() {
|
|
|
|
if (mCurrentRow == null)
|
|
|
|
throw new InflateException("orphant end row tag");
|
|
|
|
mCurrentY += mCurrentRow.verticalGap + mCurrentRow.defaultHeight;
|
|
|
|
mCurrentRow = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void endKey(Key key) {
|
|
|
|
mCurrentX += key.gap + key.width;
|
|
|
|
if (mCurrentX > mMaxRowWidth)
|
|
|
|
mMaxRowWidth = mCurrentX;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void endKeyboard(int defaultVerticalGap) {
|
|
|
|
mTotalHeight = mCurrentY - defaultVerticalGap;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setSpacer(int gap) {
|
|
|
|
mCurrentX += gap;
|
2010-10-21 10:05:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static int getDimensionOrFraction(TypedArray a, int index, int base, int defValue) {
|
|
|
|
final TypedValue value = a.peekValue(index);
|
|
|
|
if (value == null)
|
|
|
|
return defValue;
|
|
|
|
if (value.type == TypedValue.TYPE_DIMENSION) {
|
|
|
|
return a.getDimensionPixelOffset(index, defValue);
|
|
|
|
} else if (value.type == TypedValue.TYPE_FRACTION) {
|
|
|
|
// Round it to avoid values like 47.9999 from getting truncated
|
|
|
|
return Math.round(a.getFraction(index, base, base, defValue));
|
|
|
|
}
|
|
|
|
return defValue;
|
|
|
|
}
|
2010-10-21 16:24:23 +00:00
|
|
|
|
|
|
|
@SuppressWarnings("serial")
|
|
|
|
private static class ParseException extends InflateException {
|
|
|
|
public ParseException(String msg, XmlResourceParser parser) {
|
|
|
|
super(msg + " at line " + parser.getLineNumber());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@SuppressWarnings("serial")
|
|
|
|
private static class IllegalStartTag extends ParseException {
|
|
|
|
public IllegalStartTag(XmlResourceParser parser, String parent) {
|
|
|
|
super("Illegal start tag " + parser.getName() + " in " + parent, parser);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@SuppressWarnings("serial")
|
|
|
|
private static class IllegalEndTag extends ParseException {
|
|
|
|
public IllegalEndTag(XmlResourceParser parser, String parent) {
|
|
|
|
super("Illegal end tag " + parser.getName() + " in " + parent, parser);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@SuppressWarnings("serial")
|
|
|
|
private static class NonEmptyTag extends ParseException {
|
|
|
|
public NonEmptyTag(String tag, XmlResourceParser parser) {
|
|
|
|
super(tag + " must be empty tag", parser);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void debugEnterMethod(String title, boolean skip) {
|
|
|
|
Log.d(TAG, title + ": enter" + (skip ? " skip" : ""));
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void debugLeaveMethod(String title, boolean skip) {
|
|
|
|
Log.d(TAG, title + ": leave" + (skip ? " skip" : ""));
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void debugStartTag(String title, String tag, boolean skip) {
|
|
|
|
Log.d(TAG, title + ": <" + tag + ">" + (skip ? " skip" : ""));
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void debugEndTag(String title, String tag, boolean skip) {
|
|
|
|
Log.d(TAG, title + ": </" + tag + ">" + (skip ? " skip" : ""));
|
|
|
|
}
|
|
|
|
}
|