HarmonyOS技术社区 · 2021年02月20日

【软通动力】SurfaceProvider实现视频播放Demo-热乎乎的拜年视频

目录:

一.创建布局

二、在MainAbilitySlice实现播放功能的代码

三、在MainAbilitySlice的其他生命周对player进行资源管理

四、实现效果视频展示

辛丑年即将到来,基于HOS,利用SurfaceProvider控件实现了一个视频播放的Demo,有开始、暂停和循环播放功能,邀请大家一起来看看。提前祝51CTO的广大开发爱好者新春快乐!牛气冲天!
奉上视频一睹为快。

具体实现步骤如下:
一.创建布局:

`<?xml version="1.0" encoding="utf-8"?>
 <StackLayout
       xmlns:ohos="[http://schemas.huawei.com/res/ohos](http://schemas.huawei.com/res/ohos)"
       ohos:width="match_parent"
       ohos:height="match_parent">
      <ohos.agp.components.surfaceprovider.SurfaceProvider
        ohos:id="$+id:surfaceProvider"
        ohos:width="match_parent"
        ohos:height="match_parent"/>
       <Image
           ohos:id="$+id:img"
           ohos:height="match_content"
           ohos:width="match_content"
           ohos:image_src="$media:player"
           ohos:layout_alignment="center"
           ohos:visibility="hide" data-tomark-pass />
   </StackLayout>
`

二、在MainAbilitySlice实现播放功能的代码

`onStart()中添加:
//设置沉浸式状态栏
getWindow().addFlags(WindowManager.LayoutConfig.MARK_TRANSLUCENT_STATUS);
initPlayer();
需要重写两个回调:VideoSurfaceCallback 、VideoPlayerCallback
private void initPlayer() {
    sfProvider=(SurfaceProvider) findComponentById(ResourceTable.Id_surfaceProvider);
    image=(Image) findComponentById(ResourceTable.Id_img);
    sfProvider.getSurfaceOps().get().addCallback(new VideoSurfaceCallback());
    //sfProvider.pinToZTop(boolean)--如果设置为true, 视频控件会在最上次展示,但是设置为false时,虽然不在最上层展示,却出现黑屏,
    // 需加上一行代码:WindowManager.getInstance().getTopWindow().get().setTransparent(true);
    sfProvider.pinToZTop(false);
    WindowManager.getInstance().getTopWindow().get().setTransparent(true);
    player=new Player(getContext());
//sfProvider添加监听事件
    sfProvider.setClickedListener(new Component.ClickedListener() {
        @Override
        public void onClick(Component component) {
            if(player.isNowPlaying()){
                //如果正在播放,就暂停
                player.pause();
                //播放按钮可见
                image.setVisibility(Component.VISIBLE);
            }else {
                //如果暂停,点击继续播放
                player.play();
                //播放按钮隐藏
                image.setVisibility(Component.HIDE);
            }
        }
    });
}
private void playLocalFile(Surface surface) {
    try {
        RawFileDescriptor filDescriptor = getResourceManager().getRawFileEntry("resources/rawfile/123.mp4").openRawFileDescriptor();
        Source source = new Source(filDescriptor.getFileDescriptor(),filDescriptor.getStartPosition(),filDescriptor.getFileSize());
        player.setSource(source);
        player.setVideoSurface(surface);
        player.setPlayerCallback(new VideoPlayerCallback());
        player.prepare();
        sfProvider.setTop(0);
        player.play();
    } catch (Exception e) {
        HiLog.info(logLabel,"playUrl Exception:" + e.getMessage());
    }
}
private class VideoSurfaceCallback implements SurfaceOps.Callback {
    @Override
    public void surfaceCreated(SurfaceOps surfaceOps) {
        HiLog.info(logLabel,"surfaceCreated() called.");
        if (sfProvider.getSurfaceOps().isPresent()) {
            Surface surface = sfProvider.getSurfaceOps().get().getSurface();
            playLocalFile(surface);
        }
    }
    @Override
    public void surfaceChanged(SurfaceOps surfaceOps, int i, int i1, int i2) {
        HiLog.info(logLabel,"surfaceChanged() called.");
    }
    @Override
    public void surfaceDestroyed(SurfaceOps surfaceOps) {
        HiLog.info(logLabel,"surfaceDestroyed() called.");
    }
}
private class VideoPlayerCallback implements Player.IPlayerCallback {
    @Override
    public void onPrepared() {
        HiLog.info(logLabel,"onPrepared");
    }
    @Override
    public void onMessage(int i, int i1) {
        HiLog.info(logLabel,"onMessage");
    }
    @Override
    public void onError(int i, int i1) {
        HiLog.info(logLabel,"onError: i=" + i + ", i1=" + i1);
    }
    @Override
    public void onResolutionChanged(int i, int i1) {
        HiLog.info(logLabel,"onResolutionChanged");
    }
    @Override
    public void onPlayBackComplete() {
        //播放完成回调,重新播放
        if (player != null) {
            player.prepare();
            player.play();
        }
    }
    @Override
    public void onRewindToComplete() {
        HiLog.info(logLabel,"onRewindToComplete");
    }
    @Override
    public void onBufferingChange(int i) {
        HiLog.info(logLabel,"onBufferingChange");
    }
    @Override
    public void onNewTimedMetaData(Player.MediaTimedMetaData mediaTimedMetaData) {
        HiLog.info(logLabel,"onNewTimedMetaData");
    }
    @Override
    public void onMediaTimeIncontinuity(Player.MediaTimeInfo mediaTimeInfo) {
        HiLog.info(logLabel,"onMediaTimeIncontinuity");
    }
}`

三、在MainAbilitySlice的其他生命周对player进行资源管理:

`@Override
public void onActive() {
    super.onActive();
    player.play();
}
@Override
protected void onBackground() {
    super.onBackground();
    player.pause();
}
@Override
protected void onStop() {
    super.onStop();
    player.stop();
    player.release();
}`

四、实现效果:详细请查看视频。harmonyos.51cto.com/show/3119【软通动力】SurfaceProvider实现视频播放Demo-热乎乎的拜年视频

欢迎交流:HWIS-HOS@isoftstone.com

作者:软通赵海鹏

想了解更多内容,请访问: 51CTO和华为官方战略合作共建的鸿蒙技术社区https://harmonyos.51cto.com




21_9.jpg

推荐阅读
关注数
3010
内容数
446
华为鸿蒙相关技术,活动及资讯,欢迎关注及加入创作
目录
极术微信服务号
关注极术微信号
实时接收点赞提醒和评论通知
安谋科技学堂公众号
关注安谋科技学堂
实时获取安谋科技及 Arm 教学资源
安谋科技招聘公众号
关注安谋科技招聘
实时获取安谋科技中国职位信息