Merge "Don't assume that correctable words are invalid"
This commit is contained in:
commit
e4619f029e
5 changed files with 72 additions and 25 deletions
|
@ -674,7 +674,8 @@ public class DictionaryFacilitatorImpl implements DictionaryFacilitator {
|
|||
long proximityInfoHandle = keyboard.getProximityInfo().getNativeProximityInfo();
|
||||
final DictionaryGroup[] dictionaryGroups = mDictionaryGroups;
|
||||
final SuggestionResults suggestionResults = new SuggestionResults(
|
||||
SuggestedWords.MAX_SUGGESTIONS, ngramContext.isBeginningOfSentenceContext());
|
||||
SuggestedWords.MAX_SUGGESTIONS, ngramContext.isBeginningOfSentenceContext(),
|
||||
false /* firstSuggestionExceedsConfidenceThreshold */);
|
||||
final float[] weightOfLangModelVsSpatialModel =
|
||||
new float[] { Dictionary.NOT_A_WEIGHT_OF_LANG_MODEL_VS_SPATIAL_MODEL };
|
||||
for (final DictionaryGroup dictionaryGroup : dictionaryGroups) {
|
||||
|
|
|
@ -210,7 +210,8 @@ public final class Suggest {
|
|||
}
|
||||
}
|
||||
|
||||
SuggestedWordInfo.removeDups(typedWordString, suggestionsContainer);
|
||||
final int firstOcurrenceOfTypedWordInSuggestions =
|
||||
SuggestedWordInfo.removeDups(typedWordString, suggestionsContainer);
|
||||
|
||||
final SuggestedWordInfo whitelistedWordInfo =
|
||||
getWhitelistedWordInfoOrNull(suggestionsContainer);
|
||||
|
@ -273,7 +274,8 @@ public final class Suggest {
|
|||
hasAutoCorrection = false;
|
||||
} else {
|
||||
final SuggestedWordInfo firstSuggestion = suggestionResults.first();
|
||||
if (suggestionResults.mAutocorrectRecommendation) {
|
||||
if (suggestionResults.mFirstSuggestionExceedsConfidenceThreshold
|
||||
&& firstOcurrenceOfTypedWordInSuggestions != 0) {
|
||||
hasAutoCorrection = true;
|
||||
} else if (!AutoCorrectionUtils.suggestionExceedsThreshold(
|
||||
firstSuggestion, consideredWord, mAutoCorrectionThreshold)) {
|
||||
|
@ -314,12 +316,12 @@ public final class Suggest {
|
|||
} else {
|
||||
inputStyle = inputStyleIfNotPrediction;
|
||||
}
|
||||
|
||||
final boolean isTypedWordValid = firstOcurrenceOfTypedWordInSuggestions > -1
|
||||
|| (!resultsArePredictions && !allowsToBeAutoCorrected);
|
||||
callback.onGetSuggestedWords(new SuggestedWords(suggestionsList,
|
||||
suggestionResults.mRawSuggestions, typedWordInfo,
|
||||
// TODO: this first argument is lying. If this is a whitelisted word which is an
|
||||
// actual word, it says typedWordValid = false, which looks wrong. We should either
|
||||
// rename the attribute or change the value.
|
||||
!resultsArePredictions && !allowsToBeAutoCorrected /* typedWordValid */,
|
||||
isTypedWordValid,
|
||||
hasAutoCorrection /* willAutoCorrect */,
|
||||
false /* isObsoleteSuggestions */, inputStyle, sequenceNumber));
|
||||
}
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
package com.android.inputmethod.latin;
|
||||
|
||||
import android.text.TextUtils;
|
||||
import android.util.Log;
|
||||
import android.view.inputmethod.CompletionInfo;
|
||||
|
||||
import com.android.inputmethod.annotations.UsedForTesting;
|
||||
|
@ -375,31 +376,45 @@ public class SuggestedWords {
|
|||
return mWord + " (" + mDebugString + ")";
|
||||
}
|
||||
|
||||
// This will always remove the higher index if a duplicate is found.
|
||||
public static void removeDups(@Nullable final String typedWord,
|
||||
@Nonnull ArrayList<SuggestedWordInfo> candidates) {
|
||||
/**
|
||||
* This will always remove the higher index if a duplicate is found.
|
||||
*
|
||||
* @return position of typed word in the candidate list
|
||||
*/
|
||||
public static int removeDups(
|
||||
@Nullable final String typedWord,
|
||||
@Nonnull final ArrayList<SuggestedWordInfo> candidates) {
|
||||
if (candidates.isEmpty()) {
|
||||
return;
|
||||
return -1;
|
||||
}
|
||||
int firstOccurrenceOfWord = -1;
|
||||
if (!TextUtils.isEmpty(typedWord)) {
|
||||
removeSuggestedWordInfoFromList(typedWord, candidates, -1 /* startIndexExclusive */);
|
||||
firstOccurrenceOfWord = removeSuggestedWordInfoFromList(
|
||||
typedWord, candidates, -1 /* startIndexExclusive */);
|
||||
}
|
||||
for (int i = 0; i < candidates.size(); ++i) {
|
||||
removeSuggestedWordInfoFromList(candidates.get(i).mWord, candidates,
|
||||
i /* startIndexExclusive */);
|
||||
removeSuggestedWordInfoFromList(
|
||||
candidates.get(i).mWord, candidates, i /* startIndexExclusive */);
|
||||
}
|
||||
return firstOccurrenceOfWord;
|
||||
}
|
||||
|
||||
private static void removeSuggestedWordInfoFromList(
|
||||
@Nonnull final String word, @Nonnull final ArrayList<SuggestedWordInfo> candidates,
|
||||
private static int removeSuggestedWordInfoFromList(
|
||||
@Nonnull final String word,
|
||||
@Nonnull final ArrayList<SuggestedWordInfo> candidates,
|
||||
final int startIndexExclusive) {
|
||||
int firstOccurrenceOfWord = -1;
|
||||
for (int i = startIndexExclusive + 1; i < candidates.size(); ++i) {
|
||||
final SuggestedWordInfo previous = candidates.get(i);
|
||||
if (word.equals(previous.mWord)) {
|
||||
if (firstOccurrenceOfWord == -1) {
|
||||
firstOccurrenceOfWord = i;
|
||||
}
|
||||
candidates.remove(i);
|
||||
--i;
|
||||
}
|
||||
}
|
||||
return firstOccurrenceOfWord;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -33,21 +33,18 @@ public final class SuggestionResults extends TreeSet<SuggestedWordInfo> {
|
|||
// TODO: Instead of a boolean , we may want to include the context of this suggestion results,
|
||||
// such as {@link NgramContext}.
|
||||
public final boolean mIsBeginningOfSentence;
|
||||
public final boolean mAutocorrectRecommendation;
|
||||
public final boolean mFirstSuggestionExceedsConfidenceThreshold;
|
||||
private final int mCapacity;
|
||||
|
||||
public SuggestionResults(final int capacity, final boolean isBeginningOfSentence) {
|
||||
this(sSuggestedWordInfoComparator, capacity, isBeginningOfSentence, false);
|
||||
}
|
||||
|
||||
public SuggestionResults(final int capacity, final boolean isBeginningOfSentence,
|
||||
final boolean autocorrectRecommendation) {
|
||||
final boolean firstSuggestionExceedsConfidenceThreshold) {
|
||||
this(sSuggestedWordInfoComparator, capacity, isBeginningOfSentence,
|
||||
autocorrectRecommendation);
|
||||
firstSuggestionExceedsConfidenceThreshold);
|
||||
}
|
||||
|
||||
private SuggestionResults(final Comparator<SuggestedWordInfo> comparator, final int capacity,
|
||||
final boolean isBeginningOfSentence, final boolean autocorrectRecommendation) {
|
||||
final boolean isBeginningOfSentence,
|
||||
final boolean firstSuggestionExceedsConfidenceThreshold) {
|
||||
super(comparator);
|
||||
mCapacity = capacity;
|
||||
if (ProductionFlags.INCLUDE_RAW_SUGGESTIONS) {
|
||||
|
@ -56,7 +53,7 @@ public final class SuggestionResults extends TreeSet<SuggestedWordInfo> {
|
|||
mRawSuggestions = null;
|
||||
}
|
||||
mIsBeginningOfSentence = isBeginningOfSentence;
|
||||
mAutocorrectRecommendation = autocorrectRecommendation;
|
||||
mFirstSuggestionExceedsConfidenceThreshold = firstSuggestionExceedsConfidenceThreshold;
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -59,6 +59,14 @@ public class SuggestedWordsTests extends AndroidTestCase {
|
|||
SuggestedWordInfo.NOT_A_CONFIDENCE /* autoCommitFirstWordConfidence */);
|
||||
}
|
||||
|
||||
private static ArrayList<SuggestedWordInfo> createCorrectionWordInfos(final String... words) {
|
||||
final ArrayList<SuggestedWordInfo> infos = new ArrayList<>();
|
||||
for (final String word : words) {
|
||||
infos.add(createCorrectionWordInfo(word));
|
||||
}
|
||||
return infos;
|
||||
}
|
||||
|
||||
// Helper for testGetTransformedWordInfo
|
||||
private static SuggestedWordInfo transformWordInfo(final String info,
|
||||
final int trailingSingleQuotesCount) {
|
||||
|
@ -72,6 +80,30 @@ public class SuggestedWordsTests extends AndroidTestCase {
|
|||
return returnedWordInfo;
|
||||
}
|
||||
|
||||
public void testRemoveDupesNoDupes() {
|
||||
final ArrayList<SuggestedWordInfo> infos = createCorrectionWordInfos("a", "c");
|
||||
assertEquals(-1, SuggestedWordInfo.removeDups("b", infos));
|
||||
assertEquals(2, infos.size());
|
||||
}
|
||||
|
||||
public void testRemoveDupesTypedWordNotDupe() {
|
||||
final ArrayList<SuggestedWordInfo> infos = createCorrectionWordInfos("a", "a", "c");
|
||||
assertEquals(-1, SuggestedWordInfo.removeDups("b", infos));
|
||||
assertEquals(2, infos.size());
|
||||
}
|
||||
|
||||
public void testRemoveDupesTypedWordOnlyDupe() {
|
||||
final ArrayList<SuggestedWordInfo> infos = createCorrectionWordInfos("a", "b", "c");
|
||||
assertEquals(1, SuggestedWordInfo.removeDups("b", infos));
|
||||
assertEquals(2, infos.size());
|
||||
}
|
||||
|
||||
public void testRemoveDupesTypedWordNotOnlyDupe() {
|
||||
final ArrayList<SuggestedWordInfo> infos = createCorrectionWordInfos("a", "b", "b", "c");
|
||||
assertEquals(1, SuggestedWordInfo.removeDups("b", infos));
|
||||
assertEquals(2, infos.size());
|
||||
}
|
||||
|
||||
public void testGetTransformedSuggestedWordInfo() {
|
||||
SuggestedWordInfo result = transformWordInfo("word", 0);
|
||||
assertEquals(result.mWord, "word");
|
||||
|
|
Loading…
Reference in a new issue