123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507 |
- <template>
- <div
- ref="custom-video_container"
- class="custom-video_container"
- @mouseover="handleControls($event, 'start')"
- @mouseleave="handleControls($event, 'end')"
- >
- <video
- ref="custom-video"
- :poster="videoOption.poster"
- class="custom-video_video"
- >
- <source :src="videoSrc" type="video/mp4">
- <p>设备不支持</p>
- </video>
- <span
- v-if="videoState.play"
- ref="videoPlay"
- class="custom-video_play custom-video_play-pause iconfont icon-zanting"
- @click="pause('btn')"
- />
- <span
- v-else
- class="custom-video_play custom-video_play-play iconfont icon-bofang"
- @click="play('btn')"
- />
- <!-- 控制区域背景 -->
- <transition
- name="fade"
- >
- <div
- v-show="!videoState.hideControl || !videoState.play"
- class="custom-video_control"
- >
- <!-- 进度条 -->
- <div
- class="custom-video_control-bg"
- @mousedown="handlePrograssDown"
- @mousemove="handlePrograssMove"
- @mouseup="handlePrograssUp"
- >
- <div
- ref="custom-video_control-bg-outside"
- class="custom-video_control-bg-outside"
- >
- <span
- ref="custom-video_control-bg-inside"
- class="custom-video_control-bg-inside"
- />
- <span
- ref="custom-video_control-bg-inside-point"
- class="custom-video_control-bg-inside-point"
- />
- </div>
- </div>
- <!-- 声音 -->
- <div
- class="custom-video_control-voice"
- >
- <span
- class="custom-video_control-voice-play iconfont icon-shengyin"
- />
- <div
- ref="custom-video_control-voice-bg"
- class="custom-video_control-voice-bg"
- @mousedown="handleVolPrograssDown"
- @mousemove="handleVolPrograssMove"
- @mouseup="handleVolPrograssUp"
- >
- <div
- ref="custom-video_control-voice-bg-outside"
- class="custom-video_control-voice-bg-outside"
- >
- <span
- ref="custom-video_control-voice-bg-inside"
- class="custom-video_control-voice-bg-inside"
- />
- <span
- ref="custom-video_control-voice-bg-point"
- class="custom-video_control-voice-bg-point"
- />
- </div>
- </div>
- </div>
- <!-- 时间 -->
- <div
- class="custom-video_control-time"
- >
- <span>{{ currentTime ? currentTime : "00:00" }}</span>
- /
- <span>{{ duration ? duration : "00:00" }}</span>
- </div>
- <!-- 全屏缩放 -->
- <span
- class="custom-video_control-full iconfont icon-quanping"
- @click="handleScreen"
- />
- </div>
- </transition>
- </div>
- </template>
- <script>
- export default {
- name: 'Video',
- props: {
- videoSrc: {
- type: '',
- default: ''
- },
- videoName: {
- type: String,
- default: ''
- }
- },
- data() {
- return {
- videoOption: {
- poster: '', // 初始化占位图片
- volume: 20 // 初始化声音
- },
- videoState: {
- play: false, // 播放状态
- hideControl: false, // 控制栏状态
- distance: 0, // 移动的距离
- downState: false, // 鼠标点击进度条
- playState: false,
- leftInit: 0, // 当前进度初始偏移量
- screenState: false
- },
- voiceState: { // 同上
- distance: 0,
- downState: false,
- topInit: 0
- },
- videoDom: null, // video
- videoProOut: null, // 视频总进度条
- videoPro: null, // 视频进度条
- videoPoi: null, // 视频进度点
- duration: 0, // 视频总时长
- currentTime: 0, // 视频当前播放时长
- processWidth: 0, // 视频进度条总长度
- voiceProOut: null, // 音频总进度条
- voicePro: null, // 音频进度条
- voicePoi: null, // 音频进度点
- volProcessHeight: 0
- }
- },
- mounted() {
- // 初始化相关元数据
- this.videoDom = this.$refs['custom-video']
- this.videoProOut = this.$refs['custom-video_control-bg-outside']
- this.videoPro = this.$refs['custom-video_control-bg-inside']
- this.videoPoi = this.$refs['custom-video_control-bg-inside-point']
- this.voiceProOut = this.$refs['custom-video_control-voice-bg-outside']
- this.voicePro = this.$refs['custom-video_control-voice-bg-inside']
- this.voicePoi = this.$refs['custom-video_control-voice-bg-point']
- this.processWidth = this.videoProOut.clientWidth
- this.videoState.leftInit = this.getOffset(this.videoProOut).left
- this.videoDom.volume = this.videoOption.volume / 100 // 设置初始化声音
- this.initMedaData()
- this.play('btn')
- },
- methods: {
- initMedaData() { // 初始化video相关事件
- this.videoDom.addEventListener('loadedmetadata', () => { // 获取视频总时长
- this.duration = this.timeTranslate(this.videoDom.duration)
- })
- this.videoDom.addEventListener('click', () => { // 点击视频区域可以进行播放或者暂停
- this.$emit('myFull', this.videoName)
- // 点击视频区域改为可以放大缩小
- // if (this.videoDom.paused || this.videoDom.ended) {
- // if (this.videoDom.ended) {
- // this.videoDom.currentTime = 0
- // }
- // this.play('btn')
- // } else {
- // this.pause('btn')
- // }
- })
- this.videoDom.addEventListener('timeupdate', () => { // 监听视频播放过程中的时间
- const percentage = 100 * this.videoDom.currentTime / this.videoDom.duration
- this.videoPro.style.width = percentage + '%'
- this.videoPoi.style.left = percentage - 1 + '%'
- this.currentTime = this.timeTranslate(this.videoDom.currentTime)
- })
- this.videoDom.addEventListener('ended', () => { // 监听结束播放事件
- this.videoPro.style.width = 0
- this.videoPoi.style.left = 0
- this.currentTime = 0
- this.videoState.play = false
- this.videoState.hideControl = false
- })
- this.videoDom.addEventListener('volumechange', () => {
- const percentage = this.videoDom.volume * 100
- this.voicePro.style.height = percentage + '%'
- this.voicePoi.style.bottom = percentage + '%'
- })
- },
- play(flag) { // 播放按钮事件
- if (flag) this.videoState.playState = true
- this.videoState.play = true
- console.log(this.videoDom)
- this.videoDom.play()
- },
- pause(flag) { // 暂停按钮事件
- if (flag) this.videoState.playState = false
- this.videoDom.pause()
- this.videoState.play = false
- },
- handlePrograssDown(ev) { // 监听点击进度条事件,方便获取初始点击的位置
- // 视频暂停
- this.videoState.downState = true // 按下鼠标标志
- this.pause()
- this.videoState.distance = ev.clientX - this.videoState.leftInit
- },
- handlePrograssMove(ev) { // 监听移动进度条事件,同步播放相关事件
- if (!this.videoState.downState) return
- let disX = ev.clientX - this.videoState.leftInit
- if (disX > this.processWidth) {
- disX = this.processWidth
- }
- if (disX < 0) {
- disX = 0
- }
- this.videoState.distance = disX
- this.videoDom.currentTime = this.videoState.distance / this.processWidth * this.videoDom.duration
- },
- handlePrograssUp() { // 松开鼠标,播放当前进度条视频
- this.videoState.downState = false
- // 视频播放
- this.videoDom.currentTime = this.videoState.distance / this.processWidth * this.videoDom.duration
- this.currentTime = this.timeTranslate(this.videoDom.currentTime)
- if (this.videoState.playState) {
- this.play()
- }
- },
- handleVolPrograssDown(ev) { // 监听声音点击事件
- this.voiceState.topInit = this.getOffset(this.voiceProOut).top
- this.volProcessHeight = this.voiceProOut.clientHeight
- this.voiceState.downState = true // 按下鼠标标志
- this.voiceState.distance = ev.clientY - this.voiceState.topInit
- },
- handleVolPrograssMove(ev) { // 监听声音进度条移动事件
- if (!this.voiceState.downState) return
- let disY = this.voiceState.topInit + this.volProcessHeight - ev.clientY
- if (disY > this.volProcessHeight - 2) {
- disY = this.volProcessHeight - 2
- }
- if (disY < 0) {
- disY = 0
- }
- this.voiceState.distance = disY
- this.videoDom.volume = this.voiceState.distance / this.volProcessHeight
- this.videoOption.volume = Math.round(this.videoDom.volume * 100)
- },
- handleVolPrograssUp() { // 监听声音鼠标离开事件
- this.voiceState.downState = false // 按下鼠标标志
- this.videoDom.volume = this.voiceState.distance / this.volProcessHeight
- this.videoOption.volume = Math.round(this.videoDom.volume * 100)
- },
- handleControls(ev, flag) { // 监听离开或者进入视频区域隐藏或者展示控制栏
- // let yanshi = null
- switch (flag) {
- case 'start':
- // clearTimeout(yanshi)
- this.videoState.hideControl = false
- break
- case 'end':
- setTimeout(() => {
- this.videoState.hideControl = true
- }, 3000)
- break
- default:
- break
- }
- },
- handleScreen() { // 全屏操作
- this.$emit('myFull', this.videoName)
- // 全屏事件暂时注释
- // this.videoState.screenState = !this.videoState.screenState
- // if (this.videoState.screenState) {
- // this.fullScreen()
- // } else {
- // this.exitFullscreen()
- // }
- },
- timeTranslate(t) { // 时间转化
- let m = Math.floor(t / 60)
- m < 10 && (m = '0' + m)
- return m + ':' + (t % 60 / 100).toFixed(2).slice(-2)
- },
- getOffset(node, offset) { // 获取当前屏幕下进度条的左偏移量和又偏移量
- if (!offset) {
- offset = {}
- offset.left = 0
- offset.top = 0
- }
- if (node === document.body || node === null) {
- return offset
- }
- offset.top += node.offsetTop
- offset.left += node.offsetLeft
- return this.getOffset(node.offsetParent, offset)
- },
- fullScreen() {
- const ele = document.documentElement
- if (ele.requestFullscreen) {
- ele.requestFullscreen()
- } else if (ele.mozRequestFullScreen) {
- ele.mozRequestFullScreen()
- } else if (ele.webkitRequestFullScreen) {
- ele.webkitRequestFullScreen()
- }
- this.$refs['custom-video_container'].style.width = '100%'
- this.$refs['custom-video_container'].style.height = '100%'
- },
- exitFullscreen() {
- const de = document
- if (de.exitFullscreen) {
- de.exitFullscreen()
- } else if (de.mozCancelFullScreen) {
- de.mozCancelFullScreen()
- } else if (de.webkitCancelFullScreen) {
- de.webkitCancelFullScreen()
- }
- this.$refs['custom-video_container'].style.width = '100%'
- this.$refs['custom-video_container'].style.height = '100%'
- }
- }
- }
- </script>
- <style scoped>
- /* 总容器 */
- .custom-video_container{
- width: 100%;
- height: 100%;
- position: relative;
- overflow: hidden;
- background: #000;
- }
- /* 视频标签 */
- .custom-video_video{
- width: 100%;
- height: 100%;
- }
- /* 暂停 或者 播放 */
- .custom-video_play{
- display: inline-block;
- position: absolute;
- right: 50px;
- bottom: 50px;
- width: 30px;
- height: 30px;
- border-radius: 50%;
- font-size: 30px;
- color: cornflowerblue;
- }
- /* 暂停隐藏 */
- .custom-video_play-pause{
- display: none;
- }
- /* hover 显示 */
- .custom-video_container:hover > .custom-video_play-pause{
- display: inline-block;
- }
- /* hover 播放按钮动画 */
- .custom-video_play:hover{
- box-shadow: 0 0 10px #5A4180;
- transition: all 0.4s;
- }
- /* 控制栏 */
- .custom-video_control{
- position: absolute;
- width: 100%;
- height: 50px;
- left: 0;
- bottom: 0;
- background-color: rgba(0, 0, 0, .55);
- display: flex;
- flex-direction: row;
- align-items: center;
- }
- /* 控制栏进度条 */
- .custom-video_control-bg{
- flex: 1;
- height: 100%;
- display: flex;
- justify-content: center;
- align-items: center;
- margin: 0 10px;
- }
- /* 控制栏进度条 —— 总长度 */
- .custom-video_control-bg-outside{
- width: 100%;
- height: 5px;
- border-radius: 2.5px;
- background-color: #aaa;
- position: relative;
- cursor: pointer;
- }
- /* 控制栏进度条 —— 播放长度 */
- .custom-video_control-bg-inside{
- position: absolute;
- display: inline-block;
- width: 0;
- height: 100%;
- border-radius: 2.5px;
- left: 0;
- top: 0;
- background-color: #fff;
- transition: all 0.2s;
- }
- /* 控制栏进度条 —— 播放点 */
- .custom-video_control-bg-inside-point{
- display: inline-block;
- width: 10px;
- height: 10px;
- background-color: #fff;
- border-radius: 50%;
- position: absolute;
- top: -2.5px;
- left: -1%;
- transition: all 0.2s;
- }
- /* 控制栏 —— 声音、时间、全屏缩放 */
- .custom-video_control-voice,
- .custom-video_control-time,
- .custom-video_control-full{
- display: flex;
- flex-direction: row;
- justify-content: center;
- align-items: center;
- color: #fff;
- position: relative;
- }
- .custom-video_control-voice:hover > .custom-video_control-voice-bg{
- display: block;
- }
- .custom-video_control-voice-play{
- z-index: 10;
- }
- .custom-video_control-voice-bg{
- display: none;
- position: absolute;
- width: 30px;
- height: 100px;
- background-color: rgba(0, 0, 0, .55);
- left: 0;
- bottom: 0px;
- border-radius: 15px;
- }
- .custom-video_control-voice-bg-outside{
- width: 5px;
- height: 70px;
- border-radius: 2.5px;
- background-color: #aaa;
- position: absolute;
- left: 50%;
- transform: translate3d(-50%, 10%, 0);
- cursor: pointer;
- }
- .custom-video_control-voice-bg-inside{
- display: inline-block;
- position: absolute;
- width: 100%;
- bottom: 0;
- left: 0;
- border-radius: 2.5px;
- background-color: #fff;
- height: 0;
- }
- .custom-video_control-voice-bg-point{
- display: inline-block;
- width: 10px;
- height: 10px;
- background-color: #fff;
- border-radius: 50%;
- position: absolute;
- left: -2.5px;
- bottom: -1px;
- }
- .custom-video_control-time{
- font-size: 12px;
- }
- .custom-video_control-full{
- font-size: 14px;
- }
- .custom-video_control-voice,
- .custom-video_control-full{
- width: 30px;
- height: 30px;
- cursor: pointer;
- }
- /* 控制栏隐藏动画 */
- .fade-enter-active {
- transition: all .3s ease;
- }
- .fade-leave-active {
- transition: all .8s cubic-bezier(1.0, 0.5, 0.8, 1.0);
- }
- .fade-enter, .fade-leave-to {
- transform: translateY(50px);
- opacity: 0;
- }
- </style>
|