2012-07-31 04:55:42 +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.research;
|
|
|
|
|
2012-08-09 19:20:45 +00:00
|
|
|
import android.util.Log;
|
|
|
|
|
2012-10-29 05:46:34 +00:00
|
|
|
import com.android.inputmethod.latin.Constants;
|
2012-12-18 02:19:58 +00:00
|
|
|
import com.android.inputmethod.latin.define.ProductionFlag;
|
2012-07-31 04:55:42 +00:00
|
|
|
|
|
|
|
public class Statistics {
|
2012-08-09 19:20:45 +00:00
|
|
|
private static final String TAG = Statistics.class.getSimpleName();
|
2012-12-18 02:19:58 +00:00
|
|
|
private static final boolean DEBUG = false && ProductionFlag.IS_EXPERIMENTAL_DEBUG;
|
2012-08-09 19:20:45 +00:00
|
|
|
|
2012-07-31 04:55:42 +00:00
|
|
|
// Number of characters entered during a typing session
|
|
|
|
int mCharCount;
|
|
|
|
// Number of letter characters entered during a typing session
|
|
|
|
int mLetterCount;
|
|
|
|
// Number of number characters entered
|
|
|
|
int mNumberCount;
|
|
|
|
// Number of space characters entered
|
|
|
|
int mSpaceCount;
|
|
|
|
// Number of delete operations entered (taps on the backspace key)
|
|
|
|
int mDeleteKeyCount;
|
|
|
|
// Number of words entered during a session.
|
|
|
|
int mWordCount;
|
2012-08-10 21:21:18 +00:00
|
|
|
// Number of words found in the dictionary.
|
|
|
|
int mDictionaryWordCount;
|
|
|
|
// Number of words split and spaces automatically entered.
|
|
|
|
int mSplitWordsCount;
|
|
|
|
// Number of gestures that were input.
|
2012-12-21 20:22:06 +00:00
|
|
|
int mGesturesInputCount;
|
2012-08-13 17:30:12 +00:00
|
|
|
// Number of gestures that were deleted.
|
|
|
|
int mGesturesDeletedCount;
|
|
|
|
// Total number of characters in words entered by gesture.
|
2012-12-21 20:22:06 +00:00
|
|
|
int mGesturesCharsCount;
|
2012-07-31 04:55:42 +00:00
|
|
|
// Whether the text field was empty upon editing
|
|
|
|
boolean mIsEmptyUponStarting;
|
|
|
|
boolean mIsEmptinessStateKnown;
|
|
|
|
|
|
|
|
// Timers to count average time to enter a key, first press a delete key,
|
|
|
|
// between delete keys, and then to return typing after a delete key.
|
|
|
|
final AverageTimeCounter mKeyCounter = new AverageTimeCounter();
|
|
|
|
final AverageTimeCounter mBeforeDeleteKeyCounter = new AverageTimeCounter();
|
|
|
|
final AverageTimeCounter mDuringRepeatedDeleteKeysCounter = new AverageTimeCounter();
|
|
|
|
final AverageTimeCounter mAfterDeleteKeyCounter = new AverageTimeCounter();
|
|
|
|
|
|
|
|
static class AverageTimeCounter {
|
|
|
|
int mCount;
|
|
|
|
int mTotalTime;
|
|
|
|
|
|
|
|
public void reset() {
|
|
|
|
mCount = 0;
|
|
|
|
mTotalTime = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void add(long deltaTime) {
|
|
|
|
mCount++;
|
|
|
|
mTotalTime += deltaTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getAverageTime() {
|
|
|
|
if (mCount == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return mTotalTime / mCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// To account for the interruptions when the user's attention is directed elsewhere, times
|
|
|
|
// longer than MIN_TYPING_INTERMISSION are not counted when estimating this statistic.
|
2012-08-03 03:22:29 +00:00
|
|
|
public static final int MIN_TYPING_INTERMISSION = 2 * 1000; // in milliseconds
|
|
|
|
public static final int MIN_DELETION_INTERMISSION = 10 * 1000; // in milliseconds
|
2012-07-31 04:55:42 +00:00
|
|
|
|
|
|
|
// The last time that a tap was performed
|
|
|
|
private long mLastTapTime;
|
|
|
|
// The type of the last keypress (delete key or not)
|
|
|
|
boolean mIsLastKeyDeleteKey;
|
|
|
|
|
|
|
|
private static final Statistics sInstance = new Statistics();
|
|
|
|
|
|
|
|
public static Statistics getInstance() {
|
|
|
|
return sInstance;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Statistics() {
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void reset() {
|
|
|
|
mCharCount = 0;
|
|
|
|
mLetterCount = 0;
|
|
|
|
mNumberCount = 0;
|
|
|
|
mSpaceCount = 0;
|
|
|
|
mDeleteKeyCount = 0;
|
|
|
|
mWordCount = 0;
|
2012-12-21 20:22:06 +00:00
|
|
|
mDictionaryWordCount = 0;
|
|
|
|
mSplitWordsCount = 0;
|
|
|
|
mGesturesInputCount = 0;
|
|
|
|
mGesturesDeletedCount = 0;
|
2012-07-31 04:55:42 +00:00
|
|
|
mIsEmptyUponStarting = true;
|
|
|
|
mIsEmptinessStateKnown = false;
|
|
|
|
mKeyCounter.reset();
|
|
|
|
mBeforeDeleteKeyCounter.reset();
|
|
|
|
mDuringRepeatedDeleteKeysCounter.reset();
|
|
|
|
mAfterDeleteKeyCounter.reset();
|
2012-12-21 20:22:06 +00:00
|
|
|
mGesturesCharsCount = 0;
|
2012-08-13 17:30:12 +00:00
|
|
|
mGesturesDeletedCount = 0;
|
2012-07-31 04:55:42 +00:00
|
|
|
|
|
|
|
mLastTapTime = 0;
|
|
|
|
mIsLastKeyDeleteKey = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void recordChar(int codePoint, long time) {
|
2012-08-09 19:20:45 +00:00
|
|
|
if (DEBUG) {
|
|
|
|
Log.d(TAG, "recordChar() called");
|
|
|
|
}
|
2012-07-31 04:55:42 +00:00
|
|
|
final long delta = time - mLastTapTime;
|
2012-10-29 05:46:34 +00:00
|
|
|
if (codePoint == Constants.CODE_DELETE) {
|
2012-07-31 04:55:42 +00:00
|
|
|
mDeleteKeyCount++;
|
|
|
|
if (delta < MIN_DELETION_INTERMISSION) {
|
|
|
|
if (mIsLastKeyDeleteKey) {
|
|
|
|
mDuringRepeatedDeleteKeysCounter.add(delta);
|
|
|
|
} else {
|
|
|
|
mBeforeDeleteKeyCounter.add(delta);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mIsLastKeyDeleteKey = true;
|
|
|
|
} else {
|
|
|
|
mCharCount++;
|
|
|
|
if (Character.isDigit(codePoint)) {
|
|
|
|
mNumberCount++;
|
|
|
|
}
|
|
|
|
if (Character.isLetter(codePoint)) {
|
|
|
|
mLetterCount++;
|
|
|
|
}
|
|
|
|
if (Character.isSpaceChar(codePoint)) {
|
|
|
|
mSpaceCount++;
|
|
|
|
}
|
|
|
|
if (mIsLastKeyDeleteKey && delta < MIN_DELETION_INTERMISSION) {
|
|
|
|
mAfterDeleteKeyCounter.add(delta);
|
|
|
|
} else if (!mIsLastKeyDeleteKey && delta < MIN_TYPING_INTERMISSION) {
|
|
|
|
mKeyCounter.add(delta);
|
|
|
|
}
|
|
|
|
mIsLastKeyDeleteKey = false;
|
|
|
|
}
|
|
|
|
mLastTapTime = time;
|
|
|
|
}
|
|
|
|
|
2012-08-10 21:21:18 +00:00
|
|
|
public void recordWordEntered(final boolean isDictionaryWord) {
|
2012-07-31 04:55:42 +00:00
|
|
|
mWordCount++;
|
2012-08-10 21:21:18 +00:00
|
|
|
if (isDictionaryWord) {
|
|
|
|
mDictionaryWordCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void recordSplitWords() {
|
|
|
|
mSplitWordsCount++;
|
|
|
|
}
|
|
|
|
|
2012-08-13 17:30:12 +00:00
|
|
|
public void recordGestureInput(final int numCharsEntered) {
|
2012-12-21 20:22:06 +00:00
|
|
|
mGesturesInputCount++;
|
|
|
|
mGesturesCharsCount += numCharsEntered;
|
2012-07-31 04:55:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void setIsEmptyUponStarting(final boolean isEmpty) {
|
|
|
|
mIsEmptyUponStarting = isEmpty;
|
|
|
|
mIsEmptinessStateKnown = true;
|
|
|
|
}
|
2012-08-13 17:30:12 +00:00
|
|
|
|
|
|
|
public void recordGestureDelete() {
|
|
|
|
mGesturesDeletedCount++;
|
|
|
|
}
|
2012-07-31 04:55:42 +00:00
|
|
|
}
|