这两天开始做一个音乐播放器,慢慢完善功能,打算把遇到的一些问题记下来,故对代码没有什么解释。。希望读者能有所收获。
为了能让读者看的不那么难受,我把代码拆开来讲(完整的代码就是拆分的拼起来),包括笔者遇到的两个报异常的地方,解决后终于成功开启。
效果图在最后(简陋版看看就好)
如果哪里有误欢迎指正,不,请务必指正。
先初始化我们定义的Layout
这个Layout就是下拉菜单栏显示的样式。
views = new RemoteViews(getPackageName(), R.layout.notification_music);
views.setTextViewText(R.id.tv_playmusic, musics.get(curMusic).getMusic());
views.setTextViewText(R.id.tv_playsinger, musics.get(curMusic).getSinger());
RemoteViews能够支持更改Layout中控件用到的资源(如更换背景),使用
setInt(int viewId, String methodName, int value)
,
其中methodName是调用的方法名(实际是以方法名通过反射),value是资源文件的id。
定义Layout中控件的点击效果
//next music
Intent intentSkipNext = new Intent(MusicChangedReceiver.Action_Notification_SkipNext);
PendingIntent skipNextPendingIntent = PendingIntent.getBroadcast(this, NEXT_PENDINGINTENT_REQUESTCODE,
intentSkipNext, PendingIntent.FLAG_CANCEL_CURRENT);
views.setOnClickPendingIntent(R.id.btn_skip_next, skipNextPendingIntent);
这部分只贴一个,实际是通过发送广播实现的,需要绑定一个BroadcastReceiver来接受广播,在通过switch分Action进行对应处理。
定义点击Notification的动作(就是Layout除去定义了点击事件的控件的位置)
我这里的动作是启动MainActivity(设置了singleTask的启动模式)
注意这里没有把PendingIntent放入RemoteViews中,而是要直接设置在Notification中(下一部分)
Intent intentContent = new Intent(this, MainActivity.class);
PendingIntent contentPendingIntent = PendingIntent.getActivity(this, CONTENT_PENDINGINTENT_REQUESTCODE,
intentContent, PendingIntent.FLAG_CANCEL_CURRENT);
定义Notification移除时的动作
我没有定义,暂时没这个需求。
创建Notification.Builder
从这里就可以看出来Notification使用了Builder模式
其中 setSmallIcon, setContentTitle, setContentText 是必须要有的,不然会出现异常前台开启失败,反正最后不会显示出来(因为设置了自定义的布局)
builder = new Notification.Builder(this)
// 设置小图标
.setSmallIcon(R.drawable.ic_stop_red_40dp)
// 设置标题
.setContentTitle("nemuniPlayer")
// 设置内容
.setContentText("content")
.setAutoCancel(false)
//这里设置点击Notification的动作
//这里设置点击Notification的动作
//这里设置点击Notification的动作
.setContentIntent(contentPendingIntent)
.setContent(views);
获取NotificationManager
notificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
!重点!设置NotificationChannel,8.0以上不设会报异常启动失败
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
String CHANNEL_ID = "com.example.nemuni.channel";
String CHANNEL_NAME = "Music Channel";
NotificationChannel notificationChannel= new NotificationChannel(CHANNEL_ID, CHANNEL_NAME,
NotificationManager.IMPORTANCE_HIGH);
notificationChannel.setLockscreenVisibility(Notification.VISIBILITY_PUBLIC);
notificationManager.createNotificationChannel(notificationChannel);
builder.setChannelId(CHANNEL_ID);
}
(我是不会说ID和NAME的命名都是我随意的,确保ID唯一应该就没问题了吧)
剩下的就是启动了
startForeground(NOTIFICATION_PENDINGINTENT_ID, builder.build());
这里的ID我使用了进程的ID(反正我也就启动这一个前台服务)
private static final int NOTIFICATION_PENDINGINTENT_ID = android.os.Process.myPid();
顺便贴onDestroy()里的取消操作吧
if (notificationManager != null) {
notificationManager.cancel(NOTIFICATION_PENDINGINTENT_ID);
stopForeground(true);
}