Support fine grained hover event handling
This CL also adds visual feedback of hovering key in accessibility mode. Bug: 12491371 Change-Id: I7b1f28a2e421ca60b2738ed94e7bdb406f75039b
This commit is contained in:
parent
d3642a2a58
commit
639e431fa2
1 changed files with 113 additions and 57 deletions
|
@ -17,11 +17,11 @@
|
||||||
package com.android.inputmethod.accessibility;
|
package com.android.inputmethod.accessibility;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.SystemClock;
|
|
||||||
import android.support.v4.view.AccessibilityDelegateCompat;
|
import android.support.v4.view.AccessibilityDelegateCompat;
|
||||||
import android.support.v4.view.ViewCompat;
|
import android.support.v4.view.ViewCompat;
|
||||||
import android.support.v4.view.accessibility.AccessibilityEventCompat;
|
import android.support.v4.view.accessibility.AccessibilityEventCompat;
|
||||||
import android.support.v4.view.accessibility.AccessibilityNodeInfoCompat;
|
import android.support.v4.view.accessibility.AccessibilityNodeInfoCompat;
|
||||||
|
import android.util.Log;
|
||||||
import android.view.MotionEvent;
|
import android.view.MotionEvent;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.ViewParent;
|
import android.view.ViewParent;
|
||||||
|
@ -31,6 +31,7 @@ import com.android.inputmethod.keyboard.Key;
|
||||||
import com.android.inputmethod.keyboard.KeyDetector;
|
import com.android.inputmethod.keyboard.KeyDetector;
|
||||||
import com.android.inputmethod.keyboard.Keyboard;
|
import com.android.inputmethod.keyboard.Keyboard;
|
||||||
import com.android.inputmethod.keyboard.KeyboardView;
|
import com.android.inputmethod.keyboard.KeyboardView;
|
||||||
|
import com.android.inputmethod.keyboard.PointerTracker;
|
||||||
|
|
||||||
public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
||||||
extends AccessibilityDelegateCompat {
|
extends AccessibilityDelegateCompat {
|
||||||
|
@ -66,10 +67,18 @@ public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
||||||
mKeyboard = keyboard;
|
mKeyboard = keyboard;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Keyboard getKeyboard() {
|
protected final Keyboard getKeyboard() {
|
||||||
return mKeyboard;
|
return mKeyboard;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected final void setCurrentHoverKey(final Key key) {
|
||||||
|
mCurrentHoverKey = key;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected final Key getCurrentHoverKey() {
|
||||||
|
return mCurrentHoverKey;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a window state change event with the specified string resource id.
|
* Sends a window state change event with the specified string resource id.
|
||||||
*
|
*
|
||||||
|
@ -127,6 +136,19 @@ public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
||||||
return mAccessibilityNodeProvider;
|
return mAccessibilityNodeProvider;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a key that a hover event is on.
|
||||||
|
*
|
||||||
|
* @param event The hover event.
|
||||||
|
* @return key The key that the <code>event</code> is on.
|
||||||
|
*/
|
||||||
|
protected final Key getHoverKey(final MotionEvent event) {
|
||||||
|
final int actionIndex = event.getActionIndex();
|
||||||
|
final int x = (int)event.getX(actionIndex);
|
||||||
|
final int y = (int)event.getY(actionIndex);
|
||||||
|
return mKeyDetector.detectHitKey(x, y);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Receives hover events when touch exploration is turned on in SDK versions ICS and higher.
|
* Receives hover events when touch exploration is turned on in SDK versions ICS and higher.
|
||||||
*
|
*
|
||||||
|
@ -134,29 +156,44 @@ public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
||||||
* @return {@code true} if the event is handled.
|
* @return {@code true} if the event is handled.
|
||||||
*/
|
*/
|
||||||
public boolean onHoverEvent(final MotionEvent event) {
|
public boolean onHoverEvent(final MotionEvent event) {
|
||||||
final int x = (int) event.getX();
|
switch (event.getActionMasked()) {
|
||||||
final int y = (int) event.getY();
|
|
||||||
final Key previousKey = mCurrentHoverKey;
|
|
||||||
final Key key = mKeyDetector.detectHitKey(x, y);
|
|
||||||
mCurrentHoverKey = key;
|
|
||||||
|
|
||||||
switch (event.getAction()) {
|
|
||||||
case MotionEvent.ACTION_HOVER_EXIT:
|
|
||||||
// Make sure we're not getting an EXIT event because the user slid
|
|
||||||
// off the keyboard area, then force a key press.
|
|
||||||
if (key != null) {
|
|
||||||
final long downTime = simulateKeyPress(key);
|
|
||||||
simulateKeyRelease(key, downTime);
|
|
||||||
onHoverExitKey(key);
|
|
||||||
}
|
|
||||||
mCurrentHoverKey = null;
|
|
||||||
break;
|
|
||||||
case MotionEvent.ACTION_HOVER_ENTER:
|
case MotionEvent.ACTION_HOVER_ENTER:
|
||||||
|
onHoverEnter(event);
|
||||||
|
break;
|
||||||
|
case MotionEvent.ACTION_HOVER_MOVE:
|
||||||
|
onHoverMove(event);
|
||||||
|
break;
|
||||||
|
case MotionEvent.ACTION_HOVER_EXIT:
|
||||||
|
onHoverExit(event);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
Log.w(getClass().getSimpleName(), "Unknown hover event: " + event);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process {@link MotionEvent#ACTION_HOVER_ENTER} event.
|
||||||
|
*
|
||||||
|
* @param event A hover enter event.
|
||||||
|
*/
|
||||||
|
protected void onHoverEnter(final MotionEvent event) {
|
||||||
|
final Key key = getHoverKey(event);
|
||||||
if (key != null) {
|
if (key != null) {
|
||||||
onHoverEnterKey(key);
|
onHoverEnterKey(key);
|
||||||
}
|
}
|
||||||
break;
|
setCurrentHoverKey(key);
|
||||||
case MotionEvent.ACTION_HOVER_MOVE:
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process {@link MotionEvent#ACTION_HOVER_MOVE} event.
|
||||||
|
*
|
||||||
|
* @param event A hover move event.
|
||||||
|
*/
|
||||||
|
protected void onHoverMove(final MotionEvent event) {
|
||||||
|
final Key previousKey = getCurrentHoverKey();
|
||||||
|
final Key key = getHoverKey(event);
|
||||||
if (key != previousKey) {
|
if (key != previousKey) {
|
||||||
if (previousKey != null) {
|
if (previousKey != null) {
|
||||||
onHoverExitKey(previousKey);
|
onHoverExitKey(previousKey);
|
||||||
|
@ -168,43 +205,58 @@ public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
||||||
if (key != null) {
|
if (key != null) {
|
||||||
onHoverMoveKey(key);
|
onHoverMoveKey(key);
|
||||||
}
|
}
|
||||||
break;
|
setCurrentHoverKey(key);
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simulates a key press by injecting touch an event into the keyboard view.
|
* Process {@link MotionEvent#ACTION_HOVER_EXIT} event.
|
||||||
* This avoids the complexity of trackers and listeners within the keyboard.
|
|
||||||
*
|
*
|
||||||
* @param key The key to press.
|
* @param event A hover exit event.
|
||||||
* @return the event time of the simulated key press.
|
|
||||||
*/
|
*/
|
||||||
private long simulateKeyPress(final Key key) {
|
protected void onHoverExit(final MotionEvent event) {
|
||||||
final int x = key.getHitBox().centerX();
|
final Key key = getHoverKey(event);
|
||||||
final int y = key.getHitBox().centerY();
|
// Make sure we're not getting an EXIT event because the user slid
|
||||||
final long downTime = SystemClock.uptimeMillis();
|
// off the keyboard area, then force a key press.
|
||||||
final MotionEvent downEvent = MotionEvent.obtain(
|
if (key != null) {
|
||||||
downTime, downTime, MotionEvent.ACTION_DOWN, x, y, 0);
|
simulateTouchEvent(MotionEvent.ACTION_DOWN, event);
|
||||||
mKeyboardView.onTouchEvent(downEvent);
|
simulateTouchEvent(MotionEvent.ACTION_UP, event);
|
||||||
downEvent.recycle();
|
onHoverExitKey(key);
|
||||||
return downTime;
|
}
|
||||||
|
setCurrentHoverKey(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simulates a key release by injecting touch an event into the keyboard view.
|
* Simulating a touch event by injecting a synthesized touch event into {@link PointerTracker}.
|
||||||
* This avoids the complexity of trackers and listeners within the keyboard.
|
|
||||||
*
|
*
|
||||||
* @param key The key to release.
|
* @param touchAction The action of the synthesizing touch event.
|
||||||
* @param downTime the event time of the key press.
|
* @param hoverEvent The base hover event from that the touch event is synthesized.
|
||||||
*/
|
*/
|
||||||
private void simulateKeyRelease(final Key key, final long downTime) {
|
protected void simulateTouchEvent(final int touchAction, final MotionEvent hoverEvent) {
|
||||||
final int x = key.getHitBox().centerX();
|
final MotionEvent touchEvent = synthesizeTouchEvent(touchAction, hoverEvent);
|
||||||
final int y = key.getHitBox().centerY();
|
final int actionIndex = touchEvent.getActionIndex();
|
||||||
final MotionEvent upEvent = MotionEvent.obtain(
|
final int pointerId = touchEvent.getPointerId(actionIndex);
|
||||||
downTime, SystemClock.uptimeMillis(), MotionEvent.ACTION_UP, x, y, 0);
|
final PointerTracker tracker = PointerTracker.getPointerTracker(pointerId);
|
||||||
mKeyboardView.onTouchEvent(upEvent);
|
tracker.processMotionEvent(touchEvent, mKeyDetector);
|
||||||
upEvent.recycle();
|
touchEvent.recycle();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Synthesize a touch event from a hover event.
|
||||||
|
*
|
||||||
|
* @param touchAction The action of the synthesizing touch event.
|
||||||
|
* @param event The base hover event from that the touch event is synthesized.
|
||||||
|
* @return The synthesized touch event of <code>touchAction</code> that has pointer information
|
||||||
|
* of <code>event</code>.
|
||||||
|
*/
|
||||||
|
protected static MotionEvent synthesizeTouchEvent(final int touchAction,
|
||||||
|
final MotionEvent event) {
|
||||||
|
final long downTime = event.getDownTime();
|
||||||
|
final long eventTime = event.getEventTime();
|
||||||
|
final int actionIndex = event.getActionIndex();
|
||||||
|
final float x = event.getX(actionIndex);
|
||||||
|
final float y = event.getY(actionIndex);
|
||||||
|
final int pointerId = event.getPointerId(actionIndex);
|
||||||
|
return MotionEvent.obtain(downTime, eventTime, touchAction, x, y, pointerId);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -213,6 +265,8 @@ public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
||||||
* @param key The currently hovered key.
|
* @param key The currently hovered key.
|
||||||
*/
|
*/
|
||||||
protected void onHoverEnterKey(final Key key) {
|
protected void onHoverEnterKey(final Key key) {
|
||||||
|
key.onPressed();
|
||||||
|
mKeyboardView.invalidateKey(key);
|
||||||
final KeyboardAccessibilityNodeProvider provider = getAccessibilityNodeProvider();
|
final KeyboardAccessibilityNodeProvider provider = getAccessibilityNodeProvider();
|
||||||
provider.sendAccessibilityEventForKey(key, AccessibilityEventCompat.TYPE_VIEW_HOVER_ENTER);
|
provider.sendAccessibilityEventForKey(key, AccessibilityEventCompat.TYPE_VIEW_HOVER_ENTER);
|
||||||
provider.performActionForKey(key, AccessibilityNodeInfoCompat.ACTION_ACCESSIBILITY_FOCUS);
|
provider.performActionForKey(key, AccessibilityNodeInfoCompat.ACTION_ACCESSIBILITY_FOCUS);
|
||||||
|
@ -231,6 +285,8 @@ public class KeyboardAccessibilityDelegate<KV extends KeyboardView>
|
||||||
* @param key The currently hovered key.
|
* @param key The currently hovered key.
|
||||||
*/
|
*/
|
||||||
protected void onHoverExitKey(final Key key) {
|
protected void onHoverExitKey(final Key key) {
|
||||||
|
key.onReleased();
|
||||||
|
mKeyboardView.invalidateKey(key);
|
||||||
final KeyboardAccessibilityNodeProvider provider = getAccessibilityNodeProvider();
|
final KeyboardAccessibilityNodeProvider provider = getAccessibilityNodeProvider();
|
||||||
provider.sendAccessibilityEventForKey(key, AccessibilityEventCompat.TYPE_VIEW_HOVER_EXIT);
|
provider.sendAccessibilityEventForKey(key, AccessibilityEventCompat.TYPE_VIEW_HOVER_EXIT);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue