2014-01-15 07:04:05 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 The Android Open Source Project
|
|
|
|
*
|
|
|
|
* 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.latin.utils;
|
|
|
|
|
2014-01-15 08:44:52 +00:00
|
|
|
import android.util.Log;
|
|
|
|
|
2014-01-15 07:04:05 +00:00
|
|
|
import com.android.inputmethod.latin.Dictionary;
|
2014-05-23 00:30:55 +00:00
|
|
|
import com.android.inputmethod.latin.DictionaryFacilitator;
|
2014-05-19 04:55:40 +00:00
|
|
|
import com.android.inputmethod.latin.PrevWordsInfo;
|
2014-01-15 08:44:52 +00:00
|
|
|
import com.android.inputmethod.latin.settings.SpacingAndPunctuations;
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
2014-05-23 07:10:36 +00:00
|
|
|
import java.util.List;
|
2014-01-15 08:44:52 +00:00
|
|
|
import java.util.Locale;
|
2014-01-15 07:04:05 +00:00
|
|
|
|
2014-01-15 09:39:36 +00:00
|
|
|
// Note: this class is used as a parameter type of a native method. You should be careful when you
|
|
|
|
// rename this class or field name. See BinaryDictionary#addMultipleDictionaryEntriesNative().
|
|
|
|
public final class LanguageModelParam {
|
2014-01-15 08:44:52 +00:00
|
|
|
private static final String TAG = LanguageModelParam.class.getSimpleName();
|
|
|
|
private static final boolean DEBUG = false;
|
|
|
|
private static final boolean DEBUG_TOKEN = false;
|
|
|
|
|
|
|
|
// For now, these probability values are being referred to only when we add new entries to
|
|
|
|
// decaying dynamic binary dictionaries. When these are referred to, what matters is 0 or
|
|
|
|
// non-0. Thus, it's not meaningful to compare 10, 100, and so on.
|
|
|
|
// TODO: Revise the logic in ForgettingCurveUtils in native code.
|
|
|
|
private static final int UNIGRAM_PROBABILITY_FOR_VALID_WORD = 100;
|
2014-02-26 08:50:44 +00:00
|
|
|
private static final int UNIGRAM_PROBABILITY_FOR_OOV_WORD = Dictionary.NOT_A_PROBABILITY;
|
|
|
|
private static final int BIGRAM_PROBABILITY_FOR_VALID_WORD = 10;
|
|
|
|
private static final int BIGRAM_PROBABILITY_FOR_OOV_WORD = Dictionary.NOT_A_PROBABILITY;
|
2014-01-15 08:44:52 +00:00
|
|
|
|
2014-01-15 07:04:05 +00:00
|
|
|
public final String mTargetWord;
|
|
|
|
public final int[] mWord0;
|
|
|
|
public final int[] mWord1;
|
|
|
|
// TODO: this needs to be a list of shortcuts
|
|
|
|
public final int[] mShortcutTarget;
|
|
|
|
public final int mUnigramProbability;
|
|
|
|
public final int mBigramProbability;
|
|
|
|
public final int mShortcutProbability;
|
|
|
|
public final boolean mIsNotAWord;
|
|
|
|
public final boolean mIsBlacklisted;
|
|
|
|
// Time stamp in seconds.
|
|
|
|
public final int mTimestamp;
|
|
|
|
|
|
|
|
// Constructor for unigram. TODO: support shortcuts
|
|
|
|
public LanguageModelParam(final String word, final int unigramProbability,
|
|
|
|
final int timestamp) {
|
2014-01-15 07:17:53 +00:00
|
|
|
this(null /* word0 */, word, unigramProbability, Dictionary.NOT_A_PROBABILITY, timestamp);
|
2014-01-15 07:04:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Constructor for unigram and bigram.
|
|
|
|
public LanguageModelParam(final String word0, final String word1,
|
|
|
|
final int unigramProbability, final int bigramProbability,
|
|
|
|
final int timestamp) {
|
|
|
|
mTargetWord = word1;
|
2014-01-15 07:17:53 +00:00
|
|
|
mWord0 = (word0 == null) ? null : StringUtils.toCodePointArray(word0);
|
2014-01-15 07:04:05 +00:00
|
|
|
mWord1 = StringUtils.toCodePointArray(word1);
|
|
|
|
mShortcutTarget = null;
|
|
|
|
mUnigramProbability = unigramProbability;
|
|
|
|
mBigramProbability = bigramProbability;
|
|
|
|
mShortcutProbability = Dictionary.NOT_A_PROBABILITY;
|
|
|
|
mIsNotAWord = false;
|
|
|
|
mIsBlacklisted = false;
|
|
|
|
mTimestamp = timestamp;
|
|
|
|
}
|
2014-01-15 08:44:52 +00:00
|
|
|
|
|
|
|
// Process a list of words and return a list of {@link LanguageModelParam} objects.
|
|
|
|
public static ArrayList<LanguageModelParam> createLanguageModelParamsFrom(
|
2014-05-23 07:10:36 +00:00
|
|
|
final List<String> tokens, final int timestamp,
|
2014-05-23 00:30:55 +00:00
|
|
|
final DictionaryFacilitator dictionaryFacilitator,
|
2014-04-24 20:31:19 +00:00
|
|
|
final SpacingAndPunctuations spacingAndPunctuations,
|
|
|
|
final DistracterFilter distracterFilter) {
|
2014-05-23 11:18:17 +00:00
|
|
|
final ArrayList<LanguageModelParam> languageModelParams = new ArrayList<>();
|
2014-01-15 08:44:52 +00:00
|
|
|
final int N = tokens.size();
|
2014-05-23 14:19:33 +00:00
|
|
|
PrevWordsInfo prevWordsInfo = PrevWordsInfo.EMPTY_PREV_WORDS_INFO;
|
2014-01-15 08:44:52 +00:00
|
|
|
for (int i = 0; i < N; ++i) {
|
|
|
|
final String tempWord = tokens.get(i);
|
|
|
|
if (StringUtils.isEmptyStringOrWhiteSpaces(tempWord)) {
|
|
|
|
// just skip this token
|
|
|
|
if (DEBUG_TOKEN) {
|
|
|
|
Log.d(TAG, "--- isEmptyStringOrWhiteSpaces: \"" + tempWord + "\"");
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!DictionaryInfoUtils.looksValidForDictionaryInsertion(
|
|
|
|
tempWord, spacingAndPunctuations)) {
|
|
|
|
if (DEBUG_TOKEN) {
|
|
|
|
Log.d(TAG, "--- not looksValidForDictionaryInsertion: \""
|
|
|
|
+ tempWord + "\"");
|
|
|
|
}
|
|
|
|
// Sentence terminator found. Split.
|
2014-05-23 14:19:33 +00:00
|
|
|
prevWordsInfo = PrevWordsInfo.EMPTY_PREV_WORDS_INFO;
|
2014-01-15 08:44:52 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (DEBUG_TOKEN) {
|
|
|
|
Log.d(TAG, "--- word: \"" + tempWord + "\"");
|
|
|
|
}
|
|
|
|
final LanguageModelParam languageModelParam =
|
|
|
|
detectWhetherVaildWordOrNotAndGetLanguageModelParam(
|
2014-05-19 04:55:40 +00:00
|
|
|
prevWordsInfo, tempWord, timestamp, dictionaryFacilitator,
|
2014-04-24 20:31:19 +00:00
|
|
|
distracterFilter);
|
2014-03-28 03:17:44 +00:00
|
|
|
if (languageModelParam == null) {
|
|
|
|
continue;
|
|
|
|
}
|
2014-01-15 08:44:52 +00:00
|
|
|
languageModelParams.add(languageModelParam);
|
2014-05-19 04:55:40 +00:00
|
|
|
prevWordsInfo = new PrevWordsInfo(languageModelParam.mTargetWord);
|
2014-01-15 08:44:52 +00:00
|
|
|
}
|
|
|
|
return languageModelParams;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static LanguageModelParam detectWhetherVaildWordOrNotAndGetLanguageModelParam(
|
2014-05-19 04:55:40 +00:00
|
|
|
final PrevWordsInfo prevWordsInfo, final String targetWord, final int timestamp,
|
2014-05-23 00:30:55 +00:00
|
|
|
final DictionaryFacilitator dictionaryFacilitator,
|
2014-04-24 20:31:19 +00:00
|
|
|
final DistracterFilter distracterFilter) {
|
2014-03-25 06:35:20 +00:00
|
|
|
final Locale locale = dictionaryFacilitator.getLocale();
|
2014-03-28 03:17:44 +00:00
|
|
|
if (locale == null) {
|
|
|
|
return null;
|
|
|
|
}
|
2014-04-24 22:36:21 +00:00
|
|
|
// TODO: Though targetWord is an IV (in-vocabulary) word, we should still apply
|
|
|
|
// distracterFilter in the following code. If targetWord is a distracter,
|
|
|
|
// it should be filtered out.
|
2014-01-15 08:44:52 +00:00
|
|
|
if (dictionaryFacilitator.isValidWord(targetWord, false /* ignoreCase */)) {
|
2014-05-19 04:55:40 +00:00
|
|
|
return createAndGetLanguageModelParamOfWord(prevWordsInfo, targetWord, timestamp,
|
2014-01-15 08:44:52 +00:00
|
|
|
true /* isValidWord */, locale);
|
|
|
|
}
|
2014-04-24 20:31:19 +00:00
|
|
|
|
2014-01-15 08:44:52 +00:00
|
|
|
final String lowerCaseTargetWord = targetWord.toLowerCase(locale);
|
|
|
|
if (dictionaryFacilitator.isValidWord(lowerCaseTargetWord, false /* ignoreCase */)) {
|
|
|
|
// Add the lower-cased word.
|
2014-05-19 04:55:40 +00:00
|
|
|
return createAndGetLanguageModelParamOfWord(prevWordsInfo, lowerCaseTargetWord,
|
2014-01-15 08:44:52 +00:00
|
|
|
timestamp, true /* isValidWord */, locale);
|
|
|
|
}
|
2014-04-24 20:31:19 +00:00
|
|
|
|
|
|
|
// Treat the word as an OOV word. The following statement checks whether this OOV
|
|
|
|
// is a distracter to words in dictionaries. Being a distracter means the OOV word is
|
|
|
|
// too close to a common word in dictionaries (e.g., the OOV "mot" is very close to "not").
|
|
|
|
// Adding such a word to dictonaries would interfere with entering in-dictionary words. For
|
|
|
|
// example, adding "mot" to dictionaries might interfere with entering "not".
|
|
|
|
// This kind of OOV should be filtered out.
|
2014-05-19 04:55:40 +00:00
|
|
|
if (distracterFilter.isDistracterToWordsInDictionaries(prevWordsInfo, targetWord, locale)) {
|
2014-04-24 20:31:19 +00:00
|
|
|
return null;
|
|
|
|
}
|
2014-05-19 04:55:40 +00:00
|
|
|
return createAndGetLanguageModelParamOfWord(prevWordsInfo, targetWord, timestamp,
|
2014-01-15 08:44:52 +00:00
|
|
|
false /* isValidWord */, locale);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static LanguageModelParam createAndGetLanguageModelParamOfWord(
|
2014-05-19 04:55:40 +00:00
|
|
|
final PrevWordsInfo prevWordsInfo, final String targetWord, final int timestamp,
|
2014-01-15 08:44:52 +00:00
|
|
|
final boolean isValidWord, final Locale locale) {
|
|
|
|
final String word;
|
|
|
|
if (StringUtils.getCapitalizationType(targetWord) == StringUtils.CAPITALIZE_FIRST
|
2014-05-19 04:55:40 +00:00
|
|
|
&& prevWordsInfo.mPrevWord == null && !isValidWord) {
|
2014-01-15 08:44:52 +00:00
|
|
|
word = targetWord.toLowerCase(locale);
|
|
|
|
} else {
|
|
|
|
word = targetWord;
|
|
|
|
}
|
|
|
|
final int unigramProbability = isValidWord ?
|
|
|
|
UNIGRAM_PROBABILITY_FOR_VALID_WORD : UNIGRAM_PROBABILITY_FOR_OOV_WORD;
|
2014-05-19 04:55:40 +00:00
|
|
|
if (prevWordsInfo.mPrevWord == null) {
|
2014-01-15 08:44:52 +00:00
|
|
|
if (DEBUG) {
|
|
|
|
Log.d(TAG, "--- add unigram: current("
|
|
|
|
+ (isValidWord ? "Valid" : "OOV") + ") = " + word);
|
|
|
|
}
|
|
|
|
return new LanguageModelParam(word, unigramProbability, timestamp);
|
|
|
|
}
|
|
|
|
if (DEBUG) {
|
2014-05-19 04:55:40 +00:00
|
|
|
Log.d(TAG, "--- add bigram: prev = " + prevWordsInfo.mPrevWord + ", current("
|
2014-01-15 08:44:52 +00:00
|
|
|
+ (isValidWord ? "Valid" : "OOV") + ") = " + word);
|
|
|
|
}
|
|
|
|
final int bigramProbability = isValidWord ?
|
|
|
|
BIGRAM_PROBABILITY_FOR_VALID_WORD : BIGRAM_PROBABILITY_FOR_OOV_WORD;
|
2014-05-19 04:55:40 +00:00
|
|
|
return new LanguageModelParam(prevWordsInfo.mPrevWord, word, unigramProbability,
|
2014-01-15 08:44:52 +00:00
|
|
|
bigramProbability, timestamp);
|
|
|
|
}
|
2014-01-15 07:04:05 +00:00
|
|
|
}
|