2009-03-13 22:11:42 +00:00
|
|
|
/*
|
2010-03-26 22:07:10 +00:00
|
|
|
* Copyright (C) 2008 The Android Open Source Project
|
2011-12-20 08:52:29 +00:00
|
|
|
*
|
2009-03-13 22:11:42 +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
|
2011-12-20 08:52:29 +00:00
|
|
|
*
|
2009-03-13 22:11:42 +00:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2011-12-20 08:52:29 +00:00
|
|
|
*
|
2009-03-13 22:11:42 +00:00
|
|
|
* 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;
|
|
|
|
|
2011-11-22 02:35:40 +00:00
|
|
|
import com.android.inputmethod.keyboard.Key;
|
2011-12-17 23:36:16 +00:00
|
|
|
import com.android.inputmethod.keyboard.Keyboard;
|
2011-02-10 11:53:58 +00:00
|
|
|
|
2011-11-18 11:03:38 +00:00
|
|
|
import java.util.Arrays;
|
2009-03-13 22:11:42 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A place to store the currently composing word with information such as adjacent key codes as well
|
|
|
|
*/
|
2012-09-27 09:16:16 +00:00
|
|
|
public final class WordComposer {
|
2012-03-27 06:21:54 +00:00
|
|
|
private static final int N = BinaryDictionary.MAX_WORD_LENGTH;
|
2011-02-22 08:28:55 +00:00
|
|
|
|
2012-08-21 10:49:25 +00:00
|
|
|
public static final int CAPS_MODE_OFF = 0;
|
|
|
|
// 1 is shift bit, 2 is caps bit, 4 is auto bit but this is just a convention as these bits
|
|
|
|
// aren't used anywhere in the code
|
|
|
|
public static final int CAPS_MODE_MANUAL_SHIFTED = 0x1;
|
|
|
|
public static final int CAPS_MODE_MANUAL_SHIFT_LOCKED = 0x3;
|
|
|
|
public static final int CAPS_MODE_AUTO_SHIFTED = 0x5;
|
|
|
|
public static final int CAPS_MODE_AUTO_SHIFT_LOCKED = 0x7;
|
|
|
|
|
2012-03-27 06:21:54 +00:00
|
|
|
private int[] mPrimaryKeyCodes;
|
2012-07-18 11:31:09 +00:00
|
|
|
private final InputPointers mInputPointers = new InputPointers(N);
|
2012-07-06 11:00:22 +00:00
|
|
|
private final StringBuilder mTypedWord;
|
2012-01-26 08:29:57 +00:00
|
|
|
private CharSequence mAutoCorrection;
|
2012-04-26 09:01:13 +00:00
|
|
|
private boolean mIsResumed;
|
2012-07-10 01:46:13 +00:00
|
|
|
private boolean mIsBatchMode;
|
2009-07-23 19:17:48 +00:00
|
|
|
|
2012-01-26 08:29:57 +00:00
|
|
|
// Cache these values for performance
|
2009-07-23 19:17:48 +00:00
|
|
|
private int mCapsCount;
|
2012-07-27 14:13:28 +00:00
|
|
|
private int mDigitsCount;
|
2012-08-21 10:49:25 +00:00
|
|
|
private int mCapitalizedMode;
|
2011-11-29 05:15:41 +00:00
|
|
|
private int mTrailingSingleQuotesCount;
|
2012-03-27 06:21:54 +00:00
|
|
|
private int mCodePointSize;
|
2011-11-18 11:03:38 +00:00
|
|
|
|
2009-03-13 22:11:42 +00:00
|
|
|
/**
|
2010-09-27 15:32:35 +00:00
|
|
|
* Whether the user chose to capitalize the first char of the word.
|
2009-03-13 22:11:42 +00:00
|
|
|
*/
|
2010-09-27 15:32:35 +00:00
|
|
|
private boolean mIsFirstCharCapitalized;
|
2009-03-13 22:11:42 +00:00
|
|
|
|
2010-08-20 05:35:02 +00:00
|
|
|
public WordComposer() {
|
2012-03-27 06:21:54 +00:00
|
|
|
mPrimaryKeyCodes = new int[N];
|
2012-01-26 08:29:57 +00:00
|
|
|
mTypedWord = new StringBuilder(N);
|
|
|
|
mAutoCorrection = null;
|
2011-11-29 05:15:41 +00:00
|
|
|
mTrailingSingleQuotesCount = 0;
|
2012-04-26 09:01:13 +00:00
|
|
|
mIsResumed = false;
|
2012-07-10 01:46:13 +00:00
|
|
|
mIsBatchMode = false;
|
2012-03-27 06:21:54 +00:00
|
|
|
refreshSize();
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2011-05-11 06:19:24 +00:00
|
|
|
public WordComposer(WordComposer source) {
|
2012-03-27 06:21:54 +00:00
|
|
|
mPrimaryKeyCodes = Arrays.copyOf(source.mPrimaryKeyCodes, source.mPrimaryKeyCodes.length);
|
2012-01-26 08:29:57 +00:00
|
|
|
mTypedWord = new StringBuilder(source.mTypedWord);
|
2012-06-29 09:42:15 +00:00
|
|
|
mInputPointers.copy(source.mInputPointers);
|
2011-09-13 08:46:23 +00:00
|
|
|
mCapsCount = source.mCapsCount;
|
2012-07-27 14:13:28 +00:00
|
|
|
mDigitsCount = source.mDigitsCount;
|
2011-09-13 08:46:23 +00:00
|
|
|
mIsFirstCharCapitalized = source.mIsFirstCharCapitalized;
|
2012-08-21 10:49:25 +00:00
|
|
|
mCapitalizedMode = source.mCapitalizedMode;
|
2011-11-29 05:15:41 +00:00
|
|
|
mTrailingSingleQuotesCount = source.mTrailingSingleQuotesCount;
|
2012-04-26 09:01:13 +00:00
|
|
|
mIsResumed = source.mIsResumed;
|
2012-07-10 01:46:13 +00:00
|
|
|
mIsBatchMode = source.mIsBatchMode;
|
2012-03-27 06:21:54 +00:00
|
|
|
refreshSize();
|
2010-08-20 05:35:02 +00:00
|
|
|
}
|
|
|
|
|
2009-03-13 22:11:42 +00:00
|
|
|
/**
|
|
|
|
* Clear out the keys registered so far.
|
|
|
|
*/
|
|
|
|
public void reset() {
|
2012-01-26 08:29:57 +00:00
|
|
|
mTypedWord.setLength(0);
|
|
|
|
mAutoCorrection = null;
|
2009-07-23 19:17:48 +00:00
|
|
|
mCapsCount = 0;
|
2012-07-27 14:13:28 +00:00
|
|
|
mDigitsCount = 0;
|
2011-09-13 08:46:23 +00:00
|
|
|
mIsFirstCharCapitalized = false;
|
2011-11-29 05:15:41 +00:00
|
|
|
mTrailingSingleQuotesCount = 0;
|
2012-04-26 09:01:13 +00:00
|
|
|
mIsResumed = false;
|
2012-07-10 01:46:13 +00:00
|
|
|
mIsBatchMode = false;
|
2012-03-27 06:21:54 +00:00
|
|
|
refreshSize();
|
|
|
|
}
|
|
|
|
|
2012-06-27 05:35:24 +00:00
|
|
|
private final void refreshSize() {
|
2012-03-27 06:21:54 +00:00
|
|
|
mCodePointSize = mTypedWord.codePointCount(0, mTypedWord.length());
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of keystrokes in the composing word.
|
|
|
|
* @return the number of keystrokes
|
|
|
|
*/
|
2011-09-13 08:46:23 +00:00
|
|
|
public final int size() {
|
2012-03-27 06:21:54 +00:00
|
|
|
return mCodePointSize;
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2011-12-13 14:24:37 +00:00
|
|
|
public final boolean isComposingWord() {
|
2012-03-27 06:21:54 +00:00
|
|
|
return size() > 0;
|
2011-12-13 14:24:37 +00:00
|
|
|
}
|
|
|
|
|
2012-03-28 02:32:11 +00:00
|
|
|
// TODO: make sure that the index should not exceed MAX_WORD_LENGTH
|
2012-03-27 06:21:54 +00:00
|
|
|
public int getCodeAt(int index) {
|
2012-03-28 02:32:11 +00:00
|
|
|
if (index >= BinaryDictionary.MAX_WORD_LENGTH) {
|
|
|
|
return -1;
|
|
|
|
}
|
2012-03-27 06:21:54 +00:00
|
|
|
return mPrimaryKeyCodes[index];
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2012-06-29 09:42:15 +00:00
|
|
|
public InputPointers getInputPointers() {
|
|
|
|
return mInputPointers;
|
2011-02-22 08:28:55 +00:00
|
|
|
}
|
|
|
|
|
2011-09-13 08:46:23 +00:00
|
|
|
private static boolean isFirstCharCapitalized(int index, int codePoint, boolean previous) {
|
|
|
|
if (index == 0) return Character.isUpperCase(codePoint);
|
2011-09-16 03:28:13 +00:00
|
|
|
return previous && !Character.isUpperCase(codePoint);
|
2011-09-13 08:46:23 +00:00
|
|
|
}
|
|
|
|
|
2009-03-13 22:11:42 +00:00
|
|
|
/**
|
2012-03-29 06:07:53 +00:00
|
|
|
* Add a new keystroke, with the pressed key's code point with the touch point coordinates.
|
2009-03-13 22:11:42 +00:00
|
|
|
*/
|
2012-07-04 07:55:51 +00:00
|
|
|
public void add(int primaryCode, int keyX, int keyY) {
|
2012-03-27 06:21:54 +00:00
|
|
|
final int newIndex = size();
|
2012-02-03 07:05:48 +00:00
|
|
|
mTypedWord.appendCodePoint(primaryCode);
|
2012-03-27 06:21:54 +00:00
|
|
|
refreshSize();
|
2011-09-13 08:46:23 +00:00
|
|
|
if (newIndex < BinaryDictionary.MAX_WORD_LENGTH) {
|
2012-03-28 07:30:52 +00:00
|
|
|
mPrimaryKeyCodes[newIndex] = primaryCode >= Keyboard.CODE_SPACE
|
|
|
|
? Character.toLowerCase(primaryCode) : primaryCode;
|
2012-06-12 10:40:37 +00:00
|
|
|
// In the batch input mode, the {@code mInputPointers} holds batch input points and
|
|
|
|
// shouldn't be overridden by the "typed key" coordinates
|
|
|
|
// (See {@link #setBatchInputWord}).
|
|
|
|
if (!mIsBatchMode) {
|
|
|
|
// TODO: Set correct pointer id and time
|
|
|
|
mInputPointers.addPointer(newIndex, keyX, keyY, 0, 0);
|
|
|
|
}
|
2011-02-22 08:28:55 +00:00
|
|
|
}
|
2011-09-13 08:46:23 +00:00
|
|
|
mIsFirstCharCapitalized = isFirstCharCapitalized(
|
|
|
|
newIndex, primaryCode, mIsFirstCharCapitalized);
|
|
|
|
if (Character.isUpperCase(primaryCode)) mCapsCount++;
|
2012-07-27 14:13:28 +00:00
|
|
|
if (Character.isDigit(primaryCode)) mDigitsCount++;
|
2011-11-29 05:15:41 +00:00
|
|
|
if (Keyboard.CODE_SINGLE_QUOTE == primaryCode) {
|
|
|
|
++mTrailingSingleQuotesCount;
|
|
|
|
} else {
|
|
|
|
mTrailingSingleQuotesCount = 0;
|
|
|
|
}
|
2012-01-26 08:29:57 +00:00
|
|
|
mAutoCorrection = null;
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2012-07-10 01:46:13 +00:00
|
|
|
public void setBatchInputPointers(InputPointers batchPointers) {
|
2012-06-12 10:40:37 +00:00
|
|
|
mInputPointers.set(batchPointers);
|
2012-07-10 01:46:13 +00:00
|
|
|
mIsBatchMode = true;
|
|
|
|
}
|
|
|
|
|
2012-06-12 10:40:37 +00:00
|
|
|
public void setBatchInputWord(CharSequence word) {
|
|
|
|
reset();
|
|
|
|
mIsBatchMode = true;
|
|
|
|
final int length = word.length();
|
|
|
|
for (int i = 0; i < length; i = Character.offsetByCodePoints(word, i, 1)) {
|
|
|
|
final int codePoint = Character.codePointAt(word, i);
|
|
|
|
// We don't want to override the batch input points that are held in mInputPointers
|
|
|
|
// (See {@link #add(int,int,int)}).
|
2012-08-21 05:05:57 +00:00
|
|
|
add(codePoint, Constants.NOT_A_COORDINATE, Constants.NOT_A_COORDINATE);
|
2012-06-12 10:40:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-22 02:35:40 +00:00
|
|
|
/**
|
|
|
|
* Internal method to retrieve reasonable proximity info for a character.
|
|
|
|
*/
|
2012-03-23 08:55:11 +00:00
|
|
|
private void addKeyInfo(final int codePoint, final Keyboard keyboard) {
|
2012-07-04 07:31:57 +00:00
|
|
|
final Key key = keyboard.getKey(codePoint);
|
|
|
|
if (key != null) {
|
|
|
|
final int x = key.mX + key.mWidth / 2;
|
|
|
|
final int y = key.mY + key.mHeight / 2;
|
|
|
|
add(codePoint, x, y);
|
|
|
|
return;
|
2011-11-22 02:35:40 +00:00
|
|
|
}
|
2012-08-21 05:05:57 +00:00
|
|
|
add(codePoint, Constants.NOT_A_COORDINATE, Constants.NOT_A_COORDINATE);
|
2011-11-22 02:35:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the currently composing word to the one passed as an argument.
|
|
|
|
* This will register NOT_A_COORDINATE for X and Ys, and use the passed keyboard for proximity.
|
|
|
|
*/
|
2012-03-23 08:55:11 +00:00
|
|
|
public void setComposingWord(final CharSequence word, final Keyboard keyboard) {
|
2011-11-22 02:35:40 +00:00
|
|
|
reset();
|
|
|
|
final int length = word.length();
|
2012-02-03 07:05:48 +00:00
|
|
|
for (int i = 0; i < length; i = Character.offsetByCodePoints(word, i, 1)) {
|
|
|
|
int codePoint = Character.codePointAt(word, i);
|
2012-03-23 08:55:11 +00:00
|
|
|
addKeyInfo(codePoint, keyboard);
|
2011-11-22 02:35:40 +00:00
|
|
|
}
|
2012-04-26 09:01:13 +00:00
|
|
|
mIsResumed = true;
|
2011-11-22 02:35:40 +00:00
|
|
|
}
|
|
|
|
|
2009-03-13 22:11:42 +00:00
|
|
|
/**
|
|
|
|
* Delete the last keystroke as a result of hitting backspace.
|
|
|
|
*/
|
|
|
|
public void deleteLast() {
|
2012-03-27 06:21:54 +00:00
|
|
|
final int size = size();
|
2011-09-13 08:46:23 +00:00
|
|
|
if (size > 0) {
|
2012-02-03 07:05:48 +00:00
|
|
|
// Note: mTypedWord.length() and mCodes.length differ when there are surrogate pairs
|
|
|
|
final int stringBuilderLength = mTypedWord.length();
|
|
|
|
if (stringBuilderLength < size) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"In WordComposer: mCodes and mTypedWords have non-matching lengths");
|
|
|
|
}
|
|
|
|
final int lastChar = mTypedWord.codePointBefore(stringBuilderLength);
|
|
|
|
if (Character.isSupplementaryCodePoint(lastChar)) {
|
|
|
|
mTypedWord.delete(stringBuilderLength - 2, stringBuilderLength);
|
|
|
|
} else {
|
|
|
|
mTypedWord.deleteCharAt(stringBuilderLength - 1);
|
|
|
|
}
|
2011-09-13 08:46:23 +00:00
|
|
|
if (Character.isUpperCase(lastChar)) mCapsCount--;
|
2012-07-27 14:13:28 +00:00
|
|
|
if (Character.isDigit(lastChar)) mDigitsCount--;
|
2012-03-27 06:21:54 +00:00
|
|
|
refreshSize();
|
2010-09-23 03:31:22 +00:00
|
|
|
}
|
2012-02-03 07:05:48 +00:00
|
|
|
// We may have deleted the last one.
|
2012-03-27 06:21:54 +00:00
|
|
|
if (0 == size()) {
|
2011-09-13 08:46:23 +00:00
|
|
|
mIsFirstCharCapitalized = false;
|
2011-11-29 05:15:41 +00:00
|
|
|
}
|
|
|
|
if (mTrailingSingleQuotesCount > 0) {
|
|
|
|
--mTrailingSingleQuotesCount;
|
2011-11-18 11:03:38 +00:00
|
|
|
} else {
|
2012-02-03 03:22:02 +00:00
|
|
|
int i = mTypedWord.length();
|
|
|
|
while (i > 0) {
|
|
|
|
i = mTypedWord.offsetByCodePoints(i, -1);
|
2012-01-26 08:29:57 +00:00
|
|
|
if (Keyboard.CODE_SINGLE_QUOTE != mTypedWord.codePointAt(i)) break;
|
2011-11-29 05:15:41 +00:00
|
|
|
++mTrailingSingleQuotesCount;
|
|
|
|
}
|
2011-02-22 08:28:55 +00:00
|
|
|
}
|
2012-01-26 08:29:57 +00:00
|
|
|
mAutoCorrection = null;
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the word as it was typed, without any correction applied.
|
2011-12-13 10:38:36 +00:00
|
|
|
* @return the word that was typed so far. Never returns null.
|
2009-03-13 22:11:42 +00:00
|
|
|
*/
|
2011-09-15 06:42:21 +00:00
|
|
|
public String getTypedWord() {
|
2012-01-26 08:29:57 +00:00
|
|
|
return mTypedWord.toString();
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether or not the user typed a capital letter as the first letter in the word
|
|
|
|
* @return capitalization preference
|
|
|
|
*/
|
2010-09-27 15:32:35 +00:00
|
|
|
public boolean isFirstCharCapitalized() {
|
|
|
|
return mIsFirstCharCapitalized;
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
2010-09-27 15:32:35 +00:00
|
|
|
|
2011-11-29 05:15:41 +00:00
|
|
|
public int trailingSingleQuotesCount() {
|
|
|
|
return mTrailingSingleQuotesCount;
|
2011-11-18 11:03:38 +00:00
|
|
|
}
|
|
|
|
|
2010-09-27 15:32:35 +00:00
|
|
|
/**
|
|
|
|
* Whether or not all of the user typed chars are upper case
|
|
|
|
* @return true if all user typed chars are upper case, false otherwise
|
|
|
|
*/
|
|
|
|
public boolean isAllUpperCase() {
|
2012-09-19 03:52:06 +00:00
|
|
|
if (size() <= 1) {
|
|
|
|
return mCapitalizedMode == CAPS_MODE_AUTO_SHIFT_LOCKED
|
|
|
|
|| mCapitalizedMode == CAPS_MODE_MANUAL_SHIFT_LOCKED;
|
|
|
|
} else {
|
|
|
|
return mCapsCount == size();
|
|
|
|
}
|
2012-08-21 10:57:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean wasShiftedNoLock() {
|
|
|
|
return mCapitalizedMode == CAPS_MODE_AUTO_SHIFTED
|
|
|
|
|| mCapitalizedMode == CAPS_MODE_MANUAL_SHIFTED;
|
2010-09-27 15:32:35 +00:00
|
|
|
}
|
|
|
|
|
2009-07-23 19:17:48 +00:00
|
|
|
/**
|
|
|
|
* Returns true if more than one character is upper case, otherwise returns false.
|
|
|
|
*/
|
|
|
|
public boolean isMostlyCaps() {
|
|
|
|
return mCapsCount > 1;
|
|
|
|
}
|
2010-01-24 15:34:07 +00:00
|
|
|
|
2012-07-27 14:13:28 +00:00
|
|
|
/**
|
|
|
|
* Returns true if we have digits in the composing word.
|
|
|
|
*/
|
|
|
|
public boolean hasDigits() {
|
|
|
|
return mDigitsCount > 0;
|
|
|
|
}
|
|
|
|
|
2011-12-20 08:52:29 +00:00
|
|
|
/**
|
2012-08-21 10:49:25 +00:00
|
|
|
* Saves the caps mode at the start of composing.
|
|
|
|
*
|
|
|
|
* WordComposer needs to know about this for several reasons. The first is, we need to know
|
|
|
|
* after the fact what the reason was, to register the correct form into the user history
|
|
|
|
* dictionary: if the word was automatically capitalized, we should insert it in all-lower
|
|
|
|
* case but if it's a manual pressing of shift, then it should be inserted as is.
|
|
|
|
* Also, batch input needs to know about the current caps mode to display correctly
|
|
|
|
* capitalized suggestions.
|
|
|
|
* @param mode the mode at the time of start
|
2010-01-24 15:34:07 +00:00
|
|
|
*/
|
2012-08-21 10:49:25 +00:00
|
|
|
public void setCapitalizedModeAtStartComposingTime(final int mode) {
|
|
|
|
mCapitalizedMode = mode;
|
2010-01-24 15:34:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns whether the word was automatically capitalized.
|
|
|
|
* @return whether the word was automatically capitalized
|
|
|
|
*/
|
2012-08-21 10:49:25 +00:00
|
|
|
public boolean wasAutoCapitalized() {
|
|
|
|
return mCapitalizedMode == CAPS_MODE_AUTO_SHIFT_LOCKED
|
|
|
|
|| mCapitalizedMode == CAPS_MODE_AUTO_SHIFTED;
|
2010-01-24 15:34:07 +00:00
|
|
|
}
|
2011-12-13 10:38:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the auto-correction for this word.
|
|
|
|
*/
|
|
|
|
public void setAutoCorrection(final CharSequence correction) {
|
2012-01-26 08:29:57 +00:00
|
|
|
mAutoCorrection = correction;
|
2011-12-13 10:38:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-12-13 14:08:12 +00:00
|
|
|
* @return the auto-correction for this word, or null if none.
|
2011-12-13 10:38:36 +00:00
|
|
|
*/
|
|
|
|
public CharSequence getAutoCorrectionOrNull() {
|
2012-01-26 08:29:57 +00:00
|
|
|
return mAutoCorrection;
|
2011-12-13 10:38:36 +00:00
|
|
|
}
|
2011-12-13 14:12:22 +00:00
|
|
|
|
2012-04-26 09:01:13 +00:00
|
|
|
/**
|
|
|
|
* @return whether we started composing this word by resuming suggestion on an existing string
|
|
|
|
*/
|
|
|
|
public boolean isResumed() {
|
|
|
|
return mIsResumed;
|
|
|
|
}
|
|
|
|
|
2012-01-26 06:52:55 +00:00
|
|
|
// `type' should be one of the LastComposedWord.COMMIT_TYPE_* constants above.
|
2012-02-22 07:17:54 +00:00
|
|
|
public LastComposedWord commitWord(final int type, final String committedWord,
|
2012-08-27 11:09:46 +00:00
|
|
|
final String separatorString, final CharSequence prevWord) {
|
2012-02-22 08:05:19 +00:00
|
|
|
// Note: currently, we come here whenever we commit a word. If it's a MANUAL_PICK
|
|
|
|
// or a DECIDED_WORD we may cancel the commit later; otherwise, we should deactivate
|
|
|
|
// the last composed word to ensure this does not happen.
|
2012-03-27 06:21:54 +00:00
|
|
|
final int[] primaryKeyCodes = mPrimaryKeyCodes;
|
|
|
|
mPrimaryKeyCodes = new int[N];
|
|
|
|
final LastComposedWord lastComposedWord = new LastComposedWord(primaryKeyCodes,
|
2012-08-27 11:09:46 +00:00
|
|
|
mInputPointers, mTypedWord.toString(), committedWord, separatorString,
|
2012-05-24 03:11:02 +00:00
|
|
|
prevWord);
|
2012-06-29 09:42:15 +00:00
|
|
|
mInputPointers.reset();
|
2012-02-22 08:05:19 +00:00
|
|
|
if (type != LastComposedWord.COMMIT_TYPE_DECIDED_WORD
|
|
|
|
&& type != LastComposedWord.COMMIT_TYPE_MANUAL_PICK) {
|
2012-01-26 10:05:59 +00:00
|
|
|
lastComposedWord.deactivate();
|
|
|
|
}
|
2012-07-13 04:31:27 +00:00
|
|
|
mCapsCount = 0;
|
2012-07-27 14:13:28 +00:00
|
|
|
mDigitsCount = 0;
|
2012-07-13 04:31:27 +00:00
|
|
|
mIsBatchMode = false;
|
2012-01-26 08:29:57 +00:00
|
|
|
mTypedWord.setLength(0);
|
2012-07-10 09:43:42 +00:00
|
|
|
mTrailingSingleQuotesCount = 0;
|
2012-07-13 04:31:27 +00:00
|
|
|
mIsFirstCharCapitalized = false;
|
2012-03-27 06:21:54 +00:00
|
|
|
refreshSize();
|
2012-01-26 08:29:57 +00:00
|
|
|
mAutoCorrection = null;
|
2012-04-26 09:01:13 +00:00
|
|
|
mIsResumed = false;
|
2012-01-26 07:05:09 +00:00
|
|
|
return lastComposedWord;
|
2011-12-13 14:12:22 +00:00
|
|
|
}
|
|
|
|
|
2012-01-26 07:53:38 +00:00
|
|
|
public void resumeSuggestionOnLastComposedWord(final LastComposedWord lastComposedWord) {
|
2012-03-27 06:21:54 +00:00
|
|
|
mPrimaryKeyCodes = lastComposedWord.mPrimaryKeyCodes;
|
2012-06-29 09:42:15 +00:00
|
|
|
mInputPointers.set(lastComposedWord.mInputPointers);
|
2012-01-26 08:29:57 +00:00
|
|
|
mTypedWord.setLength(0);
|
|
|
|
mTypedWord.append(lastComposedWord.mTypedWord);
|
2012-03-27 06:21:54 +00:00
|
|
|
refreshSize();
|
2012-02-22 07:11:07 +00:00
|
|
|
mAutoCorrection = null; // This will be filled by the next call to updateSuggestion.
|
2012-04-26 09:01:13 +00:00
|
|
|
mIsResumed = true;
|
2011-12-14 11:13:16 +00:00
|
|
|
}
|
2012-07-10 01:46:13 +00:00
|
|
|
|
|
|
|
public boolean isBatchMode() {
|
|
|
|
return mIsBatchMode;
|
|
|
|
}
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|