ZoomButtonsController.java revision c39a6e0c51e182338deb8b63d07933b585134929
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2008 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.widget;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
19b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Projectimport android.app.AlertDialog;
20b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Projectimport android.app.Dialog;
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.BroadcastReceiver;
22b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Projectimport android.content.ContentResolver;
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Context;
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Intent;
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.IntentFilter;
26b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Projectimport android.content.SharedPreferences;
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.PixelFormat;
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Rect;
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Handler;
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Message;
31b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Projectimport android.os.SystemClock;
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.provider.Settings;
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.Gravity;
349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.KeyEvent;
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.LayoutInflater;
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.MotionEvent;
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.View;
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.ViewConfiguration;
39c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Projectimport android.view.ViewGroup;
40b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Projectimport android.view.Window;
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.WindowManager;
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.View.OnClickListener;
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.WindowManager.LayoutParams;
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// TODO: make sure no px values exist, only dip (scale if necessary from Viewconfiguration)
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/**
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * TODO: Docs
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * If you are using this with a custom View, please call
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #setVisible(boolean) setVisible(false)} from the
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link View#onDetachedFromWindow}.
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @hide
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
56b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Projectpublic class ZoomButtonsController implements View.OnTouchListener, View.OnKeyListener {
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final String TAG = "ZoomButtonsController";
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final int ZOOM_CONTROLS_TIMEOUT =
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            (int) ViewConfiguration.getZoomControlsTimeout();
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // TODO: scaled to density
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final int ZOOM_CONTROLS_TOUCH_PADDING = 20;
65c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Context mContext;
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private WindowManager mWindowManager;
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
70b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * The view that is being zoomed by this zoom controller.
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View mOwnerView;
739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * The bounds of the owner view in global coordinates. This is recalculated
76b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * each time the zoom controller is shown.
779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Rect mOwnerViewBounds = new Rect();
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * The container that is added as a window.
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private FrameLayout mContainer;
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private LayoutParams mContainerLayoutParams;
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int[] mContainerLocation = new int[2];
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private ZoomControls mControls;
88c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * The view (or null) that should receive touch events. This will get set if
919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the touch down hits the container. It will be reset on the touch up.
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View mTouchTargetView;
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * The {@link #mTouchTargetView}'s location in window, set on touch down.
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int[] mTouchTargetLocationInWindow = new int[2];
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
99b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * If the zoom controller is dismissed but the user is still in a touch
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * interaction, we set this to true. This will ignore all touch events until
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * up/cancel, and then set the owner's touch listener to null.
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mReleaseTouchListenerOnUp;
104c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
105b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    private boolean mIsSecondTapDown;
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mIsVisible;
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Rect mTempRect = new Rect();
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private OnZoomListener mCallback;
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * When showing the zoom, we add the view as a new window. However, there is
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * logic that needs to know the size of the zoom which is determined after
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * it's laid out. Therefore, we must post this logic onto the UI thread so
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * it will be exceuted AFTER the layout. This is the logic.
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Runnable mPostedVisibleInitializer;
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private IntentFilter mConfigurationChangedFilter =
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            new IntentFilter(Intent.ACTION_CONFIGURATION_CHANGED);
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private BroadcastReceiver mConfigurationChangedReceiver = new BroadcastReceiver() {
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void onReceive(Context context, Intent intent) {
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!mIsVisible) return;
1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mHandler.removeMessages(MSG_POST_CONFIGURATION_CHANGED);
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mHandler.sendEmptyMessage(MSG_POST_CONFIGURATION_CHANGED);
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    };
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
134b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    /**
135b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * The setting name that tracks whether we've shown the zoom tutorial.
136b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     */
137b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    private static final String SETTING_NAME_SHOWN_TUTORIAL = "shown_zoom_tutorial";
138b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    private static Dialog sTutorialDialog;
139b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /** When configuration changes, this is called after the UI thread is idle. */
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final int MSG_POST_CONFIGURATION_CHANGED = 2;
142b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    /** Used to delay the zoom controller dismissal. */
143b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    private static final int MSG_DISMISS_ZOOM_CONTROLS = 3;
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * If setVisible(true) is called and the owner view's window token is null,
1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * we delay the setVisible(true) call until it is not null.
1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final int MSG_POST_SET_VISIBLE = 4;
149c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Handler mHandler = new Handler() {
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void handleMessage(Message msg) {
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            switch (msg.what) {
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                case MSG_POST_CONFIGURATION_CHANGED:
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    onPostConfigurationChanged();
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    break;
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
158b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                case MSG_DISMISS_ZOOM_CONTROLS:
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    setVisible(false);
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    break;
161c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                case MSG_POST_SET_VISIBLE:
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (mOwnerView.getWindowToken() == null) {
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        // Doh, it is still null, throw an exception
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        throw new IllegalArgumentException(
166b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                                "Cannot make the zoom controller visible if the owner view is " +
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                "not attached to a window.");
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    setVisible(true);
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    break;
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    };
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public ZoomButtonsController(Context context, View ownerView) {
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mContext = context;
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mWindowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mOwnerView = ownerView;
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mContainer = createContainer();
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
183c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
184b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public void setZoomInEnabled(boolean enabled) {
185b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        mControls.setIsZoomInEnabled(enabled);
186b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
187c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
188b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public void setZoomOutEnabled(boolean enabled) {
189b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        mControls.setIsZoomOutEnabled(enabled);
190b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
191c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
192b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public void setZoomSpeed(long speed) {
193b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        mControls.setZoomSpeed(speed);
194b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
195c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private FrameLayout createContainer() {
1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        LayoutParams lp = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        lp.gravity = Gravity.BOTTOM | Gravity.CENTER;
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        lp.flags = LayoutParams.FLAG_NOT_TOUCHABLE |
200b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                LayoutParams.FLAG_NOT_FOCUSABLE |
2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LayoutParams.FLAG_LAYOUT_NO_LIMITS;
2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        lp.height = LayoutParams.WRAP_CONTENT;
2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        lp.width = LayoutParams.FILL_PARENT;
2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        lp.type = LayoutParams.TYPE_APPLICATION_PANEL;
2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        lp.format = PixelFormat.TRANSPARENT;
206c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        lp.windowAnimations = com.android.internal.R.style.Animation_ZoomButtons;
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mContainerLayoutParams = lp;
208c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        FrameLayout container = new FrameLayout(mContext);
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        container.setLayoutParams(lp);
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        container.setMeasureAllChildren(true);
212b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        container.setOnKeyListener(this);
213c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        LayoutInflater inflater = (LayoutInflater) mContext
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
216c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        inflater.inflate(com.android.internal.R.layout.zoom_container, container);
217c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mControls = (ZoomControls) container.findViewById(com.android.internal.R.id.zoomControls);
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mControls.setOnZoomInClickListener(new OnClickListener() {
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            public void onClick(View v) {
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (mCallback != null) mCallback.onZoom(true);
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        });
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mControls.setOnZoomOutClickListener(new OnClickListener() {
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            public void onClick(View v) {
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (mCallback != null) mCallback.onZoom(false);
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        });
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return container;
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
234c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setCallback(OnZoomListener callback) {
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mCallback = callback;
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setFocusable(boolean focusable) {
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (focusable) {
241c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            mContainerLayoutParams.flags &= ~LayoutParams.FLAG_NOT_FOCUSABLE;
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mContainerLayoutParams.flags |= LayoutParams.FLAG_NOT_FOCUSABLE;
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
245c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mIsVisible) {
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWindowManager.updateViewLayout(mContainer, mContainerLayoutParams);
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean isVisible() {
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mIsVisible;
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setVisible(boolean visible) {
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!useThisZoom(mContext)) return;
258c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (visible) {
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mOwnerView.getWindowToken() == null) {
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                /*
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                 * We need a window token to show ourselves, maybe the owner's
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                 * window hasn't been created yet but it will have been by the
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                 * time the looper is idle, so post the setVisible(true) call.
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                 */
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (!mHandler.hasMessages(MSG_POST_SET_VISIBLE)) {
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mHandler.sendEmptyMessage(MSG_POST_SET_VISIBLE);
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return;
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
271c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mIsVisible == visible) {
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mIsVisible = visible;
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (visible) {
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mContainerLayoutParams.token == null) {
2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mContainerLayoutParams.token = mOwnerView.getWindowToken();
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWindowManager.addView(mContainer, mContainerLayoutParams);
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mPostedVisibleInitializer == null) {
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mPostedVisibleInitializer = new Runnable() {
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    public void run() {
2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        refreshPositioningVariables();
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        if (mCallback != null) {
2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            mCallback.onVisibilityChanged(true);
2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        }
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                };
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mHandler.post(mPostedVisibleInitializer);
3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Handle configuration changes when visible
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mContext.registerReceiver(mConfigurationChangedReceiver, mConfigurationChangedFilter);
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Steal touches events from the owner
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mOwnerView.setOnTouchListener(this);
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReleaseTouchListenerOnUp = false;
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Don't want to steal any more touches
3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mTouchTargetView != null) {
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // We are still stealing the touch events for this touch
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // sequence, so release the touch listener later
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mReleaseTouchListenerOnUp = true;
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mOwnerView.setOnTouchListener(null);
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // No longer care about configuration changes
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mContext.unregisterReceiver(mConfigurationChangedReceiver);
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWindowManager.removeView(mContainer);
3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mHandler.removeCallbacks(mPostedVisibleInitializer);
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mCallback != null) {
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mCallback.onVisibilityChanged(false);
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * TODO: docs
3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Notes:
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * - Please ensure you set your View to INVISIBLE not GONE when hiding it.
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return TODO
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
339c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project    public ViewGroup getContainer() {
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mContainer;
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void dismissControlsDelayed(int delay) {
344b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        mHandler.removeMessages(MSG_DISMISS_ZOOM_CONTROLS);
345b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        mHandler.sendEmptyMessageDelayed(MSG_DISMISS_ZOOM_CONTROLS, delay);
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Should be called by the client for each event belonging to the second tap
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * (the down, move, up, and cancel events).
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param event The event belonging to the second tap.
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return Whether the event was consumed.
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean handleDoubleTapEvent(MotionEvent event) {
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!useThisZoom(mContext)) return false;
357c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int action = event.getAction();
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (action == MotionEvent.ACTION_DOWN) {
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int x = (int) event.getX();
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int y = (int) event.getY();
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
364b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            /*
365b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project             * This class will consume all events in the second tap (down,
366b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project             * move(s), up). But, the owner already got the second tap's down,
367b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project             * so cancel that. Do this before setVisible, since that call
368b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project             * will set us as a touch listener.
369b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project             */
370b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            MotionEvent cancelEvent = MotionEvent.obtain(event.getDownTime(),
371b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                    SystemClock.elapsedRealtime(),
372b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                    MotionEvent.ACTION_CANCEL, 0, 0, 0);
373b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            mOwnerView.dispatchTouchEvent(cancelEvent);
374b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            cancelEvent.recycle();
375b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setVisible(true);
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            centerPoint(x, y);
378b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            mIsSecondTapDown = true;
3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return true;
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void refreshPositioningVariables() {
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Calculate the owner view's bounds
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mOwnerView.getGlobalVisibleRect(mOwnerViewBounds);
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mContainer.getLocationOnScreen(mContainerLocation);
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Centers the point (in owner view's coordinates).
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void centerPoint(int x, int y) {
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCallback != null) {
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mCallback.onCenter(x, y);
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
399b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public boolean onKey(View v, int keyCode, KeyEvent event) {
400b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
401b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        return false;
402b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
403b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean onTouch(View v, MotionEvent event) {
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int action = event.getAction();
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
407b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        // Consume all events during the second-tap interaction (down, move, up/cancel)
408b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        boolean consumeEvent = mIsSecondTapDown;
409b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        if ((action == MotionEvent.ACTION_UP) || (action == MotionEvent.ACTION_CANCEL)) {
410b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            // The second tap can no longer be down
411b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            mIsSecondTapDown = false;
412b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        }
413c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mReleaseTouchListenerOnUp) {
415b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            // The controls were dismissed but we need to throw away all events until the up
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_CANCEL) {
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mOwnerView.setOnTouchListener(null);
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                setTouchTargetView(null);
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mReleaseTouchListenerOnUp = false;
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Eat this event
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // TODO: optimize this (it ends up removing message and queuing another)
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
428c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        View targetView = mTouchTargetView;
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (action) {
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case MotionEvent.ACTION_DOWN:
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                targetView = getViewForTouch((int) event.getRawX(), (int) event.getRawY());
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                setTouchTargetView(targetView);
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case MotionEvent.ACTION_UP:
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case MotionEvent.ACTION_CANCEL:
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                setTouchTargetView(null);
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (targetView != null) {
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // The upperleft corner of the target view in raw coordinates
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int targetViewRawX = mContainerLocation[0] + mTouchTargetLocationInWindow[0];
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int targetViewRawY = mContainerLocation[1] + mTouchTargetLocationInWindow[1];
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            MotionEvent containerEvent = MotionEvent.obtain(event);
4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Convert the motion event into the target view's coordinates (from
4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // owner view's coordinates)
4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            containerEvent.offsetLocation(mOwnerViewBounds.left - targetViewRawX,
4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mOwnerViewBounds.top - targetViewRawY);
4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            boolean retValue = targetView.dispatchTouchEvent(containerEvent);
4549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            containerEvent.recycle();
455b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            return retValue || consumeEvent;
4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
458b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            return consumeEvent;
4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void setTouchTargetView(View view) {
4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTouchTargetView = view;
4649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (view != null) {
4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            view.getLocationInWindow(mTouchTargetLocationInWindow);
4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns the View that should receive a touch at the given coordinates.
4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param rawX The raw X.
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param rawY The raw Y.
4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return The view that should receive the touches, or null if there is not one.
4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View getViewForTouch(int rawX, int rawY) {
4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Reverse order so the child drawn on top gets first dibs.
4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int containerCoordsX = rawX - mContainerLocation[0];
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int containerCoordsY = rawY - mContainerLocation[1];
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Rect frame = mTempRect;
4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (int i = mContainer.getChildCount() - 1; i >= 0; i--) {
4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            View child = mContainer.getChildAt(i);
4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (child.getVisibility() != View.VISIBLE) {
4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                continue;
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            child.getHitRect(frame);
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Expand the touch region
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            frame.top -= ZOOM_CONTROLS_TOUCH_PADDING;
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (frame.contains(containerCoordsX, containerCoordsY)) {
4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return child;
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return null;
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void onPostConfigurationChanged() {
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dismissControlsDelayed(ZOOM_CONTROLS_TIMEOUT);
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        refreshPositioningVariables();
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
503b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    /*
504b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * This is static so Activities can call this instead of the Views
505b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * (Activities usually do not have a reference to the ZoomButtonsController
506b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * instance.)
507b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     */
508b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    /**
509b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * Shows a "tutorial" (some text) to the user teaching her the new zoom
510b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * invocation method. Must call from the main thread.
511b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * <p>
512b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * It checks the global system setting to ensure this has not been seen
513b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * before. Furthermore, if the application does not have privilege to write
514b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * to the system settings, it will store this bit locally in a shared
515b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * preference.
516b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     *
517b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     * @hide This should only be used by our main apps--browser, maps, and
518b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     *       gallery
519b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project     */
520b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public static void showZoomTutorialOnce(Context context) {
521b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        ContentResolver cr = context.getContentResolver();
522b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        if (Settings.System.getInt(cr, SETTING_NAME_SHOWN_TUTORIAL, 0) == 1) {
523b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            return;
524b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        }
525b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
526b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        SharedPreferences sp = context.getSharedPreferences("_zoom", Context.MODE_PRIVATE);
527b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        if (sp.getInt(SETTING_NAME_SHOWN_TUTORIAL, 0) == 1) {
528b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            return;
529b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        }
530b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
531b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        if (sTutorialDialog != null && sTutorialDialog.isShowing()) {
532b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            sTutorialDialog.dismiss();
533b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        }
534b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
535b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        LayoutInflater layoutInflater =
536b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
537b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        TextView textView = (TextView) layoutInflater.inflate(
538b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                com.android.internal.R.layout.alert_dialog_simple_text, null)
539b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                .findViewById(android.R.id.text1);
540b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        textView.setText(com.android.internal.R.string.tutorial_double_tap_to_zoom_message_short);
541c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
542b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        sTutorialDialog = new AlertDialog.Builder(context)
543b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                .setView(textView)
544b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                .setIcon(0)
545b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                .create();
546b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
547b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        Window window = sTutorialDialog.getWindow();
548b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        window.setGravity(Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM);
549b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        window.clearFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND |
550b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                WindowManager.LayoutParams.FLAG_BLUR_BEHIND);
551b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        window.addFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE |
552b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE);
553b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
554b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        sTutorialDialog.show();
555b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
556b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
557b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    /** @hide Should only be used by Android platform apps */
558b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public static void finishZoomTutorial(Context context, boolean userNotified) {
559b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        if (sTutorialDialog == null) return;
560b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
561b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        sTutorialDialog.dismiss();
562b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        sTutorialDialog = null;
563b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
564b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        // Record that they have seen the tutorial
565b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        if (userNotified) {
566b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            try {
567b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                Settings.System.putInt(context.getContentResolver(), SETTING_NAME_SHOWN_TUTORIAL,
568b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                        1);
569b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            } catch (SecurityException e) {
570b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                /*
571b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                 * The app does not have permission to clear this global flag, make
572b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                 * sure the user does not see the message when he comes back to this
573b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                 * same app at least.
574b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                 */
575b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                SharedPreferences sp = context.getSharedPreferences("_zoom", Context.MODE_PRIVATE);
576b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project                sp.edit().putInt(SETTING_NAME_SHOWN_TUTORIAL, 1).commit();
577b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project            }
578b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        }
579b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
580b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
581b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    /** @hide Should only be used by Android platform apps */
582b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public void finishZoomTutorial() {
583b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        finishZoomTutorial(mContext, true);
584b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
585b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
586b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    // Temporary methods for different zoom types
587b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    static int getZoomType(Context context) {
588c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        return Settings.System.getInt(context.getContentResolver(), "zoom", 2);
589b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
590b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
591b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    public static boolean useOldZoom(Context context) {
592b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        return getZoomType(context) == 0;
593b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project    }
594b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static boolean useThisZoom(Context context) {
596b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project        return getZoomType(context) == 2;
5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
598b2a3dd88a53cc8c6d19f6dc8ec4f3d6c4abd9b54The Android Open Source Project
5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public interface OnZoomListener {
6009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        void onCenter(int x, int y);
6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        void onVisibilityChanged(boolean visible);
6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        void onZoom(boolean zoomIn);
6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
605