BrowserSettings.java revision 8e4c299e254969836c9cf8952b82fc8659de76fe
1
2/*
3 * Copyright (C) 2007 The Android Open Source Project
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 *      http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 */
17
18package com.android.browser;
19
20import com.google.android.providers.GoogleSettings.Partner;
21
22import android.content.ContentResolver;
23import android.content.Context;
24import android.content.pm.ActivityInfo;
25import android.content.SharedPreferences;
26import android.content.SharedPreferences.Editor;
27import android.preference.PreferenceActivity;
28import android.preference.PreferenceScreen;
29import android.webkit.CookieManager;
30import android.webkit.GeolocationPermissions;
31import android.webkit.ValueCallback;
32import android.webkit.WebView;
33import android.webkit.WebViewDatabase;
34import android.webkit.WebIconDatabase;
35import android.webkit.WebSettings;
36import android.webkit.WebStorage;
37import android.preference.PreferenceManager;
38import android.provider.Browser;
39
40import java.util.HashMap;
41import java.util.Map;
42import java.util.Set;
43import java.util.Observable;
44
45/*
46 * Package level class for storing various WebView and Browser settings. To use
47 * this class:
48 * BrowserSettings s = BrowserSettings.getInstance();
49 * s.addObserver(webView.getSettings());
50 * s.loadFromDb(context); // Only needed on app startup
51 * s.javaScriptEnabled = true;
52 * ... // set any other settings
53 * s.update(); // this will update all the observers
54 *
55 * To remove an observer:
56 * s.deleteObserver(webView.getSettings());
57 */
58class BrowserSettings extends Observable {
59
60    // Private variables for settings
61    // NOTE: these defaults need to be kept in sync with the XML
62    // until the performance of PreferenceManager.setDefaultValues()
63    // is improved.
64    private boolean loadsImagesAutomatically = true;
65    private boolean javaScriptEnabled = true;
66    private boolean pluginsEnabled = true;
67    private boolean javaScriptCanOpenWindowsAutomatically = false;
68    private boolean showSecurityWarnings = true;
69    private boolean rememberPasswords = true;
70    private boolean saveFormData = true;
71    private boolean openInBackground = false;
72    private String defaultTextEncodingName;
73    private String homeUrl = "";
74    private boolean loginInitialized = false;
75    private boolean autoFitPage = true;
76    private boolean landscapeOnly = false;
77    private boolean loadsPageInOverviewMode = true;
78    private boolean showDebugSettings = false;
79    // HTML5 API flags
80    private boolean appCacheEnabled = true;
81    private boolean databaseEnabled = true;
82    private boolean domStorageEnabled = true;
83    private boolean geolocationEnabled = true;
84    private boolean workersEnabled = true;  // only affects V8. JSC does not have a similar setting
85    // HTML5 API configuration params
86    private long appCacheMaxSize = Long.MAX_VALUE;
87    private String appCachePath;  // default value set in loadFromDb().
88    private String databasePath; // default value set in loadFromDb()
89    private String geolocationDatabasePath; // default value set in loadFromDb()
90    private WebStorageSizeManager webStorageSizeManager;
91
92    private String jsFlags = "";
93
94    private final static String TAG = "BrowserSettings";
95
96    // Development settings
97    public WebSettings.LayoutAlgorithm layoutAlgorithm =
98        WebSettings.LayoutAlgorithm.NARROW_COLUMNS;
99    private boolean useWideViewPort = true;
100    private int userAgent = 0;
101    private boolean tracing = false;
102    private boolean lightTouch = false;
103    private boolean navDump = false;
104
105    // By default the error console is shown once the user navigates to about:debug.
106    // The setting can be then toggled from the settings menu.
107    private boolean showConsole = true;
108
109    // Browser only settings
110    private boolean doFlick = false;
111
112    // Private preconfigured values
113    private static int minimumFontSize = 8;
114    private static int minimumLogicalFontSize = 8;
115    private static int defaultFontSize = 16;
116    private static int defaultFixedFontSize = 13;
117    private static WebSettings.TextSize textSize =
118        WebSettings.TextSize.NORMAL;
119    private static WebSettings.ZoomDensity zoomDensity =
120        WebSettings.ZoomDensity.MEDIUM;
121
122    // Preference keys that are used outside this class
123    public final static String PREF_CLEAR_CACHE = "privacy_clear_cache";
124    public final static String PREF_CLEAR_COOKIES = "privacy_clear_cookies";
125    public final static String PREF_CLEAR_HISTORY = "privacy_clear_history";
126    public final static String PREF_HOMEPAGE = "homepage";
127    public final static String PREF_CLEAR_FORM_DATA =
128            "privacy_clear_form_data";
129    public final static String PREF_CLEAR_PASSWORDS =
130            "privacy_clear_passwords";
131    public final static String PREF_EXTRAS_RESET_DEFAULTS =
132            "reset_default_preferences";
133    public final static String PREF_DEBUG_SETTINGS = "debug_menu";
134    public final static String PREF_WEBSITE_SETTINGS = "website_settings";
135    public final static String PREF_TEXT_SIZE = "text_size";
136    public final static String PREF_DEFAULT_ZOOM = "default_zoom";
137    public final static String PREF_DEFAULT_TEXT_ENCODING =
138            "default_text_encoding";
139    public final static String PREF_CLEAR_GEOLOCATION_ACCESS =
140            "privacy_clear_geolocation_access";
141
142    private static final String DESKTOP_USERAGENT = "Mozilla/5.0 (Macintosh; " +
143            "U; Intel Mac OS X 10_5_7; en-us) AppleWebKit/530.17 (KHTML, " +
144            "like Gecko) Version/4.0 Safari/530.17";
145
146    private static final String IPHONE_USERAGENT = "Mozilla/5.0 (iPhone; U; " +
147            "CPU iPhone OS 3_0 like Mac OS X; en-us) AppleWebKit/528.18 " +
148            "(KHTML, like Gecko) Version/4.0 Mobile/7A341 Safari/528.16";
149
150    // Value to truncate strings when adding them to a TextView within
151    // a ListView
152    public final static int MAX_TEXTVIEW_LEN = 80;
153
154    private TabControl mTabControl;
155
156    // Single instance of the BrowserSettings for use in the Browser app.
157    private static BrowserSettings sSingleton;
158
159    // Private map of WebSettings to Observer objects used when deleting an
160    // observer.
161    private HashMap<WebSettings,Observer> mWebSettingsToObservers =
162        new HashMap<WebSettings,Observer>();
163
164    /*
165     * An observer wrapper for updating a WebSettings object with the new
166     * settings after a call to BrowserSettings.update().
167     */
168    static class Observer implements java.util.Observer {
169        // Private WebSettings object that will be updated.
170        private WebSettings mSettings;
171
172        Observer(WebSettings w) {
173            mSettings = w;
174        }
175
176        public void update(Observable o, Object arg) {
177            BrowserSettings b = (BrowserSettings)o;
178            WebSettings s = mSettings;
179
180            s.setLayoutAlgorithm(b.layoutAlgorithm);
181            if (b.userAgent == 0) {
182                // use the default ua string
183                s.setUserAgentString(null);
184            } else if (b.userAgent == 1) {
185                s.setUserAgentString(DESKTOP_USERAGENT);
186            } else if (b.userAgent == 2) {
187                s.setUserAgentString(IPHONE_USERAGENT);
188            }
189            s.setUseWideViewPort(b.useWideViewPort);
190            s.setLoadsImagesAutomatically(b.loadsImagesAutomatically);
191            s.setJavaScriptEnabled(b.javaScriptEnabled);
192            s.setPluginsEnabled(b.pluginsEnabled);
193            s.setJavaScriptCanOpenWindowsAutomatically(
194                    b.javaScriptCanOpenWindowsAutomatically);
195            s.setDefaultTextEncodingName(b.defaultTextEncodingName);
196            s.setMinimumFontSize(b.minimumFontSize);
197            s.setMinimumLogicalFontSize(b.minimumLogicalFontSize);
198            s.setDefaultFontSize(b.defaultFontSize);
199            s.setDefaultFixedFontSize(b.defaultFixedFontSize);
200            s.setNavDump(b.navDump);
201            s.setTextSize(b.textSize);
202            s.setDefaultZoom(b.zoomDensity);
203            s.setLightTouchEnabled(b.lightTouch);
204            s.setSaveFormData(b.saveFormData);
205            s.setSavePassword(b.rememberPasswords);
206            s.setLoadWithOverviewMode(b.loadsPageInOverviewMode);
207
208            // WebView inside Browser doesn't want initial focus to be set.
209            s.setNeedInitialFocus(false);
210            // Browser supports multiple windows
211            s.setSupportMultipleWindows(true);
212
213            // HTML5 API flags
214            s.setAppCacheEnabled(b.appCacheEnabled);
215            s.setDatabaseEnabled(b.databaseEnabled);
216            s.setDomStorageEnabled(b.domStorageEnabled);
217            s.setWorkersEnabled(b.workersEnabled);  // This only affects V8.
218            s.setGeolocationEnabled(b.geolocationEnabled);
219
220            // HTML5 configuration parameters.
221            s.setAppCacheMaxSize(b.appCacheMaxSize);
222            s.setAppCachePath(b.appCachePath);
223            s.setDatabasePath(b.databasePath);
224            s.setGeolocationDatabasePath(b.geolocationDatabasePath);
225
226            // Enable/Disable the error console.
227            b.mTabControl.getBrowserActivity().setShouldShowErrorConsole(
228                    b.showDebugSettings && b.showConsole);
229        }
230    }
231
232    /**
233     * Load settings from the browser app's database.
234     * NOTE: Strings used for the preferences must match those specified
235     * in the browser_preferences.xml
236     * @param ctx A Context object used to query the browser's settings
237     *            database. If the database exists, the saved settings will be
238     *            stored in this BrowserSettings object. This will update all
239     *            observers of this object.
240     */
241    public void loadFromDb(Context ctx) {
242        SharedPreferences p =
243                PreferenceManager.getDefaultSharedPreferences(ctx);
244        // Set the default value for the Application Caches path.
245        appCachePath = ctx.getDir("appcache", 0).getPath();
246        // Determine the maximum size of the application cache.
247        webStorageSizeManager = new WebStorageSizeManager(
248                ctx,
249                new WebStorageSizeManager.StatFsDiskInfo(appCachePath),
250                new WebStorageSizeManager.WebKitAppCacheInfo(appCachePath));
251        appCacheMaxSize = webStorageSizeManager.getAppCacheMaxSize();
252        // Set the default value for the Database path.
253        databasePath = ctx.getDir("databases", 0).getPath();
254        // Set the default value for the Geolocation database path.
255        geolocationDatabasePath = ctx.getDir("geolocation", 0).getPath();
256
257        homeUrl = getFactoryResetHomeUrl(ctx);
258
259        // Load the defaults from the xml
260        // This call is TOO SLOW, need to manually keep the defaults
261        // in sync
262        //PreferenceManager.setDefaultValues(ctx, R.xml.browser_preferences);
263        syncSharedPreferences(p);
264    }
265
266    /* package */ void syncSharedPreferences(SharedPreferences p) {
267
268        homeUrl =
269            p.getString(PREF_HOMEPAGE, homeUrl);
270
271        loadsImagesAutomatically = p.getBoolean("load_images",
272                loadsImagesAutomatically);
273        javaScriptEnabled = p.getBoolean("enable_javascript",
274                javaScriptEnabled);
275        pluginsEnabled = p.getBoolean("enable_plugins",
276                pluginsEnabled);
277        javaScriptCanOpenWindowsAutomatically = !p.getBoolean(
278            "block_popup_windows",
279            !javaScriptCanOpenWindowsAutomatically);
280        showSecurityWarnings = p.getBoolean("show_security_warnings",
281                showSecurityWarnings);
282        rememberPasswords = p.getBoolean("remember_passwords",
283                rememberPasswords);
284        saveFormData = p.getBoolean("save_formdata",
285                saveFormData);
286        boolean accept_cookies = p.getBoolean("accept_cookies",
287                CookieManager.getInstance().acceptCookie());
288        CookieManager.getInstance().setAcceptCookie(accept_cookies);
289        openInBackground = p.getBoolean("open_in_background", openInBackground);
290        loginInitialized = p.getBoolean("login_initialized", loginInitialized);
291        textSize = WebSettings.TextSize.valueOf(
292                p.getString(PREF_TEXT_SIZE, textSize.name()));
293        zoomDensity = WebSettings.ZoomDensity.valueOf(
294                p.getString(PREF_DEFAULT_ZOOM, zoomDensity.name()));
295        autoFitPage = p.getBoolean("autofit_pages", autoFitPage);
296        loadsPageInOverviewMode = p.getBoolean("load_page",
297                loadsPageInOverviewMode);
298        boolean landscapeOnlyTemp =
299                p.getBoolean("landscape_only", landscapeOnly);
300        if (landscapeOnlyTemp != landscapeOnly) {
301            landscapeOnly = landscapeOnlyTemp;
302            mTabControl.getBrowserActivity().setRequestedOrientation(
303                    landscapeOnly ? ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE
304                    : ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED);
305        }
306        useWideViewPort = true; // use wide view port for either setting
307        if (autoFitPage) {
308            layoutAlgorithm = WebSettings.LayoutAlgorithm.NARROW_COLUMNS;
309        } else {
310            layoutAlgorithm = WebSettings.LayoutAlgorithm.NORMAL;
311        }
312        defaultTextEncodingName =
313                p.getString(PREF_DEFAULT_TEXT_ENCODING,
314                        defaultTextEncodingName);
315
316        showDebugSettings =
317                p.getBoolean(PREF_DEBUG_SETTINGS, showDebugSettings);
318        // Debug menu items have precidence if the menu is visible
319        if (showDebugSettings) {
320            boolean small_screen = p.getBoolean("small_screen",
321                    layoutAlgorithm ==
322                    WebSettings.LayoutAlgorithm.SINGLE_COLUMN);
323            if (small_screen) {
324                layoutAlgorithm = WebSettings.LayoutAlgorithm.SINGLE_COLUMN;
325            } else {
326                boolean normal_layout = p.getBoolean("normal_layout",
327                        layoutAlgorithm == WebSettings.LayoutAlgorithm.NORMAL);
328                if (normal_layout) {
329                    layoutAlgorithm = WebSettings.LayoutAlgorithm.NORMAL;
330                } else {
331                    layoutAlgorithm =
332                            WebSettings.LayoutAlgorithm.NARROW_COLUMNS;
333                }
334            }
335            useWideViewPort = p.getBoolean("wide_viewport", useWideViewPort);
336            tracing = p.getBoolean("enable_tracing", tracing);
337            lightTouch = p.getBoolean("enable_light_touch", lightTouch);
338            navDump = p.getBoolean("enable_nav_dump", navDump);
339            doFlick = p.getBoolean("enable_flick", doFlick);
340            userAgent = Integer.parseInt(p.getString("user_agent", "0"));
341        }
342        // JS flags is loaded from DB even if showDebugSettings is false,
343        // so that it can be set once and be effective all the time.
344        jsFlags = p.getString("js_engine_flags", "");
345
346        // Read the setting for showing/hiding the JS Console always so that should the
347        // user enable debug settings, we already know if we should show the console.
348        // The user will never see the console unless they navigate to about:debug,
349        // regardless of the setting we read here. This setting is only used after debug
350        // is enabled.
351        showConsole = p.getBoolean("javascript_console", showConsole);
352        mTabControl.getBrowserActivity().setShouldShowErrorConsole(
353                showDebugSettings && showConsole);
354
355        // HTML5 API flags
356        appCacheEnabled = p.getBoolean("enable_appcache", appCacheEnabled);
357        databaseEnabled = p.getBoolean("enable_database", databaseEnabled);
358        domStorageEnabled = p.getBoolean("enable_domstorage", domStorageEnabled);
359        geolocationEnabled = p.getBoolean("enable_geolocation", geolocationEnabled);
360        workersEnabled = p.getBoolean("enable_workers", workersEnabled);
361
362        update();
363    }
364
365    public String getHomePage() {
366        return homeUrl;
367    }
368
369    public String getJsFlags() {
370        return jsFlags;
371    }
372
373    public WebStorageSizeManager getWebStorageSizeManager() {
374        return webStorageSizeManager;
375    }
376
377    public void setHomePage(Context context, String url) {
378        Editor ed = PreferenceManager.
379                getDefaultSharedPreferences(context).edit();
380        ed.putString(PREF_HOMEPAGE, url);
381        ed.commit();
382        homeUrl = url;
383    }
384
385    public boolean isLoginInitialized() {
386        return loginInitialized;
387    }
388
389    public void setLoginInitialized(Context context) {
390        loginInitialized = true;
391        Editor ed = PreferenceManager.
392                getDefaultSharedPreferences(context).edit();
393        ed.putBoolean("login_initialized", loginInitialized);
394        ed.commit();
395    }
396
397    public WebSettings.TextSize getTextSize() {
398        return textSize;
399    }
400
401    public WebSettings.ZoomDensity getDefaultZoom() {
402        return zoomDensity;
403    }
404
405    public boolean openInBackground() {
406        return openInBackground;
407    }
408
409    public boolean showSecurityWarnings() {
410        return showSecurityWarnings;
411    }
412
413    public boolean isTracing() {
414        return tracing;
415    }
416
417    public boolean isLightTouch() {
418        return lightTouch;
419    }
420
421    public boolean isNavDump() {
422        return navDump;
423    }
424
425    public boolean doFlick() {
426        return doFlick;
427    }
428
429    public boolean showDebugSettings() {
430        return showDebugSettings;
431    }
432
433    public void toggleDebugSettings() {
434        showDebugSettings = !showDebugSettings;
435        navDump = showDebugSettings;
436        update();
437    }
438
439    /**
440     * Add a WebSettings object to the list of observers that will be updated
441     * when update() is called.
442     *
443     * @param s A WebSettings object that is strictly tied to the life of a
444     *            WebView.
445     */
446    public Observer addObserver(WebSettings s) {
447        Observer old = mWebSettingsToObservers.get(s);
448        if (old != null) {
449            super.deleteObserver(old);
450        }
451        Observer o = new Observer(s);
452        mWebSettingsToObservers.put(s, o);
453        super.addObserver(o);
454        return o;
455    }
456
457    /**
458     * Delete the given WebSettings observer from the list of observers.
459     * @param s The WebSettings object to be deleted.
460     */
461    public void deleteObserver(WebSettings s) {
462        Observer o = mWebSettingsToObservers.get(s);
463        if (o != null) {
464            mWebSettingsToObservers.remove(s);
465            super.deleteObserver(o);
466        }
467    }
468
469    /*
470     * Package level method for obtaining a single app instance of the
471     * BrowserSettings.
472     */
473    /*package*/ static BrowserSettings getInstance() {
474        if (sSingleton == null ) {
475            sSingleton = new BrowserSettings();
476        }
477        return sSingleton;
478    }
479
480    /*
481     * Package level method for associating the BrowserSettings with TabControl
482     */
483    /* package */void setTabControl(TabControl tabControl) {
484        mTabControl = tabControl;
485    }
486
487    /*
488     * Update all the observers of the object.
489     */
490    /*package*/ void update() {
491        setChanged();
492        notifyObservers();
493    }
494
495    /*package*/ void clearCache(Context context) {
496        WebIconDatabase.getInstance().removeAllIcons();
497        if (mTabControl != null) {
498            WebView current = mTabControl.getCurrentWebView();
499            if (current != null) {
500                current.clearCache(true);
501            }
502        }
503    }
504
505    /*package*/ void clearCookies(Context context) {
506        CookieManager.getInstance().removeAllCookie();
507    }
508
509    /* package */void clearHistory(Context context) {
510        ContentResolver resolver = context.getContentResolver();
511        Browser.clearHistory(resolver);
512        Browser.clearSearches(resolver);
513    }
514
515    /* package */ void clearFormData(Context context) {
516        WebViewDatabase.getInstance(context).clearFormData();
517        if (mTabControl != null) {
518            mTabControl.getCurrentTopWebView().clearFormData();
519        }
520    }
521
522    /*package*/ void clearPasswords(Context context) {
523        WebViewDatabase db = WebViewDatabase.getInstance(context);
524        db.clearUsernamePassword();
525        db.clearHttpAuthUsernamePassword();
526    }
527
528    private void maybeDisableWebsiteSettings(Context context) {
529        PreferenceActivity activity = (PreferenceActivity) context;
530        final PreferenceScreen screen = (PreferenceScreen)
531            activity.findPreference(BrowserSettings.PREF_WEBSITE_SETTINGS);
532        screen.setEnabled(false);
533        WebStorage.getInstance().getOrigins(new ValueCallback<Map>() {
534            public void onReceiveValue(Map webStorageOrigins) {
535                if ((webStorageOrigins != null) && !webStorageOrigins.isEmpty()) {
536                    screen.setEnabled(true);
537                }
538            }
539        });
540
541        GeolocationPermissions.getInstance().getOrigins(new ValueCallback<Set<String> >() {
542            public void onReceiveValue(Set<String> geolocationOrigins) {
543                if ((geolocationOrigins != null) && !geolocationOrigins.isEmpty()) {
544                    screen.setEnabled(true);
545                }
546            }
547        });
548    }
549
550    /*package*/ void clearDatabases(Context context) {
551        WebStorage.getInstance().deleteAllData();
552        maybeDisableWebsiteSettings(context);
553    }
554
555    /*package*/ void clearLocationAccess(Context context) {
556        GeolocationPermissions.getInstance().clearAll();
557        maybeDisableWebsiteSettings(context);
558    }
559
560    /*package*/ void resetDefaultPreferences(Context ctx) {
561        SharedPreferences p =
562            PreferenceManager.getDefaultSharedPreferences(ctx);
563        p.edit().clear().commit();
564        PreferenceManager.setDefaultValues(ctx, R.xml.browser_preferences,
565                true);
566        // reset homeUrl
567        setHomePage(ctx, getFactoryResetHomeUrl(ctx));
568        // reset appcache max size
569        appCacheMaxSize = webStorageSizeManager.getAppCacheMaxSize();
570    }
571
572    private String getFactoryResetHomeUrl(Context context) {
573        String url = context.getResources().getString(R.string.homepage_base);
574        if (url.indexOf("{CID}") != -1) {
575            url = url.replace("{CID}", Partner.getString(context
576                    .getContentResolver(), Partner.CLIENT_ID, "android-google"));
577        }
578        return url;
579    }
580
581    // Private constructor that does nothing.
582    private BrowserSettings() {
583    }
584}
585