Android Media3(五)— 实现视频截图功能

最近在某个技术交流群中,有人问怎么实现视频截图。本文简单介绍下如何使用Media3实现视频截图功能。

添加依赖

在app module下的build.gradle中添加代码,如下:

scss 复制代码
dependencies {
    implementation("androidx.media3:media3-ui:1.1.0")
    implementation("androidx.media3:media3-session:1.1.0")
    implementation("androidx.media3:media3-exoplayer:1.1.0")
}

实现视频截图功能

TextureView提供了获取视频截图的方法getBitmap(),可以获取Bitmap.Config.ARGB_8888像素格式的图片,图片的宽高与TextureView的宽高一致。

  • 设置PlayerViewsurface_type

在布局文件中将PlayerViewsurface_type设置为texture_view

ini 复制代码
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    ......

    <androidx.media3.ui.AspectRatioFrameLayout
        android:id="@+id/arfl_player_container"
        android:layout_width="match_parent"
        android:layout_height="320dp"
        android:layout_marginTop="10dp"
        app:layout_constraintTop_toBottomOf="@id/btn_screenshot">

        <androidx.media3.ui.PlayerView
            android:id="@+id/play_view"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            app:resize_mode="fill"
            app:show_buffering="always"
            app:surface_type="texture_view"
            app:use_controller="false" />

    </androidx.media3.ui.AspectRatioFrameLayout>

    ......
    
</androidx.constraintlayout.widget.ConstraintLayout>
  • 生成截图

通过PlayViewgetVideoSurfaceView()获取TextureView,并生成截图,代码如下:

kotlin 复制代码
class Media3ExampleActivity : AppCompatActivity() {

    private lateinit var binding: LayoutMedia3ExampleActivityBinding

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = LayoutMedia3ExampleActivityBinding.inflate(layoutInflater)
        setContentView(binding.root)
        binding.includeTitle.tvTitle.text = "Media3 Example"
        binding.playView.player = ExoPlayer.Builder(this)
            .build()
        binding.playView.player?.run {
            addListener(object : Player.Listener {
                ......
            })
            repeatMode = Player.REPEAT_MODE_ALL
            playWhenReady = true
        }
        binding.btnPlaySingleVideo.setOnClickListener {
            binding.playView.player?.run {
                stop()
                setMediaItem(MediaItem.fromUri("https://minigame.vip/Uploads/images/2021/09/18/1631951892_page_img.mp4"))
                prepare()
            }
        }
        binding.btnScreenshot.setOnClickListener {
            // 获取TextureView并生成截图
            (binding.playView.videoSurfaceView as? TextureView)?.bitmap?.let {
                binding.ivScreenshotContainer.setImageBitmap(it)
            }
        }
    }

    override fun onResume() {
        super.onResume()
        binding.playView.onResume()
    }

    override fun onPause() {
        super.onPause()
        binding.playView.onPause()
    }

    override fun onDestroy() {
        super.onDestroy()
        binding.playView.player?.release()
        binding.playView.player = null
    }
}

效果如图:

示例

演示代码已在示例Demo中添加。

ExampleDemo github

ExampleDemo gitee

相关推荐
杉氧2 小时前
Navigation Compose 深度实践:如何优雅地串联起你的全栈 App?
android·架构·android jetpack
雨白6 小时前
指针与数组的核心机制
android
黄林晴10 小时前
Room 3.0 正式发布!包名彻底重构,KMP 成为核心主线
android·android jetpack
三少爷的鞋11 小时前
Kotlin 协程环境下的 DCL 懒加载:别把线程时代的经验直接搬过来
android
plainGeekDev11 小时前
Gson → kotlinx.serialization
android·java·kotlin
CYY951 天前
Compose 入门篇
android·kotlin
杉氧1 天前
Compose 时代的 MVI 架构:如何用单向数据流驱动复杂 UI?
android·架构·android jetpack
杉氧1 天前
Modifier 的艺术:为什么链式调用的顺序决定了UI 的生命周期?
android·架构·android jetpack
李斯维1 天前
腾讯 XLog 日志框架 Android 端接入
android·android studio·android jetpack