| /* |
| * Copyright (C) 2009 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.browser.preferences; |
| |
| import android.app.AlertDialog; |
| import android.app.ListFragment; |
| import android.content.Context; |
| import android.content.DialogInterface; |
| import android.database.Cursor; |
| import android.graphics.Bitmap; |
| import android.graphics.BitmapFactory; |
| import android.net.Uri; |
| import android.os.AsyncTask; |
| import android.os.Bundle; |
| import android.os.Parcel; |
| import android.os.Parcelable; |
| import android.preference.PreferenceActivity; |
| import android.provider.BrowserContract.Bookmarks; |
| import android.util.Log; |
| import android.view.LayoutInflater; |
| import android.view.View; |
| import android.view.View.OnClickListener; |
| import android.view.ViewGroup; |
| import android.webkit.GeolocationPermissions; |
| import android.webkit.ValueCallback; |
| import android.webkit.WebStorage; |
| import android.widget.AdapterView; |
| import android.widget.ArrayAdapter; |
| import android.widget.ImageView; |
| import android.widget.TextView; |
| |
| import com.android.browser.R; |
| import com.android.browser.WebStorageSizeManager; |
| |
| import java.util.HashMap; |
| import java.util.HashSet; |
| import java.util.Iterator; |
| import java.util.Map; |
| import java.util.Set; |
| |
| /** |
| * Manage the settings for an origin. |
| * We use it to keep track of the 'HTML5' settings, i.e. database (webstorage) |
| * and Geolocation. |
| */ |
| public class WebsiteSettingsFragment extends ListFragment implements OnClickListener { |
| |
| private static final String EXTRA_SITE = "site"; |
| private String LOGTAG = "WebsiteSettingsActivity"; |
| private static String sMBStored = null; |
| private SiteAdapter mAdapter = null; |
| private Site mSite = null; |
| |
| static class Site implements Parcelable { |
| private String mOrigin; |
| private String mTitle; |
| private Bitmap mIcon; |
| private int mFeatures; |
| |
| // These constants provide the set of features that a site may support |
| // They must be consecutive. To add a new feature, add a new FEATURE_XXX |
| // variable with value equal to the current value of FEATURE_COUNT, then |
| // increment FEATURE_COUNT. |
| final static int FEATURE_WEB_STORAGE = 0; |
| final static int FEATURE_GEOLOCATION = 1; |
| // The number of features available. |
| final static int FEATURE_COUNT = 2; |
| |
| public Site(String origin) { |
| mOrigin = origin; |
| mTitle = null; |
| mIcon = null; |
| mFeatures = 0; |
| } |
| |
| public void addFeature(int feature) { |
| mFeatures |= (1 << feature); |
| } |
| |
| public void removeFeature(int feature) { |
| mFeatures &= ~(1 << feature); |
| } |
| |
| public boolean hasFeature(int feature) { |
| return (mFeatures & (1 << feature)) != 0; |
| } |
| |
| /** |
| * Gets the number of features supported by this site. |
| */ |
| public int getFeatureCount() { |
| int count = 0; |
| for (int i = 0; i < FEATURE_COUNT; ++i) { |
| count += hasFeature(i) ? 1 : 0; |
| } |
| return count; |
| } |
| |
| /** |
| * Gets the ID of the nth (zero-based) feature supported by this site. |
| * The return value is a feature ID - one of the FEATURE_XXX values. |
| * This is required to determine which feature is displayed at a given |
| * position in the list of features for this site. This is used both |
| * when populating the view and when responding to clicks on the list. |
| */ |
| public int getFeatureByIndex(int n) { |
| int j = -1; |
| for (int i = 0; i < FEATURE_COUNT; ++i) { |
| j += hasFeature(i) ? 1 : 0; |
| if (j == n) { |
| return i; |
| } |
| } |
| return -1; |
| } |
| |
| public String getOrigin() { |
| return mOrigin; |
| } |
| |
| public void setTitle(String title) { |
| mTitle = title; |
| } |
| |
| public void setIcon(Bitmap icon) { |
| mIcon = icon; |
| } |
| |
| public Bitmap getIcon() { |
| return mIcon; |
| } |
| |
| public String getPrettyOrigin() { |
| return mTitle == null ? null : hideHttp(mOrigin); |
| } |
| |
| public String getPrettyTitle() { |
| return mTitle == null ? hideHttp(mOrigin) : mTitle; |
| } |
| |
| private String hideHttp(String str) { |
| Uri uri = Uri.parse(str); |
| return "http".equals(uri.getScheme()) ? str.substring(7) : str; |
| } |
| |
| @Override |
| public int describeContents() { |
| return 0; |
| } |
| |
| @Override |
| public void writeToParcel(Parcel dest, int flags) { |
| dest.writeString(mOrigin); |
| dest.writeString(mTitle); |
| dest.writeInt(mFeatures); |
| dest.writeParcelable(mIcon, flags); |
| } |
| |
| private Site(Parcel in) { |
| mOrigin = in.readString(); |
| mTitle = in.readString(); |
| mFeatures = in.readInt(); |
| mIcon = in.readParcelable(null); |
| } |
| |
| public static final Parcelable.Creator<Site> CREATOR |
| = new Parcelable.Creator<Site>() { |
| public Site createFromParcel(Parcel in) { |
| return new Site(in); |
| } |
| |
| public Site[] newArray(int size) { |
| return new Site[size]; |
| } |
| }; |
| |
| } |
| |
| class SiteAdapter extends ArrayAdapter<Site> |
| implements AdapterView.OnItemClickListener { |
| private int mResource; |
| private LayoutInflater mInflater; |
| private Bitmap mDefaultIcon; |
| private Bitmap mUsageEmptyIcon; |
| private Bitmap mUsageLowIcon; |
| private Bitmap mUsageHighIcon; |
| private Bitmap mLocationAllowedIcon; |
| private Bitmap mLocationDisallowedIcon; |
| private Site mCurrentSite; |
| |
| public SiteAdapter(Context context, int rsc) { |
| this(context, rsc, null); |
| } |
| |
| public SiteAdapter(Context context, int rsc, Site site) { |
| super(context, rsc); |
| mResource = rsc; |
| mInflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); |
| mDefaultIcon = BitmapFactory.decodeResource(getResources(), |
| R.drawable.app_web_browser_sm); |
| mUsageEmptyIcon = BitmapFactory.decodeResource(getResources(), |
| R.drawable.ic_list_data_off); |
| mUsageLowIcon = BitmapFactory.decodeResource(getResources(), |
| R.drawable.ic_list_data_small); |
| mUsageHighIcon = BitmapFactory.decodeResource(getResources(), |
| R.drawable.ic_list_data_large); |
| mLocationAllowedIcon = BitmapFactory.decodeResource(getResources(), |
| R.drawable.ic_gps_on_holo_dark); |
| mLocationDisallowedIcon = BitmapFactory.decodeResource(getResources(), |
| R.drawable.ic_gps_denied_holo_dark); |
| mCurrentSite = site; |
| if (mCurrentSite == null) { |
| askForOrigins(); |
| } |
| } |
| |
| /** |
| * Adds the specified feature to the site corresponding to supplied |
| * origin in the map. Creates the site if it does not already exist. |
| */ |
| private void addFeatureToSite(Map<String, Site> sites, String origin, int feature) { |
| Site site = null; |
| if (sites.containsKey(origin)) { |
| site = (Site) sites.get(origin); |
| } else { |
| site = new Site(origin); |
| sites.put(origin, site); |
| } |
| site.addFeature(feature); |
| } |
| |
| public void askForOrigins() { |
| // Get the list of origins we want to display. |
| // All 'HTML 5 modules' (Database, Geolocation etc) form these |
| // origin strings using WebCore::SecurityOrigin::toString(), so it's |
| // safe to group origins here. Note that WebCore::SecurityOrigin |
| // uses 0 (which is not printed) for the port if the port is the |
| // default for the protocol. Eg http://www.google.com and |
| // http://www.google.com:80 both record a port of 0 and hence |
| // toString() == 'http://www.google.com' for both. |
| |
| WebStorage.getInstance().getOrigins(new ValueCallback<Map>() { |
| public void onReceiveValue(Map origins) { |
| Map<String, Site> sites = new HashMap<String, Site>(); |
| if (origins != null) { |
| Iterator<String> iter = origins.keySet().iterator(); |
| while (iter.hasNext()) { |
| addFeatureToSite(sites, iter.next(), Site.FEATURE_WEB_STORAGE); |
| } |
| } |
| askForGeolocation(sites); |
| } |
| }); |
| } |
| |
| public void askForGeolocation(final Map<String, Site> sites) { |
| GeolocationPermissions.getInstance().getOrigins(new ValueCallback<Set<String> >() { |
| public void onReceiveValue(Set<String> origins) { |
| if (origins != null) { |
| Iterator<String> iter = origins.iterator(); |
| while (iter.hasNext()) { |
| addFeatureToSite(sites, iter.next(), Site.FEATURE_GEOLOCATION); |
| } |
| } |
| populateIcons(sites); |
| populateOrigins(sites); |
| } |
| }); |
| } |
| |
| public void populateIcons(Map<String, Site> sites) { |
| // Create a map from host to origin. This is used to add metadata |
| // (title, icon) for this origin from the bookmarks DB. We must do |
| // the DB access on a background thread. |
| new UpdateFromBookmarksDbTask(this.getContext(), sites).execute(); |
| } |
| |
| private class UpdateFromBookmarksDbTask extends AsyncTask<Void, Void, Void> { |
| |
| private Context mContext; |
| private boolean mDataSetChanged; |
| private Map<String, Site> mSites; |
| |
| public UpdateFromBookmarksDbTask(Context ctx, Map<String, Site> sites) { |
| mContext = ctx.getApplicationContext(); |
| mSites = sites; |
| } |
| |
| protected Void doInBackground(Void... unused) { |
| HashMap<String, Set<Site>> hosts = new HashMap<String, Set<Site>>(); |
| Set<Map.Entry<String, Site>> elements = mSites.entrySet(); |
| Iterator<Map.Entry<String, Site>> originIter = elements.iterator(); |
| while (originIter.hasNext()) { |
| Map.Entry<String, Site> entry = originIter.next(); |
| Site site = entry.getValue(); |
| String host = Uri.parse(entry.getKey()).getHost(); |
| Set<Site> hostSites = null; |
| if (hosts.containsKey(host)) { |
| hostSites = (Set<Site>)hosts.get(host); |
| } else { |
| hostSites = new HashSet<Site>(); |
| hosts.put(host, hostSites); |
| } |
| hostSites.add(site); |
| } |
| |
| // Check the bookmark DB. If we have data for a host used by any of |
| // our origins, use it to set their title and favicon |
| Cursor c = mContext.getContentResolver().query(Bookmarks.CONTENT_URI, |
| new String[] { Bookmarks.URL, Bookmarks.TITLE, Bookmarks.FAVICON }, |
| Bookmarks.IS_FOLDER + " == 0", null, null); |
| |
| if (c != null) { |
| if (c.moveToFirst()) { |
| int urlIndex = c.getColumnIndex(Bookmarks.URL); |
| int titleIndex = c.getColumnIndex(Bookmarks.TITLE); |
| int faviconIndex = c.getColumnIndex(Bookmarks.FAVICON); |
| do { |
| String url = c.getString(urlIndex); |
| String host = Uri.parse(url).getHost(); |
| if (hosts.containsKey(host)) { |
| String title = c.getString(titleIndex); |
| Bitmap bmp = null; |
| byte[] data = c.getBlob(faviconIndex); |
| if (data != null) { |
| bmp = BitmapFactory.decodeByteArray(data, 0, data.length); |
| } |
| Set matchingSites = (Set) hosts.get(host); |
| Iterator<Site> sitesIter = matchingSites.iterator(); |
| while (sitesIter.hasNext()) { |
| Site site = sitesIter.next(); |
| // We should only set the title if the bookmark is for the root |
| // (i.e. www.google.com), as website settings act on the origin |
| // as a whole rather than a single page under that origin. If the |
| // user has bookmarked a page under the root but *not* the root, |
| // then we risk displaying the title of that page which may or |
| // may not have any relevance to the origin. |
| if (url.equals(site.getOrigin()) || |
| (new String(site.getOrigin()+"/")).equals(url)) { |
| mDataSetChanged = true; |
| site.setTitle(title); |
| } |
| |
| if (bmp != null) { |
| mDataSetChanged = true; |
| site.setIcon(bmp); |
| } |
| } |
| } |
| } while (c.moveToNext()); |
| } |
| c.close(); |
| } |
| return null; |
| } |
| |
| protected void onPostExecute(Void unused) { |
| if (mDataSetChanged) { |
| notifyDataSetChanged(); |
| } |
| } |
| } |
| |
| |
| public void populateOrigins(Map<String, Site> sites) { |
| clear(); |
| |
| // We can now simply populate our array with Site instances |
| Set<Map.Entry<String, Site>> elements = sites.entrySet(); |
| Iterator<Map.Entry<String, Site>> entryIterator = elements.iterator(); |
| while (entryIterator.hasNext()) { |
| Map.Entry<String, Site> entry = entryIterator.next(); |
| Site site = entry.getValue(); |
| add(site); |
| } |
| |
| notifyDataSetChanged(); |
| |
| if (getCount() == 0) { |
| finish(); // we close the screen |
| } |
| } |
| |
| public int getCount() { |
| if (mCurrentSite == null) { |
| return super.getCount(); |
| } |
| return mCurrentSite.getFeatureCount(); |
| } |
| |
| public String sizeValueToString(long bytes) { |
| // We display the size in MB, to 1dp, rounding up to the next 0.1MB. |
| // bytes should always be greater than zero. |
| if (bytes <= 0) { |
| Log.e(LOGTAG, "sizeValueToString called with non-positive value: " + bytes); |
| return "0"; |
| } |
| float megabytes = (float) bytes / (1024.0F * 1024.0F); |
| int truncated = (int) Math.ceil(megabytes * 10.0F); |
| float result = (float) (truncated / 10.0F); |
| return String.valueOf(result); |
| } |
| |
| /* |
| * If we receive the back event and are displaying |
| * site's settings, we want to go back to the main |
| * list view. If not, we just do nothing (see |
| * dispatchKeyEvent() below). |
| */ |
| public boolean backKeyPressed() { |
| if (mCurrentSite != null) { |
| mCurrentSite = null; |
| askForOrigins(); |
| return true; |
| } |
| return false; |
| } |
| |
| /** |
| * @hide |
| * Utility function |
| * Set the icon according to the usage |
| */ |
| public void setIconForUsage(ImageView usageIcon, long usageInBytes) { |
| float usageInMegabytes = (float) usageInBytes / (1024.0F * 1024.0F); |
| // We set the correct icon: |
| // 0 < empty < 0.1MB |
| // 0.1MB < low < 5MB |
| // 5MB < high |
| if (usageInMegabytes <= 0.1) { |
| usageIcon.setImageBitmap(mUsageEmptyIcon); |
| } else if (usageInMegabytes > 0.1 && usageInMegabytes <= 5) { |
| usageIcon.setImageBitmap(mUsageLowIcon); |
| } else if (usageInMegabytes > 5) { |
| usageIcon.setImageBitmap(mUsageHighIcon); |
| } |
| } |
| |
| @Override |
| public View getView(int position, View convertView, ViewGroup parent) { |
| View view; |
| final TextView title; |
| final TextView subtitle; |
| final ImageView icon; |
| final ImageView usageIcon; |
| final ImageView locationIcon; |
| final ImageView featureIcon; |
| |
| if (convertView == null) { |
| view = mInflater.inflate(mResource, parent, false); |
| } else { |
| view = convertView; |
| } |
| |
| title = (TextView) view.findViewById(R.id.title); |
| subtitle = (TextView) view.findViewById(R.id.subtitle); |
| icon = (ImageView) view.findViewById(R.id.icon); |
| featureIcon = (ImageView) view.findViewById(R.id.feature_icon); |
| usageIcon = (ImageView) view.findViewById(R.id.usage_icon); |
| locationIcon = (ImageView) view.findViewById(R.id.location_icon); |
| usageIcon.setVisibility(View.GONE); |
| locationIcon.setVisibility(View.GONE); |
| |
| if (mCurrentSite == null) { |
| |
| Site site = getItem(position); |
| title.setText(site.getPrettyTitle()); |
| String subtitleText = site.getPrettyOrigin(); |
| if (subtitleText != null) { |
| title.setMaxLines(1); |
| title.setSingleLine(true); |
| subtitle.setVisibility(View.VISIBLE); |
| subtitle.setText(subtitleText); |
| } else { |
| subtitle.setVisibility(View.GONE); |
| title.setMaxLines(2); |
| title.setSingleLine(false); |
| } |
| |
| icon.setVisibility(View.VISIBLE); |
| usageIcon.setVisibility(View.INVISIBLE); |
| locationIcon.setVisibility(View.INVISIBLE); |
| featureIcon.setVisibility(View.GONE); |
| Bitmap bmp = site.getIcon(); |
| if (bmp == null) { |
| bmp = mDefaultIcon; |
| } |
| icon.setImageBitmap(bmp); |
| // We set the site as the view's tag, |
| // so that we can get it in onItemClick() |
| view.setTag(site); |
| |
| String origin = site.getOrigin(); |
| if (site.hasFeature(Site.FEATURE_WEB_STORAGE)) { |
| WebStorage.getInstance().getUsageForOrigin(origin, new ValueCallback<Long>() { |
| public void onReceiveValue(Long value) { |
| if (value != null) { |
| setIconForUsage(usageIcon, value.longValue()); |
| usageIcon.setVisibility(View.VISIBLE); |
| } |
| } |
| }); |
| } |
| |
| if (site.hasFeature(Site.FEATURE_GEOLOCATION)) { |
| locationIcon.setVisibility(View.VISIBLE); |
| GeolocationPermissions.getInstance().getAllowed(origin, new ValueCallback<Boolean>() { |
| public void onReceiveValue(Boolean allowed) { |
| if (allowed != null) { |
| if (allowed.booleanValue()) { |
| locationIcon.setImageBitmap(mLocationAllowedIcon); |
| } else { |
| locationIcon.setImageBitmap(mLocationDisallowedIcon); |
| } |
| } |
| } |
| }); |
| } |
| } else { |
| icon.setVisibility(View.GONE); |
| locationIcon.setVisibility(View.GONE); |
| usageIcon.setVisibility(View.GONE); |
| featureIcon.setVisibility(View.VISIBLE); |
| String origin = mCurrentSite.getOrigin(); |
| switch (mCurrentSite.getFeatureByIndex(position)) { |
| case Site.FEATURE_WEB_STORAGE: |
| WebStorage.getInstance().getUsageForOrigin(origin, new ValueCallback<Long>() { |
| public void onReceiveValue(Long value) { |
| if (value != null) { |
| String usage = sizeValueToString(value.longValue()) + " " + sMBStored; |
| title.setText(R.string.webstorage_clear_data_title); |
| subtitle.setText(usage); |
| subtitle.setVisibility(View.VISIBLE); |
| setIconForUsage(featureIcon, value.longValue()); |
| } |
| } |
| }); |
| break; |
| case Site.FEATURE_GEOLOCATION: |
| title.setText(R.string.geolocation_settings_page_title); |
| GeolocationPermissions.getInstance().getAllowed(origin, new ValueCallback<Boolean>() { |
| public void onReceiveValue(Boolean allowed) { |
| if (allowed != null) { |
| if (allowed.booleanValue()) { |
| subtitle.setText(R.string.geolocation_settings_page_summary_allowed); |
| featureIcon.setImageBitmap(mLocationAllowedIcon); |
| } else { |
| subtitle.setText(R.string.geolocation_settings_page_summary_not_allowed); |
| featureIcon.setImageBitmap(mLocationDisallowedIcon); |
| } |
| subtitle.setVisibility(View.VISIBLE); |
| } |
| } |
| }); |
| break; |
| } |
| } |
| |
| return view; |
| } |
| |
| public void onItemClick(AdapterView<?> parent, |
| View view, |
| int position, |
| long id) { |
| if (mCurrentSite != null) { |
| switch (mCurrentSite.getFeatureByIndex(position)) { |
| case Site.FEATURE_WEB_STORAGE: |
| new AlertDialog.Builder(getContext()) |
| .setMessage(R.string.webstorage_clear_data_dialog_message) |
| .setPositiveButton(R.string.webstorage_clear_data_dialog_ok_button, |
| new AlertDialog.OnClickListener() { |
| public void onClick(DialogInterface dlg, int which) { |
| WebStorage.getInstance().deleteOrigin(mCurrentSite.getOrigin()); |
| // If this site has no more features, then go back to the |
| // origins list. |
| mCurrentSite.removeFeature(Site.FEATURE_WEB_STORAGE); |
| if (mCurrentSite.getFeatureCount() == 0) { |
| finish(); |
| } |
| askForOrigins(); |
| notifyDataSetChanged(); |
| }}) |
| .setNegativeButton(R.string.webstorage_clear_data_dialog_cancel_button, null) |
| .setIconAttribute(android.R.attr.alertDialogIcon) |
| .show(); |
| break; |
| case Site.FEATURE_GEOLOCATION: |
| new AlertDialog.Builder(getContext()) |
| .setMessage(R.string.geolocation_settings_page_dialog_message) |
| .setPositiveButton(R.string.geolocation_settings_page_dialog_ok_button, |
| new AlertDialog.OnClickListener() { |
| public void onClick(DialogInterface dlg, int which) { |
| GeolocationPermissions.getInstance().clear(mCurrentSite.getOrigin()); |
| mCurrentSite.removeFeature(Site.FEATURE_GEOLOCATION); |
| if (mCurrentSite.getFeatureCount() == 0) { |
| finish(); |
| } |
| askForOrigins(); |
| notifyDataSetChanged(); |
| }}) |
| .setNegativeButton(R.string.geolocation_settings_page_dialog_cancel_button, null) |
| .setIconAttribute(android.R.attr.alertDialogIcon) |
| .show(); |
| break; |
| } |
| } else { |
| Site site = (Site) view.getTag(); |
| PreferenceActivity activity = (PreferenceActivity) getActivity(); |
| if (activity != null) { |
| Bundle args = new Bundle(); |
| args.putParcelable(EXTRA_SITE, site); |
| activity.startPreferencePanel(WebsiteSettingsFragment.class.getName(), args, 0, |
| site.getPrettyTitle(), null, 0); |
| } |
| } |
| } |
| |
| public Site currentSite() { |
| return mCurrentSite; |
| } |
| } |
| |
| @Override |
| public View onCreateView(LayoutInflater inflater, ViewGroup container, |
| Bundle savedInstanceState) { |
| View view = inflater.inflate(R.layout.website_settings, container, false); |
| Bundle args = getArguments(); |
| if (args != null) { |
| mSite = (Site) args.getParcelable(EXTRA_SITE); |
| } |
| if (mSite == null) { |
| View clear = view.findViewById(R.id.clear_all_button); |
| clear.setVisibility(View.VISIBLE); |
| clear.setOnClickListener(this); |
| } |
| return view; |
| } |
| |
| @Override |
| public void onActivityCreated(Bundle savedInstanceState) { |
| super.onActivityCreated(savedInstanceState); |
| if (sMBStored == null) { |
| sMBStored = getString(R.string.webstorage_origin_summary_mb_stored); |
| } |
| mAdapter = new SiteAdapter(getActivity(), R.layout.website_settings_row); |
| if (mSite != null) { |
| mAdapter.mCurrentSite = mSite; |
| } |
| getListView().setAdapter(mAdapter); |
| getListView().setOnItemClickListener(mAdapter); |
| } |
| |
| @Override |
| public void onResume() { |
| super.onResume(); |
| mAdapter.askForOrigins(); |
| } |
| |
| private void finish() { |
| PreferenceActivity activity = (PreferenceActivity) getActivity(); |
| if (activity != null) { |
| activity.finishPreferencePanel(this, 0, null); |
| } |
| } |
| |
| @Override |
| public void onClick(View v) { |
| switch (v.getId()) { |
| case R.id.clear_all_button: |
| // Show the prompt to clear all origins of their data and geolocation permissions. |
| new AlertDialog.Builder(getActivity()) |
| .setMessage(R.string.website_settings_clear_all_dialog_message) |
| .setPositiveButton(R.string.website_settings_clear_all_dialog_ok_button, |
| new AlertDialog.OnClickListener() { |
| public void onClick(DialogInterface dlg, int which) { |
| WebStorage.getInstance().deleteAllData(); |
| GeolocationPermissions.getInstance().clearAll(); |
| WebStorageSizeManager.resetLastOutOfSpaceNotificationTime(); |
| mAdapter.askForOrigins(); |
| finish(); |
| }}) |
| .setNegativeButton(R.string.website_settings_clear_all_dialog_cancel_button, null) |
| .setIconAttribute(android.R.attr.alertDialogIcon) |
| .show(); |
| break; |
| } |
| } |
| } |