Android 7.0相機拍照適配
(1)首先必須獲取拍照的權(quán)限
簡單一點的可以直接用ActivityCompat
的requestPermissions
方法
ActivityCompat.requestPermissions(context,
new String[]{permission},
requestCode);
權(quán)限請求的結(jié)果會在onRequestPermissionsResult
中回調(diào)
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode) {
case 1://對應(yīng)requestPermissions的requestCode
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {//如果一次申請多個權(quán)限,就按順序依次grantResults[1]坪仇、grantResults[2]判斷
Toasts.showShort("再次點擊即可拍照");
} else {
// Permission Denied
}
break;
}
}
(2)調(diào)用相機拍照
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
UUID uuid = UUID.randomUUID();
File file = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM), uuid.toString() + ".jpeg");//拍照文件的路徑
//判斷是否是AndroidN以及更高的版本
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
Uri uri = FileProvider.getUriForFile(getContext(), BuildConfig.APPLICATION_ID + ".fileProvider", file);
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
} else {
Uri uri = Uri.fromFile(file);
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
}
startActivityForResult(intent, requestCode);
主要的適配就是Android 7.0以上發(fā)起調(diào)用的Uri必須通過FileProvider
來獲取啤它,也就是下面這句:
Uri uri = FileProvider.getUriForFile(getContext(), BuildConfig.APPLICATION_ID + ".fileProvider", file);
FileProvider
的getUriForFile
方法有三個參數(shù):
- 第一個參數(shù)為
Context
住闯,這個好理解 - 第二個參數(shù)為
FileProvider
的簽名,也就是一個唯一標(biāo)識,這個一個應(yīng)用里面必須是唯一的兼蜈,否則會有問題,后面會單獨講拙友。一般為包名+自定義的標(biāo)識为狸,這個必須和AndroidManifest
中配置的Provider的authorities
屬性一致 - 第三個參數(shù)也好立即,就是拍照將要生成的文件了
所以我們還需要對FileProvider
進(jìn)行特別的處理
(3)在AndroidManifest文件中注冊FileProvider
<!--Android N 相機-->
<provider
android:name=".provider.MyFileProvider"
android:authorities="${applicationId}.myfileprovider"http://這里表示授權(quán)信息
android:grantUriPermissions="true"http://必須為true遗契,表示同意權(quán)限
android:exported="false">//必須為false辐棒,否則會報錯
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
-
name
屬性為FileProvider的路徑 -
authorities
為授權(quán)的簽名,一般為包名+自定義的標(biāo)識牍蜂,與生成的Uri的第二個參數(shù)對應(yīng) -
grantUriPermissions
屬性必須為true -
exported
必須為false -
meta-data
中的resource
屬性必須配置漾根,為共享的文件路徑,也就是系統(tǒng)相機應(yīng)用和我們的應(yīng)用共享的文件路徑
(4)創(chuàng)建上面配置的file_paths
共享文件配置
文件內(nèi)容就是指定的共享路徑了
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name="external_files" path="."/>
</paths>
(5)最后在onActivityResult中接收照片
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
//拍照成功以后
if (resultCode == RESULT_OK) {
switch (requestCode) {
case FROM_CAMERA:
//做照片的處理
...
}
} else {
ToastUtil.shortToast(this, R.string.cancel);
}
}
適配需要注意的坑
(1)FileProvider路徑不一致
這里要特別注意上面獲取Uri時的第二個參數(shù)鲫竞,也就是包名+自定義的標(biāo)識必須和
AndroidManifest
文件中配置的一致-
第二個就比較坑了辐怕,如果是模塊化開發(fā),需要特別注意包名的一致
再來看看那句代碼:
Uri uri = FileProvider.getUriForFile(getContext(), BuildConfig.APPLICATION_ID + ".fileProvider", file);
坑就在
BuildConfig.APPLICATION_ID
這里贡茅,如果用BuildConfig
來獲取APPLICATION_ID
秘蛇,在模塊化開發(fā)中就會出現(xiàn)不一致的情況。比如打開相機的模塊是camera
顶考,我們應(yīng)用的包名是com.my.app
赁还,那么BuildConfig.APPLICATION_ID
的值就不一樣了:camera模塊:com.my.app.camera
我們app:com.my.app
這樣導(dǎo)致的后果就是導(dǎo)調(diào)用相機的時候閃退,并且每次通過IDE安裝應(yīng)用的時候都會彈出下面的提示要重新安裝App:
正確的姿勢是用getPackageName()
的方式獲取驹沿,也就是改成下面這樣:
Uri uri = FileProvider.getUriForFile(mContext, mContext.getPackageName() + ".myfileprovider", file);
(2)FileProvider重復(fù)
這個也出現(xiàn)在模塊話開發(fā)中艘策,或是引用的三方庫中也定義了FileProvider,就會報FileProvider重復(fù)的錯誤渊季。
Attribute provider#android.support.v4.content.FileProvider@authorities value=(***.fileProvider) from AndroidManifest.xml:352:13-62 is also present at ...
解決方法也很簡單朋蔫,就是定義一個我們自己的FileProvider:
public class MyFileProvider extends FileProvider {
}
是的,其他什么也不用干却汉,直接繼承FileProvider創(chuàng)建一個自己的FileProvider就好
然后驯妄,AndroidManifest文件中定義的FileProvider的name屬性改成上面的MyFileProvider
的路徑就好
<provider
android:name=".xtreme.provider.MyFileProvider" //自定義的FileProvider的路徑
android:authorities="${applicationId}.myfileprovider"
android:grantUriPermissions="true"
android:exported="false">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>