proximity_info_state.cpp revision 08f00cf55f2e083c1ed254a32495b622c9ad9862
13e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka/*
23e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * Copyright (C) 2012 The Android Open Source Project
33e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka *
43e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * Licensed under the Apache License, Version 2.0 (the "License");
53e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * you may not use this file except in compliance with the License.
63e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * You may obtain a copy of the License at
73e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka *
83e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka *      http://www.apache.org/licenses/LICENSE-2.0
93e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka *
103e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * Unless required by applicable law or agreed to in writing, software
113e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * distributed under the License is distributed on an "AS IS" BASIS,
123e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
133e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * See the License for the specific language governing permissions and
143e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka * limitations under the License.
153e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka */
163e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka
1777e8e81ad95cfc1eb8f8407fc872674b8d08bbe9Ken Wakasa#include <cstring> // for memset()
183e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka#include <stdint.h>
193e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka
203e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka#define LOG_TAG "LatinIME: proximity_info_state.cpp"
213e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka
223e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka#include "defines.h"
23687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka#include "geometry_utils.h"
243e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka#include "proximity_info.h"
253e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka#include "proximity_info_state.h"
263e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka
273e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataokanamespace latinime {
28233aad5e5c7567a97af30f38f50a65365f729dfeSatoshi Kataokavoid ProximityInfoState::initInputParams(const int pointerId, const float maxPointToKeyLength,
2908f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        const ProximityInfo *proximityInfo, const int32_t *const inputCodes, const int inputSize,
30687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const int *const xCoordinates, const int *const yCoordinates, const int *const times,
31687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const int *const pointerIds, const bool isGeometric) {
324a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mProximityInfo = proximityInfo;
334a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mHasTouchPositionCorrectionData = proximityInfo->hasTouchPositionCorrectionData();
344a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mMostCommonKeyWidthSquare = proximityInfo->getMostCommonKeyWidthSquare();
354a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mLocaleStr = proximityInfo->getLocaleStr();
364a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mKeyCount = proximityInfo->getKeyCount();
374a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mCellHeight = proximityInfo->getCellHeight();
384a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mCellWidth = proximityInfo->getCellWidth();
394a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mGridHeight = proximityInfo->getGridWidth();
404a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    mGridWidth = proximityInfo->getGridHeight();
413e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka
4208f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka    memset(mInputCodes, 0, sizeof(mInputCodes));
433e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka
4408f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka    if (!isGeometric && pointerId == 0) {
4508f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        // Initialize
4608f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        // - mInputCodes
4708f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        // - mNormalizedSquaredDistances
4808f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        // TODO: Merge
49687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        for (int i = 0; i < inputSize; ++i) {
5008f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            const int32_t primaryKey = inputCodes[i];
5108f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            const int x = xCoordinates[i];
5208f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            const int y = yCoordinates[i];
5308f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            int *proximities = &mInputCodes[i * MAX_PROXIMITY_CHARS_SIZE_INTERNAL];
5408f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            mProximityInfo->calculateNearbyKeyCodes(x, y, primaryKey, proximities);
5508f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        }
5608f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka
5708f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        if (DEBUG_PROXIMITY_CHARS) {
5808f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            for (int i = 0; i < inputSize; ++i) {
5908f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                AKLOGI("---");
6008f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                for (int j = 0; j < MAX_PROXIMITY_CHARS_SIZE_INTERNAL; ++j) {
6108f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    int icc = mInputCodes[i * MAX_PROXIMITY_CHARS_SIZE_INTERNAL + j];
6208f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    int icfjc = inputCodes[i * MAX_PROXIMITY_CHARS_SIZE_INTERNAL + j];
6308f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    icc += 0;
6408f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    icfjc += 0;
6508f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    AKLOGI("--- (%d)%c,%c", i, icc, icfjc); AKLOGI("--- A<%d>,B<%d>", icc, icfjc);
6608f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                }
673e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka            }
683e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka        }
693e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka    }
70687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
71687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    ///////////////////////
72687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    // Setup touch points
7308f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka    mMaxPointToKeyLength = maxPointToKeyLength;
74687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mInputXs.clear();
75687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mInputYs.clear();
76687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mTimes.clear();
77687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mLengthCache.clear();
78687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mDistanceCache.clear();
79687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
80687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mInputSize = 0;
81687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    if (xCoordinates && yCoordinates) {
82687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const bool proximityOnly = !isGeometric && (xCoordinates[0] < 0 || yCoordinates[0] < 0);
83687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        for (int i = 0; i < inputSize; ++i) {
84687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            // Assuming pointerId == 0 if pointerIds is null.
85687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            const int pid = pointerIds ? pointerIds[i] : 0;
86687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            if (pointerId == pid) {
8708f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                ++mInputSize;
88687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                const int c = isGeometric ? NOT_A_COORDINATE : getPrimaryCharAt(i);
89687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                const int x = proximityOnly ? NOT_A_COORDINATE : xCoordinates[i];
90687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                const int y = proximityOnly ? NOT_A_COORDINATE : yCoordinates[i];
91687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                const int time = times ? times[i] : -1;
92687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                pushTouchPoint(c, x, y, time, isGeometric);
93687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            }
94687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        }
953e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka    }
96687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
97687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    if (mInputSize > 0) {
98687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const int keyCount = mProximityInfo->getKeyCount();
99687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        mDistanceCache.resize(mInputSize * keyCount);
100687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        for (int i = 0; i < mInputSize; ++i) {
101687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            for (int k = 0; k < keyCount; ++k) {
102687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                const int index = i * keyCount + k;
103687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                const int x = mInputXs[i];
104687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                const int y = mInputYs[i];
105687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                mDistanceCache[index] =
106687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka                        mProximityInfo->getNormalizedSquaredDistanceFromCenterFloat(k, x, y);
107687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            }
108687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        }
109687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    }
11008f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka
111687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    // end
112687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    ///////////////////////
113687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
11408f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka    memset(mNormalizedSquaredDistances, NOT_A_DISTANCE, sizeof(mNormalizedSquaredDistances));
11508f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka    memset(mPrimaryInputWord, 0, sizeof(mPrimaryInputWord));
116233aad5e5c7567a97af30f38f50a65365f729dfeSatoshi Kataoka    mTouchPositionCorrectionEnabled = mInputSize > 0 && mHasTouchPositionCorrectionData
117233aad5e5c7567a97af30f38f50a65365f729dfeSatoshi Kataoka            && xCoordinates && yCoordinates && !isGeometric;
11808f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka    if (!isGeometric && pointerId == 0) {
11908f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        for (int i = 0; i < inputSize; ++i) {
12008f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            mPrimaryInputWord[i] = getPrimaryCharAt(i);
1213e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka        }
12208f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka
12308f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka        for (int i = 0; i < mInputSize && mTouchPositionCorrectionEnabled; ++i) {
12408f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            const int *proximityChars = getProximityCharsAt(i);
12508f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            const int primaryKey = proximityChars[0];
12608f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            const int x = xCoordinates[i];
12708f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            const int y = yCoordinates[i];
1283e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka            if (DEBUG_PROXIMITY_CHARS) {
12908f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                int a = x + y + primaryKey;
13008f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                a += 0;
13108f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                AKLOGI("--- Primary = %c, x = %d, y = %d", primaryKey, x, y);
13208f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            }
13308f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka            for (int j = 0; j < MAX_PROXIMITY_CHARS_SIZE_INTERNAL && proximityChars[j] > 0; ++j) {
13408f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                const int currentChar = proximityChars[j];
13508f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                const float squaredDistance =
13608f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                        hasInputCoordinates() ? calculateNormalizedSquaredDistance(
13708f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                                mProximityInfo->getKeyIndex(currentChar), i) :
13808f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                                NOT_A_DISTANCE_FLOAT;
13908f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                if (squaredDistance >= 0.0f) {
14008f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    mNormalizedSquaredDistances[i * MAX_PROXIMITY_CHARS_SIZE_INTERNAL + j] =
14108f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                            (int) (squaredDistance * NORMALIZED_SQUARED_DISTANCE_SCALING_FACTOR);
14208f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                } else {
14308f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    mNormalizedSquaredDistances[i * MAX_PROXIMITY_CHARS_SIZE_INTERNAL + j] =
14408f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                            (j == 0) ? EQUIVALENT_CHAR_WITHOUT_DISTANCE_INFO :
14508f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                                    PROXIMITY_CHAR_WITHOUT_DISTANCE_INFO;
14608f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                }
14708f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                if (DEBUG_PROXIMITY_CHARS) {
14808f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                    AKLOGI("--- Proximity (%d) = %c", j, currentChar);
14908f00cf55f2e083c1ed254a32495b622c9ad9862Satoshi Kataoka                }
1503e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka            }
1513e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka        }
1523e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka    }
1533e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka}
1544a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka
155687a244703a02323ebd64433cbaead5def499861Satoshi Kataokavoid ProximityInfoState::pushTouchPoint(const int nodeChar, int x, int y,
156687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const int time, const bool sample) {
157687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    const uint32_t size = mInputXs.size();
158687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    // TODO: Should have a const variable for 10
159687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    const int sampleRate = mProximityInfo->getMostCommonKeyWidth() / 10;
160687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    if (size > 0) {
161687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const int dist = getDistanceInt(x, y, mInputXs[size - 1], mInputYs[size - 1]);
162687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        if (sample && dist < sampleRate) {
163687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            return;
164687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        }
165687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        mLengthCache.push_back(mLengthCache[size - 1] + dist);
166687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    } else {
167687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        mLengthCache.push_back(0);
168687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    }
169687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    if (nodeChar >= 0 && (x < 0 || y < 0)) {
170687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const int keyId = mProximityInfo->getKeyIndex(nodeChar);
171687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        if (keyId >= 0) {
172687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            x = mProximityInfo->getKeyCenterXOfIdG(keyId);
173687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka            y = mProximityInfo->getKeyCenterYOfIdG(keyId);
174687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        }
175687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    }
176687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mInputXs.push_back(x);
177687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mInputYs.push_back(y);
178687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    mTimes.push_back(time);
179687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka}
180687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
1814a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataokafloat ProximityInfoState::calculateNormalizedSquaredDistance(
1824a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka        const int keyIndex, const int inputIndex) const {
1834a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    if (keyIndex == NOT_AN_INDEX) {
1844a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka        return NOT_A_DISTANCE_FLOAT;
1854a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    }
1864a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    if (!mProximityInfo->hasSweetSpotData(keyIndex)) {
1874a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka        return NOT_A_DISTANCE_FLOAT;
1884a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    }
189687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    if (NOT_A_COORDINATE == mInputXs[inputIndex]) {
1904a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka        return NOT_A_DISTANCE_FLOAT;
1914a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    }
1924a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    const float squaredDistance = calculateSquaredDistanceFromSweetSpotCenter(
1934a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka            keyIndex, inputIndex);
1944a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    const float squaredRadius = square(mProximityInfo->getSweetSpotRadiiAt(keyIndex));
1954a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    return squaredDistance / squaredRadius;
1964a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka}
1974a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka
198687a244703a02323ebd64433cbaead5def499861Satoshi Kataokaint ProximityInfoState::getDuration(const int index) const {
199687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    if (mTimes.size() == 0 || index <= 0 || index >= static_cast<int>(mInputSize) - 1) {
200687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        return 0;
201687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    }
202687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    return mTimes[index + 1] - mTimes[index - 1];
203687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka}
204687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
205687a244703a02323ebd64433cbaead5def499861Satoshi Kataokafloat ProximityInfoState::getPointToKeyLength(int inputIndex, int charCode, float scale) {
206687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    const int keyId = mProximityInfo->getKeyIndex(charCode);
207687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    if (keyId >= 0) {
208687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        const int index = inputIndex * mProximityInfo->getKeyCount() + keyId;
209687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka        return min(mDistanceCache[index] * scale, mMaxPointToKeyLength);
210687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    }
211687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    return 0;
212687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka}
213687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
214687a244703a02323ebd64433cbaead5def499861Satoshi Kataokaint ProximityInfoState::getKeyKeyDistance(int key0, int key1) {
215687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    return mProximityInfo->getKeyKeyDistanceG(key0, key1);
216687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka}
217687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
218687a244703a02323ebd64433cbaead5def499861Satoshi Kataokaint ProximityInfoState::getSpaceY() {
219687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    const int keyId = mProximityInfo->getKeyIndex(' ');
220687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    return mProximityInfo->getKeyCenterYOfIdG(keyId);
221687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka}
222687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka
2234a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataokafloat ProximityInfoState::calculateSquaredDistanceFromSweetSpotCenter(
2244a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka        const int keyIndex, const int inputIndex) const {
2254a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    const float sweetSpotCenterX = mProximityInfo->getSweetSpotCenterXAt(keyIndex);
2264a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    const float sweetSpotCenterY = mProximityInfo->getSweetSpotCenterYAt(keyIndex);
227687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    const float inputX = static_cast<float>(mInputXs[inputIndex]);
228687a244703a02323ebd64433cbaead5def499861Satoshi Kataoka    const float inputY = static_cast<float>(mInputYs[inputIndex]);
2294a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka    return square(inputX - sweetSpotCenterX) + square(inputY - sweetSpotCenterY);
2304a3db7057f77dc85311fb1f94934b5a004ab613eSatoshi Kataoka}
2313e8c58f68d53e6cc9dbf59201c7bdfb8ad04a1cdSatoshi Kataoka} // namespace latinime
232