Android权限 - 第二篇

2507次阅读  |  发布于5年以前

原文链接 : Permissions – Part 2

在Marshmallow(棉花糖,Android6.0版本)中Android添加了一个新的权限模块,需要开发者在授权的时候做一些不同的处理。在这一系列中,我们从技术角度看下如何处理请求的权限和如何提供流畅的用户体验。

Icon_no_permission

之前我们讨论了如何检查我们需要的权限是否被授予,但是我们没有讨论在适当的地方请求缺失的权限。在这篇文章中,我们来看一看在所有Acitivity中,在没有很多重复代码的情况下我们是如何请求和检测必要权限的。请记住,接下来的都是在Marshmallow或之后的版本(更早OS层Manifest声明的权限会被直接授权),如果你在builds文件中特别标明targetSdkVersion=23或更高,你就需要实现这种检测。

所以第一件事情是我们要知道权限请求模块是如何工作的。就像我们已经讨论过的,标准权限会被直接授权,但是危险的权限的请求需要用户授予。用户授予需要的权限是很容易的事情,但我们需要考虑到用户没有授予权限。针对这个app,我们最终编写成一个对用户来说并不是非常明显的我们需要RECORD_AUDIO权限的原因,所以,我们要准备通知用户为什么这个是需要的。

从用户的角度来看,工作方式是第一次运行app的时候请求用户授权需要的权限:

Part2-first-run

如果他们授权了这个权限,万事好说,我们就不急了。但是如果他们拒绝了这个权限,我们就要反复的询问需要被授权的权限:

Part2-second-run

请注意如果用户之前拒绝了权限申请,它们会显示一个不再显示这个权限请求的选项。如果用户选择了一个选项,代码中任何进一步的尝试权限请求会被自动拒绝。做为开发者要清楚这个问题,做好预留措施。

用户可以进一步设置,进到应用的设置页面,可以授权或拒绝任何应用需要的权限。这就是为什么不仅仅在应用启动的时候检测需要的权限是否被授予的重要性,而且每个activity的权限都随时都有可能发生变化。

我们将要管理这个的模式是一个单独的负责请求权限的Activity,app中所有的其他Activity都需要检测所需要的权限,如果他们没有被授权PermissionsActivity会被回调。

让我稍微更新下MainActivity

MainActivity.java

public class MainActivity extends AppCompatActivity {

    static final String[] PERMISSIONS = new String[]{Manifest.permission.RECORD_AUDIO, Manifest.permission.MODIFY_AUDIO_SETTINGS};
    private PermissionsChecker checker;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        checker = new PermissionsChecker(this);
    }

    @Override
    protected void onResume() {
        super.onResume();

        if (checker.lacksPermissions(PERMISSIONS)) {
            startPermissionsActivity();
        }
    }

    private void startPermissionsActivity() {
        PermissionsActivity.startActivity(this, PERMISSIONS);
    }
}

我们把权限检测代码放到onResume()。这就解决了用户暂停我们的app,切换到设置页面,拒绝了权限,然后又返回到我们的app的情况。好,这只是一个小概率事件,但对于预防崩溃是值得做的。

我们实现的这个基础模式是无论何时Activity被恢复我们都可以确定有操作Activity的所需要的权限。如果我们没有权限,我们将控制权交给负责获得需要的权限的PermissionsActivity。虽说这感觉像是一个防御性方案,但我认为这个一个不需要大量代码的明智的方案。PermissionsChecker内封装好了所有检测逻辑,请求调用是在PermissionsActivity。

轻量级的权限检测组件是重要的,因为我们可以相对简单的检测需要的权限,在Activity有必要请求权限的地方请求是更浪费的。

在下一篇文章中我们会看到PermissionsActivity如何处理这个权限的请求,探索如果用户拒绝了一个我们需要的权限,如何告知用户我们为什么需要。

这篇文章的源码在这里。PermissionsActivity代码中有一个占位符,我会在下篇文章中扩展,所以它现在不是一个完整功能的代码。

© 2016, Mark Allison. All rights reserved.

Copyright© 2013-2019

京ICP备2023019179号-2