Android应用商店——Splash页面的实现,Android运行时权限的使用
参考:
1. http://blog.csdn.net/lmj623565791/article/details/50709663
2. 《Android进阶之光》
上篇介绍了该项目的总体介绍,有兴趣的童鞋可以点击查看:
Android应用商店——项目介绍
Splash页面开发
Splash页面一般情况下用于展示app的logo、本版本的相关信息,多为一张图片,初始化完毕后,便进入程序的主体UI界面。本项目闪屏页面只展示相关logo,点击“进入首页”时,需用户手动授予读取SD卡的权限。
Android运行时权限
实现支持运行时权限
想必大家都知道,Android 6.0之前,安装app会列出安装此app的相关访问权限,而且只有安装时会出现一次,一旦我们同意并安装了此app,这个app就可以在用户毫不知晓的情况下访问权限内的所有东西(流氓软件全家桶)。Android 6.0以后的版本,google对此类情况进行了整改,app不得不在运行时一个一个询问用户来授予权限。对于开发者而言,你需要在每个需要权限的地方检查权限,否则等待你的就是app崩溃!!!
google将权限分为两类:一类是Normal Permissions,一般不涉及用户隐私,无需用户授权,而且用户不能取消这类授权,比如手机震动、访问网络等,只要在清单文件中声明即可;
- Normal Permissions如下
1. ACCESS_LOCATION_EXTRA_COMMANDS 2. ACCESS_NETWORK_STATE 3. ACCESS_NOTIFICATION_POLICY 4. ACCESS_WIFI_STATE 5. BLUETOOTH 6. BLUETOOTH_ADMIN 7. BROADCAST_STICKY 8. CHANGE_NETWORK_STATE 9. CHANGE_WIFI_MULTICAST_STATE 10. CHANGE_WIFI_STATE 11. DISABLE_KEYGUARD 12. EXPAND_STATUS_BAR 13. GET_PACKAGE_SIZE 14. INSTALL_SHORTCUT 15. INTERNET 16. KILL_BACKGROUND_PROCESSES 17. MODIFY_AUDIO_SETTINGS 18. NFC 19. READ_SYNC_SETTINGS 20. READ_SYNC_STATS 21. RECEIVE_BOOT_COMPLETED 22. REORDER_TASKS 23. REQUEST_INSTALL_PACKAGES 24. SET_ALARM 25. SET_TIME_ZONE 26. SET_WALLPAPER 27. SET_WALLPAPER_HINTS 28. TRANSMIT_IR 29. UNINSTALL_SHORTCUT 30. USE_FINGERPRINT 31. VIBRATE 32. WAKE_LOCK 33. WRITE_SYNC_SETTINGS
第二类是Dangerous Permissions,一般会涉及用户隐私,需要用户手动进行授权,比如读取sdcard、访问通信录等。
- Dangerous Permissions:
1. group:android.permission-group.CONTACTS permission:android.permission.WRITE_CONTACTS permission:android.permission.GET_ACCOUNTS permission:android.permission.READ_CONTACTS2. group:android.permission-group.PHONE permission:android.permission.READ_CALL_LOG permission:android.permission.READ_PHONE_STATE permission:android.permission.CALL_PHONE permission:android.permission.WRITE_CALL_LOG permission:android.permission.USE_SIP permission:android.permission.PROCESS_OUTGOING_CALLS permission:com.android.voicemail.permission.ADD_VOICEMAIL3. group:android.permission-group.CALENDAR permission:android.permission.READ_CALENDAR permission:android.permission.WRITE_CALENDAR4. group:android.permission-group.CAMERA permission:android.permission.CAMERA5. group:android.permission-group.SENSORS permission:android.permission.BODY_SENSORS6. group:android.permission-group.LOCATION permission:android.permission.ACCESS_FINE_LOCATION permission:android.permission.ACCESS_COARSE_LOCATION7. group:android.permission-group.STORAGE permission:android.permission.READ_EXTERNAL_STORAGE permission:android.permission.WRITE_EXTERNAL_STORAGE8. group:android.permission-group.MICROPHONE permission:android.permission.RECORD_AUDIO9. group:android.permission-group.SMS permission:android.permission.READ_SMS permission:android.permission.RECEIVE_WAP_PUSH permission:android.permission.RECEIVE_MMS permission:android.permission.RECEIVE_SMS permission:android.permission.SEND_SMS permission:android.permission.READ_CELL_BROADCASTS
Dangerous Permission是以分组的形式给出的,同一组的任何一个权限被授权了,其他权限也自动被授权。此外,对于申请时弹出的提示框上面的文本说明,也是对整个权限组的说明,而不是单个权限的说明。
public class SplashActivity extends Activity { /** * 用于保存是否第一次进入闪屏页面的配置文件 */ private SharedPreferences mSp; @BindView(R.id.bt_enter_home) Button mBt_enter_home; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_splash); ButterKnife.bind(this); mSp = getSharedPreferences(Constant.PREFERENCE_NAME,MODE_PRIVATE); boolean isFirstComeIn = mSp.getBoolean(Constant.PREFERENCE_KEY_IS_FIRST_COME_IN,true); if (!isFirstComeIn) {//直接进入 enterHomeActivity(); } } /** * 校验权限 */ private void verifyPermission() { //检查是否有读写SD的权限 if (ActivityCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {//如果没有该权限,去申请 ActivityCompat.requestPermissions(this,Constant.permissions,Constant.PERMISSION_REQUEST_CODE); } else { enterHomeActivity(); } } /** * “进入首页”Button的点击事件,进入主界面 */ @OnClick(R.id.bt_enter_home) protected void onEnterHome(){ verifyPermission(); } private void enterHomeActivity() { mSp.edit().putBoolean(Constant.PREFERENCE_KEY_IS_FIRST_COME_IN,false).commit(); Intent intent = new Intent(this, HomeActivity.class); startActivity(intent); finish(); } /** * 申请权限的回调 * @param requestCode * @param permissions * @param grantResults */ @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode) { case Constant.PERMISSION_REQUEST_CODE: if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { Toast.makeText(this,"授权SD卡成功",Toast.LENGTH_SHORT).show(); enterHomeActivity(); } else { Toast.makeText(this,"没有授权SD卡,可能会影响应用的使用",Toast.LENGTH_SHORT).show(); mBt_enter_home.setEnabled(false); } break; default: break; } }}
如果用户选择“允许”,则进入主界面UI,下一次就不会弹出权限申请提示框了;若选择“拒绝”,就弹出Toast显示权限被拒绝,下一次还会弹出权限申请提示框,只不过这一次会多一个选择项——“不再询问”,如下:
如果我们勾选了该选项,则下一次就不会弹出权限申请提示框,而直接调用onRequestPermissionsResult,回调结果为最后一次用户的选择,也就是弹出我们定义的Toast:”没有授权SD卡,可能会影响应用的使用”。
如果想要再次打开该权限,需要在 设置 —> 应用 —> 配置应用 —> 应用访问授权 —> 存储空间权限中操作(此设置为Android 7.1.1的原生设置,不同rom下的操作可能不同),如下图:
处理“不再询问”选项
如果用户选择了“不再询问”,那么每次我们调用需要访问该权限的API时都会失效,这显然是不好的用户体验。所以我们需要做的就是给用户一个友好的提。这时候需要使用shouldShowRequestPermissionRationale()方法,这个方法用来帮助开发者向用户解释权限的情况。如果用户选择了“不再询问”选项,则shouldShowRequestPermissionRationale()方法会返回false,这时候我们可以弹出AlertDialog来提示用户允许访问该权限的重要性:
....../** * 申请权限的回调 * @param requestCode * @param permissions * @param grantResults */ @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode) { case Constant.PERMISSION_REQUEST_CODE: if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { Toast.makeText(this,"授权SD卡成功",Toast.LENGTH_SHORT).show(); enterHomeActivity(); } else { Toast.makeText(this,"没有授权SD卡,可能会影响应用的使用",Toast.LENGTH_SHORT).show(); if (!ActivityCompat.shouldShowRequestPermissionRationale(this,Manifest.permission.READ_EXTERNAL_STORAGE)) { AlertDialog dialog = new AlertDialog.Builder(this).setMessage("该功能需要访问SD卡的权限,不开启将无法正常工作!").setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { } }).create(); dialog.show(); } mBt_enter_home.setEnabled(false); } break; default: break; }......
如下图:
PermissionsDispatcher框架的使用
以上繁琐的步骤可用第三方框架PermissionsDispatcher解决,具体使用可自行google或者github上直接查看
更多相关文章
- android task与back stack 开发文档翻译 - 3
- Android(安卓)Property System | Android属性系统
- Android权限管理之Permission权限机制及使用
- 基于ANDROID的网上订餐系统
- Android层次化安全架构及核心组件概览
- Android(安卓)防界面劫持方案,无视Android系统版本限制,无需操作栈
- android task与back stack 开发文档翻译 - 3
- AndroidManifest.xml文件详解
- Android系统权限和root权限