2011-04-27 05:14:45 +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.keyboard;
|
|
|
|
|
|
|
|
import android.content.Context;
|
|
|
|
import android.util.AttributeSet;
|
2012-12-03 02:47:52 +00:00
|
|
|
import android.view.MotionEvent;
|
2011-04-27 05:14:45 +00:00
|
|
|
import android.view.View;
|
2013-12-16 07:13:12 +00:00
|
|
|
import android.view.ViewGroup;
|
2011-04-27 05:14:45 +00:00
|
|
|
|
2014-05-26 02:09:07 +00:00
|
|
|
import com.android.inputmethod.accessibility.AccessibilityUtils;
|
|
|
|
import com.android.inputmethod.accessibility.MoreKeysKeyboardAccessibilityDelegate;
|
2012-08-21 05:05:57 +00:00
|
|
|
import com.android.inputmethod.latin.Constants;
|
2011-05-25 06:22:03 +00:00
|
|
|
import com.android.inputmethod.latin.R;
|
2013-06-23 16:11:32 +00:00
|
|
|
import com.android.inputmethod.latin.utils.CoordinateUtils;
|
2011-05-25 06:22:03 +00:00
|
|
|
|
2011-04-27 05:14:45 +00:00
|
|
|
/**
|
2012-02-08 07:12:11 +00:00
|
|
|
* A view that renders a virtual {@link MoreKeysKeyboard}. It handles rendering of keys and
|
|
|
|
* detecting key presses and touch movements.
|
2011-04-27 05:14:45 +00:00
|
|
|
*/
|
2012-12-03 02:47:52 +00:00
|
|
|
public class MoreKeysKeyboardView extends KeyboardView implements MoreKeysPanel {
|
2012-11-28 08:25:23 +00:00
|
|
|
private final int[] mCoordinates = CoordinateUtils.newInstance();
|
2011-04-27 05:14:45 +00:00
|
|
|
|
2014-06-13 02:15:38 +00:00
|
|
|
protected final KeyDetector mKeyDetector;
|
2013-07-25 08:03:16 +00:00
|
|
|
private Controller mController = EMPTY_CONTROLLER;
|
2012-12-03 02:47:52 +00:00
|
|
|
protected KeyboardActionListener mListener;
|
2011-04-27 05:14:45 +00:00
|
|
|
private int mOriginX;
|
|
|
|
private int mOriginY;
|
2012-12-03 02:47:52 +00:00
|
|
|
private Key mCurrentKey;
|
2011-07-11 01:09:15 +00:00
|
|
|
|
2012-12-03 02:47:52 +00:00
|
|
|
private int mActivePointerId;
|
2011-04-27 05:14:45 +00:00
|
|
|
|
2014-05-26 02:09:07 +00:00
|
|
|
protected MoreKeysKeyboardAccessibilityDelegate mAccessibilityDelegate;
|
2014-05-26 02:09:07 +00:00
|
|
|
|
2012-10-03 06:19:43 +00:00
|
|
|
public MoreKeysKeyboardView(final Context context, final AttributeSet attrs) {
|
2012-02-08 07:12:11 +00:00
|
|
|
this(context, attrs, R.attr.moreKeysKeyboardViewStyle);
|
2011-04-27 05:14:45 +00:00
|
|
|
}
|
|
|
|
|
2012-10-03 06:19:43 +00:00
|
|
|
public MoreKeysKeyboardView(final Context context, final AttributeSet attrs,
|
|
|
|
final int defStyle) {
|
2011-04-27 05:14:45 +00:00
|
|
|
super(context, attrs, defStyle);
|
2014-05-26 02:09:07 +00:00
|
|
|
mKeyDetector = new MoreKeysDetector(getResources().getDimension(
|
|
|
|
R.dimen.config_more_keys_keyboard_slide_allowance));
|
2011-04-27 05:14:45 +00:00
|
|
|
}
|
|
|
|
|
2011-07-21 06:57:00 +00:00
|
|
|
@Override
|
2012-10-03 06:19:43 +00:00
|
|
|
protected void onMeasure(final int widthMeasureSpec, final int heightMeasureSpec) {
|
2011-07-21 06:57:00 +00:00
|
|
|
final Keyboard keyboard = getKeyboard();
|
|
|
|
if (keyboard != null) {
|
2011-07-29 00:05:40 +00:00
|
|
|
final int width = keyboard.mOccupiedWidth + getPaddingLeft() + getPaddingRight();
|
|
|
|
final int height = keyboard.mOccupiedHeight + getPaddingTop() + getPaddingBottom();
|
2011-07-21 06:57:00 +00:00
|
|
|
setMeasuredDimension(width, height);
|
|
|
|
} else {
|
|
|
|
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-11 01:09:15 +00:00
|
|
|
@Override
|
2012-10-03 06:19:43 +00:00
|
|
|
public void setKeyboard(final Keyboard keyboard) {
|
2011-07-11 01:09:15 +00:00
|
|
|
super.setKeyboard(keyboard);
|
2014-06-13 02:15:38 +00:00
|
|
|
mKeyDetector.setKeyboard(
|
|
|
|
keyboard, -getPaddingLeft(), -getPaddingTop() + getVerticalCorrection());
|
2014-05-26 02:09:07 +00:00
|
|
|
if (AccessibilityUtils.getInstance().isTouchExplorationEnabled()) {
|
2014-06-07 14:28:13 +00:00
|
|
|
if (mAccessibilityDelegate == null) {
|
|
|
|
mAccessibilityDelegate = new MoreKeysKeyboardAccessibilityDelegate(
|
|
|
|
this, mKeyDetector);
|
|
|
|
mAccessibilityDelegate.setOpenAnnounce(R.string.spoken_open_more_keys_keyboard);
|
|
|
|
mAccessibilityDelegate.setCloseAnnounce(R.string.spoken_close_more_keys_keyboard);
|
|
|
|
}
|
2014-05-26 02:09:07 +00:00
|
|
|
mAccessibilityDelegate.setKeyboard(keyboard);
|
|
|
|
} else {
|
|
|
|
mAccessibilityDelegate = null;
|
|
|
|
}
|
2011-07-11 01:09:15 +00:00
|
|
|
}
|
|
|
|
|
2011-08-29 08:02:52 +00:00
|
|
|
@Override
|
2012-10-03 06:19:43 +00:00
|
|
|
public void showMoreKeysPanel(final View parentView, final Controller controller,
|
2012-12-03 06:49:10 +00:00
|
|
|
final int pointX, final int pointY, final KeyboardActionListener listener) {
|
2011-08-29 08:02:52 +00:00
|
|
|
mController = controller;
|
|
|
|
mListener = listener;
|
2012-12-03 06:49:10 +00:00
|
|
|
final View container = getContainerView();
|
2012-03-07 10:09:36 +00:00
|
|
|
// The coordinates of panel's left-top corner in parentView's coordinate system.
|
2013-12-13 08:09:16 +00:00
|
|
|
// We need to consider background drawable paddings.
|
|
|
|
final int x = pointX - getDefaultCoordX() - container.getPaddingLeft() - getPaddingLeft();
|
|
|
|
final int y = pointY - container.getMeasuredHeight() + container.getPaddingBottom()
|
|
|
|
+ getPaddingBottom();
|
2011-04-27 05:14:45 +00:00
|
|
|
|
2012-03-07 10:09:36 +00:00
|
|
|
parentView.getLocationInWindow(mCoordinates);
|
2013-12-13 08:09:16 +00:00
|
|
|
// Ensure the horizontal position of the panel does not extend past the parentView edges.
|
2012-12-03 06:49:10 +00:00
|
|
|
final int maxX = parentView.getMeasuredWidth() - container.getMeasuredWidth();
|
2012-12-14 04:57:50 +00:00
|
|
|
final int panelX = Math.max(0, Math.min(maxX, x)) + CoordinateUtils.x(mCoordinates);
|
2012-12-03 06:49:10 +00:00
|
|
|
final int panelY = y + CoordinateUtils.y(mCoordinates);
|
|
|
|
container.setX(panelX);
|
|
|
|
container.setY(panelY);
|
2011-04-27 05:14:45 +00:00
|
|
|
|
2012-03-07 10:09:36 +00:00
|
|
|
mOriginX = x + container.getPaddingLeft();
|
|
|
|
mOriginY = y + container.getPaddingTop();
|
2012-12-03 06:49:10 +00:00
|
|
|
controller.onShowMoreKeysPanel(this);
|
2014-05-26 02:09:07 +00:00
|
|
|
final MoreKeysKeyboardAccessibilityDelegate accessibilityDelegate = mAccessibilityDelegate;
|
|
|
|
if (accessibilityDelegate != null) {
|
|
|
|
accessibilityDelegate.onShowMoreKeysKeyboard();
|
|
|
|
}
|
2011-08-26 06:45:05 +00:00
|
|
|
}
|
|
|
|
|
2012-12-03 02:47:52 +00:00
|
|
|
/**
|
|
|
|
* Returns the default x coordinate for showing this panel.
|
|
|
|
*/
|
|
|
|
protected int getDefaultCoordX() {
|
|
|
|
return ((MoreKeysKeyboard)getKeyboard()).getDefaultCoordX();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onDownEvent(final int x, final int y, final int pointerId, final long eventTime) {
|
|
|
|
mActivePointerId = pointerId;
|
2014-06-25 05:21:57 +00:00
|
|
|
mCurrentKey = detectKey(x, y);
|
2012-12-03 02:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2014-06-02 02:38:39 +00:00
|
|
|
public void onMoveEvent(final int x, final int y, final int pointerId, final long eventTime) {
|
2012-12-10 20:38:29 +00:00
|
|
|
if (mActivePointerId != pointerId) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
final boolean hasOldKey = (mCurrentKey != null);
|
2014-06-25 05:21:57 +00:00
|
|
|
mCurrentKey = detectKey(x, y);
|
2012-12-10 20:38:29 +00:00
|
|
|
if (hasOldKey && mCurrentKey == null) {
|
2014-06-02 02:38:39 +00:00
|
|
|
// A more keys keyboard is canceled when detecting no key.
|
2014-03-03 12:02:35 +00:00
|
|
|
mController.onCancelMoreKeysPanel();
|
2012-12-10 20:38:29 +00:00
|
|
|
}
|
2012-12-03 02:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onUpEvent(final int x, final int y, final int pointerId, final long eventTime) {
|
2014-06-02 02:38:39 +00:00
|
|
|
if (mActivePointerId != pointerId) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Calling {@link #detectKey(int,int,int)} here is harmless because the last move event and
|
|
|
|
// the following up event share the same coordinates.
|
2014-06-25 05:21:57 +00:00
|
|
|
mCurrentKey = detectKey(x, y);
|
2014-06-02 02:38:39 +00:00
|
|
|
if (mCurrentKey != null) {
|
2012-12-03 02:47:52 +00:00
|
|
|
updateReleaseKeyGraphics(mCurrentKey);
|
2014-05-20 08:31:24 +00:00
|
|
|
onKeyInput(mCurrentKey, x, y);
|
2012-12-03 02:47:52 +00:00
|
|
|
mCurrentKey = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Performs the specific action for this panel when the user presses a key on the panel.
|
|
|
|
*/
|
2014-05-20 08:31:24 +00:00
|
|
|
protected void onKeyInput(final Key key, final int x, final int y) {
|
|
|
|
final int code = key.getCode();
|
2012-12-03 02:47:52 +00:00
|
|
|
if (code == Constants.CODE_OUTPUT_TEXT) {
|
|
|
|
mListener.onTextInput(mCurrentKey.getOutputText());
|
|
|
|
} else if (code != Constants.CODE_UNSPECIFIED) {
|
2013-12-27 07:43:07 +00:00
|
|
|
if (getKeyboard().hasProximityCharsCorrection(code)) {
|
2014-03-31 10:43:12 +00:00
|
|
|
mListener.onCodeInput(code, x, y, false /* isKeyRepeat */);
|
2013-12-27 07:43:07 +00:00
|
|
|
} else {
|
2014-03-31 10:43:12 +00:00
|
|
|
mListener.onCodeInput(code, Constants.NOT_A_COORDINATE, Constants.NOT_A_COORDINATE,
|
|
|
|
false /* isKeyRepeat */);
|
2013-12-27 07:43:07 +00:00
|
|
|
}
|
2012-12-03 02:47:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-25 05:21:57 +00:00
|
|
|
private Key detectKey(int x, int y) {
|
2012-12-03 02:47:52 +00:00
|
|
|
final Key oldKey = mCurrentKey;
|
|
|
|
final Key newKey = mKeyDetector.detectHitKey(x, y);
|
2014-06-02 02:38:39 +00:00
|
|
|
if (newKey == oldKey) {
|
|
|
|
return newKey;
|
|
|
|
}
|
|
|
|
// A new key is detected.
|
|
|
|
if (oldKey != null) {
|
|
|
|
updateReleaseKeyGraphics(oldKey);
|
|
|
|
invalidateKey(oldKey);
|
|
|
|
}
|
|
|
|
if (newKey != null) {
|
|
|
|
updatePressKeyGraphics(newKey);
|
|
|
|
invalidateKey(newKey);
|
2012-12-03 02:47:52 +00:00
|
|
|
}
|
2014-06-02 02:38:39 +00:00
|
|
|
return newKey;
|
2012-12-03 02:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private void updateReleaseKeyGraphics(final Key key) {
|
|
|
|
key.onReleased();
|
|
|
|
invalidateKey(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void updatePressKeyGraphics(final Key key) {
|
|
|
|
key.onPressed();
|
|
|
|
invalidateKey(key);
|
|
|
|
}
|
|
|
|
|
2011-07-23 08:16:56 +00:00
|
|
|
@Override
|
2013-07-25 08:03:16 +00:00
|
|
|
public void dismissMoreKeysPanel() {
|
|
|
|
if (!isShowingInParent()) {
|
|
|
|
return;
|
|
|
|
}
|
2014-06-13 10:57:48 +00:00
|
|
|
final MoreKeysKeyboardAccessibilityDelegate accessibilityDelegate = mAccessibilityDelegate;
|
|
|
|
if (accessibilityDelegate != null) {
|
|
|
|
accessibilityDelegate.onDismissMoreKeysKeyboard();
|
|
|
|
}
|
2014-03-03 12:02:35 +00:00
|
|
|
mController.onDismissMoreKeysPanel();
|
2011-07-23 08:16:56 +00:00
|
|
|
}
|
|
|
|
|
2011-04-27 05:14:45 +00:00
|
|
|
@Override
|
2012-10-03 06:19:43 +00:00
|
|
|
public int translateX(final int x) {
|
2011-07-11 01:09:15 +00:00
|
|
|
return x - mOriginX;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2012-10-03 06:19:43 +00:00
|
|
|
public int translateY(final int y) {
|
2011-07-11 01:09:15 +00:00
|
|
|
return y - mOriginY;
|
2011-04-27 05:14:45 +00:00
|
|
|
}
|
2012-12-03 06:49:10 +00:00
|
|
|
|
2012-12-03 02:47:52 +00:00
|
|
|
@Override
|
|
|
|
public boolean onTouchEvent(final MotionEvent me) {
|
|
|
|
final int action = me.getActionMasked();
|
|
|
|
final long eventTime = me.getEventTime();
|
|
|
|
final int index = me.getActionIndex();
|
|
|
|
final int x = (int)me.getX(index);
|
|
|
|
final int y = (int)me.getY(index);
|
|
|
|
final int pointerId = me.getPointerId(index);
|
|
|
|
switch (action) {
|
|
|
|
case MotionEvent.ACTION_DOWN:
|
|
|
|
case MotionEvent.ACTION_POINTER_DOWN:
|
|
|
|
onDownEvent(x, y, pointerId, eventTime);
|
|
|
|
break;
|
|
|
|
case MotionEvent.ACTION_UP:
|
|
|
|
case MotionEvent.ACTION_POINTER_UP:
|
|
|
|
onUpEvent(x, y, pointerId, eventTime);
|
|
|
|
break;
|
|
|
|
case MotionEvent.ACTION_MOVE:
|
|
|
|
onMoveEvent(x, y, pointerId, eventTime);
|
|
|
|
break;
|
|
|
|
}
|
2013-08-01 07:55:55 +00:00
|
|
|
return true;
|
2012-12-03 02:47:52 +00:00
|
|
|
}
|
|
|
|
|
2014-05-26 02:09:07 +00:00
|
|
|
/**
|
|
|
|
* {@inheritDoc}
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public boolean onHoverEvent(final MotionEvent event) {
|
|
|
|
final MoreKeysKeyboardAccessibilityDelegate accessibilityDelegate = mAccessibilityDelegate;
|
|
|
|
if (accessibilityDelegate != null) {
|
|
|
|
return accessibilityDelegate.onHoverEvent(event);
|
|
|
|
}
|
|
|
|
return super.onHoverEvent(event);
|
|
|
|
}
|
|
|
|
|
2013-12-16 07:13:12 +00:00
|
|
|
private View getContainerView() {
|
2012-12-03 06:49:10 +00:00
|
|
|
return (View)getParent();
|
|
|
|
}
|
|
|
|
|
2013-12-16 07:13:12 +00:00
|
|
|
@Override
|
|
|
|
public void showInParent(final ViewGroup parentView) {
|
|
|
|
removeFromParent();
|
|
|
|
parentView.addView(getContainerView());
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void removeFromParent() {
|
|
|
|
final View containerView = getContainerView();
|
|
|
|
final ViewGroup currentParent = (ViewGroup)containerView.getParent();
|
|
|
|
if (currentParent != null) {
|
|
|
|
currentParent.removeView(containerView);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-03 06:49:10 +00:00
|
|
|
@Override
|
|
|
|
public boolean isShowingInParent() {
|
|
|
|
return (getContainerView().getParent() != null);
|
|
|
|
}
|
2011-04-27 05:14:45 +00:00
|
|
|
}
|