1 /* 2 * Copyright (C) 2008-2012 OMRON SOFTWARE Co., Ltd. 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 /* This file is porting from Android framework. 17 * frameworks/base/core/java/android/inputmethodservice/KeyboardView.java 18 * 19 *package android.inputmethodservice; 20 */ 21 22 package jp.co.omronsoft.openwnn; 23 24 import android.content.Context; 25 import android.content.res.TypedArray; 26 import android.graphics.Bitmap; 27 import android.graphics.Canvas; 28 import android.graphics.Paint; 29 import android.graphics.PorterDuff; 30 import android.graphics.Rect; 31 import android.graphics.Typeface; 32 import android.graphics.Paint.Align; 33 import android.graphics.Region.Op; 34 import android.graphics.drawable.Drawable; 35 import android.os.Handler; 36 import android.os.Message; 37 import android.util.AttributeSet; 38 import android.util.TypedValue; 39 import android.view.GestureDetector; 40 import android.view.Gravity; 41 import android.view.LayoutInflater; 42 import android.view.MotionEvent; 43 import android.view.View; 44 import android.view.ViewConfiguration; 45 import android.view.ViewGroup.LayoutParams; 46 import android.widget.PopupWindow; 47 import android.widget.TextView; 48 49 import jp.co.omronsoft.openwnn.Keyboard; 50 import jp.co.omronsoft.openwnn.Keyboard.Key; 51 52 import java.util.Arrays; 53 import java.util.HashMap; 54 import java.util.List; 55 import java.util.Map; 56 57 /** 58 * A view that renders a virtual {@link Keyboard}. It handles rendering of keys and 59 * detecting key presses and touch movements. 60 */ 61 public class KeyboardView extends View implements View.OnClickListener { 62 63 /** 64 * Listener for virtual keyboard events. 65 */ 66 public interface OnKeyboardActionListener { 67 68 /** 69 * Called when the user presses a key. This is sent before the {@link #onKey} is called. 70 * For keys that repeat, this is only called once. 71 * @param primaryCode the unicode of the key being pressed. If the touch is not on a valid 72 * key, the value will be zero. 73 */ onPress(int primaryCode)74 void onPress(int primaryCode); 75 76 /** 77 * Called when the user releases a key. This is sent after the {@link #onKey} is called. 78 * For keys that repeat, this is only called once. 79 * @param primaryCode the code of the key that was released 80 */ onRelease(int primaryCode)81 void onRelease(int primaryCode); 82 83 /** 84 * Send a key press to the listener. 85 * @param primaryCode this is the key that was pressed 86 * @param keyCodes the codes for all the possible alternative keys 87 * with the primary code being the first. If the primary key code is 88 * a single character such as an alphabet or number or symbol, the alternatives 89 * will include other characters that may be on the same key or adjacent keys. 90 * These codes are useful to correct for accidental presses of a key adjacent to 91 * the intended key. 92 */ onKey(int primaryCode, int[] keyCodes)93 void onKey(int primaryCode, int[] keyCodes); 94 95 /** 96 * Sends a sequence of characters to the listener. 97 * @param text the sequence of characters to be displayed. 98 */ onText(CharSequence text)99 void onText(CharSequence text); 100 101 /** 102 * Called when the user quickly moves the finger from right to left. 103 */ swipeLeft()104 void swipeLeft(); 105 106 /** 107 * Called when the user quickly moves the finger from left to right. 108 */ swipeRight()109 void swipeRight(); 110 111 /** 112 * Called when the user quickly moves the finger from up to down. 113 */ swipeDown()114 void swipeDown(); 115 116 /** 117 * Called when the user quickly moves the finger from down to up. 118 */ swipeUp()119 void swipeUp(); 120 121 /** 122 * Called when the user long presses a key. 123 * @param popupKey the key that was long pressed 124 * @return true if the long press is handled, false otherwise. 125 */ onLongPress(Keyboard.Key key)126 boolean onLongPress(Keyboard.Key key); 127 } 128 129 private static final int NOT_A_KEY = -1; 130 private static final int[] KEY_DELETE = { Keyboard.KEYCODE_DELETE }; 131 private static final int[] LONG_PRESSABLE_STATE_SET = { 132 android.R.attr.state_long_pressable 133 }; 134 135 private Keyboard mKeyboard; 136 private int mCurrentKeyIndex = NOT_A_KEY; 137 private int mLabelTextSize; 138 private int mKeyTextSize; 139 private int mKeyTextColor; 140 private int mKeyTextColor2nd; 141 private float mShadowRadius; 142 private int mShadowColor; 143 private float mBackgroundDimAmount; 144 145 private TextView mPreviewText; 146 private PopupWindow mPreviewPopup; 147 private int mPreviewTextSizeLarge; 148 private int mPreviewOffset; 149 private int mPreviewHeight; 150 private int[] mOffsetInWindow; 151 152 private PopupWindow mPopupKeyboard; 153 private View mMiniKeyboardContainer; 154 private KeyboardView mMiniKeyboard; 155 private boolean mMiniKeyboardOnScreen; 156 private View mPopupParent; 157 private int mMiniKeyboardOffsetX; 158 private int mMiniKeyboardOffsetY; 159 private Map<Key,View> mMiniKeyboardCache; 160 private int[] mWindowOffset; 161 private Key[] mKeys; 162 163 /** Listener for {@link OnKeyboardActionListener}. */ 164 private OnKeyboardActionListener mKeyboardActionListener; 165 166 private static final int MSG_SHOW_PREVIEW = 1; 167 private static final int MSG_REMOVE_PREVIEW = 2; 168 private static final int MSG_REPEAT = 3; 169 private static final int MSG_LONGPRESS = 4; 170 171 private static final int DELAY_BEFORE_PREVIEW = 0; 172 private static final int DELAY_AFTER_PREVIEW = 70; 173 private static final int DEBOUNCE_TIME = 70; 174 175 private int mVerticalCorrection; 176 private int mProximityThreshold; 177 178 private boolean mPreviewCentered = false; 179 private boolean mShowPreview = true; 180 private boolean mShowTouchPoints = true; 181 private int mPopupPreviewX; 182 private int mPopupPreviewY; 183 private int mWindowY; 184 185 private int mLastX; 186 private int mLastY; 187 private int mStartX; 188 private int mStartY; 189 190 private boolean mProximityCorrectOn; 191 192 private Paint mPaint; 193 private Rect mPadding; 194 195 private long mDownTime; 196 private long mLastMoveTime; 197 private int mLastKey; 198 private int mLastCodeX; 199 private int mLastCodeY; 200 private int mCurrentKey = NOT_A_KEY; 201 private int mDownKey = NOT_A_KEY; 202 private long mLastKeyTime; 203 private long mCurrentKeyTime; 204 private int[] mKeyIndices = new int[12]; 205 private GestureDetector mGestureDetector; 206 private int mPopupX; 207 private int mPopupY; 208 private int mRepeatKeyIndex = NOT_A_KEY; 209 private int mPopupLayout; 210 private boolean mAbortKey; 211 private Key mInvalidatedKey; 212 private Rect mClipRegion = new Rect(0, 0, 0, 0); 213 private boolean mPossiblePoly; 214 private SwipeTracker mSwipeTracker = new SwipeTracker(); 215 private int mSwipeThreshold; 216 private boolean mDisambiguateSwipe; 217 218 private int mOldPointerCount = 1; 219 private float mOldPointerX; 220 private float mOldPointerY; 221 222 private Drawable mKeyBackground; 223 private Drawable mKeyBackground2nd; 224 225 private static final int REPEAT_INTERVAL = 50; 226 private static final int REPEAT_START_DELAY = 400; 227 private static final int LONGPRESS_TIMEOUT = ViewConfiguration.getLongPressTimeout(); 228 229 private static int MAX_NEARBY_KEYS = 12; 230 private int[] mDistances = new int[MAX_NEARBY_KEYS]; 231 232 private int mLastSentIndex; 233 private int mTapCount; 234 private long mLastTapTime; 235 private boolean mInMultiTap; 236 private static final int MULTITAP_INTERVAL = 800; 237 private StringBuilder mPreviewLabel = new StringBuilder(1); 238 239 /** Whether the keyboard bitmap needs to be redrawn before it's blitted. **/ 240 private boolean mDrawPending; 241 /** The dirty region in the keyboard bitmap */ 242 private Rect mDirtyRect = new Rect(); 243 /** The keyboard bitmap for faster updates */ 244 private Bitmap mBuffer; 245 /** Notes if the keyboard just changed, so that we could possibly reallocate the mBuffer. */ 246 private boolean mKeyboardChanged; 247 /** The canvas for the above mutable keyboard bitmap */ 248 private Canvas mCanvas; 249 250 Handler mHandler = new Handler() { 251 @Override 252 public void handleMessage(Message msg) { 253 switch (msg.what) { 254 case MSG_SHOW_PREVIEW: 255 showKey(msg.arg1); 256 break; 257 case MSG_REMOVE_PREVIEW: 258 mPreviewText.setVisibility(INVISIBLE); 259 break; 260 case MSG_REPEAT: 261 if (repeatKey()) { 262 Message repeat = Message.obtain(this, MSG_REPEAT); 263 sendMessageDelayed(repeat, REPEAT_INTERVAL); 264 } 265 break; 266 case MSG_LONGPRESS: 267 openPopupIfRequired((MotionEvent) msg.obj); 268 break; 269 } 270 } 271 }; 272 273 /** Constructor */ KeyboardView(Context context, AttributeSet attrs)274 public KeyboardView(Context context, AttributeSet attrs) { 275 this(context, attrs, 0); 276 } 277 278 /** Constructor */ KeyboardView(Context context, AttributeSet attrs, int defStyle)279 public KeyboardView(Context context, AttributeSet attrs, int defStyle) { 280 super(context, attrs, defStyle); 281 282 TypedArray a = 283 context.obtainStyledAttributes( 284 attrs, android.R.styleable.KeyboardView, defStyle, R.style.WnnKeyboardView); 285 286 LayoutInflater inflate = 287 (LayoutInflater) context 288 .getSystemService(Context.LAYOUT_INFLATER_SERVICE); 289 290 int previewLayout = 0; 291 int keyTextSize = 0; 292 293 int n = a.getIndexCount(); 294 295 for (int i = 0; i < n; i++) { 296 int attr = a.getIndex(i); 297 298 switch (attr) { 299 case android.R.styleable.KeyboardView_keyBackground: 300 mKeyBackground = a.getDrawable(attr); 301 break; 302 case android.R.styleable.KeyboardView_verticalCorrection: 303 mVerticalCorrection = a.getDimensionPixelOffset(attr, 0); 304 break; 305 case android.R.styleable.KeyboardView_keyPreviewLayout: 306 previewLayout = a.getResourceId(attr, 0); 307 break; 308 case android.R.styleable.KeyboardView_keyPreviewOffset: 309 mPreviewOffset = a.getDimensionPixelOffset(attr, 0); 310 break; 311 case android.R.styleable.KeyboardView_keyPreviewHeight: 312 mPreviewHeight = a.getDimensionPixelSize(attr, 80); 313 break; 314 case android.R.styleable.KeyboardView_keyTextSize: 315 mKeyTextSize = a.getDimensionPixelSize(attr, 18); 316 break; 317 case android.R.styleable.KeyboardView_keyTextColor: 318 mKeyTextColor = a.getColor(attr, 0xFF000000); 319 break; 320 case android.R.styleable.KeyboardView_labelTextSize: 321 mLabelTextSize = a.getDimensionPixelSize(attr, 14); 322 break; 323 case android.R.styleable.KeyboardView_popupLayout: 324 mPopupLayout = a.getResourceId(attr, 0); 325 break; 326 case android.R.styleable.KeyboardView_shadowColor: 327 mShadowColor = a.getColor(attr, 0); 328 break; 329 case android.R.styleable.KeyboardView_shadowRadius: 330 mShadowRadius = a.getFloat(attr, 0f); 331 break; 332 } 333 } 334 335 a.recycle(); 336 a = context.obtainStyledAttributes(attrs, R.styleable.WnnKeyboardView, 0, 0); 337 mKeyBackground2nd = a.getDrawable(R.styleable.WnnKeyboardView_keyBackground2nd); 338 mKeyTextColor2nd = a.getColor(R.styleable.WnnKeyboardView_keyTextColor2nd, 0xFF000000); 339 340 a.recycle(); 341 a = mContext.obtainStyledAttributes( 342 android.R.styleable.Theme); 343 mBackgroundDimAmount = a.getFloat(android.R.styleable.Theme_backgroundDimAmount, 0.5f); 344 345 mPreviewPopup = new PopupWindow(context); 346 if (previewLayout != 0) { 347 mPreviewText = (TextView) inflate.inflate(previewLayout, null); 348 mPreviewTextSizeLarge = (int) mPreviewText.getTextSize(); 349 mPreviewPopup.setContentView(mPreviewText); 350 mPreviewPopup.setBackgroundDrawable(null); 351 } else { 352 mShowPreview = false; 353 } 354 355 mPreviewPopup.setTouchable(false); 356 357 mPopupKeyboard = new PopupWindow(context); 358 mPopupKeyboard.setBackgroundDrawable(null); 359 360 mPopupParent = this; 361 362 mPaint = new Paint(); 363 mPaint.setAntiAlias(true); 364 mPaint.setTextSize(keyTextSize); 365 mPaint.setTextAlign(Align.CENTER); 366 mPaint.setAlpha(255); 367 368 mPadding = new Rect(0, 0, 0, 0); 369 mMiniKeyboardCache = new HashMap<Key,View>(); 370 mKeyBackground.getPadding(mPadding); 371 372 mSwipeThreshold = (int) (500 * getResources().getDisplayMetrics().density); 373 374 mDisambiguateSwipe = true; 375 376 resetMultiTap(); 377 initGestureDetector(); 378 } 379 initGestureDetector()380 private void initGestureDetector() { 381 mGestureDetector = new GestureDetector(getContext(), new GestureDetector.SimpleOnGestureListener() { 382 @Override 383 public boolean onFling(MotionEvent me1, MotionEvent me2, 384 float velocityX, float velocityY) { 385 if (mPossiblePoly) return false; 386 final float absX = Math.abs(velocityX); 387 final float absY = Math.abs(velocityY); 388 float deltaX = me2.getX() - me1.getX(); 389 float deltaY = me2.getY() - me1.getY(); 390 int travelX = getWidth() / 2; 391 int travelY = getHeight() / 2; 392 mSwipeTracker.computeCurrentVelocity(1000); 393 final float endingVelocityX = mSwipeTracker.getXVelocity(); 394 final float endingVelocityY = mSwipeTracker.getYVelocity(); 395 boolean sendDownKey = false; 396 if (velocityX > mSwipeThreshold && absY < absX && deltaX > travelX) { 397 if (mDisambiguateSwipe && endingVelocityX < velocityX / 4) { 398 sendDownKey = true; 399 } else { 400 swipeRight(); 401 return true; 402 } 403 } else if (velocityX < -mSwipeThreshold && absY < absX && deltaX < -travelX) { 404 if (mDisambiguateSwipe && endingVelocityX > velocityX / 4) { 405 sendDownKey = true; 406 } else { 407 swipeLeft(); 408 return true; 409 } 410 } else if (velocityY < -mSwipeThreshold && absX < absY && deltaY < -travelY) { 411 if (mDisambiguateSwipe && endingVelocityY > velocityY / 4) { 412 sendDownKey = true; 413 } else { 414 swipeUp(); 415 return true; 416 } 417 } else if (velocityY > mSwipeThreshold && absX < absY / 2 && deltaY > travelY) { 418 if (mDisambiguateSwipe && endingVelocityY < velocityY / 4) { 419 sendDownKey = true; 420 } else { 421 swipeDown(); 422 return true; 423 } 424 } 425 426 if (sendDownKey) { 427 detectAndSendKey(mDownKey, mStartX, mStartY, me1.getEventTime()); 428 } 429 return false; 430 } 431 }); 432 433 mGestureDetector.setIsLongpressEnabled(false); 434 } 435 436 /** 437 * Set the {@link OnKeyboardActionListener} object. 438 * @param listener The OnKeyboardActionListener to set. 439 */ setOnKeyboardActionListener(OnKeyboardActionListener listener)440 public void setOnKeyboardActionListener(OnKeyboardActionListener listener) { 441 mKeyboardActionListener = listener; 442 } 443 444 /** 445 * Returns the {@link OnKeyboardActionListener} object. 446 * @return the listener attached to this keyboard 447 */ getOnKeyboardActionListener()448 protected OnKeyboardActionListener getOnKeyboardActionListener() { 449 return mKeyboardActionListener; 450 } 451 452 /** 453 * Attaches a keyboard to this view. The keyboard can be switched at any time and the 454 * view will re-layout itself to accommodate the keyboard. 455 * @see Keyboard 456 * @see #getKeyboard() 457 * @param keyboard the keyboard to display in this view 458 */ setKeyboard(Keyboard keyboard)459 public void setKeyboard(Keyboard keyboard) { 460 if (!keyboard.equals(mKeyboard)) { 461 clearWindowInfo(); 462 } 463 int oldRepeatKeyCode = NOT_A_KEY; 464 if (mKeyboard != null) { 465 showPreview(NOT_A_KEY); 466 if ((mRepeatKeyIndex != NOT_A_KEY) && (mRepeatKeyIndex < mKeys.length)) { 467 oldRepeatKeyCode = mKeys[mRepeatKeyIndex].codes[0]; 468 } 469 } 470 removeMessages(); 471 mKeyboard = keyboard; 472 List<Key> keys = mKeyboard.getKeys(); 473 mKeys = keys.toArray(new Key[keys.size()]); 474 requestLayout(); 475 mKeyboardChanged = true; 476 invalidateAllKeys(); 477 computeProximityThreshold(keyboard); 478 mMiniKeyboardCache.clear(); 479 boolean abort = true; 480 if (oldRepeatKeyCode != NOT_A_KEY) { 481 int keyIndex = getKeyIndices(mStartX, mStartY, null); 482 if ((keyIndex != NOT_A_KEY) 483 && (keyIndex < mKeys.length) 484 && (oldRepeatKeyCode == mKeys[keyIndex].codes[0])) { 485 abort = false; 486 mRepeatKeyIndex = keyIndex; 487 } 488 } 489 if (abort) { 490 mHandler.removeMessages(MSG_REPEAT); 491 } 492 mAbortKey = abort; 493 } 494 495 /** 496 * Returns the current keyboard being displayed by this view. 497 * @return the currently attached keyboard 498 * @see #setKeyboard(Keyboard) 499 */ getKeyboard()500 public Keyboard getKeyboard() { 501 return mKeyboard; 502 } 503 504 /** 505 * Sets the state of the shift key of the keyboard, if any. 506 * @param shifted whether or not to enable the state of the shift key 507 * @return true if the shift key state changed, false if there was no change 508 * @see KeyboardView#isShifted() 509 */ setShifted(boolean shifted)510 public boolean setShifted(boolean shifted) { 511 if (mKeyboard != null) { 512 if (mKeyboard.setShifted(shifted)) { 513 invalidateAllKeys(); 514 return true; 515 } 516 } 517 return false; 518 } 519 520 /** 521 * Returns the state of the shift key of the keyboard, if any. 522 * @return true if the shift is in a pressed state, false otherwise. If there is 523 * no shift key on the keyboard or there is no keyboard attached, it returns false. 524 * @see KeyboardView#setShifted(boolean) 525 */ isShifted()526 public boolean isShifted() { 527 if (mKeyboard != null) { 528 return mKeyboard.isShifted(); 529 } 530 return false; 531 } 532 533 /** 534 * Enables or disables the key feedback popup. This is a popup that shows a magnified 535 * version of the depressed key. By default the preview is enabled. 536 * @param previewEnabled whether or not to enable the key feedback popup 537 * @see #isPreviewEnabled() 538 */ setPreviewEnabled(boolean previewEnabled)539 public void setPreviewEnabled(boolean previewEnabled) { 540 mShowPreview = previewEnabled; 541 } 542 543 /** 544 * Returns the enabled state of the key feedback popup. 545 * @return whether or not the key feedback popup is enabled 546 * @see #setPreviewEnabled(boolean) 547 */ isPreviewEnabled()548 public boolean isPreviewEnabled() { 549 return mShowPreview; 550 } 551 552 /** 553 * Returns the root parent has the enabled state of the key feedback popup. 554 * @return whether or not the key feedback popup is enabled 555 * @see #setPreviewEnabled(boolean) 556 */ isParentPreviewEnabled()557 public boolean isParentPreviewEnabled() { 558 if ((mPopupParent != null) && (mPopupParent != this) 559 && (mPopupParent instanceof KeyboardView)) { 560 return ((KeyboardView)mPopupParent).isParentPreviewEnabled(); 561 } else { 562 return mShowPreview; 563 } 564 } 565 setVerticalCorrection(int verticalOffset)566 public void setVerticalCorrection(int verticalOffset) { 567 568 } 569 570 /** 571 * Set View on the PopupParent. 572 * @param v The View to set. 573 */ setPopupParent(View v)574 public void setPopupParent(View v) { 575 mPopupParent = v; 576 } 577 578 /** 579 * Set parameters on the KeyboardOffset. 580 * @param x The value of KeyboardOffset. 581 * @param y The value of KeyboardOffset. 582 */ setPopupOffset(int x, int y)583 public void setPopupOffset(int x, int y) { 584 mMiniKeyboardOffsetX = x; 585 mMiniKeyboardOffsetY = y; 586 if (mPreviewPopup.isShowing()) { 587 mPreviewPopup.dismiss(); 588 } 589 } 590 591 /** 592 * When enabled, calls to {@link OnKeyboardActionListener#onKey} will include key 593 * codes for adjacent keys. When disabled, only the primary key code will be 594 * reported. 595 * @param enabled whether or not the proximity correction is enabled 596 */ setProximityCorrectionEnabled(boolean enabled)597 public void setProximityCorrectionEnabled(boolean enabled) { 598 mProximityCorrectOn = enabled; 599 } 600 601 /** 602 * Returns true if proximity correction is enabled. 603 */ isProximityCorrectionEnabled()604 public boolean isProximityCorrectionEnabled() { 605 return mProximityCorrectOn; 606 } 607 608 /** 609 * Popup keyboard close button clicked. 610 * @hide 611 */ onClick(View v)612 public void onClick(View v) { 613 dismissPopupKeyboard(); 614 } 615 adjustCase(CharSequence label)616 private CharSequence adjustCase(CharSequence label) { 617 if (mKeyboard.isShifted() && label != null && label.length() < 3 618 && Character.isLowerCase(label.charAt(0))) { 619 label = label.toString().toUpperCase(); 620 } 621 return label; 622 } 623 624 @Override onMeasure(int widthMeasureSpec, int heightMeasureSpec)625 public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 626 if (mKeyboard == null) { 627 setMeasuredDimension(mPaddingLeft + mPaddingRight, mPaddingTop + mPaddingBottom); 628 } else { 629 int width = mKeyboard.getMinWidth() + mPaddingLeft + mPaddingRight; 630 if (MeasureSpec.getSize(widthMeasureSpec) < width + 10) { 631 width = MeasureSpec.getSize(widthMeasureSpec); 632 } 633 setMeasuredDimension(width, mKeyboard.getHeight() + mPaddingTop + mPaddingBottom); 634 } 635 } 636 637 /** 638 * Compute the average distance between adjacent keys (horizontally and vertically) 639 * and square it to get the proximity threshold. We use a square here and in computing 640 * the touch distance from a key's center to avoid taking a square root. 641 * @param keyboard 642 */ computeProximityThreshold(Keyboard keyboard)643 private void computeProximityThreshold(Keyboard keyboard) { 644 if (keyboard == null) return; 645 final Key[] keys = mKeys; 646 if (keys == null) return; 647 int length = keys.length; 648 int dimensionSum = 0; 649 for (int i = 0; i < length; i++) { 650 Key key = keys[i]; 651 dimensionSum += Math.min(key.width, key.height) + key.gap; 652 } 653 if (dimensionSum < 0 || length == 0) return; 654 mProximityThreshold = (int) (dimensionSum * 1.4f / length); 655 mProximityThreshold *= mProximityThreshold; 656 } 657 658 @Override onSizeChanged(int w, int h, int oldw, int oldh)659 public void onSizeChanged(int w, int h, int oldw, int oldh) { 660 super.onSizeChanged(w, h, oldw, oldh); 661 mBuffer = null; 662 } 663 664 @Override onDraw(Canvas canvas)665 public void onDraw(Canvas canvas) { 666 super.onDraw(canvas); 667 if (mDrawPending || mBuffer == null || mKeyboardChanged) { 668 onBufferDraw(); 669 } 670 canvas.drawBitmap(mBuffer, 0, 0, null); 671 } 672 onBufferDraw()673 private void onBufferDraw() { 674 boolean isBufferNull = (mBuffer == null); 675 if (isBufferNull || mKeyboardChanged) { 676 if (isBufferNull || mKeyboardChanged && 677 (mBuffer.getWidth() != getWidth() || mBuffer.getHeight() != getHeight())) { 678 final int width = Math.max(1, getWidth()); 679 final int height = Math.max(1, getHeight()); 680 mBuffer = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); 681 mCanvas = new Canvas(mBuffer); 682 } 683 invalidateAllKeys(); 684 mKeyboardChanged = false; 685 } 686 final Canvas canvas = mCanvas; 687 canvas.clipRect(mDirtyRect, Op.REPLACE); 688 689 if (mKeyboard == null) return; 690 691 final Paint paint = mPaint; 692 final Rect clipRegion = mClipRegion; 693 final Rect padding = mPadding; 694 final int kbdPaddingLeft = mPaddingLeft; 695 final int kbdPaddingTop = mPaddingTop; 696 final Key[] keys = mKeys; 697 final Key invalidKey = mInvalidatedKey; 698 699 paint.setColor(mKeyTextColor); 700 boolean drawSingleKey = false; 701 if (invalidKey != null && canvas.getClipBounds(clipRegion)) { 702 if (invalidKey.x + kbdPaddingLeft - 1 <= clipRegion.left && 703 invalidKey.y + kbdPaddingTop - 1 <= clipRegion.top && 704 invalidKey.x + invalidKey.width + kbdPaddingLeft + 1 >= clipRegion.right && 705 invalidKey.y + invalidKey.height + kbdPaddingTop + 1 >= clipRegion.bottom) { 706 drawSingleKey = true; 707 } 708 } 709 canvas.drawColor(0x00000000, PorterDuff.Mode.CLEAR); 710 final int keyCount = keys.length; 711 for (int i = 0; i < keyCount; i++) { 712 final Key key = keys[i]; 713 if (drawSingleKey && invalidKey != key) { 714 continue; 715 } 716 717 paint.setColor(key.isSecondKey ? mKeyTextColor2nd : mKeyTextColor); 718 Drawable keyBackground = key.isSecondKey ? mKeyBackground2nd : mKeyBackground; 719 int[] drawableState = key.getCurrentDrawableState(); 720 keyBackground.setState(drawableState); 721 722 String label = key.label == null? null : adjustCase(key.label).toString(); 723 724 final Rect bounds = keyBackground.getBounds(); 725 if (key.width != bounds.right || 726 key.height != bounds.bottom) { 727 keyBackground.setBounds(0, 0, key.width, key.height); 728 } 729 canvas.translate(key.x + kbdPaddingLeft, key.y + kbdPaddingTop); 730 keyBackground.draw(canvas); 731 732 if (label != null) { 733 if (OpenWnn.isXLarge()) { 734 if (label.length() > 1 && key.codes.length < 2) { 735 paint.setTextSize(mLabelTextSize); 736 paint.setTypeface(Typeface.DEFAULT); 737 } else { 738 paint.setTextSize(mKeyTextSize); 739 paint.setTypeface(Typeface.DEFAULT_BOLD); 740 } 741 } else { 742 if (label.length() > 1 && key.codes.length < 2) { 743 paint.setTextSize(mLabelTextSize); 744 paint.setTypeface(Typeface.DEFAULT_BOLD); 745 } else { 746 paint.setTextSize(mKeyTextSize); 747 paint.setTypeface(Typeface.DEFAULT_BOLD); 748 } 749 } 750 paint.setShadowLayer(mShadowRadius, 0, 0, mShadowColor); 751 if (OpenWnn.isXLarge()) { 752 canvas.drawText(label, 753 (key.width - padding.left + 7 - padding.right) / 2 754 + padding.left, 755 (key.height - padding.top + 7 - padding.bottom) / 2 756 + (paint.getTextSize() - paint.descent()) / 2 + padding.top, 757 paint); 758 } else { 759 canvas.drawText(label, 760 (key.width - padding.left - padding.right) / 2 761 + padding.left, 762 (key.height - padding.top - padding.bottom) / 2 763 + (paint.getTextSize() - paint.descent()) / 2 + padding.top, 764 paint); 765 } 766 paint.setShadowLayer(0, 0, 0, 0); 767 } else if (key.icon != null) { 768 int drawableX; 769 int drawableY; 770 if (OpenWnn.isXLarge()) { 771 drawableX = (key.width - padding.left + 12 - padding.right 772 - key.icon.getIntrinsicWidth()) / 2 + padding.left; 773 drawableY = (key.height - padding.top + 9 - padding.bottom 774 - key.icon.getIntrinsicHeight()) / 2 + padding.top; 775 } else { 776 drawableX = (key.width - padding.left - padding.right 777 - key.icon.getIntrinsicWidth()) / 2 + padding.left; 778 drawableY = (key.height - padding.top - padding.bottom 779 - key.icon.getIntrinsicHeight()) / 2 + padding.top; 780 } 781 canvas.translate(drawableX, drawableY); 782 key.icon.setBounds(0, 0, 783 key.icon.getIntrinsicWidth(), key.icon.getIntrinsicHeight()); 784 key.icon.draw(canvas); 785 canvas.translate(-drawableX, -drawableY); 786 } 787 canvas.translate(-key.x - kbdPaddingLeft, -key.y - kbdPaddingTop); 788 } 789 mInvalidatedKey = null; 790 if (mMiniKeyboardOnScreen) { 791 paint.setColor((int) (mBackgroundDimAmount * 0xFF) << 24); 792 canvas.drawRect(0, 0, getWidth(), getHeight(), paint); 793 } 794 795 mDrawPending = false; 796 mDirtyRect.setEmpty(); 797 } 798 getKeyIndices(int x, int y, int[] allKeys)799 private int getKeyIndices(int x, int y, int[] allKeys) { 800 final Key[] keys = mKeys; 801 int primaryIndex = NOT_A_KEY; 802 int closestKey = NOT_A_KEY; 803 int closestKeyDist = mProximityThreshold + 1; 804 java.util.Arrays.fill(mDistances, Integer.MAX_VALUE); 805 int [] nearestKeyIndices = mKeyboard.getNearestKeys(x, y); 806 final int keyCount = nearestKeyIndices.length; 807 for (int i = 0; i < keyCount; i++) { 808 final Key key = keys[nearestKeyIndices[i]]; 809 int dist = 0; 810 boolean isInside = key.isInside(x,y); 811 if (isInside) { 812 primaryIndex = nearestKeyIndices[i]; 813 } 814 815 if (((mProximityCorrectOn 816 && (dist = key.squaredDistanceFrom(x, y)) < mProximityThreshold) 817 || isInside) 818 && key.codes[0] > 32) { 819 final int nCodes = key.codes.length; 820 if (dist < closestKeyDist) { 821 closestKeyDist = dist; 822 closestKey = nearestKeyIndices[i]; 823 } 824 825 if (allKeys == null) continue; 826 827 for (int j = 0; j < mDistances.length; j++) { 828 if (mDistances[j] > dist) { 829 System.arraycopy(mDistances, j, mDistances, j + nCodes, 830 mDistances.length - j - nCodes); 831 System.arraycopy(allKeys, j, allKeys, j + nCodes, 832 allKeys.length - j - nCodes); 833 for (int c = 0; c < nCodes; c++) { 834 allKeys[j + c] = key.codes[c]; 835 mDistances[j + c] = dist; 836 } 837 break; 838 } 839 } 840 } 841 } 842 if (primaryIndex == NOT_A_KEY) { 843 primaryIndex = closestKey; 844 } 845 return primaryIndex; 846 } 847 detectAndSendKey(int index, int x, int y, long eventTime)848 private void detectAndSendKey(int index, int x, int y, long eventTime) { 849 if (index != NOT_A_KEY && index < mKeys.length) { 850 final Key key = mKeys[index]; 851 if (key.text != null) { 852 mKeyboardActionListener.onText(key.text); 853 mKeyboardActionListener.onRelease(NOT_A_KEY); 854 } else { 855 int code = key.codes[0]; 856 int[] codes = new int[MAX_NEARBY_KEYS]; 857 Arrays.fill(codes, NOT_A_KEY); 858 getKeyIndices(x, y, codes); 859 if (mInMultiTap) { 860 if (mTapCount != -1) { 861 mKeyboardActionListener.onKey(Keyboard.KEYCODE_DELETE, KEY_DELETE); 862 } else { 863 mTapCount = 0; 864 } 865 code = key.codes[mTapCount]; 866 } 867 mKeyboardActionListener.onKey(code, codes); 868 mKeyboardActionListener.onRelease(code); 869 } 870 mLastSentIndex = index; 871 mLastTapTime = eventTime; 872 } 873 } 874 875 /** 876 * Handle multi-tap keys by producing the key label for the current multi-tap state. 877 */ getPreviewText(Key key)878 private CharSequence getPreviewText(Key key) { 879 if (mInMultiTap) { 880 mPreviewLabel.setLength(0); 881 mPreviewLabel.append((char) key.codes[mTapCount < 0 ? 0 : mTapCount]); 882 return adjustCase(mPreviewLabel); 883 } else { 884 return adjustCase(key.label); 885 } 886 } 887 showPreview(int keyIndex)888 private void showPreview(int keyIndex) { 889 int oldKeyIndex = mCurrentKeyIndex; 890 final PopupWindow previewPopup = mPreviewPopup; 891 892 mCurrentKeyIndex = keyIndex; 893 final Key[] keys = mKeys; 894 if (oldKeyIndex != mCurrentKeyIndex) { 895 if (oldKeyIndex != NOT_A_KEY && keys.length > oldKeyIndex) { 896 keys[oldKeyIndex].onReleased(mCurrentKeyIndex == NOT_A_KEY); 897 invalidateKey(oldKeyIndex); 898 } 899 if (mCurrentKeyIndex != NOT_A_KEY && keys.length > mCurrentKeyIndex) { 900 keys[mCurrentKeyIndex].onPressed(); 901 invalidateKey(mCurrentKeyIndex); 902 } 903 } 904 if (oldKeyIndex != mCurrentKeyIndex && mShowPreview && isParentPreviewEnabled()) { 905 mHandler.removeMessages(MSG_SHOW_PREVIEW); 906 if (previewPopup.isShowing()) { 907 if (keyIndex == NOT_A_KEY) { 908 mHandler.sendMessageDelayed(mHandler 909 .obtainMessage(MSG_REMOVE_PREVIEW), 910 DELAY_AFTER_PREVIEW); 911 } 912 } 913 if (keyIndex != NOT_A_KEY) { 914 if (previewPopup.isShowing() && mPreviewText.getVisibility() == VISIBLE) { 915 showKey(keyIndex); 916 } else { 917 mHandler.sendMessageDelayed( 918 mHandler.obtainMessage(MSG_SHOW_PREVIEW, keyIndex, 0), 919 DELAY_BEFORE_PREVIEW); 920 } 921 } 922 } 923 } 924 showKey(final int keyIndex)925 private void showKey(final int keyIndex) { 926 final PopupWindow previewPopup = mPreviewPopup; 927 final Key[] keys = mKeys; 928 if (keyIndex < 0 || keyIndex >= mKeys.length) return; 929 Key key = keys[keyIndex]; 930 931 mPreviewText.setBackgroundDrawable(getContext().getResources().getDrawable(R.drawable.keyboard_key_feedback)); 932 933 if (key.icon != null) { 934 mPreviewText.setCompoundDrawables(null, null, null, 935 key.iconPreview != null ? key.iconPreview : key.icon); 936 mPreviewText.setText(null); 937 mPreviewText.setPadding(5, 0, 5, 20); 938 } else { 939 mPreviewText.setCompoundDrawables(null, null, null, null); 940 mPreviewText.setText(getPreviewText(key)); 941 if (key.label.length() > 1 && key.codes.length < 2) { 942 mPreviewText.setTextSize(TypedValue.COMPLEX_UNIT_PX, mKeyTextSize); 943 mPreviewText.setTypeface(Typeface.DEFAULT_BOLD); 944 } else { 945 mPreviewText.setTextSize(TypedValue.COMPLEX_UNIT_PX, mPreviewTextSizeLarge); 946 mPreviewText.setTypeface(Typeface.DEFAULT); 947 } 948 mPreviewText.setPadding(0, 0, 0, 10); 949 } 950 mPreviewText.measure(MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED), 951 MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED)); 952 int popupWidth = Math.max(mPreviewText.getMeasuredWidth(), key.width 953 + mPreviewText.getPaddingLeft() + mPreviewText.getPaddingRight()); 954 final int popupHeight = mPreviewHeight; 955 LayoutParams lp = mPreviewText.getLayoutParams(); 956 if (lp != null) { 957 lp.width = popupWidth; 958 lp.height = popupHeight; 959 } 960 if (!mPreviewCentered) { 961 mPopupPreviewX = key.x - (Math.abs(popupWidth - key.width) / 2 ); 962 mPopupPreviewY = key.y - popupHeight + mPreviewOffset; 963 } else { 964 mPopupPreviewX = 160 - mPreviewText.getMeasuredWidth() / 2; 965 mPopupPreviewY = - mPreviewText.getMeasuredHeight(); 966 } 967 mPopupPreviewY = mPopupPreviewY + 20; 968 mHandler.removeMessages(MSG_REMOVE_PREVIEW); 969 if (mOffsetInWindow == null) { 970 mOffsetInWindow = new int[2]; 971 getLocationInWindow(mOffsetInWindow); 972 mOffsetInWindow[0] += mMiniKeyboardOffsetX; 973 mOffsetInWindow[1] += mMiniKeyboardOffsetY; 974 int[] mWindowLocation = new int[2]; 975 getLocationOnScreen(mWindowLocation); 976 mWindowY = mWindowLocation[1]; 977 } 978 mPreviewText.getBackground().setState( 979 key.popupResId != 0 ? LONG_PRESSABLE_STATE_SET : EMPTY_STATE_SET); 980 mPopupPreviewX += mOffsetInWindow[0]; 981 mPopupPreviewY += mOffsetInWindow[1]; 982 983 if (mPopupPreviewY + mWindowY < 0) { 984 if (key.x + key.width <= getWidth() / 2) { 985 mPopupPreviewX += (int) (key.width * 2.5); 986 } else { 987 mPopupPreviewX -= (int) (key.width * 2.5); 988 } 989 mPopupPreviewY += popupHeight; 990 } 991 992 if (previewPopup.isShowing()) { 993 previewPopup.update(mPopupPreviewX, mPopupPreviewY, 994 popupWidth, popupHeight); 995 } else { 996 previewPopup.setWidth(popupWidth); 997 previewPopup.setHeight(popupHeight); 998 previewPopup.showAtLocation(mPopupParent, Gravity.NO_GRAVITY, 999 mPopupPreviewX, mPopupPreviewY); 1000 } 1001 mPreviewText.setVisibility(VISIBLE); 1002 } 1003 1004 /** 1005 * Requests a redraw of the entire keyboard. Calling {@link #invalidate} is not sufficient 1006 * because the keyboard renders the keys to an off-screen buffer and an invalidate() only 1007 * draws the cached buffer. 1008 * @see #invalidateKey(int) 1009 */ invalidateAllKeys()1010 public void invalidateAllKeys() { 1011 mDirtyRect.union(0, 0, getWidth(), getHeight()); 1012 mDrawPending = true; 1013 invalidate(); 1014 } 1015 1016 /** 1017 * Invalidates a key so that it will be redrawn on the next repaint. Use this method if only 1018 * one key is changing it's content. Any changes that affect the position or size of the key 1019 * may not be honored. 1020 * @param keyIndex the index of the key in the attached {@link Keyboard}. 1021 * @see #invalidateAllKeys 1022 */ invalidateKey(int keyIndex)1023 public void invalidateKey(int keyIndex) { 1024 if (mKeys == null) return; 1025 if (keyIndex < 0 || keyIndex >= mKeys.length) { 1026 return; 1027 } 1028 final Key key = mKeys[keyIndex]; 1029 mInvalidatedKey = key; 1030 mDirtyRect.union(key.x + mPaddingLeft, key.y + mPaddingTop, 1031 key.x + key.width + mPaddingLeft, key.y + key.height + mPaddingTop); 1032 onBufferDraw(); 1033 invalidate(key.x + mPaddingLeft, key.y + mPaddingTop, 1034 key.x + key.width + mPaddingLeft, key.y + key.height + mPaddingTop); 1035 } 1036 openPopupIfRequired(MotionEvent me)1037 private boolean openPopupIfRequired(MotionEvent me) { 1038 if (mPopupLayout == 0) { 1039 return false; 1040 } 1041 if (mCurrentKey < 0 || mCurrentKey >= mKeys.length) { 1042 return false; 1043 } 1044 1045 Key popupKey = mKeys[mCurrentKey]; 1046 boolean result = onLongPress(popupKey); 1047 if (result) { 1048 mAbortKey = true; 1049 showPreview(NOT_A_KEY); 1050 } 1051 return result; 1052 } 1053 1054 /** 1055 * Called when a key is long pressed. By default this will open any popup keyboard associated 1056 * with this key through the attributes popupLayout and popupCharacters. 1057 * @param popupKey the key that was long pressed 1058 * @return true if the long press is handled, false otherwise. Subclasses should call the 1059 * method on the base class if the subclass doesn't wish to handle the call. 1060 */ onLongPress(Key popupKey)1061 protected boolean onLongPress(Key popupKey) { 1062 if (mKeyboardActionListener.onLongPress(popupKey)) { 1063 return true; 1064 } 1065 int popupKeyboardId = popupKey.popupResId; 1066 if (popupKeyboardId != 0) { 1067 mMiniKeyboardContainer = mMiniKeyboardCache.get(popupKey); 1068 if (mMiniKeyboardContainer == null) { 1069 LayoutInflater inflater = (LayoutInflater) getContext().getSystemService( 1070 Context.LAYOUT_INFLATER_SERVICE); 1071 mMiniKeyboardContainer = inflater.inflate(mPopupLayout, null); 1072 mMiniKeyboard = (KeyboardView) mMiniKeyboardContainer.findViewById(R.id.keyboardView); 1073 View closeButton = mMiniKeyboardContainer.findViewById(R.id.closeButton); 1074 if (closeButton != null) closeButton.setOnClickListener(this); 1075 mMiniKeyboard.setOnKeyboardActionListener(new OnKeyboardActionListener() { 1076 public void onKey(int primaryCode, int[] keyCodes) { 1077 mKeyboardActionListener.onKey(primaryCode, keyCodes); 1078 dismissPopupKeyboard(); 1079 } 1080 1081 public void onText(CharSequence text) { 1082 mKeyboardActionListener.onText(text); 1083 dismissPopupKeyboard(); 1084 } 1085 1086 public void swipeLeft() { } 1087 public void swipeRight() { } 1088 public void swipeUp() { } 1089 public void swipeDown() { } 1090 public void onPress(int primaryCode) { 1091 mKeyboardActionListener.onPress(primaryCode); 1092 } 1093 public void onRelease(int primaryCode) { 1094 mKeyboardActionListener.onRelease(primaryCode); 1095 } 1096 public boolean onLongPress(Keyboard.Key key) { 1097 return false; 1098 } 1099 }); 1100 Keyboard keyboard; 1101 if (popupKey.popupCharacters != null) { 1102 keyboard = new Keyboard(getContext(), popupKeyboardId, 1103 popupKey.popupCharacters, -1, getPaddingLeft() + getPaddingRight()); 1104 } else { 1105 keyboard = new Keyboard(getContext(), popupKeyboardId); 1106 } 1107 mMiniKeyboard.setKeyboard(keyboard); 1108 mMiniKeyboard.setPopupParent(this); 1109 mMiniKeyboardContainer.measure( 1110 MeasureSpec.makeMeasureSpec(getWidth(), MeasureSpec.AT_MOST), 1111 MeasureSpec.makeMeasureSpec(getHeight(), MeasureSpec.AT_MOST)); 1112 1113 mMiniKeyboardCache.put(popupKey, mMiniKeyboardContainer); 1114 } else { 1115 mMiniKeyboard = (KeyboardView) mMiniKeyboardContainer.findViewById(R.id.keyboardView); 1116 } 1117 if (mWindowOffset == null) { 1118 mWindowOffset = new int[2]; 1119 getLocationInWindow(mWindowOffset); 1120 } 1121 mPopupX = popupKey.x + mPaddingLeft; 1122 mPopupY = popupKey.y + mPaddingTop; 1123 mPopupX = mPopupX + popupKey.width - mMiniKeyboardContainer.getMeasuredWidth(); 1124 mPopupY = mPopupY - mMiniKeyboardContainer.getMeasuredHeight(); 1125 final int x = mPopupX + mMiniKeyboardContainer.getPaddingRight() + mWindowOffset[0]; 1126 final int y = mPopupY + mMiniKeyboardContainer.getPaddingBottom() + mWindowOffset[1]; 1127 mMiniKeyboard.setPopupOffset(x < 0 ? 0 : x, y); 1128 mMiniKeyboard.setShifted(isShifted()); 1129 mPopupKeyboard.setContentView(mMiniKeyboardContainer); 1130 mPopupKeyboard.setWidth(mMiniKeyboardContainer.getMeasuredWidth()); 1131 mPopupKeyboard.setHeight(mMiniKeyboardContainer.getMeasuredHeight()); 1132 mPopupKeyboard.showAtLocation(this, Gravity.NO_GRAVITY, x, y); 1133 mMiniKeyboardOnScreen = true; 1134 invalidateAllKeys(); 1135 return true; 1136 } 1137 return false; 1138 } 1139 1140 private long mOldEventTime; 1141 private boolean mUsedVelocity; 1142 1143 @Override 1144 public boolean onTouchEvent(MotionEvent me) { 1145 final int pointerCount = me.getPointerCount(); 1146 final int action = me.getAction(); 1147 boolean result = false; 1148 final long now = me.getEventTime(); 1149 final boolean isPointerCountOne = (pointerCount == 1); 1150 1151 if (pointerCount != mOldPointerCount) { 1152 if (isPointerCountOne) { 1153 MotionEvent down = MotionEvent.obtain(now, now, MotionEvent.ACTION_DOWN, 1154 me.getX(), me.getY(), me.getMetaState()); 1155 result = onModifiedTouchEvent(down, false); 1156 down.recycle(); 1157 if (action == MotionEvent.ACTION_UP) { 1158 result = onModifiedTouchEvent(me, true); 1159 } 1160 } else { 1161 MotionEvent up = MotionEvent.obtain(now, now, MotionEvent.ACTION_UP, 1162 mOldPointerX, mOldPointerY, me.getMetaState()); 1163 result = onModifiedTouchEvent(up, true); 1164 up.recycle(); 1165 } 1166 } else { 1167 if (isPointerCountOne) { 1168 result = onModifiedTouchEvent(me, false); 1169 mOldPointerX = me.getX(); 1170 mOldPointerY = me.getY(); 1171 } else { 1172 result = true; 1173 } 1174 } 1175 mOldPointerCount = pointerCount; 1176 1177 return result; 1178 } 1179 1180 private boolean onModifiedTouchEvent(MotionEvent me, boolean possiblePoly) { 1181 int touchX = (int) me.getX() - mPaddingLeft; 1182 int touchY = (int) me.getY() + mVerticalCorrection - mPaddingTop; 1183 final int action = me.getAction(); 1184 final long eventTime = me.getEventTime(); 1185 mOldEventTime = eventTime; 1186 int keyIndex = getKeyIndices(touchX, touchY, null); 1187 mPossiblePoly = possiblePoly; 1188 1189 if (action == MotionEvent.ACTION_DOWN) mSwipeTracker.clear(); 1190 mSwipeTracker.addMovement(me); 1191 1192 if (mAbortKey 1193 && action != MotionEvent.ACTION_DOWN && action != MotionEvent.ACTION_CANCEL) { 1194 return true; 1195 } 1196 1197 if (mGestureDetector.onTouchEvent(me)) { 1198 showPreview(NOT_A_KEY); 1199 mHandler.removeMessages(MSG_REPEAT); 1200 mHandler.removeMessages(MSG_LONGPRESS); 1201 return true; 1202 } 1203 1204 if (mMiniKeyboardOnScreen && action != MotionEvent.ACTION_CANCEL) { 1205 return true; 1206 } 1207 1208 switch (action) { 1209 case MotionEvent.ACTION_DOWN: 1210 mAbortKey = false; 1211 mStartX = touchX; 1212 mStartY = touchY; 1213 mLastCodeX = touchX; 1214 mLastCodeY = touchY; 1215 mLastKeyTime = 0; 1216 mCurrentKeyTime = 0; 1217 mLastKey = NOT_A_KEY; 1218 mCurrentKey = keyIndex; 1219 mDownKey = keyIndex; 1220 mDownTime = me.getEventTime(); 1221 mLastMoveTime = mDownTime; 1222 checkMultiTap(eventTime, keyIndex); 1223 mKeyboardActionListener.onPress(keyIndex != NOT_A_KEY ? 1224 mKeys[keyIndex].codes[0] : 0); 1225 if (mCurrentKey >= 0 && mKeys[mCurrentKey].repeatable) { 1226 mRepeatKeyIndex = mCurrentKey; 1227 Message msg = mHandler.obtainMessage(MSG_REPEAT); 1228 mHandler.sendMessageDelayed(msg, REPEAT_START_DELAY); 1229 repeatKey(); 1230 if (mAbortKey) { 1231 mRepeatKeyIndex = NOT_A_KEY; 1232 break; 1233 } 1234 } 1235 if (mCurrentKey != NOT_A_KEY) { 1236 Message msg = mHandler.obtainMessage(MSG_LONGPRESS, me); 1237 mHandler.sendMessageDelayed(msg, LONGPRESS_TIMEOUT); 1238 } 1239 showPreview(keyIndex); 1240 break; 1241 1242 case MotionEvent.ACTION_MOVE: 1243 boolean continueLongPress = false; 1244 if (keyIndex != NOT_A_KEY) { 1245 if (mCurrentKey == NOT_A_KEY) { 1246 mCurrentKey = keyIndex; 1247 mCurrentKeyTime = eventTime - mDownTime; 1248 } else { 1249 if (keyIndex == mCurrentKey) { 1250 mCurrentKeyTime += eventTime - mLastMoveTime; 1251 continueLongPress = true; 1252 } else if (mRepeatKeyIndex == NOT_A_KEY) { 1253 resetMultiTap(); 1254 mLastKey = mCurrentKey; 1255 mLastCodeX = mLastX; 1256 mLastCodeY = mLastY; 1257 mLastKeyTime = 1258 mCurrentKeyTime + eventTime - mLastMoveTime; 1259 mCurrentKey = keyIndex; 1260 mCurrentKeyTime = 0; 1261 } 1262 } 1263 } 1264 if (!continueLongPress) { 1265 mHandler.removeMessages(MSG_LONGPRESS); 1266 if (keyIndex != NOT_A_KEY) { 1267 Message msg = mHandler.obtainMessage(MSG_LONGPRESS, me); 1268 mHandler.sendMessageDelayed(msg, LONGPRESS_TIMEOUT); 1269 } 1270 } 1271 showPreview(mCurrentKey); 1272 mLastMoveTime = eventTime; 1273 break; 1274 1275 case MotionEvent.ACTION_UP: 1276 removeMessages(); 1277 if (keyIndex == mCurrentKey) { 1278 mCurrentKeyTime += eventTime - mLastMoveTime; 1279 } else { 1280 resetMultiTap(); 1281 mLastKey = mCurrentKey; 1282 mLastKeyTime = mCurrentKeyTime + eventTime - mLastMoveTime; 1283 mCurrentKey = keyIndex; 1284 mCurrentKeyTime = 0; 1285 } 1286 if (mCurrentKeyTime < mLastKeyTime && mCurrentKeyTime < DEBOUNCE_TIME 1287 && mLastKey != NOT_A_KEY) { 1288 mCurrentKey = mLastKey; 1289 touchX = mLastCodeX; 1290 touchY = mLastCodeY; 1291 } 1292 showPreview(NOT_A_KEY); 1293 Arrays.fill(mKeyIndices, NOT_A_KEY); 1294 if (mRepeatKeyIndex == NOT_A_KEY && !mMiniKeyboardOnScreen && !mAbortKey) { 1295 detectAndSendKey(mCurrentKey, touchX, touchY, eventTime); 1296 } 1297 invalidateKey(keyIndex); 1298 mRepeatKeyIndex = NOT_A_KEY; 1299 break; 1300 case MotionEvent.ACTION_CANCEL: 1301 removeMessages(); 1302 dismissPopupKeyboard(); 1303 mAbortKey = true; 1304 showPreview(NOT_A_KEY); 1305 invalidateKey(mCurrentKey); 1306 break; 1307 } 1308 mLastX = touchX; 1309 mLastY = touchY; 1310 return true; 1311 } 1312 repeatKey()1313 private boolean repeatKey() { 1314 Key key = mKeys[mRepeatKeyIndex]; 1315 detectAndSendKey(mCurrentKey, key.x, key.y, mLastTapTime); 1316 return true; 1317 } 1318 swipeRight()1319 protected void swipeRight() { 1320 mKeyboardActionListener.swipeRight(); 1321 } 1322 swipeLeft()1323 protected void swipeLeft() { 1324 mKeyboardActionListener.swipeLeft(); 1325 } 1326 swipeUp()1327 protected void swipeUp() { 1328 mKeyboardActionListener.swipeUp(); 1329 } 1330 swipeDown()1331 protected void swipeDown() { 1332 mKeyboardActionListener.swipeDown(); 1333 } 1334 closing()1335 public void closing() { 1336 if (mPreviewPopup.isShowing()) { 1337 mPreviewPopup.dismiss(); 1338 } 1339 removeMessages(); 1340 1341 dismissPopupKeyboard(); 1342 mBuffer = null; 1343 mCanvas = null; 1344 mMiniKeyboardCache.clear(); 1345 } 1346 removeMessages()1347 private void removeMessages() { 1348 mHandler.removeMessages(MSG_REPEAT); 1349 mHandler.removeMessages(MSG_LONGPRESS); 1350 mHandler.removeMessages(MSG_SHOW_PREVIEW); 1351 } 1352 1353 @Override onDetachedFromWindow()1354 public void onDetachedFromWindow() { 1355 super.onDetachedFromWindow(); 1356 closing(); 1357 } 1358 dismissPopupKeyboard()1359 private void dismissPopupKeyboard() { 1360 if (mPopupKeyboard.isShowing()) { 1361 mPopupKeyboard.dismiss(); 1362 mMiniKeyboardOnScreen = false; 1363 invalidateAllKeys(); 1364 } 1365 } 1366 handleBack()1367 public boolean handleBack() { 1368 if (mPopupKeyboard.isShowing()) { 1369 dismissPopupKeyboard(); 1370 return true; 1371 } 1372 return false; 1373 } 1374 resetMultiTap()1375 private void resetMultiTap() { 1376 mLastSentIndex = NOT_A_KEY; 1377 mTapCount = 0; 1378 mLastTapTime = -1; 1379 mInMultiTap = false; 1380 } 1381 checkMultiTap(long eventTime, int keyIndex)1382 private void checkMultiTap(long eventTime, int keyIndex) { 1383 if (keyIndex == NOT_A_KEY) return; 1384 Key key = mKeys[keyIndex]; 1385 if (key.codes.length > 1) { 1386 mInMultiTap = true; 1387 if (eventTime < mLastTapTime + MULTITAP_INTERVAL 1388 && keyIndex == mLastSentIndex) { 1389 mTapCount = (mTapCount + 1) % key.codes.length; 1390 return; 1391 } else { 1392 mTapCount = -1; 1393 return; 1394 } 1395 } 1396 if (eventTime > mLastTapTime + MULTITAP_INTERVAL || keyIndex != mLastSentIndex) { 1397 resetMultiTap(); 1398 } 1399 } 1400 1401 private static class SwipeTracker { 1402 1403 static final int NUM_PAST = 4; 1404 static final int LONGEST_PAST_TIME = 200; 1405 1406 final float mPastX[] = new float[NUM_PAST]; 1407 final float mPastY[] = new float[NUM_PAST]; 1408 final long mPastTime[] = new long[NUM_PAST]; 1409 1410 float mYVelocity; 1411 float mXVelocity; 1412 clear()1413 public void clear() { 1414 mPastTime[0] = 0; 1415 } 1416 addMovement(MotionEvent ev)1417 public void addMovement(MotionEvent ev) { 1418 long time = ev.getEventTime(); 1419 final int N = ev.getHistorySize(); 1420 for (int i=0; i<N; i++) { 1421 addPoint(ev.getHistoricalX(i), ev.getHistoricalY(i), 1422 ev.getHistoricalEventTime(i)); 1423 } 1424 addPoint(ev.getX(), ev.getY(), time); 1425 } 1426 addPoint(float x, float y, long time)1427 private void addPoint(float x, float y, long time) { 1428 int drop = -1; 1429 int i; 1430 final long[] pastTime = mPastTime; 1431 for (i=0; i<NUM_PAST; i++) { 1432 if (pastTime[i] == 0) { 1433 break; 1434 } else if (pastTime[i] < time-LONGEST_PAST_TIME) { 1435 drop = i; 1436 } 1437 } 1438 if (i == NUM_PAST && drop < 0) { 1439 drop = 0; 1440 } 1441 if (drop == i) drop--; 1442 final float[] pastX = mPastX; 1443 final float[] pastY = mPastY; 1444 if (drop >= 0) { 1445 final int start = drop+1; 1446 final int count = NUM_PAST-drop-1; 1447 System.arraycopy(pastX, start, pastX, 0, count); 1448 System.arraycopy(pastY, start, pastY, 0, count); 1449 System.arraycopy(pastTime, start, pastTime, 0, count); 1450 i -= (drop+1); 1451 } 1452 pastX[i] = x; 1453 pastY[i] = y; 1454 pastTime[i] = time; 1455 i++; 1456 if (i < NUM_PAST) { 1457 pastTime[i] = 0; 1458 } 1459 } 1460 computeCurrentVelocity(int units)1461 public void computeCurrentVelocity(int units) { 1462 computeCurrentVelocity(units, Float.MAX_VALUE); 1463 } 1464 computeCurrentVelocity(int units, float maxVelocity)1465 public void computeCurrentVelocity(int units, float maxVelocity) { 1466 final float[] pastX = mPastX; 1467 final float[] pastY = mPastY; 1468 final long[] pastTime = mPastTime; 1469 1470 final float oldestX = pastX[0]; 1471 final float oldestY = pastY[0]; 1472 final long oldestTime = pastTime[0]; 1473 float accumX = 0; 1474 float accumY = 0; 1475 int N=0; 1476 while (N < NUM_PAST) { 1477 if (pastTime[N] == 0) { 1478 break; 1479 } 1480 N++; 1481 } 1482 1483 for (int i=1; i < N; i++) { 1484 final int dur = (int)(pastTime[i] - oldestTime); 1485 if (dur == 0) continue; 1486 float dist = pastX[i] - oldestX; 1487 float vel = (dist/dur) * units; 1488 if (accumX == 0) accumX = vel; 1489 else accumX = (accumX + vel) * .5f; 1490 1491 dist = pastY[i] - oldestY; 1492 vel = (dist/dur) * units; 1493 if (accumY == 0) accumY = vel; 1494 else accumY = (accumY + vel) * .5f; 1495 } 1496 mXVelocity = accumX < 0.0f ? Math.max(accumX, -maxVelocity) 1497 : Math.min(accumX, maxVelocity); 1498 mYVelocity = accumY < 0.0f ? Math.max(accumY, -maxVelocity) 1499 : Math.min(accumY, maxVelocity); 1500 } 1501 1502 public float getXVelocity() { 1503 return mXVelocity; 1504 } 1505 1506 public float getYVelocity() { 1507 return mYVelocity; 1508 } 1509 } 1510 1511 /** 1512 * Clear window info. 1513 */ 1514 public void clearWindowInfo() { 1515 mOffsetInWindow = null; 1516 } 1517 } 1518