SettingsHelper.java revision 11bfc2261f653f1b3154b88bf692241dbd6fc477
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
17package com.android.providers.settings;
18
19import android.app.ActivityManagerNative;
20import android.app.IActivityManager;
21import android.app.backup.IBackupManager;
22import android.content.Context;
23import android.content.res.Configuration;
24import android.location.LocationManager;
25import android.media.AudioManager;
26import android.media.RingtoneManager;
27import android.net.Uri;
28import android.os.IPowerManager;
29import android.os.RemoteException;
30import android.os.ServiceManager;
31import android.os.UserManager;
32import android.provider.Settings;
33import android.text.TextUtils;
34
35import java.util.Locale;
36
37public class SettingsHelper {
38    private static final String SILENT_RINGTONE = "_silent";
39    private Context mContext;
40    private AudioManager mAudioManager;
41
42    public SettingsHelper(Context context) {
43        mContext = context;
44        mAudioManager = (AudioManager) context
45                .getSystemService(Context.AUDIO_SERVICE);
46    }
47
48    /**
49     * Sets the property via a call to the appropriate API, if any, and returns
50     * whether or not the setting should be saved to the database as well.
51     * @param name the name of the setting
52     * @param value the string value of the setting
53     * @return whether to continue with writing the value to the database. In
54     * some cases the data will be written by the call to the appropriate API,
55     * and in some cases the property value needs to be modified before setting.
56     */
57    public boolean restoreValue(String name, String value) {
58        if (Settings.System.SCREEN_BRIGHTNESS.equals(name)) {
59            setBrightness(Integer.parseInt(value));
60        } else if (Settings.System.SOUND_EFFECTS_ENABLED.equals(name)) {
61            setSoundEffects(Integer.parseInt(value) == 1);
62        } else if (Settings.Secure.LOCATION_PROVIDERS_ALLOWED.equals(name)) {
63            setGpsLocation(value);
64            return false;
65        } else if (Settings.Secure.BACKUP_AUTO_RESTORE.equals(name)) {
66            setAutoRestore(Integer.parseInt(value) == 1);
67        } else if (isAlreadyConfiguredCriticalAccessibilitySetting(name)) {
68            return false;
69        } else if (Settings.System.RINGTONE.equals(name)
70                || Settings.System.NOTIFICATION_SOUND.equals(name)) {
71            setRingtone(name, value);
72            return false;
73        }
74        return true;
75    }
76
77    public String onBackupValue(String name, String value) {
78        // Special processing for backing up ringtones
79        if (Settings.System.RINGTONE.equals(name)
80                || Settings.System.NOTIFICATION_SOUND.equals(name)) {
81            if (value == null) {
82                // Silent ringtone
83                return SILENT_RINGTONE;
84            } else {
85                return getCanonicalRingtoneValue(value);
86            }
87        }
88        // Return the original value
89        return value;
90    }
91
92    /**
93     * Sets the ringtone of type specified by the name.
94     *
95     * @param name should be Settings.System.RINGTONE or Settings.System.NOTIFICATION_SOUND.
96     * @param value can be a canonicalized uri or "_silent" to indicate a silent (null) ringtone.
97     */
98    private void setRingtone(String name, String value) {
99        // If it's null, don't change the default
100        if (value == null) return;
101        Uri ringtoneUri = null;
102        if (SILENT_RINGTONE.equals(value)) {
103            ringtoneUri = null;
104        } else {
105            Uri canonicalUri = Uri.parse(value);
106            ringtoneUri = mContext.getContentResolver().uncanonicalize(canonicalUri);
107            if (ringtoneUri == null) {
108                // Unrecognized or invalid Uri, don't restore
109                return;
110            }
111        }
112        final int ringtoneType = Settings.System.RINGTONE.equals(name)
113                ? RingtoneManager.TYPE_RINGTONE : RingtoneManager.TYPE_NOTIFICATION;
114        RingtoneManager.setActualDefaultRingtoneUri(mContext, ringtoneType, ringtoneUri);
115    }
116
117    private String getCanonicalRingtoneValue(String value) {
118        final Uri ringtoneUri = Uri.parse(value);
119        final Uri canonicalUri = mContext.getContentResolver().canonicalize(ringtoneUri);
120        return canonicalUri == null ? null : canonicalUri.toString();
121    }
122
123    private boolean isAlreadyConfiguredCriticalAccessibilitySetting(String name) {
124        // These are the critical accessibility settings that are required for a
125        // blind user to be able to interact with the device. If these settings are
126        // already configured, we will not overwrite them. If they are already set,
127        // it means that the user has performed a global gesture to enable accessibility
128        // and definitely needs these features working after the restore.
129        if (Settings.Secure.ACCESSIBILITY_ENABLED.equals(name)
130                || Settings.Secure.ACCESSIBILITY_SCRIPT_INJECTION.equals(name)
131                || Settings.Secure.ACCESSIBILITY_SPEAK_PASSWORD.equals(name)
132                || Settings.Secure.TOUCH_EXPLORATION_ENABLED.equals(name)) {
133            return Settings.Secure.getInt(mContext.getContentResolver(), name, 0) != 0;
134        } else if (Settings.Secure.TOUCH_EXPLORATION_GRANTED_ACCESSIBILITY_SERVICES.equals(name)
135                || Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES.equals(name)) {
136            return !TextUtils.isEmpty(Settings.Secure.getString(
137                    mContext.getContentResolver(), name));
138        }
139        return false;
140    }
141
142    private void setAutoRestore(boolean enabled) {
143        try {
144            IBackupManager bm = IBackupManager.Stub.asInterface(
145                    ServiceManager.getService(Context.BACKUP_SERVICE));
146            if (bm != null) {
147                bm.setAutoRestore(enabled);
148            }
149        } catch (RemoteException e) {}
150    }
151
152    private void setGpsLocation(String value) {
153        UserManager um = (UserManager) mContext.getSystemService(Context.USER_SERVICE);
154        if (um.hasUserRestriction(UserManager.DISALLOW_SHARE_LOCATION)) {
155            return;
156        }
157        final String GPS = LocationManager.GPS_PROVIDER;
158        boolean enabled =
159                GPS.equals(value) ||
160                value.startsWith(GPS + ",") ||
161                value.endsWith("," + GPS) ||
162                value.contains("," + GPS + ",");
163        Settings.Secure.setLocationProviderEnabled(
164                mContext.getContentResolver(), GPS, enabled);
165    }
166
167    private void setSoundEffects(boolean enable) {
168        if (enable) {
169            mAudioManager.loadSoundEffects();
170        } else {
171            mAudioManager.unloadSoundEffects();
172        }
173    }
174
175    private void setBrightness(int brightness) {
176        try {
177            IPowerManager power = IPowerManager.Stub.asInterface(
178                    ServiceManager.getService("power"));
179            if (power != null) {
180                power.setTemporaryScreenBrightnessSettingOverride(brightness);
181            }
182        } catch (RemoteException doe) {
183
184        }
185    }
186
187    byte[] getLocaleData() {
188        Configuration conf = mContext.getResources().getConfiguration();
189        final Locale loc = conf.locale;
190        String localeString = loc.getLanguage();
191        String country = loc.getCountry();
192        if (!TextUtils.isEmpty(country)) {
193            localeString += "-" + country;
194        }
195        return localeString.getBytes();
196    }
197
198    /**
199     * Sets the locale specified. Input data is the byte representation of a
200     * BCP-47 language tag. For backwards compatibility, strings of the form
201     * {@code ll_CC} are also accepted, where {@code ll} is a two letter language
202     * code and {@code CC} is a two letter country code.
203     *
204     * @param data the locale string in bytes.
205     */
206    void setLocaleData(byte[] data, int size) {
207        // Check if locale was set by the user:
208        Configuration conf = mContext.getResources().getConfiguration();
209        // TODO: The following is not working as intended because the network is forcing a locale
210        // change after registering. Need to find some other way to detect if the user manually
211        // changed the locale
212        if (conf.userSetLocale) return; // Don't change if user set it in the SetupWizard
213
214        final String[] availableLocales = mContext.getAssets().getLocales();
215        // Replace "_" with "-" to deal with older backups.
216        String localeCode = new String(data, 0, size).replace('_', '-');
217        Locale loc = null;
218        for (int i = 0; i < availableLocales.length; i++) {
219            if (availableLocales[i].equals(localeCode)) {
220                loc = Locale.forLanguageTag(localeCode);
221                break;
222            }
223        }
224        if (loc == null) return; // Couldn't find the saved locale in this version of the software
225
226        try {
227            IActivityManager am = ActivityManagerNative.getDefault();
228            Configuration config = am.getConfiguration();
229            config.locale = loc;
230            // indicate this isn't some passing default - the user wants this remembered
231            config.userSetLocale = true;
232
233            am.updateConfiguration(config);
234        } catch (RemoteException e) {
235            // Intentionally left blank
236        }
237    }
238
239    /**
240     * Informs the audio service of changes to the settings so that
241     * they can be re-read and applied.
242     */
243    void applyAudioSettings() {
244        AudioManager am = new AudioManager(mContext);
245        am.reloadAudioSettings();
246    }
247}
248