添加依赖
implementation 'com.github.bumptech.glide:glide:4.2.0' annotationProcessor 'com.github.bumptech.glide:compiler:4.2.0'activity_main.xml文件
<?xml version="1.0" encoding="utf-8"?> <android.support.v7.widget.LinearLayoutCompat xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <android.support.v7.widget.AppCompatButton android:id="@+id/take_photo" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="take photo"/> <android.support.v7.widget.AppCompatImageView android:id="@+id/picture" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center_horizontal" /> </android.support.v7.widget.LinearLayoutCompat>MainActivity
public class MainActivity extends AppCompatActivity { public static final int TAKE_PHOTO = 1; private AppCompatImageView picture; private Uri imageUri; private static final RequestOptions OPTIONS = new RequestOptions() .diskCacheStrategy(DiskCacheStrategy.ALL) .centerCrop() .dontAnimate(); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); AppCompatButton takePhoto = findViewById(R.id.take_photo); picture = findViewById(R.id.picture); takePhoto.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { //创建File对象,用于存储拍照后的图片 //getExternalCacheDir() 可以得到SD卡中专门用于存放当前应用缓存数据的位置,叫做应用关联缓存目录 // 具体路径是/sdcard/Android/data/<package name>/cache //为什么要用关联缓存目录来存放照片,因为6.0以后,读写SD卡是危险权限,图片存放在SD卡的其他任何目录,都需要运行时权限处理,应用关联缓存目录可以跳过这一步 File outputImage = new File(getExternalCacheDir(), "output_img.jpg"); try { if (outputImage.exists()) { outputImage.delete(); } outputImage.createNewFile(); } catch (IOException e) { e.printStackTrace(); } if (Build.VERSION.SDK_INT >= 24) {//大于7.0 //FileProvider.getUriForFile() 将File对象转换成一个封装过的Uri对象 7.0 以后直接使用本地真实路径的Uri被认为是不安全的, // 而FileProvider书=是一种特殊的内容提供器,它使用了和内容提供器类似的机制来对数据进行保护,可以选择性的将封装过的Uri共享给外部,提高了应用的安全性 imageUri = FileProvider.getUriForFile(MainActivity.this, "com.zsp.cameraalbumtest.fileprovider", outputImage); } else { imageUri = Uri.fromFile(outputImage); } //启动相机程序 Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); //指定图片的输出地址 intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); startActivityForResult(intent, TAKE_PHOTO); } }); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { switch (requestCode) { case TAKE_PHOTO: if (resultCode == RESULT_OK) { //将拍摄的照片显示出来 try { Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri)); //picture.setImageBitmap(bitmap); 我的小米手机测试不显示照片,所以选择了Glide加载图片 Glide.with(this) .load(imageUri) .apply(OPTIONS) .into(picture); } catch (FileNotFoundException e) { e.printStackTrace(); } } break; default: break; } } }创建@xml/file_paths,右击res目录---New---Directory,创建一个xml目录,接着右击xml目录---New --File,
创建file_paths.xml文件
修改file_paths.xml
<?xml version="1.0" encoding="utf-8"?> <pahts xmlns:xmls="http://schemas.android.com/apk/res/android"> <external-path name="my_images" path=""/> </pahts>external-path就是用来指定Uri共享的,name属性值可以随便填,path属性的值表示共享的具体路径,设置为空表示将整个SD卡共享。在AndroidManifest.xml中对内容提供器进行注册
<provider android:name="android.support.v4.content.FileProvider" android:authorities="com.zsp.cameraalbumtest.fileprovider" android:exported="false" android:grantUriPermissions="true"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" /> </provider>其中 android:name="android.support.v4.content.FileProvider"属性的值时固定的
android:authorities的值和getUriForFile()的第二个参数保持一致
<meta-data>指定Uri的共享路径
申明权限
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>运行效果
在activity_main.xml文件中添加按钮
<android.support.v7.widget.AppCompatButton android:id="@+id/choose_from_album" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="choose_from_album" />修改MainActivity中的代码
public class MainActivity extends AppCompatActivity { public static final int TAKE_PHOTO = 1; public static final int CHOOSE_PHOTO = 2; private AppCompatImageView picture; private Uri imageUri; private static final RequestOptions OPTIONS = new RequestOptions() .diskCacheStrategy(DiskCacheStrategy.ALL) .centerCrop() .dontAnimate(); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); AppCompatButton takePhoto = findViewById(R.id.take_photo); picture = findViewById(R.id.picture); takePhoto.setOnClickListener(new View.OnClickListener() {//拍照 @Override public void onClick(View view) { //创建File对象,用于存储拍照后的图片 //getExternalCacheDir() 可以得到SD卡中专门用于存放当前应用缓存数据的位置,叫做应用关联缓存目录 // 具体路径是/sdcard/Android/data/<package name>/cache //为什么要用关联缓存目录来存放照片,因为6.0以后,读写SD卡是危险权限,图片存放在SD卡的其他任何目录,都需要运行时权限处理,应用关联缓存目录可以跳过这一步 File outputImage = new File(getExternalCacheDir(), "output_img.jpg"); try { if (outputImage.exists()) { outputImage.delete(); } outputImage.createNewFile(); } catch (IOException e) { e.printStackTrace(); } if (Build.VERSION.SDK_INT >= 24) {//大于7.0 //FileProvider.getUriForFile() 将File对象转换成一个封装过的Uri对象 7.0 以后直接使用本地真实路径的Uri被认为是不安全的, // 而FileProvider书=是一种特殊的内容提供器,它使用了和内容提供器类似的机制来对数据进行保护,可以选择性的将封装过的Uri共享给外部,提高了应用的安全性 imageUri = FileProvider.getUriForFile(MainActivity.this, "com.zsp.cameraalbumtest.fileprovider", outputImage); } else { imageUri = Uri.fromFile(outputImage); } //启动相机程序 Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); //指定图片的输出地址 intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); startActivityForResult(intent, TAKE_PHOTO); } }); AppCompatButton chooseFromAlbum = findViewById(R.id.choose_from_album); chooseFromAlbum.setOnClickListener(new View.OnClickListener() {//从相册中选取 @Override public void onClick(View view) { if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1); } else { openAlbum(); } } }); } private void openAlbum() { Intent intent = new Intent("android.intent.action.GET_CONTENT"); intent.setType("image/*"); //打开相册 startActivityForResult(intent, CHOOSE_PHOTO); } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { switch (requestCode) { case 1: if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { openAlbum(); } else { Toast.makeText(this, "You denied the permission", Toast.LENGTH_LONG).show(); } break; default: break; } } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { switch (requestCode) { case TAKE_PHOTO: if (resultCode == RESULT_OK) { //将拍摄的照片显示出来 try { Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri)); //picture.setImageBitmap(bitmap); 我的小米手机测试不显示照片,所以选择了Glide加载图片 Glide.with(this) .load(imageUri) .apply(OPTIONS) .into(picture); } catch (FileNotFoundException e) { e.printStackTrace(); } } break; case CHOOSE_PHOTO: if (resultCode == RESULT_OK) { //判断手机版本号 if (Build.VERSION.SDK_INT >= 19) { //4.4及以后 //4.4 以后选取相册中的图片不再返回图片的真实uri,而是一个封装过的uri handleImageOnKitKat(data); } else { //4.4以前 handleImageBeforeKitKat(data); } } break; default: break; } } @RequiresApi(api = Build.VERSION_CODES.KITKAT) private void handleImageOnKitKat(Intent data) { String imagePath = null; Uri uri = data.getData(); if (DocumentsContract.isDocumentUri(this, uri)) { //如果是Document类型的uri,则通过Document id处理 String docId = DocumentsContract.getDocumentId(uri); if ("com.android.providers.media.documents".equals(uri.getAuthority())) { String id = docId.split(":")[1];//解析出数字格式的id String selection = MediaStore.Images.Media._ID + "=" + id; imagePath = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,selection); }else if ("com.android.providers.downloads.documents".equals(uri.getAuthority())){ Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/publick_downloads"),Long.valueOf(docId)); imagePath = getImagePath(contentUri,null); } }else if ("content".equalsIgnoreCase(uri.getScheme())){ //如果是content类型的Uri,则用普通方式处理 imagePath = getImagePath(uri,null); }else if ("file".equalsIgnoreCase(uri.getScheme())){ //如果是 file类型的Uri,直接获取图片路径即可 imagePath = uri.getPath(); } //根据图片路径显示图片 displayImage(imagePath); } private void handleImageBeforeKitKat(Intent data) { Uri uri = data.getData(); String imagePath = getImagePath(uri,null); displayImage(imagePath); } private String getImagePath(Uri uri, String selection) { String path = null; //通过uri和selection来获取图片的真实路径 Cursor cursor = getContentResolver().query(uri,null,selection,null,null); if (cursor!=null){ if (cursor.moveToFirst()){ path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA)); } cursor.close(); } return path; } private void displayImage(String imagePath) { if (imagePath!=null){ Glide.with(this) .load(imagePath) .apply(OPTIONS) .into(picture); }else { Toast.makeText(this,"failed to get image",Toast.LENGTH_LONG).show(); } } }运行效果
打开相册选择图片
展示图片