Loading res/xml/unrestricted_data_access_settings.xml +5 −1 Original line number Diff line number Diff line Loading @@ -17,4 +17,8 @@ <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" android:title="@string/unrestricted_data_saver" /> xmlns:settings="http://schemas.android.com/apk/res-auto" android:key="unrestricted_data" android:title="@string/unrestricted_data_saver" settings:controller="com.android.settings.datausage.UnrestrictedDataAccessPreferenceController"> </PreferenceScreen> src/com/android/settings/datausage/UnrestrictedDataAccess.java +17 −295 Original line number Diff line number Diff line Loading @@ -14,64 +14,35 @@ package com.android.settings.datausage; import static com.android.settingslib.RestrictedLockUtils.checkIfMeteredDataRestricted; import android.app.Application; import android.content.Context; import android.os.Bundle; import android.os.UserHandle; import androidx.preference.Preference; import androidx.preference.PreferenceViewHolder; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.logging.nano.MetricsProto.MetricsEvent; import com.android.settings.R; import com.android.settings.SettingsPreferenceFragment; import com.android.settings.applications.AppStateBaseBridge; import com.android.settings.applications.appinfo.AppInfoDashboardFragment; import com.android.settings.datausage.AppStateDataUsageBridge.DataUsageState; import com.android.settings.overlay.FeatureFactory; import com.android.settings.widget.AppSwitchPreference; import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import com.android.settingslib.RestrictedPreferenceHelper; import com.android.settings.dashboard.DashboardFragment; import com.android.settingslib.applications.ApplicationsState; import com.android.settingslib.applications.ApplicationsState.AppEntry; import com.android.settingslib.applications.ApplicationsState.AppFilter; import java.util.ArrayList; public class UnrestrictedDataAccess extends DashboardFragment { public class UnrestrictedDataAccess extends SettingsPreferenceFragment implements ApplicationsState.Callbacks, AppStateBaseBridge.Callback, Preference.OnPreferenceChangeListener { private static final String TAG = "UnrestrictedDataAccess"; private static final int MENU_SHOW_SYSTEM = Menu.FIRST + 42; private static final String EXTRA_SHOW_SYSTEM = "show_system"; private ApplicationsState mApplicationsState; private AppStateDataUsageBridge mDataUsageBridge; private ApplicationsState.Session mSession; private DataSaverBackend mDataSaverBackend; private boolean mShowSystem; private boolean mExtraLoaded; private AppFilter mFilter; @Override public void onCreate(Bundle icicle) { super.onCreate(icicle); setAnimationAllowed(true); mApplicationsState = ApplicationsState.getInstance( (Application) getContext().getApplicationContext()); mDataSaverBackend = new DataSaverBackend(getContext()); mDataUsageBridge = new AppStateDataUsageBridge(mApplicationsState, this, mDataSaverBackend); mSession = mApplicationsState.newSession(this, getLifecycle()); mShowSystem = icicle != null && icicle.getBoolean(EXTRA_SHOW_SYSTEM); mFilter = mShowSystem ? ApplicationsState.FILTER_ALL_ENABLED : ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER; setHasOptionsMenu(true); use(UnrestrictedDataAccessPreferenceController.class).setParentFragment(this); } @Override Loading @@ -89,9 +60,10 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment item.setTitle(mShowSystem ? R.string.menu_hide_system : R.string.menu_show_system); mFilter = mShowSystem ? ApplicationsState.FILTER_ALL_ENABLED : ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER; if (mExtraLoaded) { rebuild(); } use(UnrestrictedDataAccessPreferenceController.class).setFilter(mFilter); use(UnrestrictedDataAccessPreferenceController.class).rebuild(); break; } return super.onOptionsItemSelected(item); Loading @@ -106,31 +78,15 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment @Override public void onViewCreated(View view, Bundle savedInstanceState) { super.onViewCreated(view, savedInstanceState); setLoading(true, false); } @Override public void onResume() { super.onResume(); mDataUsageBridge.resume(); } @Override public void onPause() { super.onPause(); mDataUsageBridge.pause(); } @Override public void onDestroy() { super.onDestroy(); mDataUsageBridge.release(); } @Override public void onExtraInfoUpdated() { mExtraLoaded = true; rebuild(); public void onAttach(Context context) { super.onAttach(context); mFilter = mShowSystem ? ApplicationsState.FILTER_ALL_ENABLED : ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER; use(UnrestrictedDataAccessPreferenceController.class).setSession(getLifecycle()); use(UnrestrictedDataAccessPreferenceController.class).setFilter(mFilter); } @Override Loading @@ -138,74 +94,9 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment return R.string.help_url_unrestricted_data_access; } private void rebuild() { ArrayList<AppEntry> apps = mSession.rebuild(mFilter, ApplicationsState.ALPHA_COMPARATOR); if (apps != null) { onRebuildComplete(apps); } } @Override public void onRunningStateChanged(boolean running) { } @Override public void onPackageListChanged() { } @Override public void onRebuildComplete(ArrayList<AppEntry> apps) { if (getContext() == null) return; cacheRemoveAllPrefs(getPreferenceScreen()); final int N = apps.size(); for (int i = 0; i < N; i++) { AppEntry entry = apps.get(i); if (!shouldAddPreference(entry)) { continue; } String key = entry.info.packageName + "|" + entry.info.uid; AccessPreference preference = (AccessPreference) getCachedPreference(key); if (preference == null) { preference = new AccessPreference(getPrefContext(), entry); preference.setKey(key); preference.setOnPreferenceChangeListener(this); getPreferenceScreen().addPreference(preference); } else { preference.setDisabledByAdmin(checkIfMeteredDataRestricted(getContext(), entry.info.packageName, UserHandle.getUserId(entry.info.uid))); preference.reuse(); } preference.setOrder(i); } setLoading(false, true); removeCachedPrefs(getPreferenceScreen()); } @Override public void onPackageIconChanged() { } @Override public void onPackageSizeChanged(String packageName) { } @Override public void onAllSizesComputed() { } @Override public void onLauncherInfoChanged() { } @Override public void onLoadEntriesCompleted() { protected String getLogTag() { return TAG; } @Override Loading @@ -217,173 +108,4 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment protected int getPreferenceScreenResId() { return R.xml.unrestricted_data_access_settings; } @Override public boolean onPreferenceChange(Preference preference, Object newValue) { if (preference instanceof AccessPreference) { AccessPreference accessPreference = (AccessPreference) preference; boolean whitelisted = newValue == Boolean.TRUE; logSpecialPermissionChange(whitelisted, accessPreference.mEntry.info.packageName); mDataSaverBackend.setIsWhitelisted(accessPreference.mEntry.info.uid, accessPreference.mEntry.info.packageName, whitelisted); accessPreference.mState.isDataSaverWhitelisted = whitelisted; return true; } return false; } @VisibleForTesting void logSpecialPermissionChange(boolean whitelisted, String packageName) { int logCategory = whitelisted ? MetricsEvent.APP_SPECIAL_PERMISSION_UNL_DATA_ALLOW : MetricsEvent.APP_SPECIAL_PERMISSION_UNL_DATA_DENY; FeatureFactory.getFactory(getContext()).getMetricsFeatureProvider().action(getContext(), logCategory, packageName); } @VisibleForTesting boolean shouldAddPreference(AppEntry app) { return app != null && UserHandle.isApp(app.info.uid); } @VisibleForTesting class AccessPreference extends AppSwitchPreference implements DataSaverBackend.Listener { private final AppEntry mEntry; private final DataUsageState mState; private final RestrictedPreferenceHelper mHelper; public AccessPreference(final Context context, AppEntry entry) { super(context); setWidgetLayoutResource(R.layout.restricted_switch_widget); mHelper = new RestrictedPreferenceHelper(context, this, null); mEntry = entry; mState = (DataUsageState) mEntry.extraInfo; mEntry.ensureLabel(getContext()); setDisabledByAdmin(checkIfMeteredDataRestricted(context, entry.info.packageName, UserHandle.getUserId(entry.info.uid))); setState(); if (mEntry.icon != null) { setIcon(mEntry.icon); } } @Override public void onAttached() { super.onAttached(); mDataSaverBackend.addListener(this); } @Override public void onDetached() { mDataSaverBackend.remListener(this); super.onDetached(); } @Override protected void onClick() { if (mState.isDataSaverBlacklisted) { // app is blacklisted, launch App Data Usage screen AppInfoDashboardFragment.startAppInfoFragment(AppDataUsage.class, R.string.app_data_usage, null /* arguments */, UnrestrictedDataAccess.this, mEntry); } else { // app is not blacklisted, let superclass handle toggle switch super.onClick(); } } @Override public void performClick() { if (!mHelper.performClick()) { super.performClick(); } } // Sets UI state based on whitelist/blacklist status. private void setState() { setTitle(mEntry.label); if (mState != null) { setChecked(mState.isDataSaverWhitelisted); if (isDisabledByAdmin()) { setSummary(R.string.disabled_by_admin); } else if (mState.isDataSaverBlacklisted) { setSummary(R.string.restrict_background_blacklisted); } else { setSummary(""); } } } public void reuse() { setState(); notifyChanged(); } @Override public void onBindViewHolder(PreferenceViewHolder holder) { if (mEntry.icon == null) { holder.itemView.post(new Runnable() { @Override public void run() { // Ensure we have an icon before binding. mApplicationsState.ensureIcon(mEntry); // This might trigger us to bind again, but it gives an easy way to only // load the icon once its needed, so its probably worth it. setIcon(mEntry.icon); } }); } final boolean disabledByAdmin = isDisabledByAdmin(); final View widgetFrame = holder.findViewById(android.R.id.widget_frame); if (disabledByAdmin) { widgetFrame.setVisibility(View.VISIBLE); } else { widgetFrame.setVisibility(mState != null && mState.isDataSaverBlacklisted ? View.INVISIBLE : View.VISIBLE); } super.onBindViewHolder(holder); mHelper.onBindViewHolder(holder); holder.findViewById(R.id.restricted_icon).setVisibility( disabledByAdmin ? View.VISIBLE : View.GONE); holder.findViewById(android.R.id.switch_widget).setVisibility( disabledByAdmin ? View.GONE : View.VISIBLE); } @Override public void onDataSaverChanged(boolean isDataSaving) { } @Override public void onWhitelistStatusChanged(int uid, boolean isWhitelisted) { if (mState != null && mEntry.info.uid == uid) { mState.isDataSaverWhitelisted = isWhitelisted; reuse(); } } @Override public void onBlacklistStatusChanged(int uid, boolean isBlacklisted) { if (mState != null && mEntry.info.uid == uid) { mState.isDataSaverBlacklisted = isBlacklisted; reuse(); } } public void setDisabledByAdmin(EnforcedAdmin admin) { mHelper.setDisabledByAdmin(admin); } public boolean isDisabledByAdmin() { return mHelper.isDisabledByAdmin(); } @VisibleForTesting public AppEntry getEntryForTest() { return mEntry; } } } src/com/android/settings/datausage/UnrestrictedDataAccessPreference.java 0 → 100644 +185 −0 Original line number Diff line number Diff line /* * Copyright (C) 2018 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.settings.datausage; import static com.android.settingslib.RestrictedLockUtils.checkIfMeteredDataRestricted; import android.content.Context; import android.os.UserHandle; import android.view.View; import com.android.settings.R; import com.android.settings.applications.appinfo.AppInfoDashboardFragment; import com.android.settings.dashboard.DashboardFragment; import com.android.settings.widget.AppSwitchPreference; import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import com.android.settingslib.RestrictedPreferenceHelper; import com.android.settingslib.applications.ApplicationsState; import com.android.settingslib.applications.ApplicationsState.AppEntry; import androidx.preference.PreferenceViewHolder; public class UnrestrictedDataAccessPreference extends AppSwitchPreference implements DataSaverBackend.Listener { private final ApplicationsState mApplicationsState; private final AppEntry mEntry; private final AppStateDataUsageBridge.DataUsageState mDataUsageState; private final DataSaverBackend mDataSaverBackend; private final DashboardFragment mParentFragment; private final RestrictedPreferenceHelper mHelper; public UnrestrictedDataAccessPreference(final Context context, AppEntry entry, ApplicationsState applicationsState, DataSaverBackend dataSaverBackend, DashboardFragment parentFragment) { super(context); setWidgetLayoutResource(R.layout.restricted_switch_widget); mHelper = new RestrictedPreferenceHelper(context, this, null); mEntry = entry; mDataUsageState = (AppStateDataUsageBridge.DataUsageState) mEntry.extraInfo; mEntry.ensureLabel(context); mApplicationsState = applicationsState; mDataSaverBackend = dataSaverBackend; mParentFragment = parentFragment; setDisabledByAdmin(checkIfMeteredDataRestricted(context, entry.info.packageName, UserHandle.getUserId(entry.info.uid))); updateState(); setKey(generateKey(mEntry)); if (mEntry.icon != null) { setIcon(mEntry.icon); } } static String generateKey(final AppEntry entry) { return entry.info.packageName + "|" + entry.info.uid; } @Override public void onAttached() { super.onAttached(); mDataSaverBackend.addListener(this); } @Override public void onDetached() { mDataSaverBackend.remListener(this); super.onDetached(); } @Override protected void onClick() { if (mDataUsageState.isDataSaverBlacklisted) { // app is blacklisted, launch App Data Usage screen AppInfoDashboardFragment.startAppInfoFragment(AppDataUsage.class, R.string.app_data_usage, null /* arguments */, mParentFragment, mEntry); } else { // app is not blacklisted, let superclass handle toggle switch super.onClick(); } } @Override public void performClick() { if (!mHelper.performClick()) { super.performClick(); } } @Override public void onBindViewHolder(PreferenceViewHolder holder) { if (mEntry.icon == null) { holder.itemView.post(new Runnable() { @Override public void run() { // Ensure we have an icon before binding. mApplicationsState.ensureIcon(mEntry); // This might trigger us to bind again, but it gives an easy way to only // load the icon once its needed, so its probably worth it. setIcon(mEntry.icon); } }); } final boolean disabledByAdmin = isDisabledByAdmin(); final View widgetFrame = holder.findViewById(android.R.id.widget_frame); if (disabledByAdmin) { widgetFrame.setVisibility(View.VISIBLE); } else { widgetFrame.setVisibility( mDataUsageState != null && mDataUsageState.isDataSaverBlacklisted ? View.INVISIBLE : View.VISIBLE); } super.onBindViewHolder(holder); mHelper.onBindViewHolder(holder); holder.findViewById(R.id.restricted_icon).setVisibility( disabledByAdmin ? View.VISIBLE : View.GONE); holder.findViewById(android.R.id.switch_widget).setVisibility( disabledByAdmin ? View.GONE : View.VISIBLE); } @Override public void onDataSaverChanged(boolean isDataSaving) { } @Override public void onWhitelistStatusChanged(int uid, boolean isWhitelisted) { if (mDataUsageState != null && mEntry.info.uid == uid) { mDataUsageState.isDataSaverWhitelisted = isWhitelisted; updateState(); } } @Override public void onBlacklistStatusChanged(int uid, boolean isBlacklisted) { if (mDataUsageState != null && mEntry.info.uid == uid) { mDataUsageState.isDataSaverBlacklisted = isBlacklisted; updateState(); } } public AppStateDataUsageBridge.DataUsageState getDataUsageState() { return mDataUsageState; } public AppEntry getEntry() { return mEntry; } public boolean isDisabledByAdmin() { return mHelper.isDisabledByAdmin(); } public void setDisabledByAdmin(EnforcedAdmin admin) { mHelper.setDisabledByAdmin(admin); } // Sets UI state based on whitelist/blacklist status. public void updateState() { setTitle(mEntry.label); if (mDataUsageState != null) { setChecked(mDataUsageState.isDataSaverWhitelisted); if (isDisabledByAdmin()) { setSummary(R.string.disabled_by_admin); } else if (mDataUsageState.isDataSaverBlacklisted) { setSummary(R.string.restrict_background_blacklisted); } else { setSummary(""); } } notifyChanged(); } } src/com/android/settings/datausage/UnrestrictedDataAccessPreferenceController.java 0 → 100644 +235 −0 File added.Preview size limit exceeded, changes collapsed. Show changes tests/robotests/assets/grandfather_not_implementing_index_provider +1 −0 Original line number Diff line number Diff line com.android.settings.applications.appinfo.AppInfoDashboardFragment com.android.settings.datausage.UnrestrictedDataAccess com.android.settings.bluetooth.DevicePickerFragment com.android.settings.bluetooth.BluetoothDeviceDetailsFragment com.android.settings.bluetooth.BluetoothPairingDetail Loading Loading
res/xml/unrestricted_data_access_settings.xml +5 −1 Original line number Diff line number Diff line Loading @@ -17,4 +17,8 @@ <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" android:title="@string/unrestricted_data_saver" /> xmlns:settings="http://schemas.android.com/apk/res-auto" android:key="unrestricted_data" android:title="@string/unrestricted_data_saver" settings:controller="com.android.settings.datausage.UnrestrictedDataAccessPreferenceController"> </PreferenceScreen>
src/com/android/settings/datausage/UnrestrictedDataAccess.java +17 −295 Original line number Diff line number Diff line Loading @@ -14,64 +14,35 @@ package com.android.settings.datausage; import static com.android.settingslib.RestrictedLockUtils.checkIfMeteredDataRestricted; import android.app.Application; import android.content.Context; import android.os.Bundle; import android.os.UserHandle; import androidx.preference.Preference; import androidx.preference.PreferenceViewHolder; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.logging.nano.MetricsProto.MetricsEvent; import com.android.settings.R; import com.android.settings.SettingsPreferenceFragment; import com.android.settings.applications.AppStateBaseBridge; import com.android.settings.applications.appinfo.AppInfoDashboardFragment; import com.android.settings.datausage.AppStateDataUsageBridge.DataUsageState; import com.android.settings.overlay.FeatureFactory; import com.android.settings.widget.AppSwitchPreference; import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import com.android.settingslib.RestrictedPreferenceHelper; import com.android.settings.dashboard.DashboardFragment; import com.android.settingslib.applications.ApplicationsState; import com.android.settingslib.applications.ApplicationsState.AppEntry; import com.android.settingslib.applications.ApplicationsState.AppFilter; import java.util.ArrayList; public class UnrestrictedDataAccess extends DashboardFragment { public class UnrestrictedDataAccess extends SettingsPreferenceFragment implements ApplicationsState.Callbacks, AppStateBaseBridge.Callback, Preference.OnPreferenceChangeListener { private static final String TAG = "UnrestrictedDataAccess"; private static final int MENU_SHOW_SYSTEM = Menu.FIRST + 42; private static final String EXTRA_SHOW_SYSTEM = "show_system"; private ApplicationsState mApplicationsState; private AppStateDataUsageBridge mDataUsageBridge; private ApplicationsState.Session mSession; private DataSaverBackend mDataSaverBackend; private boolean mShowSystem; private boolean mExtraLoaded; private AppFilter mFilter; @Override public void onCreate(Bundle icicle) { super.onCreate(icicle); setAnimationAllowed(true); mApplicationsState = ApplicationsState.getInstance( (Application) getContext().getApplicationContext()); mDataSaverBackend = new DataSaverBackend(getContext()); mDataUsageBridge = new AppStateDataUsageBridge(mApplicationsState, this, mDataSaverBackend); mSession = mApplicationsState.newSession(this, getLifecycle()); mShowSystem = icicle != null && icicle.getBoolean(EXTRA_SHOW_SYSTEM); mFilter = mShowSystem ? ApplicationsState.FILTER_ALL_ENABLED : ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER; setHasOptionsMenu(true); use(UnrestrictedDataAccessPreferenceController.class).setParentFragment(this); } @Override Loading @@ -89,9 +60,10 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment item.setTitle(mShowSystem ? R.string.menu_hide_system : R.string.menu_show_system); mFilter = mShowSystem ? ApplicationsState.FILTER_ALL_ENABLED : ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER; if (mExtraLoaded) { rebuild(); } use(UnrestrictedDataAccessPreferenceController.class).setFilter(mFilter); use(UnrestrictedDataAccessPreferenceController.class).rebuild(); break; } return super.onOptionsItemSelected(item); Loading @@ -106,31 +78,15 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment @Override public void onViewCreated(View view, Bundle savedInstanceState) { super.onViewCreated(view, savedInstanceState); setLoading(true, false); } @Override public void onResume() { super.onResume(); mDataUsageBridge.resume(); } @Override public void onPause() { super.onPause(); mDataUsageBridge.pause(); } @Override public void onDestroy() { super.onDestroy(); mDataUsageBridge.release(); } @Override public void onExtraInfoUpdated() { mExtraLoaded = true; rebuild(); public void onAttach(Context context) { super.onAttach(context); mFilter = mShowSystem ? ApplicationsState.FILTER_ALL_ENABLED : ApplicationsState.FILTER_DOWNLOADED_AND_LAUNCHER; use(UnrestrictedDataAccessPreferenceController.class).setSession(getLifecycle()); use(UnrestrictedDataAccessPreferenceController.class).setFilter(mFilter); } @Override Loading @@ -138,74 +94,9 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment return R.string.help_url_unrestricted_data_access; } private void rebuild() { ArrayList<AppEntry> apps = mSession.rebuild(mFilter, ApplicationsState.ALPHA_COMPARATOR); if (apps != null) { onRebuildComplete(apps); } } @Override public void onRunningStateChanged(boolean running) { } @Override public void onPackageListChanged() { } @Override public void onRebuildComplete(ArrayList<AppEntry> apps) { if (getContext() == null) return; cacheRemoveAllPrefs(getPreferenceScreen()); final int N = apps.size(); for (int i = 0; i < N; i++) { AppEntry entry = apps.get(i); if (!shouldAddPreference(entry)) { continue; } String key = entry.info.packageName + "|" + entry.info.uid; AccessPreference preference = (AccessPreference) getCachedPreference(key); if (preference == null) { preference = new AccessPreference(getPrefContext(), entry); preference.setKey(key); preference.setOnPreferenceChangeListener(this); getPreferenceScreen().addPreference(preference); } else { preference.setDisabledByAdmin(checkIfMeteredDataRestricted(getContext(), entry.info.packageName, UserHandle.getUserId(entry.info.uid))); preference.reuse(); } preference.setOrder(i); } setLoading(false, true); removeCachedPrefs(getPreferenceScreen()); } @Override public void onPackageIconChanged() { } @Override public void onPackageSizeChanged(String packageName) { } @Override public void onAllSizesComputed() { } @Override public void onLauncherInfoChanged() { } @Override public void onLoadEntriesCompleted() { protected String getLogTag() { return TAG; } @Override Loading @@ -217,173 +108,4 @@ public class UnrestrictedDataAccess extends SettingsPreferenceFragment protected int getPreferenceScreenResId() { return R.xml.unrestricted_data_access_settings; } @Override public boolean onPreferenceChange(Preference preference, Object newValue) { if (preference instanceof AccessPreference) { AccessPreference accessPreference = (AccessPreference) preference; boolean whitelisted = newValue == Boolean.TRUE; logSpecialPermissionChange(whitelisted, accessPreference.mEntry.info.packageName); mDataSaverBackend.setIsWhitelisted(accessPreference.mEntry.info.uid, accessPreference.mEntry.info.packageName, whitelisted); accessPreference.mState.isDataSaverWhitelisted = whitelisted; return true; } return false; } @VisibleForTesting void logSpecialPermissionChange(boolean whitelisted, String packageName) { int logCategory = whitelisted ? MetricsEvent.APP_SPECIAL_PERMISSION_UNL_DATA_ALLOW : MetricsEvent.APP_SPECIAL_PERMISSION_UNL_DATA_DENY; FeatureFactory.getFactory(getContext()).getMetricsFeatureProvider().action(getContext(), logCategory, packageName); } @VisibleForTesting boolean shouldAddPreference(AppEntry app) { return app != null && UserHandle.isApp(app.info.uid); } @VisibleForTesting class AccessPreference extends AppSwitchPreference implements DataSaverBackend.Listener { private final AppEntry mEntry; private final DataUsageState mState; private final RestrictedPreferenceHelper mHelper; public AccessPreference(final Context context, AppEntry entry) { super(context); setWidgetLayoutResource(R.layout.restricted_switch_widget); mHelper = new RestrictedPreferenceHelper(context, this, null); mEntry = entry; mState = (DataUsageState) mEntry.extraInfo; mEntry.ensureLabel(getContext()); setDisabledByAdmin(checkIfMeteredDataRestricted(context, entry.info.packageName, UserHandle.getUserId(entry.info.uid))); setState(); if (mEntry.icon != null) { setIcon(mEntry.icon); } } @Override public void onAttached() { super.onAttached(); mDataSaverBackend.addListener(this); } @Override public void onDetached() { mDataSaverBackend.remListener(this); super.onDetached(); } @Override protected void onClick() { if (mState.isDataSaverBlacklisted) { // app is blacklisted, launch App Data Usage screen AppInfoDashboardFragment.startAppInfoFragment(AppDataUsage.class, R.string.app_data_usage, null /* arguments */, UnrestrictedDataAccess.this, mEntry); } else { // app is not blacklisted, let superclass handle toggle switch super.onClick(); } } @Override public void performClick() { if (!mHelper.performClick()) { super.performClick(); } } // Sets UI state based on whitelist/blacklist status. private void setState() { setTitle(mEntry.label); if (mState != null) { setChecked(mState.isDataSaverWhitelisted); if (isDisabledByAdmin()) { setSummary(R.string.disabled_by_admin); } else if (mState.isDataSaverBlacklisted) { setSummary(R.string.restrict_background_blacklisted); } else { setSummary(""); } } } public void reuse() { setState(); notifyChanged(); } @Override public void onBindViewHolder(PreferenceViewHolder holder) { if (mEntry.icon == null) { holder.itemView.post(new Runnable() { @Override public void run() { // Ensure we have an icon before binding. mApplicationsState.ensureIcon(mEntry); // This might trigger us to bind again, but it gives an easy way to only // load the icon once its needed, so its probably worth it. setIcon(mEntry.icon); } }); } final boolean disabledByAdmin = isDisabledByAdmin(); final View widgetFrame = holder.findViewById(android.R.id.widget_frame); if (disabledByAdmin) { widgetFrame.setVisibility(View.VISIBLE); } else { widgetFrame.setVisibility(mState != null && mState.isDataSaverBlacklisted ? View.INVISIBLE : View.VISIBLE); } super.onBindViewHolder(holder); mHelper.onBindViewHolder(holder); holder.findViewById(R.id.restricted_icon).setVisibility( disabledByAdmin ? View.VISIBLE : View.GONE); holder.findViewById(android.R.id.switch_widget).setVisibility( disabledByAdmin ? View.GONE : View.VISIBLE); } @Override public void onDataSaverChanged(boolean isDataSaving) { } @Override public void onWhitelistStatusChanged(int uid, boolean isWhitelisted) { if (mState != null && mEntry.info.uid == uid) { mState.isDataSaverWhitelisted = isWhitelisted; reuse(); } } @Override public void onBlacklistStatusChanged(int uid, boolean isBlacklisted) { if (mState != null && mEntry.info.uid == uid) { mState.isDataSaverBlacklisted = isBlacklisted; reuse(); } } public void setDisabledByAdmin(EnforcedAdmin admin) { mHelper.setDisabledByAdmin(admin); } public boolean isDisabledByAdmin() { return mHelper.isDisabledByAdmin(); } @VisibleForTesting public AppEntry getEntryForTest() { return mEntry; } } }
src/com/android/settings/datausage/UnrestrictedDataAccessPreference.java 0 → 100644 +185 −0 Original line number Diff line number Diff line /* * Copyright (C) 2018 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.settings.datausage; import static com.android.settingslib.RestrictedLockUtils.checkIfMeteredDataRestricted; import android.content.Context; import android.os.UserHandle; import android.view.View; import com.android.settings.R; import com.android.settings.applications.appinfo.AppInfoDashboardFragment; import com.android.settings.dashboard.DashboardFragment; import com.android.settings.widget.AppSwitchPreference; import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import com.android.settingslib.RestrictedPreferenceHelper; import com.android.settingslib.applications.ApplicationsState; import com.android.settingslib.applications.ApplicationsState.AppEntry; import androidx.preference.PreferenceViewHolder; public class UnrestrictedDataAccessPreference extends AppSwitchPreference implements DataSaverBackend.Listener { private final ApplicationsState mApplicationsState; private final AppEntry mEntry; private final AppStateDataUsageBridge.DataUsageState mDataUsageState; private final DataSaverBackend mDataSaverBackend; private final DashboardFragment mParentFragment; private final RestrictedPreferenceHelper mHelper; public UnrestrictedDataAccessPreference(final Context context, AppEntry entry, ApplicationsState applicationsState, DataSaverBackend dataSaverBackend, DashboardFragment parentFragment) { super(context); setWidgetLayoutResource(R.layout.restricted_switch_widget); mHelper = new RestrictedPreferenceHelper(context, this, null); mEntry = entry; mDataUsageState = (AppStateDataUsageBridge.DataUsageState) mEntry.extraInfo; mEntry.ensureLabel(context); mApplicationsState = applicationsState; mDataSaverBackend = dataSaverBackend; mParentFragment = parentFragment; setDisabledByAdmin(checkIfMeteredDataRestricted(context, entry.info.packageName, UserHandle.getUserId(entry.info.uid))); updateState(); setKey(generateKey(mEntry)); if (mEntry.icon != null) { setIcon(mEntry.icon); } } static String generateKey(final AppEntry entry) { return entry.info.packageName + "|" + entry.info.uid; } @Override public void onAttached() { super.onAttached(); mDataSaverBackend.addListener(this); } @Override public void onDetached() { mDataSaverBackend.remListener(this); super.onDetached(); } @Override protected void onClick() { if (mDataUsageState.isDataSaverBlacklisted) { // app is blacklisted, launch App Data Usage screen AppInfoDashboardFragment.startAppInfoFragment(AppDataUsage.class, R.string.app_data_usage, null /* arguments */, mParentFragment, mEntry); } else { // app is not blacklisted, let superclass handle toggle switch super.onClick(); } } @Override public void performClick() { if (!mHelper.performClick()) { super.performClick(); } } @Override public void onBindViewHolder(PreferenceViewHolder holder) { if (mEntry.icon == null) { holder.itemView.post(new Runnable() { @Override public void run() { // Ensure we have an icon before binding. mApplicationsState.ensureIcon(mEntry); // This might trigger us to bind again, but it gives an easy way to only // load the icon once its needed, so its probably worth it. setIcon(mEntry.icon); } }); } final boolean disabledByAdmin = isDisabledByAdmin(); final View widgetFrame = holder.findViewById(android.R.id.widget_frame); if (disabledByAdmin) { widgetFrame.setVisibility(View.VISIBLE); } else { widgetFrame.setVisibility( mDataUsageState != null && mDataUsageState.isDataSaverBlacklisted ? View.INVISIBLE : View.VISIBLE); } super.onBindViewHolder(holder); mHelper.onBindViewHolder(holder); holder.findViewById(R.id.restricted_icon).setVisibility( disabledByAdmin ? View.VISIBLE : View.GONE); holder.findViewById(android.R.id.switch_widget).setVisibility( disabledByAdmin ? View.GONE : View.VISIBLE); } @Override public void onDataSaverChanged(boolean isDataSaving) { } @Override public void onWhitelistStatusChanged(int uid, boolean isWhitelisted) { if (mDataUsageState != null && mEntry.info.uid == uid) { mDataUsageState.isDataSaverWhitelisted = isWhitelisted; updateState(); } } @Override public void onBlacklistStatusChanged(int uid, boolean isBlacklisted) { if (mDataUsageState != null && mEntry.info.uid == uid) { mDataUsageState.isDataSaverBlacklisted = isBlacklisted; updateState(); } } public AppStateDataUsageBridge.DataUsageState getDataUsageState() { return mDataUsageState; } public AppEntry getEntry() { return mEntry; } public boolean isDisabledByAdmin() { return mHelper.isDisabledByAdmin(); } public void setDisabledByAdmin(EnforcedAdmin admin) { mHelper.setDisabledByAdmin(admin); } // Sets UI state based on whitelist/blacklist status. public void updateState() { setTitle(mEntry.label); if (mDataUsageState != null) { setChecked(mDataUsageState.isDataSaverWhitelisted); if (isDisabledByAdmin()) { setSummary(R.string.disabled_by_admin); } else if (mDataUsageState.isDataSaverBlacklisted) { setSummary(R.string.restrict_background_blacklisted); } else { setSummary(""); } } notifyChanged(); } }
src/com/android/settings/datausage/UnrestrictedDataAccessPreferenceController.java 0 → 100644 +235 −0 File added.Preview size limit exceeded, changes collapsed. Show changes
tests/robotests/assets/grandfather_not_implementing_index_provider +1 −0 Original line number Diff line number Diff line com.android.settings.applications.appinfo.AppInfoDashboardFragment com.android.settings.datausage.UnrestrictedDataAccess com.android.settings.bluetooth.DevicePickerFragment com.android.settings.bluetooth.BluetoothDeviceDetailsFragment com.android.settings.bluetooth.BluetoothPairingDetail Loading