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-12-14 11:13:16 +00:00
|
|
|
import android.text.TextUtils;
|
|
|
|
|
2011-11-22 02:35:40 +00:00
|
|
|
import com.android.inputmethod.keyboard.Key;
|
2011-02-10 11:53:58 +00:00
|
|
|
import com.android.inputmethod.keyboard.KeyDetector;
|
2011-12-17 23:36:16 +00:00
|
|
|
import com.android.inputmethod.keyboard.Keyboard;
|
2011-02-10 11:53:58 +00:00
|
|
|
|
2009-03-13 22:11:42 +00:00
|
|
|
import java.util.ArrayList;
|
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
|
|
|
|
*/
|
|
|
|
public class WordComposer {
|
2011-02-22 08:28:55 +00:00
|
|
|
|
2011-02-10 11:53:58 +00:00
|
|
|
public static final int NOT_A_CODE = KeyDetector.NOT_A_CODE;
|
2011-02-22 08:28:55 +00:00
|
|
|
public static final int NOT_A_COORDINATE = -1;
|
2011-02-10 11:53:58 +00:00
|
|
|
|
2011-12-13 13:20:20 +00:00
|
|
|
// Storage for all the info about the current input.
|
|
|
|
private static class CharacterStore {
|
|
|
|
/**
|
|
|
|
* The list of unicode values for each keystroke (including surrounding keys)
|
|
|
|
*/
|
|
|
|
ArrayList<int[]> mCodes;
|
|
|
|
int[] mXCoordinates;
|
|
|
|
int[] mYCoordinates;
|
|
|
|
StringBuilder mTypedWord;
|
2011-12-14 11:13:16 +00:00
|
|
|
CharSequence mAutoCorrection;
|
2011-12-13 13:20:20 +00:00
|
|
|
CharacterStore() {
|
|
|
|
final int N = BinaryDictionary.MAX_WORD_LENGTH;
|
|
|
|
mCodes = new ArrayList<int[]>(N);
|
|
|
|
mTypedWord = new StringBuilder(N);
|
|
|
|
mXCoordinates = new int[N];
|
|
|
|
mYCoordinates = new int[N];
|
2011-12-14 11:13:16 +00:00
|
|
|
mAutoCorrection = null;
|
2011-12-13 13:20:20 +00:00
|
|
|
}
|
|
|
|
CharacterStore(final CharacterStore that) {
|
|
|
|
mCodes = new ArrayList<int[]>(that.mCodes);
|
|
|
|
mTypedWord = new StringBuilder(that.mTypedWord);
|
|
|
|
mXCoordinates = Arrays.copyOf(that.mXCoordinates, that.mXCoordinates.length);
|
|
|
|
mYCoordinates = Arrays.copyOf(that.mYCoordinates, that.mYCoordinates.length);
|
|
|
|
}
|
|
|
|
void reset() {
|
|
|
|
// For better performance than creating a new character store.
|
|
|
|
mCodes.clear();
|
|
|
|
mTypedWord.setLength(0);
|
2011-12-14 11:13:16 +00:00
|
|
|
mAutoCorrection = null;
|
2011-12-13 13:20:20 +00:00
|
|
|
}
|
|
|
|
}
|
2011-02-22 08:28:55 +00:00
|
|
|
|
2011-12-13 14:12:22 +00:00
|
|
|
// The currently typing word. May not be null.
|
2011-12-13 13:20:20 +00:00
|
|
|
private CharacterStore mCurrentWord;
|
2009-07-23 19:17:48 +00:00
|
|
|
|
|
|
|
private int mCapsCount;
|
2010-01-24 15:34:07 +00:00
|
|
|
|
|
|
|
private boolean mAutoCapitalized;
|
2011-11-18 11:03:38 +00:00
|
|
|
// Cache this value for performance
|
2011-11-29 05:15:41 +00:00
|
|
|
private int mTrailingSingleQuotesCount;
|
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() {
|
2011-12-13 13:20:20 +00:00
|
|
|
mCurrentWord = new CharacterStore();
|
2011-11-29 05:15:41 +00:00
|
|
|
mTrailingSingleQuotesCount = 0;
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2011-05-11 06:19:24 +00:00
|
|
|
public WordComposer(WordComposer source) {
|
2011-05-10 07:40:28 +00:00
|
|
|
init(source);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void init(WordComposer source) {
|
2011-12-13 13:20:20 +00:00
|
|
|
mCurrentWord = new CharacterStore(source.mCurrentWord);
|
2011-09-13 08:46:23 +00:00
|
|
|
mCapsCount = source.mCapsCount;
|
|
|
|
mIsFirstCharCapitalized = source.mIsFirstCharCapitalized;
|
|
|
|
mAutoCapitalized = source.mAutoCapitalized;
|
2011-11-29 05:15:41 +00:00
|
|
|
mTrailingSingleQuotesCount = source.mTrailingSingleQuotesCount;
|
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() {
|
2011-12-13 13:20:20 +00:00
|
|
|
mCurrentWord.reset();
|
2009-07-23 19:17:48 +00:00
|
|
|
mCapsCount = 0;
|
2011-09-13 08:46:23 +00:00
|
|
|
mIsFirstCharCapitalized = false;
|
2011-11-29 05:15:41 +00:00
|
|
|
mTrailingSingleQuotesCount = 0;
|
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() {
|
2011-12-13 13:20:20 +00:00
|
|
|
return mCurrentWord.mTypedWord.length();
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2011-12-13 14:24:37 +00:00
|
|
|
public final boolean isComposingWord() {
|
|
|
|
return size() > 0;
|
|
|
|
}
|
|
|
|
|
2009-03-13 22:11:42 +00:00
|
|
|
/**
|
|
|
|
* Returns the codes at a particular position in the word.
|
|
|
|
* @param index the position in the word
|
|
|
|
* @return the unicode for the pressed and surrounding keys
|
|
|
|
*/
|
|
|
|
public int[] getCodesAt(int index) {
|
2011-12-13 13:20:20 +00:00
|
|
|
return mCurrentWord.mCodes.get(index);
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2011-02-22 08:28:55 +00:00
|
|
|
public int[] getXCoordinates() {
|
2011-12-13 13:20:20 +00:00
|
|
|
return mCurrentWord.mXCoordinates;
|
2011-02-22 08:28:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public int[] getYCoordinates() {
|
2011-12-13 13:20:20 +00:00
|
|
|
return mCurrentWord.mYCoordinates;
|
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
|
|
|
/**
|
|
|
|
* Add a new keystroke, with codes[0] containing the pressed key's unicode and the rest of
|
|
|
|
* the array containing unicode for adjacent keys, sorted by reducing probability/proximity.
|
|
|
|
* @param codes the array of unicode values
|
|
|
|
*/
|
2011-02-22 08:28:55 +00:00
|
|
|
public void add(int primaryCode, int[] codes, int x, int y) {
|
2011-09-13 08:46:23 +00:00
|
|
|
final int newIndex = size();
|
2011-12-13 13:20:20 +00:00
|
|
|
mCurrentWord.mTypedWord.append((char) primaryCode);
|
2010-03-15 05:53:16 +00:00
|
|
|
correctPrimaryJuxtapos(primaryCode, codes);
|
2011-12-13 13:20:20 +00:00
|
|
|
mCurrentWord.mCodes.add(codes);
|
2011-09-13 08:46:23 +00:00
|
|
|
if (newIndex < BinaryDictionary.MAX_WORD_LENGTH) {
|
2011-12-13 13:20:20 +00:00
|
|
|
mCurrentWord.mXCoordinates[newIndex] = x;
|
|
|
|
mCurrentWord.mYCoordinates[newIndex] = y;
|
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++;
|
2011-11-29 05:15:41 +00:00
|
|
|
if (Keyboard.CODE_SINGLE_QUOTE == primaryCode) {
|
|
|
|
++mTrailingSingleQuotesCount;
|
|
|
|
} else {
|
|
|
|
mTrailingSingleQuotesCount = 0;
|
|
|
|
}
|
2011-12-14 11:13:16 +00:00
|
|
|
mCurrentWord.mAutoCorrection = null;
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|
|
|
|
|
2011-11-22 02:35:40 +00:00
|
|
|
/**
|
|
|
|
* Internal method to retrieve reasonable proximity info for a character.
|
|
|
|
*/
|
2011-12-17 23:36:16 +00:00
|
|
|
private void addKeyInfo(final int codePoint, final Keyboard keyboard,
|
2011-11-22 02:35:40 +00:00
|
|
|
final KeyDetector keyDetector) {
|
|
|
|
for (final Key key : keyboard.mKeys) {
|
|
|
|
if (key.mCode == codePoint) {
|
|
|
|
final int x = key.mX + key.mWidth / 2;
|
|
|
|
final int y = key.mY + key.mHeight / 2;
|
|
|
|
final int[] codes = keyDetector.newCodeArray();
|
2011-11-29 07:56:27 +00:00
|
|
|
keyDetector.getKeyAndNearbyCodes(x, y, codes);
|
2011-11-22 02:35:40 +00:00
|
|
|
add(codePoint, codes, x, y);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
add(codePoint, new int[] { codePoint },
|
|
|
|
WordComposer.NOT_A_COORDINATE, WordComposer.NOT_A_COORDINATE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2011-12-17 23:36:16 +00:00
|
|
|
public void setComposingWord(final CharSequence word, final Keyboard keyboard,
|
2011-11-22 02:35:40 +00:00
|
|
|
final KeyDetector keyDetector) {
|
|
|
|
reset();
|
|
|
|
final int length = word.length();
|
|
|
|
for (int i = 0; i < length; ++i) {
|
|
|
|
int codePoint = word.charAt(i);
|
|
|
|
addKeyInfo(codePoint, keyboard, keyDetector);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shortcut for the above method, this will create a new KeyDetector for the passed keyboard.
|
|
|
|
*/
|
2011-12-17 23:36:16 +00:00
|
|
|
public void setComposingWord(final CharSequence word, final Keyboard keyboard) {
|
2011-11-22 02:35:40 +00:00
|
|
|
final KeyDetector keyDetector = new KeyDetector(0);
|
|
|
|
keyDetector.setKeyboard(keyboard, 0, 0);
|
|
|
|
keyDetector.setProximityCorrectionEnabled(true);
|
|
|
|
keyDetector.setProximityThreshold(keyboard.mMostCommonKeyWidth);
|
|
|
|
setComposingWord(word, keyboard, keyDetector);
|
|
|
|
}
|
|
|
|
|
2010-03-15 05:53:16 +00:00
|
|
|
/**
|
|
|
|
* Swaps the first and second values in the codes array if the primary code is not the first
|
|
|
|
* value in the array but the second. This happens when the preferred key is not the key that
|
|
|
|
* the user released the finger on.
|
|
|
|
* @param primaryCode the preferred character
|
|
|
|
* @param codes array of codes based on distance from touch point
|
|
|
|
*/
|
2011-10-28 04:31:31 +00:00
|
|
|
private static void correctPrimaryJuxtapos(int primaryCode, int[] codes) {
|
2010-03-15 05:53:16 +00:00
|
|
|
if (codes.length < 2) return;
|
|
|
|
if (codes[0] > 0 && codes[1] > 0 && codes[0] != primaryCode && codes[1] == primaryCode) {
|
|
|
|
codes[1] = codes[0];
|
|
|
|
codes[0] = primaryCode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-13 22:11:42 +00:00
|
|
|
/**
|
|
|
|
* Delete the last keystroke as a result of hitting backspace.
|
|
|
|
*/
|
|
|
|
public void deleteLast() {
|
2011-09-13 08:46:23 +00:00
|
|
|
final int size = size();
|
|
|
|
if (size > 0) {
|
|
|
|
final int lastPos = size - 1;
|
2011-12-13 13:20:20 +00:00
|
|
|
char lastChar = mCurrentWord.mTypedWord.charAt(lastPos);
|
|
|
|
mCurrentWord.mCodes.remove(lastPos);
|
|
|
|
mCurrentWord.mTypedWord.deleteCharAt(lastPos);
|
2011-09-13 08:46:23 +00:00
|
|
|
if (Character.isUpperCase(lastChar)) mCapsCount--;
|
2010-09-23 03:31:22 +00:00
|
|
|
}
|
2011-09-13 08:46:23 +00:00
|
|
|
if (size() == 0) {
|
|
|
|
mIsFirstCharCapitalized = false;
|
2011-11-29 05:15:41 +00:00
|
|
|
}
|
|
|
|
if (mTrailingSingleQuotesCount > 0) {
|
|
|
|
--mTrailingSingleQuotesCount;
|
2011-11-18 11:03:38 +00:00
|
|
|
} else {
|
2011-12-13 13:20:20 +00:00
|
|
|
for (int i = mCurrentWord.mTypedWord.length() - 1; i >= 0; --i) {
|
|
|
|
if (Keyboard.CODE_SINGLE_QUOTE != mCurrentWord.mTypedWord.codePointAt(i)) break;
|
2011-11-29 05:15:41 +00:00
|
|
|
++mTrailingSingleQuotesCount;
|
|
|
|
}
|
2011-02-22 08:28:55 +00:00
|
|
|
}
|
2011-12-14 11:13:16 +00:00
|
|
|
mCurrentWord.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() {
|
2011-12-13 13:20:20 +00:00
|
|
|
return mCurrentWord.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() {
|
|
|
|
return (mCapsCount > 0) && (mCapsCount == size());
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2011-12-20 08:52:29 +00:00
|
|
|
/**
|
2010-01-24 15:34:07 +00:00
|
|
|
* Saves the reason why the word is capitalized - whether it was automatic or
|
|
|
|
* due to the user hitting shift in the middle of a sentence.
|
|
|
|
* @param auto whether it was an automatic capitalization due to start of sentence
|
|
|
|
*/
|
|
|
|
public void setAutoCapitalized(boolean auto) {
|
|
|
|
mAutoCapitalized = auto;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns whether the word was automatically capitalized.
|
|
|
|
* @return whether the word was automatically capitalized
|
|
|
|
*/
|
|
|
|
public boolean isAutoCapitalized() {
|
|
|
|
return mAutoCapitalized;
|
|
|
|
}
|
2011-12-13 10:38:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the auto-correction for this word.
|
|
|
|
*/
|
|
|
|
public void setAutoCorrection(final CharSequence correction) {
|
2011-12-14 11:13:16 +00:00
|
|
|
mCurrentWord.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() {
|
2011-12-14 11:13:16 +00:00
|
|
|
return mCurrentWord.mAutoCorrection;
|
2011-12-13 10:38:36 +00:00
|
|
|
}
|
2011-12-13 14:12:22 +00:00
|
|
|
|
2012-01-26 06:52:55 +00:00
|
|
|
// `type' should be one of the LastComposedWord.COMMIT_TYPE_* constants above.
|
2012-01-26 07:37:47 +00:00
|
|
|
public LastComposedWord commitWord(final int type) {
|
2011-12-21 13:34:08 +00:00
|
|
|
// Note: currently, we come here whenever we commit a word. If it's any *other* kind than
|
2011-12-20 08:52:29 +00:00
|
|
|
// DECIDED_WORD, we should reset mAutoCorrection so that we don't attempt to cancel later.
|
|
|
|
// If it's a DECIDED_WORD, it may be an actual auto-correction by the IME, or what the user
|
|
|
|
// typed because the IME decided *not* to auto-correct for whatever reason.
|
|
|
|
// Ideally we would also null it when it was a DECIDED_WORD that was not an auto-correct.
|
|
|
|
// As it happens these two cases should behave differently, because the former can be
|
|
|
|
// canceled while the latter can't. Currently, we figure this out in
|
2012-01-26 07:53:38 +00:00
|
|
|
// LastComposedWord#didAutoCorrectToAnotherWord with #equals(). It would be marginally
|
|
|
|
// cleaner to do it here, but it would be slower (since we would #equals() for each commit,
|
|
|
|
// instead of only on cancel), and ultimately we want to figure it out even earlier anyway.
|
2012-01-26 07:05:09 +00:00
|
|
|
final LastComposedWord lastComposedWord = new LastComposedWord(type, mCurrentWord.mCodes,
|
|
|
|
mCurrentWord.mXCoordinates, mCurrentWord.mYCoordinates,
|
2012-01-26 07:37:47 +00:00
|
|
|
mCurrentWord.mTypedWord.toString(),
|
2012-01-26 07:53:38 +00:00
|
|
|
(type != LastComposedWord.COMMIT_TYPE_DECIDED_WORD)
|
|
|
|
|| (null == mCurrentWord.mAutoCorrection)
|
|
|
|
? null : mCurrentWord.mAutoCorrection.toString());
|
2011-12-13 14:12:22 +00:00
|
|
|
// TODO: improve performance by swapping buffers instead of creating a new object.
|
|
|
|
mCurrentWord = new CharacterStore();
|
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) {
|
|
|
|
mCurrentWord.mCodes = lastComposedWord.mCodes;
|
|
|
|
mCurrentWord.mXCoordinates = lastComposedWord.mXCoordinates;
|
|
|
|
mCurrentWord.mYCoordinates = lastComposedWord.mYCoordinates;
|
|
|
|
mCurrentWord.mTypedWord.setLength(0);
|
|
|
|
mCurrentWord.mTypedWord.append(lastComposedWord.mTypedWord);
|
|
|
|
mCurrentWord.mAutoCorrection = lastComposedWord.mAutoCorrection;
|
2011-12-14 11:13:16 +00:00
|
|
|
}
|
2009-03-13 22:11:42 +00:00
|
|
|
}
|