Android(Kotlin):如何从多张图片创建视频
创始人
2024-10-13 07:31:11
0

要从多张图片创建视频,你可以使用Android提供的MediaCodec和MediaMuxer类来实现。下面是一个使用Kotlin编写的示例代码,演示了如何从多张图片创建视频:

import android.media.MediaCodec
import android.media.MediaCodecInfo
import android.media.MediaFormat
import android.media.MediaMuxer
import android.os.Environment
import android.util.Log
import java.io.File
import java.io.IOException
import java.nio.ByteBuffer

class ImageToVideoConverter {
    private val TAG = "ImageToVideoConverter"

    private val MIME_TYPE = "video/avc"
    private val FRAME_RATE = 10
    private val IFRAME_INTERVAL = 1
    private val TIMEOUT_US = 10000L

    private var mediaCodec: MediaCodec? = null
    private var mediaMuxer: MediaMuxer? = null
    private var bufferInfo: MediaCodec.BufferInfo? = null
    private var outputFile: File? = null
    private var input: ArrayList? = null

    fun convertImagesToVideo(imagePaths: ArrayList, outputFilePath: String) {
        input = imagePaths
        outputFile = File(outputFilePath)

        try {
            mediaCodec = MediaCodec.createEncoderByType(MIME_TYPE)
            val format = MediaFormat.createVideoFormat(MIME_TYPE, 720, 1280)
            format.setInteger(MediaFormat.KEY_COLOR_FORMAT, MediaCodecInfo.CodecCapabilities.COLOR_FormatSurface)
            format.setInteger(MediaFormat.KEY_BIT_RATE, 2000000)
            format.setInteger(MediaFormat.KEY_FRAME_RATE, FRAME_RATE)
            format.setInteger(MediaFormat.KEY_I_FRAME_INTERVAL, IFRAME_INTERVAL)
            mediaCodec?.configure(format, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE)
            mediaCodec?.start()
            bufferInfo = MediaCodec.BufferInfo()

            mediaMuxer = MediaMuxer(outputFile?.absolutePath, MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4)

            val inputSurface = mediaCodec?.createInputSurface()
            val videoEncoderThread = VideoEncoderThread(inputSurface)
            videoEncoderThread.start()
            mediaMuxer?.start()

            val presentationTimeUs = 0L
            for (i in input!!.indices) {
                val imagePath = input!![i]
                val bitmap = BitmapFactory.decodeFile(imagePath)
                val frame = Bitmap.createBitmap(bitmap.width, bitmap.height, Bitmap.Config.ARGB_8888)
                val canvas = Canvas(frame)
                canvas.drawBitmap(bitmap, 0f, 0f, null)
                val byteBuffer = ByteBuffer.allocate(frame.byteCount)
                frame.copyPixelsToBuffer(byteBuffer)
                byteBuffer.rewind()
                val buffer = byteBuffer.array()
                val inputBufferIndex = mediaCodec?.dequeueInputBuffer(TIMEOUT_US)
                if (inputBufferIndex!! >= 0) {
                    val inputBuffer = mediaCodec?.getInputBuffer(inputBufferIndex)
                    inputBuffer?.clear()
                    inputBuffer?.put(buffer)
                    mediaCodec?.queueInputBuffer(inputBufferIndex, 0, buffer.size, presentationTimeUs, 0)
                }
                mediaCodec?.dequeueOutputBuffer(bufferInfo!!, TIMEOUT_US)
                while (bufferInfo?.flags!! and MediaCodec.BUFFER_FLAG_CODEC_CONFIG != 0) {
                    mediaCodec?.releaseOutputBuffer(inputBufferIndex!!, false)
                    inputBufferIndex = mediaCodec?.dequeueOutputBuffer(bufferInfo!!, TIMEOUT_US)
                }
                mediaCodec?.stop()
                mediaCodec?.release()

                mediaCodec = null
                mediaCodec = MediaCodec.createEncoderByType(MIME_TYPE)
                mediaCodec?.configure(format, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE)
                mediaCodec?.start()
            }

            videoEncoderThread.quit()
            mediaMuxer?.stop()
            mediaMuxer?.release()
        } catch (e: IOException) {
            Log.e(TAG, "Failed to create video from images: ${e.message}")
        }
    }

    private inner class VideoEncoderThread(private val inputSurface: Surface?) : Thread() {
        private var isRunning = true

        override fun run() {
            val bufferInfo = MediaCodec.BufferInfo()
            while (isRunning) {
                val outputBufferIndex = mediaCodec?.dequeueOutputBuffer(bufferInfo, TIMEOUT_US)
                if (outputBufferIndex!! >= 0) {
                    val outputBuffer = mediaCodec?.getOutputBuffer(outputBufferIndex)
                    mediaMuxer?.writeSampleData(0, outputBuffer!!, bufferInfo)

                    mediaCodec?.releaseOutputBuffer(outputBufferIndex, false)

                    if (bufferInfo.flags and MediaCodec.BUFFER_FLAG_END_OF_STREAM != 0) {
                        break
                    }
                }
            }

相关内容

热门资讯

玩家实测!fishpoker透... 玩家实测!fishpoker透视底牌,xpoker透视辅助,分享教程(有挂技巧)1、点击下载安装,微...
揭秘几款!hhpoker有辅助... 1、揭秘几款!hhpoker有辅助吗,wepoker脚本下载,第三方教程(有挂辅助);详细教程。2、...
总算清楚!wepoker祈福有... 总算清楚!wepoker祈福有用吗,wepoker私人局怎么玩,攻略教程(有挂方法);玩家必备必赢加...
记者揭秘!aapoker辅助怎... 记者揭秘!aapoker辅助怎么用,hhpoker是真的还是假的,专业教程(有挂技巧);hhpoke...
盘点一款!epoker透视底牌... 1、盘点一款!epoker透视底牌,大菠萝免费辅助,详细教程(有挂辅助);详细教程。2、大菠萝免费辅...
重大通报!wepoker高级辅... 1、重大通报!wepoker高级辅助,wepoker国外版透视,规律教程(有挂技巧);详细教程。2、...
一秒答解!wepoker究竟有... 一秒答解!wepoker究竟有没有透视,hhpoker脚本下载,介绍教程(有挂透明);建议优先通过w...
每日必备!aapoker公共底... 1、每日必备!aapoker公共底牌,wepoker底牌透视脚本,攻略方法(有挂软件)(UU pok...
今日百科!wepoker破解游... 今日百科!wepoker破解游戏盒子,hardrock透视工具,新2025教程(有挂软件)是由北京得...
1.9分钟了解!wepoker... 自定义wepoker私人局俱乐部辅助系统规律,只需要输入自己想要的开挂功能,一键便可以生成出微扑克专...