2011-05-24 11:30:30 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2011 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.compat;
|
|
|
|
|
|
|
|
import android.content.Context;
|
|
|
|
import android.text.Spannable;
|
|
|
|
import android.text.SpannableString;
|
|
|
|
import android.text.Spanned;
|
2011-06-15 05:40:34 +00:00
|
|
|
import android.text.TextUtils;
|
2012-12-17 10:35:09 +00:00
|
|
|
import android.text.style.SuggestionSpan;
|
2011-05-24 11:30:30 +00:00
|
|
|
|
2014-10-03 13:59:38 +00:00
|
|
|
import com.android.inputmethod.annotations.UsedForTesting;
|
2012-08-21 07:34:55 +00:00
|
|
|
import com.android.inputmethod.latin.SuggestedWords;
|
2013-12-19 12:45:08 +00:00
|
|
|
import com.android.inputmethod.latin.SuggestedWords.SuggestedWordInfo;
|
2012-08-21 07:34:55 +00:00
|
|
|
import com.android.inputmethod.latin.SuggestionSpanPickedNotificationReceiver;
|
2014-10-03 13:59:38 +00:00
|
|
|
import com.android.inputmethod.latin.define.DebugFlags;
|
2014-10-03 14:06:47 +00:00
|
|
|
import com.android.inputmethod.latin.utils.LocaleUtils;
|
2012-08-21 07:34:55 +00:00
|
|
|
|
2011-10-28 10:02:59 +00:00
|
|
|
import java.lang.reflect.Field;
|
2011-06-15 05:40:34 +00:00
|
|
|
import java.util.ArrayList;
|
2014-10-03 14:06:47 +00:00
|
|
|
import java.util.Locale;
|
|
|
|
|
|
|
|
import javax.annotation.Nullable;
|
2011-05-24 11:30:30 +00:00
|
|
|
|
2012-08-29 08:26:00 +00:00
|
|
|
public final class SuggestionSpanUtils {
|
2013-01-06 02:10:27 +00:00
|
|
|
// Note that SuggestionSpan.FLAG_AUTO_CORRECTION has been introduced
|
|
|
|
// in API level 15 (Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1).
|
2014-08-01 07:57:31 +00:00
|
|
|
private static final Field FIELD_FLAG_AUTO_CORRECTION = CompatUtils.getField(
|
2013-01-06 02:10:27 +00:00
|
|
|
SuggestionSpan.class, "FLAG_AUTO_CORRECTION");
|
2014-08-01 07:57:31 +00:00
|
|
|
private static final Integer OBJ_FLAG_AUTO_CORRECTION = (Integer) CompatUtils.getFieldValue(
|
2013-01-06 02:10:27 +00:00
|
|
|
null /* receiver */, null /* defaultValue */, FIELD_FLAG_AUTO_CORRECTION);
|
2011-10-28 10:02:59 +00:00
|
|
|
|
2011-05-25 09:30:31 +00:00
|
|
|
static {
|
2014-07-17 01:41:46 +00:00
|
|
|
if (DebugFlags.DEBUG_ENABLED) {
|
2012-12-17 10:35:09 +00:00
|
|
|
if (OBJ_FLAG_AUTO_CORRECTION == null) {
|
2011-12-05 03:41:11 +00:00
|
|
|
throw new RuntimeException("Field is accidentially null.");
|
2011-10-28 10:02:59 +00:00
|
|
|
}
|
|
|
|
}
|
2011-05-25 09:30:31 +00:00
|
|
|
}
|
2011-05-24 11:30:30 +00:00
|
|
|
|
2012-04-03 09:01:04 +00:00
|
|
|
private SuggestionSpanUtils() {
|
|
|
|
// This utility class is not publicly instantiable.
|
|
|
|
}
|
|
|
|
|
2014-10-03 13:59:38 +00:00
|
|
|
@UsedForTesting
|
2011-09-28 06:59:11 +00:00
|
|
|
public static CharSequence getTextWithAutoCorrectionIndicatorUnderline(
|
2012-10-03 06:19:43 +00:00
|
|
|
final Context context, final String text) {
|
2012-12-17 10:35:09 +00:00
|
|
|
if (TextUtils.isEmpty(text) || OBJ_FLAG_AUTO_CORRECTION == null) {
|
2011-09-28 06:59:11 +00:00
|
|
|
return text;
|
|
|
|
}
|
2012-10-03 06:19:43 +00:00
|
|
|
final Spannable spannable = new SpannableString(text);
|
2014-10-03 13:59:38 +00:00
|
|
|
// TODO: Set locale if it is feasible.
|
2013-01-06 02:10:27 +00:00
|
|
|
final SuggestionSpan suggestionSpan = new SuggestionSpan(context, null /* locale */,
|
2013-07-30 07:00:43 +00:00
|
|
|
new String[] {} /* suggestions */, OBJ_FLAG_AUTO_CORRECTION,
|
2013-01-06 02:10:27 +00:00
|
|
|
SuggestionSpanPickedNotificationReceiver.class);
|
2012-12-17 10:35:09 +00:00
|
|
|
spannable.setSpan(suggestionSpan, 0, text.length(),
|
2011-10-14 16:31:50 +00:00
|
|
|
Spanned.SPAN_EXCLUSIVE_EXCLUSIVE | Spanned.SPAN_COMPOSING);
|
2011-09-28 06:59:11 +00:00
|
|
|
return spannable;
|
|
|
|
}
|
|
|
|
|
2014-10-03 13:59:38 +00:00
|
|
|
@UsedForTesting
|
2012-10-03 06:19:43 +00:00
|
|
|
public static CharSequence getTextWithSuggestionSpan(final Context context,
|
2013-12-19 12:45:08 +00:00
|
|
|
final String pickedWord, final SuggestedWords suggestedWords) {
|
|
|
|
if (TextUtils.isEmpty(pickedWord) || suggestedWords.isEmpty()
|
2014-08-26 09:54:08 +00:00
|
|
|
|| suggestedWords.isPrediction() || suggestedWords.isPunctuationSuggestions()) {
|
2011-06-17 09:25:36 +00:00
|
|
|
return pickedWord;
|
2011-05-24 11:30:30 +00:00
|
|
|
}
|
|
|
|
|
2014-05-23 11:18:17 +00:00
|
|
|
final ArrayList<String> suggestionsList = new ArrayList<>();
|
2011-06-15 05:40:34 +00:00
|
|
|
for (int i = 0; i < suggestedWords.size(); ++i) {
|
2012-12-17 10:35:09 +00:00
|
|
|
if (suggestionsList.size() >= SuggestionSpan.SUGGESTIONS_MAX_SIZE) {
|
2011-06-15 05:40:34 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-12-19 12:45:08 +00:00
|
|
|
final SuggestedWordInfo info = suggestedWords.getInfo(i);
|
2014-05-26 09:45:32 +00:00
|
|
|
if (info.isKindOf(SuggestedWordInfo.KIND_PREDICTION)) {
|
2013-12-19 12:45:08 +00:00
|
|
|
continue;
|
|
|
|
}
|
2012-10-03 06:19:43 +00:00
|
|
|
final String word = suggestedWords.getWord(i);
|
2011-06-17 09:25:36 +00:00
|
|
|
if (!TextUtils.equals(pickedWord, word)) {
|
2011-06-15 05:40:34 +00:00
|
|
|
suggestionsList.add(word.toString());
|
|
|
|
}
|
2011-05-24 11:30:30 +00:00
|
|
|
}
|
2014-10-03 13:59:38 +00:00
|
|
|
// TODO: Set locale if it is feasible.
|
2013-01-06 02:10:27 +00:00
|
|
|
final SuggestionSpan suggestionSpan = new SuggestionSpan(context, null /* locale */,
|
|
|
|
suggestionsList.toArray(new String[suggestionsList.size()]), 0 /* flags */,
|
2012-12-17 10:35:09 +00:00
|
|
|
SuggestionSpanPickedNotificationReceiver.class);
|
2013-12-19 12:45:08 +00:00
|
|
|
final Spannable spannable = new SpannableString(pickedWord);
|
2012-12-17 10:35:09 +00:00
|
|
|
spannable.setSpan(suggestionSpan, 0, pickedWord.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
|
2011-05-24 11:30:30 +00:00
|
|
|
return spannable;
|
|
|
|
}
|
2014-10-03 14:06:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns first {@link Locale} found in the given array of {@link SuggestionSpan}.
|
|
|
|
* @param suggestionSpans the array of {@link SuggestionSpan} to be examined.
|
|
|
|
* @return the first {@link Locale} found in {@code suggestionSpans}. {@code null} when not
|
|
|
|
* found.
|
|
|
|
*/
|
|
|
|
@UsedForTesting
|
|
|
|
@Nullable
|
|
|
|
public static Locale findFirstLocaleFromSuggestionSpans(
|
|
|
|
final SuggestionSpan[] suggestionSpans) {
|
|
|
|
for (final SuggestionSpan suggestionSpan : suggestionSpans) {
|
|
|
|
final String localeString = suggestionSpan.getLocale();
|
|
|
|
if (TextUtils.isEmpty(localeString)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
final Locale locale = LocaleUtils.constructLocaleFromString(localeString);
|
|
|
|
if (locale == null) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
return locale;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
2011-05-24 11:30:30 +00:00
|
|
|
}
|