开启前台服务

这两天开始做一个音乐播放器,慢慢完善功能,打算把遇到的一些问题记下来,故对代码没有什么解释。。希望读者能有所收获。
为了能让读者看的不那么难受,我把代码拆开来讲(完整的代码就是拆分的拼起来),包括笔者遇到的两个报异常的地方,解决后终于成功开启。
效果图在最后(简陋版看看就好)

如果哪里有误欢迎指正,不,请务必指正。

先初始化我们定义的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);
        }
效果图
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容