今天看完郭神关于M版本运行权限的视频,突然想到何不把最近项目中关于N版本适配的内容总结一下,和大家分享,下面几个问题有两个前提:手机是N系统以及Gradle中targetSDK配置为24。
setRingerMode(int ringerMode)
首先,我先卖个关子,大家先看下下面代码有什么问题没有:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
AudioManager audioManager = (AudioManager) getApplicationContext().getSystemService(Context.AUDIO_SERVICE);
audioManager.setRingerMode(AudioManager.RINGER_MODE_SILENT);
}
});
}
}
代码很简单,主界面上一个button,点击之后启动静音模式,在N之前的版本,这样做是没问题的,但是在N版本上,会出现Fatal Exception。
com.example.saberhao.newfeatures E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.example.saberhao.newfeatures, PID: 23714
java.lang.SecurityException: Not allowed to change Do Not Disturb state
at android.os.Parcel.readException(Parcel.java:1693)
at android.os.Parcel.readException(Parcel.java:1646)
at android.media.IAudioService$Stub$Proxy.setRingerModeExternal(IAudioService.java:1264)
at android.media.AudioManager.setRingerMode(AudioManager.java:1311)
at com.example.saberhao.newfeatures.MainActivity.onClick(MainActivity.java:27)
在 API 24 上,有如下说明
From N onward, ringer mode adjustments that would toggle Do Not Disturb are not allowed unless the app has been granted Do Not Disturb Access. See isNotificationPolicyAccessGranted().
从N版本之后,App必须取得勿打扰权限才能使用setRingerMode去更改铃声状态,通过isNotificationPolicyAccessGranted() 可以判断当前是否已经获得这个权限
那么我们怎么获取勿打扰权限呢,API里也有说明:
Request policy access by sending the user to the activity that matches the system intent action ACTION_NOTIFICATION_POLICY_ACCESS_SETTINGS.
发送intent进行请求,这个权限的申请和M版本是有区别的,下面给出完整的方案:
1.在Androidmenifest.xml中声明权限
<uses-permission android:name="android.permission.ACCESS_NOTIFICATION_POLICY"/>
2.在改变铃声状态前加入权限判断和申请逻辑
NotificationManager notificationManager = (NotificationManager)getApplicationContext.getSystemService(Context.NOTIFICATION_SERVICE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N
&& !notificationManager.isNotificationPolicyAccessGranted()) {
Intent intent = new Intent(Settings.ACTION_NOTIFICATION_POLICY_ACCESS_SETTINGS);
getApplicationContext().startActivity(intent);
return;
}
加入权限判断后,会启动如下页面进行权限请求:
在获得权限之后,就可以对系统铃声状态进行更改。
multi-window
谷歌在7.0上加入了多窗口的功能,当然多窗口并非谷歌首创,如果我没记错的话,应该是三星第一个支持,由于android N正式支持多窗口,三星也放弃了自家多窗口的框架,投入谷歌的怀抱,针对N版本的App默认是启动多窗口功能的,当然谷歌也为这个功能提供了开关
android:resizeableActivity=[“true” | “false”]
如果产品经理说,我们不需要支持多窗口,那么可以赶紧把 android:resizeableActivity=“false” 加到你的androidmenifest中吧,毕竟要做好适配估计也要花些功夫。
顺便说下三星自家的多窗口,其实市面上很多App都是不支持的,因为默认就是不支持,估计大家也懒得适配,其实三星官方也为我们提供了多窗口的sample,我们只需要在androidmenifest加入如下说明即可:
<meta-data android:name="com.samsung.android.sdk.multiwindow.enable" android:value="true"/>
<meta-data android:name="android.intent.category.MULTIWINDOW_LAUNCHER" android:value="true"/>
<meta-data android:name="com.sec.android.intent.category.MULTIWINDOW_LAUNCHER" android:value="true"/>
删除部分隐式intent
android手机由于各种大家都懂得原因,安卓N删除了CONNECTIVITY_ACTION、ACTION_NEW_PICTURE 和ACTION_NEW_VIDEO 三个隐式intent,以优化耗电性能,虽然谷歌删除了删除这些Intent,但是Android提供了多个解决方案来缓解App对这些intent的需求,比如我们可以结合jobscheduler来对网络状态进行判断
private static void addScheduleJob(Context context) {
final int SCHEDULE_JOB = 0;
if (android.os.Build.VERSION.SDK_INT < Build.VERSION_CODES.N)
return;
ComponentName serviceName = new ComponentName(context, ScheduleJobService.class);
JobScheduler jobscheduler = (JobScheduler)context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
JobInfo.Builder builder = new JobInfo.Builder(SCHEDULE_JOB, serviceName);
builder.setPeriodic(TimeUnit.HOURS.toMillis(1)); //One Hour
builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY);
builder.setRequiresCharging(false);
int ret = jobscheduler.schedule(builder.build());
if (ret < 0) {
Log.e("addScheduleJob","failed");
}
}
这段代码的意思是,每隔30分钟,对当前网络状态进行检查,如果当前是联网状态,则启动ScheduleJobService.class 中 SCHEDULE_JOB 对应的操作。我想这样让App变成系统可控的任务,比起原来监听隐式广播,对用户友好很多,也是维护安卓生态的有力措施。
由于真正接触N也不是很久,还有许多不清楚的地方,如果大家有更好的经验分享,请不吝赐教。