<template>
|
<uni-video
|
:id="id"
|
v-on="$listeners"
|
>
|
<div
|
ref="container"
|
class="uni-video-container"
|
@touchstart="touchstart"
|
@touchend="touchend"
|
@touchmove="touchmove"
|
@fullscreenchange.stop="onFullscreenChange"
|
@webkitfullscreenchange.stop="onFullscreenChange($event,true)"
|
>
|
<video
|
ref="video"
|
:style="{objectFit:objectFit}"
|
:muted="muted"
|
:loop="loop"
|
:src="srcSync"
|
:poster="poster"
|
:autoplay="autoplay"
|
v-bind="$attrs"
|
class="uni-video-video"
|
webkit-playsinline
|
playsinline
|
@click="triggerControls"
|
@durationchange="onDurationChange"
|
@loadedmetadata="onLoadedMetadata"
|
@progress="onProgress"
|
@waiting="onWaiting"
|
@error="onVideoError"
|
@play="onPlay"
|
@pause="onPause"
|
@ended="onEnded"
|
@timeupdate="onTimeUpdate"
|
@webkitbeginfullscreen="emitFullscreenChange(true)"
|
@x5videoenterfullscreen="emitFullscreenChange(true)"
|
@webkitendfullscreen="emitFullscreenChange(false)"
|
@x5videoexitfullscreen="emitFullscreenChange(false)"
|
/>
|
<div
|
v-show="controlsShow"
|
class="uni-video-bar uni-video-bar-full"
|
@click.stop
|
>
|
<div class="uni-video-controls">
|
<div
|
v-show="showPlayBtn"
|
:class="{'uni-video-control-button-play':!playing,'uni-video-control-button-pause':playing}"
|
class="uni-video-control-button"
|
@click.stop="trigger"
|
/>
|
<div
|
v-show="showProgress"
|
class="uni-video-current-time"
|
>
|
{{ currentTime|time }}
|
</div>
|
<div
|
v-show="showProgress"
|
ref="progress"
|
class="uni-video-progress-container"
|
@click.stop="clickProgress($event)"
|
>
|
<div class="uni-video-progress">
|
<div
|
:style="{width:buffered+'%'}"
|
class="uni-video-progress-buffered"
|
/>
|
<div
|
ref="ball"
|
:style="{left:progress+'%'}"
|
class="uni-video-ball"
|
>
|
<div class="uni-video-inner" />
|
</div>
|
</div>
|
</div>
|
<div
|
v-show="showProgress"
|
class="uni-video-duration"
|
>
|
{{ (duration||durationTime)|time }}
|
</div>
|
</div>
|
<div
|
v-if="danmuBtn"
|
:class="{'uni-video-danmu-button-active':enableDanmuSync}"
|
class="uni-video-danmu-button"
|
@click.stop="triggerDanmu"
|
>
|
{{ $$t("uni.video.danmu") }}
|
</div>
|
<div
|
v-show="showFullscreenBtn"
|
:class="{'uni-video-type-fullscreen':fullscreen}"
|
class="uni-video-fullscreen"
|
@click.stop="triggerFullscreen(!fullscreen)"
|
/>
|
</div>
|
<div
|
v-show="start&&enableDanmuSync"
|
ref="danmu"
|
style="z-index: 0;"
|
class="uni-video-danmu"
|
/>
|
<div
|
v-if="centerPlayBtnShow"
|
class="uni-video-cover"
|
@click.stop
|
>
|
<div
|
class="uni-video-cover-play-button"
|
@click.stop="play"
|
/>
|
<p class="uni-video-cover-duration">
|
{{ (duration||durationTime)|time }}
|
</p>
|
</div>
|
<div
|
:class="{'uni-video-toast-volume':gestureType==='volume'}"
|
class="uni-video-toast"
|
>
|
<div class="uni-video-toast-title">
|
{{ $$t("uni.video.volume") }}
|
</div>
|
<svg
|
class="uni-video-toast-icon"
|
width="200px"
|
height="200px"
|
viewBox="0 0 1024 1024"
|
version="1.1"
|
xmlns="http://www.w3.org/2000/svg"
|
>
|
<path
|
d="M475.400704 201.19552l0 621.674496q0 14.856192-10.856448 25.71264t-25.71264 10.856448-25.71264-10.856448l-190.273536-190.273536-149.704704 0q-14.856192 0-25.71264-10.856448t-10.856448-25.71264l0-219.414528q0-14.856192 10.856448-25.71264t25.71264-10.856448l149.704704 0 190.273536-190.273536q10.856448-10.856448 25.71264-10.856448t25.71264 10.856448 10.856448 25.71264zm219.414528 310.837248q0 43.425792-24.28416 80.851968t-64.2816 53.425152q-5.71392 2.85696-14.2848 2.85696-14.856192 0-25.71264-10.570752t-10.856448-25.998336q0-11.999232 6.856704-20.284416t16.570368-14.2848 19.427328-13.142016 16.570368-20.284416 6.856704-32.569344-6.856704-32.569344-16.570368-20.284416-19.427328-13.142016-16.570368-14.2848-6.856704-20.284416q0-15.427584 10.856448-25.998336t25.71264-10.570752q8.57088 0 14.2848 2.85696 39.99744 15.427584 64.2816 53.139456t24.28416 81.137664zm146.276352 0q0 87.422976-48.56832 161.41824t-128.5632 107.707392q-7.428096 2.85696-14.2848 2.85696-15.427584 0-26.284032-10.856448t-10.856448-25.71264q0-22.284288 22.284288-33.712128 31.997952-16.570368 43.425792-25.141248 42.283008-30.855168 65.995776-77.423616t23.712768-99.136512-23.712768-99.136512-65.995776-77.423616q-11.42784-8.57088-43.425792-25.141248-22.284288-11.42784-22.284288-33.712128 0-14.856192 10.856448-25.71264t25.71264-10.856448q7.428096 0 14.856192 2.85696 79.99488 33.712128 128.5632 107.707392t48.56832 161.41824zm146.276352 0q0 131.42016-72.566784 241.41312t-193.130496 161.989632q-7.428096 2.85696-14.856192 2.85696-14.856192 0-25.71264-10.856448t-10.856448-25.71264q0-20.570112 22.284288-33.712128 3.999744-2.285568 12.85632-5.999616t12.85632-5.999616q26.284032-14.2848 46.854144-29.140992 70.281216-51.996672 109.707264-129.705984t39.426048-165.132288-39.426048-165.132288-109.707264-129.705984q-20.570112-14.856192-46.854144-29.140992-3.999744-2.285568-12.85632-5.999616t-12.85632-5.999616q-22.284288-13.142016-22.284288-33.712128 0-14.856192 10.856448-25.71264t25.71264-10.856448q7.428096 0 14.856192 2.85696 120.563712 51.996672 193.130496 161.989632t72.566784 241.41312z"
|
/>
|
</svg>
|
<div class="uni-video-toast-value">
|
<div
|
:style="{width:volumeNew*100+'%'}"
|
class="uni-video-toast-value-content"
|
>
|
<div class="uni-video-toast-volume-grids">
|
<div
|
v-for="(item,index) in 10"
|
:key="index"
|
class="uni-video-toast-volume-grids-item"
|
/>
|
</div>
|
</div>
|
</div>
|
</div>
|
<div
|
:class="{'uni-video-toast-progress':gestureType=='progress'}"
|
class="uni-video-toast"
|
>
|
<div class="uni-video-toast-title">
|
{{ currentTimeNew|time }} / {{ durationTime|time }}
|
</div>
|
</div>
|
<div class="uni-video-slots">
|
<slot />
|
</div>
|
</div>
|
</uni-video>
|
</template>
|
<script>
|
import {
|
subscriber,
|
interact
|
} from 'uni-mixins'
|
import {
|
supportsPassive
|
} from 'uni-shared'
|
import {
|
i18nMixin
|
} from 'uni-core/helpers/i18n'
|
|
const passiveOptions = supportsPassive ? {
|
passive: false
|
} : false
|
|
const GestureType = {
|
NONE: 'none',
|
STOP: 'stop',
|
VOLUME: 'volume',
|
PROGRESS: 'progress'
|
}
|
|
export default {
|
name: 'Video',
|
filters: {
|
time (val) {
|
val = val > 0 && val < Infinity ? val : 0
|
let h = Math.floor(val / 3600)
|
let m = Math.floor(val % 3600 / 60)
|
let s = Math.floor(val % 3600 % 60)
|
h = (h < 10 ? '0' : '') + h
|
m = (m < 10 ? '0' : '') + m
|
s = (s < 10 ? '0' : '') + s
|
let str = m + ':' + s
|
if (h !== '00') {
|
str = h + ':' + str
|
}
|
return str
|
}
|
},
|
mixins: [i18nMixin, subscriber, interact],
|
props: {
|
id: {
|
type: String,
|
default: ''
|
},
|
src: {
|
type: String,
|
default: ''
|
},
|
duration: {
|
type: [Number, String],
|
default: ''
|
},
|
controls: {
|
type: [Boolean, String],
|
default: true
|
},
|
danmuList: {
|
type: Array,
|
default () {
|
return []
|
}
|
},
|
danmuBtn: {
|
type: [Boolean, String],
|
default: false
|
},
|
enableDanmu: {
|
type: [Boolean, String],
|
default: false
|
},
|
autoplay: {
|
type: [Boolean, String],
|
default: false
|
},
|
loop: {
|
type: [Boolean, String],
|
default: false
|
},
|
muted: {
|
type: [Boolean, String],
|
default: false
|
},
|
objectFit: {
|
type: String,
|
default: 'contain'
|
},
|
poster: {
|
type: String,
|
default: ''
|
},
|
direction: {
|
type: [String, Number],
|
default: ''
|
},
|
showProgress: {
|
type: Boolean,
|
default: true
|
},
|
initialTime: {
|
type: [String, Number],
|
default: 0
|
},
|
showFullscreenBtn: {
|
type: [Boolean, String],
|
default: true
|
},
|
pageGesture: {
|
type: [Boolean, String],
|
default: false
|
},
|
enableProgressGesture: {
|
type: [Boolean, String],
|
default: true
|
},
|
showPlayBtn: {
|
type: [Boolean, String],
|
default: true
|
},
|
showCenterPlayBtn: {
|
type: [Boolean, String],
|
default: true
|
}
|
},
|
data () {
|
return {
|
start: false,
|
playing: false,
|
currentTime: 0,
|
durationTime: 0,
|
progress: 0,
|
touching: false,
|
enableDanmuSync: Boolean(this.enableDanmu),
|
controlsVisible: true,
|
fullscreen: false,
|
controlsTouching: false,
|
touchStartOrigin: {
|
x: 0,
|
y: 0
|
},
|
gestureType: GestureType.NONE,
|
currentTimeOld: 0,
|
currentTimeNew: 0,
|
volumeOld: null,
|
volumeNew: null,
|
buffered: 0,
|
isSafari: /^Apple/.test(navigator.vendor)
|
}
|
},
|
computed: {
|
centerPlayBtnShow () {
|
return this.showCenterPlayBtn && !this.start
|
},
|
controlsShow () {
|
return !this.centerPlayBtnShow && this.controls && this.controlsVisible
|
},
|
autoHideContorls () {
|
return this.controlsShow && this.playing && !this.controlsTouching
|
},
|
srcSync () {
|
return this.$getRealPath(this.src)
|
}
|
},
|
watch: {
|
enableDanmuSync (val) {
|
this.$emit('update:enableDanmu', val)
|
},
|
autoHideContorls (val) {
|
if (val) {
|
this.autoHideStart()
|
} else {
|
this.autoHideEnd()
|
}
|
},
|
srcSync (val) {
|
this.playing = false
|
this.currentTime = 0
|
},
|
currentTime () {
|
this.updateProgress()
|
},
|
duration () {
|
this.updateProgress()
|
},
|
buffered (buffered) {
|
if (buffered !== 0) {
|
this.$trigger('progress', {}, {
|
buffered
|
})
|
}
|
}
|
},
|
created () {
|
this.otherData = {
|
danmuList: [],
|
danmuIndex: {
|
time: 0,
|
index: -1
|
},
|
hideTiming: null
|
}
|
const danmuList = this.otherData.danmuList = JSON.parse(JSON.stringify(this.danmuList || []))
|
danmuList.sort(function (a, b) {
|
return (a.time || 0) - (b.time || 0)
|
})
|
},
|
mounted () {
|
const self = this
|
let originX
|
let originY
|
let moveOnce = true
|
let originProgress
|
const ball = this.$refs.ball
|
ball.addEventListener('touchstart', (event) => {
|
this.controlsTouching = true
|
const toucher = event.targetTouches[0]
|
originX = toucher.pageX
|
originY = toucher.pageY
|
originProgress = this.progress
|
moveOnce = true
|
this.touching = true
|
ball.addEventListener('touchmove', touchmove, passiveOptions)
|
})
|
|
function touchmove (event) {
|
const toucher = event.targetTouches[0]
|
const pageX = toucher.pageX
|
const pageY = toucher.pageY
|
if (moveOnce && Math.abs(pageX - originX) < Math.abs(pageY - originY)) {
|
touchend()
|
return
|
}
|
moveOnce = false
|
const w = self.$refs.progress.offsetWidth
|
let progress = originProgress + (pageX - originX) / w * 100
|
if (progress < 0) {
|
progress = 0
|
} else if (progress > 100) {
|
progress = 100
|
}
|
self.progress = progress
|
event.preventDefault()
|
event.stopPropagation()
|
}
|
|
function touchend (event) {
|
self.controlsTouching = false
|
if (self.touching) {
|
ball.removeEventListener('touchmove', touchmove, passiveOptions)
|
if (!moveOnce) {
|
event.preventDefault()
|
event.stopPropagation()
|
self.seek(self.$refs.video.duration * self.progress / 100)
|
}
|
self.touching = false
|
}
|
}
|
ball.addEventListener('touchend', touchend)
|
ball.addEventListener('touchcancel', touchend)
|
},
|
beforeDestroy () {
|
this.triggerFullscreen(false)
|
clearTimeout(this.otherData.hideTiming)
|
},
|
methods: {
|
_handleSubscribe ({
|
type,
|
data = {}
|
}) {
|
const methods = ['play', 'pause', 'seek', 'sendDanmu', 'playbackRate', 'requestFullScreen', 'exitFullScreen']
|
let options
|
switch (type) {
|
case 'seek':
|
options = data.position
|
break
|
case 'sendDanmu':
|
options = data
|
break
|
case 'playbackRate':
|
options = data.rate
|
break
|
}
|
if (methods.indexOf(type) >= 0) {
|
this[type](options)
|
}
|
},
|
trigger () {
|
if (this.playing) {
|
this.$refs.video.pause()
|
} else {
|
this.$refs.video.play()
|
}
|
},
|
play () {
|
this.start = true
|
this.$refs.video.play()
|
},
|
pause () {
|
this.$refs.video.pause()
|
},
|
seek (position) {
|
position = Number(position)
|
if (typeof position === 'number' && !isNaN(position)) {
|
this.$refs.video.currentTime = position
|
}
|
},
|
clickProgress (event) {
|
const $progress = this.$refs.progress
|
let element = event.target
|
let x = event.offsetX
|
while (element !== $progress) {
|
x += element.offsetLeft
|
element = element.parentNode
|
}
|
const w = $progress.offsetWidth
|
let progress = 0
|
if (x >= 0 && x <= w) {
|
progress = x / w
|
this.seek(this.$refs.video.duration * progress)
|
}
|
},
|
triggerDanmu () {
|
this.enableDanmuSync = !this.enableDanmuSync
|
},
|
playDanmu (danmu) {
|
const p = document.createElement('p')
|
p.className = 'uni-video-danmu-item'
|
p.innerText = danmu.text
|
let style = `bottom: ${Math.random() * 100}%;color: ${danmu.color};`
|
p.setAttribute('style', style)
|
this.$refs.danmu.appendChild(p)
|
setTimeout(function () {
|
style += 'left: 0;-webkit-transform: translateX(-100%);transform: translateX(-100%);'
|
p.setAttribute('style', style)
|
setTimeout(function () {
|
p.remove()
|
}, 4000)
|
}, 17)
|
},
|
sendDanmu (danmu) {
|
const otherData = this.otherData
|
otherData.danmuList.splice(otherData.danmuIndex.index + 1, 0, {
|
text: String(danmu.text),
|
color: danmu.color,
|
time: this.$refs.video.currentTime || 0
|
})
|
},
|
playbackRate (rate) {
|
this.$refs.video.playbackRate = rate
|
},
|
triggerFullscreen (val) {
|
const container = this.$refs.container
|
const video = this.$refs.video
|
let mockFullScreen
|
if (val) {
|
if ((document.fullscreenEnabled || document.webkitFullscreenEnabled) && (!this.isSafari || this.userInteract)) {
|
container[document.fullscreenEnabled ? 'requestFullscreen' : 'webkitRequestFullscreen']()
|
} else if (video.webkitEnterFullScreen) {
|
video.webkitEnterFullScreen()
|
} else {
|
mockFullScreen = true
|
container.remove()
|
container.classList.add('uni-video-type-fullscreen')
|
document.body.appendChild(container)
|
}
|
} else {
|
if (document.fullscreenEnabled || document.webkitFullscreenEnabled) {
|
if (document.fullscreenElement) {
|
document.exitFullscreen()
|
} else if (document.webkitFullscreenElement) {
|
document.webkitExitFullscreen()
|
}
|
} else if (video.webkitExitFullScreen) {
|
video.webkitExitFullScreen()
|
} else {
|
mockFullScreen = true
|
container.remove()
|
container.classList.remove('uni-video-type-fullscreen')
|
this.$el.appendChild(container)
|
}
|
}
|
if (mockFullScreen) {
|
this.emitFullscreenChange(val)
|
}
|
},
|
onFullscreenChange ($event, webkit) {
|
if (webkit && document.fullscreenEnabled) {
|
return
|
}
|
this.emitFullscreenChange(!!(document.fullscreenElement || document.webkitFullscreenElement))
|
},
|
emitFullscreenChange (val) {
|
this.fullscreen = val
|
this.$trigger('fullscreenchange', {}, {
|
fullScreen: val,
|
direction: 'vertical'
|
})
|
},
|
requestFullScreen () {
|
this.triggerFullscreen(true)
|
},
|
exitFullScreen () {
|
this.triggerFullscreen(false)
|
},
|
onDurationChange ({ target }) {
|
this.durationTime = target.duration
|
},
|
onLoadedMetadata ($event) {
|
const initialTime = Number(this.initialTime) || 0
|
const video = $event.target
|
if (initialTime > 0) {
|
video.currentTime = initialTime
|
}
|
this.$trigger('loadedmetadata', $event, {
|
width: video.videoWidth,
|
height: video.videoHeight,
|
duration: video.duration
|
})
|
this.onProgress($event)
|
},
|
onProgress ($event) {
|
const video = $event.target
|
const buffered = video.buffered
|
if (buffered.length) {
|
this.buffered = buffered.end(buffered.length - 1) / video.duration * 100
|
}
|
},
|
onWaiting ($event) {
|
this.$trigger('waiting', $event, {})
|
},
|
onVideoError ($event) {
|
this.playing = false
|
this.$trigger('error', $event, {})
|
},
|
onPlay ($event) {
|
this.start = true
|
this.playing = true
|
this.$trigger('play', $event, {})
|
},
|
onPause ($event) {
|
this.playing = false
|
this.$trigger('pause', $event, {})
|
},
|
onEnded ($event) {
|
this.playing = false
|
this.$trigger('ended', $event, {})
|
},
|
onTimeUpdate ($event) {
|
const video = $event.target
|
const otherData = this.otherData
|
const currentTime = this.currentTime = video.currentTime
|
const oldDanmuIndex = otherData.danmuIndex
|
const danmuIndex = {
|
time: currentTime,
|
index: oldDanmuIndex.index
|
}
|
const danmuList = otherData.danmuList
|
if (currentTime > oldDanmuIndex.time) {
|
for (let index = oldDanmuIndex.index + 1; index < danmuList.length; index++) {
|
const element = danmuList[index]
|
if (currentTime >= (element.time || 0)) {
|
danmuIndex.index = index
|
if (this.playing && this.enableDanmuSync) {
|
this.playDanmu(element)
|
}
|
} else {
|
break
|
}
|
}
|
} else if (currentTime < oldDanmuIndex.time) {
|
for (let index = oldDanmuIndex.index - 1; index > -1; index--) {
|
const element = danmuList[index]
|
if (currentTime <= (element.time || 0)) {
|
danmuIndex.index = index - 1
|
} else {
|
break
|
}
|
}
|
}
|
otherData.danmuIndex = danmuIndex
|
this.$trigger('timeupdate', $event, {
|
currentTime,
|
duration: video.duration
|
})
|
},
|
triggerControls () {
|
this.controlsVisible = !this.controlsVisible
|
},
|
touchstart (event) {
|
const toucher = event.targetTouches[0]
|
this.touchStartOrigin = {
|
x: toucher.pageX,
|
y: toucher.pageY
|
}
|
this.gestureType = GestureType.NONE
|
this.volumeOld = null
|
this.currentTimeOld = this.currentTimeNew = 0
|
},
|
touchmove (event) {
|
function stop () {
|
event.stopPropagation()
|
event.preventDefault()
|
}
|
if (this.fullscreen) {
|
stop()
|
}
|
const gestureType = this.gestureType
|
if (gestureType === GestureType.STOP) {
|
return
|
}
|
const toucher = event.targetTouches[0]
|
const pageX = toucher.pageX
|
const pageY = toucher.pageY
|
const origin = this.touchStartOrigin
|
if (gestureType === GestureType.PROGRESS) {
|
this.changeProgress(pageX - origin.x)
|
} else if (gestureType === GestureType.VOLUME) {
|
this.changeVolume(pageY - origin.y)
|
}
|
if (gestureType !== GestureType.NONE) {
|
return
|
}
|
if (Math.abs(pageX - origin.x) > Math.abs(pageY - origin.y)) {
|
if (!this.enableProgressGesture) {
|
this.gestureType = GestureType.STOP
|
return
|
}
|
this.gestureType = GestureType.PROGRESS
|
this.currentTimeOld = this.currentTimeNew = this.$refs.video.currentTime
|
if (!this.fullscreen) {
|
stop()
|
}
|
} else {
|
if (!this.pageGesture) {
|
this.gestureType = GestureType.STOP
|
return
|
}
|
this.gestureType = GestureType.VOLUME
|
this.volumeOld = this.$refs.video.volume
|
if (!this.fullscreen) {
|
stop()
|
}
|
}
|
},
|
touchend (event) {
|
if (this.gestureType !== GestureType.NONE && this.gestureType !== GestureType.STOP) {
|
event.stopPropagation()
|
event.preventDefault()
|
}
|
if (this.gestureType === GestureType.PROGRESS && this.currentTimeOld !== this.currentTimeNew) {
|
this.$refs.video.currentTime = this.currentTimeNew
|
}
|
this.gestureType = GestureType.NONE
|
},
|
changeProgress (x) {
|
const duration = this.$refs.video.duration
|
let currentTimeNew = x / 600 * duration + this.currentTimeOld
|
if (currentTimeNew < 0) {
|
currentTimeNew = 0
|
} else if (currentTimeNew > duration) {
|
currentTimeNew = duration
|
}
|
this.currentTimeNew = currentTimeNew
|
},
|
changeVolume (y) {
|
const valueOld = this.volumeOld
|
let value
|
if (typeof valueOld === 'number') {
|
value = valueOld - y / 200
|
if (value < 0) {
|
value = 0
|
} else if (value > 1) {
|
value = 1
|
}
|
this.$refs.video.volume = value
|
this.volumeNew = value
|
}
|
},
|
autoHideStart () {
|
this.otherData.hideTiming = setTimeout(() => {
|
this.controlsVisible = false
|
}, 3000)
|
},
|
autoHideEnd () {
|
const otherData = this.otherData
|
if (otherData.hideTiming) {
|
clearTimeout(otherData.hideTiming)
|
otherData.hideTiming = null
|
}
|
},
|
updateProgress () {
|
if (!this.touching) {
|
this.progress = this.currentTime / this.durationTime * 100
|
}
|
}
|
}
|
}
|
</script>
|
|
<style>
|
uni-video {
|
width: 300px;
|
height: 225px;
|
display: inline-block;
|
line-height: 0;
|
overflow: hidden;
|
position: relative;
|
}
|
|
uni-video[hidden] {
|
display: none;
|
}
|
|
.uni-video-container {
|
width: 100%;
|
height: 100%;
|
background-color: black;
|
display: inline-block;
|
position: absolute;
|
top: 0;
|
left: 0;
|
overflow: hidden;
|
object-position: inherit;
|
}
|
|
.uni-video-container.uni-video-type-fullscreen {
|
position: fixed;
|
z-index: 999;
|
}
|
|
.uni-video-video {
|
width: 100%;
|
height: 100%;
|
object-position: inherit;
|
}
|
|
.uni-video-cover {
|
position: absolute;
|
top: 0;
|
left: 0;
|
bottom: 0;
|
width: 100%;
|
display: flex;
|
flex-direction: column;
|
justify-content: center;
|
align-items: center;
|
background-color: rgba(1, 1, 1, 0.5);
|
z-index: 1;
|
}
|
|
.uni-video-slots {
|
position: absolute;
|
left: 0;
|
top: 0;
|
width: 100%;
|
height: 100%;
|
overflow: hidden;
|
pointer-events: none;
|
}
|
|
.uni-video-cover-play-button {
|
width: 40px;
|
height: 40px;
|
background-size: 50%;
|
background-repeat: no-repeat;
|
background-position: 50% 50%;
|
cursor: pointer;
|
}
|
|
.uni-video-cover-duration {
|
color: #fff;
|
font-size: 16px;
|
line-height: 1;
|
margin-top: 10px;
|
}
|
|
.uni-video-bar {
|
height: 44px;
|
background-color: rgba(0, 0, 0, 0.5);
|
overflow: hidden;
|
position: absolute;
|
bottom: 0;
|
right: 0;
|
display: flex;
|
display: flex;
|
align-items: center;
|
align-items: center;
|
padding: 0 10px;
|
z-index: 0;
|
/* 解决全屏后被 video 遮挡的问题 */
|
transform: translate3d(0, 0, 0);
|
}
|
|
.uni-video-bar.uni-video-bar-full {
|
left: 0;
|
}
|
|
.uni-video-controls {
|
display: flex;
|
display: flex;
|
flex-grow: 1;
|
flex-grow: 1;
|
margin: 0 8.5px;
|
}
|
|
.uni-video-control-button {
|
width: 13px;
|
height: 15px;
|
padding: 14.5px 12.5px 14.5px 12.5px;
|
margin-left: -8.5px;
|
box-sizing: content-box;
|
cursor: pointer;
|
}
|
|
.uni-video-control-button::after {
|
content: "";
|
display: block;
|
width: 100%;
|
height: 100%;
|
background-size: 100%;
|
background-position: 50% 50%;
|
background-repeat: no-repeat;
|
}
|
|
.uni-video-control-button.uni-video-control-button-play::after,
|
.uni-video-cover-play-button {
|
background-image: url("");
|
}
|
|
.uni-video-control-button.uni-video-control-button-pause::after {
|
background-image: url("");
|
}
|
|
.uni-video-current-time,
|
.uni-video-duration {
|
height: 14.5px;
|
line-height: 14.5px;
|
margin-top: 15px;
|
margin-bottom: 14.5px;
|
font-size: 12px;
|
color: #cbcbcb;
|
}
|
|
.uni-video-progress-container {
|
flex-grow: 2;
|
flex-grow: 2;
|
position: relative;
|
}
|
|
.uni-video-progress {
|
height: 2px;
|
margin: 21px 12px;
|
background-color: rgba(255, 255, 255, 0.4);
|
position: relative;
|
cursor: pointer;
|
}
|
|
.uni-video-progress-buffered {
|
position: absolute;
|
left: 0;
|
top: 0;
|
width: 0;
|
height: 100%;
|
transition: width 0.1s;
|
background-color: rgba(255, 255, 255, 0.8);
|
}
|
|
.uni-video-ball {
|
width: 16px;
|
height: 16px;
|
padding: 14px;
|
position: absolute;
|
top: -21px;
|
box-sizing: content-box;
|
left: 0%;
|
margin-left: -22px;
|
}
|
|
.uni-video-inner {
|
width: 100%;
|
height: 100%;
|
background-color: #ffffff;
|
border-radius: 50%;
|
}
|
|
.uni-video-danmu-button {
|
white-space: nowrap;
|
line-height: 1;
|
padding: 2px 10px;
|
border: 1px solid #fff;
|
border-radius: 5px;
|
font-size: 13px;
|
color: #fff;
|
margin: 0 8.5px;
|
cursor: pointer;
|
}
|
|
.uni-video-danmu-button.uni-video-danmu-button-active {
|
border-color: #48c23d;
|
color: #48c23d;
|
}
|
|
.uni-video-fullscreen {
|
width: 17px;
|
height: 17px;
|
padding: 8.5px;
|
box-sizing: content-box;
|
background-image: url("");
|
background-size: 50%;
|
background-position: 50% 50%;
|
background-repeat: no-repeat;
|
cursor: pointer;
|
}
|
|
.uni-video-fullscreen.uni-video-type-fullscreen {
|
background-image: url("");
|
}
|
|
.uni-video-danmu {
|
position: absolute;
|
top: 0;
|
left: 0;
|
bottom: 0;
|
width: 100%;
|
margin-top: 14px;
|
margin-bottom: 44px;
|
font-size: 14px;
|
line-height: 14px;
|
overflow: visible;
|
}
|
|
.uni-video-danmu-item {
|
line-height: 1;
|
position: absolute;
|
color: #ffffff;
|
white-space: nowrap;
|
left: 100%;
|
transform: translatex(0);
|
transition-property: left, transform;
|
transition-duration: 3s;
|
transition-timing-function: linear;
|
}
|
|
.uni-video-toast {
|
pointer-events: none;
|
position: absolute;
|
left: 50%;
|
top: 50%;
|
transform: translate(-50%, -50%);
|
border-radius: 5px;
|
background-color: rgba(255, 255, 255, 0.8);
|
color: #000000;
|
display: none;
|
}
|
|
.uni-video-toast.uni-video-toast-volume {
|
width: 100px;
|
height: 100px;
|
display: block;
|
}
|
|
.uni-video-toast-volume .uni-video-toast-title {
|
width: 100%;
|
font-size: 12px;
|
line-height: 16px;
|
text-align: center;
|
margin-top: 10px;
|
display: block;
|
}
|
|
.uni-video-toast-volume .uni-video-toast-icon {
|
fill: #000000;
|
width: 50%;
|
height: 50%;
|
margin-left: 25%;
|
display: block;
|
}
|
|
.uni-video-toast-volume .uni-video-toast-value {
|
width: 80px;
|
height: 5px;
|
margin-top: 5px;
|
margin-left: 10px;
|
}
|
|
.uni-video-toast-volume
|
.uni-video-toast-value
|
> .uni-video-toast-value-content {
|
overflow: hidden;
|
}
|
|
.uni-video-toast-volume-grids {
|
width: 80px;
|
height: 5px;
|
}
|
|
.uni-video-toast-volume-grids-item {
|
float: left;
|
width: 7.1px;
|
height: 5px;
|
background-color: #000000;
|
}
|
|
.uni-video-toast-volume-grids-item:not(:first-child) {
|
margin-left: 1px;
|
}
|
|
.uni-video-toast.uni-video-toast-progress {
|
display: block;
|
background-color: rgba(0, 0, 0, 0.8);
|
color: white;
|
font-size: 14px;
|
line-height: 18px;
|
padding: 6px;
|
}
|
</style>
|