1 /* 2 * Copyright (C) 2012 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.internal.widget.multiwaveview; 18 19 import android.animation.Animator; 20 import android.animation.Animator.AnimatorListener; 21 import android.animation.AnimatorListenerAdapter; 22 import android.animation.TimeInterpolator; 23 import android.animation.ValueAnimator; 24 import android.animation.ValueAnimator.AnimatorUpdateListener; 25 import android.content.ComponentName; 26 import android.content.Context; 27 import android.content.pm.PackageManager; 28 import android.content.pm.PackageManager.NameNotFoundException; 29 import android.content.res.Resources; 30 import android.content.res.TypedArray; 31 import android.graphics.Canvas; 32 import android.graphics.drawable.Drawable; 33 import android.os.Bundle; 34 import android.os.UserHandle; 35 import android.os.Vibrator; 36 import android.provider.Settings; 37 import android.text.TextUtils; 38 import android.util.AttributeSet; 39 import android.util.Log; 40 import android.util.TypedValue; 41 import android.view.Gravity; 42 import android.view.MotionEvent; 43 import android.view.View; 44 import android.view.accessibility.AccessibilityManager; 45 46 import com.android.internal.R; 47 48 import java.util.ArrayList; 49 50 /** 51 * A re-usable widget containing a center, outer ring and wave animation. 52 */ 53 public class GlowPadView extends View { 54 private static final String TAG = "GlowPadView"; 55 private static final boolean DEBUG = false; 56 57 // Wave state machine 58 private static final int STATE_IDLE = 0; 59 private static final int STATE_START = 1; 60 private static final int STATE_FIRST_TOUCH = 2; 61 private static final int STATE_TRACKING = 3; 62 private static final int STATE_SNAP = 4; 63 private static final int STATE_FINISH = 5; 64 65 // Animation properties. 66 private static final float SNAP_MARGIN_DEFAULT = 20.0f; // distance to ring before we snap to it 67 68 public interface OnTriggerListener { 69 int NO_HANDLE = 0; 70 int CENTER_HANDLE = 1; onGrabbed(View v, int handle)71 public void onGrabbed(View v, int handle); onReleased(View v, int handle)72 public void onReleased(View v, int handle); onTrigger(View v, int target)73 public void onTrigger(View v, int target); onGrabbedStateChange(View v, int handle)74 public void onGrabbedStateChange(View v, int handle); onFinishFinalAnimation()75 public void onFinishFinalAnimation(); 76 } 77 78 // Tuneable parameters for animation 79 private static final int WAVE_ANIMATION_DURATION = 1350; 80 private static final int RETURN_TO_HOME_DELAY = 1200; 81 private static final int RETURN_TO_HOME_DURATION = 200; 82 private static final int HIDE_ANIMATION_DELAY = 200; 83 private static final int HIDE_ANIMATION_DURATION = 200; 84 private static final int SHOW_ANIMATION_DURATION = 200; 85 private static final int SHOW_ANIMATION_DELAY = 50; 86 private static final int INITIAL_SHOW_HANDLE_DURATION = 200; 87 private static final int REVEAL_GLOW_DELAY = 0; 88 private static final int REVEAL_GLOW_DURATION = 0; 89 90 private static final float TAP_RADIUS_SCALE_ACCESSIBILITY_ENABLED = 1.3f; 91 private static final float TARGET_SCALE_EXPANDED = 1.0f; 92 private static final float TARGET_SCALE_COLLAPSED = 0.8f; 93 private static final float RING_SCALE_EXPANDED = 1.0f; 94 private static final float RING_SCALE_COLLAPSED = 0.5f; 95 96 private ArrayList<TargetDrawable> mTargetDrawables = new ArrayList<TargetDrawable>(); 97 private AnimationBundle mWaveAnimations = new AnimationBundle(); 98 private AnimationBundle mTargetAnimations = new AnimationBundle(); 99 private AnimationBundle mGlowAnimations = new AnimationBundle(); 100 private ArrayList<String> mTargetDescriptions; 101 private ArrayList<String> mDirectionDescriptions; 102 private OnTriggerListener mOnTriggerListener; 103 private TargetDrawable mHandleDrawable; 104 private TargetDrawable mOuterRing; 105 private Vibrator mVibrator; 106 107 private int mFeedbackCount = 3; 108 private int mVibrationDuration = 0; 109 private int mGrabbedState; 110 private int mActiveTarget = -1; 111 private float mGlowRadius; 112 private float mWaveCenterX; 113 private float mWaveCenterY; 114 private int mMaxTargetHeight; 115 private int mMaxTargetWidth; 116 private float mRingScaleFactor = 1f; 117 private boolean mAllowScaling; 118 119 private float mOuterRadius = 0.0f; 120 private float mSnapMargin = 0.0f; 121 private float mFirstItemOffset = 0.0f; 122 private boolean mMagneticTargets = false; 123 private boolean mDragging; 124 private int mNewTargetResources; 125 126 private class AnimationBundle extends ArrayList<Tweener> { 127 private static final long serialVersionUID = 0xA84D78726F127468L; 128 private boolean mSuspended; 129 start()130 public void start() { 131 if (mSuspended) return; // ignore attempts to start animations 132 final int count = size(); 133 for (int i = 0; i < count; i++) { 134 Tweener anim = get(i); 135 anim.animator.start(); 136 } 137 } 138 cancel()139 public void cancel() { 140 final int count = size(); 141 for (int i = 0; i < count; i++) { 142 Tweener anim = get(i); 143 anim.animator.cancel(); 144 } 145 clear(); 146 } 147 stop()148 public void stop() { 149 final int count = size(); 150 for (int i = 0; i < count; i++) { 151 Tweener anim = get(i); 152 anim.animator.end(); 153 } 154 clear(); 155 } 156 setSuspended(boolean suspend)157 public void setSuspended(boolean suspend) { 158 mSuspended = suspend; 159 } 160 }; 161 162 private AnimatorListener mResetListener = new AnimatorListenerAdapter() { 163 public void onAnimationEnd(Animator animator) { 164 switchToState(STATE_IDLE, mWaveCenterX, mWaveCenterY); 165 dispatchOnFinishFinalAnimation(); 166 } 167 }; 168 169 private AnimatorListener mResetListenerWithPing = new AnimatorListenerAdapter() { 170 public void onAnimationEnd(Animator animator) { 171 ping(); 172 switchToState(STATE_IDLE, mWaveCenterX, mWaveCenterY); 173 dispatchOnFinishFinalAnimation(); 174 } 175 }; 176 177 private AnimatorUpdateListener mUpdateListener = new AnimatorUpdateListener() { 178 public void onAnimationUpdate(ValueAnimator animation) { 179 invalidate(); 180 } 181 }; 182 183 private boolean mAnimatingTargets; 184 private AnimatorListener mTargetUpdateListener = new AnimatorListenerAdapter() { 185 public void onAnimationEnd(Animator animator) { 186 if (mNewTargetResources != 0) { 187 internalSetTargetResources(mNewTargetResources); 188 mNewTargetResources = 0; 189 hideTargets(false, false); 190 } 191 mAnimatingTargets = false; 192 } 193 }; 194 private int mTargetResourceId; 195 private int mTargetDescriptionsResourceId; 196 private int mDirectionDescriptionsResourceId; 197 private boolean mAlwaysTrackFinger; 198 private int mHorizontalInset; 199 private int mVerticalInset; 200 private int mGravity = Gravity.TOP; 201 private boolean mInitialLayout = true; 202 private Tweener mBackgroundAnimator; 203 private PointCloud mPointCloud; 204 private float mInnerRadius; 205 private int mPointerId; 206 GlowPadView(Context context)207 public GlowPadView(Context context) { 208 this(context, null); 209 } 210 GlowPadView(Context context, AttributeSet attrs)211 public GlowPadView(Context context, AttributeSet attrs) { 212 super(context, attrs); 213 Resources res = context.getResources(); 214 215 TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.GlowPadView); 216 mInnerRadius = a.getDimension(R.styleable.GlowPadView_innerRadius, mInnerRadius); 217 mOuterRadius = a.getDimension(R.styleable.GlowPadView_outerRadius, mOuterRadius); 218 mSnapMargin = a.getDimension(R.styleable.GlowPadView_snapMargin, mSnapMargin); 219 mFirstItemOffset = (float) Math.toRadians( 220 a.getFloat(R.styleable.GlowPadView_firstItemOffset, 221 (float) Math.toDegrees(mFirstItemOffset))); 222 mVibrationDuration = a.getInt(R.styleable.GlowPadView_vibrationDuration, 223 mVibrationDuration); 224 mFeedbackCount = a.getInt(R.styleable.GlowPadView_feedbackCount, 225 mFeedbackCount); 226 mAllowScaling = a.getBoolean(R.styleable.GlowPadView_allowScaling, false); 227 TypedValue handle = a.peekValue(R.styleable.GlowPadView_handleDrawable); 228 mHandleDrawable = new TargetDrawable(res, handle != null ? handle.resourceId : 0); 229 mHandleDrawable.setState(TargetDrawable.STATE_INACTIVE); 230 mOuterRing = new TargetDrawable(res, 231 getResourceId(a, R.styleable.GlowPadView_outerRingDrawable)); 232 233 mAlwaysTrackFinger = a.getBoolean(R.styleable.GlowPadView_alwaysTrackFinger, false); 234 mMagneticTargets = a.getBoolean(R.styleable.GlowPadView_magneticTargets, mMagneticTargets); 235 236 int pointId = getResourceId(a, R.styleable.GlowPadView_pointDrawable); 237 Drawable pointDrawable = pointId != 0 ? res.getDrawable(pointId) : null; 238 mGlowRadius = a.getDimension(R.styleable.GlowPadView_glowRadius, 0.0f); 239 240 TypedValue outValue = new TypedValue(); 241 242 // Read array of target drawables 243 if (a.getValue(R.styleable.GlowPadView_targetDrawables, outValue)) { 244 internalSetTargetResources(outValue.resourceId); 245 } 246 if (mTargetDrawables == null || mTargetDrawables.size() == 0) { 247 throw new IllegalStateException("Must specify at least one target drawable"); 248 } 249 250 // Read array of target descriptions 251 if (a.getValue(R.styleable.GlowPadView_targetDescriptions, outValue)) { 252 final int resourceId = outValue.resourceId; 253 if (resourceId == 0) { 254 throw new IllegalStateException("Must specify target descriptions"); 255 } 256 setTargetDescriptionsResourceId(resourceId); 257 } 258 259 // Read array of direction descriptions 260 if (a.getValue(R.styleable.GlowPadView_directionDescriptions, outValue)) { 261 final int resourceId = outValue.resourceId; 262 if (resourceId == 0) { 263 throw new IllegalStateException("Must specify direction descriptions"); 264 } 265 setDirectionDescriptionsResourceId(resourceId); 266 } 267 268 mGravity = a.getInt(R.styleable.GlowPadView_gravity, Gravity.TOP); 269 270 a.recycle(); 271 272 setVibrateEnabled(mVibrationDuration > 0); 273 274 assignDefaultsIfNeeded(); 275 276 mPointCloud = new PointCloud(pointDrawable); 277 mPointCloud.makePointCloud(mInnerRadius, mOuterRadius); 278 mPointCloud.glowManager.setRadius(mGlowRadius); 279 } 280 getResourceId(TypedArray a, int id)281 private int getResourceId(TypedArray a, int id) { 282 TypedValue tv = a.peekValue(id); 283 return tv == null ? 0 : tv.resourceId; 284 } 285 dump()286 private void dump() { 287 Log.v(TAG, "Outer Radius = " + mOuterRadius); 288 Log.v(TAG, "SnapMargin = " + mSnapMargin); 289 Log.v(TAG, "FeedbackCount = " + mFeedbackCount); 290 Log.v(TAG, "VibrationDuration = " + mVibrationDuration); 291 Log.v(TAG, "GlowRadius = " + mGlowRadius); 292 Log.v(TAG, "WaveCenterX = " + mWaveCenterX); 293 Log.v(TAG, "WaveCenterY = " + mWaveCenterY); 294 } 295 suspendAnimations()296 public void suspendAnimations() { 297 mWaveAnimations.setSuspended(true); 298 mTargetAnimations.setSuspended(true); 299 mGlowAnimations.setSuspended(true); 300 } 301 resumeAnimations()302 public void resumeAnimations() { 303 mWaveAnimations.setSuspended(false); 304 mTargetAnimations.setSuspended(false); 305 mGlowAnimations.setSuspended(false); 306 mWaveAnimations.start(); 307 mTargetAnimations.start(); 308 mGlowAnimations.start(); 309 } 310 311 @Override getSuggestedMinimumWidth()312 protected int getSuggestedMinimumWidth() { 313 // View should be large enough to contain the background + handle and 314 // target drawable on either edge. 315 return (int) (Math.max(mOuterRing.getWidth(), 2 * mOuterRadius) + mMaxTargetWidth); 316 } 317 318 @Override getSuggestedMinimumHeight()319 protected int getSuggestedMinimumHeight() { 320 // View should be large enough to contain the unlock ring + target and 321 // target drawable on either edge 322 return (int) (Math.max(mOuterRing.getHeight(), 2 * mOuterRadius) + mMaxTargetHeight); 323 } 324 325 /** 326 * This gets the suggested width accounting for the ring's scale factor. 327 */ getScaledSuggestedMinimumWidth()328 protected int getScaledSuggestedMinimumWidth() { 329 return (int) (mRingScaleFactor * Math.max(mOuterRing.getWidth(), 2 * mOuterRadius) 330 + mMaxTargetWidth); 331 } 332 333 /** 334 * This gets the suggested height accounting for the ring's scale factor. 335 */ getScaledSuggestedMinimumHeight()336 protected int getScaledSuggestedMinimumHeight() { 337 return (int) (mRingScaleFactor * Math.max(mOuterRing.getHeight(), 2 * mOuterRadius) 338 + mMaxTargetHeight); 339 } 340 resolveMeasured(int measureSpec, int desired)341 private int resolveMeasured(int measureSpec, int desired) 342 { 343 int result = 0; 344 int specSize = MeasureSpec.getSize(measureSpec); 345 switch (MeasureSpec.getMode(measureSpec)) { 346 case MeasureSpec.UNSPECIFIED: 347 result = desired; 348 break; 349 case MeasureSpec.AT_MOST: 350 result = Math.min(specSize, desired); 351 break; 352 case MeasureSpec.EXACTLY: 353 default: 354 result = specSize; 355 } 356 return result; 357 } 358 switchToState(int state, float x, float y)359 private void switchToState(int state, float x, float y) { 360 switch (state) { 361 case STATE_IDLE: 362 deactivateTargets(); 363 hideGlow(0, 0, 0.0f, null); 364 startBackgroundAnimation(0, 0.0f); 365 mHandleDrawable.setState(TargetDrawable.STATE_INACTIVE); 366 mHandleDrawable.setAlpha(1.0f); 367 break; 368 369 case STATE_START: 370 startBackgroundAnimation(0, 0.0f); 371 break; 372 373 case STATE_FIRST_TOUCH: 374 mHandleDrawable.setAlpha(0.0f); 375 deactivateTargets(); 376 showTargets(true); 377 startBackgroundAnimation(INITIAL_SHOW_HANDLE_DURATION, 1.0f); 378 setGrabbedState(OnTriggerListener.CENTER_HANDLE); 379 if (AccessibilityManager.getInstance(mContext).isEnabled()) { 380 announceTargets(); 381 } 382 break; 383 384 case STATE_TRACKING: 385 mHandleDrawable.setAlpha(0.0f); 386 showGlow(REVEAL_GLOW_DURATION , REVEAL_GLOW_DELAY, 1.0f, null); 387 break; 388 389 case STATE_SNAP: 390 // TODO: Add transition states (see list_selector_background_transition.xml) 391 mHandleDrawable.setAlpha(0.0f); 392 showGlow(REVEAL_GLOW_DURATION , REVEAL_GLOW_DELAY, 0.0f, null); 393 break; 394 395 case STATE_FINISH: 396 doFinish(); 397 break; 398 } 399 } 400 showGlow(int duration, int delay, float finalAlpha, AnimatorListener finishListener)401 private void showGlow(int duration, int delay, float finalAlpha, 402 AnimatorListener finishListener) { 403 mGlowAnimations.cancel(); 404 mGlowAnimations.add(Tweener.to(mPointCloud.glowManager, duration, 405 "ease", Ease.Cubic.easeIn, 406 "delay", delay, 407 "alpha", finalAlpha, 408 "onUpdate", mUpdateListener, 409 "onComplete", finishListener)); 410 mGlowAnimations.start(); 411 } 412 hideGlow(int duration, int delay, float finalAlpha, AnimatorListener finishListener)413 private void hideGlow(int duration, int delay, float finalAlpha, 414 AnimatorListener finishListener) { 415 mGlowAnimations.cancel(); 416 mGlowAnimations.add(Tweener.to(mPointCloud.glowManager, duration, 417 "ease", Ease.Quart.easeOut, 418 "delay", delay, 419 "alpha", finalAlpha, 420 "x", 0.0f, 421 "y", 0.0f, 422 "onUpdate", mUpdateListener, 423 "onComplete", finishListener)); 424 mGlowAnimations.start(); 425 } 426 deactivateTargets()427 private void deactivateTargets() { 428 final int count = mTargetDrawables.size(); 429 for (int i = 0; i < count; i++) { 430 TargetDrawable target = mTargetDrawables.get(i); 431 target.setState(TargetDrawable.STATE_INACTIVE); 432 } 433 mActiveTarget = -1; 434 } 435 436 /** 437 * Dispatches a trigger event to listener. Ignored if a listener is not set. 438 * @param whichTarget the target that was triggered. 439 */ dispatchTriggerEvent(int whichTarget)440 private void dispatchTriggerEvent(int whichTarget) { 441 vibrate(); 442 if (mOnTriggerListener != null) { 443 mOnTriggerListener.onTrigger(this, whichTarget); 444 } 445 } 446 dispatchOnFinishFinalAnimation()447 private void dispatchOnFinishFinalAnimation() { 448 if (mOnTriggerListener != null) { 449 mOnTriggerListener.onFinishFinalAnimation(); 450 } 451 } 452 doFinish()453 private void doFinish() { 454 final int activeTarget = mActiveTarget; 455 final boolean targetHit = activeTarget != -1; 456 457 if (targetHit) { 458 if (DEBUG) Log.v(TAG, "Finish with target hit = " + targetHit); 459 460 highlightSelected(activeTarget); 461 462 // Inform listener of any active targets. Typically only one will be active. 463 hideGlow(RETURN_TO_HOME_DURATION, RETURN_TO_HOME_DELAY, 0.0f, mResetListener); 464 dispatchTriggerEvent(activeTarget); 465 if (!mAlwaysTrackFinger) { 466 // Force ring and targets to finish animation to final expanded state 467 mTargetAnimations.stop(); 468 } 469 } else { 470 // Animate handle back to the center based on current state. 471 hideGlow(HIDE_ANIMATION_DURATION, 0, 0.0f, mResetListenerWithPing); 472 hideTargets(true, false); 473 } 474 475 setGrabbedState(OnTriggerListener.NO_HANDLE); 476 } 477 highlightSelected(int activeTarget)478 private void highlightSelected(int activeTarget) { 479 // Highlight the given target and fade others 480 mTargetDrawables.get(activeTarget).setState(TargetDrawable.STATE_ACTIVE); 481 hideUnselected(activeTarget); 482 } 483 hideUnselected(int active)484 private void hideUnselected(int active) { 485 for (int i = 0; i < mTargetDrawables.size(); i++) { 486 if (i != active) { 487 mTargetDrawables.get(i).setAlpha(0.0f); 488 } 489 } 490 } 491 hideTargets(boolean animate, boolean expanded)492 private void hideTargets(boolean animate, boolean expanded) { 493 mTargetAnimations.cancel(); 494 // Note: these animations should complete at the same time so that we can swap out 495 // the target assets asynchronously from the setTargetResources() call. 496 mAnimatingTargets = animate; 497 final int duration = animate ? HIDE_ANIMATION_DURATION : 0; 498 final int delay = animate ? HIDE_ANIMATION_DELAY : 0; 499 500 final float targetScale = expanded ? 501 TARGET_SCALE_EXPANDED : TARGET_SCALE_COLLAPSED; 502 final int length = mTargetDrawables.size(); 503 final TimeInterpolator interpolator = Ease.Cubic.easeOut; 504 for (int i = 0; i < length; i++) { 505 TargetDrawable target = mTargetDrawables.get(i); 506 target.setState(TargetDrawable.STATE_INACTIVE); 507 mTargetAnimations.add(Tweener.to(target, duration, 508 "ease", interpolator, 509 "alpha", 0.0f, 510 "scaleX", targetScale, 511 "scaleY", targetScale, 512 "delay", delay, 513 "onUpdate", mUpdateListener)); 514 } 515 516 float ringScaleTarget = expanded ? 517 RING_SCALE_EXPANDED : RING_SCALE_COLLAPSED; 518 ringScaleTarget *= mRingScaleFactor; 519 mTargetAnimations.add(Tweener.to(mOuterRing, duration, 520 "ease", interpolator, 521 "alpha", 0.0f, 522 "scaleX", ringScaleTarget, 523 "scaleY", ringScaleTarget, 524 "delay", delay, 525 "onUpdate", mUpdateListener, 526 "onComplete", mTargetUpdateListener)); 527 528 mTargetAnimations.start(); 529 } 530 showTargets(boolean animate)531 private void showTargets(boolean animate) { 532 mTargetAnimations.stop(); 533 mAnimatingTargets = animate; 534 final int delay = animate ? SHOW_ANIMATION_DELAY : 0; 535 final int duration = animate ? SHOW_ANIMATION_DURATION : 0; 536 final int length = mTargetDrawables.size(); 537 for (int i = 0; i < length; i++) { 538 TargetDrawable target = mTargetDrawables.get(i); 539 target.setState(TargetDrawable.STATE_INACTIVE); 540 mTargetAnimations.add(Tweener.to(target, duration, 541 "ease", Ease.Cubic.easeOut, 542 "alpha", 1.0f, 543 "scaleX", 1.0f, 544 "scaleY", 1.0f, 545 "delay", delay, 546 "onUpdate", mUpdateListener)); 547 } 548 549 float ringScale = mRingScaleFactor * RING_SCALE_EXPANDED; 550 mTargetAnimations.add(Tweener.to(mOuterRing, duration, 551 "ease", Ease.Cubic.easeOut, 552 "alpha", 1.0f, 553 "scaleX", ringScale, 554 "scaleY", ringScale, 555 "delay", delay, 556 "onUpdate", mUpdateListener, 557 "onComplete", mTargetUpdateListener)); 558 559 mTargetAnimations.start(); 560 } 561 vibrate()562 private void vibrate() { 563 final boolean hapticEnabled = Settings.System.getIntForUser( 564 mContext.getContentResolver(), Settings.System.HAPTIC_FEEDBACK_ENABLED, 1, 565 UserHandle.USER_CURRENT) != 0; 566 if (mVibrator != null && hapticEnabled) { 567 mVibrator.vibrate(mVibrationDuration); 568 } 569 } 570 loadDrawableArray(int resourceId)571 private ArrayList<TargetDrawable> loadDrawableArray(int resourceId) { 572 Resources res = getContext().getResources(); 573 TypedArray array = res.obtainTypedArray(resourceId); 574 final int count = array.length(); 575 ArrayList<TargetDrawable> drawables = new ArrayList<TargetDrawable>(count); 576 for (int i = 0; i < count; i++) { 577 TypedValue value = array.peekValue(i); 578 TargetDrawable target = new TargetDrawable(res, value != null ? value.resourceId : 0); 579 drawables.add(target); 580 } 581 array.recycle(); 582 return drawables; 583 } 584 internalSetTargetResources(int resourceId)585 private void internalSetTargetResources(int resourceId) { 586 final ArrayList<TargetDrawable> targets = loadDrawableArray(resourceId); 587 mTargetDrawables = targets; 588 mTargetResourceId = resourceId; 589 590 int maxWidth = mHandleDrawable.getWidth(); 591 int maxHeight = mHandleDrawable.getHeight(); 592 final int count = targets.size(); 593 for (int i = 0; i < count; i++) { 594 TargetDrawable target = targets.get(i); 595 maxWidth = Math.max(maxWidth, target.getWidth()); 596 maxHeight = Math.max(maxHeight, target.getHeight()); 597 } 598 if (mMaxTargetWidth != maxWidth || mMaxTargetHeight != maxHeight) { 599 mMaxTargetWidth = maxWidth; 600 mMaxTargetHeight = maxHeight; 601 requestLayout(); // required to resize layout and call updateTargetPositions() 602 } else { 603 updateTargetPositions(mWaveCenterX, mWaveCenterY); 604 updatePointCloudPosition(mWaveCenterX, mWaveCenterY); 605 } 606 } 607 608 /** 609 * Loads an array of drawables from the given resourceId. 610 * 611 * @param resourceId 612 */ setTargetResources(int resourceId)613 public void setTargetResources(int resourceId) { 614 if (mAnimatingTargets) { 615 // postpone this change until we return to the initial state 616 mNewTargetResources = resourceId; 617 } else { 618 internalSetTargetResources(resourceId); 619 } 620 } 621 getTargetResourceId()622 public int getTargetResourceId() { 623 return mTargetResourceId; 624 } 625 626 /** 627 * Sets the resource id specifying the target descriptions for accessibility. 628 * 629 * @param resourceId The resource id. 630 */ setTargetDescriptionsResourceId(int resourceId)631 public void setTargetDescriptionsResourceId(int resourceId) { 632 mTargetDescriptionsResourceId = resourceId; 633 if (mTargetDescriptions != null) { 634 mTargetDescriptions.clear(); 635 } 636 } 637 638 /** 639 * Gets the resource id specifying the target descriptions for accessibility. 640 * 641 * @return The resource id. 642 */ getTargetDescriptionsResourceId()643 public int getTargetDescriptionsResourceId() { 644 return mTargetDescriptionsResourceId; 645 } 646 647 /** 648 * Sets the resource id specifying the target direction descriptions for accessibility. 649 * 650 * @param resourceId The resource id. 651 */ setDirectionDescriptionsResourceId(int resourceId)652 public void setDirectionDescriptionsResourceId(int resourceId) { 653 mDirectionDescriptionsResourceId = resourceId; 654 if (mDirectionDescriptions != null) { 655 mDirectionDescriptions.clear(); 656 } 657 } 658 659 /** 660 * Gets the resource id specifying the target direction descriptions. 661 * 662 * @return The resource id. 663 */ getDirectionDescriptionsResourceId()664 public int getDirectionDescriptionsResourceId() { 665 return mDirectionDescriptionsResourceId; 666 } 667 668 /** 669 * Enable or disable vibrate on touch. 670 * 671 * @param enabled 672 */ setVibrateEnabled(boolean enabled)673 public void setVibrateEnabled(boolean enabled) { 674 if (enabled && mVibrator == null) { 675 mVibrator = (Vibrator) getContext().getSystemService(Context.VIBRATOR_SERVICE); 676 } else { 677 mVibrator = null; 678 } 679 } 680 681 /** 682 * Starts wave animation. 683 * 684 */ ping()685 public void ping() { 686 if (mFeedbackCount > 0) { 687 boolean doWaveAnimation = true; 688 final AnimationBundle waveAnimations = mWaveAnimations; 689 690 // Don't do a wave if there's already one in progress 691 if (waveAnimations.size() > 0 && waveAnimations.get(0).animator.isRunning()) { 692 long t = waveAnimations.get(0).animator.getCurrentPlayTime(); 693 if (t < WAVE_ANIMATION_DURATION/2) { 694 doWaveAnimation = false; 695 } 696 } 697 698 if (doWaveAnimation) { 699 startWaveAnimation(); 700 } 701 } 702 } 703 stopAndHideWaveAnimation()704 private void stopAndHideWaveAnimation() { 705 mWaveAnimations.cancel(); 706 mPointCloud.waveManager.setAlpha(0.0f); 707 } 708 startWaveAnimation()709 private void startWaveAnimation() { 710 mWaveAnimations.cancel(); 711 mPointCloud.waveManager.setAlpha(1.0f); 712 mPointCloud.waveManager.setRadius(mHandleDrawable.getWidth()/2.0f); 713 mWaveAnimations.add(Tweener.to(mPointCloud.waveManager, WAVE_ANIMATION_DURATION, 714 "ease", Ease.Quad.easeOut, 715 "delay", 0, 716 "radius", 2.0f * mOuterRadius, 717 "onUpdate", mUpdateListener, 718 "onComplete", 719 new AnimatorListenerAdapter() { 720 public void onAnimationEnd(Animator animator) { 721 mPointCloud.waveManager.setRadius(0.0f); 722 mPointCloud.waveManager.setAlpha(0.0f); 723 } 724 })); 725 mWaveAnimations.start(); 726 } 727 728 /** 729 * Resets the widget to default state and cancels all animation. If animate is 'true', will 730 * animate objects into place. Otherwise, objects will snap back to place. 731 * 732 * @param animate 733 */ reset(boolean animate)734 public void reset(boolean animate) { 735 mGlowAnimations.stop(); 736 mTargetAnimations.stop(); 737 startBackgroundAnimation(0, 0.0f); 738 stopAndHideWaveAnimation(); 739 hideTargets(animate, false); 740 hideGlow(0, 0, 0.0f, null); 741 Tweener.reset(); 742 } 743 startBackgroundAnimation(int duration, float alpha)744 private void startBackgroundAnimation(int duration, float alpha) { 745 final Drawable background = getBackground(); 746 if (mAlwaysTrackFinger && background != null) { 747 if (mBackgroundAnimator != null) { 748 mBackgroundAnimator.animator.cancel(); 749 } 750 mBackgroundAnimator = Tweener.to(background, duration, 751 "ease", Ease.Cubic.easeIn, 752 "alpha", (int)(255.0f * alpha), 753 "delay", SHOW_ANIMATION_DELAY); 754 mBackgroundAnimator.animator.start(); 755 } 756 } 757 758 @Override onTouchEvent(MotionEvent event)759 public boolean onTouchEvent(MotionEvent event) { 760 final int action = event.getActionMasked(); 761 boolean handled = false; 762 switch (action) { 763 case MotionEvent.ACTION_POINTER_DOWN: 764 case MotionEvent.ACTION_DOWN: 765 if (DEBUG) Log.v(TAG, "*** DOWN ***"); 766 handleDown(event); 767 handleMove(event); 768 handled = true; 769 break; 770 771 case MotionEvent.ACTION_MOVE: 772 if (DEBUG) Log.v(TAG, "*** MOVE ***"); 773 handleMove(event); 774 handled = true; 775 break; 776 777 case MotionEvent.ACTION_POINTER_UP: 778 case MotionEvent.ACTION_UP: 779 if (DEBUG) Log.v(TAG, "*** UP ***"); 780 handleMove(event); 781 handleUp(event); 782 handled = true; 783 break; 784 785 case MotionEvent.ACTION_CANCEL: 786 if (DEBUG) Log.v(TAG, "*** CANCEL ***"); 787 handleMove(event); 788 handleCancel(event); 789 handled = true; 790 break; 791 792 } 793 invalidate(); 794 return handled ? true : super.onTouchEvent(event); 795 } 796 updateGlowPosition(float x, float y)797 private void updateGlowPosition(float x, float y) { 798 float dx = x - mOuterRing.getX(); 799 float dy = y - mOuterRing.getY(); 800 dx *= 1f / mRingScaleFactor; 801 dy *= 1f / mRingScaleFactor; 802 mPointCloud.glowManager.setX(mOuterRing.getX() + dx); 803 mPointCloud.glowManager.setY(mOuterRing.getY() + dy); 804 } 805 handleDown(MotionEvent event)806 private void handleDown(MotionEvent event) { 807 int actionIndex = event.getActionIndex(); 808 float eventX = event.getX(actionIndex); 809 float eventY = event.getY(actionIndex); 810 switchToState(STATE_START, eventX, eventY); 811 if (!trySwitchToFirstTouchState(eventX, eventY)) { 812 mDragging = false; 813 } else { 814 mPointerId = event.getPointerId(actionIndex); 815 updateGlowPosition(eventX, eventY); 816 } 817 } 818 handleUp(MotionEvent event)819 private void handleUp(MotionEvent event) { 820 if (DEBUG && mDragging) Log.v(TAG, "** Handle RELEASE"); 821 int actionIndex = event.getActionIndex(); 822 if (event.getPointerId(actionIndex) == mPointerId) { 823 switchToState(STATE_FINISH, event.getX(actionIndex), event.getY(actionIndex)); 824 } 825 } 826 handleCancel(MotionEvent event)827 private void handleCancel(MotionEvent event) { 828 if (DEBUG && mDragging) Log.v(TAG, "** Handle CANCEL"); 829 830 // Drop the active target if canceled. 831 mActiveTarget = -1; 832 833 int actionIndex = event.findPointerIndex(mPointerId); 834 actionIndex = actionIndex == -1 ? 0 : actionIndex; 835 switchToState(STATE_FINISH, event.getX(actionIndex), event.getY(actionIndex)); 836 } 837 handleMove(MotionEvent event)838 private void handleMove(MotionEvent event) { 839 int activeTarget = -1; 840 final int historySize = event.getHistorySize(); 841 ArrayList<TargetDrawable> targets = mTargetDrawables; 842 int ntargets = targets.size(); 843 float x = 0.0f; 844 float y = 0.0f; 845 float activeAngle = 0.0f; 846 int actionIndex = event.findPointerIndex(mPointerId); 847 848 if (actionIndex == -1) { 849 return; // no data for this pointer 850 } 851 852 for (int k = 0; k < historySize + 1; k++) { 853 float eventX = k < historySize ? event.getHistoricalX(actionIndex, k) 854 : event.getX(actionIndex); 855 float eventY = k < historySize ? event.getHistoricalY(actionIndex, k) 856 : event.getY(actionIndex); 857 // tx and ty are relative to wave center 858 float tx = eventX - mWaveCenterX; 859 float ty = eventY - mWaveCenterY; 860 float touchRadius = (float) Math.sqrt(dist2(tx, ty)); 861 final float scale = touchRadius > mOuterRadius ? mOuterRadius / touchRadius : 1.0f; 862 float limitX = tx * scale; 863 float limitY = ty * scale; 864 double angleRad = Math.atan2(-ty, tx); 865 866 if (!mDragging) { 867 trySwitchToFirstTouchState(eventX, eventY); 868 } 869 870 if (mDragging) { 871 // For multiple targets, snap to the one that matches 872 final float snapRadius = mRingScaleFactor * mOuterRadius - mSnapMargin; 873 final float snapDistance2 = snapRadius * snapRadius; 874 // Find first target in range 875 for (int i = 0; i < ntargets; i++) { 876 TargetDrawable target = targets.get(i); 877 878 double targetMinRad = mFirstItemOffset + (i - 0.5) * 2 * Math.PI / ntargets; 879 double targetMaxRad = mFirstItemOffset + (i + 0.5) * 2 * Math.PI / ntargets; 880 if (target.isEnabled()) { 881 boolean angleMatches = 882 (angleRad > targetMinRad && angleRad <= targetMaxRad) || 883 (angleRad + 2 * Math.PI > targetMinRad && 884 angleRad + 2 * Math.PI <= targetMaxRad) || 885 (angleRad - 2 * Math.PI > targetMinRad && 886 angleRad - 2 * Math.PI <= targetMaxRad); 887 if (angleMatches && (dist2(tx, ty) > snapDistance2)) { 888 activeTarget = i; 889 activeAngle = (float) -angleRad; 890 } 891 } 892 } 893 } 894 x = limitX; 895 y = limitY; 896 } 897 898 if (!mDragging) { 899 return; 900 } 901 902 if (activeTarget != -1) { 903 switchToState(STATE_SNAP, x,y); 904 updateGlowPosition(x, y); 905 } else { 906 switchToState(STATE_TRACKING, x, y); 907 updateGlowPosition(x, y); 908 } 909 910 if (mActiveTarget != activeTarget) { 911 // Defocus the old target 912 if (mActiveTarget != -1) { 913 TargetDrawable target = targets.get(mActiveTarget); 914 if (target.hasState(TargetDrawable.STATE_FOCUSED)) { 915 target.setState(TargetDrawable.STATE_INACTIVE); 916 } 917 if (mMagneticTargets) { 918 updateTargetPosition(mActiveTarget, mWaveCenterX, mWaveCenterY); 919 } 920 } 921 // Focus the new target 922 if (activeTarget != -1) { 923 TargetDrawable target = targets.get(activeTarget); 924 if (target.hasState(TargetDrawable.STATE_FOCUSED)) { 925 target.setState(TargetDrawable.STATE_FOCUSED); 926 } 927 if (mMagneticTargets) { 928 updateTargetPosition(activeTarget, mWaveCenterX, mWaveCenterY, activeAngle); 929 } 930 if (AccessibilityManager.getInstance(mContext).isEnabled()) { 931 String targetContentDescription = getTargetDescription(activeTarget); 932 announceForAccessibility(targetContentDescription); 933 } 934 } 935 } 936 mActiveTarget = activeTarget; 937 } 938 939 @Override 940 public boolean onHoverEvent(MotionEvent event) { 941 if (AccessibilityManager.getInstance(mContext).isTouchExplorationEnabled()) { 942 final int action = event.getAction(); 943 switch (action) { 944 case MotionEvent.ACTION_HOVER_ENTER: 945 event.setAction(MotionEvent.ACTION_DOWN); 946 break; 947 case MotionEvent.ACTION_HOVER_MOVE: 948 event.setAction(MotionEvent.ACTION_MOVE); 949 break; 950 case MotionEvent.ACTION_HOVER_EXIT: 951 event.setAction(MotionEvent.ACTION_UP); 952 break; 953 } 954 onTouchEvent(event); 955 event.setAction(action); 956 } 957 super.onHoverEvent(event); 958 return true; 959 } 960 961 /** 962 * Sets the current grabbed state, and dispatches a grabbed state change 963 * event to our listener. 964 */ 965 private void setGrabbedState(int newState) { 966 if (newState != mGrabbedState) { 967 if (newState != OnTriggerListener.NO_HANDLE) { 968 vibrate(); 969 } 970 mGrabbedState = newState; 971 if (mOnTriggerListener != null) { 972 if (newState == OnTriggerListener.NO_HANDLE) { 973 mOnTriggerListener.onReleased(this, OnTriggerListener.CENTER_HANDLE); 974 } else { 975 mOnTriggerListener.onGrabbed(this, OnTriggerListener.CENTER_HANDLE); 976 } 977 mOnTriggerListener.onGrabbedStateChange(this, newState); 978 } 979 } 980 } 981 982 private boolean trySwitchToFirstTouchState(float x, float y) { 983 final float tx = x - mWaveCenterX; 984 final float ty = y - mWaveCenterY; 985 if (mAlwaysTrackFinger || dist2(tx,ty) <= getScaledGlowRadiusSquared()) { 986 if (DEBUG) Log.v(TAG, "** Handle HIT"); 987 switchToState(STATE_FIRST_TOUCH, x, y); 988 updateGlowPosition(tx, ty); 989 mDragging = true; 990 return true; 991 } 992 return false; 993 } 994 995 private void assignDefaultsIfNeeded() { 996 if (mOuterRadius == 0.0f) { 997 mOuterRadius = Math.max(mOuterRing.getWidth(), mOuterRing.getHeight())/2.0f; 998 } 999 if (mSnapMargin == 0.0f) { 1000 mSnapMargin = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 1001 SNAP_MARGIN_DEFAULT, getContext().getResources().getDisplayMetrics()); 1002 } 1003 if (mInnerRadius == 0.0f) { 1004 mInnerRadius = mHandleDrawable.getWidth() / 10.0f; 1005 } 1006 } 1007 1008 private void computeInsets(int dx, int dy) { 1009 final int layoutDirection = getLayoutDirection(); 1010 final int absoluteGravity = Gravity.getAbsoluteGravity(mGravity, layoutDirection); 1011 1012 switch (absoluteGravity & Gravity.HORIZONTAL_GRAVITY_MASK) { 1013 case Gravity.LEFT: 1014 mHorizontalInset = 0; 1015 break; 1016 case Gravity.RIGHT: 1017 mHorizontalInset = dx; 1018 break; 1019 case Gravity.CENTER_HORIZONTAL: 1020 default: 1021 mHorizontalInset = dx / 2; 1022 break; 1023 } 1024 switch (absoluteGravity & Gravity.VERTICAL_GRAVITY_MASK) { 1025 case Gravity.TOP: 1026 mVerticalInset = 0; 1027 break; 1028 case Gravity.BOTTOM: 1029 mVerticalInset = dy; 1030 break; 1031 case Gravity.CENTER_VERTICAL: 1032 default: 1033 mVerticalInset = dy / 2; 1034 break; 1035 } 1036 } 1037 1038 /** 1039 * Given the desired width and height of the ring and the allocated width and height, compute 1040 * how much we need to scale the ring. 1041 */ 1042 private float computeScaleFactor(int desiredWidth, int desiredHeight, 1043 int actualWidth, int actualHeight) { 1044 1045 // Return unity if scaling is not allowed. 1046 if (!mAllowScaling) return 1f; 1047 1048 final int layoutDirection = getLayoutDirection(); 1049 final int absoluteGravity = Gravity.getAbsoluteGravity(mGravity, layoutDirection); 1050 1051 float scaleX = 1f; 1052 float scaleY = 1f; 1053 1054 // We use the gravity as a cue for whether we want to scale on a particular axis. 1055 // We only scale to fit horizontally if we're not pinned to the left or right. Likewise, 1056 // we only scale to fit vertically if we're not pinned to the top or bottom. In these 1057 // cases, we want the ring to hang off the side or top/bottom, respectively. 1058 switch (absoluteGravity & Gravity.HORIZONTAL_GRAVITY_MASK) { 1059 case Gravity.LEFT: 1060 case Gravity.RIGHT: 1061 break; 1062 case Gravity.CENTER_HORIZONTAL: 1063 default: 1064 if (desiredWidth > actualWidth) { 1065 scaleX = (1f * actualWidth - mMaxTargetWidth) / 1066 (desiredWidth - mMaxTargetWidth); 1067 } 1068 break; 1069 } 1070 switch (absoluteGravity & Gravity.VERTICAL_GRAVITY_MASK) { 1071 case Gravity.TOP: 1072 case Gravity.BOTTOM: 1073 break; 1074 case Gravity.CENTER_VERTICAL: 1075 default: 1076 if (desiredHeight > actualHeight) { 1077 scaleY = (1f * actualHeight - mMaxTargetHeight) / 1078 (desiredHeight - mMaxTargetHeight); 1079 } 1080 break; 1081 } 1082 return Math.min(scaleX, scaleY); 1083 } 1084 1085 @Override onMeasure(int widthMeasureSpec, int heightMeasureSpec)1086 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 1087 final int minimumWidth = getSuggestedMinimumWidth(); 1088 final int minimumHeight = getSuggestedMinimumHeight(); 1089 int computedWidth = resolveMeasured(widthMeasureSpec, minimumWidth); 1090 int computedHeight = resolveMeasured(heightMeasureSpec, minimumHeight); 1091 1092 mRingScaleFactor = computeScaleFactor(minimumWidth, minimumHeight, 1093 computedWidth, computedHeight); 1094 1095 int scaledWidth = getScaledSuggestedMinimumWidth(); 1096 int scaledHeight = getScaledSuggestedMinimumHeight(); 1097 1098 computeInsets(computedWidth - scaledWidth, computedHeight - scaledHeight); 1099 setMeasuredDimension(computedWidth, computedHeight); 1100 } 1101 getRingWidth()1102 private float getRingWidth() { 1103 return mRingScaleFactor * Math.max(mOuterRing.getWidth(), 2 * mOuterRadius); 1104 } 1105 getRingHeight()1106 private float getRingHeight() { 1107 return mRingScaleFactor * Math.max(mOuterRing.getHeight(), 2 * mOuterRadius); 1108 } 1109 1110 @Override onLayout(boolean changed, int left, int top, int right, int bottom)1111 protected void onLayout(boolean changed, int left, int top, int right, int bottom) { 1112 super.onLayout(changed, left, top, right, bottom); 1113 final int width = right - left; 1114 final int height = bottom - top; 1115 1116 // Target placement width/height. This puts the targets on the greater of the ring 1117 // width or the specified outer radius. 1118 final float placementWidth = getRingWidth(); 1119 final float placementHeight = getRingHeight(); 1120 float newWaveCenterX = mHorizontalInset 1121 + Math.max(width, mMaxTargetWidth + placementWidth) / 2; 1122 float newWaveCenterY = mVerticalInset 1123 + Math.max(height, + mMaxTargetHeight + placementHeight) / 2; 1124 1125 if (mInitialLayout) { 1126 stopAndHideWaveAnimation(); 1127 hideTargets(false, false); 1128 mInitialLayout = false; 1129 } 1130 1131 mOuterRing.setPositionX(newWaveCenterX); 1132 mOuterRing.setPositionY(newWaveCenterY); 1133 1134 mPointCloud.setScale(mRingScaleFactor); 1135 1136 mHandleDrawable.setPositionX(newWaveCenterX); 1137 mHandleDrawable.setPositionY(newWaveCenterY); 1138 1139 updateTargetPositions(newWaveCenterX, newWaveCenterY); 1140 updatePointCloudPosition(newWaveCenterX, newWaveCenterY); 1141 updateGlowPosition(newWaveCenterX, newWaveCenterY); 1142 1143 mWaveCenterX = newWaveCenterX; 1144 mWaveCenterY = newWaveCenterY; 1145 1146 if (DEBUG) dump(); 1147 } 1148 updateTargetPosition(int i, float centerX, float centerY)1149 private void updateTargetPosition(int i, float centerX, float centerY) { 1150 final float angle = getAngle(getSliceAngle(), i); 1151 updateTargetPosition(i, centerX, centerY, angle); 1152 } 1153 updateTargetPosition(int i, float centerX, float centerY, float angle)1154 private void updateTargetPosition(int i, float centerX, float centerY, float angle) { 1155 final float placementRadiusX = getRingWidth() / 2; 1156 final float placementRadiusY = getRingHeight() / 2; 1157 if (i >= 0) { 1158 ArrayList<TargetDrawable> targets = mTargetDrawables; 1159 final TargetDrawable targetIcon = targets.get(i); 1160 targetIcon.setPositionX(centerX); 1161 targetIcon.setPositionY(centerY); 1162 targetIcon.setX(placementRadiusX * (float) Math.cos(angle)); 1163 targetIcon.setY(placementRadiusY * (float) Math.sin(angle)); 1164 } 1165 } 1166 updateTargetPositions(float centerX, float centerY)1167 private void updateTargetPositions(float centerX, float centerY) { 1168 updateTargetPositions(centerX, centerY, false); 1169 } 1170 updateTargetPositions(float centerX, float centerY, boolean skipActive)1171 private void updateTargetPositions(float centerX, float centerY, boolean skipActive) { 1172 final int size = mTargetDrawables.size(); 1173 final float alpha = getSliceAngle(); 1174 // Reposition the target drawables if the view changed. 1175 for (int i = 0; i < size; i++) { 1176 if (!skipActive || i != mActiveTarget) { 1177 updateTargetPosition(i, centerX, centerY, getAngle(alpha, i)); 1178 } 1179 } 1180 } 1181 getAngle(float alpha, int i)1182 private float getAngle(float alpha, int i) { 1183 return mFirstItemOffset + alpha * i; 1184 } 1185 getSliceAngle()1186 private float getSliceAngle() { 1187 return (float) (-2.0f * Math.PI / mTargetDrawables.size()); 1188 } 1189 updatePointCloudPosition(float centerX, float centerY)1190 private void updatePointCloudPosition(float centerX, float centerY) { 1191 mPointCloud.setCenter(centerX, centerY); 1192 } 1193 1194 @Override onDraw(Canvas canvas)1195 protected void onDraw(Canvas canvas) { 1196 mPointCloud.draw(canvas); 1197 mOuterRing.draw(canvas); 1198 final int ntargets = mTargetDrawables.size(); 1199 for (int i = 0; i < ntargets; i++) { 1200 TargetDrawable target = mTargetDrawables.get(i); 1201 if (target != null) { 1202 target.draw(canvas); 1203 } 1204 } 1205 mHandleDrawable.draw(canvas); 1206 } 1207 setOnTriggerListener(OnTriggerListener listener)1208 public void setOnTriggerListener(OnTriggerListener listener) { 1209 mOnTriggerListener = listener; 1210 } 1211 square(float d)1212 private float square(float d) { 1213 return d * d; 1214 } 1215 dist2(float dx, float dy)1216 private float dist2(float dx, float dy) { 1217 return dx*dx + dy*dy; 1218 } 1219 getScaledGlowRadiusSquared()1220 private float getScaledGlowRadiusSquared() { 1221 final float scaledTapRadius; 1222 if (AccessibilityManager.getInstance(mContext).isEnabled()) { 1223 scaledTapRadius = TAP_RADIUS_SCALE_ACCESSIBILITY_ENABLED * mGlowRadius; 1224 } else { 1225 scaledTapRadius = mGlowRadius; 1226 } 1227 return square(scaledTapRadius); 1228 } 1229 announceTargets()1230 private void announceTargets() { 1231 StringBuilder utterance = new StringBuilder(); 1232 final int targetCount = mTargetDrawables.size(); 1233 for (int i = 0; i < targetCount; i++) { 1234 String targetDescription = getTargetDescription(i); 1235 String directionDescription = getDirectionDescription(i); 1236 if (!TextUtils.isEmpty(targetDescription) 1237 && !TextUtils.isEmpty(directionDescription)) { 1238 String text = String.format(directionDescription, targetDescription); 1239 utterance.append(text); 1240 } 1241 } 1242 if (utterance.length() > 0) { 1243 announceForAccessibility(utterance.toString()); 1244 } 1245 } 1246 getTargetDescription(int index)1247 private String getTargetDescription(int index) { 1248 if (mTargetDescriptions == null || mTargetDescriptions.isEmpty()) { 1249 mTargetDescriptions = loadDescriptions(mTargetDescriptionsResourceId); 1250 if (mTargetDrawables.size() != mTargetDescriptions.size()) { 1251 Log.w(TAG, "The number of target drawables must be" 1252 + " equal to the number of target descriptions."); 1253 return null; 1254 } 1255 } 1256 return mTargetDescriptions.get(index); 1257 } 1258 getDirectionDescription(int index)1259 private String getDirectionDescription(int index) { 1260 if (mDirectionDescriptions == null || mDirectionDescriptions.isEmpty()) { 1261 mDirectionDescriptions = loadDescriptions(mDirectionDescriptionsResourceId); 1262 if (mTargetDrawables.size() != mDirectionDescriptions.size()) { 1263 Log.w(TAG, "The number of target drawables must be" 1264 + " equal to the number of direction descriptions."); 1265 return null; 1266 } 1267 } 1268 return mDirectionDescriptions.get(index); 1269 } 1270 loadDescriptions(int resourceId)1271 private ArrayList<String> loadDescriptions(int resourceId) { 1272 TypedArray array = getContext().getResources().obtainTypedArray(resourceId); 1273 final int count = array.length(); 1274 ArrayList<String> targetContentDescriptions = new ArrayList<String>(count); 1275 for (int i = 0; i < count; i++) { 1276 String contentDescription = array.getString(i); 1277 targetContentDescriptions.add(contentDescription); 1278 } 1279 array.recycle(); 1280 return targetContentDescriptions; 1281 } 1282 getResourceIdForTarget(int index)1283 public int getResourceIdForTarget(int index) { 1284 final TargetDrawable drawable = mTargetDrawables.get(index); 1285 return drawable == null ? 0 : drawable.getResourceId(); 1286 } 1287 setEnableTarget(int resourceId, boolean enabled)1288 public void setEnableTarget(int resourceId, boolean enabled) { 1289 for (int i = 0; i < mTargetDrawables.size(); i++) { 1290 final TargetDrawable target = mTargetDrawables.get(i); 1291 if (target.getResourceId() == resourceId) { 1292 target.setEnabled(enabled); 1293 break; // should never be more than one match 1294 } 1295 } 1296 } 1297 1298 /** 1299 * Gets the position of a target in the array that matches the given resource. 1300 * @param resourceId 1301 * @return the index or -1 if not found 1302 */ getTargetPosition(int resourceId)1303 public int getTargetPosition(int resourceId) { 1304 for (int i = 0; i < mTargetDrawables.size(); i++) { 1305 final TargetDrawable target = mTargetDrawables.get(i); 1306 if (target.getResourceId() == resourceId) { 1307 return i; // should never be more than one match 1308 } 1309 } 1310 return -1; 1311 } 1312 replaceTargetDrawables(Resources res, int existingResourceId, int newResourceId)1313 private boolean replaceTargetDrawables(Resources res, int existingResourceId, 1314 int newResourceId) { 1315 if (existingResourceId == 0 || newResourceId == 0) { 1316 return false; 1317 } 1318 1319 boolean result = false; 1320 final ArrayList<TargetDrawable> drawables = mTargetDrawables; 1321 final int size = drawables.size(); 1322 for (int i = 0; i < size; i++) { 1323 final TargetDrawable target = drawables.get(i); 1324 if (target != null && target.getResourceId() == existingResourceId) { 1325 target.setDrawable(res, newResourceId); 1326 result = true; 1327 } 1328 } 1329 1330 if (result) { 1331 requestLayout(); // in case any given drawable's size changes 1332 } 1333 1334 return result; 1335 } 1336 1337 /** 1338 * Searches the given package for a resource to use to replace the Drawable on the 1339 * target with the given resource id 1340 * @param component of the .apk that contains the resource 1341 * @param name of the metadata in the .apk 1342 * @param existingResId the resource id of the target to search for 1343 * @return true if found in the given package and replaced at least one target Drawables 1344 */ replaceTargetDrawablesIfPresent(ComponentName component, String name, int existingResId)1345 public boolean replaceTargetDrawablesIfPresent(ComponentName component, String name, 1346 int existingResId) { 1347 if (existingResId == 0) return false; 1348 1349 boolean replaced = false; 1350 if (component != null) { 1351 try { 1352 PackageManager packageManager = mContext.getPackageManager(); 1353 // Look for the search icon specified in the activity meta-data 1354 Bundle metaData = packageManager.getActivityInfo( 1355 component, PackageManager.GET_META_DATA).metaData; 1356 if (metaData != null) { 1357 int iconResId = metaData.getInt(name); 1358 if (iconResId != 0) { 1359 Resources res = packageManager.getResourcesForActivity(component); 1360 replaced = replaceTargetDrawables(res, existingResId, iconResId); 1361 } 1362 } 1363 } catch (NameNotFoundException e) { 1364 Log.w(TAG, "Failed to swap drawable; " 1365 + component.flattenToShortString() + " not found", e); 1366 } catch (Resources.NotFoundException nfe) { 1367 Log.w(TAG, "Failed to swap drawable from " 1368 + component.flattenToShortString(), nfe); 1369 } 1370 } 1371 if (!replaced) { 1372 // Restore the original drawable 1373 replaceTargetDrawables(mContext.getResources(), existingResId, existingResId); 1374 } 1375 return replaced; 1376 } 1377 } 1378