/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.support.v4.app; import android.app.Notification; import android.app.Service; import android.content.Intent; import android.os.Build; import android.os.IBinder; import android.os.RemoteException; /** * Abstract service to receive side channel notifications sent from * {@link android.support.v4.app.NotificationManagerCompat}. * *
To receive side channel notifications, extend this service and register it in your * android manifest with an intent filter for the BIND_NOTIFICATION_SIDE_CHANNEL action. * Note: you must also have an enabled * {@link android.service.notification.NotificationListenerService} within your package. * *
Example AndroidManifest.xml addition: *
* <service android:name="com.example.NotificationSideChannelService"> * <intent-filter> * <action android:name="android.support.BIND_NOTIFICATION_SIDE_CHANNEL" /> * </intent-filter> * </service>* */ public abstract class NotificationCompatSideChannelService extends Service { // In support lib, we cannot reference version codes >= 4 from android.os.Build. private static final int BUILD_VERSION_CODE_KITKAT_WATCH = 20; @Override public IBinder onBind(Intent intent) { if (intent.getAction().equals(NotificationManagerCompat.ACTION_BIND_SIDE_CHANNEL)) { // Group support is the only current reason to use side channel, // so disallow clients to bind for side channel on devices past KITKAT_WATCH for now. if (Build.VERSION.SDK_INT >= BUILD_VERSION_CODE_KITKAT_WATCH) { return null; } return new NotificationSideChannelStub(); } return null; } /** * Handle a side-channeled notification being posted. */ public abstract void notify(String packageName, int id, String tag, Notification notification); /** * Handle a side-channelled notification being cancelled. */ public abstract void cancel(String packageName, int id, String tag); /** * Handle the side-channelled cancelling of all notifications for a package. */ public abstract void cancelAll(String packageName); private class NotificationSideChannelStub extends INotificationSideChannel.Stub { @Override public void notify(String packageName, int id, String tag, Notification notification) throws RemoteException { checkPermission(getCallingUid(), packageName); long idToken = clearCallingIdentity(); try { NotificationCompatSideChannelService.this.notify(packageName, id, tag, notification); } finally { restoreCallingIdentity(idToken); } } @Override public void cancel(String packageName, int id, String tag) throws RemoteException { checkPermission(getCallingUid(), packageName); long idToken = clearCallingIdentity(); try { NotificationCompatSideChannelService.this.cancel(packageName, id, tag); } finally { restoreCallingIdentity(idToken); } } @Override public void cancelAll(String packageName) { checkPermission(getCallingUid(), packageName); long idToken = clearCallingIdentity(); try { NotificationCompatSideChannelService.this.cancelAll(packageName); } finally { restoreCallingIdentity(idToken); } } } private void checkPermission(int callingUid, String packageName) { for (String validPackage : getPackageManager().getPackagesForUid(callingUid)) { if (validPackage.equals(packageName)) { return; } } throw new SecurityException("NotificationSideChannelService: Uid " + callingUid + " is not authorized for package " + packageName); } }