VideoPlayer播放组件详解-Unity使用官方组件播放视频

本文介绍了如何在Unity中创建一个基本的视频播放控制器,包括设置场景、创建UI元素、编写控制逻辑,并展示了如何处理拖拽事件来调整视频帧。作者建议使用AVPRO插件,因为它在WebGL支持和稳定性方面表现更优,并分享了遇到的官方API在WebGL下播放问题的解决方案。
摘要由CSDN通过智能技术生成

本次Demo演示工具介绍:Unity版本:2020.3.25,Visual Studio版本: 2020;
1:使用Unity新建一个工程,新建空场景并保存,然后在Project面板下新建几个文件夹:Res,Resources,Scence,Scripts,此三个文件夹放一个根目录VideoPlayerDemo,目录如下;这是我的习惯,大家随意;
在这里插入图片描述
2:Res文件夹下创建一个Render Texture,Hierarchy面板下创建一个画布,再创建RawImage,相机设置为纯色渲染,Projection设置为Orthographic模式,画布设置ScreenSpace_Camera模式,并将主相机赋值给画布的渲染相机,RawImage设置为随画布拉伸的方式,锚点的右下角那个点一下;在这里插入图片描述
3:新建一个面板Controller,将再该面板中编写相关播放控制功能;Slider-进度条,Dropdown-播放速度,两个Text文本设置播放时间,还有个播放按钮。
在这里插入图片描述
4:新建类VideoController挂载到控制面板上;

    /// <summary>
    /// 绑定拖拽功能
    /// </summary>
    /// <param name="obj"></param>
    /// <param name="eventID"></param>
    /// <param name="action"></param>
    public static void AddTriggersListener(GameObject obj, EventTriggerType eventID, UnityAction<BaseEventData> action)
    {
        EventTrigger trigger = obj.GetComponent<EventTrigger>();
        if (trigger == null)
        {
            trigger = obj.AddComponent<EventTrigger>();
        }

        if (trigger.triggers.Count == 0)
        {
            trigger.triggers = new List<EventTrigger.Entry>();
        }

        UnityAction<BaseEventData> callback = new UnityAction<BaseEventData>(action);
        EventTrigger.Entry entry = new EventTrigger.Entry();
        entry.eventID = eventID;
        entry.callback.AddListener(callback);
        trigger.triggers.Add(entry);
    }
   public class VideoController : MonoBehaviour
    {
        private Slider slider;
        private Text totleTimer;
        private Text culTimer;
        private Dropdown speedValue;
        private Button playbtn;
        private Sprite PlaySP;
        private Sprite PuaseSP;
        bool isPause = false;
        private VideoPlayer video;

        private void Awake()
        {
            slider =Config<Slider>.GetComponent(transform, "ProsseSlider");
            speedValue = Config<Dropdown>.GetComponent(transform, "SpeedDropdown");
            totleTimer = Config<Text>.GetComponent(transform, "TotleTimer");
            culTimer = Config<Text>.GetComponent(transform, "CulTimer");
            playbtn = Config<Button>.GetComponent(transform, "PlayeBtn");
            PlaySP= Resources.Load<Sprite>("UITexture/播放");
            PuaseSP = Resources.Load<Sprite>("UITexture/暂停");
            video = Config<VideoPlayer>.GetComponent(transform.parent, "VideoPlayer");
            speedValue.onValueChanged.AddListener(DropdownOnClick);
        }

        private void DropdownOnClick(int arg0)
        {
            var value =float.Parse( speedValue.options[arg0].text.Replace("倍速 x ", ""));

            video.playbackSpeed = value;
        }
       
        // Start is called before the first frame update
        void Start()
        { 
            playbtn.GetComponent<Image>().sprite = PuaseSP;
            //视频播放结束事件
            video.loopPointReached += EndWithVideoPlay;
            //播放按钮监听
            playbtn.onClick.AddListener(PlayBtnOnClick);
            //添加Slider拖拽事件
            Utils.AddTriggersListener(slider.gameObject, EventTriggerType.Drag, OnDrag);
        }

        private void OnDrag(BaseEventData arg0)
        { 
            //踩坑记: 强制转换  long 的时候先将算法算出来之后得到float类型,然后再将结果强转,直接算法强转结果一直为0;
            float tatol = slider.value * video.frameCount;
            video.frame = (long)tatol;
        }
        /// <summary>
        /// 清除上一帧渲染
        /// </summary>
        private void RemoveTargetframe()
        {
          video.targetTexture.Release();
          video.targetTexture.MarkRestoreExpected(); 
        }
        /// <summary>
        /// 显示当前视频的时间
        /// </summary>
        private void ShowVideoTime() 
        { 
            try
            {
                if (video != null && video.isPlaying)
                {
                    //视频帧率  每秒多少帧;
                    float frameRate = video.frameRate;
                    int clipHour = (int)((float)video.frameCount / frameRate / (float)3600);
                    int clipMinute = (int)((float)video.frameCount / frameRate - (float)clipHour * (float)3600) / 60;
                    int clipSecond = (int)((float)video.frameCount / frameRate - (float)clipHour * (float)3600 - (float)clipMinute * (float)60);
                    // 当前的视频播放时间
                    int currentHour = (int)video.time / 3600;
                    int currentMinute = (int)(video.time - currentHour * 3600) / 60;
                    int currentSecond = (int)(video.time - currentHour * 3600 - currentMinute * 60);

                    string culTime = string.Format("{0:D2}:{1:D2}:{2:D2}", currentHour, currentMinute, currentSecond);
                    string totalTime = string.Format("{0:D2}:{1:D2}:{2:D2}", clipHour, clipMinute, clipSecond);
                    culTimer.text = culTime;
                    totleTimer.text = totalTime;
                    slider.value = (float)video.frame / (float)video.frameCount;
                }
            }
            catch (Exception e)
            {
                print(e.Message);
            }

        }
        private void PlayBtnOnClick()
        {
            isPause = isPause ? false : true;
            playbtn.GetComponent<Image>().sprite = isPause ? PlaySP : PuaseSP;
            if (isPause)
                video.Pause();
            else
                video.Play();
        }

        private void EndWithVideoPlay(VideoPlayer source)
        {
            print("视频播放结束");
        }

        // Update is called once per frame
        void Update()
        {
            ShowVideoTime();
        }
    }

虽然该有的功能都有,但还是建议大家使用AVPRO插件,个人觉得它比较友好,支持webgl,官方的api我使用在webgl的时候发现老是黑屏,播放不了的情况。AVPRO解决了我的问题。
最后看效果吧:
在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

有点朦

您的鼓励将是我创作的最大动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值