2012-08-20 03:57:34 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 The Android Open Source Project
|
|
|
|
*
|
2013-01-21 12:52:57 +00:00
|
|
|
* 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
|
2012-08-20 03:57:34 +00:00
|
|
|
*
|
2013-01-21 12:52:57 +00:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2012-08-20 03:57:34 +00:00
|
|
|
*
|
2013-01-21 12:52:57 +00:00
|
|
|
* 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.
|
2012-08-20 03:57:34 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
package com.android.inputmethod.keyboard.internal;
|
|
|
|
|
2013-05-07 08:40:47 +00:00
|
|
|
import android.content.res.TypedArray;
|
|
|
|
|
|
|
|
import com.android.inputmethod.latin.R;
|
2012-08-20 03:57:34 +00:00
|
|
|
import com.android.inputmethod.latin.ResizableIntArray;
|
|
|
|
|
2012-09-27 09:16:16 +00:00
|
|
|
public final class GestureStrokeWithPreviewPoints extends GestureStroke {
|
2012-08-20 03:57:34 +00:00
|
|
|
public static final int PREVIEW_CAPACITY = 256;
|
|
|
|
|
|
|
|
private final ResizableIntArray mPreviewEventTimes = new ResizableIntArray(PREVIEW_CAPACITY);
|
|
|
|
private final ResizableIntArray mPreviewXCoordinates = new ResizableIntArray(PREVIEW_CAPACITY);
|
|
|
|
private final ResizableIntArray mPreviewYCoordinates = new ResizableIntArray(PREVIEW_CAPACITY);
|
|
|
|
|
2013-05-07 08:40:47 +00:00
|
|
|
private final GestureStrokePreviewParams mPreviewParams;
|
|
|
|
|
2012-08-20 03:57:34 +00:00
|
|
|
private int mStrokeId;
|
|
|
|
private int mLastPreviewSize;
|
2013-03-07 10:18:13 +00:00
|
|
|
private final HermiteInterpolator mInterpolator = new HermiteInterpolator();
|
|
|
|
private int mLastInterpolatedPreviewIndex;
|
2012-08-20 03:57:34 +00:00
|
|
|
|
2012-09-14 09:10:39 +00:00
|
|
|
private int mLastX;
|
|
|
|
private int mLastY;
|
2013-03-07 10:18:13 +00:00
|
|
|
private double mDistanceFromLastSample;
|
2013-05-07 08:40:47 +00:00
|
|
|
|
|
|
|
public static final class GestureStrokePreviewParams {
|
|
|
|
public final double mMinSamplingDistance; // in pixel
|
|
|
|
public final double mMaxInterpolationAngularThreshold; // in radian
|
|
|
|
public final double mMaxInterpolationDistanceThreshold; // in pixel
|
|
|
|
public final int mMaxInterpolationSegments;
|
|
|
|
|
|
|
|
public static final GestureStrokePreviewParams DEFAULT = new GestureStrokePreviewParams();
|
|
|
|
|
|
|
|
private static final int DEFAULT_MAX_INTERPOLATION_ANGULAR_THRESHOLD = 15; // in degree
|
|
|
|
|
|
|
|
private GestureStrokePreviewParams() {
|
|
|
|
mMinSamplingDistance = 0.0d;
|
|
|
|
mMaxInterpolationAngularThreshold =
|
|
|
|
degreeToRadian(DEFAULT_MAX_INTERPOLATION_ANGULAR_THRESHOLD);
|
|
|
|
mMaxInterpolationDistanceThreshold = mMinSamplingDistance;
|
|
|
|
mMaxInterpolationSegments = 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static double degreeToRadian(final int degree) {
|
2013-05-20 22:24:58 +00:00
|
|
|
return degree / 180.0d * Math.PI;
|
2013-05-07 08:40:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public GestureStrokePreviewParams(final TypedArray mainKeyboardViewAttr) {
|
|
|
|
mMinSamplingDistance = mainKeyboardViewAttr.getDimension(
|
2013-05-13 04:12:12 +00:00
|
|
|
R.styleable.MainKeyboardView_gestureTrailMinSamplingDistance,
|
2013-05-07 08:40:47 +00:00
|
|
|
(float)DEFAULT.mMinSamplingDistance);
|
|
|
|
final int interpolationAngularDegree = mainKeyboardViewAttr.getInteger(R.styleable
|
2013-05-13 04:12:12 +00:00
|
|
|
.MainKeyboardView_gestureTrailMaxInterpolationAngularThreshold, 0);
|
2013-05-07 08:40:47 +00:00
|
|
|
mMaxInterpolationAngularThreshold = (interpolationAngularDegree <= 0)
|
|
|
|
? DEFAULT.mMaxInterpolationAngularThreshold
|
|
|
|
: degreeToRadian(interpolationAngularDegree);
|
|
|
|
mMaxInterpolationDistanceThreshold = mainKeyboardViewAttr.getDimension(R.styleable
|
2013-05-13 04:12:12 +00:00
|
|
|
.MainKeyboardView_gestureTrailMaxInterpolationDistanceThreshold,
|
2013-05-07 08:40:47 +00:00
|
|
|
(float)DEFAULT.mMaxInterpolationDistanceThreshold);
|
|
|
|
mMaxInterpolationSegments = mainKeyboardViewAttr.getInteger(
|
2013-05-13 04:12:12 +00:00
|
|
|
R.styleable.MainKeyboardView_gestureTrailMaxInterpolationSegments,
|
2013-05-07 08:40:47 +00:00
|
|
|
DEFAULT.mMaxInterpolationSegments);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public GestureStrokeWithPreviewPoints(final int pointerId,
|
|
|
|
final GestureStrokeParams strokeParams,
|
|
|
|
final GestureStrokePreviewParams previewParams) {
|
|
|
|
super(pointerId, strokeParams);
|
|
|
|
mPreviewParams = previewParams;
|
2012-08-20 03:57:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2012-09-27 10:01:17 +00:00
|
|
|
protected void reset() {
|
2012-08-20 03:57:34 +00:00
|
|
|
super.reset();
|
|
|
|
mStrokeId++;
|
|
|
|
mLastPreviewSize = 0;
|
2013-03-07 10:18:13 +00:00
|
|
|
mLastInterpolatedPreviewIndex = 0;
|
2012-08-20 03:57:34 +00:00
|
|
|
mPreviewEventTimes.setLength(0);
|
|
|
|
mPreviewXCoordinates.setLength(0);
|
|
|
|
mPreviewYCoordinates.setLength(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getGestureStrokeId() {
|
|
|
|
return mStrokeId;
|
|
|
|
}
|
|
|
|
|
2013-05-07 06:51:23 +00:00
|
|
|
private boolean needsSampling(final int x, final int y) {
|
2013-05-07 05:27:34 +00:00
|
|
|
mDistanceFromLastSample += Math.hypot(x - mLastX, y - mLastY);
|
|
|
|
mLastX = x;
|
|
|
|
mLastY = y;
|
2013-05-07 06:51:23 +00:00
|
|
|
final boolean isDownEvent = (mPreviewEventTimes.getLength() == 0);
|
2013-05-07 08:40:47 +00:00
|
|
|
if (mDistanceFromLastSample >= mPreviewParams.mMinSamplingDistance || isDownEvent) {
|
2013-05-07 05:27:34 +00:00
|
|
|
mDistanceFromLastSample = 0.0d;
|
2013-03-07 10:18:13 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2012-09-14 09:10:39 +00:00
|
|
|
}
|
|
|
|
|
2012-08-20 03:57:34 +00:00
|
|
|
@Override
|
2012-11-22 06:39:28 +00:00
|
|
|
public boolean addPointOnKeyboard(final int x, final int y, final int time,
|
|
|
|
final boolean isMajorEvent) {
|
2013-05-07 06:51:23 +00:00
|
|
|
if (needsSampling(x, y)) {
|
2012-09-14 09:10:39 +00:00
|
|
|
mPreviewEventTimes.add(time);
|
|
|
|
mPreviewXCoordinates.add(x);
|
|
|
|
mPreviewYCoordinates.add(y);
|
|
|
|
}
|
2013-03-07 10:18:13 +00:00
|
|
|
return super.addPointOnKeyboard(x, y, time, isMajorEvent);
|
|
|
|
|
2012-08-20 03:57:34 +00:00
|
|
|
}
|
|
|
|
|
2013-05-20 22:24:58 +00:00
|
|
|
/**
|
|
|
|
* Append sampled preview points.
|
|
|
|
*
|
|
|
|
* @param eventTimes the event time array of gesture trail to be drawn.
|
|
|
|
* @param xCoords the x-coordinates array of gesture trail to be drawn.
|
|
|
|
* @param yCoords the y-coordinates array of gesture trail to be drawn.
|
|
|
|
* @param types the point types array of gesture trail. This is valid only when
|
|
|
|
* {@link GestureTrail#DEBUG_SHOW_POINTS} is true.
|
|
|
|
*/
|
2012-08-20 03:57:34 +00:00
|
|
|
public void appendPreviewStroke(final ResizableIntArray eventTimes,
|
2013-05-20 22:24:58 +00:00
|
|
|
final ResizableIntArray xCoords, final ResizableIntArray yCoords,
|
|
|
|
final ResizableIntArray types) {
|
2012-08-20 03:57:34 +00:00
|
|
|
final int length = mPreviewEventTimes.getLength() - mLastPreviewSize;
|
|
|
|
if (length <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
eventTimes.append(mPreviewEventTimes, mLastPreviewSize, length);
|
|
|
|
xCoords.append(mPreviewXCoordinates, mLastPreviewSize, length);
|
|
|
|
yCoords.append(mPreviewYCoordinates, mLastPreviewSize, length);
|
2013-05-20 22:24:58 +00:00
|
|
|
if (GestureTrail.DEBUG_SHOW_POINTS) {
|
|
|
|
types.fill(GestureTrail.POINT_TYPE_SAMPLED, types.getLength(), length);
|
|
|
|
}
|
2012-08-20 03:57:34 +00:00
|
|
|
mLastPreviewSize = mPreviewEventTimes.getLength();
|
|
|
|
}
|
2013-03-07 10:18:13 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate interpolated points between the last interpolated point and the end of the trail.
|
|
|
|
* And return the start index of the last interpolated segment of input arrays because it
|
|
|
|
* may need to recalculate the interpolated points in the segment if further segments are
|
|
|
|
* added to this stroke.
|
|
|
|
*
|
|
|
|
* @param lastInterpolatedIndex the start index of the last interpolated segment of
|
|
|
|
* <code>eventTimes</code>, <code>xCoords</code>, and <code>yCoords</code>.
|
2013-05-13 04:12:12 +00:00
|
|
|
* @param eventTimes the event time array of gesture trail to be drawn.
|
|
|
|
* @param xCoords the x-coordinates array of gesture trail to be drawn.
|
|
|
|
* @param yCoords the y-coordinates array of gesture trail to be drawn.
|
2013-05-20 22:24:58 +00:00
|
|
|
* @param types the point types array of gesture trail. This is valid only when
|
|
|
|
* {@link GestureTrail#DEBUG_SHOW_POINTS} is true.
|
2013-03-07 10:18:13 +00:00
|
|
|
* @return the start index of the last interpolated segment of input arrays.
|
|
|
|
*/
|
|
|
|
public int interpolateStrokeAndReturnStartIndexOfLastSegment(final int lastInterpolatedIndex,
|
|
|
|
final ResizableIntArray eventTimes, final ResizableIntArray xCoords,
|
2013-05-09 10:16:11 +00:00
|
|
|
final ResizableIntArray yCoords, final ResizableIntArray types) {
|
2013-03-07 10:18:13 +00:00
|
|
|
final int size = mPreviewEventTimes.getLength();
|
|
|
|
final int[] pt = mPreviewEventTimes.getPrimitiveArray();
|
|
|
|
final int[] px = mPreviewXCoordinates.getPrimitiveArray();
|
|
|
|
final int[] py = mPreviewYCoordinates.getPrimitiveArray();
|
|
|
|
mInterpolator.reset(px, py, 0, size);
|
|
|
|
// The last segment of gesture stroke needs to be interpolated again because the slope of
|
|
|
|
// the tangent at the last point isn't determined.
|
|
|
|
int lastInterpolatedDrawIndex = lastInterpolatedIndex;
|
|
|
|
int d1 = lastInterpolatedIndex;
|
|
|
|
for (int p2 = mLastInterpolatedPreviewIndex + 1; p2 < size; p2++) {
|
|
|
|
final int p1 = p2 - 1;
|
|
|
|
final int p0 = p1 - 1;
|
|
|
|
final int p3 = p2 + 1;
|
|
|
|
mLastInterpolatedPreviewIndex = p1;
|
|
|
|
lastInterpolatedDrawIndex = d1;
|
|
|
|
mInterpolator.setInterval(p0, p1, p2, p3);
|
|
|
|
final double m1 = Math.atan2(mInterpolator.mSlope1Y, mInterpolator.mSlope1X);
|
|
|
|
final double m2 = Math.atan2(mInterpolator.mSlope2Y, mInterpolator.mSlope2X);
|
2013-05-07 09:41:16 +00:00
|
|
|
final double deltaAngle = Math.abs(angularDiff(m2, m1));
|
2013-05-07 08:40:47 +00:00
|
|
|
final int segmentsByAngle = (int)Math.ceil(
|
|
|
|
deltaAngle / mPreviewParams.mMaxInterpolationAngularThreshold);
|
2013-05-07 09:41:16 +00:00
|
|
|
final double deltaDistance = Math.hypot(mInterpolator.mP1X - mInterpolator.mP2X,
|
|
|
|
mInterpolator.mP1Y - mInterpolator.mP2Y);
|
2013-05-07 08:40:47 +00:00
|
|
|
final int segmentsByDistance = (int)Math.ceil(deltaDistance
|
|
|
|
/ mPreviewParams.mMaxInterpolationDistanceThreshold);
|
|
|
|
final int segments = Math.min(mPreviewParams.mMaxInterpolationSegments,
|
|
|
|
Math.max(segmentsByAngle, segmentsByDistance));
|
2013-03-07 10:18:13 +00:00
|
|
|
final int t1 = eventTimes.get(d1);
|
|
|
|
final int dt = pt[p2] - pt[p1];
|
|
|
|
d1++;
|
2013-05-07 08:40:47 +00:00
|
|
|
for (int i = 1; i < segments; i++) {
|
|
|
|
final float t = i / (float)segments;
|
2013-03-07 10:18:13 +00:00
|
|
|
mInterpolator.interpolate(t);
|
|
|
|
eventTimes.add(d1, (int)(dt * t) + t1);
|
|
|
|
xCoords.add(d1, (int)mInterpolator.mInterpolatedX);
|
|
|
|
yCoords.add(d1, (int)mInterpolator.mInterpolatedY);
|
2013-05-20 22:24:58 +00:00
|
|
|
if (GestureTrail.DEBUG_SHOW_POINTS) {
|
2013-05-13 04:12:12 +00:00
|
|
|
types.add(d1, GestureTrail.POINT_TYPE_INTERPOLATED);
|
2013-05-09 10:16:11 +00:00
|
|
|
}
|
2013-03-07 10:18:13 +00:00
|
|
|
d1++;
|
|
|
|
}
|
|
|
|
eventTimes.add(d1, pt[p2]);
|
|
|
|
xCoords.add(d1, px[p2]);
|
|
|
|
yCoords.add(d1, py[p2]);
|
2013-05-20 22:24:58 +00:00
|
|
|
if (GestureTrail.DEBUG_SHOW_POINTS) {
|
2013-05-13 04:12:12 +00:00
|
|
|
types.add(d1, GestureTrail.POINT_TYPE_SAMPLED);
|
2013-05-09 10:16:11 +00:00
|
|
|
}
|
2013-03-07 10:18:13 +00:00
|
|
|
}
|
|
|
|
return lastInterpolatedDrawIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static final double TWO_PI = Math.PI * 2.0d;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate the angular of rotation from <code>a0</code> to <code>a1</code>.
|
|
|
|
*
|
|
|
|
* @param a1 the angular to which the rotation ends.
|
|
|
|
* @param a0 the angular from which the rotation starts.
|
|
|
|
* @return the angular rotation value from a0 to a1, normalized to [-PI, +PI].
|
|
|
|
*/
|
|
|
|
private static double angularDiff(final double a1, final double a0) {
|
|
|
|
double deltaAngle = a1 - a0;
|
|
|
|
while (deltaAngle > Math.PI) {
|
|
|
|
deltaAngle -= TWO_PI;
|
|
|
|
}
|
|
|
|
while (deltaAngle < -Math.PI) {
|
|
|
|
deltaAngle += TWO_PI;
|
|
|
|
}
|
|
|
|
return deltaAngle;
|
|
|
|
}
|
2012-08-20 03:57:34 +00:00
|
|
|
}
|