1.前言
本文介紹兼容AndroidQ的“圖片文件(包括gif文件)插入相冊”方案。
以前的發(fā)送廣播方式已失效憋飞,并且insertImage(ContentResolver cr, String imagePath, String name, String description)也已被廢棄他炊。
因此需要使用新的插入相冊實現(xiàn)方式争剿。并針對AndroidQ進行版本適配已艰。
2. “圖片插入相冊”具體實現(xiàn):
這里有一個需要注意的地方:put(MediaStore.Images.Media.IS_PENDING, 1)。這個設(shè)置是做耗時操作時秒梅,需要獨占資源旗芬。但是使用結(jié)束后,務(wù)必注意解除獨占捆蜀。
fun insertImage(context: Context, filePath: String) {
if (!checkFile(filePath)) {
return
}
val resolver = context.applicationContext.contentResolver
val imageCollection = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
MediaStore.Images.Media.getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY)
} else {
MediaStore.Images.Media.EXTERNAL_CONTENT_URI
}
val contentValues = ContentValues().apply {
val saveFile = File(filePath)
put(
MediaStore.Images.Media.DISPLAY_NAME,
saveFile.nameWithoutExtension + System.currentTimeMillis()
)
//設(shè)置文件類型為image/*
put(MediaStore.MediaColumns.MIME_TYPE, getPhotoMimeType(filePath))
val imageTime = System.currentTimeMillis()
put(MediaStore.MediaColumns.DATE_ADDED, imageTime / 1000)
put(MediaStore.MediaColumns.DATE_MODIFIED, imageTime / 1000)
// 判斷是否android10 以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
// 設(shè)置相對路徑(自動創(chuàng)建文件夾)
val folder = context.resources.getString(R.string.app_name)
val relativeName =
Environment.DIRECTORY_PICTURES + File.separator + folder + File.separator
put(MediaStore.MediaColumns.RELATIVE_PATH, relativeName)
// 設(shè)置獨占鎖:耗時操作疮丛,獨占訪問權(quán)限,完成操作需復(fù)位
put(MediaStore.Images.Media.IS_PENDING, 1)
}
}
resolver.insert(imageCollection, contentValues)?.let { insertUri ->
val isSuccess = copyFile(context, resolver, filePath, insertUri)
//判斷是否android10 以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
// 復(fù)位(解除)獨占權(quán)限
contentValues.clear()
contentValues.put(MediaStore.Images.Media.IS_PENDING, 0)
resolver.update(insertUri, contentValues, null, null)
}
}
}
3. 以下是其他工具方法:
3.1 獲取圖片類型(mine_type)
/**
* 獲取照片的mine_type
*
* @param path
* @return
*/
private fun getPhotoMimeType(path: String): String? {
val lowerPath = path.toLowerCase()
if (lowerPath.endsWith("jpg") || lowerPath.endsWith("jpeg")) {
return "image/jpeg"
} else if (lowerPath.endsWith("png")) {
return "image/png"
} else if (lowerPath.endsWith("webp")) {
return "image/webp"
} else if (lowerPath.endsWith("gif")) {
return "image/gif"
}
return "image/jpeg"
}
3.2 檢測文件是否存在
/**
* 檢測文件存在
*
* @param filePath
* @return
*/
private fun checkFile(filePath: String): Boolean {
//boolean result = FileUtil.fileIsExist(filePath);
var result = false
val mFile = File(filePath)
if (mFile.exists()) {
result = true
}
val logStr = if (result) "文件已存在" else "文件不存在"
Log.e(TAG, "$logStr, path = $filePath")
return result
}
3.3 圖片使用流寫入相冊
/**
* 通過兩個路徑地址生成對應(yīng)的輸入輸出流
* 主要方式獲取ContentResolver.openOutputStream(insertUri)
*/
private fun copyFile(
context: Context,
contentResolver: ContentResolver,
sourceFilePath: String,
insertUri: Uri
): Boolean {
var inputStream: InputStream? = null //輸入流
var outputStream: OutputStream? = null //輸出流
return try {
var isCopySuccess = false
outputStream = contentResolver.openOutputStream(insertUri)?.also { outStream ->
val sourceFile = File(sourceFilePath)
if (sourceFile.exists()) { // 文件存在時
// 讀入原文件
inputStream = FileInputStream(sourceFile).also { inStream ->
//輸入流讀取文件辆它,輸出流寫入指定目錄
isCopySuccess = copyFileWithStream(outStream, inStream)
}
}
}
isCopySuccess
} catch (e: Exception) {
e.printStackTrace()
false
} finally {
try {
inputStream?.close()
outputStream?.close()
} catch (e: IOException) {
e.printStackTrace()
}
}
}
/**
* 利用輸入輸出流邊讀邊寫誊薄,將文件(圖片)寫入指定路徑
*/
private fun copyFileWithStream(outputStream: OutputStream, inputStream: InputStream): Boolean {
var isSuccess = false
try {
inputStream.use { ins ->
val buf = ByteArray(2048)
var len: Int
while (ins.read(buf).also { len = it } != -1) {
outputStream.write(buf, 0, len)
}
outputStream.flush()
}
isSuccess = true
} catch (e: IOException) {
Log.e(TAG, Log.getStackTraceString(e))
isSuccess = false
} finally {
try {
outputStream.close()
inputStream.close()
} catch (e: IOException) {
e.printStackTrace()
}
}
return isSuccess
}