跳到主要内容

使用LPP播放器播放视频 (C/C++)

从API version 20开始,使用LPP(low power player)播放器可以通过低功耗实现从媒体源到渲染的视频通路能力。本指南通过播放本地视频的示例,讲解如何使用LowPowerPlayer播放视频。

LowPowerPlayer播放器不支持纯视频和纯音频播放。纯音频低功耗播放请参考低功耗音频播放

播放流程包含:创建解封装器、创建播放器、设置回调监听函数、配置播放参数、播放控制(播放/暂停/继续/倍速/音量/停止/重置)、销毁播放器实例。

图1 播放状态变化示意图

播放流程包含:创建(created)、初始化(initialized)、就绪(ready)、解码(decoding)和渲染(rendering)五个阶段。

应用通过调用CreateByMime初始化创建解码器实例。完成解码器参数配置Configure,切换到initialized(已初始化)状态。

在初始化完成的状态下,完成解码器资源预加载(Prepare),视频进入准备就绪状态(ready)。然后启动解码流程(StartDecode),切换到解码(decoding)状态,触发首帧渲染(RenderFirstFrame)。最后启动渲染(StartRender)流程,切换到渲染(rendering)状态。

在渲染过程中,遇到暂停(Pause)会切换到paused状态,此时解码与渲染被临时挂起,资源未释放。当恢复渲染(Resume)时,视频将恢复并回到渲染(rendering)状态。当遇到停止(Stop)时,会切换到stopped状态,该状态代表解码器已停止工作,但实例仍存在。当渲染过程中流结束(EoS)时,会切换到eos状态。

在播放过程中,如果遇到错误(OnError),会出现异常,需要重置或进入释放阶段(released)销毁解码器实例,释放所有资源。

当播放处于ready/decoding/rendering/paused/stopped状态时,播放引擎此时处于工作状态会占用较多的系统资源。当暂停使用播放器时,可调用reset或destroy回收资源。

开发建议

当前指导仅介绍如何实现媒体资源播放,在应用开发过程中可能会涉及后台播放、播放冲突等情况,请根据实际需要参考以下说明。

开发步骤及注意事项

在CMake脚本中链接动态库。

target_link_libraries(sample PUBLIC liblowpower_avsink.so)

头文件引入。

#include "multimedia/player_framework/lowpower_audio_sink_base.h"
#include "multimedia/player_framework/lowpower_audio_sink.h"
#include "multimedia/player_framework/lowpower_video_sink.h"
#include "multimedia/player_framework/lowpower_video_sink_base.h"

开发者使用系统日志能力时,需引入如下头文件:

#include <hilog/log.h>

并需要在CMake脚本中链接如下动态库:

target_link_libraries(sample PUBLIC libhilog_ndk.z.so)

使用该模块时,需要链接的库如下所示:解封装、基础解码、显示渲染等能力。

set(BASE_LIBRARY
libnative_media_codecbase.so libnative_media_core.so libnative_media_vdec.so libnative_window.so
libnative_media_venc.so libnative_media_acodec.so libnative_media_avdemuxer.so libnative_media_avsource.so
libohaudio.so
)
target_link_libraries(sample PUBLIC ${BASE_LIBRARY})

开发者通过引入lowpower_audio_sink_base.hlowpower_audio_sink.hlowpower_video_sink.hlowpower_video_sink_base.h 头文件,使用音视频播放相关API。

  1. 创建播放器。

    根据实际情况,应用可使用自研解封装或可通过OH_AVSource_CreateWithDataSource()/OH_AVSource_CreateWithFD()/OH_AVSource_CreateWithURI()来创建OH_AVSource ,通过OH_AVSource调用OH_AVDemuxer_CreateWithSource(),创建解封装器,获取视频的元信息。

    source_ = OH_AVSource_CreateWithFD(info.inputFd, info.inputFileOffset, info.inputFileSize);
    demuxer_ = OH_AVDemuxer_CreateWithSource(source_);
    int32_t ret = GetTrackInfo(sourceFormat, info);
  2. 根据视频元信息,调用 OH_LowPowerAudioSink_CreateByMimeOH_LowPowerVideoSink_CreateByMime来创建播放器。

    lppVideoStreamer_ = OH_LowPowerVideoSink_CreateByMime(codecMime.c_str());
    lppAudioStreamer_ = OH_LowPowerAudioSink_CreateByMime(codecMime.c_str());
  3. 设置回调监听函数。

    调用OH_LowPowerAudioSinkCallback_CreateOH_LowPowerVideoSinkCallback_Create创建OH_LowPowerAudioSinkCallbackOH_LowPowerVideoSinkCallback的回调函数的整合,通过setListener函数向该结构体添加对应的回调函数,完成registerCallback的一次性注册。

    lppAudioStreamerCallback_ = OH_LowPowerAudioSinkCallback_Create();
    OH_LowPowerAudioSinkCallback_SetDataNeededListener(lppAudioStreamerCallback_, LppCallback::OnDataNeeded, lppUserData);
    OH_LowPowerAudioSinkCallback_SetPositionUpdateListener(lppAudioStreamerCallback_, LppCallback::OnPositionUpdated, lppUserData);
    ret = OH_LowPowerAudioSink_RegisterCallback(lppAudioStreamer_, lppAudioStreamerCallback_);
  4. 配置播放器。

    根据之前通过解封装获得的元信息,创建并配置OH_AVFormat。通过configure接口 OH_LowPowerAudioSink_Configure / OH_LowPowerVideoSink_Configure进行播放器的配置,详细参数可参考示例代码。视频流需要使用OH_LowPowerVideoSink_SetVideoSurface接口来设置显示窗口。

    OH_AVFormat *format = OH_AVFormat_Create();

    OH_AVFormat_SetIntValue(format, OH_MD_KEY_WIDTH, sampleInfo.videoWidth);
    OH_AVFormat_SetIntValue(format, OH_MD_KEY_HEIGHT, sampleInfo.videoHeight);
    OH_AVFormat_SetDoubleValue(format, OH_MD_KEY_FRAME_RATE, sampleInfo.frameRate);
    OH_AVFormat_SetIntValue(format, OH_MD_KEY_PIXEL_FORMAT, sampleInfo.pixelFormat);
    OH_AVFormat_SetIntValue(format, OH_MD_KEY_ROTATION, sampleInfo.rotation);

    int ret = OH_LowPowerVideoSink_Configure(lppVideoStreamer_, format);
  5. 准备播放。

    准备播放前,需要调用OH_LowPowerVideoSink_SetSyncAudioSink设置音画同步绑定。然后调用prepare方法,OH_LowPowerAudioSink_PrepareOH_LowPowerVideoSink_Prepare进入'准备'阶段。

    OH_LowPowerVideoSink_Prepare(lppVideoStreamer_);
  6. 开始播放。

    调用OH_LowPowerAudioSink_StartOH_LowPowerVideoSink_StartRenderer开始渲染。视频流需要在渲染开始前调用OH_LowPowerVideoSink_StartDecoder开始解码或调用 OH_LowPowerVideoSink_RenderFirstFrame开始解码并送显首帧'接口'进入解码。

    OH_LowPowerVideoSink_StartDecoder(lppVideoStreamer_);
    OH_LowPowerVideoSink_StartRenderer(lppVideoStreamer_);
  7. 播放控制(可选)。

    暂停:OH_LowPowerAudioSink_PauseOH_LowPowerVideoSink_Pause

    恢复播放:OH_LowPowerAudioSink_ResumeOH_LowPowerVideoSink_Resume

    停止:OH_LowPowerAudioSink_StopOH_LowPowerVideoSink_Stop

    设置声音:OH_LowPowerAudioSink_SetVolume

    设置播放速度:OH_LowPowerAudioSink_SetPlaybackSpeedOH_LowPowerVideoSink_SetPlaybackSpeed

    清空缓存数据,可用于seek操作:OH_LowPowerAudioSink_FlushOH_LowPowerVideoSink_Flush

  8. 更换资源(可选)。

    调用OH_LowPowerAudioSink_ResetOH_LowPowerVideoSink_Reset重置资源,允许更换资源,重新配置播放器。

  9. 退出播放。

    调用OH_LowPowerAudioSink_DestroyOH_LowPowerVideoSink_Destroy销毁实例,AVPlayer进入'RELEASED'状态,退出播放。

运行完整示例

  1. 新建工程。下载示例工程,并将示例工程的以下资源复制到对应目录。

    lpp_demo-sample/entry/src/main/
    ├── cpp # Native层
    │ ├── capbilities # 能力接口和实现
    │ │ ├── include # 能力接口
    │ │ ├── demuxer.cpp # 解封装实现
    │ │ ├── lpp_audio_streamer.cpp # 低功耗音频流实现
    │ │ └── lpp_video_streamer.cpp # 低功耗视频流实现
    │ ├── common # 公共模块
    │ │ ├── dfx # 日志
    │ │ ├── lpp_callback.cpp # 低功耗音视频回调实现
    │ │ ├── lpp_callback.h # 低功耗音视频回调接口
    │ │ └── sample_info.h # 功能实现公共类
    │ ├── render # 送显模块接口和实现 * window player设置
    │ │ ├── include # 送显模块接口
    │ │ ├── egl_core.cpp # 送显参数设置
    │ │ ├── plugin_manager.cpp # 送显模块管理实现
    │ │ └── plugin_render.cpp # 送显逻辑实现
    │ ├── sample # Native层
    │ │ ├── player # Native层播放接口和实现
    │ │ │ ├── Player.cpp # Native层播放功能调用逻辑的实现
    │ │ │ ├── Player.h # Native层播放功能调用逻辑的接口
    │ │ │ ├── PlayerNative.cpp # Native层播放的入口
    │ │ │ └── PlayerNative.h # Native层暴露上来的接口
    │ ├── types #
    │ │ └── libplayer # 播放模块暴露给UI层的接口
    │ └── CMakeLists.txt # 编译入口
    ├── ets # UI层
    │ ├── common # 公共模块
    │ │ ├── utils # 共用的工具类
    │ │ │ ├── DateTimeUtils.ets # 获取当前时间
    │ │ │ └── Logger.ts # 日志工具
    │ | └───CommonConstants.ets # 参数常量
    │ ├── entryability # 应用的入口
    │ │ └── EntryAbility.ts # 申请权限弹窗实现
    │ ├── pages # EntryAbility包含的页面
    │ │ └── Index.ets # 首页/播放页面
    ├── resources # 用于存放应用所用到的资源文件
    │ ├── base # 该目录下的资源文件会被赋予唯一的ID
    │ │ ├── element # 用于存放字体和颜色
    │ │ ├── media # 用于存放图片
    │ │ └── profile # 应用入口首页
    │ ├── en_US # 设备语言是美式英文时,优先匹配此目录下资源
    │ └── zh_CN # 设备语言是简体中文时,优先匹配此目录下资源
    └── module.json5 # 模块配置信息
  2. 编译新建工程并运行。