1 /* 2 * Copyright (C) 2015 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License 15 */ 16 17 package com.android.systemui.classifier; 18 19 import android.view.MotionEvent; 20 21 import java.util.ArrayList; 22 import java.util.HashMap; 23 import java.util.List; 24 25 /** 26 * A classifier which calculates the variance of differences between successive angles in a stroke. 27 * For each stroke it keeps its last three points. If some successive points are the same, it 28 * ignores the repetitions. If a new point is added, the classifier calculates the angle between 29 * the last three points. After that, it calculates the difference between this angle and the 30 * previously calculated angle. Then it calculates the variance of the differences from a stroke. 31 * To the differences there is artificially added value 0.0 and the difference between the first 32 * angle and PI (angles are in radians). It helps with strokes which have few points and punishes 33 * more strokes which are not smooth. 34 * 35 * This classifier also tries to split the stroke into two parts in the place in which the biggest 36 * angle is. It calculates the angle variance of the two parts and sums them up. The reason the 37 * classifier is doing this, is because some human swipes at the beginning go for a moment in one 38 * direction and then they rapidly change direction for the rest of the stroke (like a tick). The 39 * final result is the minimum of angle variance of the whole stroke and the sum of angle variances 40 * of the two parts split up. The classifier tries the tick option only if the first part is 41 * shorter than the second part. 42 * 43 * Additionally, the classifier classifies the angles as left angles (those angles which value is 44 * in [0.0, PI - ANGLE_DEVIATION) interval), straight angles 45 * ([PI - ANGLE_DEVIATION, PI + ANGLE_DEVIATION] interval) and right angles 46 * ((PI + ANGLE_DEVIATION, 2 * PI) interval) and then calculates the percentage of angles which are 47 * in the same direction (straight angles can be left angels or right angles) 48 */ 49 public class AnglesClassifier extends StrokeClassifier { 50 private HashMap<Stroke, Data> mStrokeMap = new HashMap<>(); 51 AnglesClassifier(ClassifierData classifierData)52 public AnglesClassifier(ClassifierData classifierData) { 53 mClassifierData = classifierData; 54 } 55 56 @Override getTag()57 public String getTag() { 58 return "ANG"; 59 } 60 61 @Override onTouchEvent(MotionEvent event)62 public void onTouchEvent(MotionEvent event) { 63 int action = event.getActionMasked(); 64 65 if (action == MotionEvent.ACTION_DOWN) { 66 mStrokeMap.clear(); 67 } 68 69 for (int i = 0; i < event.getPointerCount(); i++) { 70 Stroke stroke = mClassifierData.getStroke(event.getPointerId(i)); 71 72 if (mStrokeMap.get(stroke) == null) { 73 mStrokeMap.put(stroke, new Data()); 74 } 75 mStrokeMap.get(stroke).addPoint(stroke.getPoints().get(stroke.getPoints().size() - 1)); 76 } 77 } 78 79 @Override getFalseTouchEvaluation(int type, Stroke stroke)80 public float getFalseTouchEvaluation(int type, Stroke stroke) { 81 Data data = mStrokeMap.get(stroke); 82 return AnglesVarianceEvaluator.evaluate(data.getAnglesVariance()) 83 + AnglesPercentageEvaluator.evaluate(data.getAnglesPercentage()); 84 } 85 86 private static class Data { 87 private final float ANGLE_DEVIATION = (float) Math.PI / 20.0f; 88 89 private List<Point> mLastThreePoints = new ArrayList<>(); 90 private float mFirstAngleVariance; 91 private float mPreviousAngle; 92 private float mBiggestAngle; 93 private float mSumSquares; 94 private float mSecondSumSquares; 95 private float mSum; 96 private float mSecondSum; 97 private float mCount; 98 private float mSecondCount; 99 private float mFirstLength; 100 private float mLength; 101 private float mAnglesCount; 102 private float mLeftAngles; 103 private float mRightAngles; 104 private float mStraightAngles; 105 Data()106 public Data() { 107 mFirstAngleVariance = 0.0f; 108 mPreviousAngle = (float) Math.PI; 109 mBiggestAngle = 0.0f; 110 mSumSquares = mSecondSumSquares = 0.0f; 111 mSum = mSecondSum = 0.0f; 112 mCount = mSecondCount = 1.0f; 113 mLength = mFirstLength = 0.0f; 114 mAnglesCount = mLeftAngles = mRightAngles = mStraightAngles = 0.0f; 115 } 116 addPoint(Point point)117 public void addPoint(Point point) { 118 // Checking if the added point is different than the previously added point 119 // Repetitions are being ignored so that proper angles are calculated. 120 if (mLastThreePoints.isEmpty() 121 || !mLastThreePoints.get(mLastThreePoints.size() - 1).equals(point)) { 122 if (!mLastThreePoints.isEmpty()) { 123 mLength += mLastThreePoints.get(mLastThreePoints.size() - 1).dist(point); 124 } 125 mLastThreePoints.add(point); 126 if (mLastThreePoints.size() == 4) { 127 mLastThreePoints.remove(0); 128 129 float angle = mLastThreePoints.get(1).getAngle(mLastThreePoints.get(0), 130 mLastThreePoints.get(2)); 131 132 mAnglesCount++; 133 if (angle < Math.PI - ANGLE_DEVIATION) { 134 mLeftAngles++; 135 } else if (angle <= Math.PI + ANGLE_DEVIATION) { 136 mStraightAngles++; 137 } else { 138 mRightAngles++; 139 } 140 141 float difference = angle - mPreviousAngle; 142 143 // If this is the biggest angle of the stroke so then we save the value of 144 // the angle variance so far and start to count the values for the angle 145 // variance of the second part. 146 if (mBiggestAngle < angle) { 147 mBiggestAngle = angle; 148 mFirstLength = mLength; 149 mFirstAngleVariance = getAnglesVariance(mSumSquares, mSum, mCount); 150 mSecondSumSquares = 0.0f; 151 mSecondSum = 0.0f; 152 mSecondCount = 1.0f; 153 } else { 154 mSecondSum += difference; 155 mSecondSumSquares += difference * difference; 156 mSecondCount += 1.0; 157 } 158 159 mSum += difference; 160 mSumSquares += difference * difference; 161 mCount += 1.0; 162 mPreviousAngle = angle; 163 } 164 } 165 } 166 getAnglesVariance(float sumSquares, float sum, float count)167 public float getAnglesVariance(float sumSquares, float sum, float count) { 168 return sumSquares / count - (sum / count) * (sum / count); 169 } 170 getAnglesVariance()171 public float getAnglesVariance() { 172 float anglesVariance = getAnglesVariance(mSumSquares, mSum, mCount); 173 if (mFirstLength < mLength / 2f) { 174 anglesVariance = Math.min(anglesVariance, mFirstAngleVariance 175 + getAnglesVariance(mSecondSumSquares, mSecondSum, mSecondCount)); 176 } 177 return anglesVariance; 178 } 179 getAnglesPercentage()180 public float getAnglesPercentage() { 181 if (mAnglesCount == 0.0f) { 182 return 1.0f; 183 } 184 return (Math.max(mLeftAngles, mRightAngles) + mStraightAngles) / mAnglesCount; 185 } 186 } 187 }