1 /* 2 * Copyright (C) 2013 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 android.app.Notification; 20 import android.app.NotificationManager; 21 import android.app.TaskStackBuilder; 22 import android.content.BroadcastReceiver; 23 import android.content.ContentResolver; 24 import android.content.Context; 25 import android.content.Intent; 26 import android.content.IntentFilter; 27 import android.database.ContentObserver; 28 import android.net.NetworkInfo; 29 import android.net.wifi.ScanResult; 30 import android.net.wifi.WifiManager; 31 import android.os.Handler; 32 import android.os.Message; 33 import android.os.UserHandle; 34 import android.provider.Settings; 35 36 import java.io.FileDescriptor; 37 import java.io.PrintWriter; 38 import java.util.List; 39 40 /* Takes care of handling the "open wi-fi network available" notification @hide */ 41 final class WifiNotificationController { 42 /** 43 * The icon to show in the 'available networks' notification. This will also 44 * be the ID of the Notification given to the NotificationManager. 45 */ 46 private static final int ICON_NETWORKS_AVAILABLE = 47 com.android.internal.R.drawable.stat_notify_wifi_in_range; 48 /** 49 * When a notification is shown, we wait this amount before possibly showing it again. 50 */ 51 private final long NOTIFICATION_REPEAT_DELAY_MS; 52 /** 53 * Whether the user has set the setting to show the 'available networks' notification. 54 */ 55 private boolean mNotificationEnabled; 56 /** 57 * Observes the user setting to keep {@link #mNotificationEnabled} in sync. 58 */ 59 private NotificationEnabledSettingObserver mNotificationEnabledSettingObserver; 60 /** 61 * The {@link System#currentTimeMillis()} must be at least this value for us 62 * to show the notification again. 63 */ 64 private long mNotificationRepeatTime; 65 /** 66 * The Notification object given to the NotificationManager. 67 */ 68 private Notification mNotification; 69 /** 70 * Whether the notification is being shown, as set by us. That is, if the 71 * user cancels the notification, we will not receive the callback so this 72 * will still be true. We only guarantee if this is false, then the 73 * notification is not showing. 74 */ 75 private boolean mNotificationShown; 76 /** 77 * The number of continuous scans that must occur before consider the 78 * supplicant in a scanning state. This allows supplicant to associate with 79 * remembered networks that are in the scan results. 80 */ 81 private static final int NUM_SCANS_BEFORE_ACTUALLY_SCANNING = 3; 82 /** 83 * The number of scans since the last network state change. When this 84 * exceeds {@link #NUM_SCANS_BEFORE_ACTUALLY_SCANNING}, we consider the 85 * supplicant to actually be scanning. When the network state changes to 86 * something other than scanning, we reset this to 0. 87 */ 88 private int mNumScansSinceNetworkStateChange; 89 90 private final Context mContext; 91 private final WifiStateMachine mWifiStateMachine; 92 private NetworkInfo mNetworkInfo; 93 private volatile int mWifiState; 94 95 WifiNotificationController(Context context, WifiStateMachine wsm) { 96 mContext = context; 97 mWifiStateMachine = wsm; 98 mWifiState = WifiManager.WIFI_STATE_UNKNOWN; 99 100 IntentFilter filter = new IntentFilter(); 101 filter.addAction(WifiManager.WIFI_STATE_CHANGED_ACTION); 102 filter.addAction(WifiManager.NETWORK_STATE_CHANGED_ACTION); 103 filter.addAction(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION); 104 105 mContext.registerReceiver( 106 new BroadcastReceiver() { 107 @Override 108 public void onReceive(Context context, Intent intent) { 109 if (intent.getAction().equals(WifiManager.WIFI_STATE_CHANGED_ACTION)) { 110 mWifiState = intent.getIntExtra(WifiManager.EXTRA_WIFI_STATE, 111 WifiManager.WIFI_STATE_UNKNOWN); 112 resetNotification(); 113 } else if (intent.getAction().equals( 114 WifiManager.NETWORK_STATE_CHANGED_ACTION)) { 115 mNetworkInfo = (NetworkInfo) intent.getParcelableExtra( 116 WifiManager.EXTRA_NETWORK_INFO); 117 // reset & clear notification on a network connect & disconnect 118 switch(mNetworkInfo.getDetailedState()) { 119 case CONNECTED: 120 case DISCONNECTED: 121 case CAPTIVE_PORTAL_CHECK: 122 resetNotification(); 123 break; 124 } 125 } else if (intent.getAction().equals( 126 WifiManager.SCAN_RESULTS_AVAILABLE_ACTION)) { 127 checkAndSetNotification(mNetworkInfo, 128 mWifiStateMachine.syncGetScanResultsList()); 129 } 130 } 131 }, filter); 132 133 // Setting is in seconds 134 NOTIFICATION_REPEAT_DELAY_MS = Settings.Global.getInt(context.getContentResolver(), 135 Settings.Global.WIFI_NETWORKS_AVAILABLE_REPEAT_DELAY, 900) * 1000l; 136 mNotificationEnabledSettingObserver = new NotificationEnabledSettingObserver(new Handler()); 137 mNotificationEnabledSettingObserver.register(); 138 } 139 140 private synchronized void checkAndSetNotification(NetworkInfo networkInfo, 141 List<ScanResult> scanResults) { 142 143 // TODO: unregister broadcast so we do not have to check here 144 // If we shouldn't place a notification on available networks, then 145 // don't bother doing any of the following 146 if (!mNotificationEnabled) return; 147 if (mWifiState != WifiManager.WIFI_STATE_ENABLED) return; 148 149 NetworkInfo.State state = NetworkInfo.State.DISCONNECTED; 150 if (networkInfo != null) 151 state = networkInfo.getState(); 152 153 if ((state == NetworkInfo.State.DISCONNECTED) 154 || (state == NetworkInfo.State.UNKNOWN)) { 155 if (scanResults != null) { 156 int numOpenNetworks = 0; 157 for (int i = scanResults.size() - 1; i >= 0; i--) { 158 ScanResult scanResult = scanResults.get(i); 159 160 //A capability of [ESS] represents an open access point 161 //that is available for an STA to connect 162 if (scanResult.capabilities != null && 163 scanResult.capabilities.equals("[ESS]")) { 164 numOpenNetworks++; 165 } 166 } 167 168 if (numOpenNetworks > 0) { 169 if (++mNumScansSinceNetworkStateChange >= NUM_SCANS_BEFORE_ACTUALLY_SCANNING) { 170 /* 171 * We've scanned continuously at least 172 * NUM_SCANS_BEFORE_NOTIFICATION times. The user 173 * probably does not have a remembered network in range, 174 * since otherwise supplicant would have tried to 175 * associate and thus resetting this counter. 176 */ 177 setNotificationVisible(true, numOpenNetworks, false, 0); 178 } 179 return; 180 } 181 } 182 } 183 184 // No open networks in range, remove the notification 185 setNotificationVisible(false, 0, false, 0); 186 } 187 188 /** 189 * Clears variables related to tracking whether a notification has been 190 * shown recently and clears the current notification. 191 */ 192 private synchronized void resetNotification() { 193 mNotificationRepeatTime = 0; 194 mNumScansSinceNetworkStateChange = 0; 195 setNotificationVisible(false, 0, false, 0); 196 } 197 198 /** 199 * Display or don't display a notification that there are open Wi-Fi networks. 200 * @param visible {@code true} if notification should be visible, {@code false} otherwise 201 * @param numNetworks the number networks seen 202 * @param force {@code true} to force notification to be shown/not-shown, 203 * even if it is already shown/not-shown. 204 * @param delay time in milliseconds after which the notification should be made 205 * visible or invisible. 206 */ 207 private void setNotificationVisible(boolean visible, int numNetworks, boolean force, 208 int delay) { 209 210 // Since we use auto cancel on the notification, when the 211 // mNetworksAvailableNotificationShown is true, the notification may 212 // have actually been canceled. However, when it is false we know 213 // for sure that it is not being shown (it will not be shown any other 214 // place than here) 215 216 // If it should be hidden and it is already hidden, then noop 217 if (!visible && !mNotificationShown && !force) { 218 return; 219 } 220 221 NotificationManager notificationManager = (NotificationManager) mContext 222 .getSystemService(Context.NOTIFICATION_SERVICE); 223 224 Message message; 225 if (visible) { 226 227 // Not enough time has passed to show the notification again 228 if (System.currentTimeMillis() < mNotificationRepeatTime) { 229 return; 230 } 231 232 if (mNotification == null) { 233 // Cache the Notification object. 234 mNotification = new Notification(); 235 mNotification.when = 0; 236 mNotification.icon = ICON_NETWORKS_AVAILABLE; 237 mNotification.flags = Notification.FLAG_AUTO_CANCEL; 238 mNotification.contentIntent = TaskStackBuilder.create(mContext) 239 .addNextIntentWithParentStack( 240 new Intent(WifiManager.ACTION_PICK_WIFI_NETWORK)) 241 .getPendingIntent(0, 0, null, UserHandle.CURRENT); 242 } 243 244 CharSequence title = mContext.getResources().getQuantityText( 245 com.android.internal.R.plurals.wifi_available, numNetworks); 246 CharSequence details = mContext.getResources().getQuantityText( 247 com.android.internal.R.plurals.wifi_available_detailed, numNetworks); 248 mNotification.tickerText = title; 249 mNotification.color = mContext.getResources().getColor( 250 com.android.internal.R.color.system_notification_accent_color); 251 mNotification.setLatestEventInfo(mContext, title, details, mNotification.contentIntent); 252 253 mNotificationRepeatTime = System.currentTimeMillis() + NOTIFICATION_REPEAT_DELAY_MS; 254 255 notificationManager.notifyAsUser(null, ICON_NETWORKS_AVAILABLE, mNotification, 256 UserHandle.ALL); 257 } else { 258 notificationManager.cancelAsUser(null, ICON_NETWORKS_AVAILABLE, UserHandle.ALL); 259 } 260 261 mNotificationShown = visible; 262 } 263 264 void dump(FileDescriptor fd, PrintWriter pw, String[] args) { 265 pw.println("mNotificationEnabled " + mNotificationEnabled); 266 pw.println("mNotificationRepeatTime " + mNotificationRepeatTime); 267 pw.println("mNotificationShown " + mNotificationShown); 268 pw.println("mNumScansSinceNetworkStateChange " + mNumScansSinceNetworkStateChange); 269 } 270 271 private class NotificationEnabledSettingObserver extends ContentObserver { 272 public NotificationEnabledSettingObserver(Handler handler) { 273 super(handler); 274 } 275 276 public void register() { 277 ContentResolver cr = mContext.getContentResolver(); 278 cr.registerContentObserver(Settings.Global.getUriFor( 279 Settings.Global.WIFI_NETWORKS_AVAILABLE_NOTIFICATION_ON), true, this); 280 synchronized (WifiNotificationController.this) { 281 mNotificationEnabled = getValue(); 282 } 283 } 284 285 @Override 286 public void onChange(boolean selfChange) { 287 super.onChange(selfChange); 288 289 synchronized (WifiNotificationController.this) { 290 mNotificationEnabled = getValue(); 291 resetNotification(); 292 } 293 } 294 295 private boolean getValue() { 296 return Settings.Global.getInt(mContext.getContentResolver(), 297 Settings.Global.WIFI_NETWORKS_AVAILABLE_NOTIFICATION_ON, 1) == 1; 298 } 299 } 300 301 } 302