2011-01-19 11:59:57 +00:00
|
|
|
/*
|
2011-05-20 03:09:57 +00:00
|
|
|
* Copyright (C) 2011 The Android Open Source Project
|
2011-01-19 11:59: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
|
|
|
|
*
|
|
|
|
* 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.keyboard;
|
|
|
|
|
2011-08-23 03:08:36 +00:00
|
|
|
import android.graphics.Paint;
|
|
|
|
|
|
|
|
import com.android.inputmethod.keyboard.internal.KeyboardBuilder;
|
|
|
|
import com.android.inputmethod.keyboard.internal.KeyboardParams;
|
|
|
|
import com.android.inputmethod.keyboard.internal.PopupCharactersParser;
|
|
|
|
import com.android.inputmethod.latin.R;
|
2011-01-19 11:59:57 +00:00
|
|
|
|
|
|
|
public class MiniKeyboard extends Keyboard {
|
2011-07-29 00:05:40 +00:00
|
|
|
private final int mDefaultKeyCoordX;
|
2011-01-19 11:59:57 +00:00
|
|
|
|
2011-08-23 03:08:36 +00:00
|
|
|
private MiniKeyboard(Builder.MiniKeyboardParams params) {
|
2011-07-29 00:05:40 +00:00
|
|
|
super(params);
|
|
|
|
mDefaultKeyCoordX = params.getDefaultKeyCoordX() + params.mDefaultKeyWidth / 2;
|
2011-01-19 11:59:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public int getDefaultCoordX() {
|
|
|
|
return mDefaultKeyCoordX;
|
|
|
|
}
|
2011-08-23 03:08:36 +00:00
|
|
|
|
|
|
|
public static class Builder extends KeyboardBuilder<Builder.MiniKeyboardParams> {
|
|
|
|
private final CharSequence[] mPopupCharacters;
|
|
|
|
|
|
|
|
public static class MiniKeyboardParams extends KeyboardParams {
|
|
|
|
/* package */int mTopRowAdjustment;
|
|
|
|
public int mNumRows;
|
|
|
|
public int mNumColumns;
|
|
|
|
public int mLeftKeys;
|
|
|
|
public int mRightKeys; // includes default key.
|
|
|
|
|
|
|
|
public MiniKeyboardParams() {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* package for test */MiniKeyboardParams(int numKeys, int maxColumns, int keyWidth,
|
|
|
|
int rowHeight, int coordXInParent, int parentKeyboardWidth) {
|
|
|
|
super();
|
|
|
|
setParameters(numKeys, maxColumns, keyWidth, rowHeight, coordXInParent,
|
|
|
|
parentKeyboardWidth);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set keyboard parameters of mini keyboard.
|
|
|
|
*
|
|
|
|
* @param numKeys number of keys in this mini keyboard.
|
|
|
|
* @param maxColumns number of maximum columns of this mini keyboard.
|
|
|
|
* @param keyWidth mini keyboard key width in pixel, including horizontal gap.
|
|
|
|
* @param rowHeight mini keyboard row height in pixel, including vertical gap.
|
|
|
|
* @param coordXInParent coordinate x of the popup key in parent keyboard.
|
|
|
|
* @param parentKeyboardWidth parent keyboard width in pixel.
|
|
|
|
*/
|
|
|
|
public void setParameters(int numKeys, int maxColumns, int keyWidth, int rowHeight,
|
|
|
|
int coordXInParent, int parentKeyboardWidth) {
|
|
|
|
if (parentKeyboardWidth / keyWidth < maxColumns) {
|
|
|
|
throw new IllegalArgumentException(
|
|
|
|
"Keyboard is too small to hold mini keyboard: " + parentKeyboardWidth
|
|
|
|
+ " " + keyWidth + " " + maxColumns);
|
|
|
|
}
|
|
|
|
mDefaultKeyWidth = keyWidth;
|
|
|
|
mDefaultRowHeight = rowHeight;
|
|
|
|
|
|
|
|
final int numRows = (numKeys + maxColumns - 1) / maxColumns;
|
|
|
|
mNumRows = numRows;
|
|
|
|
final int numColumns = getOptimizedColumns(numKeys, maxColumns);
|
|
|
|
mNumColumns = numColumns;
|
|
|
|
|
|
|
|
final int numLeftKeys = (numColumns - 1) / 2;
|
|
|
|
final int numRightKeys = numColumns - numLeftKeys; // including default key.
|
|
|
|
final int maxLeftKeys = coordXInParent / keyWidth;
|
|
|
|
final int maxRightKeys = Math.max(1, (parentKeyboardWidth - coordXInParent)
|
|
|
|
/ keyWidth);
|
|
|
|
int leftKeys, rightKeys;
|
|
|
|
if (numLeftKeys > maxLeftKeys) {
|
|
|
|
leftKeys = maxLeftKeys;
|
|
|
|
rightKeys = numColumns - maxLeftKeys;
|
|
|
|
} else if (numRightKeys > maxRightKeys) {
|
|
|
|
leftKeys = numColumns - maxRightKeys;
|
|
|
|
rightKeys = maxRightKeys;
|
|
|
|
} else {
|
|
|
|
leftKeys = numLeftKeys;
|
|
|
|
rightKeys = numRightKeys;
|
|
|
|
}
|
|
|
|
// Shift right if the left edge of mini keyboard is on the edge of parent keyboard
|
|
|
|
// unless the parent key is on the left edge.
|
|
|
|
if (leftKeys * keyWidth >= coordXInParent && leftKeys > 0) {
|
|
|
|
leftKeys--;
|
|
|
|
rightKeys++;
|
|
|
|
}
|
|
|
|
// Shift left if the right edge of mini keyboard is on the edge of parent keyboard
|
|
|
|
// unless the parent key is on the right edge.
|
|
|
|
if (rightKeys * keyWidth + coordXInParent >= parentKeyboardWidth && rightKeys > 1) {
|
|
|
|
leftKeys++;
|
|
|
|
rightKeys--;
|
|
|
|
}
|
|
|
|
mLeftKeys = leftKeys;
|
|
|
|
mRightKeys = rightKeys;
|
|
|
|
|
|
|
|
// Centering of the top row.
|
|
|
|
final boolean onEdge = (leftKeys == 0 || rightKeys == 1);
|
|
|
|
if (numRows < 2 || onEdge || getTopRowEmptySlots(numKeys, numColumns) % 2 == 0) {
|
|
|
|
mTopRowAdjustment = 0;
|
|
|
|
} else if (mLeftKeys < mRightKeys - 1) {
|
|
|
|
mTopRowAdjustment = 1;
|
|
|
|
} else {
|
|
|
|
mTopRowAdjustment = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
mWidth = mOccupiedWidth = mNumColumns * mDefaultKeyWidth;
|
|
|
|
mHeight = mOccupiedHeight = mNumRows * mDefaultRowHeight + mVerticalGap;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return key position according to column count (0 is default).
|
|
|
|
/* package */int getColumnPos(int n) {
|
|
|
|
final int col = n % mNumColumns;
|
|
|
|
if (col == 0) {
|
|
|
|
// default position.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
int pos = 0;
|
|
|
|
int right = 1; // include default position key.
|
|
|
|
int left = 0;
|
|
|
|
int i = 0;
|
|
|
|
while (true) {
|
|
|
|
// Assign right key if available.
|
|
|
|
if (right < mRightKeys) {
|
|
|
|
pos = right;
|
|
|
|
right++;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
if (i >= col)
|
|
|
|
break;
|
|
|
|
// Assign left key if available.
|
|
|
|
if (left < mLeftKeys) {
|
|
|
|
left++;
|
|
|
|
pos = -left;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
if (i >= col)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int getTopRowEmptySlots(int numKeys, int numColumns) {
|
|
|
|
final int remainingKeys = numKeys % numColumns;
|
|
|
|
if (remainingKeys == 0) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return numColumns - remainingKeys;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private int getOptimizedColumns(int numKeys, int maxColumns) {
|
|
|
|
int numColumns = Math.min(numKeys, maxColumns);
|
|
|
|
while (getTopRowEmptySlots(numKeys, numColumns) >= mNumRows) {
|
|
|
|
numColumns--;
|
|
|
|
}
|
|
|
|
return numColumns;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getDefaultKeyCoordX() {
|
|
|
|
return mLeftKeys * mDefaultKeyWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getX(int n, int row) {
|
|
|
|
final int x = getColumnPos(n) * mDefaultKeyWidth + getDefaultKeyCoordX();
|
|
|
|
if (isTopRow(row)) {
|
|
|
|
return x + mTopRowAdjustment * (mDefaultKeyWidth / 2);
|
|
|
|
}
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getY(int row) {
|
|
|
|
return (mNumRows - 1 - row) * mDefaultRowHeight + mTopPadding;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getRowFlags(int row) {
|
|
|
|
int rowFlags = 0;
|
|
|
|
if (row == 0)
|
|
|
|
rowFlags |= Keyboard.EDGE_TOP;
|
|
|
|
if (isTopRow(row))
|
|
|
|
rowFlags |= Keyboard.EDGE_BOTTOM;
|
|
|
|
return rowFlags;
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean isTopRow(int rowCount) {
|
|
|
|
return rowCount == mNumRows - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Builder(KeyboardView view, int xmlId, Key parentKey, Keyboard parentKeyboard) {
|
|
|
|
super(view.getContext(), new MiniKeyboardParams());
|
|
|
|
load(parentKeyboard.mId.cloneWithNewXml(mResources.getResourceEntryName(xmlId), xmlId));
|
|
|
|
|
|
|
|
// HACK: Current mini keyboard design totally relies on the 9-patch
|
|
|
|
// padding about horizontal
|
|
|
|
// and vertical key spacing. To keep the visual of mini keyboard as
|
|
|
|
// is, these hacks are
|
|
|
|
// needed to keep having the same horizontal and vertical key
|
|
|
|
// spacing.
|
|
|
|
mParams.mHorizontalGap = 0;
|
|
|
|
mParams.mVerticalGap = mParams.mTopPadding = parentKeyboard.mVerticalGap / 2;
|
|
|
|
// TODO: When we have correctly padded key background 9-patch
|
|
|
|
// drawables for mini keyboard,
|
|
|
|
// revert the above hacks and uncomment the following lines.
|
|
|
|
// mParams.mHorizontalGap = parentKeyboard.mHorizontalGap;
|
|
|
|
// mParams.mVerticalGap = parentKeyboard.mVerticalGap;
|
|
|
|
|
|
|
|
mParams.mIsRtlKeyboard = parentKeyboard.mIsRtlKeyboard;
|
|
|
|
mPopupCharacters = parentKey.mPopupCharacters;
|
|
|
|
|
|
|
|
final int keyWidth = getMaxKeyWidth(view, mPopupCharacters, mParams.mDefaultKeyWidth);
|
2011-08-31 05:22:53 +00:00
|
|
|
mParams.setParameters(mPopupCharacters.length, parentKey.mMaxMiniKeyboardColumn,
|
|
|
|
keyWidth, parentKeyboard.mDefaultRowHeight, parentKey.mX
|
2011-08-23 03:08:36 +00:00
|
|
|
+ (mParams.mDefaultKeyWidth - keyWidth) / 2, view.getMeasuredWidth());
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int getMaxKeyWidth(KeyboardView view, CharSequence[] popupCharacters,
|
|
|
|
int minKeyWidth) {
|
2011-08-30 07:48:38 +00:00
|
|
|
final int padding = (int) view.getContext().getResources()
|
|
|
|
.getDimension(R.dimen.mini_keyboard_key_horizontal_padding);
|
2011-08-23 03:08:36 +00:00
|
|
|
Paint paint = null;
|
2011-08-29 12:12:36 +00:00
|
|
|
int maxWidth = minKeyWidth;
|
2011-08-23 03:08:36 +00:00
|
|
|
for (CharSequence popupSpec : popupCharacters) {
|
|
|
|
final CharSequence label = PopupCharactersParser.getLabel(popupSpec.toString());
|
|
|
|
// If the label is single letter, minKeyWidth is enough to hold
|
|
|
|
// the label.
|
|
|
|
if (label != null && label.length() > 1) {
|
|
|
|
if (paint == null) {
|
|
|
|
paint = new Paint();
|
|
|
|
paint.setAntiAlias(true);
|
|
|
|
}
|
2011-08-30 07:48:38 +00:00
|
|
|
final int width = (int)view.getDefaultLabelWidth(label, paint) + padding;
|
2011-08-29 12:12:36 +00:00
|
|
|
if (maxWidth < width) {
|
|
|
|
maxWidth = width;
|
|
|
|
}
|
2011-08-23 03:08:36 +00:00
|
|
|
}
|
|
|
|
}
|
2011-08-30 07:48:38 +00:00
|
|
|
return maxWidth;
|
2011-08-23 03:08:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public MiniKeyboard build() {
|
|
|
|
final MiniKeyboardParams params = mParams;
|
|
|
|
for (int n = 0; n < mPopupCharacters.length; n++) {
|
2011-08-30 12:41:52 +00:00
|
|
|
final String popupSpec = mPopupCharacters[n].toString();
|
2011-08-23 03:08:36 +00:00
|
|
|
final int row = n / params.mNumColumns;
|
2011-08-30 12:41:52 +00:00
|
|
|
final Key key = new Key(mResources, params, popupSpec, params.getX(n, row),
|
2011-08-23 03:08:36 +00:00
|
|
|
params.getY(row), params.mDefaultKeyWidth, params.mDefaultRowHeight,
|
|
|
|
params.getRowFlags(row));
|
|
|
|
params.onAddKey(key);
|
|
|
|
}
|
|
|
|
return new MiniKeyboard(params);
|
|
|
|
}
|
|
|
|
}
|
2011-01-19 11:59:57 +00:00
|
|
|
}
|