RotationPolicy.java revision 79578b29bf4ba1d210586b9d7bb832eddf0960b7
1/*
2 * Copyright (C) 2012 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.internal.view;
18
19import android.content.Context;
20import android.content.pm.PackageManager;
21import android.database.ContentObserver;
22import android.net.Uri;
23import android.os.AsyncTask;
24import android.os.Build;
25import android.os.Handler;
26import android.os.RemoteException;
27import android.os.UserHandle;
28import android.provider.Settings;
29import android.util.Log;
30import android.view.IWindowManager;
31import android.view.Surface;
32import android.view.WindowManagerGlobal;
33
34/**
35 * Provides helper functions for configuring the display rotation policy.
36 */
37public final class RotationPolicy {
38    private static final String TAG = "RotationPolicy";
39
40    private RotationPolicy() {
41    }
42
43    /**
44     * Gets whether the device supports rotation. In general such a
45     * device has an accelerometer and has the portrait and landscape
46     * features.
47     *
48     * @param context Context for accessing system resources.
49     * @return Whether the device supports rotation.
50     */
51    public static boolean isRotationSupported(Context context) {
52        PackageManager pm = context.getPackageManager();
53        return pm.hasSystemFeature(PackageManager.FEATURE_SENSOR_ACCELEROMETER)
54                && pm.hasSystemFeature(PackageManager.FEATURE_SCREEN_PORTRAIT)
55                && pm.hasSystemFeature(PackageManager.FEATURE_SCREEN_LANDSCAPE);
56    }
57
58    /**
59     * Returns true if the device supports the rotation-lock toggle feature
60     * in the system UI or system bar.
61     *
62     * When the rotation-lock toggle is supported, the "auto-rotate screen" option in
63     * Display settings should be hidden, but it should remain available in Accessibility
64     * settings.
65     */
66    public static boolean isRotationLockToggleSupported(Context context) {
67        return isRotationSupported(context)
68                && context.getResources().getConfiguration().smallestScreenWidthDp >= 600;
69    }
70
71    /**
72     * Returns true if the rotation-lock toggle should be shown in the UI.
73     */
74    public static boolean isRotationLockToggleVisible(Context context) {
75        return isRotationLockToggleSupported(context) &&
76                Settings.System.getIntForUser(context.getContentResolver(),
77                        Settings.System.HIDE_ROTATION_LOCK_TOGGLE_FOR_ACCESSIBILITY, 0,
78                        UserHandle.USER_CURRENT) == 0;
79    }
80
81    /**
82     * Returns true if rotation lock is enabled.
83     */
84    public static boolean isRotationLocked(Context context) {
85        return Settings.System.getIntForUser(context.getContentResolver(),
86                Settings.System.ACCELEROMETER_ROTATION, 0, UserHandle.USER_CURRENT) == 0;
87    }
88
89    /**
90     * Enables or disables rotation lock.
91     *
92     * Should be used by the rotation lock toggle.
93     */
94    public static void setRotationLock(Context context, final boolean enabled) {
95        Settings.System.putIntForUser(context.getContentResolver(),
96                Settings.System.HIDE_ROTATION_LOCK_TOGGLE_FOR_ACCESSIBILITY, 0,
97                UserHandle.USER_CURRENT);
98
99        AsyncTask.execute(new Runnable() {
100            @Override
101            public void run() {
102                try {
103                    IWindowManager wm = WindowManagerGlobal.getWindowManagerService();
104                    if (enabled) {
105                        wm.freezeRotation(-1);
106                    } else {
107                        wm.thawRotation();
108                    }
109                } catch (RemoteException exc) {
110                    Log.w(TAG, "Unable to save auto-rotate setting");
111                }
112            }
113        });
114    }
115
116    /**
117     * Enables or disables rotation lock and adjusts whether the rotation lock toggle
118     * should be hidden for accessibility purposes.
119     *
120     * Should be used by Display settings and Accessibility settings.
121     */
122    public static void setRotationLockForAccessibility(Context context, final boolean enabled) {
123        Settings.System.putIntForUser(context.getContentResolver(),
124                Settings.System.HIDE_ROTATION_LOCK_TOGGLE_FOR_ACCESSIBILITY, enabled ? 1 : 0,
125                        UserHandle.USER_CURRENT);
126
127        AsyncTask.execute(new Runnable() {
128            @Override
129            public void run() {
130                try {
131                    IWindowManager wm = WindowManagerGlobal.getWindowManagerService();
132                    if (enabled) {
133                        wm.freezeRotation(Surface.ROTATION_0);
134                    } else {
135                        wm.thawRotation();
136                    }
137                } catch (RemoteException exc) {
138                    Log.w(TAG, "Unable to save auto-rotate setting");
139                }
140            }
141        });
142    }
143
144    /**
145     * Registers a listener for rotation policy changes affecting the caller's user
146     */
147    public static void registerRotationPolicyListener(Context context,
148            RotationPolicyListener listener) {
149        registerRotationPolicyListener(context, listener, UserHandle.getCallingUserId());
150    }
151
152    /**
153     * Registers a listener for rotation policy changes affecting a specific user,
154     * or USER_ALL for all users.
155     */
156    public static void registerRotationPolicyListener(Context context,
157            RotationPolicyListener listener, int userHandle) {
158        context.getContentResolver().registerContentObserver(Settings.System.getUriFor(
159                Settings.System.ACCELEROMETER_ROTATION),
160                false, listener.mObserver, userHandle);
161        context.getContentResolver().registerContentObserver(Settings.System.getUriFor(
162                Settings.System.HIDE_ROTATION_LOCK_TOGGLE_FOR_ACCESSIBILITY),
163                false, listener.mObserver, userHandle);
164    }
165
166    /**
167     * Unregisters a listener for rotation policy changes.
168     */
169    public static void unregisterRotationPolicyListener(Context context,
170            RotationPolicyListener listener) {
171        context.getContentResolver().unregisterContentObserver(listener.mObserver);
172    }
173
174    /**
175     * Listener that is invoked whenever a change occurs that might affect the rotation policy.
176     */
177    public static abstract class RotationPolicyListener {
178        final ContentObserver mObserver = new ContentObserver(new Handler()) {
179            public void onChange(boolean selfChange, Uri uri) {
180                RotationPolicyListener.this.onChange();
181            }
182        };
183
184        public abstract void onChange();
185    }
186}