1 /* 2 * Copyright (C) 2006 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 android.text.method; 18 19 import android.text.Editable; 20 import android.text.NoCopySpan; 21 import android.text.Spannable; 22 import android.text.Spanned; 23 import android.view.KeyCharacterMap; 24 import android.view.KeyEvent; 25 import android.view.View; 26 27 /** 28 * This base class encapsulates the behavior for tracking the state of 29 * meta keys such as SHIFT, ALT and SYM as well as the pseudo-meta state of selecting text. 30 * <p> 31 * Key listeners that care about meta state should inherit from this class; 32 * you should not instantiate this class directly in a client. 33 * </p><p> 34 * This class provides two mechanisms for tracking meta state that can be used 35 * together or independently. 36 * </p> 37 * <ul> 38 * <li>Methods such as {@link #handleKeyDown(long, int, KeyEvent)} and 39 * {@link #getMetaState(long)} operate on a meta key state bit mask.</li> 40 * <li>Methods such as {@link #onKeyDown(View, Editable, int, KeyEvent)} and 41 * {@link #getMetaState(CharSequence, int)} operate on meta key state flags stored 42 * as spans in an {@link Editable} text buffer. The spans only describe the current 43 * meta key state of the text editor; they do not carry any positional information.</li> 44 * </ul> 45 * <p> 46 * The behavior of this class varies according to the keyboard capabilities 47 * described by the {@link KeyCharacterMap} of the keyboard device such as 48 * the {@link KeyCharacterMap#getModifierBehavior() key modifier behavior}. 49 * </p><p> 50 * {@link MetaKeyKeyListener} implements chorded and toggled key modifiers. 51 * When key modifiers are toggled into a latched or locked state, the state 52 * of the modifier is stored in the {@link Editable} text buffer or in a 53 * meta state integer managed by the client. These latched or locked modifiers 54 * should be considered to be held <b>in addition to</b> those that the 55 * keyboard already reported as being pressed in {@link KeyEvent#getMetaState()}. 56 * In other words, the {@link MetaKeyKeyListener} augments the meta state 57 * provided by the keyboard; it does not replace it. This distinction is important 58 * to ensure that meta keys not handled by {@link MetaKeyKeyListener} such as 59 * {@link KeyEvent#KEYCODE_CAPS_LOCK} or {@link KeyEvent#KEYCODE_NUM_LOCK} are 60 * taken into consideration. 61 * </p><p> 62 * To ensure correct meta key behavior, the following pattern should be used 63 * when mapping key codes to characters: 64 * </p> 65 * <code> 66 * private char getUnicodeChar(TextKeyListener listener, KeyEvent event, Editable textBuffer) { 67 * // Use the combined meta states from the event and the key listener. 68 * int metaState = event.getMetaState() | listener.getMetaState(textBuffer); 69 * return event.getUnicodeChar(metaState); 70 * } 71 * </code> 72 */ 73 public abstract class MetaKeyKeyListener { 74 /** 75 * Flag that indicates that the SHIFT key is on. 76 * Value equals {@link KeyEvent#META_SHIFT_ON}. 77 */ 78 public static final int META_SHIFT_ON = KeyEvent.META_SHIFT_ON; 79 /** 80 * Flag that indicates that the ALT key is on. 81 * Value equals {@link KeyEvent#META_ALT_ON}. 82 */ 83 public static final int META_ALT_ON = KeyEvent.META_ALT_ON; 84 /** 85 * Flag that indicates that the SYM key is on. 86 * Value equals {@link KeyEvent#META_SYM_ON}. 87 */ 88 public static final int META_SYM_ON = KeyEvent.META_SYM_ON; 89 90 /** 91 * Flag that indicates that the SHIFT key is locked in CAPS mode. 92 */ 93 public static final int META_CAP_LOCKED = KeyEvent.META_CAP_LOCKED; 94 /** 95 * Flag that indicates that the ALT key is locked. 96 */ 97 public static final int META_ALT_LOCKED = KeyEvent.META_ALT_LOCKED; 98 /** 99 * Flag that indicates that the SYM key is locked. 100 */ 101 public static final int META_SYM_LOCKED = KeyEvent.META_SYM_LOCKED; 102 103 /** 104 * @hide pending API review 105 */ 106 public static final int META_SELECTING = KeyEvent.META_SELECTING; 107 108 // These bits are privately used by the meta key key listener. 109 // They are deliberately assigned values outside of the representable range of an 'int' 110 // so as not to conflict with any meta key states publicly defined by KeyEvent. 111 private static final long META_CAP_USED = 1L << 32; 112 private static final long META_ALT_USED = 1L << 33; 113 private static final long META_SYM_USED = 1L << 34; 114 115 private static final long META_CAP_PRESSED = 1L << 40; 116 private static final long META_ALT_PRESSED = 1L << 41; 117 private static final long META_SYM_PRESSED = 1L << 42; 118 119 private static final long META_CAP_RELEASED = 1L << 48; 120 private static final long META_ALT_RELEASED = 1L << 49; 121 private static final long META_SYM_RELEASED = 1L << 50; 122 123 private static final long META_SHIFT_MASK = META_SHIFT_ON 124 | META_CAP_LOCKED | META_CAP_USED 125 | META_CAP_PRESSED | META_CAP_RELEASED; 126 private static final long META_ALT_MASK = META_ALT_ON 127 | META_ALT_LOCKED | META_ALT_USED 128 | META_ALT_PRESSED | META_ALT_RELEASED; 129 private static final long META_SYM_MASK = META_SYM_ON 130 | META_SYM_LOCKED | META_SYM_USED 131 | META_SYM_PRESSED | META_SYM_RELEASED; 132 133 private static final Object CAP = new NoCopySpan.Concrete(); 134 private static final Object ALT = new NoCopySpan.Concrete(); 135 private static final Object SYM = new NoCopySpan.Concrete(); 136 private static final Object SELECTING = new NoCopySpan.Concrete(); 137 138 private static final int PRESSED_RETURN_VALUE = 1; 139 private static final int LOCKED_RETURN_VALUE = 2; 140 141 /** 142 * Resets all meta state to inactive. 143 */ resetMetaState(Spannable text)144 public static void resetMetaState(Spannable text) { 145 text.removeSpan(CAP); 146 text.removeSpan(ALT); 147 text.removeSpan(SYM); 148 text.removeSpan(SELECTING); 149 } 150 151 /** 152 * Gets the state of the meta keys. 153 * 154 * @param text the buffer in which the meta key would have been pressed. 155 * 156 * @return an integer in which each bit set to one represents a pressed 157 * or locked meta key. 158 */ getMetaState(CharSequence text)159 public static final int getMetaState(CharSequence text) { 160 return getActive(text, CAP, META_SHIFT_ON, META_CAP_LOCKED) | 161 getActive(text, ALT, META_ALT_ON, META_ALT_LOCKED) | 162 getActive(text, SYM, META_SYM_ON, META_SYM_LOCKED) | 163 getActive(text, SELECTING, META_SELECTING, META_SELECTING); 164 } 165 166 /** 167 * Gets the state of the meta keys for a specific key event. 168 * 169 * For input devices that use toggled key modifiers, the `toggled' state 170 * is stored into the text buffer. This method retrieves the meta state 171 * for this event, accounting for the stored state. If the event has been 172 * created by a device that does not support toggled key modifiers, like 173 * a virtual device for example, the stored state is ignored. 174 * 175 * @param text the buffer in which the meta key would have been pressed. 176 * @param event the event for which to evaluate the meta state. 177 * @return an integer in which each bit set to one represents a pressed 178 * or locked meta key. 179 */ getMetaState(final CharSequence text, final KeyEvent event)180 public static final int getMetaState(final CharSequence text, final KeyEvent event) { 181 int metaState = event.getMetaState(); 182 if (event.getKeyCharacterMap().getModifierBehavior() 183 == KeyCharacterMap.MODIFIER_BEHAVIOR_CHORDED_OR_TOGGLED) { 184 metaState |= getMetaState(text); 185 } 186 return metaState; 187 } 188 189 // As META_SELECTING is @hide we should not mention it in public comments, hence the 190 // omission in @param meta 191 /** 192 * Gets the state of a particular meta key. 193 * 194 * @param meta META_SHIFT_ON, META_ALT_ON, META_SYM_ON 195 * @param text the buffer in which the meta key would have been pressed. 196 * 197 * @return 0 if inactive, 1 if active, 2 if locked. 198 */ getMetaState(CharSequence text, int meta)199 public static final int getMetaState(CharSequence text, int meta) { 200 switch (meta) { 201 case META_SHIFT_ON: 202 return getActive(text, CAP, PRESSED_RETURN_VALUE, LOCKED_RETURN_VALUE); 203 204 case META_ALT_ON: 205 return getActive(text, ALT, PRESSED_RETURN_VALUE, LOCKED_RETURN_VALUE); 206 207 case META_SYM_ON: 208 return getActive(text, SYM, PRESSED_RETURN_VALUE, LOCKED_RETURN_VALUE); 209 210 case META_SELECTING: 211 return getActive(text, SELECTING, PRESSED_RETURN_VALUE, LOCKED_RETURN_VALUE); 212 213 default: 214 return 0; 215 } 216 } 217 218 /** 219 * Gets the state of a particular meta key to use with a particular key event. 220 * 221 * If the key event has been created by a device that does not support toggled 222 * key modifiers, like a virtual keyboard for example, only the meta state in 223 * the key event is considered. 224 * 225 * @param meta META_SHIFT_ON, META_ALT_ON, META_SYM_ON 226 * @param text the buffer in which the meta key would have been pressed. 227 * @param event the event for which to evaluate the meta state. 228 * @return 0 if inactive, 1 if active, 2 if locked. 229 */ getMetaState(final CharSequence text, final int meta, final KeyEvent event)230 public static final int getMetaState(final CharSequence text, final int meta, 231 final KeyEvent event) { 232 int metaState = event.getMetaState(); 233 if (event.getKeyCharacterMap().getModifierBehavior() 234 == KeyCharacterMap.MODIFIER_BEHAVIOR_CHORDED_OR_TOGGLED) { 235 metaState |= getMetaState(text); 236 } 237 if (META_SELECTING == meta) { 238 // #getMetaState(long, int) does not support META_SELECTING, but we want the same 239 // behavior as #getMetaState(CharSequence, int) so we need to do it here 240 if ((metaState & META_SELECTING) != 0) { 241 // META_SELECTING is only ever set to PRESSED and can't be LOCKED, so return 1 242 return 1; 243 } 244 return 0; 245 } 246 return getMetaState(metaState, meta); 247 } 248 getActive(CharSequence text, Object meta, int on, int lock)249 private static int getActive(CharSequence text, Object meta, 250 int on, int lock) { 251 if (!(text instanceof Spanned)) { 252 return 0; 253 } 254 255 Spanned sp = (Spanned) text; 256 int flag = sp.getSpanFlags(meta); 257 258 if (flag == LOCKED) { 259 return lock; 260 } else if (flag != 0) { 261 return on; 262 } else { 263 return 0; 264 } 265 } 266 267 /** 268 * Call this method after you handle a keypress so that the meta 269 * state will be reset to unshifted (if it is not still down) 270 * or primed to be reset to unshifted (once it is released). 271 */ adjustMetaAfterKeypress(Spannable content)272 public static void adjustMetaAfterKeypress(Spannable content) { 273 adjust(content, CAP); 274 adjust(content, ALT); 275 adjust(content, SYM); 276 } 277 278 /** 279 * Returns true if this object is one that this class would use to 280 * keep track of any meta state in the specified text. 281 */ isMetaTracker(CharSequence text, Object what)282 public static boolean isMetaTracker(CharSequence text, Object what) { 283 return what == CAP || what == ALT || what == SYM || 284 what == SELECTING; 285 } 286 287 /** 288 * Returns true if this object is one that this class would use to 289 * keep track of the selecting meta state in the specified text. 290 */ isSelectingMetaTracker(CharSequence text, Object what)291 public static boolean isSelectingMetaTracker(CharSequence text, Object what) { 292 return what == SELECTING; 293 } 294 adjust(Spannable content, Object what)295 private static void adjust(Spannable content, Object what) { 296 int current = content.getSpanFlags(what); 297 298 if (current == PRESSED) 299 content.setSpan(what, 0, 0, USED); 300 else if (current == RELEASED) 301 content.removeSpan(what); 302 } 303 304 /** 305 * Call this if you are a method that ignores the locked meta state 306 * (arrow keys, for example) and you handle a key. 307 */ resetLockedMeta(Spannable content)308 protected static void resetLockedMeta(Spannable content) { 309 resetLock(content, CAP); 310 resetLock(content, ALT); 311 resetLock(content, SYM); 312 resetLock(content, SELECTING); 313 } 314 resetLock(Spannable content, Object what)315 private static void resetLock(Spannable content, Object what) { 316 int current = content.getSpanFlags(what); 317 318 if (current == LOCKED) 319 content.removeSpan(what); 320 } 321 322 /** 323 * Handles presses of the meta keys. 324 */ onKeyDown(View view, Editable content, int keyCode, KeyEvent event)325 public boolean onKeyDown(View view, Editable content, int keyCode, KeyEvent event) { 326 if (keyCode == KeyEvent.KEYCODE_SHIFT_LEFT || keyCode == KeyEvent.KEYCODE_SHIFT_RIGHT) { 327 press(content, CAP); 328 return true; 329 } 330 331 if (keyCode == KeyEvent.KEYCODE_ALT_LEFT || keyCode == KeyEvent.KEYCODE_ALT_RIGHT 332 || keyCode == KeyEvent.KEYCODE_NUM) { 333 press(content, ALT); 334 return true; 335 } 336 337 if (keyCode == KeyEvent.KEYCODE_SYM) { 338 press(content, SYM); 339 return true; 340 } 341 342 return false; // no super to call through to 343 } 344 press(Editable content, Object what)345 private void press(Editable content, Object what) { 346 int state = content.getSpanFlags(what); 347 348 if (state == PRESSED) 349 ; // repeat before use 350 else if (state == RELEASED) 351 content.setSpan(what, 0, 0, LOCKED); 352 else if (state == USED) 353 ; // repeat after use 354 else if (state == LOCKED) 355 content.removeSpan(what); 356 else 357 content.setSpan(what, 0, 0, PRESSED); 358 } 359 360 /** 361 * Start selecting text. 362 * @hide pending API review 363 */ startSelecting(View view, Spannable content)364 public static void startSelecting(View view, Spannable content) { 365 content.setSpan(SELECTING, 0, 0, PRESSED); 366 } 367 368 /** 369 * Stop selecting text. This does not actually collapse the selection; 370 * call {@link android.text.Selection#setSelection} too. 371 * @hide pending API review 372 */ stopSelecting(View view, Spannable content)373 public static void stopSelecting(View view, Spannable content) { 374 content.removeSpan(SELECTING); 375 } 376 377 /** 378 * Handles release of the meta keys. 379 */ onKeyUp(View view, Editable content, int keyCode, KeyEvent event)380 public boolean onKeyUp(View view, Editable content, int keyCode, KeyEvent event) { 381 if (keyCode == KeyEvent.KEYCODE_SHIFT_LEFT || keyCode == KeyEvent.KEYCODE_SHIFT_RIGHT) { 382 release(content, CAP, event); 383 return true; 384 } 385 386 if (keyCode == KeyEvent.KEYCODE_ALT_LEFT || keyCode == KeyEvent.KEYCODE_ALT_RIGHT 387 || keyCode == KeyEvent.KEYCODE_NUM) { 388 release(content, ALT, event); 389 return true; 390 } 391 392 if (keyCode == KeyEvent.KEYCODE_SYM) { 393 release(content, SYM, event); 394 return true; 395 } 396 397 return false; // no super to call through to 398 } 399 release(Editable content, Object what, KeyEvent event)400 private void release(Editable content, Object what, KeyEvent event) { 401 int current = content.getSpanFlags(what); 402 403 switch (event.getKeyCharacterMap().getModifierBehavior()) { 404 case KeyCharacterMap.MODIFIER_BEHAVIOR_CHORDED_OR_TOGGLED: 405 if (current == USED) 406 content.removeSpan(what); 407 else if (current == PRESSED) 408 content.setSpan(what, 0, 0, RELEASED); 409 break; 410 411 default: 412 content.removeSpan(what); 413 break; 414 } 415 } 416 clearMetaKeyState(View view, Editable content, int states)417 public void clearMetaKeyState(View view, Editable content, int states) { 418 clearMetaKeyState(content, states); 419 } 420 clearMetaKeyState(Editable content, int states)421 public static void clearMetaKeyState(Editable content, int states) { 422 if ((states&META_SHIFT_ON) != 0) content.removeSpan(CAP); 423 if ((states&META_ALT_ON) != 0) content.removeSpan(ALT); 424 if ((states&META_SYM_ON) != 0) content.removeSpan(SYM); 425 if ((states&META_SELECTING) != 0) content.removeSpan(SELECTING); 426 } 427 428 /** 429 * Call this if you are a method that ignores the locked meta state 430 * (arrow keys, for example) and you handle a key. 431 */ resetLockedMeta(long state)432 public static long resetLockedMeta(long state) { 433 if ((state & META_CAP_LOCKED) != 0) { 434 state &= ~META_SHIFT_MASK; 435 } 436 if ((state & META_ALT_LOCKED) != 0) { 437 state &= ~META_ALT_MASK; 438 } 439 if ((state & META_SYM_LOCKED) != 0) { 440 state &= ~META_SYM_MASK; 441 } 442 return state; 443 } 444 445 // --------------------------------------------------------------------- 446 // Version of API that operates on a state bit mask 447 // --------------------------------------------------------------------- 448 449 /** 450 * Gets the state of the meta keys. 451 * 452 * @param state the current meta state bits. 453 * 454 * @return an integer in which each bit set to one represents a pressed 455 * or locked meta key. 456 */ getMetaState(long state)457 public static final int getMetaState(long state) { 458 int result = 0; 459 460 if ((state & META_CAP_LOCKED) != 0) { 461 result |= META_CAP_LOCKED; 462 } else if ((state & META_SHIFT_ON) != 0) { 463 result |= META_SHIFT_ON; 464 } 465 466 if ((state & META_ALT_LOCKED) != 0) { 467 result |= META_ALT_LOCKED; 468 } else if ((state & META_ALT_ON) != 0) { 469 result |= META_ALT_ON; 470 } 471 472 if ((state & META_SYM_LOCKED) != 0) { 473 result |= META_SYM_LOCKED; 474 } else if ((state & META_SYM_ON) != 0) { 475 result |= META_SYM_ON; 476 } 477 478 return result; 479 } 480 481 /** 482 * Gets the state of a particular meta key. 483 * 484 * @param state the current state bits. 485 * @param meta META_SHIFT_ON, META_ALT_ON, or META_SYM_ON 486 * 487 * @return 0 if inactive, 1 if active, 2 if locked. 488 */ getMetaState(long state, int meta)489 public static final int getMetaState(long state, int meta) { 490 switch (meta) { 491 case META_SHIFT_ON: 492 if ((state & META_CAP_LOCKED) != 0) return LOCKED_RETURN_VALUE; 493 if ((state & META_SHIFT_ON) != 0) return PRESSED_RETURN_VALUE; 494 return 0; 495 496 case META_ALT_ON: 497 if ((state & META_ALT_LOCKED) != 0) return LOCKED_RETURN_VALUE; 498 if ((state & META_ALT_ON) != 0) return PRESSED_RETURN_VALUE; 499 return 0; 500 501 case META_SYM_ON: 502 if ((state & META_SYM_LOCKED) != 0) return LOCKED_RETURN_VALUE; 503 if ((state & META_SYM_ON) != 0) return PRESSED_RETURN_VALUE; 504 return 0; 505 506 default: 507 return 0; 508 } 509 } 510 511 /** 512 * Call this method after you handle a keypress so that the meta 513 * state will be reset to unshifted (if it is not still down) 514 * or primed to be reset to unshifted (once it is released). Takes 515 * the current state, returns the new state. 516 */ adjustMetaAfterKeypress(long state)517 public static long adjustMetaAfterKeypress(long state) { 518 if ((state & META_CAP_PRESSED) != 0) { 519 state = (state & ~META_SHIFT_MASK) | META_SHIFT_ON | META_CAP_USED; 520 } else if ((state & META_CAP_RELEASED) != 0) { 521 state &= ~META_SHIFT_MASK; 522 } 523 524 if ((state & META_ALT_PRESSED) != 0) { 525 state = (state & ~META_ALT_MASK) | META_ALT_ON | META_ALT_USED; 526 } else if ((state & META_ALT_RELEASED) != 0) { 527 state &= ~META_ALT_MASK; 528 } 529 530 if ((state & META_SYM_PRESSED) != 0) { 531 state = (state & ~META_SYM_MASK) | META_SYM_ON | META_SYM_USED; 532 } else if ((state & META_SYM_RELEASED) != 0) { 533 state &= ~META_SYM_MASK; 534 } 535 return state; 536 } 537 538 /** 539 * Handles presses of the meta keys. 540 */ handleKeyDown(long state, int keyCode, KeyEvent event)541 public static long handleKeyDown(long state, int keyCode, KeyEvent event) { 542 if (keyCode == KeyEvent.KEYCODE_SHIFT_LEFT || keyCode == KeyEvent.KEYCODE_SHIFT_RIGHT) { 543 return press(state, META_SHIFT_ON, META_SHIFT_MASK, 544 META_CAP_LOCKED, META_CAP_PRESSED, META_CAP_RELEASED, META_CAP_USED); 545 } 546 547 if (keyCode == KeyEvent.KEYCODE_ALT_LEFT || keyCode == KeyEvent.KEYCODE_ALT_RIGHT 548 || keyCode == KeyEvent.KEYCODE_NUM) { 549 return press(state, META_ALT_ON, META_ALT_MASK, 550 META_ALT_LOCKED, META_ALT_PRESSED, META_ALT_RELEASED, META_ALT_USED); 551 } 552 553 if (keyCode == KeyEvent.KEYCODE_SYM) { 554 return press(state, META_SYM_ON, META_SYM_MASK, 555 META_SYM_LOCKED, META_SYM_PRESSED, META_SYM_RELEASED, META_SYM_USED); 556 } 557 return state; 558 } 559 press(long state, int what, long mask, long locked, long pressed, long released, long used)560 private static long press(long state, int what, long mask, 561 long locked, long pressed, long released, long used) { 562 if ((state & pressed) != 0) { 563 // repeat before use 564 } else if ((state & released) != 0) { 565 state = (state &~ mask) | what | locked; 566 } else if ((state & used) != 0) { 567 // repeat after use 568 } else if ((state & locked) != 0) { 569 state &= ~mask; 570 } else { 571 state |= what | pressed; 572 } 573 return state; 574 } 575 576 /** 577 * Handles release of the meta keys. 578 */ handleKeyUp(long state, int keyCode, KeyEvent event)579 public static long handleKeyUp(long state, int keyCode, KeyEvent event) { 580 if (keyCode == KeyEvent.KEYCODE_SHIFT_LEFT || keyCode == KeyEvent.KEYCODE_SHIFT_RIGHT) { 581 return release(state, META_SHIFT_ON, META_SHIFT_MASK, 582 META_CAP_PRESSED, META_CAP_RELEASED, META_CAP_USED, event); 583 } 584 585 if (keyCode == KeyEvent.KEYCODE_ALT_LEFT || keyCode == KeyEvent.KEYCODE_ALT_RIGHT 586 || keyCode == KeyEvent.KEYCODE_NUM) { 587 return release(state, META_ALT_ON, META_ALT_MASK, 588 META_ALT_PRESSED, META_ALT_RELEASED, META_ALT_USED, event); 589 } 590 591 if (keyCode == KeyEvent.KEYCODE_SYM) { 592 return release(state, META_SYM_ON, META_SYM_MASK, 593 META_SYM_PRESSED, META_SYM_RELEASED, META_SYM_USED, event); 594 } 595 return state; 596 } 597 release(long state, int what, long mask, long pressed, long released, long used, KeyEvent event)598 private static long release(long state, int what, long mask, 599 long pressed, long released, long used, KeyEvent event) { 600 switch (event.getKeyCharacterMap().getModifierBehavior()) { 601 case KeyCharacterMap.MODIFIER_BEHAVIOR_CHORDED_OR_TOGGLED: 602 if ((state & used) != 0) { 603 state &= ~mask; 604 } else if ((state & pressed) != 0) { 605 state |= what | released; 606 } 607 break; 608 609 default: 610 state &= ~mask; 611 break; 612 } 613 return state; 614 } 615 616 /** 617 * Clears the state of the specified meta key if it is locked. 618 * @param state the meta key state 619 * @param which meta keys to clear, may be a combination of {@link #META_SHIFT_ON}, 620 * {@link #META_ALT_ON} or {@link #META_SYM_ON}. 621 */ clearMetaKeyState(long state, int which)622 public long clearMetaKeyState(long state, int which) { 623 if ((which & META_SHIFT_ON) != 0 && (state & META_CAP_LOCKED) != 0) { 624 state &= ~META_SHIFT_MASK; 625 } 626 if ((which & META_ALT_ON) != 0 && (state & META_ALT_LOCKED) != 0) { 627 state &= ~META_ALT_MASK; 628 } 629 if ((which & META_SYM_ON) != 0 && (state & META_SYM_LOCKED) != 0) { 630 state &= ~META_SYM_MASK; 631 } 632 return state; 633 } 634 635 /** 636 * The meta key has been pressed but has not yet been used. 637 */ 638 private static final int PRESSED = 639 Spannable.SPAN_MARK_MARK | (1 << Spannable.SPAN_USER_SHIFT); 640 641 /** 642 * The meta key has been pressed and released but has still 643 * not yet been used. 644 */ 645 private static final int RELEASED = 646 Spannable.SPAN_MARK_MARK | (2 << Spannable.SPAN_USER_SHIFT); 647 648 /** 649 * The meta key has been pressed and used but has not yet been released. 650 */ 651 private static final int USED = 652 Spannable.SPAN_MARK_MARK | (3 << Spannable.SPAN_USER_SHIFT); 653 654 /** 655 * The meta key has been pressed and released without use, and then 656 * pressed again; it may also have been released again. 657 */ 658 private static final int LOCKED = 659 Spannable.SPAN_MARK_MARK | (4 << Spannable.SPAN_USER_SHIFT); 660 } 661