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