2012-08-03 03:22:29 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 The Android Open Source Project
|
|
|
|
*
|
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
|
2012-08-03 03:22:29 +00:00
|
|
|
*
|
2013-01-21 12:52:57 +00:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2012-08-03 03:22:29 +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.
|
2012-08-03 03:22:29 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
package com.android.inputmethod.research;
|
|
|
|
|
2012-12-18 21:59:22 +00:00
|
|
|
import android.content.SharedPreferences;
|
2013-01-15 21:48:15 +00:00
|
|
|
import android.text.TextUtils;
|
2012-12-18 21:59:22 +00:00
|
|
|
import android.util.JsonWriter;
|
|
|
|
import android.util.Log;
|
2012-12-23 23:40:37 +00:00
|
|
|
import android.view.MotionEvent;
|
2012-12-18 21:59:22 +00:00
|
|
|
import android.view.inputmethod.CompletionInfo;
|
|
|
|
|
|
|
|
import com.android.inputmethod.keyboard.Key;
|
|
|
|
import com.android.inputmethod.latin.SuggestedWords;
|
|
|
|
import com.android.inputmethod.latin.SuggestedWords.SuggestedWordInfo;
|
2012-08-24 05:45:23 +00:00
|
|
|
import com.android.inputmethod.latin.Utils;
|
2012-12-18 21:59:22 +00:00
|
|
|
import com.android.inputmethod.latin.define.ProductionFlag;
|
2012-08-10 08:54:06 +00:00
|
|
|
import com.android.inputmethod.research.ResearchLogger.LogStatement;
|
2012-08-03 03:22:29 +00:00
|
|
|
|
2012-12-18 21:59:22 +00:00
|
|
|
import java.io.IOException;
|
2012-08-24 05:45:23 +00:00
|
|
|
import java.io.StringWriter;
|
2012-08-10 08:54:06 +00:00
|
|
|
import java.util.ArrayList;
|
2012-08-09 22:58:25 +00:00
|
|
|
import java.util.List;
|
2012-12-18 21:59:22 +00:00
|
|
|
import java.util.Map;
|
2012-08-03 03:22:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A group of log statements related to each other.
|
|
|
|
*
|
|
|
|
* A LogUnit is collection of LogStatements, each of which is generated by at a particular point
|
|
|
|
* in the code. (There is no LogStatement class; the data is stored across the instance variables
|
|
|
|
* here.) A single LogUnit's statements can correspond to all the calls made while in the same
|
|
|
|
* composing region, or all the calls between committing the last composing region, and the first
|
|
|
|
* character of the next composing region.
|
|
|
|
*
|
|
|
|
* Individual statements in a log may be marked as potentially private. If so, then they are only
|
|
|
|
* published to a ResearchLog if the ResearchLogger determines that publishing the entire LogUnit
|
|
|
|
* will not violate the user's privacy. Checks for this may include whether other LogUnits have
|
|
|
|
* been published recently, or whether the LogUnit contains numbers, etc.
|
|
|
|
*/
|
|
|
|
/* package */ class LogUnit {
|
2012-12-18 21:59:22 +00:00
|
|
|
private static final String TAG = LogUnit.class.getSimpleName();
|
|
|
|
private static final boolean DEBUG = false && ProductionFlag.IS_EXPERIMENTAL_DEBUG;
|
2012-08-10 08:54:06 +00:00
|
|
|
private final ArrayList<LogStatement> mLogStatementList;
|
|
|
|
private final ArrayList<Object[]> mValuesList;
|
2012-08-09 22:58:25 +00:00
|
|
|
// Assume that mTimeList is sorted in increasing order. Do not insert null values into
|
|
|
|
// mTimeList.
|
2012-08-10 08:54:06 +00:00
|
|
|
private final ArrayList<Long> mTimeList;
|
2013-01-15 21:48:15 +00:00
|
|
|
// Word that this LogUnit generates. Should be null if the LogUnit does not generate a genuine
|
|
|
|
// word (i.e. separators alone do not count as a word). Should never be empty.
|
2012-08-03 03:22:29 +00:00
|
|
|
private String mWord;
|
2012-08-09 22:58:25 +00:00
|
|
|
private boolean mMayContainDigit;
|
2012-08-10 08:54:06 +00:00
|
|
|
private boolean mIsPartOfMegaword;
|
2013-01-11 22:17:17 +00:00
|
|
|
private boolean mContainsCorrection;
|
2012-08-03 03:22:29 +00:00
|
|
|
|
2013-01-15 21:48:15 +00:00
|
|
|
// mCorrectionType indicates whether the word was corrected at all, and if so, whether it was
|
|
|
|
// to a different word or just a "typo" correction. It is considered a "typo" if the final
|
|
|
|
// word was listed in the suggestions available the first time the word was gestured or
|
|
|
|
// tapped.
|
|
|
|
private int mCorrectionType;
|
|
|
|
public static final int CORRECTIONTYPE_NO_CORRECTION = 0;
|
|
|
|
public static final int CORRECTIONTYPE_CORRECTION = 1;
|
|
|
|
public static final int CORRECTIONTYPE_DIFFERENT_WORD = 2;
|
|
|
|
public static final int CORRECTIONTYPE_TYPO = 3;
|
|
|
|
|
|
|
|
private SuggestedWords mSuggestedWords;
|
|
|
|
|
2012-08-09 22:58:25 +00:00
|
|
|
public LogUnit() {
|
2012-08-10 08:54:06 +00:00
|
|
|
mLogStatementList = new ArrayList<LogStatement>();
|
|
|
|
mValuesList = new ArrayList<Object[]>();
|
|
|
|
mTimeList = new ArrayList<Long>();
|
|
|
|
mIsPartOfMegaword = false;
|
2013-01-15 21:48:15 +00:00
|
|
|
mCorrectionType = CORRECTIONTYPE_NO_CORRECTION;
|
|
|
|
mSuggestedWords = null;
|
2012-08-09 22:58:25 +00:00
|
|
|
}
|
|
|
|
|
2012-08-10 08:54:06 +00:00
|
|
|
private LogUnit(final ArrayList<LogStatement> logStatementList,
|
|
|
|
final ArrayList<Object[]> valuesList,
|
|
|
|
final ArrayList<Long> timeList,
|
|
|
|
final boolean isPartOfMegaword) {
|
|
|
|
mLogStatementList = logStatementList;
|
2012-08-09 22:58:25 +00:00
|
|
|
mValuesList = valuesList;
|
|
|
|
mTimeList = timeList;
|
2012-08-10 08:54:06 +00:00
|
|
|
mIsPartOfMegaword = isPartOfMegaword;
|
2013-01-15 21:48:15 +00:00
|
|
|
mCorrectionType = CORRECTIONTYPE_NO_CORRECTION;
|
|
|
|
mSuggestedWords = null;
|
2012-08-09 22:58:25 +00:00
|
|
|
}
|
|
|
|
|
2012-08-10 21:21:18 +00:00
|
|
|
private static final Object[] NULL_VALUES = new Object[0];
|
2012-08-09 22:58:25 +00:00
|
|
|
/**
|
|
|
|
* Adds a new log statement. The time parameter in successive calls to this method must be
|
|
|
|
* monotonically increasing, or splitByTime() will not work.
|
|
|
|
*/
|
2012-08-10 21:21:18 +00:00
|
|
|
public void addLogStatement(final LogStatement logStatement, final long time,
|
|
|
|
Object... values) {
|
|
|
|
if (values == null) {
|
|
|
|
values = NULL_VALUES;
|
|
|
|
}
|
2012-08-10 08:54:06 +00:00
|
|
|
mLogStatementList.add(logStatement);
|
2012-08-03 03:22:29 +00:00
|
|
|
mValuesList.add(values);
|
2012-08-09 22:58:25 +00:00
|
|
|
mTimeList.add(time);
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|
|
|
|
|
2012-12-18 21:59:22 +00:00
|
|
|
/**
|
|
|
|
* Publish the contents of this LogUnit to researchLog.
|
|
|
|
*/
|
|
|
|
public synchronized void publishTo(final ResearchLog researchLog,
|
2013-01-12 00:49:54 +00:00
|
|
|
final boolean canIncludePrivateData) {
|
2012-08-24 05:45:23 +00:00
|
|
|
// Prepare debugging output if necessary
|
|
|
|
final StringWriter debugStringWriter;
|
|
|
|
final JsonWriter debugJsonWriter;
|
|
|
|
if (DEBUG) {
|
|
|
|
debugStringWriter = new StringWriter();
|
|
|
|
debugJsonWriter = new JsonWriter(debugStringWriter);
|
|
|
|
debugJsonWriter.setIndent(" ");
|
|
|
|
try {
|
|
|
|
debugJsonWriter.beginArray();
|
|
|
|
} catch (IOException e) {
|
|
|
|
Log.e(TAG, "Could not open array in JsonWriter", e);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
debugStringWriter = null;
|
|
|
|
debugJsonWriter = null;
|
|
|
|
}
|
2012-12-18 21:59:22 +00:00
|
|
|
// Write out any logStatement that passes the privacy filter.
|
2013-01-09 02:29:17 +00:00
|
|
|
final int size = mLogStatementList.size();
|
|
|
|
if (size != 0) {
|
|
|
|
// Note that jsonWriter is only set to a non-null value if the logUnit start text is
|
|
|
|
// output and at least one logStatement is output.
|
|
|
|
JsonWriter jsonWriter = null;
|
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
final LogStatement logStatement = mLogStatementList.get(i);
|
2013-01-12 00:49:54 +00:00
|
|
|
if (!canIncludePrivateData && logStatement.mIsPotentiallyPrivate) {
|
2013-01-09 02:29:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (mIsPartOfMegaword && logStatement.mIsPotentiallyRevealing) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Only retrieve the jsonWriter if we need to. If we don't get this far, then
|
|
|
|
// researchLog.getValidJsonWriterLocked() will not ever be called, and the file
|
|
|
|
// will not have been opened for writing.
|
|
|
|
if (jsonWriter == null) {
|
|
|
|
jsonWriter = researchLog.getValidJsonWriterLocked();
|
2013-01-12 00:49:54 +00:00
|
|
|
outputLogUnitStart(jsonWriter, canIncludePrivateData);
|
2013-01-09 02:29:17 +00:00
|
|
|
}
|
|
|
|
outputLogStatementToLocked(jsonWriter, mLogStatementList.get(i), mValuesList.get(i),
|
|
|
|
mTimeList.get(i));
|
|
|
|
if (DEBUG) {
|
|
|
|
outputLogStatementToLocked(debugJsonWriter, mLogStatementList.get(i),
|
|
|
|
mValuesList.get(i), mTimeList.get(i));
|
|
|
|
}
|
2012-08-10 08:54:06 +00:00
|
|
|
}
|
2013-01-09 02:29:17 +00:00
|
|
|
if (jsonWriter != null) {
|
|
|
|
// We must have called logUnitStart earlier, so emit a logUnitStop.
|
2013-01-12 00:49:54 +00:00
|
|
|
outputLogUnitStop(jsonWriter);
|
2012-08-24 05:45:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (DEBUG) {
|
|
|
|
try {
|
|
|
|
debugJsonWriter.endArray();
|
|
|
|
debugJsonWriter.flush();
|
|
|
|
} catch (IOException e) {
|
|
|
|
Log.e(TAG, "Could not close array in JsonWriter", e);
|
|
|
|
}
|
|
|
|
final String bigString = debugStringWriter.getBuffer().toString();
|
|
|
|
final String[] lines = bigString.split("\n");
|
|
|
|
for (String line : lines) {
|
|
|
|
Log.d(TAG, line);
|
|
|
|
}
|
2012-12-18 21:59:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static final String CURRENT_TIME_KEY = "_ct";
|
|
|
|
private static final String UPTIME_KEY = "_ut";
|
|
|
|
private static final String EVENT_TYPE_KEY = "_ty";
|
2013-01-09 02:29:17 +00:00
|
|
|
private static final String WORD_KEY = "_wo";
|
2013-01-15 21:48:15 +00:00
|
|
|
private static final String CORRECTION_TYPE_KEY = "_corType";
|
2013-01-09 02:29:17 +00:00
|
|
|
private static final String LOG_UNIT_BEGIN_KEY = "logUnitStart";
|
|
|
|
private static final String LOG_UNIT_END_KEY = "logUnitEnd";
|
|
|
|
|
2012-10-24 17:49:30 +00:00
|
|
|
private void outputLogUnitStart(final JsonWriter jsonWriter,
|
2013-01-12 00:49:54 +00:00
|
|
|
final boolean canIncludePrivateData) {
|
2013-01-09 02:29:17 +00:00
|
|
|
try {
|
|
|
|
jsonWriter.beginObject();
|
|
|
|
jsonWriter.name(CURRENT_TIME_KEY).value(System.currentTimeMillis());
|
2013-01-12 00:49:54 +00:00
|
|
|
if (canIncludePrivateData) {
|
2012-10-24 17:49:30 +00:00
|
|
|
jsonWriter.name(WORD_KEY).value(getWord());
|
2013-01-15 21:48:15 +00:00
|
|
|
jsonWriter.name(CORRECTION_TYPE_KEY).value(getCorrectionType());
|
2012-10-24 17:49:30 +00:00
|
|
|
}
|
2013-01-09 02:29:17 +00:00
|
|
|
jsonWriter.name(EVENT_TYPE_KEY).value(LOG_UNIT_BEGIN_KEY);
|
|
|
|
jsonWriter.endObject();
|
|
|
|
} catch (IOException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
Log.w(TAG, "Error in JsonWriter; cannot write LogUnitStart");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-12 00:49:54 +00:00
|
|
|
private void outputLogUnitStop(final JsonWriter jsonWriter) {
|
2013-01-09 02:29:17 +00:00
|
|
|
try {
|
|
|
|
jsonWriter.beginObject();
|
|
|
|
jsonWriter.name(CURRENT_TIME_KEY).value(System.currentTimeMillis());
|
|
|
|
jsonWriter.name(EVENT_TYPE_KEY).value(LOG_UNIT_END_KEY);
|
|
|
|
jsonWriter.endObject();
|
|
|
|
} catch (IOException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
Log.w(TAG, "Error in JsonWriter; cannot write LogUnitStop");
|
|
|
|
}
|
|
|
|
}
|
2012-12-18 21:59:22 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Write the logStatement and its contents out through jsonWriter.
|
|
|
|
*
|
|
|
|
* Note that this method is not thread safe for the same jsonWriter. Callers must ensure
|
|
|
|
* thread safety.
|
|
|
|
*/
|
|
|
|
private boolean outputLogStatementToLocked(final JsonWriter jsonWriter,
|
|
|
|
final LogStatement logStatement, final Object[] values, final Long time) {
|
|
|
|
if (DEBUG) {
|
|
|
|
if (logStatement.mKeys.length != values.length) {
|
|
|
|
Log.d(TAG, "Key and Value list sizes do not match. " + logStatement.mName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
jsonWriter.beginObject();
|
|
|
|
jsonWriter.name(CURRENT_TIME_KEY).value(System.currentTimeMillis());
|
|
|
|
jsonWriter.name(UPTIME_KEY).value(time);
|
|
|
|
jsonWriter.name(EVENT_TYPE_KEY).value(logStatement.mName);
|
|
|
|
final String[] keys = logStatement.mKeys;
|
|
|
|
final int length = values.length;
|
|
|
|
for (int i = 0; i < length; i++) {
|
|
|
|
jsonWriter.name(keys[i]);
|
|
|
|
final Object value = values[i];
|
|
|
|
if (value instanceof CharSequence) {
|
|
|
|
jsonWriter.value(value.toString());
|
|
|
|
} else if (value instanceof Number) {
|
|
|
|
jsonWriter.value((Number) value);
|
|
|
|
} else if (value instanceof Boolean) {
|
|
|
|
jsonWriter.value((Boolean) value);
|
|
|
|
} else if (value instanceof CompletionInfo[]) {
|
|
|
|
JsonUtils.writeJson((CompletionInfo[]) value, jsonWriter);
|
|
|
|
} else if (value instanceof SharedPreferences) {
|
|
|
|
JsonUtils.writeJson((SharedPreferences) value, jsonWriter);
|
|
|
|
} else if (value instanceof Key[]) {
|
|
|
|
JsonUtils.writeJson((Key[]) value, jsonWriter);
|
|
|
|
} else if (value instanceof SuggestedWords) {
|
|
|
|
JsonUtils.writeJson((SuggestedWords) value, jsonWriter);
|
2012-12-23 23:40:37 +00:00
|
|
|
} else if (value instanceof MotionEvent) {
|
|
|
|
JsonUtils.writeJson((MotionEvent) value, jsonWriter);
|
2012-12-18 21:59:22 +00:00
|
|
|
} else if (value == null) {
|
|
|
|
jsonWriter.nullValue();
|
|
|
|
} else {
|
|
|
|
Log.w(TAG, "Unrecognized type to be logged: " +
|
|
|
|
(value == null ? "<null>" : value.getClass().getName()));
|
|
|
|
jsonWriter.nullValue();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
jsonWriter.endObject();
|
|
|
|
} catch (IOException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
Log.w(TAG, "Error in JsonWriter; skipping LogStatement");
|
|
|
|
return false;
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|
2012-12-18 21:59:22 +00:00
|
|
|
return true;
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|
|
|
|
|
2013-01-15 21:48:15 +00:00
|
|
|
/**
|
|
|
|
* Mark the current logUnit as containing data to generate {@code word}.
|
|
|
|
*
|
|
|
|
* If {@code setWord()} was previously called for this LogUnit, then the method will try to
|
|
|
|
* determine what kind of correction it is, and update its internal state of the correctionType
|
|
|
|
* accordingly.
|
|
|
|
*
|
|
|
|
* @param word The word this LogUnit generates. Caller should not pass null or the empty
|
|
|
|
* string.
|
|
|
|
*/
|
|
|
|
public void setWord(final String word) {
|
2012-11-17 00:10:10 +00:00
|
|
|
if (hasWord()) {
|
2013-01-15 21:48:15 +00:00
|
|
|
// The word was already set once, and it is now being changed. See if the new word
|
|
|
|
// is close to the old word. If so, then the change is probably a typo correction.
|
|
|
|
// If not, the user may have decided to enter a different word, so flag it.
|
|
|
|
if (mSuggestedWords != null) {
|
|
|
|
if (isInSuggestedWords(word, mSuggestedWords)) {
|
|
|
|
mCorrectionType = CORRECTIONTYPE_TYPO;
|
|
|
|
} else {
|
|
|
|
mCorrectionType = CORRECTIONTYPE_DIFFERENT_WORD;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// No suggested words, so it's not clear whether it's a typo or different word.
|
|
|
|
// Mark it as a generic correction.
|
|
|
|
mCorrectionType = CORRECTIONTYPE_CORRECTION;
|
|
|
|
}
|
|
|
|
}
|
2012-08-03 03:22:29 +00:00
|
|
|
mWord = word;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getWord() {
|
|
|
|
return mWord;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasWord() {
|
2012-11-17 00:10:10 +00:00
|
|
|
return mWord != null && !TextUtils.isEmpty(mWord.trim());
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|
|
|
|
|
2012-08-09 22:58:25 +00:00
|
|
|
public void setMayContainDigit() {
|
|
|
|
mMayContainDigit = true;
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|
|
|
|
|
2012-08-09 22:58:25 +00:00
|
|
|
public boolean mayContainDigit() {
|
|
|
|
return mMayContainDigit;
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|
|
|
|
|
2013-01-11 22:17:17 +00:00
|
|
|
public void setContainsCorrection() {
|
|
|
|
mContainsCorrection = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean containsCorrection() {
|
|
|
|
return mContainsCorrection;
|
|
|
|
}
|
|
|
|
|
2013-01-15 21:48:15 +00:00
|
|
|
public void setCorrectionType(final int correctionType) {
|
|
|
|
mCorrectionType = correctionType;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getCorrectionType() {
|
|
|
|
return mCorrectionType;
|
|
|
|
}
|
|
|
|
|
2012-08-03 03:22:29 +00:00
|
|
|
public boolean isEmpty() {
|
2012-08-10 08:54:06 +00:00
|
|
|
return mLogStatementList.isEmpty();
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|
2012-08-09 22:58:25 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Split this logUnit, with all events before maxTime staying in the current logUnit, and all
|
|
|
|
* events after maxTime going into a new LogUnit that is returned.
|
|
|
|
*/
|
|
|
|
public LogUnit splitByTime(final long maxTime) {
|
|
|
|
// Assume that mTimeList is in sorted order.
|
|
|
|
final int length = mTimeList.size();
|
2012-12-23 23:12:51 +00:00
|
|
|
// TODO: find time by binary search, e.g. using Collections#binarySearch()
|
2012-08-09 22:58:25 +00:00
|
|
|
for (int index = 0; index < length; index++) {
|
2012-08-10 08:54:06 +00:00
|
|
|
if (mTimeList.get(index) > maxTime) {
|
|
|
|
final List<LogStatement> laterLogStatements =
|
|
|
|
mLogStatementList.subList(index, length);
|
|
|
|
final List<Object[]> laterValues = mValuesList.subList(index, length);
|
|
|
|
final List<Long> laterTimes = mTimeList.subList(index, length);
|
|
|
|
|
|
|
|
// Create the LogUnit containing the later logStatements and associated data.
|
2012-08-09 22:58:25 +00:00
|
|
|
final LogUnit newLogUnit = new LogUnit(
|
2012-08-10 08:54:06 +00:00
|
|
|
new ArrayList<LogStatement>(laterLogStatements),
|
|
|
|
new ArrayList<Object[]>(laterValues),
|
|
|
|
new ArrayList<Long>(laterTimes),
|
|
|
|
true /* isPartOfMegaword */);
|
2012-08-09 22:58:25 +00:00
|
|
|
newLogUnit.mWord = null;
|
|
|
|
newLogUnit.mMayContainDigit = mMayContainDigit;
|
2013-01-11 22:17:17 +00:00
|
|
|
newLogUnit.mContainsCorrection = mContainsCorrection;
|
2012-08-10 08:54:06 +00:00
|
|
|
|
|
|
|
// Purge the logStatements and associated data from this LogUnit.
|
|
|
|
laterLogStatements.clear();
|
|
|
|
laterValues.clear();
|
|
|
|
laterTimes.clear();
|
|
|
|
mIsPartOfMegaword = true;
|
|
|
|
|
2012-08-09 22:58:25 +00:00
|
|
|
return newLogUnit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new LogUnit();
|
|
|
|
}
|
2012-12-23 23:12:51 +00:00
|
|
|
|
|
|
|
public void append(final LogUnit logUnit) {
|
|
|
|
mLogStatementList.addAll(logUnit.mLogStatementList);
|
|
|
|
mValuesList.addAll(logUnit.mValuesList);
|
|
|
|
mTimeList.addAll(logUnit.mTimeList);
|
|
|
|
mWord = null;
|
2013-01-15 21:48:15 +00:00
|
|
|
if (logUnit.mWord != null) {
|
|
|
|
setWord(logUnit.mWord);
|
|
|
|
}
|
2012-12-23 23:12:51 +00:00
|
|
|
mMayContainDigit = mMayContainDigit || logUnit.mMayContainDigit;
|
2013-01-11 22:17:17 +00:00
|
|
|
mContainsCorrection = mContainsCorrection || logUnit.mContainsCorrection;
|
2012-12-23 23:12:51 +00:00
|
|
|
mIsPartOfMegaword = false;
|
|
|
|
}
|
2013-01-15 21:48:15 +00:00
|
|
|
|
|
|
|
public SuggestedWords getSuggestions() {
|
|
|
|
return mSuggestedWords;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize the suggestions.
|
|
|
|
*
|
|
|
|
* Once set to a non-null value, the suggestions may not be changed again. This is to keep
|
|
|
|
* track of the list of words that are close to the user's initial effort to type the word.
|
|
|
|
* Only words that are close to the initial effort are considered typo corrections.
|
|
|
|
*/
|
|
|
|
public void initializeSuggestions(final SuggestedWords suggestedWords) {
|
|
|
|
if (mSuggestedWords == null) {
|
|
|
|
mSuggestedWords = suggestedWords;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean isInSuggestedWords(final String queryWord,
|
|
|
|
final SuggestedWords suggestedWords) {
|
|
|
|
if (TextUtils.isEmpty(queryWord)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
final int size = suggestedWords.size();
|
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
final SuggestedWordInfo wordInfo = suggestedWords.getInfo(i);
|
|
|
|
if (queryWord.equals(wordInfo.mWord)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2012-08-03 03:22:29 +00:00
|
|
|
}
|