• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2008 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.widget;
18 
19 import android.content.BroadcastReceiver;
20 import android.content.Context;
21 import android.content.Intent;
22 import android.content.IntentFilter;
23 import android.graphics.PixelFormat;
24 import android.graphics.Rect;
25 import android.os.Handler;
26 import android.os.Message;
27 import android.util.Log;
28 import android.view.Gravity;
29 import android.view.KeyEvent;
30 import android.view.LayoutInflater;
31 import android.view.MotionEvent;
32 import android.view.View;
33 import android.view.ViewConfiguration;
34 import android.view.ViewGroup;
35 import android.view.ViewParent;
36 import android.view.ViewRootImpl;
37 import android.view.WindowManager;
38 import android.view.View.OnClickListener;
39 import android.view.WindowManager.LayoutParams;
40 
41 /*
42  * Implementation notes:
43  * - The zoom controls are displayed in their own window.
44  *   (Easier for the client and better performance)
45  * - This window is never touchable, and by default is not focusable.
46  *   Its rect is quite big (fills horizontally) but has empty space between the
47  *   edges and center.  Touches there should be given to the owner.  Instead of
48  *   having the window touchable and dispatching these empty touch events to the
49  *   owner, we set the window to not touchable and steal events from owner
50  *   via onTouchListener.
51  * - To make the buttons clickable, it attaches an OnTouchListener to the owner
52  *   view and does the hit detection locally (attaches when visible, detaches when invisible).
53  * - When it is focusable, it forwards uninteresting events to the owner view's
54  *   view hierarchy.
55  */
56 /**
57  * The {@link ZoomButtonsController} handles showing and hiding the zoom
58  * controls and positioning it relative to an owner view. It also gives the
59  * client access to the zoom controls container, allowing for additional
60  * accessory buttons to be shown in the zoom controls window.
61  * <p>
62  * Typically, clients should call {@link #setVisible(boolean) setVisible(true)}
63  * on a touch down or move (no need to call {@link #setVisible(boolean)
64  * setVisible(false)} since it will time out on its own). Also, whenever the
65  * owner cannot be zoomed further, the client should update
66  * {@link #setZoomInEnabled(boolean)} and {@link #setZoomOutEnabled(boolean)}.
67  * <p>
68  * If you are using this with a custom View, please call
69  * {@link #setVisible(boolean) setVisible(false)} from
70  * {@link View#onDetachedFromWindow} and from {@link View#onVisibilityChanged}
71  * when <code>visibility != View.VISIBLE</code>.
72  *
73  */
74 public class ZoomButtonsController implements View.OnTouchListener {
75 
76     private static final String TAG = "ZoomButtonsController";
77 
78     private static final int ZOOM_CONTROLS_TIMEOUT =
79             (int) ViewConfiguration.getZoomControlsTimeout();
80 
81     private static final int ZOOM_CONTROLS_TOUCH_PADDING = 20;
82     private int mTouchPaddingScaledSq;
83 
84     private final Context mContext;
85     private final WindowManager mWindowManager;
86     private boolean mAutoDismissControls = true;
87 
88     /**
89      * The view that is being zoomed by this zoom controller.
90      */
91     private final View mOwnerView;
92 
93     /**
94      * The location of the owner view on the screen. This is recalculated
95      * each time the zoom controller is shown.
96      */
97     private final int[] mOwnerViewRawLocation = new int[2];
98 
99     /**
100      * The container that is added as a window.
101      */
102     private final FrameLayout mContainer;
103     private LayoutParams mContainerLayoutParams;
104     private final int[] mContainerRawLocation = new int[2];
105 
106     private ZoomControls mControls;
107 
108     /**
109      * The view (or null) that should receive touch events. This will get set if
110      * the touch down hits the container. It will be reset on the touch up.
111      */
112     private View mTouchTargetView;
113     /**
114      * The {@link #mTouchTargetView}'s location in window, set on touch down.
115      */
116     private final int[] mTouchTargetWindowLocation = new int[2];
117 
118     /**
119      * If the zoom controller is dismissed but the user is still in a touch
120      * interaction, we set this to true. This will ignore all touch events until
121      * up/cancel, and then set the owner's touch listener to null.
122      * <p>
123      * Otherwise, the owner view would get mismatched events (i.e., touch move
124      * even though it never got the touch down.)
125      */
126     private boolean mReleaseTouchListenerOnUp;
127 
128     /** Whether the container has been added to the window manager. */
129     private boolean mIsVisible;
130 
131     private final Rect mTempRect = new Rect();
132     private final int[] mTempIntArray = new int[2];
133 
134     private OnZoomListener mCallback;
135 
136     /**
137      * When showing the zoom, we add the view as a new window. However, there is
138      * logic that needs to know the size of the zoom which is determined after
139      * it's laid out. Therefore, we must post this logic onto the UI thread so
140      * it will be exceuted AFTER the layout. This is the logic.
141      */
142     private Runnable mPostedVisibleInitializer;
143 
144     private final IntentFilter mConfigurationChangedFilter =
145             new IntentFilter(Intent.ACTION_CONFIGURATION_CHANGED);
146 
147     /**
148      * Needed to reposition the zoom controls after configuration changes.
149      */
150     private final BroadcastReceiver mConfigurationChangedReceiver = new BroadcastReceiver() {
151         @Override
152         public void onReceive(Context context, Intent intent) {
153             if (!mIsVisible) return;
154 
155             mHandler.removeMessages(MSG_POST_CONFIGURATION_CHANGED);
156             mHandler.sendEmptyMessage(MSG_POST_CONFIGURATION_CHANGED);
157         }
158     };
159 
160     /** When configuration changes, this is called after the UI thread is idle. */
161     private static final int MSG_POST_CONFIGURATION_CHANGED = 2;
162     /** Used to delay the zoom controller dismissal. */
163     private static final int MSG_DISMISS_ZOOM_CONTROLS = 3;
164     /**
165      * If setVisible(true) is called and the owner view's window token is null,
166      * we delay the setVisible(true) call until it is not null.
167      */
168     private static final int MSG_POST_SET_VISIBLE = 4;
169 
170     private final Handler mHandler = new Handler() {
171         @Override
172         public void handleMessage(Message msg) {
173             switch (msg.what) {
174                 case MSG_POST_CONFIGURATION_CHANGED:
175                     onPostConfigurationChanged();
176                     break;
177 
178                 case MSG_DISMISS_ZOOM_CONTROLS:
179                     setVisible(false);
180                     break;
181 
182                 case MSG_POST_SET_VISIBLE:
183                     if (mOwnerView.getWindowToken() == null) {
184                         // Doh, it is still null, just ignore the set visible call
185                         Log.e(TAG,
186                                 "Cannot make the zoom controller visible if the owner view is " +
187                                 "not attached to a window.");
188                     } else {
189                         setVisible(true);
190                     }
191                     break;
192             }
193 
194         }
195     };
196 
197     /**
198      * Constructor for the {@link ZoomButtonsController}.
199      *
200      * @param ownerView The view that is being zoomed by the zoom controls. The
201      *            zoom controls will be displayed aligned with this view.
202      */
ZoomButtonsController(View ownerView)203     public ZoomButtonsController(View ownerView) {
204         mContext = ownerView.getContext();
205         mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
206         mOwnerView = ownerView;
207 
208         mTouchPaddingScaledSq = (int)
209                 (ZOOM_CONTROLS_TOUCH_PADDING * mContext.getResources().getDisplayMetrics().density);
210         mTouchPaddingScaledSq *= mTouchPaddingScaledSq;
211 
212         mContainer = createContainer();
213     }
214 
215     /**
216      * Whether to enable the zoom in control.
217      *
218      * @param enabled Whether to enable the zoom in control.
219      */
setZoomInEnabled(boolean enabled)220     public void setZoomInEnabled(boolean enabled) {
221         mControls.setIsZoomInEnabled(enabled);
222     }
223 
224     /**
225      * Whether to enable the zoom out control.
226      *
227      * @param enabled Whether to enable the zoom out control.
228      */
setZoomOutEnabled(boolean enabled)229     public void setZoomOutEnabled(boolean enabled) {
230         mControls.setIsZoomOutEnabled(enabled);
231     }
232 
233     /**
234      * Sets the delay between zoom callbacks as the user holds a zoom button.
235      *
236      * @param speed The delay in milliseconds between zoom callbacks.
237      */
setZoomSpeed(long speed)238     public void setZoomSpeed(long speed) {
239         mControls.setZoomSpeed(speed);
240     }
241 
createContainer()242     private FrameLayout createContainer() {
243         LayoutParams lp = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
244         // Controls are positioned BOTTOM | CENTER with respect to the owner view.
245         lp.gravity = Gravity.TOP | Gravity.LEFT;
246         lp.flags = LayoutParams.FLAG_NOT_TOUCHABLE |
247                 LayoutParams.FLAG_NOT_FOCUSABLE |
248                 LayoutParams.FLAG_LAYOUT_NO_LIMITS |
249                 LayoutParams.FLAG_ALT_FOCUSABLE_IM;
250         lp.height = LayoutParams.WRAP_CONTENT;
251         lp.width = LayoutParams.MATCH_PARENT;
252         lp.type = LayoutParams.TYPE_APPLICATION_PANEL;
253         lp.format = PixelFormat.TRANSLUCENT;
254         lp.windowAnimations = com.android.internal.R.style.Animation_ZoomButtons;
255         mContainerLayoutParams = lp;
256 
257         FrameLayout container = new Container(mContext);
258         container.setLayoutParams(lp);
259         container.setMeasureAllChildren(true);
260 
261         LayoutInflater inflater = (LayoutInflater) mContext
262                 .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
263         inflater.inflate(com.android.internal.R.layout.zoom_container, container);
264 
265         mControls = (ZoomControls) container.findViewById(com.android.internal.R.id.zoomControls);
266         mControls.setOnZoomInClickListener(new OnClickListener() {
267             public void onClick(View v) {
268                 dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
269                 if (mCallback != null) mCallback.onZoom(true);
270             }
271         });
272         mControls.setOnZoomOutClickListener(new OnClickListener() {
273             public void onClick(View v) {
274                 dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
275                 if (mCallback != null) mCallback.onZoom(false);
276             }
277         });
278 
279         return container;
280     }
281 
282     /**
283      * Sets the {@link OnZoomListener} listener that receives callbacks to zoom.
284      *
285      * @param listener The listener that will be told to zoom.
286      */
setOnZoomListener(OnZoomListener listener)287     public void setOnZoomListener(OnZoomListener listener) {
288         mCallback = listener;
289     }
290 
291     /**
292      * Sets whether the zoom controls should be focusable. If the controls are
293      * focusable, then trackball and arrow key interactions are possible.
294      * Otherwise, only touch interactions are possible.
295      *
296      * @param focusable Whether the zoom controls should be focusable.
297      */
setFocusable(boolean focusable)298     public void setFocusable(boolean focusable) {
299         int oldFlags = mContainerLayoutParams.flags;
300         if (focusable) {
301             mContainerLayoutParams.flags &= ~LayoutParams.FLAG_NOT_FOCUSABLE;
302         } else {
303             mContainerLayoutParams.flags |= LayoutParams.FLAG_NOT_FOCUSABLE;
304         }
305 
306         if ((mContainerLayoutParams.flags != oldFlags) && mIsVisible) {
307             mWindowManager.updateViewLayout(mContainer, mContainerLayoutParams);
308         }
309     }
310 
311     /**
312      * Whether the zoom controls will be automatically dismissed after showing.
313      *
314      * @return Whether the zoom controls will be auto dismissed after showing.
315      */
isAutoDismissed()316     public boolean isAutoDismissed() {
317         return mAutoDismissControls;
318     }
319 
320     /**
321      * Sets whether the zoom controls will be automatically dismissed after
322      * showing.
323      */
setAutoDismissed(boolean autoDismiss)324     public void setAutoDismissed(boolean autoDismiss) {
325         if (mAutoDismissControls == autoDismiss) return;
326         mAutoDismissControls = autoDismiss;
327     }
328 
329     /**
330      * Whether the zoom controls are visible to the user.
331      *
332      * @return Whether the zoom controls are visible to the user.
333      */
isVisible()334     public boolean isVisible() {
335         return mIsVisible;
336     }
337 
338     /**
339      * Sets whether the zoom controls should be visible to the user.
340      *
341      * @param visible Whether the zoom controls should be visible to the user.
342      */
setVisible(boolean visible)343     public void setVisible(boolean visible) {
344 
345         if (visible) {
346             if (mOwnerView.getWindowToken() == null) {
347                 /*
348                  * We need a window token to show ourselves, maybe the owner's
349                  * window hasn't been created yet but it will have been by the
350                  * time the looper is idle, so post the setVisible(true) call.
351                  */
352                 if (!mHandler.hasMessages(MSG_POST_SET_VISIBLE)) {
353                     mHandler.sendEmptyMessage(MSG_POST_SET_VISIBLE);
354                 }
355                 return;
356             }
357 
358             dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
359         }
360 
361         if (mIsVisible == visible) {
362             return;
363         }
364         mIsVisible = visible;
365 
366         if (visible) {
367             if (mContainerLayoutParams.token == null) {
368                 mContainerLayoutParams.token = mOwnerView.getWindowToken();
369             }
370 
371             mWindowManager.addView(mContainer, mContainerLayoutParams);
372 
373             if (mPostedVisibleInitializer == null) {
374                 mPostedVisibleInitializer = new Runnable() {
375                     public void run() {
376                         refreshPositioningVariables();
377 
378                         if (mCallback != null) {
379                             mCallback.onVisibilityChanged(true);
380                         }
381                     }
382                 };
383             }
384 
385             mHandler.post(mPostedVisibleInitializer);
386 
387             // Handle configuration changes when visible
388             mContext.registerReceiver(mConfigurationChangedReceiver, mConfigurationChangedFilter);
389 
390             // Steal touches events from the owner
391             mOwnerView.setOnTouchListener(this);
392             mReleaseTouchListenerOnUp = false;
393 
394         } else {
395             // Don't want to steal any more touches
396             if (mTouchTargetView != null) {
397                 // We are still stealing the touch events for this touch
398                 // sequence, so release the touch listener later
399                 mReleaseTouchListenerOnUp = true;
400             } else {
401                 mOwnerView.setOnTouchListener(null);
402             }
403 
404             // No longer care about configuration changes
405             mContext.unregisterReceiver(mConfigurationChangedReceiver);
406 
407             mWindowManager.removeView(mContainer);
408             mHandler.removeCallbacks(mPostedVisibleInitializer);
409 
410             if (mCallback != null) {
411                 mCallback.onVisibilityChanged(false);
412             }
413         }
414 
415     }
416 
417     /**
418      * Gets the container that is the parent of the zoom controls.
419      * <p>
420      * The client can add other views to this container to link them with the
421      * zoom controls.
422      *
423      * @return The container of the zoom controls. It will be a layout that
424      *         respects the gravity of a child's layout parameters.
425      */
getContainer()426     public ViewGroup getContainer() {
427         return mContainer;
428     }
429 
430     /**
431      * Gets the view for the zoom controls.
432      *
433      * @return The zoom controls view.
434      */
getZoomControls()435     public View getZoomControls() {
436         return mControls;
437     }
438 
dismissControlsDelayed(int delay)439     private void dismissControlsDelayed(int delay) {
440         if (mAutoDismissControls) {
441             mHandler.removeMessages(MSG_DISMISS_ZOOM_CONTROLS);
442             mHandler.sendEmptyMessageDelayed(MSG_DISMISS_ZOOM_CONTROLS, delay);
443         }
444     }
445 
refreshPositioningVariables()446     private void refreshPositioningVariables() {
447         // if the mOwnerView is detached from window then skip.
448         if (mOwnerView.getWindowToken() == null) return;
449 
450         // Position the zoom controls on the bottom of the owner view.
451         int ownerHeight = mOwnerView.getHeight();
452         int ownerWidth = mOwnerView.getWidth();
453         // The gap between the top of the owner and the top of the container
454         int containerOwnerYOffset = ownerHeight - mContainer.getHeight();
455 
456         // Calculate the owner view's bounds
457         mOwnerView.getLocationOnScreen(mOwnerViewRawLocation);
458         mContainerRawLocation[0] = mOwnerViewRawLocation[0];
459         mContainerRawLocation[1] = mOwnerViewRawLocation[1] + containerOwnerYOffset;
460 
461         int[] ownerViewWindowLoc = mTempIntArray;
462         mOwnerView.getLocationInWindow(ownerViewWindowLoc);
463 
464         // lp.x and lp.y should be relative to the owner's window top-left
465         mContainerLayoutParams.x = ownerViewWindowLoc[0];
466         mContainerLayoutParams.width = ownerWidth;
467         mContainerLayoutParams.y = ownerViewWindowLoc[1] + containerOwnerYOffset;
468         if (mIsVisible) {
469             mWindowManager.updateViewLayout(mContainer, mContainerLayoutParams);
470         }
471 
472     }
473 
474     /* This will only be called when the container has focus. */
onContainerKey(KeyEvent event)475     private boolean onContainerKey(KeyEvent event) {
476         int keyCode = event.getKeyCode();
477         if (isInterestingKey(keyCode)) {
478 
479             if (keyCode == KeyEvent.KEYCODE_BACK) {
480                 if (event.getAction() == KeyEvent.ACTION_DOWN
481                         && event.getRepeatCount() == 0) {
482                     if (mOwnerView != null) {
483                         KeyEvent.DispatcherState ds = mOwnerView.getKeyDispatcherState();
484                         if (ds != null) {
485                             ds.startTracking(event, this);
486                         }
487                     }
488                     return true;
489                 } else if (event.getAction() == KeyEvent.ACTION_UP
490                         && event.isTracking() && !event.isCanceled()) {
491                     setVisible(false);
492                     return true;
493                 }
494 
495             } else {
496                 dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
497             }
498 
499             // Let the container handle the key
500             return false;
501 
502         } else {
503 
504             ViewRootImpl viewRoot = mOwnerView.getViewRootImpl();
505             if (viewRoot != null) {
506                 viewRoot.dispatchKey(event);
507             }
508 
509             // We gave the key to the owner, don't let the container handle this key
510             return true;
511         }
512     }
513 
isInterestingKey(int keyCode)514     private boolean isInterestingKey(int keyCode) {
515         switch (keyCode) {
516             case KeyEvent.KEYCODE_DPAD_CENTER:
517             case KeyEvent.KEYCODE_DPAD_UP:
518             case KeyEvent.KEYCODE_DPAD_DOWN:
519             case KeyEvent.KEYCODE_DPAD_LEFT:
520             case KeyEvent.KEYCODE_DPAD_RIGHT:
521             case KeyEvent.KEYCODE_ENTER:
522             case KeyEvent.KEYCODE_BACK:
523                 return true;
524             default:
525                 return false;
526         }
527     }
528 
529     /**
530      * @hide The ZoomButtonsController implements the OnTouchListener, but this
531      *       does not need to be shown in its public API.
532      */
onTouch(View v, MotionEvent event)533     public boolean onTouch(View v, MotionEvent event) {
534         int action = event.getAction();
535 
536         if (event.getPointerCount() > 1) {
537             // ZoomButtonsController doesn't handle mutitouch. Give up control.
538             return false;
539         }
540 
541         if (mReleaseTouchListenerOnUp) {
542             // The controls were dismissed but we need to throw away all events until the up
543             if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_CANCEL) {
544                 mOwnerView.setOnTouchListener(null);
545                 setTouchTargetView(null);
546                 mReleaseTouchListenerOnUp = false;
547             }
548 
549             // Eat this event
550             return true;
551         }
552 
553         dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
554 
555         View targetView = mTouchTargetView;
556 
557         switch (action) {
558             case MotionEvent.ACTION_DOWN:
559                 targetView = findViewForTouch((int) event.getRawX(), (int) event.getRawY());
560                 setTouchTargetView(targetView);
561                 break;
562 
563             case MotionEvent.ACTION_UP:
564             case MotionEvent.ACTION_CANCEL:
565                 setTouchTargetView(null);
566                 break;
567         }
568 
569         if (targetView != null) {
570             // The upperleft corner of the target view in raw coordinates
571             int targetViewRawX = mContainerRawLocation[0] + mTouchTargetWindowLocation[0];
572             int targetViewRawY = mContainerRawLocation[1] + mTouchTargetWindowLocation[1];
573 
574             MotionEvent containerEvent = MotionEvent.obtain(event);
575             // Convert the motion event into the target view's coordinates (from
576             // owner view's coordinates)
577             containerEvent.offsetLocation(mOwnerViewRawLocation[0] - targetViewRawX,
578                     mOwnerViewRawLocation[1] - targetViewRawY);
579             /* Disallow negative coordinates (which can occur due to
580              * ZOOM_CONTROLS_TOUCH_PADDING) */
581             // These are floats because we need to potentially offset away this exact amount
582             float containerX = containerEvent.getX();
583             float containerY = containerEvent.getY();
584             if (containerX < 0 && containerX > -ZOOM_CONTROLS_TOUCH_PADDING) {
585                 containerEvent.offsetLocation(-containerX, 0);
586             }
587             if (containerY < 0 && containerY > -ZOOM_CONTROLS_TOUCH_PADDING) {
588                 containerEvent.offsetLocation(0, -containerY);
589             }
590             boolean retValue = targetView.dispatchTouchEvent(containerEvent);
591             containerEvent.recycle();
592             return retValue;
593 
594         } else {
595             return false;
596         }
597     }
598 
setTouchTargetView(View view)599     private void setTouchTargetView(View view) {
600         mTouchTargetView = view;
601         if (view != null) {
602             view.getLocationInWindow(mTouchTargetWindowLocation);
603         }
604     }
605 
606     /**
607      * Returns the View that should receive a touch at the given coordinates.
608      *
609      * @param rawX The raw X.
610      * @param rawY The raw Y.
611      * @return The view that should receive the touches, or null if there is not one.
612      */
findViewForTouch(int rawX, int rawY)613     private View findViewForTouch(int rawX, int rawY) {
614         // Reverse order so the child drawn on top gets first dibs.
615         int containerCoordsX = rawX - mContainerRawLocation[0];
616         int containerCoordsY = rawY - mContainerRawLocation[1];
617         Rect frame = mTempRect;
618 
619         View closestChild = null;
620         int closestChildDistanceSq = Integer.MAX_VALUE;
621 
622         for (int i = mContainer.getChildCount() - 1; i >= 0; i--) {
623             View child = mContainer.getChildAt(i);
624             if (child.getVisibility() != View.VISIBLE) {
625                 continue;
626             }
627 
628             child.getHitRect(frame);
629             if (frame.contains(containerCoordsX, containerCoordsY)) {
630                 return child;
631             }
632 
633             int distanceX;
634             if (containerCoordsX >= frame.left && containerCoordsX <= frame.right) {
635                 distanceX = 0;
636             } else {
637                 distanceX = Math.min(Math.abs(frame.left - containerCoordsX),
638                     Math.abs(containerCoordsX - frame.right));
639             }
640             int distanceY;
641             if (containerCoordsY >= frame.top && containerCoordsY <= frame.bottom) {
642                 distanceY = 0;
643             } else {
644                 distanceY = Math.min(Math.abs(frame.top - containerCoordsY),
645                         Math.abs(containerCoordsY - frame.bottom));
646             }
647             int distanceSq = distanceX * distanceX + distanceY * distanceY;
648 
649             if ((distanceSq < mTouchPaddingScaledSq) &&
650                     (distanceSq < closestChildDistanceSq)) {
651                 closestChild = child;
652                 closestChildDistanceSq = distanceSq;
653             }
654         }
655 
656         return closestChild;
657     }
658 
onPostConfigurationChanged()659     private void onPostConfigurationChanged() {
660         dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
661         refreshPositioningVariables();
662     }
663 
664     /**
665      * Interface that will be called when the user performs an interaction that
666      * triggers some action, for example zooming.
667      */
668     public interface OnZoomListener {
669 
670         /**
671          * Called when the zoom controls' visibility changes.
672          *
673          * @param visible Whether the zoom controls are visible.
674          */
onVisibilityChanged(boolean visible)675         void onVisibilityChanged(boolean visible);
676 
677         /**
678          * Called when the owner view needs to be zoomed.
679          *
680          * @param zoomIn The direction of the zoom: true to zoom in, false to zoom out.
681          */
onZoom(boolean zoomIn)682         void onZoom(boolean zoomIn);
683     }
684 
685     private class Container extends FrameLayout {
Container(Context context)686         public Container(Context context) {
687             super(context);
688         }
689 
690         /*
691          * Need to override this to intercept the key events. Otherwise, we
692          * would attach a key listener to the container but its superclass
693          * ViewGroup gives it to the focused View instead of calling the key
694          * listener, and so we wouldn't get the events.
695          */
696         @Override
dispatchKeyEvent(KeyEvent event)697         public boolean dispatchKeyEvent(KeyEvent event) {
698             return onContainerKey(event) ? true : super.dispatchKeyEvent(event);
699         }
700     }
701 
702 }
703