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.incallui.answer.impl.classifier;
18 
19 import android.util.ArrayMap;
20 import android.view.MotionEvent;
21 import java.util.ArrayList;
22 import java.util.List;
23 import java.util.Map;
24 
25 /**
26  * A classifier which for each point from a stroke, it creates a point on plane with coordinates
27  * (timeOffsetNano, distanceCoveredUpToThisPoint) (scaled by DURATION_SCALE and LENGTH_SCALE) and
28  * then it calculates the angle variance of these points like the class {@link AnglesClassifier}
29  * (without splitting it into two parts). The classifier ignores the last point of a stroke because
30  * the UP event comes in with some delay and this ruins the smoothness of this curve. Additionally,
31  * the classifier classifies calculates the percentage of angles which value is in [PI -
32  * ANGLE_DEVIATION, 2* PI) interval. The reason why the classifier does that is because the speed of
33  * a good stroke is most often increases, so most of these angels should be in this interval.
34  */
35 class SpeedAnglesClassifier extends StrokeClassifier {
36   private Map<Stroke, Data> strokeMap = new ArrayMap<>();
37 
SpeedAnglesClassifier(ClassifierData classifierData)38   public SpeedAnglesClassifier(ClassifierData classifierData) {
39     this.classifierData = classifierData;
40   }
41 
42   @Override
getTag()43   public String getTag() {
44     return "SPD_ANG";
45   }
46 
47   @Override
onTouchEvent(MotionEvent event)48   public void onTouchEvent(MotionEvent event) {
49     int action = event.getActionMasked();
50 
51     if (action == MotionEvent.ACTION_DOWN) {
52       strokeMap.clear();
53     }
54 
55     for (int i = 0; i < event.getPointerCount(); i++) {
56       Stroke stroke = classifierData.getStroke(event.getPointerId(i));
57 
58       if (strokeMap.get(stroke) == null) {
59         strokeMap.put(stroke, new Data());
60       }
61 
62       if (action != MotionEvent.ACTION_UP
63           && action != MotionEvent.ACTION_CANCEL
64           && !(action == MotionEvent.ACTION_POINTER_UP && i == event.getActionIndex())) {
65         strokeMap.get(stroke).addPoint(stroke.getPoints().get(stroke.getPoints().size() - 1));
66       }
67     }
68   }
69 
70   @Override
getFalseTouchEvaluation(Stroke stroke)71   public float getFalseTouchEvaluation(Stroke stroke) {
72     Data data = strokeMap.get(stroke);
73     return SpeedVarianceEvaluator.evaluate(data.getAnglesVariance())
74         + SpeedAnglesPercentageEvaluator.evaluate(data.getAnglesPercentage());
75   }
76 
77   private static class Data {
78     private static final float DURATION_SCALE = 1e8f;
79     private static final float LENGTH_SCALE = 1.0f;
80     private static final float ANGLE_DEVIATION = (float) Math.PI / 10.0f;
81 
82     private List<Point> lastThreePoints = new ArrayList<>();
83     private Point previousPoint;
84     private float previousAngle;
85     private float sumSquares;
86     private float sum;
87     private float count;
88     private float dist;
89     private float anglesCount;
90     private float acceleratingAngles;
91 
Data()92     public Data() {
93       previousPoint = null;
94       previousAngle = (float) Math.PI;
95       sumSquares = 0.0f;
96       sum = 0.0f;
97       count = 1.0f;
98       dist = 0.0f;
99       anglesCount = acceleratingAngles = 0.0f;
100     }
101 
addPoint(Point point)102     public void addPoint(Point point) {
103       if (previousPoint != null) {
104         dist += previousPoint.dist(point);
105       }
106 
107       previousPoint = point;
108       Point speedPoint =
109           new Point((float) point.timeOffsetNano / DURATION_SCALE, dist / LENGTH_SCALE);
110 
111       // Checking if the added point is different than the previously added point
112       // Repetitions are being ignored so that proper angles are calculated.
113       if (lastThreePoints.isEmpty()
114           || !lastThreePoints.get(lastThreePoints.size() - 1).equals(speedPoint)) {
115         lastThreePoints.add(speedPoint);
116         if (lastThreePoints.size() == 4) {
117           lastThreePoints.remove(0);
118 
119           float angle =
120               lastThreePoints.get(1).getAngle(lastThreePoints.get(0), lastThreePoints.get(2));
121 
122           anglesCount++;
123           if (angle >= (float) Math.PI - ANGLE_DEVIATION) {
124             acceleratingAngles++;
125           }
126 
127           float difference = angle - previousAngle;
128           sum += difference;
129           sumSquares += difference * difference;
130           count += 1.0f;
131           previousAngle = angle;
132         }
133       }
134     }
135 
getAnglesVariance()136     public float getAnglesVariance() {
137       return sumSquares / count - (sum / count) * (sum / count);
138     }
139 
getAnglesPercentage()140     public float getAnglesPercentage() {
141       if (anglesCount == 0.0f) {
142         return 1.0f;
143       }
144       return (acceleratingAngles) / anglesCount;
145     }
146   }
147 }
148