知乎开源的图片选择器 Matisse 在 6.0 以上手机使用相机功能

需求:实现一个图片选择器,能选择本地图片拍照

拿到需求,第一个想法就是看自己在规定时间上能否又快又好的实现。显然是不能( 。。),现在那么多前辈都把轮子都造好了,我们直接拼装不久可以了吗?目前为止,我还是这样,等功能深厚成为前辈了再撸几个轮子给后辈使用,这些都是后话,先实现这个需求吧。

先去GitHub搜索一圈图片选择器,发现知乎开源的Matisse家伙长的挺好看的,就选它了。

使用步骤看GitHub的,官方的才是最正确的使用姿势。Matisse

或者是参看一下这位仁兄的介绍:Android 一起来看看知乎开源的图片选择库

官网的使用方式默认是不开启拍照功能的,因此需要拍照功能的可以这么写

1
2
3
4
5
6
7
8
9
10
11
12
Matisse.from(PublishActivity.this)
.choose(MimeType.allOf()) // 选择 mime 的类型
.countable(true) // 显示选择的数量
.capture(true) // 开启相机,和 captureStrategy 一并使用否则报错
.captureStrategy(new CaptureStrategy(true,"com.meiqu.pianxin.ui.publish.MyFileProvider")) // 拍照的图片路径
.theme(R.style.Matisse_Dracula) // 黑色背景
.maxSelectable(9) // 图片选择的最多数量
.gridExpectedSize(getResources().getDimensionPixelSize(R.dimen.grid_expected_size)) // 列表中显示的图片大小
.restrictOrientation(ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED)
.thumbnailScale(0.85f) // 缩略图的比例
.imageEngine(new GlideEngine()) // 使用的图片加载引擎
.forResult(REQUEST_CODE_CHOOSE); // 设置作为标记的请求码,返回图片时使用
  • captureStrategy(new CaptureStrategy(true,"com.meiqu.pianxin.ui.publish.MyFileProvider")) 主要是告知系统拍照的图片存储位置
  • 第二个参数是自己实现继承FileProvider类的一个空类,需要在manifest文件中添加如下代码
1
2
3
4
5
6
7
8
9
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.meiqu.pianxin.ui.publish.MyFileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/filepaths"/>
</provider>
  • android:resource=”@xml/filepaths” 是在res/xml 目录下的创建的文件filepaths.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
<paths xmlns:android="http://schemas.android.com/apk/res/android">

<!--代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory()/DCIM/camerademo目录-->
<external-path name="mq_DCIM" path="DCIM/camerademo" />

<!--代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory()/Pictures/camerademo目录-->
<external-path name="mq_Pictures" path="Pictures/camerademo" />

<!--代表app 私有的存储区域 Context.getFilesDir()目录下的images目录 /data/user/0/com.hm.camerademo/files/images-->
<files-path name="mq_private_files" path="images" />

<!--代表app 私有的存储区域 Context.getCacheDir()目录下的images目录 /data/user/0/com.hm.camerademo/cache/images-->
<cache-path name="mq_private_cache" path="images" />

<!--代表app 外部存储区域根目录下的文件 Context.getExternalFilesDir(Environment.DIRECTORY_PICTURES)目录下的Pictures目录-->
<external-files-path name="mq_external_files" path="Pictures" />
<!--代表app 外部存储区域根目录下的文件 Context.getExternalCacheDir目录下的images目录-->
<external-cache-path name="mq_external_cache" path="" />

<root-path name="mq_external_cache" path="" />
</paths>

以上使用在 5.0 以下的手机是没毛病的,在 6.0 这个动态权限的控制下就有些问题了

6.0 以上使用 Matisse 的正确姿势

Matisse 和 PermissionsDispatcher 的使用姿势

Matisse 需要用到相机和读写本地数据的权限

1
2
3
4
> <uses-permission android:name="android.permission.CAMERA" />
> <uses-permission android:name="android.permission.READ_PHONE_STATE" />
> <uses-permission android:name="android.Manifest.permission.READ_PHONE_STATE" />
>
  • 所以在调用了这些权限的地方是使用 PermissionsDispatcher 去动态设置,看PermissionsDispatcher 官网介绍

  • 我在实际中使用

  • 1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33

    步骤:
    1. 在Activity或者Fragment上添加 @RuntimePermissions
    2. 在用到权限的方法( 例如:void initData(),不能加上 private 修饰)名上添加 @NeedsPermission(Manifest.permission.READ_PHONE_STATE)
    3. 重写 onRequestPermissionsResult 方法,MainActivityPermissionsDispatcher.onRequestPermissionsResult(MainActivity.this, requestCode, grantResults);
    4. 在onCreate方法调用,MainActivityPermissionsDispatcher.initDataWithCheck(MainActivity.this);

    实例:
    @RuntimePermissions // 必须添加
    public class MainActivity extends AppCompatActivity {
    ...
    @Override
    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    ....
    MainActivityPermissionsDispatcher.initDataWithCheck(MainActivity.this);
    ....
    }

    @NeedsPermission(Manifest.permission.READ_PHONE_STATE) // 必须添加
    void initData() {
    // 会调用用户信息权限
    DataCenter dc = DataCenter.getInstance();
    dc.initDataCenter(this);
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    MainActivityPermissionsDispatcher.onRequestPermissionsResult(MainActivity.this, requestCode, grantResults);
    }
    }

小额支持我写出更好的文章~