1 /* 2 * Copyright (C) 2018 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 17 package com.android.server.wifi; 18 19 import static com.android.server.wifi.ConnectToNetworkNotificationBuilder.ACTION_CONNECT_TO_NETWORK; 20 import static com.android.server.wifi.ConnectToNetworkNotificationBuilder.ACTION_PICK_WIFI_NETWORK; 21 import static com.android.server.wifi.ConnectToNetworkNotificationBuilder.ACTION_PICK_WIFI_NETWORK_AFTER_CONNECT_FAILURE; 22 import static com.android.server.wifi.ConnectToNetworkNotificationBuilder.ACTION_USER_DISMISSED_NOTIFICATION; 23 import static com.android.server.wifi.ConnectToNetworkNotificationBuilder.AVAILABLE_NETWORK_NOTIFIER_TAG; 24 25 import android.annotation.IntDef; 26 import android.annotation.NonNull; 27 import android.app.Notification; 28 import android.app.NotificationManager; 29 import android.content.BroadcastReceiver; 30 import android.content.Context; 31 import android.content.Intent; 32 import android.content.IntentFilter; 33 import android.database.ContentObserver; 34 import android.net.wifi.ScanResult; 35 import android.net.wifi.WifiConfiguration; 36 import android.net.wifi.WifiManager; 37 import android.os.Handler; 38 import android.os.Looper; 39 import android.os.Message; 40 import android.os.Messenger; 41 import android.os.Process; 42 import android.os.UserHandle; 43 import android.os.UserManager; 44 import android.provider.Settings; 45 import android.text.TextUtils; 46 import android.util.ArraySet; 47 import android.util.Log; 48 49 import com.android.internal.annotations.VisibleForTesting; 50 import com.android.server.wifi.nano.WifiMetricsProto.ConnectToNetworkNotificationAndActionCount; 51 import com.android.server.wifi.util.ScanResultUtil; 52 53 import java.io.FileDescriptor; 54 import java.io.PrintWriter; 55 import java.lang.annotation.Retention; 56 import java.lang.annotation.RetentionPolicy; 57 import java.util.List; 58 import java.util.Set; 59 60 /** 61 * Base class for all network notifiers (e.g. OpenNetworkNotifier, CarrierNetworkNotifier). 62 * 63 * NOTE: These API's are not thread safe and should only be used from WifiCoreThread. 64 */ 65 public class AvailableNetworkNotifier { 66 67 /** Time in milliseconds to display the Connecting notification. */ 68 private static final int TIME_TO_SHOW_CONNECTING_MILLIS = 10000; 69 70 /** Time in milliseconds to display the Connected notification. */ 71 private static final int TIME_TO_SHOW_CONNECTED_MILLIS = 5000; 72 73 /** Time in milliseconds to display the Failed To Connect notification. */ 74 private static final int TIME_TO_SHOW_FAILED_MILLIS = 5000; 75 76 /** The state of the notification */ 77 @IntDef({ 78 STATE_NO_NOTIFICATION, 79 STATE_SHOWING_RECOMMENDATION_NOTIFICATION, 80 STATE_CONNECTING_IN_NOTIFICATION, 81 STATE_CONNECTED_NOTIFICATION, 82 STATE_CONNECT_FAILED_NOTIFICATION 83 }) 84 @Retention(RetentionPolicy.SOURCE) 85 private @interface State {} 86 87 /** No recommendation is made and no notifications are shown. */ 88 private static final int STATE_NO_NOTIFICATION = 0; 89 /** The initial notification recommending a network to connect to is shown. */ 90 private static final int STATE_SHOWING_RECOMMENDATION_NOTIFICATION = 1; 91 /** The notification of status of connecting to the recommended network is shown. */ 92 private static final int STATE_CONNECTING_IN_NOTIFICATION = 2; 93 /** The notification that the connection to the recommended network was successful is shown. */ 94 private static final int STATE_CONNECTED_NOTIFICATION = 3; 95 /** The notification to show that connection to the recommended network failed is shown. */ 96 private static final int STATE_CONNECT_FAILED_NOTIFICATION = 4; 97 98 /** Current state of the notification. */ 99 @State private int mState = STATE_NO_NOTIFICATION; 100 101 /** 102 * The {@link Clock#getWallClockMillis()} must be at least this value for us 103 * to show the notification again. 104 */ 105 private long mNotificationRepeatTime; 106 /** 107 * When a notification is shown, we wait this amount before possibly showing it again. 108 */ 109 private final long mNotificationRepeatDelay; 110 /** Default repeat delay in seconds. */ 111 @VisibleForTesting 112 static final int DEFAULT_REPEAT_DELAY_SEC = 900; 113 114 /** Whether the user has set the setting to show the 'available networks' notification. */ 115 private boolean mSettingEnabled; 116 /** Whether the screen is on or not. */ 117 private boolean mScreenOn; 118 119 /** List of SSIDs blacklisted from recommendation. */ 120 private final Set<String> mBlacklistedSsids = new ArraySet<>(); 121 122 private final Context mContext; 123 private final Handler mHandler; 124 private final FrameworkFacade mFrameworkFacade; 125 private final WifiMetrics mWifiMetrics; 126 private final Clock mClock; 127 private final WifiConfigManager mConfigManager; 128 private final ClientModeImpl mClientModeImpl; 129 private final Messenger mSrcMessenger; 130 private final ConnectToNetworkNotificationBuilder mNotificationBuilder; 131 132 private ScanResult mRecommendedNetwork; 133 134 /** Tag used for logs and metrics */ 135 private final String mTag; 136 /** Identifier of the {@link SsidSetStoreData}. */ 137 private final String mStoreDataIdentifier; 138 /** Identifier for the settings toggle, used for registering ContentObserver */ 139 private final String mToggleSettingsName; 140 141 /** System wide identifier for notification in Notification Manager */ 142 private final int mSystemMessageNotificationId; 143 144 /** 145 * The nominator id for this class, from 146 * {@link com.android.server.wifi.nano.WifiMetricsProto.ConnectionEvent.ConnectionNominator} 147 */ 148 private final int mNominatorId; 149 AvailableNetworkNotifier( String tag, String storeDataIdentifier, String toggleSettingsName, int notificationIdentifier, int nominatorId, Context context, Looper looper, FrameworkFacade framework, Clock clock, WifiMetrics wifiMetrics, WifiConfigManager wifiConfigManager, WifiConfigStore wifiConfigStore, ClientModeImpl clientModeImpl, ConnectToNetworkNotificationBuilder connectToNetworkNotificationBuilder)150 public AvailableNetworkNotifier( 151 String tag, 152 String storeDataIdentifier, 153 String toggleSettingsName, 154 int notificationIdentifier, 155 int nominatorId, 156 Context context, 157 Looper looper, 158 FrameworkFacade framework, 159 Clock clock, 160 WifiMetrics wifiMetrics, 161 WifiConfigManager wifiConfigManager, 162 WifiConfigStore wifiConfigStore, 163 ClientModeImpl clientModeImpl, 164 ConnectToNetworkNotificationBuilder connectToNetworkNotificationBuilder) { 165 mTag = tag; 166 mStoreDataIdentifier = storeDataIdentifier; 167 mToggleSettingsName = toggleSettingsName; 168 mSystemMessageNotificationId = notificationIdentifier; 169 mNominatorId = nominatorId; 170 mContext = context; 171 mHandler = new Handler(looper); 172 mFrameworkFacade = framework; 173 mWifiMetrics = wifiMetrics; 174 mClock = clock; 175 mConfigManager = wifiConfigManager; 176 mClientModeImpl = clientModeImpl; 177 mNotificationBuilder = connectToNetworkNotificationBuilder; 178 mScreenOn = false; 179 mSrcMessenger = new Messenger(new Handler(looper, mConnectionStateCallback)); 180 wifiConfigStore.registerStoreData(new SsidSetStoreData(mStoreDataIdentifier, 181 new AvailableNetworkNotifierStoreData())); 182 183 // Setting is in seconds 184 mNotificationRepeatDelay = mFrameworkFacade.getIntegerSetting(context, 185 Settings.Global.WIFI_NETWORKS_AVAILABLE_REPEAT_DELAY, 186 DEFAULT_REPEAT_DELAY_SEC) * 1000L; 187 NotificationEnabledSettingObserver settingObserver = new NotificationEnabledSettingObserver( 188 mHandler); 189 settingObserver.register(); 190 191 IntentFilter filter = new IntentFilter(); 192 filter.addAction(ACTION_USER_DISMISSED_NOTIFICATION); 193 filter.addAction(ACTION_CONNECT_TO_NETWORK); 194 filter.addAction(ACTION_PICK_WIFI_NETWORK); 195 filter.addAction(ACTION_PICK_WIFI_NETWORK_AFTER_CONNECT_FAILURE); 196 mContext.registerReceiver( 197 mBroadcastReceiver, filter, null /* broadcastPermission */, mHandler); 198 } 199 200 private final BroadcastReceiver mBroadcastReceiver = 201 new BroadcastReceiver() { 202 @Override 203 public void onReceive(Context context, Intent intent) { 204 if (!mTag.equals(intent.getExtra(AVAILABLE_NETWORK_NOTIFIER_TAG))) { 205 return; 206 } 207 switch (intent.getAction()) { 208 case ACTION_USER_DISMISSED_NOTIFICATION: 209 handleUserDismissedAction(); 210 break; 211 case ACTION_CONNECT_TO_NETWORK: 212 handleConnectToNetworkAction(); 213 break; 214 case ACTION_PICK_WIFI_NETWORK: 215 handleSeeAllNetworksAction(); 216 break; 217 case ACTION_PICK_WIFI_NETWORK_AFTER_CONNECT_FAILURE: 218 handlePickWifiNetworkAfterConnectFailure(); 219 break; 220 default: 221 Log.e(mTag, "Unknown action " + intent.getAction()); 222 } 223 } 224 }; 225 226 private final Handler.Callback mConnectionStateCallback = (Message msg) -> { 227 switch (msg.what) { 228 // Success here means that an attempt to connect to the network has been initiated. 229 // Successful connection updates are received via the 230 // WifiConnectivityManager#handleConnectionStateChanged() callback. 231 case WifiManager.CONNECT_NETWORK_SUCCEEDED: 232 break; 233 case WifiManager.CONNECT_NETWORK_FAILED: 234 handleConnectionAttemptFailedToSend(); 235 break; 236 default: 237 Log.e("AvailableNetworkNotifier", "Unknown message " + msg.what); 238 } 239 return true; 240 }; 241 242 /** 243 * Clears the pending notification. This is called by {@link WifiConnectivityManager} on stop. 244 * 245 * @param resetRepeatTime resets the time delay for repeated notification if true. 246 */ clearPendingNotification(boolean resetRepeatTime)247 public void clearPendingNotification(boolean resetRepeatTime) { 248 if (resetRepeatTime) { 249 mNotificationRepeatTime = 0; 250 } 251 252 if (mState != STATE_NO_NOTIFICATION) { 253 getNotificationManager().cancel(mSystemMessageNotificationId); 254 255 if (mRecommendedNetwork != null) { 256 Log.d(mTag, "Notification with state=" 257 + mState 258 + " was cleared for recommended network: " 259 + "\"" + mRecommendedNetwork.SSID + "\""); 260 } 261 mState = STATE_NO_NOTIFICATION; 262 mRecommendedNetwork = null; 263 } 264 } 265 isControllerEnabled()266 private boolean isControllerEnabled() { 267 return mSettingEnabled && !UserManager.get(mContext) 268 .hasUserRestriction(UserManager.DISALLOW_CONFIG_WIFI, UserHandle.CURRENT); 269 } 270 271 /** 272 * If there are available networks, attempt to post a network notification. 273 * 274 * @param availableNetworks Available networks to choose from and possibly show notification 275 */ handleScanResults(@onNull List<ScanDetail> availableNetworks)276 public void handleScanResults(@NonNull List<ScanDetail> availableNetworks) { 277 if (!isControllerEnabled()) { 278 clearPendingNotification(true /* resetRepeatTime */); 279 return; 280 } 281 if (availableNetworks.isEmpty() && mState == STATE_SHOWING_RECOMMENDATION_NOTIFICATION) { 282 clearPendingNotification(false /* resetRepeatTime */); 283 return; 284 } 285 286 // Not enough time has passed to show a recommendation notification again 287 if (mState == STATE_NO_NOTIFICATION 288 && mClock.getWallClockMillis() < mNotificationRepeatTime) { 289 return; 290 } 291 292 // Do nothing when the screen is off and no notification is showing. 293 if (mState == STATE_NO_NOTIFICATION && !mScreenOn) { 294 return; 295 } 296 297 // Only show a new or update an existing recommendation notification. 298 if (mState == STATE_NO_NOTIFICATION 299 || mState == STATE_SHOWING_RECOMMENDATION_NOTIFICATION) { 300 ScanResult recommendation = 301 recommendNetwork(availableNetworks); 302 303 if (recommendation != null) { 304 postInitialNotification(recommendation); 305 } else { 306 clearPendingNotification(false /* resetRepeatTime */); 307 } 308 } 309 } 310 311 /** 312 * Recommends a network to connect to from a list of available networks, while ignoring the 313 * SSIDs in the blacklist. 314 * 315 * @param networks List of networks to select from 316 */ recommendNetwork(@onNull List<ScanDetail> networks)317 public ScanResult recommendNetwork(@NonNull List<ScanDetail> networks) { 318 ScanResult result = null; 319 int highestRssi = Integer.MIN_VALUE; 320 for (ScanDetail scanDetail : networks) { 321 ScanResult scanResult = scanDetail.getScanResult(); 322 323 if (scanResult.level > highestRssi) { 324 result = scanResult; 325 highestRssi = scanResult.level; 326 } 327 } 328 329 if (result != null && mBlacklistedSsids.contains(result.SSID)) { 330 result = null; 331 } 332 return result; 333 } 334 335 /** Handles screen state changes. */ handleScreenStateChanged(boolean screenOn)336 public void handleScreenStateChanged(boolean screenOn) { 337 mScreenOn = screenOn; 338 } 339 340 /** 341 * Called by {@link WifiConnectivityManager} when Wi-Fi is connected. If the notification 342 * was in the connecting state, update the notification to show that it has connected to the 343 * recommended network. 344 * 345 * @param ssid The connected network's ssid 346 */ handleWifiConnected(String ssid)347 public void handleWifiConnected(String ssid) { 348 removeNetworkFromBlacklist(ssid); 349 if (mState != STATE_CONNECTING_IN_NOTIFICATION) { 350 clearPendingNotification(true /* resetRepeatTime */); 351 return; 352 } 353 354 postNotification(mNotificationBuilder.createNetworkConnectedNotification(mTag, 355 mRecommendedNetwork)); 356 357 Log.d(mTag, "User connected to recommended network: " 358 + "\"" + mRecommendedNetwork.SSID + "\""); 359 mWifiMetrics.incrementConnectToNetworkNotification(mTag, 360 ConnectToNetworkNotificationAndActionCount.NOTIFICATION_CONNECTED_TO_NETWORK); 361 mState = STATE_CONNECTED_NOTIFICATION; 362 mHandler.postDelayed( 363 () -> { 364 if (mState == STATE_CONNECTED_NOTIFICATION) { 365 clearPendingNotification(true /* resetRepeatTime */); 366 } 367 }, 368 TIME_TO_SHOW_CONNECTED_MILLIS); 369 } 370 371 /** 372 * Handles when a Wi-Fi connection attempt failed. 373 */ handleConnectionFailure()374 public void handleConnectionFailure() { 375 if (mState != STATE_CONNECTING_IN_NOTIFICATION) { 376 return; 377 } 378 postNotification(mNotificationBuilder.createNetworkFailedNotification(mTag)); 379 380 Log.d(mTag, "User failed to connect to recommended network: " 381 + "\"" + mRecommendedNetwork.SSID + "\""); 382 mWifiMetrics.incrementConnectToNetworkNotification(mTag, 383 ConnectToNetworkNotificationAndActionCount.NOTIFICATION_FAILED_TO_CONNECT); 384 mState = STATE_CONNECT_FAILED_NOTIFICATION; 385 mHandler.postDelayed( 386 () -> { 387 if (mState == STATE_CONNECT_FAILED_NOTIFICATION) { 388 clearPendingNotification(false /* resetRepeatTime */); 389 } 390 }, 391 TIME_TO_SHOW_FAILED_MILLIS); 392 } 393 getNotificationManager()394 private NotificationManager getNotificationManager() { 395 return (NotificationManager) mContext.getSystemService(Context.NOTIFICATION_SERVICE); 396 } 397 postInitialNotification(ScanResult recommendedNetwork)398 private void postInitialNotification(ScanResult recommendedNetwork) { 399 if (mRecommendedNetwork != null 400 && TextUtils.equals(mRecommendedNetwork.SSID, recommendedNetwork.SSID)) { 401 return; 402 } 403 404 postNotification(mNotificationBuilder.createConnectToAvailableNetworkNotification(mTag, 405 recommendedNetwork)); 406 407 if (mState == STATE_NO_NOTIFICATION) { 408 mWifiMetrics.incrementConnectToNetworkNotification(mTag, 409 ConnectToNetworkNotificationAndActionCount.NOTIFICATION_RECOMMEND_NETWORK); 410 } else { 411 mWifiMetrics.incrementNumNetworkRecommendationUpdates(mTag); 412 } 413 mState = STATE_SHOWING_RECOMMENDATION_NOTIFICATION; 414 mRecommendedNetwork = recommendedNetwork; 415 mNotificationRepeatTime = mClock.getWallClockMillis() + mNotificationRepeatDelay; 416 } 417 postNotification(Notification notification)418 private void postNotification(Notification notification) { 419 getNotificationManager().notify(mSystemMessageNotificationId, notification); 420 } 421 handleConnectToNetworkAction()422 private void handleConnectToNetworkAction() { 423 mWifiMetrics.incrementConnectToNetworkNotificationAction(mTag, mState, 424 ConnectToNetworkNotificationAndActionCount.ACTION_CONNECT_TO_NETWORK); 425 if (mState != STATE_SHOWING_RECOMMENDATION_NOTIFICATION) { 426 return; 427 } 428 postNotification(mNotificationBuilder.createNetworkConnectingNotification(mTag, 429 mRecommendedNetwork)); 430 mWifiMetrics.incrementConnectToNetworkNotification(mTag, 431 ConnectToNetworkNotificationAndActionCount.NOTIFICATION_CONNECTING_TO_NETWORK); 432 433 Log.d(mTag, 434 "User initiated connection to recommended network: " 435 + "\"" + mRecommendedNetwork.SSID + "\""); 436 WifiConfiguration network = createRecommendedNetworkConfig(mRecommendedNetwork); 437 438 NetworkUpdateResult result = mConfigManager.addOrUpdateNetwork(network, Process.WIFI_UID); 439 if (result.isSuccess()) { 440 mWifiMetrics.setNominatorForNetwork(result.netId, mNominatorId); 441 442 Message msg = Message.obtain(); 443 msg.what = WifiManager.CONNECT_NETWORK; 444 msg.arg1 = result.netId; 445 msg.obj = null; 446 msg.replyTo = mSrcMessenger; 447 mClientModeImpl.sendMessage(msg); 448 addNetworkToBlacklist(mRecommendedNetwork.SSID); 449 } 450 451 mState = STATE_CONNECTING_IN_NOTIFICATION; 452 mHandler.postDelayed( 453 () -> { 454 if (mState == STATE_CONNECTING_IN_NOTIFICATION) { 455 handleConnectionFailure(); 456 } 457 }, 458 TIME_TO_SHOW_CONNECTING_MILLIS); 459 } 460 addNetworkToBlacklist(String ssid)461 private void addNetworkToBlacklist(String ssid) { 462 mBlacklistedSsids.add(ssid); 463 mWifiMetrics.setNetworkRecommenderBlacklistSize(mTag, mBlacklistedSsids.size()); 464 mConfigManager.saveToStore(false /* forceWrite */); 465 Log.d(mTag, "Network is added to the network notification blacklist: " 466 + "\"" + ssid + "\""); 467 } 468 removeNetworkFromBlacklist(String ssid)469 private void removeNetworkFromBlacklist(String ssid) { 470 if (ssid == null) { 471 return; 472 } 473 if (!mBlacklistedSsids.remove(ssid)) { 474 return; 475 } 476 mWifiMetrics.setNetworkRecommenderBlacklistSize(mTag, mBlacklistedSsids.size()); 477 mConfigManager.saveToStore(false /* forceWrite */); 478 Log.d(mTag, "Network is removed from the network notification blacklist: " 479 + "\"" + ssid + "\""); 480 } 481 createRecommendedNetworkConfig(ScanResult recommendedNetwork)482 WifiConfiguration createRecommendedNetworkConfig(ScanResult recommendedNetwork) { 483 return ScanResultUtil.createNetworkFromScanResult(recommendedNetwork); 484 } 485 handleSeeAllNetworksAction()486 private void handleSeeAllNetworksAction() { 487 mWifiMetrics.incrementConnectToNetworkNotificationAction(mTag, mState, 488 ConnectToNetworkNotificationAndActionCount.ACTION_PICK_WIFI_NETWORK); 489 startWifiSettings(); 490 } 491 startWifiSettings()492 private void startWifiSettings() { 493 // Close notification drawer before opening the picker. 494 mContext.sendBroadcast(new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)); 495 mContext.startActivity( 496 new Intent(Settings.ACTION_WIFI_SETTINGS) 497 .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)); 498 clearPendingNotification(false /* resetRepeatTime */); 499 } 500 handleConnectionAttemptFailedToSend()501 private void handleConnectionAttemptFailedToSend() { 502 handleConnectionFailure(); 503 mWifiMetrics.incrementNumNetworkConnectMessageFailedToSend(mTag); 504 } 505 handlePickWifiNetworkAfterConnectFailure()506 private void handlePickWifiNetworkAfterConnectFailure() { 507 mWifiMetrics.incrementConnectToNetworkNotificationAction(mTag, mState, 508 ConnectToNetworkNotificationAndActionCount 509 .ACTION_PICK_WIFI_NETWORK_AFTER_CONNECT_FAILURE); 510 startWifiSettings(); 511 } 512 handleUserDismissedAction()513 private void handleUserDismissedAction() { 514 Log.d(mTag, "User dismissed notification with state=" + mState); 515 mWifiMetrics.incrementConnectToNetworkNotificationAction(mTag, mState, 516 ConnectToNetworkNotificationAndActionCount.ACTION_USER_DISMISSED_NOTIFICATION); 517 if (mState == STATE_SHOWING_RECOMMENDATION_NOTIFICATION) { 518 // blacklist dismissed network 519 addNetworkToBlacklist(mRecommendedNetwork.SSID); 520 } 521 resetStateAndDelayNotification(); 522 } 523 resetStateAndDelayNotification()524 private void resetStateAndDelayNotification() { 525 mState = STATE_NO_NOTIFICATION; 526 mNotificationRepeatTime = System.currentTimeMillis() + mNotificationRepeatDelay; 527 mRecommendedNetwork = null; 528 } 529 530 /** Dump this network notifier's state. */ dump(FileDescriptor fd, PrintWriter pw, String[] args)531 public void dump(FileDescriptor fd, PrintWriter pw, String[] args) { 532 pw.println(mTag + ": "); 533 pw.println("mSettingEnabled " + mSettingEnabled); 534 pw.println("currentTime: " + mClock.getWallClockMillis()); 535 pw.println("mNotificationRepeatTime: " + mNotificationRepeatTime); 536 pw.println("mState: " + mState); 537 pw.println("mBlacklistedSsids: " + mBlacklistedSsids.toString()); 538 } 539 540 private class AvailableNetworkNotifierStoreData implements SsidSetStoreData.DataSource { 541 @Override getSsids()542 public Set<String> getSsids() { 543 return new ArraySet<>(mBlacklistedSsids); 544 } 545 546 @Override setSsids(Set<String> ssidList)547 public void setSsids(Set<String> ssidList) { 548 mBlacklistedSsids.addAll(ssidList); 549 mWifiMetrics.setNetworkRecommenderBlacklistSize(mTag, mBlacklistedSsids.size()); 550 } 551 } 552 553 private class NotificationEnabledSettingObserver extends ContentObserver { NotificationEnabledSettingObserver(Handler handler)554 NotificationEnabledSettingObserver(Handler handler) { 555 super(handler); 556 } 557 register()558 public void register() { 559 mFrameworkFacade.registerContentObserver(mContext, 560 Settings.Global.getUriFor(mToggleSettingsName), true, this); 561 mSettingEnabled = getValue(); 562 } 563 564 @Override onChange(boolean selfChange)565 public void onChange(boolean selfChange) { 566 super.onChange(selfChange); 567 mSettingEnabled = getValue(); 568 clearPendingNotification(true /* resetRepeatTime */); 569 } 570 getValue()571 private boolean getValue() { 572 boolean enabled = 573 mFrameworkFacade.getIntegerSetting(mContext, mToggleSettingsName, 1) == 1; 574 mWifiMetrics.setIsWifiNetworksAvailableNotificationEnabled(mTag, enabled); 575 Log.d(mTag, "Settings toggle enabled=" + enabled); 576 return enabled; 577 } 578 } 579 } 580