2013-08-28 00:14:00 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 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.keyboard;
|
|
|
|
|
|
|
|
import static com.android.inputmethod.latin.Constants.NOT_A_COORDINATE;
|
|
|
|
|
|
|
|
import android.content.Context;
|
2013-09-16 08:13:41 +00:00
|
|
|
import android.content.SharedPreferences;
|
2013-08-28 00:14:00 +00:00
|
|
|
import android.content.res.ColorStateList;
|
|
|
|
import android.content.res.Resources;
|
|
|
|
import android.content.res.TypedArray;
|
2013-09-13 10:06:22 +00:00
|
|
|
import android.graphics.Rect;
|
2013-09-06 11:16:05 +00:00
|
|
|
import android.os.Build;
|
2013-09-16 08:13:41 +00:00
|
|
|
import android.preference.PreferenceManager;
|
2013-08-28 00:14:00 +00:00
|
|
|
import android.support.v4.view.PagerAdapter;
|
|
|
|
import android.support.v4.view.ViewPager;
|
|
|
|
import android.util.AttributeSet;
|
2013-09-12 12:11:43 +00:00
|
|
|
import android.util.Log;
|
2013-09-13 12:49:53 +00:00
|
|
|
import android.util.Pair;
|
2013-08-28 00:14:00 +00:00
|
|
|
import android.util.SparseArray;
|
|
|
|
import android.view.LayoutInflater;
|
|
|
|
import android.view.View;
|
|
|
|
import android.view.ViewGroup;
|
|
|
|
import android.widget.ImageView;
|
|
|
|
import android.widget.LinearLayout;
|
|
|
|
import android.widget.TabHost;
|
|
|
|
import android.widget.TabHost.OnTabChangeListener;
|
|
|
|
import android.widget.TextView;
|
|
|
|
|
2013-09-13 07:30:16 +00:00
|
|
|
import com.android.inputmethod.keyboard.internal.DynamicGridKeyboard;
|
2013-08-28 00:14:00 +00:00
|
|
|
import com.android.inputmethod.keyboard.internal.ScrollKeyboardView;
|
|
|
|
import com.android.inputmethod.keyboard.internal.ScrollViewWithNotifier;
|
|
|
|
import com.android.inputmethod.latin.Constants;
|
|
|
|
import com.android.inputmethod.latin.R;
|
|
|
|
import com.android.inputmethod.latin.SubtypeSwitcher;
|
2013-09-16 08:13:41 +00:00
|
|
|
import com.android.inputmethod.latin.settings.Settings;
|
2013-08-28 00:14:00 +00:00
|
|
|
import com.android.inputmethod.latin.utils.CollectionUtils;
|
|
|
|
import com.android.inputmethod.latin.utils.ResourceUtils;
|
|
|
|
|
2013-09-12 12:11:43 +00:00
|
|
|
import java.util.ArrayList;
|
2013-09-13 10:06:22 +00:00
|
|
|
import java.util.Arrays;
|
|
|
|
import java.util.Comparator;
|
2013-08-28 00:14:00 +00:00
|
|
|
import java.util.HashMap;
|
2013-09-13 10:06:22 +00:00
|
|
|
import java.util.concurrent.ConcurrentHashMap;
|
2013-08-28 00:14:00 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* View class to implement Emoji keyboards.
|
|
|
|
* The Emoji keyboard consists of group of views {@link R.layout#emoji_keyboard_view}.
|
|
|
|
* <ol>
|
|
|
|
* <li> Emoji category tabs.
|
|
|
|
* <li> Delete button.
|
|
|
|
* <li> Emoji keyboard pages that can be scrolled by swiping horizontally or by selecting a tab.
|
|
|
|
* <li> Back to main keyboard button and enter button.
|
|
|
|
* </ol>
|
|
|
|
* Because of the above reasons, this class doesn't extend {@link KeyboardView}.
|
|
|
|
*/
|
|
|
|
public final class EmojiKeyboardView extends LinearLayout implements OnTabChangeListener,
|
|
|
|
ViewPager.OnPageChangeListener, View.OnClickListener,
|
|
|
|
ScrollKeyboardView.OnKeyClickListener {
|
2013-09-12 12:11:43 +00:00
|
|
|
private static final String TAG = EmojiKeyboardView.class.getSimpleName();
|
2013-08-28 00:14:00 +00:00
|
|
|
private final int mKeyBackgroundId;
|
2013-09-10 08:55:44 +00:00
|
|
|
private final int mEmojiFunctionalKeyBackgroundId;
|
2013-09-12 12:11:43 +00:00
|
|
|
private final KeyboardLayoutSet mLayoutSet;
|
2013-08-28 00:14:00 +00:00
|
|
|
private final ColorStateList mTabLabelColor;
|
2013-09-12 12:11:43 +00:00
|
|
|
private EmojiKeyboardAdapter mEmojiKeyboardAdapter;
|
2013-08-28 00:14:00 +00:00
|
|
|
|
|
|
|
private TabHost mTabHost;
|
|
|
|
private ViewPager mEmojiPager;
|
|
|
|
|
|
|
|
private KeyboardActionListener mKeyboardActionListener = KeyboardActionListener.EMPTY_LISTENER;
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
private static final int CATEGORY_ID_UNSPECIFIED = -1;
|
|
|
|
public static final int CATEGORY_ID_RECENTS = 0;
|
|
|
|
public static final int CATEGORY_ID_PEOPLE = 1;
|
|
|
|
public static final int CATEGORY_ID_OBJECTS = 2;
|
|
|
|
public static final int CATEGORY_ID_NATURE = 3;
|
|
|
|
public static final int CATEGORY_ID_PLACES = 4;
|
|
|
|
public static final int CATEGORY_ID_SYMBOLS = 5;
|
|
|
|
public static final int CATEGORY_ID_EMOTICONS = 6;
|
2013-09-13 10:06:22 +00:00
|
|
|
|
2013-09-13 12:49:53 +00:00
|
|
|
private static class CategoryProperties {
|
2013-09-16 08:13:41 +00:00
|
|
|
public int mCategoryId;
|
2013-09-13 12:49:53 +00:00
|
|
|
public int mPageCount;
|
2013-09-16 08:13:41 +00:00
|
|
|
public CategoryProperties(final int categoryId, final int pageCount) {
|
|
|
|
mCategoryId = categoryId;
|
2013-09-13 12:49:53 +00:00
|
|
|
mPageCount = pageCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-12 12:11:43 +00:00
|
|
|
private static class EmojiCategory {
|
|
|
|
private static final String[] sCategoryName = {
|
|
|
|
"recents",
|
|
|
|
"people",
|
|
|
|
"objects",
|
|
|
|
"nature",
|
|
|
|
"places",
|
|
|
|
"symbols",
|
|
|
|
"emoticons" };
|
|
|
|
private static final int[] sCategoryIcon = new int[] {
|
|
|
|
R.drawable.ic_emoji_recent_light,
|
|
|
|
R.drawable.ic_emoji_people_light,
|
|
|
|
R.drawable.ic_emoji_objects_light,
|
|
|
|
R.drawable.ic_emoji_nature_light,
|
|
|
|
R.drawable.ic_emoji_places_light,
|
|
|
|
R.drawable.ic_emoji_symbols_light,
|
|
|
|
0 };
|
|
|
|
private static final String[] sCategoryLabel =
|
|
|
|
{ null, null, null, null, null, null, ":-)" };
|
|
|
|
private static final int[] sCategoryElementId = {
|
|
|
|
KeyboardId.ELEMENT_EMOJI_RECENTS,
|
|
|
|
KeyboardId.ELEMENT_EMOJI_CATEGORY1,
|
|
|
|
KeyboardId.ELEMENT_EMOJI_CATEGORY2,
|
|
|
|
KeyboardId.ELEMENT_EMOJI_CATEGORY3,
|
|
|
|
KeyboardId.ELEMENT_EMOJI_CATEGORY4,
|
|
|
|
KeyboardId.ELEMENT_EMOJI_CATEGORY5,
|
2013-09-16 08:13:41 +00:00
|
|
|
KeyboardId.ELEMENT_EMOJI_CATEGORY6 };
|
|
|
|
private final SharedPreferences mPrefs;
|
2013-09-13 12:49:53 +00:00
|
|
|
private final int mMaxPageKeyCount;
|
2013-09-13 10:06:22 +00:00
|
|
|
private final KeyboardLayoutSet mLayoutSet;
|
2013-09-12 12:11:43 +00:00
|
|
|
private final HashMap<String, Integer> mCategoryNameToIdMap = CollectionUtils.newHashMap();
|
2013-09-13 12:49:53 +00:00
|
|
|
private final ArrayList<CategoryProperties> mShownCategories =
|
|
|
|
CollectionUtils.newArrayList();
|
2013-09-13 10:06:22 +00:00
|
|
|
private final ConcurrentHashMap<Long, DynamicGridKeyboard>
|
|
|
|
mCategoryKeyboardMap = new ConcurrentHashMap<Long, DynamicGridKeyboard>();
|
2013-09-12 12:11:43 +00:00
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
private int mCurrentCategoryId = CATEGORY_ID_UNSPECIFIED;
|
|
|
|
private int mCurrentCategoryPageId = 0;
|
2013-09-13 10:06:22 +00:00
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public EmojiCategory(final SharedPreferences prefs, final Resources res,
|
|
|
|
final KeyboardLayoutSet layoutSet) {
|
|
|
|
mPrefs = prefs;
|
2013-09-13 12:49:53 +00:00
|
|
|
mMaxPageKeyCount = res.getInteger(R.integer.emoji_keyboard_max_key_count);
|
2013-09-13 10:06:22 +00:00
|
|
|
mLayoutSet = layoutSet;
|
2013-09-12 12:11:43 +00:00
|
|
|
for (int i = 0; i < sCategoryName.length; ++i) {
|
|
|
|
mCategoryNameToIdMap.put(sCategoryName[i], i);
|
|
|
|
}
|
2013-09-16 08:13:41 +00:00
|
|
|
addShownCategoryId(CATEGORY_ID_RECENTS);
|
2013-09-12 12:11:43 +00:00
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {
|
2013-09-16 08:13:41 +00:00
|
|
|
addShownCategoryId(CATEGORY_ID_PEOPLE);
|
|
|
|
addShownCategoryId(CATEGORY_ID_OBJECTS);
|
|
|
|
addShownCategoryId(CATEGORY_ID_NATURE);
|
|
|
|
addShownCategoryId(CATEGORY_ID_PLACES);
|
|
|
|
mCurrentCategoryId = CATEGORY_ID_PEOPLE;
|
2013-09-12 12:11:43 +00:00
|
|
|
} else {
|
2013-09-16 08:13:41 +00:00
|
|
|
mCurrentCategoryId = CATEGORY_ID_SYMBOLS;
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
2013-09-16 08:13:41 +00:00
|
|
|
addShownCategoryId(CATEGORY_ID_SYMBOLS);
|
|
|
|
addShownCategoryId(CATEGORY_ID_EMOTICONS);
|
|
|
|
getKeyboard(CATEGORY_ID_RECENTS, 0 /* cagetoryPageId */)
|
|
|
|
.loadRecentKeys(mCategoryKeyboardMap.values());
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
private void addShownCategoryId(int categoryId) {
|
|
|
|
// Load a keyboard of categoryId
|
|
|
|
getKeyboard(categoryId, 0 /* cagetoryPageId */);
|
2013-09-13 12:49:53 +00:00
|
|
|
final CategoryProperties properties =
|
2013-09-16 08:13:41 +00:00
|
|
|
new CategoryProperties(categoryId, getCategoryPageCount(categoryId));
|
2013-09-13 12:49:53 +00:00
|
|
|
mShownCategories.add(properties);
|
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public String getCategoryName(int categoryId, int categoryPageId) {
|
|
|
|
return sCategoryName[categoryId] + "-" + categoryPageId;
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public int getCategoryId(String name) {
|
2013-09-13 12:49:53 +00:00
|
|
|
final String[] strings = name.split("-");
|
|
|
|
return mCategoryNameToIdMap.get(strings[0]);
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public int getCategoryIcon(int categoryId) {
|
|
|
|
return sCategoryIcon[categoryId];
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public String getCategoryLabel(int categoryId) {
|
|
|
|
return sCategoryLabel[categoryId];
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-13 12:49:53 +00:00
|
|
|
public ArrayList<CategoryProperties> getShownCategories() {
|
2013-09-12 12:11:43 +00:00
|
|
|
return mShownCategories;
|
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public int getCurrentCategoryId() {
|
|
|
|
return mCurrentCategoryId;
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public void setCurrentCategoryId(int categoryId) {
|
|
|
|
mCurrentCategoryId = categoryId;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setCurrentCategoryPageId(int id) {
|
|
|
|
mCurrentCategoryPageId = id;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void saveLastTypedCategoryPage() {
|
|
|
|
Settings.writeEmojiCategoryLastTypedId(
|
|
|
|
mPrefs, mCurrentCategoryId, mCurrentCategoryPageId);
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isInRecentTab() {
|
2013-09-16 08:13:41 +00:00
|
|
|
return mCurrentCategoryId == CATEGORY_ID_RECENTS;
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public int getTabIdFromCategoryId(int categoryId) {
|
2013-09-12 12:11:43 +00:00
|
|
|
for (int i = 0; i < mShownCategories.size(); ++i) {
|
2013-09-16 08:13:41 +00:00
|
|
|
if (mShownCategories.get(i).mCategoryId == categoryId) {
|
2013-09-12 12:11:43 +00:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
2013-09-16 08:13:41 +00:00
|
|
|
Log.w(TAG, "categoryId not found: " + categoryId);
|
2013-09-12 12:11:43 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
// Returns the view pager's page position for the categoryId
|
|
|
|
public int getPageIdFromCategoryId(int categoryId) {
|
|
|
|
final int lastSavedCategoryPageId =
|
|
|
|
Settings.readEmojiCategoryLastTypedId(mPrefs, categoryId);
|
2013-09-13 12:49:53 +00:00
|
|
|
int sum = 0;
|
|
|
|
for (int i = 0; i < mShownCategories.size(); ++i) {
|
|
|
|
final CategoryProperties props = mShownCategories.get(i);
|
2013-09-16 08:13:41 +00:00
|
|
|
if (props.mCategoryId == categoryId) {
|
|
|
|
return sum + lastSavedCategoryPageId;
|
2013-09-13 12:49:53 +00:00
|
|
|
}
|
|
|
|
sum += props.mPageCount;
|
|
|
|
}
|
2013-09-16 08:13:41 +00:00
|
|
|
Log.w(TAG, "categoryId not found: " + categoryId);
|
2013-09-13 12:49:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-09-12 12:11:43 +00:00
|
|
|
public int getRecentTabId() {
|
2013-09-16 08:13:41 +00:00
|
|
|
return getTabIdFromCategoryId(CATEGORY_ID_RECENTS);
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
private int getCategoryPageCount(int categoryId) {
|
|
|
|
final Keyboard keyboard = mLayoutSet.getKeyboard(sCategoryElementId[categoryId]);
|
2013-09-13 12:49:53 +00:00
|
|
|
return (keyboard.getKeys().length - 1) / mMaxPageKeyCount + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a pair of the category id and the category page id from the view pager's page
|
|
|
|
// position. The category page id is numbered in each category. And the view page position
|
|
|
|
// is the position of the current shown page in the view pager which contains all pages of
|
|
|
|
// all categories.
|
|
|
|
public Pair<Integer, Integer> getCategoryIdAndPageIdFromPagePosition(int position) {
|
|
|
|
int sum = 0;
|
|
|
|
for (CategoryProperties properties : mShownCategories) {
|
|
|
|
final int temp = sum;
|
|
|
|
sum += properties.mPageCount;
|
|
|
|
if (sum > position) {
|
2013-09-16 08:13:41 +00:00
|
|
|
return new Pair<Integer, Integer>(properties.mCategoryId, position - temp);
|
2013-09-13 12:49:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a keyboard from the view pager's page position.
|
|
|
|
public DynamicGridKeyboard getKeyboardFromPagePosition(int position) {
|
|
|
|
final Pair<Integer, Integer> categoryAndId =
|
|
|
|
getCategoryIdAndPageIdFromPagePosition(position);
|
|
|
|
if (categoryAndId != null) {
|
|
|
|
return getKeyboard(categoryAndId.first, categoryAndId.second);
|
|
|
|
}
|
|
|
|
return null;
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
public DynamicGridKeyboard getKeyboard(int categoryId, int id) {
|
2013-09-13 10:06:22 +00:00
|
|
|
synchronized(mCategoryKeyboardMap) {
|
2013-09-16 08:13:41 +00:00
|
|
|
final long key = (((long) categoryId) << Constants.MAX_INT_BIT_COUNT) | id;
|
2013-09-13 10:06:22 +00:00
|
|
|
final DynamicGridKeyboard kbd;
|
|
|
|
if (!mCategoryKeyboardMap.containsKey(key)) {
|
2013-09-16 08:13:41 +00:00
|
|
|
if (categoryId != CATEGORY_ID_RECENTS) {
|
2013-09-13 10:06:22 +00:00
|
|
|
final Keyboard keyboard =
|
2013-09-16 08:13:41 +00:00
|
|
|
mLayoutSet.getKeyboard(sCategoryElementId[categoryId]);
|
2013-09-13 12:49:53 +00:00
|
|
|
final Key[][] sortedKeys = sortKeys(keyboard.getKeys(), mMaxPageKeyCount);
|
|
|
|
for (int i = 0; i < sortedKeys.length; ++i) {
|
2013-09-16 08:13:41 +00:00
|
|
|
final DynamicGridKeyboard tempKbd = new DynamicGridKeyboard(mPrefs,
|
2013-09-13 12:49:53 +00:00
|
|
|
mLayoutSet.getKeyboard(KeyboardId.ELEMENT_EMOJI_RECENTS),
|
2013-09-16 08:13:41 +00:00
|
|
|
mMaxPageKeyCount, categoryId, i /* categoryPageId */);
|
2013-09-13 12:49:53 +00:00
|
|
|
for (Key emojiKey : sortedKeys[i]) {
|
|
|
|
if (emojiKey == null) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
tempKbd.addKeyLast(emojiKey);
|
2013-09-13 10:06:22 +00:00
|
|
|
}
|
2013-09-16 08:13:41 +00:00
|
|
|
mCategoryKeyboardMap.put((((long) categoryId)
|
|
|
|
<< Constants.MAX_INT_BIT_COUNT) | i, tempKbd);
|
2013-09-13 10:06:22 +00:00
|
|
|
}
|
2013-09-13 12:49:53 +00:00
|
|
|
kbd = mCategoryKeyboardMap.get(key);
|
2013-09-13 10:06:22 +00:00
|
|
|
} else {
|
2013-09-16 08:13:41 +00:00
|
|
|
kbd = new DynamicGridKeyboard(mPrefs,
|
2013-09-13 10:06:22 +00:00
|
|
|
mLayoutSet.getKeyboard(KeyboardId.ELEMENT_EMOJI_RECENTS),
|
2013-09-16 08:13:41 +00:00
|
|
|
mMaxPageKeyCount, categoryId, 0 /* categoryPageId */);
|
2013-09-13 12:49:53 +00:00
|
|
|
mCategoryKeyboardMap.put(key, kbd);
|
2013-09-13 10:06:22 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
kbd = mCategoryKeyboardMap.get(key);
|
|
|
|
}
|
|
|
|
return kbd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-13 12:49:53 +00:00
|
|
|
public int getTotalPageCountOfAllCategories() {
|
|
|
|
int sum = 0;
|
|
|
|
for (CategoryProperties properties : mShownCategories) {
|
|
|
|
sum += properties.mPageCount;
|
|
|
|
}
|
|
|
|
return sum;
|
|
|
|
}
|
|
|
|
|
2013-09-13 10:06:22 +00:00
|
|
|
private Key[][] sortKeys(Key[] inKeys, int maxPageCount) {
|
|
|
|
Key[] keys = Arrays.copyOf(inKeys, inKeys.length);
|
|
|
|
Arrays.sort(keys, 0, keys.length, new Comparator<Key>() {
|
|
|
|
@Override
|
|
|
|
public int compare(Key lhs, Key rhs) {
|
|
|
|
final Rect lHitBox = lhs.getHitBox();
|
|
|
|
final Rect rHitBox = rhs.getHitBox();
|
|
|
|
if (lHitBox.top < rHitBox.top) {
|
|
|
|
return -1;
|
|
|
|
} else if (lHitBox.top > rHitBox.top) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (lHitBox.left < rHitBox.left) {
|
|
|
|
return -1;
|
|
|
|
} else if (lHitBox.left > rHitBox.left) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (lhs.getCode() == rhs.getCode()) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return lhs.getCode() < rhs.getCode() ? -1 : 1;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
final int pageCount = (keys.length - 1) / maxPageCount + 1;
|
|
|
|
final Key[][] retval = new Key[pageCount][maxPageCount];
|
|
|
|
for (int i = 0; i < keys.length; ++i) {
|
|
|
|
retval[i / maxPageCount][i % maxPageCount] = keys[i];
|
|
|
|
}
|
|
|
|
return retval;
|
2013-09-12 12:11:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-13 10:06:22 +00:00
|
|
|
private final EmojiCategory mEmojiCategory;
|
2013-08-28 00:14:00 +00:00
|
|
|
|
|
|
|
public EmojiKeyboardView(final Context context, final AttributeSet attrs) {
|
|
|
|
this(context, attrs, R.attr.emojiKeyboardViewStyle);
|
|
|
|
}
|
|
|
|
|
|
|
|
public EmojiKeyboardView(final Context context, final AttributeSet attrs, final int defStyle) {
|
|
|
|
super(context, attrs, defStyle);
|
|
|
|
final TypedArray keyboardViewAttr = context.obtainStyledAttributes(attrs,
|
|
|
|
R.styleable.KeyboardView, defStyle, R.style.KeyboardView);
|
|
|
|
mKeyBackgroundId = keyboardViewAttr.getResourceId(
|
|
|
|
R.styleable.KeyboardView_keyBackground, 0);
|
2013-09-10 08:55:44 +00:00
|
|
|
mEmojiFunctionalKeyBackgroundId = keyboardViewAttr.getResourceId(
|
|
|
|
R.styleable.KeyboardView_keyBackgroundEmojiFunctional, 0);
|
2013-08-28 00:14:00 +00:00
|
|
|
keyboardViewAttr.recycle();
|
|
|
|
final TypedArray emojiKeyboardViewAttr = context.obtainStyledAttributes(attrs,
|
|
|
|
R.styleable.EmojiKeyboardView, defStyle, R.style.EmojiKeyboardView);
|
|
|
|
mTabLabelColor = emojiKeyboardViewAttr.getColorStateList(
|
|
|
|
R.styleable.EmojiKeyboardView_emojiTabLabelColor);
|
|
|
|
emojiKeyboardViewAttr.recycle();
|
|
|
|
final KeyboardLayoutSet.Builder builder = new KeyboardLayoutSet.Builder(
|
|
|
|
context, null /* editorInfo */);
|
|
|
|
final Resources res = context.getResources();
|
2013-09-13 12:49:53 +00:00
|
|
|
final EmojiLayoutParams emojiLp = new EmojiLayoutParams(res);
|
2013-08-28 00:14:00 +00:00
|
|
|
builder.setSubtype(SubtypeSwitcher.getInstance().getEmojiSubtype());
|
|
|
|
builder.setKeyboardGeometry(ResourceUtils.getDefaultKeyboardWidth(res),
|
2013-09-13 12:49:53 +00:00
|
|
|
emojiLp.mEmojiKeyboardHeight);
|
2013-08-28 00:14:00 +00:00
|
|
|
builder.setOptions(false, false, false /* lanuageSwitchKeyEnabled */);
|
2013-09-12 12:11:43 +00:00
|
|
|
mLayoutSet = builder.build();
|
2013-09-16 08:13:41 +00:00
|
|
|
mEmojiCategory = new EmojiCategory(PreferenceManager.getDefaultSharedPreferences(context),
|
|
|
|
context.getResources(), builder.build());
|
2013-08-28 00:14:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onMeasure(final int widthMeasureSpec, final int heightMeasureSpec) {
|
|
|
|
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
|
|
|
|
final Resources res = getContext().getResources();
|
|
|
|
// The main keyboard expands to the entire this {@link KeyboardView}.
|
|
|
|
final int width = ResourceUtils.getDefaultKeyboardWidth(res)
|
|
|
|
+ getPaddingLeft() + getPaddingRight();
|
|
|
|
final int height = ResourceUtils.getDefaultKeyboardHeight(res)
|
|
|
|
+ res.getDimensionPixelSize(R.dimen.suggestions_strip_height)
|
|
|
|
+ getPaddingTop() + getPaddingBottom();
|
|
|
|
setMeasuredDimension(width, height);
|
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
private void addTab(final TabHost host, final int categoryId) {
|
|
|
|
final String tabId = mEmojiCategory.getCategoryName(categoryId, 0 /* categoryPageId */);
|
2013-08-28 00:14:00 +00:00
|
|
|
final TabHost.TabSpec tspec = host.newTabSpec(tabId);
|
|
|
|
tspec.setContent(R.id.emoji_keyboard_dummy);
|
2013-09-16 08:13:41 +00:00
|
|
|
if (mEmojiCategory.getCategoryIcon(categoryId) != 0) {
|
2013-08-28 00:14:00 +00:00
|
|
|
final ImageView iconView = (ImageView)LayoutInflater.from(getContext()).inflate(
|
|
|
|
R.layout.emoji_keyboard_tab_icon, null);
|
2013-09-16 08:13:41 +00:00
|
|
|
iconView.setImageResource(mEmojiCategory.getCategoryIcon(categoryId));
|
2013-08-28 00:14:00 +00:00
|
|
|
tspec.setIndicator(iconView);
|
|
|
|
}
|
2013-09-16 08:13:41 +00:00
|
|
|
if (mEmojiCategory.getCategoryLabel(categoryId) != null) {
|
2013-08-28 00:14:00 +00:00
|
|
|
final TextView textView = (TextView)LayoutInflater.from(getContext()).inflate(
|
|
|
|
R.layout.emoji_keyboard_tab_label, null);
|
2013-09-16 08:13:41 +00:00
|
|
|
textView.setText(mEmojiCategory.getCategoryLabel(categoryId));
|
2013-08-28 00:14:00 +00:00
|
|
|
textView.setTextColor(mTabLabelColor);
|
|
|
|
tspec.setIndicator(textView);
|
|
|
|
}
|
|
|
|
host.addTab(tspec);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onFinishInflate() {
|
|
|
|
mTabHost = (TabHost)findViewById(R.id.emoji_category_tabhost);
|
|
|
|
mTabHost.setup();
|
2013-09-13 12:49:53 +00:00
|
|
|
for (final CategoryProperties properties : mEmojiCategory.getShownCategories()) {
|
2013-09-16 08:13:41 +00:00
|
|
|
addTab(mTabHost, properties.mCategoryId);
|
2013-09-06 11:16:05 +00:00
|
|
|
}
|
2013-08-28 00:14:00 +00:00
|
|
|
mTabHost.setOnTabChangedListener(this);
|
|
|
|
mTabHost.getTabWidget().setStripEnabled(true);
|
|
|
|
|
2013-09-12 12:11:43 +00:00
|
|
|
mEmojiKeyboardAdapter = new EmojiKeyboardAdapter(mEmojiCategory, mLayoutSet, this);
|
|
|
|
|
2013-08-28 00:14:00 +00:00
|
|
|
mEmojiPager = (ViewPager)findViewById(R.id.emoji_keyboard_pager);
|
|
|
|
mEmojiPager.setAdapter(mEmojiKeyboardAdapter);
|
|
|
|
mEmojiPager.setOnPageChangeListener(this);
|
|
|
|
mEmojiPager.setOffscreenPageLimit(0);
|
|
|
|
final Resources res = getResources();
|
2013-09-10 08:55:44 +00:00
|
|
|
final EmojiLayoutParams emojiLp = new EmojiLayoutParams(res);
|
|
|
|
emojiLp.setPagerProps(mEmojiPager);
|
2013-08-28 00:14:00 +00:00
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
setCurrentCategoryId(mEmojiCategory.getCurrentCategoryId(), true /* force */);
|
2013-08-28 00:14:00 +00:00
|
|
|
|
2013-09-10 08:55:44 +00:00
|
|
|
final LinearLayout actionBar = (LinearLayout)findViewById(R.id.emoji_action_bar);
|
|
|
|
emojiLp.setActionBarProps(actionBar);
|
|
|
|
|
2013-08-28 00:14:00 +00:00
|
|
|
// TODO: Implement auto repeat, using View.OnTouchListener?
|
2013-09-10 08:55:44 +00:00
|
|
|
final ImageView deleteKey = (ImageView)findViewById(R.id.emoji_keyboard_delete);
|
|
|
|
deleteKey.setBackgroundResource(mEmojiFunctionalKeyBackgroundId);
|
2013-08-28 00:14:00 +00:00
|
|
|
deleteKey.setTag(Constants.CODE_DELETE);
|
|
|
|
deleteKey.setOnClickListener(this);
|
2013-09-10 08:55:44 +00:00
|
|
|
final ImageView alphabetKey = (ImageView)findViewById(R.id.emoji_keyboard_alphabet);
|
|
|
|
alphabetKey.setBackgroundResource(mEmojiFunctionalKeyBackgroundId);
|
2013-08-28 00:14:00 +00:00
|
|
|
alphabetKey.setTag(Constants.CODE_SWITCH_ALPHA_SYMBOL);
|
|
|
|
alphabetKey.setOnClickListener(this);
|
2013-09-10 08:55:44 +00:00
|
|
|
final ImageView spaceKey = (ImageView)findViewById(R.id.emoji_keyboard_space);
|
|
|
|
spaceKey.setBackgroundResource(mKeyBackgroundId);
|
|
|
|
spaceKey.setTag(Constants.CODE_SPACE);
|
|
|
|
spaceKey.setOnClickListener(this);
|
|
|
|
emojiLp.setKeyProps(spaceKey);
|
|
|
|
final ImageView sendKey = (ImageView)findViewById(R.id.emoji_keyboard_send);
|
|
|
|
sendKey.setBackgroundResource(mEmojiFunctionalKeyBackgroundId);
|
2013-08-28 00:14:00 +00:00
|
|
|
sendKey.setTag(Constants.CODE_ENTER);
|
|
|
|
sendKey.setOnClickListener(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onTabChanged(final String tabId) {
|
2013-09-16 08:13:41 +00:00
|
|
|
final int categoryId = mEmojiCategory.getCategoryId(tabId);
|
|
|
|
setCurrentCategoryId(categoryId, false /* force */);
|
2013-08-28 00:14:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPageSelected(final int position) {
|
2013-09-16 08:13:41 +00:00
|
|
|
final Pair<Integer, Integer> newPos =
|
|
|
|
mEmojiCategory.getCategoryIdAndPageIdFromPagePosition(position);
|
|
|
|
setCurrentCategoryId(newPos.first /* categoryId */, false /* force */);
|
|
|
|
mEmojiCategory.setCurrentCategoryPageId(newPos.second /* categoryPageId */);
|
2013-08-28 00:14:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPageScrollStateChanged(final int state) {
|
|
|
|
// Ignore this message. Only want the actual page selected.
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onPageScrolled(final int position, final float positionOffset,
|
|
|
|
final int positionOffsetPixels) {
|
|
|
|
// Ignore this message. Only want the actual page selected.
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onClick(final View v) {
|
|
|
|
if (v.getTag() instanceof Integer) {
|
|
|
|
final int code = (Integer)v.getTag();
|
|
|
|
registerCode(code);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void registerCode(final int code) {
|
|
|
|
mKeyboardActionListener.onPressKey(code, 0 /* repeatCount */, true /* isSinglePointer */);
|
|
|
|
mKeyboardActionListener.onCodeInput(code, NOT_A_COORDINATE, NOT_A_COORDINATE);
|
|
|
|
mKeyboardActionListener.onReleaseKey(code, false /* withSliding */);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onKeyClick(final Key key) {
|
|
|
|
mEmojiKeyboardAdapter.addRecentKey(key);
|
2013-09-16 08:13:41 +00:00
|
|
|
mEmojiCategory.saveLastTypedCategoryPage();
|
2013-08-28 00:14:00 +00:00
|
|
|
final int code = key.getCode();
|
|
|
|
if (code == Constants.CODE_OUTPUT_TEXT) {
|
|
|
|
mKeyboardActionListener.onTextInput(key.getOutputText());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
registerCode(code);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setHardwareAcceleratedDrawingEnabled(final boolean enabled) {
|
|
|
|
// TODO:
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setKeyboardActionListener(final KeyboardActionListener listener) {
|
|
|
|
mKeyboardActionListener = listener;
|
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
private void setCurrentCategoryId(final int categoryId, final boolean force) {
|
|
|
|
if (mEmojiCategory.getCurrentCategoryId() == categoryId && !force) {
|
2013-08-28 00:14:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-09-16 08:13:41 +00:00
|
|
|
mEmojiCategory.setCurrentCategoryId(categoryId);
|
|
|
|
final int newTabId = mEmojiCategory.getTabIdFromCategoryId(categoryId);
|
|
|
|
final int newCategoryPageId = mEmojiCategory.getPageIdFromCategoryId(categoryId);
|
2013-09-13 12:49:53 +00:00
|
|
|
if (force || mEmojiCategory.getCategoryIdAndPageIdFromPagePosition(
|
2013-09-16 08:13:41 +00:00
|
|
|
mEmojiPager.getCurrentItem()).first != categoryId) {
|
2013-09-13 12:49:53 +00:00
|
|
|
mEmojiPager.setCurrentItem(newCategoryPageId, true /* smoothScroll */);
|
2013-08-28 00:14:00 +00:00
|
|
|
}
|
2013-09-12 12:11:43 +00:00
|
|
|
if (force || mTabHost.getCurrentTab() != newTabId) {
|
|
|
|
mTabHost.setCurrentTab(newTabId);
|
2013-08-28 00:14:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class EmojiKeyboardAdapter extends PagerAdapter {
|
|
|
|
private final ScrollKeyboardView.OnKeyClickListener mListener;
|
2013-09-13 07:30:16 +00:00
|
|
|
private final DynamicGridKeyboard mRecentsKeyboard;
|
2013-08-28 00:14:00 +00:00
|
|
|
private final SparseArray<ScrollKeyboardView> mActiveKeyboardView =
|
|
|
|
CollectionUtils.newSparseArray();
|
2013-09-12 12:11:43 +00:00
|
|
|
private final EmojiCategory mEmojiCategory;
|
|
|
|
private int mActivePosition = 0;
|
2013-08-28 00:14:00 +00:00
|
|
|
|
2013-09-12 12:11:43 +00:00
|
|
|
public EmojiKeyboardAdapter(final EmojiCategory emojiCategory,
|
|
|
|
final KeyboardLayoutSet layoutSet,
|
2013-08-28 00:14:00 +00:00
|
|
|
final ScrollKeyboardView.OnKeyClickListener listener) {
|
2013-09-12 12:11:43 +00:00
|
|
|
mEmojiCategory = emojiCategory;
|
2013-08-28 00:14:00 +00:00
|
|
|
mListener = listener;
|
2013-09-16 08:13:41 +00:00
|
|
|
mRecentsKeyboard = mEmojiCategory.getKeyboard(CATEGORY_ID_RECENTS, 0);
|
2013-08-28 00:14:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void addRecentKey(final Key key) {
|
2013-09-12 12:11:43 +00:00
|
|
|
if (mEmojiCategory.isInRecentTab()) {
|
2013-08-28 00:14:00 +00:00
|
|
|
return;
|
|
|
|
}
|
2013-09-13 10:06:22 +00:00
|
|
|
mRecentsKeyboard.addKeyFirst(key);
|
2013-09-12 12:11:43 +00:00
|
|
|
final KeyboardView recentKeyboardView =
|
|
|
|
mActiveKeyboardView.get(mEmojiCategory.getRecentTabId());
|
2013-08-28 00:14:00 +00:00
|
|
|
if (recentKeyboardView != null) {
|
|
|
|
recentKeyboardView.invalidateAllKeys();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int getCount() {
|
2013-09-13 12:49:53 +00:00
|
|
|
return mEmojiCategory.getTotalPageCountOfAllCategories();
|
2013-08-28 00:14:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void setPrimaryItem(final View container, final int position, final Object object) {
|
|
|
|
if (mActivePosition == position) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
final ScrollKeyboardView oldKeyboardView = mActiveKeyboardView.get(mActivePosition);
|
|
|
|
if (oldKeyboardView != null) {
|
|
|
|
oldKeyboardView.releaseCurrentKey();
|
|
|
|
oldKeyboardView.deallocateMemory();
|
|
|
|
}
|
|
|
|
mActivePosition = position;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Object instantiateItem(final ViewGroup container, final int position) {
|
2013-09-13 10:06:22 +00:00
|
|
|
final Keyboard keyboard =
|
2013-09-13 12:49:53 +00:00
|
|
|
mEmojiCategory.getKeyboardFromPagePosition(position);
|
2013-08-28 00:14:00 +00:00
|
|
|
final LayoutInflater inflater = LayoutInflater.from(container.getContext());
|
|
|
|
final View view = inflater.inflate(
|
|
|
|
R.layout.emoji_keyboard_page, container, false /* attachToRoot */);
|
|
|
|
final ScrollKeyboardView keyboardView = (ScrollKeyboardView)view.findViewById(
|
|
|
|
R.id.emoji_keyboard_page);
|
|
|
|
keyboardView.setKeyboard(keyboard);
|
|
|
|
keyboardView.setOnKeyClickListener(mListener);
|
|
|
|
final ScrollViewWithNotifier scrollView = (ScrollViewWithNotifier)view.findViewById(
|
|
|
|
R.id.emoji_keyboard_scroller);
|
|
|
|
keyboardView.setScrollView(scrollView);
|
|
|
|
container.addView(view);
|
|
|
|
mActiveKeyboardView.put(position, keyboardView);
|
|
|
|
return view;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean isViewFromObject(final View view, final Object object) {
|
|
|
|
return view == object;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void destroyItem(final ViewGroup container, final int position,
|
|
|
|
final Object object) {
|
|
|
|
final ScrollKeyboardView keyboardView = mActiveKeyboardView.get(position);
|
|
|
|
if (keyboardView != null) {
|
|
|
|
keyboardView.deallocateMemory();
|
|
|
|
mActiveKeyboardView.remove(position);
|
|
|
|
}
|
|
|
|
container.removeView(keyboardView);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|