2012-07-18 05:27:51 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 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.internal;
|
|
|
|
|
2012-09-21 03:15:02 +00:00
|
|
|
import android.util.Log;
|
|
|
|
|
2012-07-18 05:27:51 +00:00
|
|
|
import com.android.inputmethod.latin.InputPointers;
|
2012-07-20 07:24:54 +00:00
|
|
|
import com.android.inputmethod.latin.ResizableIntArray;
|
2012-07-18 05:27:51 +00:00
|
|
|
|
|
|
|
public class GestureStroke {
|
2012-09-21 03:15:02 +00:00
|
|
|
private static final String TAG = GestureStroke.class.getSimpleName();
|
|
|
|
private static final boolean DEBUG = false;
|
2012-09-27 10:01:17 +00:00
|
|
|
private static final boolean DEBUG_SPEED = false;
|
2012-09-21 03:15:02 +00:00
|
|
|
|
2012-07-18 11:31:09 +00:00
|
|
|
public static final int DEFAULT_CAPACITY = 128;
|
|
|
|
|
2012-07-18 05:27:51 +00:00
|
|
|
private final int mPointerId;
|
2012-07-20 07:24:54 +00:00
|
|
|
private final ResizableIntArray mEventTimes = new ResizableIntArray(DEFAULT_CAPACITY);
|
|
|
|
private final ResizableIntArray mXCoordinates = new ResizableIntArray(DEFAULT_CAPACITY);
|
|
|
|
private final ResizableIntArray mYCoordinates = new ResizableIntArray(DEFAULT_CAPACITY);
|
2012-09-21 03:15:02 +00:00
|
|
|
|
2012-09-27 10:01:17 +00:00
|
|
|
private int mKeyWidth; // pixel
|
|
|
|
// Static threshold for starting gesture detection
|
2012-09-21 03:15:02 +00:00
|
|
|
private int mDetectFastMoveSpeedThreshold; // pixel /sec
|
|
|
|
private int mDetectFastMoveTime;
|
|
|
|
private int mDetectFastMoveX;
|
|
|
|
private int mDetectFastMoveY;
|
2012-09-27 10:01:17 +00:00
|
|
|
// Dynamic threshold for gesture after fast typing
|
|
|
|
private boolean mAfterFastTyping;
|
|
|
|
private int mGestureDynamicDistanceThresholdFrom; // pixel
|
|
|
|
private int mGestureDynamicDistanceThresholdTo; // pixel
|
|
|
|
// Variables for gesture sampling
|
|
|
|
private int mGestureSamplingMinimumDistance; // pixel
|
|
|
|
private long mLastMajorEventTime;
|
|
|
|
private int mLastMajorEventX;
|
|
|
|
private int mLastMajorEventY;
|
|
|
|
// Variables for gesture recognition
|
|
|
|
private int mGestureRecognitionSpeedThreshold; // pixel / sec
|
|
|
|
private int mIncrementalRecognitionSize;
|
|
|
|
private int mLastIncrementalBatchSize;
|
2012-07-18 05:27:51 +00:00
|
|
|
|
2012-07-19 05:47:55 +00:00
|
|
|
// TODO: Move some of these to resource.
|
2012-07-18 05:27:51 +00:00
|
|
|
|
2012-09-27 10:01:17 +00:00
|
|
|
// Static threshold for gesture after fast typing
|
|
|
|
public static final int GESTURE_STATIC_TIME_THRESHOLD_AFTER_FAST_TYPING = 350; // msec
|
|
|
|
|
|
|
|
// Static threshold for starting gesture detection
|
|
|
|
private static final float DETECT_FAST_MOVE_SPEED_THRESHOLD = 1.5f; // keyWidth / sec
|
|
|
|
|
|
|
|
// Dynamic threshold for gesture after fast typing
|
|
|
|
private static final int GESTURE_DYNAMIC_THRESHOLD_DECAY_DURATION = 450; // msec
|
|
|
|
// Time based threshold values
|
|
|
|
private static final int GESTURE_DYNAMIC_TIME_THRESHOLD_FROM = 300; // msec
|
|
|
|
private static final int GESTURE_DYNAMIC_TIME_THRESHOLD_TO = 20; // msec
|
|
|
|
// Distance based threshold values
|
|
|
|
private static final float GESTURE_DYNAMIC_DISTANCE_THRESHOLD_FROM = 6.0f; // keyWidth
|
|
|
|
private static final float GESTURE_DYNAMIC_DISTANCE_THRESHOLD_TO = 0.35f; // keyWidth
|
|
|
|
|
|
|
|
// Parameters for gesture sampling
|
|
|
|
private static final float GESTURE_SAMPLING_MINIMUM_DISTANCE = 1.0f / 6.0f; // keyWidth
|
|
|
|
|
|
|
|
// Parameters for gesture recognition
|
|
|
|
private static final int GESTURE_RECOGNITION_MINIMUM_TIME = 100; // msec
|
|
|
|
private static final float GESTURE_RECOGNITION_SPEED_THRESHOLD = 5.5f; // keyWidth / sec
|
|
|
|
|
|
|
|
private static final int MSEC_PER_SEC = 1000;
|
2012-09-20 05:16:15 +00:00
|
|
|
|
2012-08-20 03:57:34 +00:00
|
|
|
public GestureStroke(final int pointerId) {
|
2012-07-18 05:27:51 +00:00
|
|
|
mPointerId = pointerId;
|
|
|
|
}
|
|
|
|
|
2012-09-14 09:10:39 +00:00
|
|
|
public void setKeyboardGeometry(final int keyWidth) {
|
2012-09-21 03:15:02 +00:00
|
|
|
mKeyWidth = keyWidth;
|
2012-07-19 05:47:55 +00:00
|
|
|
// TODO: Find an appropriate base metric for these length. Maybe diagonal length of the key?
|
2012-09-27 10:01:17 +00:00
|
|
|
mDetectFastMoveSpeedThreshold = (int)(keyWidth * DETECT_FAST_MOVE_SPEED_THRESHOLD);
|
|
|
|
mGestureDynamicDistanceThresholdFrom =
|
|
|
|
(int)(keyWidth * GESTURE_DYNAMIC_DISTANCE_THRESHOLD_FROM);
|
|
|
|
mGestureDynamicDistanceThresholdTo =
|
|
|
|
(int)(keyWidth * GESTURE_DYNAMIC_DISTANCE_THRESHOLD_TO);
|
|
|
|
mGestureSamplingMinimumDistance = (int)(keyWidth * GESTURE_SAMPLING_MINIMUM_DISTANCE);
|
2012-09-21 03:15:02 +00:00
|
|
|
mGestureRecognitionSpeedThreshold =
|
2012-09-27 10:01:17 +00:00
|
|
|
(int)(keyWidth * GESTURE_RECOGNITION_SPEED_THRESHOLD);
|
2012-09-21 03:15:02 +00:00
|
|
|
if (DEBUG) {
|
2012-09-27 10:01:17 +00:00
|
|
|
Log.d(TAG, String.format(
|
|
|
|
"[%d] setKeyboardGeometry: keyWidth=%3d tT=%3d >> %3d tD=%3d >> %3d",
|
|
|
|
mPointerId, keyWidth,
|
|
|
|
GESTURE_DYNAMIC_TIME_THRESHOLD_FROM,
|
|
|
|
GESTURE_DYNAMIC_TIME_THRESHOLD_TO,
|
|
|
|
mGestureDynamicDistanceThresholdFrom,
|
|
|
|
mGestureDynamicDistanceThresholdTo));
|
2012-09-21 03:15:02 +00:00
|
|
|
}
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-27 10:01:17 +00:00
|
|
|
public void onDownEvent(final int x, final int y, final long downTime,
|
|
|
|
final long gestureFirstDownTime, final long lastTypingTime) {
|
|
|
|
reset();
|
|
|
|
final long elapsedTimeAfterTyping = downTime - lastTypingTime;
|
|
|
|
if (elapsedTimeAfterTyping < GESTURE_STATIC_TIME_THRESHOLD_AFTER_FAST_TYPING) {
|
2012-09-25 05:23:23 +00:00
|
|
|
mAfterFastTyping = true;
|
|
|
|
}
|
|
|
|
if (DEBUG) {
|
2012-09-27 10:01:17 +00:00
|
|
|
Log.d(TAG, String.format("[%d] onDownEvent: dT=%3d%s", mPointerId,
|
|
|
|
elapsedTimeAfterTyping, mAfterFastTyping ? " afterFastTyping" : ""));
|
2012-09-25 05:23:23 +00:00
|
|
|
}
|
2012-09-27 10:01:17 +00:00
|
|
|
final int elapsedTimeFromFirstDown = (int)(downTime - gestureFirstDownTime);
|
|
|
|
addPoint(x, y, elapsedTimeFromFirstDown, true /* isMajorEvent */);
|
2012-09-25 05:23:23 +00:00
|
|
|
}
|
|
|
|
|
2012-09-27 10:01:17 +00:00
|
|
|
private int getGestureDynamicDistanceThreshold(final int deltaTime) {
|
|
|
|
if (!mAfterFastTyping || deltaTime >= GESTURE_DYNAMIC_THRESHOLD_DECAY_DURATION) {
|
|
|
|
return mGestureDynamicDistanceThresholdTo;
|
2012-09-25 05:23:23 +00:00
|
|
|
}
|
|
|
|
final int decayedThreshold =
|
2012-09-27 10:01:17 +00:00
|
|
|
(mGestureDynamicDistanceThresholdFrom - mGestureDynamicDistanceThresholdTo)
|
|
|
|
* deltaTime / GESTURE_DYNAMIC_THRESHOLD_DECAY_DURATION;
|
|
|
|
return mGestureDynamicDistanceThresholdFrom - decayedThreshold;
|
|
|
|
}
|
|
|
|
|
|
|
|
private int getGestureDynamicTimeThreshold(final int deltaTime) {
|
|
|
|
if (!mAfterFastTyping || deltaTime >= GESTURE_DYNAMIC_THRESHOLD_DECAY_DURATION) {
|
|
|
|
return GESTURE_DYNAMIC_TIME_THRESHOLD_TO;
|
|
|
|
}
|
|
|
|
final int decayedThreshold =
|
|
|
|
(GESTURE_DYNAMIC_TIME_THRESHOLD_FROM - GESTURE_DYNAMIC_TIME_THRESHOLD_TO)
|
|
|
|
* deltaTime / GESTURE_DYNAMIC_THRESHOLD_DECAY_DURATION;
|
|
|
|
return GESTURE_DYNAMIC_TIME_THRESHOLD_FROM - decayedThreshold;
|
2012-09-25 05:23:23 +00:00
|
|
|
}
|
|
|
|
|
2012-08-13 04:45:32 +00:00
|
|
|
public boolean isStartOfAGesture() {
|
2012-09-21 03:15:02 +00:00
|
|
|
if (mDetectFastMoveTime == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
2012-08-13 04:45:32 +00:00
|
|
|
final int size = mEventTimes.getLength();
|
2012-09-21 03:15:02 +00:00
|
|
|
if (size <= 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
final int lastIndex = size - 1;
|
|
|
|
final int deltaTime = mEventTimes.get(lastIndex) - mDetectFastMoveTime;
|
2012-09-27 10:01:17 +00:00
|
|
|
final int deltaDistance = getDistance(
|
2012-09-21 03:15:02 +00:00
|
|
|
mXCoordinates.get(lastIndex), mYCoordinates.get(lastIndex),
|
|
|
|
mDetectFastMoveX, mDetectFastMoveY);
|
2012-09-27 10:01:17 +00:00
|
|
|
final int distanceThreshold = getGestureDynamicDistanceThreshold(deltaTime);
|
|
|
|
final int timeThreshold = getGestureDynamicTimeThreshold(deltaTime);
|
|
|
|
final boolean isStartOfAGesture = deltaTime >= timeThreshold
|
|
|
|
&& deltaDistance >= distanceThreshold;
|
2012-09-21 03:15:02 +00:00
|
|
|
if (DEBUG) {
|
2012-09-27 10:01:17 +00:00
|
|
|
Log.d(TAG, String.format("[%d] isStartOfAGesture: dT=%3d tT=%3d dD=%3d tD=%3d%s%s",
|
|
|
|
mPointerId, deltaTime, timeThreshold,
|
|
|
|
deltaDistance, distanceThreshold,
|
|
|
|
mAfterFastTyping ? " afterFastTyping" : "",
|
|
|
|
isStartOfAGesture ? " startOfAGesture" : ""));
|
2012-09-21 03:15:02 +00:00
|
|
|
}
|
|
|
|
return isStartOfAGesture;
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-27 10:01:17 +00:00
|
|
|
protected void reset() {
|
2012-07-19 05:47:55 +00:00
|
|
|
mIncrementalRecognitionSize = 0;
|
2012-07-19 12:53:42 +00:00
|
|
|
mLastIncrementalBatchSize = 0;
|
2012-07-20 07:24:54 +00:00
|
|
|
mEventTimes.setLength(0);
|
|
|
|
mXCoordinates.setLength(0);
|
|
|
|
mYCoordinates.setLength(0);
|
2012-09-21 03:15:02 +00:00
|
|
|
mLastMajorEventTime = 0;
|
|
|
|
mDetectFastMoveTime = 0;
|
2012-09-25 05:23:23 +00:00
|
|
|
mAfterFastTyping = false;
|
2012-09-21 03:15:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private void appendPoint(final int x, final int y, final int time) {
|
|
|
|
mEventTimes.add(time);
|
|
|
|
mXCoordinates.add(x);
|
|
|
|
mYCoordinates.add(y);
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-21 03:15:02 +00:00
|
|
|
private void updateMajorEvent(final int x, final int y, final int time) {
|
|
|
|
mLastMajorEventTime = time;
|
|
|
|
mLastMajorEventX = x;
|
|
|
|
mLastMajorEventY = y;
|
|
|
|
}
|
|
|
|
|
|
|
|
private int detectFastMove(final int x, final int y, final int time) {
|
2012-07-20 07:24:54 +00:00
|
|
|
final int size = mEventTimes.getLength();
|
2012-09-21 03:15:02 +00:00
|
|
|
final int lastIndex = size - 1;
|
|
|
|
final int lastX = mXCoordinates.get(lastIndex);
|
|
|
|
final int lastY = mYCoordinates.get(lastIndex);
|
|
|
|
final int dist = getDistance(lastX, lastY, x, y);
|
|
|
|
final int msecs = time - mEventTimes.get(lastIndex);
|
|
|
|
if (msecs > 0) {
|
|
|
|
final int pixels = getDistance(lastX, lastY, x, y);
|
|
|
|
final int pixelsPerSec = pixels * MSEC_PER_SEC;
|
2012-09-27 10:01:17 +00:00
|
|
|
if (DEBUG_SPEED) {
|
2012-09-21 03:15:02 +00:00
|
|
|
final float speed = (float)pixelsPerSec / msecs / mKeyWidth;
|
2012-09-27 10:01:17 +00:00
|
|
|
Log.d(TAG, String.format("[%d] detectFastMove: speed=%5.2f", mPointerId, speed));
|
2012-09-21 03:15:02 +00:00
|
|
|
}
|
|
|
|
// Equivalent to (pixels / msecs < mStartSpeedThreshold / MSEC_PER_SEC)
|
|
|
|
if (mDetectFastMoveTime == 0 && pixelsPerSec > mDetectFastMoveSpeedThreshold * msecs) {
|
|
|
|
if (DEBUG) {
|
2012-09-27 10:01:17 +00:00
|
|
|
final float speed = (float)pixelsPerSec / msecs / mKeyWidth;
|
|
|
|
Log.d(TAG, String.format(
|
|
|
|
"[%d] detectFastMove: speed=%5.2f T=%3d points=%3d fastMove",
|
|
|
|
mPointerId, speed, time, size));
|
2012-09-21 03:15:02 +00:00
|
|
|
}
|
|
|
|
mDetectFastMoveTime = time;
|
|
|
|
mDetectFastMoveX = x;
|
|
|
|
mDetectFastMoveY = y;
|
|
|
|
}
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
2012-09-21 03:15:02 +00:00
|
|
|
return dist;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void addPoint(final int x, final int y, final int time, final boolean isMajorEvent) {
|
|
|
|
final int size = mEventTimes.getLength();
|
|
|
|
if (size <= 0) {
|
|
|
|
// Down event
|
|
|
|
appendPoint(x, y, time);
|
|
|
|
updateMajorEvent(x, y, time);
|
|
|
|
} else {
|
2012-09-27 10:01:17 +00:00
|
|
|
final int distance = detectFastMove(x, y, time);
|
|
|
|
if (distance > mGestureSamplingMinimumDistance) {
|
2012-09-21 03:15:02 +00:00
|
|
|
appendPoint(x, y, time);
|
|
|
|
}
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
2012-09-21 03:15:02 +00:00
|
|
|
if (isMajorEvent) {
|
2012-09-20 07:39:40 +00:00
|
|
|
updateIncrementalRecognitionSize(x, y, time);
|
2012-09-21 03:15:02 +00:00
|
|
|
updateMajorEvent(x, y, time);
|
2012-09-20 07:39:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void updateIncrementalRecognitionSize(final int x, final int y, final int time) {
|
2012-09-21 03:15:02 +00:00
|
|
|
final int msecs = (int)(time - mLastMajorEventTime);
|
|
|
|
if (msecs <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
final int pixels = getDistance(mLastMajorEventX, mLastMajorEventY, x, y);
|
|
|
|
final int pixelsPerSec = pixels * MSEC_PER_SEC;
|
|
|
|
// Equivalent to (pixels / msecs < mGestureRecognitionThreshold / MSEC_PER_SEC)
|
|
|
|
if (pixelsPerSec < mGestureRecognitionSpeedThreshold * msecs) {
|
|
|
|
mIncrementalRecognitionSize = mEventTimes.getLength();
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
2012-09-27 10:01:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static final boolean hasRecognitionTimePast(
|
|
|
|
final long currentTime, final long lastRecognitionTime) {
|
|
|
|
return currentTime > lastRecognitionTime + GESTURE_RECOGNITION_MINIMUM_TIME;
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void appendAllBatchPoints(final InputPointers out) {
|
2012-07-20 07:24:54 +00:00
|
|
|
appendBatchPoints(out, mEventTimes.getLength());
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void appendIncrementalBatchPoints(final InputPointers out) {
|
2012-07-20 07:24:54 +00:00
|
|
|
appendBatchPoints(out, mIncrementalRecognitionSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void appendBatchPoints(final InputPointers out, final int size) {
|
2012-08-23 06:01:46 +00:00
|
|
|
final int length = size - mLastIncrementalBatchSize;
|
|
|
|
if (length <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
2012-07-20 07:24:54 +00:00
|
|
|
out.append(mPointerId, mEventTimes, mXCoordinates, mYCoordinates,
|
2012-08-23 06:01:46 +00:00
|
|
|
mLastIncrementalBatchSize, length);
|
2012-07-20 07:24:54 +00:00
|
|
|
mLastIncrementalBatchSize = size;
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
|
|
|
|
2012-09-21 03:15:02 +00:00
|
|
|
private static int getDistance(final int x1, final int y1, final int x2, final int y2) {
|
|
|
|
final int dx = x1 - x2;
|
|
|
|
final int dy = y1 - y2;
|
2012-08-12 02:10:48 +00:00
|
|
|
// Note that, in recent versions of Android, FloatMath is actually slower than
|
|
|
|
// java.lang.Math due to the way the JIT optimizes java.lang.Math.
|
2012-09-21 03:15:02 +00:00
|
|
|
return (int)Math.sqrt(dx * dx + dy * dy);
|
2012-07-18 05:27:51 +00:00
|
|
|
}
|
|
|
|
}
|