一、相机拍照
先配置里添加文件读写权限
<!-- 读写文件权限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
在res下面的xml新建一个file_paths配置文件
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-cache-path
name="take_photo"
path="."/>
</paths>
在配置文件里添加这个配置目录,
name :配置当前 FileProvider 的实现类。
authorities:配置一个 FileProvider 的名字,它在当前系统内需要是唯一值。
exported:表示该 FileProvider 是否需要公开出去,这里不需要,所以是 false。
granUriPermissions:是否允许授权文件的临时访问权限。这里需要,所以是 true。
再加一个meta-data是文件引用目录。
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="FileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
</application>
先创建一个文件,文件类型为jpg,名字为output_image,第一个创建一个文件对象, 然后将该文件创建出来,如果存在就删除,重新创建,然后实例到一个共享的缓存目录,目录名是配置好的,然后放最后一个参数就放进去刚创建好的文件对象,变量名是imageUri,相当于有了这么一个空图片文件,启动照相机,拍完照后,带着图片参数,返回结果集
File outputImage = new File(getExternalCacheDir(), "output_image.jpg");
try {
if (outputImage.exists()) {
outputImage.delete();
}
outputImage.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
imageUri = FileProvider.getUriForFile(this,
"com.example.FileProvider", outputImage);
// 添加手机外部启动
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
// 把图片uri参数携带过去
intent.putExtra("output", imageUri);
通过结果集,调用BitmapFactory.decodeStream(),里面参数是获取上下文对象,调用写入流把图片写进刚刚创建好的共享缓存目录,把流变成了Bitmap图,图片就可以设置了。
Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
person_set_header.setImageBitmap(bitmap);
二、从相册获取图片
先checkSelfPermission判断是否获取了内存卡读写的权限,没有则调用requestPermissions选择存储卡申请读写权限,然后开始执行方法,打开相册,设置显示的类型只有图片,然后选取后会开始调用结果集,
// 内存卡读写 等于0则授权了,等于-1则未授权 如果没授权,则开启权限
if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
//相册中的照片都是存储在SD卡上的,需要申请运行时权限,WRITE_EXTERNAL_STORAGE是危险权限,表示同时授予程序对SD卡的读和写的能力
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);
} else {
openAlbum();
pop.dismiss();
}
private void openAlbum() {
/*打开图库*/
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("image/*");
startActivityForResult(intent, 2);//打开相册
}
传入结果集的data,然后getData获取uri,判断是否是document类型,因为现在普遍Android版本都是4.4以上,所以这里直接按照高版本获取路径的方法来写了,这里判断是document类型后,调用了getDocumentID,然后获取的id会带有iamge和id参数,以分号分割,写1则获取后面id的参数,写0则是前面iamge的,所以要获取后面的填1,然后拼接查询字段,写成一个 字符串,用来后面查询
private void handleImageOnKiKai(Intent data) {
String imagePath = null;
Uri uri = data.getData();
if (DocumentsContract.isDocumentUri(this, uri)) {
//如果是Document类型的Uri,则通过document id 处理
String docId = DocumentsContract.getDocumentId(uri);
// 获取Document的id 以冒号分割开始截取,0是第一块image,1是第二块
String id = docId.split(":")[1];
// 拼接字符串,拼一个字段
String selection = MediaStore.Images.Media._ID + "=" + id;
imagePath = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection);
displayImage(imagePath);
}
}
传过来的参数是MediaStore.Images.Media.EXTERNAL_CONTENT_URI, 还有selection,第一个参数图片共享文件,第二个是我们要查找图片的拼接号的id字段,然后通过cursor的query方法进行查询,然后选择第一个结果位置,就是找到的图片位置,然后getString,获取该列图片数据转换成字符串,就是图片真实路径。
private String getImagePath(Uri uri, String selection) {
String path = null;
//通过Uri 和selection获取真正的图片路径 返回的cursor是对应结果集
Cursor cursor = getContentResolver().
query(uri, null, selection, null, null);
// 指向第一个结果位置
if (cursor.moveToFirst()) {
// 获取数据,图片的真实路径
path = cursor.getString(
cursor.getColumnIndex(MediaStore.Images.Media.DATA));
}
cursor.close();
return path;
}
把找到的图片编码成图片文件资源,然后设置就大功告成。
private void displayImage(String path) {
if (path != null) {
// 将路径转换为一个Bitmap对象
Bitmap bitmap = BitmapFactory.decodeFile(path);
person_set_header.setImageBitmap(bitmap);
} else {
Toast.makeText(this, "加载失败", Toast.LENGTH_LONG).show();
}
}