1.前言
本文介绍兼容AndroidQ的“视频文插入相册”方案。
以前的发送广播方式已失效。
因此需要使用新的插入相册实现方式。并针对AndroidQ进行版本适配。
2. “视频文件插入相册”具体实现:
这里有一个需要注意的地方:put(MediaStore.Images.Media.IS_PENDING, 1)。这个设置是做耗时操作时,需要独占资源。但是使用结束后,务必注意解除独占。
fun insertVideo(context: Context, filePath: String) {
if (!checkFile(filePath)) {
return
}
val resolver = context.applicationContext.contentResolver
val videoCollection = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
MediaStore.Video.Media.getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY)
} else {
MediaStore.Video.Media.EXTERNAL_CONTENT_URI
}
val contentValues = ContentValues().apply {
val saveFile = File(filePath)
put(
MediaStore.Video.Media.DISPLAY_NAME,
saveFile.nameWithoutExtension + System.currentTimeMillis()
)
put(MediaStore.Video.Media.DURATION, getDurationOfVideo(filePath))
put(MediaStore.MediaColumns.MIME_TYPE, getVideoMimeType(filePath))
val videoTime = System.currentTimeMillis()
put(MediaStore.MediaColumns.DATE_ADDED, videoTime / 1000)
put(MediaStore.MediaColumns.DATE_MODIFIED, videoTime / 1000)
// 判断是否android10 以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
// 设置相对路径(自动创建文件夹)
val folder = context.resources.getString(R.string.app_name)
val relativeName =
Environment.DIRECTORY_MOVIES + File.separator + folder + File.separator
put(MediaStore.MediaColumns.RELATIVE_PATH, relativeName)
// 设置独占锁:耗时操作,独占访问权限,完成操作需复位
put(MediaStore.Video.Media.IS_PENDING, 1)
}
}
resolver.insert(videoCollection, contentValues)?.let { insertUri ->
val isSuccess = copyFile(context, resolver, filePath, insertUri)
//判断是否android10 以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
// 复位(解除)独占权限
contentValues.clear()
contentValues.put(MediaStore.Video.Media.IS_PENDING, 0)
resolver.update(insertUri, contentValues, null, null)
}
}
}
3. 以下是其他工具方法:
3.1 获取视频类型(mine_type)
/**
* 获取video的mine_type,暂时只支持mp4,3gp
*
* @param path
* @return
*/
private fun getVideoMimeType(path: String): String {
val lowerPath = path.toLowerCase()
if (lowerPath.endsWith("mp4") || lowerPath.endsWith("mpeg4")) {
return "video/mp4"
} else if (lowerPath.endsWith("3gp")) {
return "video/3gp"
}
return "video/mp4"
}
3.2 获取视频时长(duration)
/**
* 获取视频时长
*/
private fun getDurationOfVideo(filePath: String): kotlin.Long {
if (!checkFile(filePath)) {
return 0
}
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD_MR1) {
try {
val mmr = MediaMetadataRetriever()
mmr.setDataSource(filePath)
val durationStr = mmr.extractMetadata(MediaMetadataRetriever.METADATA_KEY_DURATION)?: "0L"
durationStr.toLong()
} catch (e: java.lang.Exception) {
0
}
} else 0
}
3.3 检测文件是否存在
/**
* 检测文件存在
*
* @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.4 视频文件使用流写入相册
/**
* 通过两个路径地址生成对应的输入输出流
* 主要方式获取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
}
相关文档:
参考文档:
官方文档地址:[https://developer.android.google.cn/training/data-storage/shared/media#add-item]