| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2016 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | package com.android.settings.dashboard; |
| 17 | |
| 18 | import android.app.Activity; |
| Fan Zhang | ff47b30 | 2018-11-09 14:52:42 -0800 | [diff] [blame] | 19 | import android.app.settings.SettingsEnums; |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 20 | import android.content.ContentResolver; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 21 | import android.content.Context; |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 22 | import android.os.Bundle; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 23 | import android.text.TextUtils; |
| 24 | import android.util.ArrayMap; |
| Sunny Shao | 3351c78 | 2019-11-18 16:38:19 +0800 | [diff] [blame] | 25 | import android.util.FeatureFlagUtils; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 26 | import android.util.Log; |
| 27 | |
| Yanting Yang | 4e56cb2 | 2019-04-18 21:37:31 +0800 | [diff] [blame] | 28 | import androidx.annotation.CallSuper; |
| Fan Zhang | 1c61a58 | 2018-07-26 11:26:11 -0700 | [diff] [blame] | 29 | import androidx.annotation.VisibleForTesting; |
| 30 | import androidx.preference.Preference; |
| Doris Ling | 8741c04 | 2018-08-27 14:11:55 -0700 | [diff] [blame] | 31 | import androidx.preference.PreferenceGroup; |
| Fan Zhang | 1c61a58 | 2018-07-26 11:26:11 -0700 | [diff] [blame] | 32 | import androidx.preference.PreferenceManager; |
| 33 | import androidx.preference.PreferenceScreen; |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 34 | import androidx.preference.SwitchPreference; |
| Fan Zhang | 1c61a58 | 2018-07-26 11:26:11 -0700 | [diff] [blame] | 35 | |
| Yanting Yang | 4e56cb2 | 2019-04-18 21:37:31 +0800 | [diff] [blame] | 36 | import com.android.settings.R; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 37 | import com.android.settings.SettingsPreferenceFragment; |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 38 | import com.android.settings.core.BasePreferenceController; |
| Sunny Shao | 3351c78 | 2019-11-18 16:38:19 +0800 | [diff] [blame] | 39 | import com.android.settings.core.FeatureFlags; |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 40 | import com.android.settings.core.PreferenceControllerListHelper; |
| tmfang | d97fba5 | 2018-06-14 14:48:35 +0800 | [diff] [blame] | 41 | import com.android.settings.core.SettingsBaseActivity; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 42 | import com.android.settings.overlay.FeatureFactory; |
| govenliu | b70ddb9 | 2020-07-16 17:09:03 +0800 | [diff] [blame] | 43 | import com.android.settings.widget.PrimarySwitchPreference; |
| Tony Mantler | 1d583e1 | 2017-06-13 13:09:25 -0700 | [diff] [blame] | 44 | import com.android.settingslib.core.AbstractPreferenceController; |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 45 | import com.android.settingslib.core.lifecycle.Lifecycle; |
| 46 | import com.android.settingslib.core.lifecycle.LifecycleObserver; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 47 | import com.android.settingslib.drawer.DashboardCategory; |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 48 | import com.android.settingslib.drawer.ProviderTile; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 49 | import com.android.settingslib.drawer.Tile; |
| Raff Tsai | 6db277e | 2019-10-02 16:43:23 +0800 | [diff] [blame] | 50 | import com.android.settingslib.search.Indexable; |
| Sunny Shao | 3351c78 | 2019-11-18 16:38:19 +0800 | [diff] [blame] | 51 | import com.android.settingslib.utils.ThreadUtils; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 52 | |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 53 | import java.util.ArrayList; |
| Yanting Yang | 4e56cb2 | 2019-04-18 21:37:31 +0800 | [diff] [blame] | 54 | import java.util.Arrays; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 55 | import java.util.Collection; |
| 56 | import java.util.List; |
| 57 | import java.util.Map; |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 58 | import java.util.Objects; |
| Jason Chiu | 20df25e | 2020-09-30 14:08:33 +0800 | [diff] [blame^] | 59 | import java.util.Set; |
| Sunny Shao | 3351c78 | 2019-11-18 16:38:19 +0800 | [diff] [blame] | 60 | import java.util.concurrent.ExecutionException; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 61 | |
| 62 | /** |
| 63 | * Base fragment for dashboard style UI containing a list of static and dynamic setting items. |
| 64 | */ |
| 65 | public abstract class DashboardFragment extends SettingsPreferenceFragment |
| tmfang | d97fba5 | 2018-06-14 14:48:35 +0800 | [diff] [blame] | 66 | implements SettingsBaseActivity.CategoryListener, Indexable, |
| Raff Tsai | 6db277e | 2019-10-02 16:43:23 +0800 | [diff] [blame] | 67 | PreferenceGroup.OnExpandButtonClickListener, |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 68 | BasePreferenceController.UiBlockListener { |
| Raff Tsai | f71db73 | 2019-12-02 10:55:29 +0800 | [diff] [blame] | 69 | public static final String CATEGORY = "category"; |
| Fan Zhang | db1112a | 2016-10-18 12:58:31 -0700 | [diff] [blame] | 70 | private static final String TAG = "DashboardFragment"; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 71 | |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 72 | @VisibleForTesting |
| 73 | final ArrayMap<String, List<DynamicDataObserver>> mDashboardTilePrefKeys = new ArrayMap<>(); |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 74 | private final Map<Class, List<AbstractPreferenceController>> mPreferenceControllers = |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 75 | new ArrayMap<>(); |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 76 | private final List<DynamicDataObserver> mRegisteredObservers = new ArrayList<>(); |
| hughchen | d49de61 | 2019-09-24 10:30:49 +0800 | [diff] [blame] | 77 | private final List<AbstractPreferenceController> mControllers = new ArrayList<>(); |
| Sunny Shao | e5e8240 | 2019-11-27 18:55:16 +0800 | [diff] [blame] | 78 | @VisibleForTesting |
| 79 | UiBlockerController mBlockerController; |
| Fan Zhang | b297abc | 2018-02-21 09:37:10 -0800 | [diff] [blame] | 80 | private DashboardFeatureProvider mDashboardFeatureProvider; |
| Fan Zhang | 9dc9c61 | 2016-11-11 13:23:21 -0800 | [diff] [blame] | 81 | private DashboardTilePlaceholderPreferenceController mPlaceholderPreferenceController; |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 82 | private boolean mListeningToCategoryChange; |
| Yanting Yang | 4e56cb2 | 2019-04-18 21:37:31 +0800 | [diff] [blame] | 83 | private List<String> mSuppressInjectedTileKeys; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 84 | |
| 85 | @Override |
| 86 | public void onAttach(Context context) { |
| 87 | super.onAttach(context); |
| Yanting Yang | 4e56cb2 | 2019-04-18 21:37:31 +0800 | [diff] [blame] | 88 | mSuppressInjectedTileKeys = Arrays.asList(context.getResources().getStringArray( |
| 89 | R.array.config_suppress_injected_tile_keys)); |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 90 | mDashboardFeatureProvider = FeatureFactory.getFactory(context). |
| 91 | getDashboardFeatureProvider(context); |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 92 | // Load preference controllers from code |
| 93 | final List<AbstractPreferenceController> controllersFromCode = |
| Fan Zhang | f7843ad | 2018-02-22 13:51:41 -0800 | [diff] [blame] | 94 | createPreferenceControllers(context); |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 95 | // Load preference controllers from xml definition |
| 96 | final List<BasePreferenceController> controllersFromXml = PreferenceControllerListHelper |
| 97 | .getPreferenceControllersFromXml(context, getPreferenceScreenResId()); |
| 98 | // Filter xml-based controllers in case a similar controller is created from code already. |
| 99 | final List<BasePreferenceController> uniqueControllerFromXml = |
| 100 | PreferenceControllerListHelper.filterControllers( |
| 101 | controllersFromXml, controllersFromCode); |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 102 | |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 103 | // Add unique controllers to list. |
| 104 | if (controllersFromCode != null) { |
| hughchen | d49de61 | 2019-09-24 10:30:49 +0800 | [diff] [blame] | 105 | mControllers.addAll(controllersFromCode); |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 106 | } |
| hughchen | d49de61 | 2019-09-24 10:30:49 +0800 | [diff] [blame] | 107 | mControllers.addAll(uniqueControllerFromXml); |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 108 | |
| 109 | // And wire up with lifecycle. |
| tmfang | 27c84de | 2018-06-28 11:39:05 +0800 | [diff] [blame] | 110 | final Lifecycle lifecycle = getSettingsLifecycle(); |
| Jason Chiu | 1a1f931 | 2019-10-01 17:35:00 +0800 | [diff] [blame] | 111 | uniqueControllerFromXml.forEach(controller -> { |
| 112 | if (controller instanceof LifecycleObserver) { |
| 113 | lifecycle.addObserver((LifecycleObserver) controller); |
| 114 | } |
| 115 | }); |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 116 | |
| Jason Chiu | 819abf7 | 2020-04-01 15:40:47 +0800 | [diff] [blame] | 117 | // Set metrics category for BasePreferenceController. |
| 118 | final int metricCategory = getMetricsCategory(); |
| 119 | mControllers.forEach(controller -> { |
| 120 | if (controller instanceof BasePreferenceController) { |
| 121 | ((BasePreferenceController) controller).setMetricsCategory(metricCategory); |
| 122 | } |
| 123 | }); |
| 124 | |
| Fan Zhang | 9dc9c61 | 2016-11-11 13:23:21 -0800 | [diff] [blame] | 125 | mPlaceholderPreferenceController = |
| 126 | new DashboardTilePlaceholderPreferenceController(context); |
| hughchen | d49de61 | 2019-09-24 10:30:49 +0800 | [diff] [blame] | 127 | mControllers.add(mPlaceholderPreferenceController); |
| 128 | for (AbstractPreferenceController controller : mControllers) { |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 129 | addPreferenceController(controller); |
| 130 | } |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 131 | } |
| 132 | |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 133 | @VisibleForTesting |
| 134 | void checkUiBlocker(List<AbstractPreferenceController> controllers) { |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 135 | final List<String> keys = new ArrayList<>(); |
| Jason Chiu | 1a1f931 | 2019-10-01 17:35:00 +0800 | [diff] [blame] | 136 | controllers.forEach(controller -> { |
| 137 | if (controller instanceof BasePreferenceController.UiBlocker |
| 138 | && controller.isAvailable()) { |
| 139 | ((BasePreferenceController) controller).setUiBlockListener(this); |
| 140 | keys.add(controller.getPreferenceKey()); |
| 141 | } |
| 142 | }); |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 143 | |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 144 | if (!keys.isEmpty()) { |
| 145 | mBlockerController = new UiBlockerController(keys); |
| Raff Tsai | 6db277e | 2019-10-02 16:43:23 +0800 | [diff] [blame] | 146 | mBlockerController.start(() -> updatePreferenceVisibility(mPreferenceControllers)); |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 147 | } |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 148 | } |
| 149 | |
| 150 | @Override |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 151 | public void onCreate(Bundle icicle) { |
| 152 | super.onCreate(icicle); |
| 153 | // Set ComparisonCallback so we get better animation when list changes. |
| 154 | getPreferenceManager().setPreferenceComparisonCallback( |
| 155 | new PreferenceManager.SimplePreferenceComparisonCallback()); |
| jeffreyhuang | bf234af | 2017-11-14 15:09:12 -0800 | [diff] [blame] | 156 | if (icicle != null) { |
| 157 | // Upon rotation configuration change we need to update preference states before any |
| 158 | // editing dialog is recreated (that would happen before onResume is called). |
| 159 | updatePreferenceStates(); |
| 160 | } |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 161 | } |
| 162 | |
| 163 | @Override |
| Jason Chiu | 20df25e | 2020-09-30 14:08:33 +0800 | [diff] [blame^] | 164 | public void onCategoriesChanged(Set<String> categories) { |
| 165 | final String categoryKey = getCategoryKey(); |
| 166 | final DashboardCategory dashboardCategory = |
| 167 | mDashboardFeatureProvider.getTilesForCategory(categoryKey); |
| 168 | if (dashboardCategory == null) { |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 169 | return; |
| 170 | } |
| Jason Chiu | 20df25e | 2020-09-30 14:08:33 +0800 | [diff] [blame^] | 171 | |
| 172 | if (categories == null) { |
| 173 | // force refreshing |
| 174 | refreshDashboardTiles(getLogTag()); |
| 175 | } else if (categories.contains(categoryKey)) { |
| 176 | Log.i(TAG, "refresh tiles for " + categoryKey); |
| 177 | refreshDashboardTiles(getLogTag()); |
| 178 | } |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 179 | } |
| 180 | |
| 181 | @Override |
| 182 | public void onCreatePreferences(Bundle savedInstanceState, String rootKey) { |
| hughchen | d49de61 | 2019-09-24 10:30:49 +0800 | [diff] [blame] | 183 | checkUiBlocker(mControllers); |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 184 | refreshAllPreferences(getLogTag()); |
| Jason Chiu | b12e3b9 | 2020-04-07 16:30:18 +0800 | [diff] [blame] | 185 | mControllers.stream() |
| 186 | .map(controller -> (Preference) findPreference(controller.getPreferenceKey())) |
| 187 | .filter(Objects::nonNull) |
| 188 | .forEach(preference -> { |
| 189 | // Give all controllers a chance to handle click. |
| 190 | preference.getExtras().putInt(CATEGORY, getMetricsCategory()); |
| 191 | }); |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 192 | } |
| 193 | |
| 194 | @Override |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 195 | public void onStart() { |
| 196 | super.onStart(); |
| Fan Zhang | fabbfb4 | 2016-10-07 12:41:43 -0700 | [diff] [blame] | 197 | final DashboardCategory category = |
| 198 | mDashboardFeatureProvider.getTilesForCategory(getCategoryKey()); |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 199 | if (category == null) { |
| 200 | return; |
| 201 | } |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 202 | final Activity activity = getActivity(); |
| tmfang | d97fba5 | 2018-06-14 14:48:35 +0800 | [diff] [blame] | 203 | if (activity instanceof SettingsBaseActivity) { |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 204 | mListeningToCategoryChange = true; |
| tmfang | d97fba5 | 2018-06-14 14:48:35 +0800 | [diff] [blame] | 205 | ((SettingsBaseActivity) activity).addCategoryListener(this); |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 206 | } |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 207 | final ContentResolver resolver = getContentResolver(); |
| 208 | mDashboardTilePrefKeys.values().stream() |
| 209 | .filter(Objects::nonNull) |
| 210 | .flatMap(List::stream) |
| 211 | .forEach(observer -> { |
| 212 | if (!mRegisteredObservers.contains(observer)) { |
| 213 | registerDynamicDataObserver(resolver, observer); |
| 214 | } |
| 215 | }); |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 216 | } |
| 217 | |
| 218 | @Override |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 219 | public void onResume() { |
| 220 | super.onResume(); |
| 221 | updatePreferenceStates(); |
| Sunny Shao | e5e8240 | 2019-11-27 18:55:16 +0800 | [diff] [blame] | 222 | writeElapsedTimeMetric(SettingsEnums.ACTION_DASHBOARD_VISIBLE_TIME, |
| 223 | "isParalleledControllers:" + isParalleledControllers()); |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 224 | } |
| 225 | |
| 226 | @Override |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 227 | public boolean onPreferenceTreeClick(Preference preference) { |
| Jason Chiu | 1281e59 | 2019-12-12 17:31:28 +0800 | [diff] [blame] | 228 | final Collection<List<AbstractPreferenceController>> controllers = |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 229 | mPreferenceControllers.values(); |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 230 | for (List<AbstractPreferenceController> controllerList : controllers) { |
| 231 | for (AbstractPreferenceController controller : controllerList) { |
| 232 | if (controller.handlePreferenceTreeClick(preference)) { |
| Jason Chiu | 1281e59 | 2019-12-12 17:31:28 +0800 | [diff] [blame] | 233 | // log here since calling super.onPreferenceTreeClick will be skipped |
| 234 | writePreferenceClickMetric(preference); |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 235 | return true; |
| 236 | } |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 237 | } |
| 238 | } |
| 239 | return super.onPreferenceTreeClick(preference); |
| 240 | } |
| 241 | |
| 242 | @Override |
| 243 | public void onStop() { |
| 244 | super.onStop(); |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 245 | unregisterDynamicDataObservers(new ArrayList<>(mRegisteredObservers)); |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 246 | if (mListeningToCategoryChange) { |
| 247 | final Activity activity = getActivity(); |
| tmfang | d97fba5 | 2018-06-14 14:48:35 +0800 | [diff] [blame] | 248 | if (activity instanceof SettingsBaseActivity) { |
| 249 | ((SettingsBaseActivity) activity).remCategoryListener(this); |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 250 | } |
| 251 | mListeningToCategoryChange = false; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 252 | } |
| 253 | } |
| 254 | |
| Fan Zhang | 9679dba | 2017-10-31 14:27:18 -0700 | [diff] [blame] | 255 | @Override |
| 256 | protected abstract int getPreferenceScreenResId(); |
| 257 | |
| Doris Ling | 8741c04 | 2018-08-27 14:11:55 -0700 | [diff] [blame] | 258 | @Override |
| 259 | public void onExpandButtonClick() { |
| Fan Zhang | ff47b30 | 2018-11-09 14:52:42 -0800 | [diff] [blame] | 260 | mMetricsFeatureProvider.action(SettingsEnums.PAGE_UNKNOWN, |
| Fan Zhang | 31b2100 | 2019-01-16 13:49:47 -0800 | [diff] [blame] | 261 | SettingsEnums.ACTION_SETTINGS_ADVANCED_BUTTON_EXPAND, |
| Fan Zhang | ff47b30 | 2018-11-09 14:52:42 -0800 | [diff] [blame] | 262 | getMetricsCategory(), null, 0); |
| Doris Ling | 8741c04 | 2018-08-27 14:11:55 -0700 | [diff] [blame] | 263 | } |
| 264 | |
| Fan Zhang | 7d5a9ee | 2018-08-09 17:32:37 -0700 | [diff] [blame] | 265 | protected boolean shouldForceRoundedIcon() { |
| 266 | return false; |
| 267 | } |
| 268 | |
| Fan Zhang | f7843ad | 2018-02-22 13:51:41 -0800 | [diff] [blame] | 269 | protected <T extends AbstractPreferenceController> T use(Class<T> clazz) { |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 270 | List<AbstractPreferenceController> controllerList = mPreferenceControllers.get(clazz); |
| 271 | if (controllerList != null) { |
| 272 | if (controllerList.size() > 1) { |
| 273 | Log.w(TAG, "Multiple controllers of Class " + clazz.getSimpleName() |
| 274 | + " found, returning first one."); |
| 275 | } |
| 276 | return (T) controllerList.get(0); |
| 277 | } |
| 278 | |
| 279 | return null; |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 280 | } |
| 281 | |
| Tony Mantler | 1d583e1 | 2017-06-13 13:09:25 -0700 | [diff] [blame] | 282 | protected void addPreferenceController(AbstractPreferenceController controller) { |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 283 | if (mPreferenceControllers.get(controller.getClass()) == null) { |
| 284 | mPreferenceControllers.put(controller.getClass(), new ArrayList<>()); |
| 285 | } |
| 286 | mPreferenceControllers.get(controller.getClass()).add(controller); |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 287 | } |
| 288 | |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 289 | /** |
| Fan Zhang | fabbfb4 | 2016-10-07 12:41:43 -0700 | [diff] [blame] | 290 | * Returns the CategoryKey for loading {@link DashboardCategory} for this fragment. |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 291 | */ |
| Fan Zhang | 26a1def | 2017-07-26 10:58:32 -0700 | [diff] [blame] | 292 | @VisibleForTesting |
| Fan Zhang | 7e6df83 | 2017-01-24 14:02:17 -0800 | [diff] [blame] | 293 | public String getCategoryKey() { |
| 294 | return DashboardFragmentRegistry.PARENT_TO_CATEGORY_KEY_MAP.get(getClass().getName()); |
| 295 | } |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 296 | |
| 297 | /** |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 298 | * Get the tag string for logging. |
| 299 | */ |
| 300 | protected abstract String getLogTag(); |
| 301 | |
| 302 | /** |
| Tony Mantler | 1d583e1 | 2017-06-13 13:09:25 -0700 | [diff] [blame] | 303 | * Get a list of {@link AbstractPreferenceController} for this fragment. |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 304 | */ |
| Fan Zhang | f7843ad | 2018-02-22 13:51:41 -0800 | [diff] [blame] | 305 | protected List<AbstractPreferenceController> createPreferenceControllers(Context context) { |
| Fan Zhang | 917f101 | 2018-02-21 15:22:25 -0800 | [diff] [blame] | 306 | return null; |
| 307 | } |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 308 | |
| 309 | /** |
| Doris Ling | 20d4b04 | 2016-11-22 16:37:06 -0800 | [diff] [blame] | 310 | * Returns true if this tile should be displayed |
| 311 | */ |
| Yanting Yang | 4e56cb2 | 2019-04-18 21:37:31 +0800 | [diff] [blame] | 312 | @CallSuper |
| Doris Ling | 20d4b04 | 2016-11-22 16:37:06 -0800 | [diff] [blame] | 313 | protected boolean displayTile(Tile tile) { |
| Yanting Yang | 4e56cb2 | 2019-04-18 21:37:31 +0800 | [diff] [blame] | 314 | if (mSuppressInjectedTileKeys != null && tile.hasKey()) { |
| 315 | // For suppressing injected tiles for OEMs. |
| 316 | return !mSuppressInjectedTileKeys.contains(tile.getKey(getContext())); |
| 317 | } |
| Doris Ling | 20d4b04 | 2016-11-22 16:37:06 -0800 | [diff] [blame] | 318 | return true; |
| 319 | } |
| 320 | |
| 321 | /** |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 322 | * Displays resource based tiles. |
| 323 | */ |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 324 | private void displayResourceTiles() { |
| 325 | final int resId = getPreferenceScreenResId(); |
| 326 | if (resId <= 0) { |
| 327 | return; |
| 328 | } |
| 329 | addPreferencesFromResource(resId); |
| 330 | final PreferenceScreen screen = getPreferenceScreen(); |
| Doris Ling | 8741c04 | 2018-08-27 14:11:55 -0700 | [diff] [blame] | 331 | screen.setOnExpandButtonClickListener(this); |
| Bonian Chen | 048d9fe | 2020-04-20 12:33:49 +0800 | [diff] [blame] | 332 | displayResourceTilesToScreen(screen); |
| 333 | } |
| 334 | |
| 335 | /** |
| 336 | * Perform {@link AbstractPreferenceController#displayPreference(PreferenceScreen)} |
| 337 | * on all {@link AbstractPreferenceController}s. |
| 338 | */ |
| 339 | protected void displayResourceTilesToScreen(PreferenceScreen screen) { |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 340 | mPreferenceControllers.values().stream().flatMap(Collection::stream).forEach( |
| 341 | controller -> controller.displayPreference(screen)); |
| Fan Zhang | 66b573a | 2016-10-06 16:33:13 -0700 | [diff] [blame] | 342 | } |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 343 | |
| 344 | /** |
| Sunny Shao | 3351c78 | 2019-11-18 16:38:19 +0800 | [diff] [blame] | 345 | * @return {@code true} if the underlying controllers should be executed in parallel. |
| 346 | * Override this function to enable/disable the behavior. |
| 347 | */ |
| 348 | protected boolean isParalleledControllers() { |
| 349 | return false; |
| 350 | } |
| 351 | |
| 352 | /** |
| Bonian Chen | 1b9bda3 | 2020-02-04 10:27:50 +0800 | [diff] [blame] | 353 | * Get current PreferenceController(s) |
| 354 | */ |
| 355 | protected Collection<List<AbstractPreferenceController>> getPreferenceControllers() { |
| 356 | return mPreferenceControllers.values(); |
| 357 | } |
| 358 | |
| 359 | /** |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 360 | * Update state of each preference managed by PreferenceController. |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 361 | */ |
| Fan Zhang | beddff8 | 2016-11-18 11:10:05 -0800 | [diff] [blame] | 362 | protected void updatePreferenceStates() { |
| Sunny Shao | 3351c78 | 2019-11-18 16:38:19 +0800 | [diff] [blame] | 363 | if (isParalleledControllers() && FeatureFlagUtils.isEnabled(getContext(), |
| 364 | FeatureFlags.CONTROLLER_ENHANCEMENT)) { |
| 365 | updatePreferenceStatesInParallel(); |
| 366 | return; |
| 367 | } |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 368 | final PreferenceScreen screen = getPreferenceScreen(); |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 369 | Collection<List<AbstractPreferenceController>> controllerLists = |
| 370 | mPreferenceControllers.values(); |
| 371 | for (List<AbstractPreferenceController> controllerList : controllerLists) { |
| 372 | for (AbstractPreferenceController controller : controllerList) { |
| 373 | if (!controller.isAvailable()) { |
| 374 | continue; |
| 375 | } |
| Jason Chiu | a42a93b | 2018-12-18 18:27:50 +0800 | [diff] [blame] | 376 | |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 377 | final String key = controller.getPreferenceKey(); |
| Jason Chiu | a42a93b | 2018-12-18 18:27:50 +0800 | [diff] [blame] | 378 | if (TextUtils.isEmpty(key)) { |
| 379 | Log.d(TAG, String.format("Preference key is %s in Controller %s", |
| 380 | key, controller.getClass().getSimpleName())); |
| 381 | continue; |
| 382 | } |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 383 | |
| Ben Lin | 9275177 | 2017-12-21 17:32:34 -0800 | [diff] [blame] | 384 | final Preference preference = screen.findPreference(key); |
| 385 | if (preference == null) { |
| 386 | Log.d(TAG, String.format("Cannot find preference with key %s in Controller %s", |
| 387 | key, controller.getClass().getSimpleName())); |
| 388 | continue; |
| 389 | } |
| 390 | controller.updateState(preference); |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 391 | } |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 392 | } |
| 393 | } |
| 394 | |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 395 | /** |
| Sunny Shao | 3351c78 | 2019-11-18 16:38:19 +0800 | [diff] [blame] | 396 | * Use parallel method to update state of each preference managed by PreferenceController. |
| 397 | */ |
| 398 | @VisibleForTesting |
| 399 | void updatePreferenceStatesInParallel() { |
| 400 | final PreferenceScreen screen = getPreferenceScreen(); |
| 401 | final Collection<List<AbstractPreferenceController>> controllerLists = |
| 402 | mPreferenceControllers.values(); |
| 403 | final List<ControllerFutureTask> taskList = new ArrayList<>(); |
| 404 | for (List<AbstractPreferenceController> controllerList : controllerLists) { |
| 405 | for (AbstractPreferenceController controller : controllerList) { |
| 406 | final ControllerFutureTask task = new ControllerFutureTask( |
| 407 | new ControllerTask(controller, screen, mMetricsFeatureProvider, |
| 408 | getMetricsCategory()), null /* result */); |
| 409 | taskList.add(task); |
| 410 | ThreadUtils.postOnBackgroundThread(task); |
| 411 | } |
| 412 | } |
| 413 | |
| 414 | for (ControllerFutureTask task : taskList) { |
| 415 | try { |
| 416 | task.get(); |
| 417 | } catch (InterruptedException | ExecutionException e) { |
| 418 | Log.w(TAG, task.getController().getPreferenceKey() + " " + e.getMessage()); |
| 419 | } |
| 420 | } |
| 421 | } |
| 422 | |
| 423 | /** |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 424 | * Refresh all preference items, including both static prefs from xml, and dynamic items from |
| 425 | * DashboardCategory. |
| 426 | */ |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 427 | private void refreshAllPreferences(final String tag) { |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 428 | final PreferenceScreen screen = getPreferenceScreen(); |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 429 | // First remove old preferences. |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 430 | if (screen != null) { |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 431 | // Intentionally do not cache PreferenceScreen because it will be recreated later. |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 432 | screen.removeAll(); |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 433 | } |
| 434 | |
| 435 | // Add resource based tiles. |
| 436 | displayResourceTiles(); |
| 437 | |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 438 | refreshDashboardTiles(tag); |
| Fan Zhang | 293883b | 2018-10-26 14:53:12 -0700 | [diff] [blame] | 439 | |
| 440 | final Activity activity = getActivity(); |
| 441 | if (activity != null) { |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 442 | Log.d(tag, "All preferences added, reporting fully drawn"); |
| Fan Zhang | 293883b | 2018-10-26 14:53:12 -0700 | [diff] [blame] | 443 | activity.reportFullyDrawn(); |
| 444 | } |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 445 | |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 446 | updatePreferenceVisibility(mPreferenceControllers); |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 447 | } |
| 448 | |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 449 | @VisibleForTesting |
| 450 | void updatePreferenceVisibility( |
| 451 | Map<Class, List<AbstractPreferenceController>> preferenceControllers) { |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 452 | final PreferenceScreen screen = getPreferenceScreen(); |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 453 | if (screen == null || preferenceControllers == null || mBlockerController == null) { |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 454 | return; |
| 455 | } |
| 456 | |
| 457 | final boolean visible = mBlockerController.isBlockerFinished(); |
| 458 | for (List<AbstractPreferenceController> controllerList : |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 459 | preferenceControllers.values()) { |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 460 | for (AbstractPreferenceController controller : controllerList) { |
| 461 | final String key = controller.getPreferenceKey(); |
| jackqdyulei | a5f1b5c | 2019-01-14 10:55:39 -0800 | [diff] [blame] | 462 | final Preference preference = findPreference(key); |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 463 | if (preference != null) { |
| 464 | preference.setVisible(visible && controller.isAvailable()); |
| 465 | } |
| 466 | } |
| 467 | } |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 468 | } |
| 469 | |
| 470 | /** |
| 471 | * Refresh preference items backed by DashboardCategory. |
| 472 | */ |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 473 | private void refreshDashboardTiles(final String tag) { |
| Fan Zhang | 8b5bca5 | 2016-10-19 12:00:32 -0700 | [diff] [blame] | 474 | final PreferenceScreen screen = getPreferenceScreen(); |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 475 | |
| Fan Zhang | fabbfb4 | 2016-10-07 12:41:43 -0700 | [diff] [blame] | 476 | final DashboardCategory category = |
| 477 | mDashboardFeatureProvider.getTilesForCategory(getCategoryKey()); |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 478 | if (category == null) { |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 479 | Log.d(tag, "NO dashboard tiles for " + tag); |
| Fan Zhang | 3692465 | 2016-10-07 08:38:48 -0700 | [diff] [blame] | 480 | return; |
| 481 | } |
| Doris Ling | bcb7635 | 2017-11-22 17:29:21 -0800 | [diff] [blame] | 482 | final List<Tile> tiles = category.getTiles(); |
| Fan Zhang | e6c60c2 | 2016-10-04 17:48:32 -0700 | [diff] [blame] | 483 | if (tiles == null) { |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 484 | Log.d(tag, "tile list is empty, skipping category " + category.key); |
| Fan Zhang | e6c60c2 | 2016-10-04 17:48:32 -0700 | [diff] [blame] | 485 | return; |
| 486 | } |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 487 | // Create a list to track which tiles are to be removed. |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 488 | final Map<String, List<DynamicDataObserver>> remove = new ArrayMap(mDashboardTilePrefKeys); |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 489 | |
| Fan Zhang | fabbfb4 | 2016-10-07 12:41:43 -0700 | [diff] [blame] | 490 | // Install dashboard tiles. |
| Fan Zhang | 7d5a9ee | 2018-08-09 17:32:37 -0700 | [diff] [blame] | 491 | final boolean forceRoundedIcons = shouldForceRoundedIcon(); |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 492 | for (Tile tile : tiles) { |
| 493 | final String key = mDashboardFeatureProvider.getDashboardKeyForTile(tile); |
| 494 | if (TextUtils.isEmpty(key)) { |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 495 | Log.d(tag, "tile does not contain a key, skipping " + tile); |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 496 | continue; |
| 497 | } |
| Doris Ling | 20d4b04 | 2016-11-22 16:37:06 -0800 | [diff] [blame] | 498 | if (!displayTile(tile)) { |
| 499 | continue; |
| 500 | } |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 501 | if (mDashboardTilePrefKeys.containsKey(key)) { |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 502 | // Have the key already, will rebind. |
| Doris Ling | 8b14a1a | 2017-08-17 14:45:12 -0700 | [diff] [blame] | 503 | final Preference preference = screen.findPreference(key); |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 504 | mDashboardFeatureProvider.bindPreferenceToTileAndGetObservers(getActivity(), |
| 505 | forceRoundedIcons, getMetricsCategory(), preference, tile, key, |
| Fan Zhang | 7d5a9ee | 2018-08-09 17:32:37 -0700 | [diff] [blame] | 506 | mPlaceholderPreferenceController.getOrder()); |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 507 | } else { |
| 508 | // Don't have this key, add it. |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 509 | final Preference pref = createPreference(tile); |
| 510 | final List<DynamicDataObserver> observers = |
| 511 | mDashboardFeatureProvider.bindPreferenceToTileAndGetObservers(getActivity(), |
| 512 | forceRoundedIcons, getMetricsCategory(), pref, tile, key, |
| 513 | mPlaceholderPreferenceController.getOrder()); |
| Doris Ling | 8b14a1a | 2017-08-17 14:45:12 -0700 | [diff] [blame] | 514 | screen.addPreference(pref); |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 515 | registerDynamicDataObservers(observers); |
| 516 | mDashboardTilePrefKeys.put(key, observers); |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 517 | } |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 518 | remove.remove(key); |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 519 | } |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 520 | // Finally remove tiles that are gone. |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 521 | for (Map.Entry<String, List<DynamicDataObserver>> entry : remove.entrySet()) { |
| 522 | final String key = entry.getKey(); |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 523 | mDashboardTilePrefKeys.remove(key); |
| Doris Ling | 8b14a1a | 2017-08-17 14:45:12 -0700 | [diff] [blame] | 524 | final Preference preference = screen.findPreference(key); |
| 525 | if (preference != null) { |
| 526 | screen.removePreference(preference); |
| 527 | } |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 528 | unregisterDynamicDataObservers(entry.getValue()); |
| Fan Zhang | a1a84e6 | 2016-10-19 14:15:34 -0700 | [diff] [blame] | 529 | } |
| 530 | } |
| jackqdyulei | 22904a0 | 2018-12-27 14:40:09 -0800 | [diff] [blame] | 531 | |
| 532 | @Override |
| 533 | public void onBlockerWorkFinished(BasePreferenceController controller) { |
| 534 | mBlockerController.countDown(controller.getPreferenceKey()); |
| 535 | } |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 536 | |
| 537 | @VisibleForTesting |
| 538 | Preference createPreference(Tile tile) { |
| 539 | return tile instanceof ProviderTile |
| 540 | ? new SwitchPreference(getPrefContext()) |
| Jason Chiu | 10e675d | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 541 | : tile.hasSwitch() |
| govenliu | b70ddb9 | 2020-07-16 17:09:03 +0800 | [diff] [blame] | 542 | ? new PrimarySwitchPreference(getPrefContext()) |
| Jason Chiu | 10e675d | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 543 | : new Preference(getPrefContext()); |
| Jason Chiu | bac5987 | 2019-10-24 17:30:34 +0800 | [diff] [blame] | 544 | } |
| 545 | |
| 546 | @VisibleForTesting |
| 547 | void registerDynamicDataObservers(List<DynamicDataObserver> observers) { |
| 548 | if (observers == null || observers.isEmpty()) { |
| 549 | return; |
| 550 | } |
| 551 | final ContentResolver resolver = getContentResolver(); |
| 552 | observers.forEach(observer -> registerDynamicDataObserver(resolver, observer)); |
| 553 | } |
| 554 | |
| 555 | private void registerDynamicDataObserver(ContentResolver resolver, |
| 556 | DynamicDataObserver observer) { |
| 557 | Log.d(TAG, "register observer: @" + Integer.toHexString(observer.hashCode()) |
| 558 | + ", uri: " + observer.getUri()); |
| 559 | resolver.registerContentObserver(observer.getUri(), false, observer); |
| 560 | mRegisteredObservers.add(observer); |
| 561 | } |
| 562 | |
| 563 | private void unregisterDynamicDataObservers(List<DynamicDataObserver> observers) { |
| 564 | if (observers == null || observers.isEmpty()) { |
| 565 | return; |
| 566 | } |
| 567 | final ContentResolver resolver = getContentResolver(); |
| 568 | observers.forEach(observer -> { |
| 569 | Log.d(TAG, "unregister observer: @" + Integer.toHexString(observer.hashCode()) |
| 570 | + ", uri: " + observer.getUri()); |
| 571 | mRegisteredObservers.remove(observer); |
| 572 | resolver.unregisterContentObserver(observer); |
| 573 | }); |
| 574 | } |
| Fan Zhang | bb6d260 | 2016-10-04 13:21:06 -0700 | [diff] [blame] | 575 | } |