blob: a9ff6c589df984b1de056c604ff6125c8be14f4d [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007 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
Joe Onorato7a0f36b2010-06-07 10:24:36 -070017package com.android.server;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080018
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080019import android.app.StatusBarManager;
20import android.content.BroadcastReceiver;
21import android.content.Context;
22import android.content.Intent;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080023import android.content.pm.PackageManager;
24import android.content.res.Resources;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080025import android.os.Binder;
Joe Onoratof3f0e052010-05-14 18:49:29 -070026import android.os.Handler;
Svetoslav Ganov6179ea32011-06-28 01:12:41 -070027import android.os.IBinder;
28import android.os.RemoteException;
Joe Onorato8a9b2202010-02-26 18:56:32 -080029import android.util.Slog;
Joe Onorato664644d2011-01-23 17:53:23 -080030import android.view.View;
Joe Onorato0cbda992010-05-02 16:28:15 -070031
32import com.android.internal.statusbar.IStatusBar;
33import com.android.internal.statusbar.IStatusBarService;
34import com.android.internal.statusbar.StatusBarIcon;
35import com.android.internal.statusbar.StatusBarIconList;
Joe Onorato18e69df2010-05-17 22:26:12 -070036import com.android.internal.statusbar.StatusBarNotification;
Dianne Hackborna924dc0d2011-02-17 14:22:17 -080037import com.android.server.wm.WindowManagerService;
The Android Open Source Project10592532009-03-18 17:39:46 -070038
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080039import java.io.FileDescriptor;
40import java.io.PrintWriter;
41import java.util.ArrayList;
42import java.util.HashMap;
Joe Onorato75199e32010-05-29 17:22:51 -040043import java.util.List;
44import java.util.Map;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080045
46
47/**
Joe Onoratof3f0e052010-05-14 18:49:29 -070048 * A note on locking: We rely on the fact that calls onto mBar are oneway or
49 * if they are local, that they just enqueue messages to not deadlock.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080050 */
Joe Onorato089de882010-04-12 08:18:45 -070051public class StatusBarManagerService extends IStatusBarService.Stub
Jeff Brown2992ea72011-01-28 22:04:14 -080052 implements WindowManagerService.OnHardKeyboardStatusChangeListener
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080053{
Joe Onorato4762c2d2010-05-17 15:42:59 -070054 static final String TAG = "StatusBarManagerService";
Joe Onorato431bb222010-10-18 19:13:23 -040055 static final boolean SPEW = false;
Joe Onoratodf7dbb62009-11-17 10:43:37 -080056
Joe Onoratof3f0e052010-05-14 18:49:29 -070057 final Context mContext;
Jeff Brown2992ea72011-01-28 22:04:14 -080058 final WindowManagerService mWindowManager;
Joe Onoratof3f0e052010-05-14 18:49:29 -070059 Handler mHandler = new Handler();
60 NotificationCallbacks mNotificationCallbacks;
Joe Onorato4762c2d2010-05-17 15:42:59 -070061 volatile IStatusBar mBar;
Joe Onoratof3f0e052010-05-14 18:49:29 -070062 StatusBarIconList mIcons = new StatusBarIconList();
Joe Onorato75199e32010-05-29 17:22:51 -040063 HashMap<IBinder,StatusBarNotification> mNotifications
64 = new HashMap<IBinder,StatusBarNotification>();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080065
Joe Onoratof3f0e052010-05-14 18:49:29 -070066 // for disabling the status bar
67 ArrayList<DisableRecord> mDisableRecords = new ArrayList<DisableRecord>();
Joe Onorato7bb8eeb2011-01-27 16:00:58 -080068 IBinder mSysUiVisToken = new Binder();
Joe Onoratof3f0e052010-05-14 18:49:29 -070069 int mDisabled = 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080070
Joe Onorato93056472010-09-10 10:30:46 -040071 Object mLock = new Object();
Daniel Sandler60ee2562011-07-22 12:34:33 -040072 // encompasses lights-out mode and other flags defined on View
73 int mSystemUiVisibility = 0;
Daniel Sandlere02d8082010-10-08 15:13:22 -040074 boolean mMenuVisible = false;
Joe Onorato857fd9b2011-01-27 15:08:35 -080075 int mImeWindowVis = 0;
76 int mImeBackDisposition;
77 IBinder mImeToken = null;
satok06487a52010-10-29 11:37:18 +090078
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080079 private class DisableRecord implements IBinder.DeathRecipient {
80 String pkg;
81 int what;
82 IBinder token;
83
84 public void binderDied() {
Joe Onorato8a9b2202010-02-26 18:56:32 -080085 Slog.i(TAG, "binder died for pkg=" + pkg);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080086 disable(0, token, pkg);
Suchi Amalapurapufff2fda2009-06-30 21:36:16 -070087 token.unlinkToDeath(this, 0);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080088 }
89 }
90
91 public interface NotificationCallbacks {
92 void onSetDisabled(int status);
93 void onClearAll();
Fred Quintana6ecaff12009-09-25 14:23:13 -070094 void onNotificationClick(String pkg, String tag, int id);
Daniel Sandler0f0b11c2010-08-04 15:54:58 -040095 void onNotificationClear(String pkg, String tag, int id);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080096 void onPanelRevealed();
Dianne Hackborn9d39d0c2010-06-24 15:57:42 -070097 void onNotificationError(String pkg, String tag, int id,
98 int uid, int initialPid, String message);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080099 }
100
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800101 /**
102 * Construct the service, add the status bar view to the window manager
103 */
Jeff Brown2992ea72011-01-28 22:04:14 -0800104 public StatusBarManagerService(Context context, WindowManagerService windowManager) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800105 mContext = context;
Jeff Brown2992ea72011-01-28 22:04:14 -0800106 mWindowManager = windowManager;
107 mWindowManager.setOnHardKeyboardStatusChangeListener(this);
Joe Onorato0cbda992010-05-02 16:28:15 -0700108
109 final Resources res = context.getResources();
Joe Onorato75144ea2010-06-07 12:36:25 -0700110 mIcons.defineSlots(res.getStringArray(com.android.internal.R.array.config_statusBarIcons));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800111 }
112
113 public void setNotificationCallbacks(NotificationCallbacks listener) {
114 mNotificationCallbacks = listener;
115 }
116
117 // ================================================================================
Joe Onorato25f95f92010-04-08 18:37:10 -0500118 // From IStatusBarService
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800119 // ================================================================================
Joe Onoratof3f0e052010-05-14 18:49:29 -0700120 public void expand() {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800121 enforceExpandStatusBar();
Joe Onorato4762c2d2010-05-17 15:42:59 -0700122
123 if (mBar != null) {
124 try {
125 mBar.animateExpand();
126 } catch (RemoteException ex) {
127 }
128 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800129 }
130
Joe Onoratof3f0e052010-05-14 18:49:29 -0700131 public void collapse() {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800132 enforceExpandStatusBar();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800133
Joe Onorato4762c2d2010-05-17 15:42:59 -0700134 if (mBar != null) {
135 try {
136 mBar.animateCollapse();
137 } catch (RemoteException ex) {
138 }
139 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800140 }
141
142 public void disable(int what, IBinder token, String pkg) {
143 enforceStatusBar();
Joe Onoratof3f0e052010-05-14 18:49:29 -0700144
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800145 synchronized (mLock) {
146 disableLocked(what, token, pkg);
147 }
148 }
149
150 private void disableLocked(int what, IBinder token, String pkg) {
Joe Onoratof3f0e052010-05-14 18:49:29 -0700151 // It's important that the the callback and the call to mBar get done
152 // in the same order when multiple threads are calling this function
153 // so they are paired correctly. The messages on the handler will be
154 // handled in the order they were enqueued, but will be outside the lock.
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800155 manageDisableListLocked(what, token, pkg);
156 final int net = gatherDisableActionsLocked();
157 if (net != mDisabled) {
158 mDisabled = net;
159 mHandler.post(new Runnable() {
160 public void run() {
161 mNotificationCallbacks.onSetDisabled(net);
Joe Onoratof3f0e052010-05-14 18:49:29 -0700162 }
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800163 });
164 if (mBar != null) {
165 try {
166 mBar.disable(net);
167 } catch (RemoteException ex) {
Joe Onoratof3f0e052010-05-14 18:49:29 -0700168 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800169 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800170 }
171 }
172
Svetoslav Ganov6179ea32011-06-28 01:12:41 -0700173 public void setIcon(String slot, String iconPackage, int iconId, int iconLevel,
174 String contentDescription) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800175 enforceStatusBar();
Joe Onorato0cbda992010-05-02 16:28:15 -0700176
177 synchronized (mIcons) {
178 int index = mIcons.getSlotIndex(slot);
179 if (index < 0) {
180 throw new SecurityException("invalid status bar icon slot: " + slot);
181 }
182
Svetoslav Ganov6179ea32011-06-28 01:12:41 -0700183 StatusBarIcon icon = new StatusBarIcon(iconPackage, iconId, iconLevel, 0,
184 contentDescription);
Joe Onorato66d7d012010-05-14 10:05:10 -0700185 //Slog.d(TAG, "setIcon slot=" + slot + " index=" + index + " icon=" + icon);
Joe Onorato0cbda992010-05-02 16:28:15 -0700186 mIcons.setIcon(index, icon);
187
Joe Onorato0cbda992010-05-02 16:28:15 -0700188 if (mBar != null) {
189 try {
190 mBar.setIcon(index, icon);
191 } catch (RemoteException ex) {
192 }
193 }
194 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800195 }
196
Joe Onorato0cbda992010-05-02 16:28:15 -0700197 public void setIconVisibility(String slot, boolean visible) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800198 enforceStatusBar();
Joe Onorato0cbda992010-05-02 16:28:15 -0700199
Joe Onorato514ad6632010-05-13 18:49:00 -0700200 synchronized (mIcons) {
201 int index = mIcons.getSlotIndex(slot);
202 if (index < 0) {
203 throw new SecurityException("invalid status bar icon slot: " + slot);
204 }
205
206 StatusBarIcon icon = mIcons.getIcon(index);
207 if (icon == null) {
208 return;
209 }
210
211 if (icon.visible != visible) {
212 icon.visible = visible;
213
Joe Onorato514ad6632010-05-13 18:49:00 -0700214 if (mBar != null) {
215 try {
216 mBar.setIcon(index, icon);
217 } catch (RemoteException ex) {
218 }
219 }
220 }
221 }
Joe Onorato0cbda992010-05-02 16:28:15 -0700222 }
223
224 public void removeIcon(String slot) {
225 enforceStatusBar();
226
227 synchronized (mIcons) {
228 int index = mIcons.getSlotIndex(slot);
229 if (index < 0) {
230 throw new SecurityException("invalid status bar icon slot: " + slot);
231 }
232
233 mIcons.removeIcon(index);
234
Joe Onorato0cbda992010-05-02 16:28:15 -0700235 if (mBar != null) {
236 try {
237 mBar.removeIcon(index);
238 } catch (RemoteException ex) {
239 }
240 }
241 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800242 }
243
Daniel Sandlere02d8082010-10-08 15:13:22 -0400244 /**
245 * Hide or show the on-screen Menu key. Only call this from the window manager, typically in
246 * response to a window with FLAG_NEEDS_MENU_KEY set.
247 */
Dianne Hackborn7d049322011-06-14 15:00:32 -0700248 public void topAppWindowChanged(final boolean menuVisible) {
Daniel Sandlere02d8082010-10-08 15:13:22 -0400249 enforceStatusBar();
250
Dianne Hackborn7d049322011-06-14 15:00:32 -0700251 if (SPEW) Slog.d(TAG, (menuVisible?"showing":"hiding") + " MENU key");
Daniel Sandlere02d8082010-10-08 15:13:22 -0400252
253 synchronized(mLock) {
Dianne Hackborn7d049322011-06-14 15:00:32 -0700254 mMenuVisible = menuVisible;
255 mHandler.post(new Runnable() {
256 public void run() {
257 if (mBar != null) {
258 try {
259 mBar.topAppWindowChanged(menuVisible);
260 } catch (RemoteException ex) {
Daniel Sandlere02d8082010-10-08 15:13:22 -0400261 }
262 }
Dianne Hackborn7d049322011-06-14 15:00:32 -0700263 }
264 });
Daniel Sandlere02d8082010-10-08 15:13:22 -0400265 }
266 }
267
Joe Onorato857fd9b2011-01-27 15:08:35 -0800268 public void setImeWindowStatus(final IBinder token, final int vis, final int backDisposition) {
satok06487a52010-10-29 11:37:18 +0900269 enforceStatusBar();
270
Joe Onorato857fd9b2011-01-27 15:08:35 -0800271 if (SPEW) {
272 Slog.d(TAG, "swetImeWindowStatus vis=" + vis + " backDisposition=" + backDisposition);
273 }
satok06487a52010-10-29 11:37:18 +0900274
275 synchronized(mLock) {
Joe Onorato857fd9b2011-01-27 15:08:35 -0800276 // In case of IME change, we need to call up setImeWindowStatus() regardless of
277 // mImeWindowVis because mImeWindowVis may not have been set to false when the
satok06e07442010-11-02 19:46:55 +0900278 // previous IME was destroyed.
Joe Onorato857fd9b2011-01-27 15:08:35 -0800279 mImeWindowVis = vis;
280 mImeBackDisposition = backDisposition;
281 mImeToken = token;
satok06e07442010-11-02 19:46:55 +0900282 mHandler.post(new Runnable() {
283 public void run() {
284 if (mBar != null) {
285 try {
Joe Onorato857fd9b2011-01-27 15:08:35 -0800286 mBar.setImeWindowStatus(token, vis, backDisposition);
satok06e07442010-11-02 19:46:55 +0900287 } catch (RemoteException ex) {
satok06487a52010-10-29 11:37:18 +0900288 }
289 }
satok06e07442010-11-02 19:46:55 +0900290 }
291 });
satok06487a52010-10-29 11:37:18 +0900292 }
293 }
294
Joe Onorato664644d2011-01-23 17:53:23 -0800295 public void setSystemUiVisibility(int vis) {
Joe Onorato55bf3802011-01-25 13:42:10 -0800296 // also allows calls from window manager which is in this process.
Joe Onoratof63b0f42010-09-12 17:03:19 -0400297 enforceStatusBarService();
298
Jeff Sharkey4519a022011-09-07 23:24:53 -0700299 if (SPEW) Slog.d(TAG, "setSystemUiVisibility(0x" + Integer.toHexString(vis) + ")");
Daniel Sandler60ee2562011-07-22 12:34:33 -0400300
Joe Onoratof63b0f42010-09-12 17:03:19 -0400301 synchronized (mLock) {
Daniel Sandler60ee2562011-07-22 12:34:33 -0400302 updateUiVisibilityLocked(vis);
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800303 disableLocked(vis & StatusBarManager.DISABLE_MASK, mSysUiVisToken,
304 "WindowManager.LayoutParams");
Joe Onoratof63b0f42010-09-12 17:03:19 -0400305 }
306 }
307
Daniel Sandler60ee2562011-07-22 12:34:33 -0400308 private void updateUiVisibilityLocked(final int vis) {
309 if (mSystemUiVisibility != vis) {
310 mSystemUiVisibility = vis;
Joe Onoratof63b0f42010-09-12 17:03:19 -0400311 mHandler.post(new Runnable() {
312 public void run() {
313 if (mBar != null) {
314 try {
Daniel Sandler60ee2562011-07-22 12:34:33 -0400315 mBar.setSystemUiVisibility(vis);
Joe Onoratof63b0f42010-09-12 17:03:19 -0400316 } catch (RemoteException ex) {
Joe Onorato93056472010-09-10 10:30:46 -0400317 }
318 }
Joe Onoratof63b0f42010-09-12 17:03:19 -0400319 }
320 });
Joe Onorato93056472010-09-10 10:30:46 -0400321 }
322 }
323
Jeff Brown2992ea72011-01-28 22:04:14 -0800324 public void setHardKeyboardEnabled(final boolean enabled) {
325 mHandler.post(new Runnable() {
326 public void run() {
327 mWindowManager.setHardKeyboardEnabled(enabled);
328 }
329 });
330 }
331
332 @Override
333 public void onHardKeyboardStatusChange(final boolean available, final boolean enabled) {
334 mHandler.post(new Runnable() {
335 public void run() {
336 if (mBar != null) {
337 try {
338 mBar.setHardKeyboardStatus(available, enabled);
339 } catch (RemoteException ex) {
340 }
341 }
342 }
343 });
344 }
345
Michael Jurka3b1fc472011-06-13 10:54:40 -0700346 @Override
347 public void toggleRecentApps() {
348 if (mBar != null) {
349 try {
350 mBar.toggleRecentApps();
351 } catch (RemoteException ex) {}
352 }
353 }
354
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800355 private void enforceStatusBar() {
Joe Onorato0cbda992010-05-02 16:28:15 -0700356 mContext.enforceCallingOrSelfPermission(android.Manifest.permission.STATUS_BAR,
Joe Onorato089de882010-04-12 08:18:45 -0700357 "StatusBarManagerService");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800358 }
359
360 private void enforceExpandStatusBar() {
Joe Onorato0cbda992010-05-02 16:28:15 -0700361 mContext.enforceCallingOrSelfPermission(android.Manifest.permission.EXPAND_STATUS_BAR,
Joe Onorato089de882010-04-12 08:18:45 -0700362 "StatusBarManagerService");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800363 }
364
Joe Onorato8bc6c512010-06-04 16:21:12 -0400365 private void enforceStatusBarService() {
366 mContext.enforceCallingOrSelfPermission(android.Manifest.permission.STATUS_BAR_SERVICE,
367 "StatusBarManagerService");
368 }
369
Joe Onorato4762c2d2010-05-17 15:42:59 -0700370 // ================================================================================
371 // Callbacks from the status bar service.
372 // ================================================================================
Joe Onorato75199e32010-05-29 17:22:51 -0400373 public void registerStatusBar(IStatusBar bar, StatusBarIconList iconList,
Joe Onorato93056472010-09-10 10:30:46 -0400374 List<IBinder> notificationKeys, List<StatusBarNotification> notifications,
satokcd7cd292010-11-20 15:46:23 +0900375 int switches[], List<IBinder> binders) {
Joe Onorato8bc6c512010-06-04 16:21:12 -0400376 enforceStatusBarService();
377
Joe Onorato0cbda992010-05-02 16:28:15 -0700378 Slog.i(TAG, "registerStatusBar bar=" + bar);
379 mBar = bar;
Joe Onorato75199e32010-05-29 17:22:51 -0400380 synchronized (mIcons) {
381 iconList.copyFrom(mIcons);
382 }
383 synchronized (mNotifications) {
384 for (Map.Entry<IBinder,StatusBarNotification> e: mNotifications.entrySet()) {
385 notificationKeys.add(e.getKey());
386 notifications.add(e.getValue());
387 }
388 }
Joe Onorato93056472010-09-10 10:30:46 -0400389 synchronized (mLock) {
Joe Onoratoe4c7b3f2010-10-30 12:15:03 -0700390 switches[0] = gatherDisableActionsLocked();
Daniel Sandler60ee2562011-07-22 12:34:33 -0400391 switches[1] = mSystemUiVisibility;
Joe Onoratoe4c7b3f2010-10-30 12:15:03 -0700392 switches[2] = mMenuVisible ? 1 : 0;
Joe Onorato857fd9b2011-01-27 15:08:35 -0800393 switches[3] = mImeWindowVis;
394 switches[4] = mImeBackDisposition;
395 binders.add(mImeToken);
Joe Onorato93056472010-09-10 10:30:46 -0400396 }
Jeff Brown2992ea72011-01-28 22:04:14 -0800397 switches[5] = mWindowManager.isHardKeyboardAvailable() ? 1 : 0;
398 switches[6] = mWindowManager.isHardKeyboardEnabled() ? 1 : 0;
Joe Onorato2314aab2010-04-08 16:41:23 -0500399 }
Joe Onoratoaaba60b2010-05-23 15:18:41 -0400400
Joe Onorato4762c2d2010-05-17 15:42:59 -0700401 /**
Joe Onoratof1f25912010-06-07 11:52:41 -0700402 * The status bar service should call this each time the user brings the panel from
403 * invisible to visible in order to clear the notification light.
Joe Onorato4762c2d2010-05-17 15:42:59 -0700404 */
Joe Onoratof1f25912010-06-07 11:52:41 -0700405 public void onPanelRevealed() {
Joe Onorato8bc6c512010-06-04 16:21:12 -0400406 enforceStatusBarService();
407
Joe Onoratof1f25912010-06-07 11:52:41 -0700408 // tell the notification manager to turn off the lights.
409 mNotificationCallbacks.onPanelRevealed();
Joe Onorato4762c2d2010-05-17 15:42:59 -0700410 }
411
Joe Onoratoaaba60b2010-05-23 15:18:41 -0400412 public void onNotificationClick(String pkg, String tag, int id) {
Joe Onorato8bc6c512010-06-04 16:21:12 -0400413 enforceStatusBarService();
414
Joe Onoratoaaba60b2010-05-23 15:18:41 -0400415 mNotificationCallbacks.onNotificationClick(pkg, tag, id);
416 }
417
Dianne Hackborn9d39d0c2010-06-24 15:57:42 -0700418 public void onNotificationError(String pkg, String tag, int id,
419 int uid, int initialPid, String message) {
Joe Onorato8bc6c512010-06-04 16:21:12 -0400420 enforceStatusBarService();
421
Joe Onorato005847b2010-06-04 16:08:02 -0400422 // WARNING: this will call back into us to do the remove. Don't hold any locks.
Dianne Hackborn9d39d0c2010-06-24 15:57:42 -0700423 mNotificationCallbacks.onNotificationError(pkg, tag, id, uid, initialPid, message);
Joe Onorato005847b2010-06-04 16:08:02 -0400424 }
425
Daniel Sandler0f0b11c2010-08-04 15:54:58 -0400426 public void onNotificationClear(String pkg, String tag, int id) {
427 enforceStatusBarService();
428
429 mNotificationCallbacks.onNotificationClear(pkg, tag, id);
430 }
431
Joe Onoratoaaba60b2010-05-23 15:18:41 -0400432 public void onClearAllNotifications() {
Joe Onorato8bc6c512010-06-04 16:21:12 -0400433 enforceStatusBarService();
434
Joe Onoratoaaba60b2010-05-23 15:18:41 -0400435 mNotificationCallbacks.onClearAll();
436 }
437
Joe Onorato18e69df2010-05-17 22:26:12 -0700438 // ================================================================================
439 // Callbacks for NotificationManagerService.
440 // ================================================================================
441 public IBinder addNotification(StatusBarNotification notification) {
442 synchronized (mNotifications) {
Joe Onoratoa0c56fe2010-05-20 10:21:52 -0700443 IBinder key = new Binder();
Joe Onorato75199e32010-05-29 17:22:51 -0400444 mNotifications.put(key, notification);
Joe Onoratoe345fff2010-05-23 15:18:27 -0400445 if (mBar != null) {
446 try {
447 mBar.addNotification(key, notification);
448 } catch (RemoteException ex) {
449 }
450 }
Joe Onorato18e69df2010-05-17 22:26:12 -0700451 return key;
452 }
Joe Onorato0cbda992010-05-02 16:28:15 -0700453 }
454
Joe Onorato18e69df2010-05-17 22:26:12 -0700455 public void updateNotification(IBinder key, StatusBarNotification notification) {
456 synchronized (mNotifications) {
Joe Onorato75199e32010-05-29 17:22:51 -0400457 if (!mNotifications.containsKey(key)) {
458 throw new IllegalArgumentException("updateNotification key not found: " + key);
459 }
460 mNotifications.put(key, notification);
Joe Onoratoe345fff2010-05-23 15:18:27 -0400461 if (mBar != null) {
462 try {
463 mBar.updateNotification(key, notification);
464 } catch (RemoteException ex) {
465 }
466 }
Joe Onorato18e69df2010-05-17 22:26:12 -0700467 }
Joe Onorato0cbda992010-05-02 16:28:15 -0700468 }
469
470 public void removeNotification(IBinder key) {
Joe Onorato18e69df2010-05-17 22:26:12 -0700471 synchronized (mNotifications) {
Joe Onorato75199e32010-05-29 17:22:51 -0400472 final StatusBarNotification n = mNotifications.remove(key);
473 if (n == null) {
474 throw new IllegalArgumentException("removeNotification key not found: " + key);
475 }
Joe Onoratoe345fff2010-05-23 15:18:27 -0400476 if (mBar != null) {
477 try {
478 mBar.removeNotification(key);
479 } catch (RemoteException ex) {
480 }
481 }
Joe Onorato18e69df2010-05-17 22:26:12 -0700482 }
Joe Onorato0cbda992010-05-02 16:28:15 -0700483 }
Joe Onorato2314aab2010-04-08 16:41:23 -0500484
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800485 // ================================================================================
486 // Can be called from any thread
487 // ================================================================================
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800488
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800489 // lock on mDisableRecords
490 void manageDisableListLocked(int what, IBinder token, String pkg) {
491 if (SPEW) {
Joe Onoratof3f0e052010-05-14 18:49:29 -0700492 Slog.d(TAG, "manageDisableList what=0x" + Integer.toHexString(what) + " pkg=" + pkg);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800493 }
494 // update the list
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800495 final int N = mDisableRecords.size();
496 DisableRecord tok = null;
497 int i;
498 for (i=0; i<N; i++) {
499 DisableRecord t = mDisableRecords.get(i);
500 if (t.token == token) {
501 tok = t;
502 break;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800503 }
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800504 }
505 if (what == 0 || !token.isBinderAlive()) {
506 if (tok != null) {
507 mDisableRecords.remove(i);
508 tok.token.unlinkToDeath(tok, 0);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800509 }
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800510 } else {
511 if (tok == null) {
512 tok = new DisableRecord();
513 try {
514 token.linkToDeath(tok, 0);
515 }
516 catch (RemoteException ex) {
517 return; // give up
518 }
519 mDisableRecords.add(tok);
520 }
521 tok.what = what;
522 tok.token = token;
523 tok.pkg = pkg;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800524 }
525 }
526
527 // lock on mDisableRecords
528 int gatherDisableActionsLocked() {
529 final int N = mDisableRecords.size();
530 // gather the new net flags
531 int net = 0;
532 for (int i=0; i<N; i++) {
533 net |= mDisableRecords.get(i).what;
534 }
535 return net;
536 }
537
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800538 // ================================================================================
539 // Always called from UI thread
540 // ================================================================================
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800541
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800542 protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
543 if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.DUMP)
544 != PackageManager.PERMISSION_GRANTED) {
545 pw.println("Permission Denial: can't dump StatusBar from from pid="
546 + Binder.getCallingPid()
547 + ", uid=" + Binder.getCallingUid());
548 return;
549 }
Joe Onorato0cbda992010-05-02 16:28:15 -0700550
Joe Onorato0cbda992010-05-02 16:28:15 -0700551 synchronized (mIcons) {
552 mIcons.dump(pw);
553 }
Joe Onorato18e69df2010-05-17 22:26:12 -0700554
555 synchronized (mNotifications) {
Joe Onorato75199e32010-05-29 17:22:51 -0400556 int i=0;
557 pw.println("Notification list:");
558 for (Map.Entry<IBinder,StatusBarNotification> e: mNotifications.entrySet()) {
559 pw.printf(" %2d: %s\n", i, e.getValue().toString());
560 i++;
561 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800562 }
Joe Onorato18e69df2010-05-17 22:26:12 -0700563
Joe Onorato7bb8eeb2011-01-27 16:00:58 -0800564 synchronized (mLock) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800565 final int N = mDisableRecords.size();
566 pw.println(" mDisableRecords.size=" + N
567 + " mDisabled=0x" + Integer.toHexString(mDisabled));
568 for (int i=0; i<N; i++) {
569 DisableRecord tok = mDisableRecords.get(i);
570 pw.println(" [" + i + "] what=0x" + Integer.toHexString(tok.what)
571 + " pkg=" + tok.pkg + " token=" + tok.token);
572 }
573 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800574 }
575
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800576 private BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() {
577 public void onReceive(Context context, Intent intent) {
578 String action = intent.getAction();
Joe Onoratof9e0e6b2009-09-08 16:24:36 -0400579 if (Intent.ACTION_CLOSE_SYSTEM_DIALOGS.equals(action)
580 || Intent.ACTION_SCREEN_OFF.equals(action)) {
Joe Onoratof3f0e052010-05-14 18:49:29 -0700581 collapse();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800582 }
Joe Onorato0cbda992010-05-02 16:28:15 -0700583 /*
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800584 else if (Telephony.Intents.SPN_STRINGS_UPDATED_ACTION.equals(action)) {
585 updateNetworkName(intent.getBooleanExtra(Telephony.Intents.EXTRA_SHOW_SPN, false),
586 intent.getStringExtra(Telephony.Intents.EXTRA_SPN),
587 intent.getBooleanExtra(Telephony.Intents.EXTRA_SHOW_PLMN, false),
588 intent.getStringExtra(Telephony.Intents.EXTRA_PLMN));
589 }
590 else if (Intent.ACTION_CONFIGURATION_CHANGED.equals(action)) {
591 updateResources();
592 }
Joe Onorato0cbda992010-05-02 16:28:15 -0700593 */
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800594 }
595 };
596
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800597}