内容如题,由于安卓10开始,官方开始对隐私相关内容进行修改,变得更加严格。
所以,对于隐私相关的内容,如文件保存等,就需要进行适配。
下面开始介绍如何进行图片在安卓高版本中适配。
开发环境:
win10+androidstudio4.4+jdk1.8
targetVersion 30
文末附相关代码链接
思路
(一)如何在保存图片,保存在哪个目录
(二)如何删除图片
实现
(一)保存图片:
如果是用于应用之间,或者图片是用于跨应用操作,如图片分享等等。建议保存在系统开发的目录,如系统的Picture目录。如果图片只用于应用内部,建议保存在应用内部的目录。
核心代码如下:
(1)应用保存在应用内目录核心代码:
String rootPath = MediaPathManager.getInstance().getAppInnerRootPath(context);
String savePath = rootPath + File.separator + relPath + File.separator;
File fileDir = new File(savePath);
if (!fileDir.exists()) {
createFile(fileDir, false);
}
File f = new File(savePath + fileName);
if (!fileDir.exists()) {
createFile(f, true);
}
try {
FileOutputStream out = new FileOutputStream(f);
source.compress(Bitmap.CompressFormat.PNG, 90, out);
out.flush();
out.close();
LogUtil.d("save app dir: " + (savePath + fileName));
return savePath + fileName;
} catch (IOException e) {
e.printStackTrace();
}
return "";
注意,如果保存在应用内部目录,是不需要权限的,只需要创建好相关目录和文件即可。
(2)保存在系统相册目录相关代码:
private static String saveMediaToSys(Context context, Bitmap bitmap, String dirType, String relativeDir,
String filename, String mimeType, String description) throws IOException {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q) {
//首先保存
File saveDir = Environment.getExternalStoragePublicDirectory(dirType);
saveDir = new File(saveDir, relativeDir);
if (!saveDir.exists() && !saveDir.mkdirs()) {
try {
throw new Exception("create directory fail!");
} catch (Exception e) {
e.printStackTrace();
}
}
File outputFile = new File(saveDir, filename);
FileOutputStream fos = null;
try {
fos = new FileOutputStream(outputFile);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
BufferedOutputStream bos = new BufferedOutputStream(fos);
bitmap.compress(Bitmap.CompressFormat.JPEG, 80, bos);
bos.flush();
bos.close();
//最后通知图库更新
context.sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.fromFile(outputFile)));
return outputFile.getPath();
} else {
String path = (!TextUtils.isEmpty(relativeDir)) ?
(Environment.DIRECTORY_PICTURES + File.separator + relativeDir) :
Environment.DIRECTORY_PICTURES;
ContentValues contentValues = new ContentValues();
contentValues.put(MediaStore.Images.Media.DISPLAY_NAME, filename);
contentValues.put(MediaStore.Images.Media.DESCRIPTION, description);
contentValues.put(MediaStore.Images.Media.RELATIVE_PATH, path);
contentValues.put(MediaStore.Images.Media.MIME_TYPE, mimeType);
//contentValues.put(MediaStore.Images.Media.IS_PENDING,1)
Uri external = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
Uri insertUri = context.getContentResolver().insert(external, contentValues);
OutputStream fos = (OutputStream) null;
if (insertUri != null) {
try {
fos = context.getContentResolver().openOutputStream(insertUri);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
if (fos != null) {
bitmap.compress(Bitmap.CompressFormat.JPEG, 90, fos);
fos.flush();
fos.close();
}
return MediaUriUtils.getFilePathFromContentUri(context, insertUri);
}
}
从上述代码中可以看出,如果是低于安卓10,是直接通过file文件进行创建,并且通过广播通知系统相册进行刷新的。而在安卓10以上(包含)则是通过MediaStore和contentResolver方法,进行插入到系统,原理其实是创建一个uri后写入文件,不过这个过程系统已经帮我们处理好,开发者只需要调用相关api即可。
(二)删除图片
原理上,其实可以参考插入图片的逻辑,因为换汤不换药的。核心代码如下:
public static void deletePicWithUri(Activity activity, String path) {
try {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q) {
new File(path).delete();
} else {
try {
Uri imageUri = MediaUriUtils.getImageContentUri(activity, new File(path));
activity.getContentResolver().delete(imageUri, null, null);
} catch (RecoverableSecurityException e1) {
//捕获 RecoverableSecurityException异常,发起请求
try {
ActivityCompat.startIntentSenderForResult(activity,
e1.getUserAction().getActionIntent().getIntentSender(),
CODE_REQ, null, 0, 0, 0, null);
} catch (IntentSender.SendIntentException e) {
e.printStackTrace();
}
}
}
} catch (Exception e) {
}
}
上述代码思路就是:如果低于安卓10版本,则进行file的操作,高于则进行uri的操作。
写在最后
(1)相关操作时,注意权限申请
(2)文件操作时,注意目录的创建,对象的释放
(3)耗时操作,注意放在子线程
that’s all--------------------------------------------------------------------
20230303更新
上述代码中,实测还是会发现存在问题的。所以经过总结,测试。有了如下的更新:
核心代码如下:
安卓10以上:
/**
* 保存到系统--Q以后
*
* @param externalUrl MediaStore.Images.Media.EXTERNAL_CONTENT_URI
* MediaStore.Video.Media.EXTERNAL_CONTENT_URI
*/
private static String saveToSystemAfterQ(Context context, ContentValues contentValues,
String fileFullPath, Uri externalUrl) throws Exception {
ContentResolver contentResolver = context.getContentResolver();
File tempFile = new File(fileFullPath);
Uri external = externalUrl;
Uri uri = contentResolver.insert(external, contentValues);
copyFileAfterQ(context, contentResolver, tempFile, uri);
contentValues.clear();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
contentValues.put(MediaStore.MediaColumns.IS_PENDING, 0);
}
context.getContentResolver().update(uri, contentValues, null, null);
context.sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, uri));
return MediaUriUtils.getFilePathFromContentUri(context, uri);
}
/**
* 获取图片的ContentValue--Q以后
*/
public static ContentValues getContentValuesAfterQ(String rootPath, File paramFile, String mimeType,
String desc) {
long timestamp = System.currentTimeMillis();
ContentValues localContentValues = new ContentValues();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
localContentValues.put(MediaStore.Images.Media.RELATIVE_PATH, rootPath);
}
localContentValues.put(MediaStore.Images.Media.TITLE, paramFile.getName());
localContentValues.put(MediaStore.Images.Media.DISPLAY_NAME, paramFile.getName());
localContentValues.put(MediaStore.Images.Media.MIME_TYPE, mimeType);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
localContentValues.put(MediaStore.Images.Media.DATE_TAKEN, timestamp);
localContentValues.put(MediaStore.Images.Media.ORIENTATION, 0);
}
localContentValues.put(MediaStore.Images.Media.DATE_MODIFIED, timestamp);
if (!TextUtils.isEmpty(desc)) {
localContentValues.put(MediaStore.Images.Media.DESCRIPTION, desc);
}
localContentValues.put(MediaStore.Images.Media.DATE_ADDED, timestamp);
localContentValues.put(MediaStore.Images.Media.DATA, paramFile.getAbsolutePath());
localContentValues.put(MediaStore.Images.Media.SIZE, paramFile.length());
return localContentValues;
}
/**
* 安卓Q以后文件复制
*/
private static void copyFileAfterQ(Context context, ContentResolver localContentResolver,
File tempFile, Uri localUri) throws IOException {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q &&
context.getApplicationInfo().targetSdkVersion >= Build.VERSION_CODES.Q) {
//拷贝文件到相册的uri,android10及以上得这么干,否则不会显示。可以参考ScreenMediaRecorder的save方法
OutputStream os = localContentResolver.openOutputStream(localUri);
Files.copy(tempFile.toPath(), os);
os.close();
tempFile.delete();
}
}
对于安卓10之前的,就直接复制就好了,这里就不一一细说。