1.前言
本文介绍兼容AndroidQ的“Bitmap插入相册”方案。
以前的发送广播方式已失效,并且insertImage(ContentResolver cr, Bitmap source, String title, String description)也已被废弃。
因此需要使用新的插入相册实现方式。并针对AndroidQ进行版本适配。
2. “Bitmap插入相册”具体实现:
这里有一个需要注意的地方:put(MediaStore.Images.Media.IS_PENDING, 1)。这个设置是做耗时操作时,需要独占资源。但是使用结束后,务必注意解除独占。
/**
* 保存Bitmap到相册
*/
fun insertBitmap(
context: Context,
picBitmap: Bitmap,
picName: String,
mimeType: String
): Boolean {
val contentValues = ContentValues().apply {
this.put(MediaStore.MediaColumns.DISPLAY_NAME, picName)
this.put(MediaStore.MediaColumns.MIME_TYPE, mimeType)
val imageTime = System.currentTimeMillis()
this.put(MediaStore.MediaColumns.DATE_ADDED, imageTime / 1000)
this.put(MediaStore.MediaColumns.DATE_MODIFIED, imageTime / 1000)
// 判断是否android10 以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
// 设置相对路径(自动创建文件夹)
val folder = context.resources.getString(R.string.app_name)
val relativeName =
Environment.DIRECTORY_PICTURES + File.separator + folder + File.separator
this.put(MediaStore.MediaColumns.RELATIVE_PATH, relativeName)
// 设置独占锁:耗时操作,独占访问权限,完成操作需复位
this.put(MediaStore.Images.Media.IS_PENDING, 1)
}
}
val resolver: ContentResolver = context.contentResolver
resolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, contentValues)?.also { uri ->
try {
// First, write the actual data for our screenshot
resolver.openOutputStream(uri).use { out ->
if (!picBitmap.compress(Bitmap.CompressFormat.PNG, 100, out)) {
throw IOException("Failed to compress")
}
}
//判断是否android10 以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
// 复位(解除)独占权限
contentValues.clear()
contentValues.put(MediaStore.Audio.Media.IS_PENDING, 0)
resolver.update(uri, contentValues, null, null)
}
} catch (e: IOException) {
// resolver.delete(uri, null)
Log.e(TAG, Log.getStackTraceString(e))
return false
}
}
return true
}
相关文档:
参考文档:
官方文档地址:[https://developer.android.google.cn/training/data-storage/shared/media#add-item]