乐于分享
好东西不私藏

uniapp:视频video,记录播放进度

uniapp:视频video,记录播放进度

需求

  • 播放暂停视频
  • 不允许快进,可以后退
  • 视频后退不会影响最高观看时长,例如看了10分钟,后退5分钟,观看时长依然是600秒
  • 监听退出记录观看时间,下次进来接着看
  • 视频看完积分
  • 自定义视频是否有倍速

uniapp原生组件video 没有倍速

<template>    <view>        <!-- id:唯一标识,@timeupdate进度条变化的事件,@ended进度条到最后的事件,initial-time指定视频初始播放位置, -->        <videoid="myVideo"style="width: 100%;" @timeupdate="timeUpdate" @ended="ended":initial-time="initialTime"                :src="course.videos" :poster="course.img">        </video>    </view></template><script>    export default {        data() {            return {                initialTime0//初始播放位置                duration0// 视频时长                videoContext''// 用来存储video对象                watchTime0// 实际观看时间                videoRealTime0// 实时播放进度                course: {                    videos"https://img.cdn.aliyun.dcloud.net.cn/guide/uniapp/%E7%AC%AC1%E8%AE%B2%EF%BC%88uni-app%E4%BA%A7%E5%93%81%E4%BB%8B%E7%BB%8D%EF%BC%89-%20DCloud%E5%AE%98%E6%96%B9%E8%A7%86%E9%A2%91%E6%95%99%E7%A8%8B@20181126-lite.m4v",                    img'https://cdn.uviewui.com/uview/swiper/swiper1.png'                }            };        },        onLoad(options) {            uni.setNavigationBarTitle({                title: options.id            })            // 调用接口取到该用户上次播放的位置(秒)             this.watchTime = 67            this.initialTime = 67        },        onReady() {            // 视频唯一ID             this.videoContext = uni.createVideoContext('myVideo')        },        methods: {            // 监听进度条变化:禁止拖动 e.detail = {currentTime, duration} 。触发频率 250ms 一次            timeUpdate(e) {                //视频时长                this.duration = parseInt(e.detail.duration)                // 记录用户当前视频进度                var jumpTime = parseInt(e.detail.currentTime)                // 判断用户当前视频进度比实际观看时间差别,这里只判断了用户快进                if (jumpTime - this.watchTime > 1) {                    // 差别过大,调用seek方法跳转到实际观看时间                    this.videoContext.seek(this.watchTime)                } else {                    this.videoRealTime = parseInt(e.detail.currentTime)                    if (this.videoRealTime > this.watchTime) {                        this.watchTime = this.videoRealTime                    }                }            },            ended() {                // 用户把进度条拉到最后,但是实际观看时间不够,跳转回去会自动暂停                if (this.watchTime < this.duration) {                    this.videoContext.play()                } else {                    console.log('看完了')                }            },            // 监听返回:监听不了ios的左滑返回,目前的采用的解决方案是在onLoad设置禁用左滑            // onLoad(option) {            //  // 单页禁止测滑返回            //  // #ifdef APP-PLUS            //  let currentWebview = this.$mp.page.$getAppWebview() //获取当前页面的webview对象            //  currentWebview.setStyle({            //      popGesture: 'none'            //  })            //  // #endif            // }            // 监听返回,记录视频的观看时长            onBackPress(e) {                //backbutton 是点击物理按键返回,navigateBack是uniapp中的返回(比如左上角的返回箭头)                console.log('返回', e, this.watchTimethis.duration);            },        },    }</script> 
使用插件 x-video 视频播放:https://ext.dcloud.net.cn/plugin?id=10979

修改组件 实现不能往前拉的功能

<template>    <divclass="video-wrap":style="{ width: `${width}`, height: `${height}` }">        <!-- video player -->        <view @click="handleControls">            <videoclass="video-player":id="videoId":style="{ width: `${width}`, height: `${height}` }"                webkit-playsinline="true" playsinline="true" x-webkit-airplay="allow" x5-video-player-type="h5-page"                x5-video-orientation="portrait" :src="src" :initial-time="initialTime" :controls="isFullScreen"                :show-center-play-btn="false" :autoplay="autoplay" :muted="isMute" :poster="poster" @play="videoPlay"                @pause="videoPause" @ended="videoEnded" @timeupdate="videoTimeUp" @loadedmetadata="videoLoaded"                @seeked="videoSeeked" @seeking="videoSeeking" @waiting="videoWaiting" @error="videoError"                @fullscreenchange="onFullScreen"></video>        </view>        <viewclass="abs-center">            <!-- 中心播放按钮 -->            <imagesrc="../../static/play-btn.png"mode=""class="play-btn"v-if="!isVideoPlay && !showLoading"                @click="videoPlayCenter"></image>            <!--  加载中 -->            <divclass="video-loading"v-if="showLoading">                <imagesrc="../../static/loading.png"mode=""class="loading-btn"></image>            </div>        </view>        <!-- 控制条 -->        <view:class="['controls-bar', controls ? 'show' : 'hide']">            <!-- 播放 -->            <viewclass="play-box" @click="videoPlayClick">                <imagesrc="../../static/pause.png"mode=""class="play-icon"v-if="isVideoPlay"></image>                <imagesrc="../../static/play.png"mode=""class="play-icon"v-else></image>            </view>            <!-- 声音 -->            <viewclass="mute-box" @click="videoMuteClick">                <imagesrc="../../static/sound.png"mode=""class="mute-icon"v-if="!isMute"></image>                <imagesrc="../../static/mute.png"mode=""class="mute-icon"v-else></image>            </view>            <!-- 进度 -->            <viewclass="progress">                <viewclass="currtime">{{ currentTimeStr }}</view>                <viewclass="slider-container">                    <slider @change="sliderChange" @changing="sliderChanging":step="step":value="sliderValue"                        backgroundColor="#9f9587" activeColor="#d6d2cc" block-color="#FFFFFF" block-size="12" />                </view>                <viewclass="druationTime">{{ druationTimeStr }}</view>            </view>            <!-- 倍速 -->            <viewclass="play-rate" @click="videoPlayRate"v-if="showRate">{{ playbackRate }}x</view>            <!-- 全屏 -->            <viewclass="play-full" @click="videoFull">                <imagesrc="../../static/fullscreen.png"mode=""class="play-icon" @click="videoFull"></image>            </view>            <!-- 倍速菜单 -->            <ulclass="play-rate-menu":style="{ height: height }"v-if="showRateMenu">                <liv-for="item in playbackRates":key="item":class="[{ activeRate: playbackRate === item }, 'play-rate-item']"                    @click="changePlayRate(item)">                    {{ item }}x                </li>            </ul>        </view>    </div></template><script>    export default {        name'XVideo',        props: {            // 视频地址            videoId: {                typeString,                default'myVideo'            },            // 视频地址            src: {                typeString            },            // 自动播放            autoplay: {                typeBoolean,                defaulttrue            },            // 封面            poster: {                typeString            },            // 步长,表示占比,取值必须大于0且为整数            step: {                typeNumber,                default1            },            // 初始播放进度,表示占比            progress: {                typeNumber            },            // 视频宽度            width: {                typeString,                default'100%'            },            // 视频高度            height: {                typeString,                default'484rpx'            },            // 播放错误提示            errorTip: {                typeString,                default'播放错误'            },            // 是否展示倍速            showRate: {                typeBoolean,                defaulttrue            },            // 播放速率            playbackRates: {                typeArray,                default() => [0.50.811.251.52]            }        },        data() {            return {                controlsfalse//显示播放控件                isVideoPlayfalse// 是否正在播放                isMutefalse// 是否静音                isVideoEndfalse// 是否播放结束                showPostertrue// 是否显示视屏封面                showLoadingfalse// 加载中                durationTime0//总播放时间 时间戳                currentTime0//当前播放时间 时间戳                watchTime0//实际最高播放时间 时间戳                druationTimeStr'00:00'//总播放时间 字符串 计算后                currentTimeStr'00:00'//当前播放时间 字符串 计算后                sliderValue0//进度条的值 百分比                isSeekedtrue//防止进度条拖拽失效                playbackRate1// 初始播放速率                showRateMenufalse//显示播放速率                initialTime0//初始播放时间                isFullScreenfalse//是否全屏                windowWidth0 //屏幕宽度            }        },        mounted() {            this.videoPlayer = uni.createVideoContext(this.videoIdthis)            // #ifdef H5            // 处理微信不能自动播放            if (this.autoplay) {                document.addEventListener(                    'WeixinJSBridgeReady',                    () => {                        this.videoPlayer.play()                        this.isVideoPlay = true                    },                    false                )            }            // #endif        },        onHide() {            clearTimeout(this.timer)        },        methods: {            // 自动隐藏控制条            hideControls() {                this.timer = setTimeout(() => {                    this.controls = false                }, 5000)            },            // 点击显示/隐藏控制条            handleControls() {                this.controls = !this.controls            },            // 根据秒获取时间            formatSeconds(second) {                second = Math.round(second)                var hh = parseInt(second / 3600)                var mm = parseInt((second - hh * 3600) / 60)                if (mm < 10) mm = '0' + mm                var ss = parseInt((second - hh * 3600) % 60)                if (ss < 10) ss = '0' + ss                if (hh < 10) hh = hh == 0 ? '' : `0${hh}:`                var length = hh + mm + ':' + ss                if (second > 0) {                    return length                } else {                    return '00:00'                }            },            // 缓冲            videoWaiting(e) {                // 没有缓冲结束事件,所以在不播放的情况触发loading                if (!this.isVideoPlaythis.showLoading = true            },            // 视频信息加载完成            videoLoaded(e) {                // console.log(e.detail.duration, this.progress)                this.durationTime = e.detail.duration                this.druationTimeStr = this.formatSeconds(this.durationTime)                this.initialTime = this.progress                this.watchTime = this.progress                this.currentTime = this.progress                this.sliderValue = this.progress * 100 / this.durationTime                this.videoPlayer.seek(this.initialTime)                this.currentTimeStr = this.formatSeconds(this.initialTime)                this.controls = true                this.showLoading = false                this.$emit('loadeddata'this.durationTimethis.videoPlayer)            },            // 播放进度更新,触发频率 250ms 一次            // videoTimeUp(e) {            //  console.log(this.initialTime,this.currentTime,this.watchTime)            //  // 记录用户当前视频进度            //  var jumpTime = parseInt(e.detail.currentTime)            //  // 判断用户当前视频进度比实际观看时间差别,这里只判断了用户快进            //  if (jumpTime - this.currentTime > 1) {            //      // 差别过大,调用seek方法跳转到实际观看时间            //      this.videoPlayer.seek(this.currentTime)            //      this.currentTimeStr = this.formatSeconds(this.currentTime)            //  } else if (jumpTime - this.currentTime < -1) {            //      let sliderValue = Math.round((e.detail.currentTime / this.durationTime) * 100)            //      if (this.isSeeked) {            //          //判断拖拽完成后才触发更新,避免拖拽失效            //          if (sliderValue % this.step === 0)            //              // 比例值能被步进值整除            //              this.sliderValue = sliderValue            //      }            //      this.currentTimeStr = this.formatSeconds(e.detail.currentTime)            //      this.$emit('timeupdate', e)            //  } else {            //      let sliderValue = Math.round((e.detail.currentTime / this.durationTime) * 100)            //      if (this.isSeeked) {            //          //判断拖拽完成后才触发更新,避免拖拽失效            //          if (sliderValue % this.step === 0)            //              // 比例值能被步进值整除            //              this.sliderValue = sliderValue            //      }            //      this.currentTimeStr = this.formatSeconds(e.detail.currentTime)            //      this.currentTime = e.detail.currentTime            //      this.watchTime = e.detail.currentTime            //      this.$emit('timeupdate', e)            //  }            // },            videoTimeUp(e) {                // console.log(this.initialTime, this.currentTime, this.watchTime)                // 记录用户当前视频进度                var jumpTime = e.detail.currentTime                // 判断用户当前视频进度比实际观看时间差别,这里只判断了用户快进                if (jumpTime - this.watchTime > 1) {                    // 差别过大,调用seek方法跳转到实际观看时间                    this.videoPlayer.seek(this.watchTime)                    // this.currentTimeStr = this.formatSeconds(this.currentTime)                } else {                    this.currentTime = e.detail.currentTime                    let sliderValue = Math.round((e.detail.currentTime / this.durationTime) * 100)                    if (this.isSeeked) {                        //判断拖拽完成后才触发更新,避免拖拽失效                        if (sliderValue % this.step === 0)                            // 比例值能被步进值整除                            this.sliderValue = sliderValue                    }                    this.currentTimeStr = this.formatSeconds(e.detail.currentTime)                    if (this.currentTime > this.watchTime) {                        this.watchTime = this.currentTime                    }                    this.$emit('timeupdate'this.watchTime)                }            },            //正在拖动slider            sliderChanging(e) {                console.log(2, e.detail.value)                this.isSeeked = false // 拖拽过程中,不允许更新进度条                this.showLoading = true                this.videoPlayer.pause()                this.$emit('seeking')            },            // 拖动slider完成后            sliderChange(e) {                console.log(1, e.detail.valuethis.durationTime, (e.detail.value / 100) * this.durationTime)                this.sliderValue = e.detail.value                let currentTime = (this.sliderValue / 100) * this.durationTime                this.showLoading = false                this.isSeeked = true // 完成拖动后允许更新滚动条                this.videoPlayer.seek(currentTime)                if (this.sliderValue < 100) {                    this.videoPlayer.play()                } else {                    this.videoPlayer.pause()                    this.videoEnded()                }                this.hideControls()                this.$emit('seeked'this.sliderValue)            },            // 点击中心播放            videoPlayCenter() {                this.videoPlayer.play()                this.$emit('play')            },            // 点击左下角播放/暂停,会触发原始播放/暂停事件,分开写,防止重复触发            videoPlayClick() {                if (this.isVideoPlay) {                    this.videoPlayer.pause()                } else {                    this.videoPlayer.play()                    this.$emit('play')                }            },            // 原始播放            videoPlay() {                if (this.pauseTimer) {                    clearTimeout(this.pauseTimer)                }                this.isVideoPlay = true                this.isVideoEnd = false                this.showLoading = false                this.hideControls()            },            // 原始暂停            videoPause() {                // 处理播放结束和拖动会先触发暂停的问题                this.pauseTimer = setTimeout(() => {                    if (this.isVideoEndreturn                    if (!this.isSeekedreturn                    this.isVideoPlay = false                    this.$emit('pause')                }, 100)            },            // 静音            videoMuteClick() {                this.isMute = !this.isMute            },            // 播放结束            videoEnded() {                // 重置状态                this.isVideoPlay = false                this.showPoster = true                this.isVideoEnd = true                this.$emit('ended')            },            // 播放错误            videoError(e) {                // uni.showToast({                //  title: this.errorTip,                //  icon: 'none'                // })                this.$emit('error')            },            // 显示倍速            videoPlayRate() {                this.showRateMenu = true            },            // 点击倍速            changePlayRate(rate) {                this.playbackRate = rate                this.videoPlayer.playbackRate(rate)                this.showRateMenu = false                this.hideControls()            },            // 创建倍速按钮            createPlayRateDOM() {                const playRateDom = document.createElement('div')                playRateDom.className = 'full-play-rate'                playRateDom.innerText = `${this.playbackRate}x`                playRateDom.onclick = () => {                    const playRateMenuDom = document.querySelector('.full-play-rate-menu')                    playRateMenuDom.style.display = 'block'                }                return playRateDom            },            // 创建倍速菜单            createPlayRateMenuDom() {                const playRateMenuDom = document.createElement('ul')                playRateMenuDom.className = `play-rate-menu full-play-rate-menu`                playRateMenuDom.style.height = this.windowWidth + 'px'                playRateMenuDom.style.display = 'none'                let liStr = ''                this.playbackRates.forEach((item) => {                    liStr += `                    <li class="${this.playbackRate === item ? 'activeRate' : ''} play-rate-item full-play-rate-item">                        ${item}x                    </li>                     `                })                playRateMenuDom.innerHTML = liStr                return playRateMenuDom            },            // 处理全屏倍速功能            videoFullRate() {                this.windowWidth = uni.getSystemInfoSync().windowWidth                const fullScreen = document.querySelector('.uni-video-fullscreen')                // 添加倍速菜单                const playRateMenuDom = document.querySelector('.full-play-rate-menu')                const videoContainer = document.querySelector('.uni-video-container')                if (!playRateMenuDom) {                    videoContainer.appendChild(this.createPlayRateMenuDom())                    const lis = document.querySelectorAll('.full-play-rate-item')                    lis.forEach((item) => {                        item.style.lineHeight = this.windowWidth / this.playbackRates.length + 'px'                        item.onclick = () => {                            let rate = +item.innerText.slice(0, -1)                            this.playbackRate = rate                            this.videoPlayer.playbackRate(rate)                            lis.forEach((li) => {                                li.classList.remove('activeRate')                                let rate = +li.innerText.slice(0, -1)                                if (this.playbackRate === rate) {                                    li.classList.add('activeRate')                                }                            })                            const playRateMenuDom = document.querySelector('.full-play-rate-menu')                            playRateMenuDom.style.display = 'none'                            const playRateDom = document.querySelector('.full-play-rate')                            playRateDom.innerText = `${this.playbackRate}x`                        }                    })                }                // 添加倍速按钮                const playRateDom = document.querySelector('.full-play-rate')                if (!playRateDom) {                    fullScreen.parentNode.insertBefore(this.createPlayRateDOM(), fullScreen)                }            },            // 点击全屏            videoFull() {                this.videoPlayer.requestFullScreen()                // #ifdef H5                if (this.showRate) {                    this.videoFullRate()                }                // #endif            },            // 监听原生全屏事件            onFullScreen({                detail            }) {                if (detail.fullScreen) {                    this.isFullScreen = true                } else {                    this.isFullScreen = false                    // #ifdef H5                    const playRateMenuDom = document.querySelector('.full-play-rate-menu')                    playRateMenuDom.style.display = 'none'                    // #endif                }            }        }    }</script><stylelang="scss"scoped>    /deep/ .uni-video-fullscreen {        margin-left30px    }    .show {        opacity1 !important;    }    .hide {        opacity0 !important;        pointer-events: none;    }    .abs-center {        position: absolute;        top50%;        left50%;        transformtranslate(-50%, -50%);    }    ::v-deep .full-play-rate {        color#cbcbcb;        font-size32rpx;        margin-left24rpx;        margin-right24rpx;    }    ::v-deep .full-play-rate-menu {        position: fixed !important;    }    ::v-deep .play-rate-menu {        list-style-type: none;        background-colorrgba(0000.7);        width24%;        position: absolute;        right0;        bottom0;        padding-left0;        box-sizing: border-box;    }    ::v-deep .play-rate-item {        line-height70rpx;        font-size28rpx;        text-align: center;        color#fff;    }    ::v-deep .activeRate {        color#5785e3;    }    .video-wrap {        position: relative;        .play-btn {            width120rpx;            height120rpx;        }        @keyframes run {            from {                transformrotate(0deg);            }            to {                transformrotate(360deg);            }        }        .loading-btn {            width120rpx;            height120rpx;            animation: run 0.8s linear 0s infinite;        }        .controls-bar {            width100%;            padding1% 1% 1% 0;            position: absolute;            bottom0;            left0;            right0;            z-index99;            display: flex;            align-items: center;            backgroundrgba(5957570.7);            color#fff;            opacity1;            transition: opacity 1s;            height84rpx;            .play-box,            .mute-box,            .play-full {                width84rpx;                height100%;                display: flex;                align-items: center;                justify-content: center;            }            .mute-icon {                width40rpx;                height40rpx;            }            .play-icon {                width34rpx;                height34rpx;            }            .progress {                display: flex;                align-items: center;                flex1;                font-size24rpx;                margin-left16rpx;                .slider-container {                    flex1;                    max-width58%;                }                .currtime {                    color#ffffff;                    width11%;                    height100%;                    text-align: center;                    margin-right20rpx;                }                .druationTime {                    color#ffffff;                    width12%;                    height100%;                    text-align: center;                }            }            .play-rate {                font-size32rpx;                margin-right24rpx;            }            .play-rate-menu {                padding-top26rpx;            }            .play-rate-item:first-child {                margin-top30rpx;            }        }    }</style>