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

相关推荐
2601_949833399 小时前
flutter_for_openharmony口腔护理app实战+预约管理实现
android·javascript·flutter
2603_9494621011 小时前
Flutter for OpenHarmony社团管理App实战:预算管理实现
android·javascript·flutter
王泰虎13 小时前
安卓开发日记,因为JCenter 关闭导致加载不了三方库应该怎么办
android
2601_9495430116 小时前
Flutter for OpenHarmony垃圾分类指南App实战:主题配置实现
android·flutter
2601_9498333918 小时前
flutter_for_openharmony口腔护理app实战+知识实现
android·javascript·flutter
晚霞的不甘18 小时前
Flutter for OpenHarmony从基础到专业:深度解析新版番茄钟的倒计时优化
android·flutter·ui·正则表达式·前端框架·鸿蒙
鸟儿不吃草18 小时前
android的Retrofit请求https://192.168.43.73:8080/报错:Handshake failed
android·retrofit
Minilinux201818 小时前
Android音频系列(09)-AudioPolicyManager代码解析
android·音视频·apm·audiopolicy·音频策略
李子红了时19 小时前
【无标题】
android
Android系统攻城狮20 小时前
Android tinyalsa深度解析之pcm_close调用流程与实战(一百零四)
android·pcm·tinyalsa·音频进阶·音频性能实战·android hal