2019-12-17 09:49:33 -06:00
|
|
|
// @ts-ignore
|
|
|
|
import * as videojs from 'video.js'
|
|
|
|
import { VideoJSComponentInterface } from '../peertube-videojs-typings'
|
|
|
|
|
|
|
|
function getMainTemplate (options: any) {
|
|
|
|
return `
|
|
|
|
<div class="vjs-upnext-top">
|
|
|
|
<span class="vjs-upnext-headtext">${options.headText}</span>
|
|
|
|
<div class="vjs-upnext-title"></div>
|
|
|
|
</div>
|
|
|
|
<div class="vjs-upnext-autoplay-icon">
|
|
|
|
<svg height="100%" version="1.1" viewbox="0 0 98 98" width="100%">
|
|
|
|
<circle class="vjs-upnext-svg-autoplay-circle" cx="49" cy="49" fill="#000" fill-opacity="0.8" r="48"></circle>
|
|
|
|
<circle class="vjs-upnext-svg-autoplay-ring" cx="-49" cy="49" fill-opacity="0" r="46.5" stroke="#FFFFFF" stroke-width="4" transform="rotate(-90)"></circle>
|
|
|
|
<polygon class="vjs-upnext-svg-autoplay-triangle" fill="#fff" points="32,27 72,49 32,71"></polygon></svg>
|
|
|
|
</div>
|
|
|
|
<span class="vjs-upnext-bottom">
|
|
|
|
<span class="vjs-upnext-cancel">
|
|
|
|
<button class="vjs-upnext-cancel-button" tabindex="0" aria-label="Cancel autoplay">${options.cancelText}</button>
|
|
|
|
</span>
|
2019-12-20 10:49:57 -06:00
|
|
|
<span class="vjs-upnext-suspended">${options.suspendedText}</span>
|
2019-12-17 09:49:33 -06:00
|
|
|
</span>
|
|
|
|
`
|
|
|
|
}
|
|
|
|
|
|
|
|
// @ts-ignore-start
|
|
|
|
const Component = videojs.getComponent('Component')
|
|
|
|
class EndCard extends Component {
|
|
|
|
options_: any
|
|
|
|
dashOffsetTotal = 586
|
|
|
|
dashOffsetStart = 293
|
|
|
|
interval = 50
|
|
|
|
upNextEvents = new videojs.EventTarget()
|
2019-12-20 10:49:57 -06:00
|
|
|
ticks = 0
|
|
|
|
totalTicks: number
|
2019-12-17 09:49:33 -06:00
|
|
|
|
|
|
|
container: HTMLElement
|
|
|
|
title: HTMLElement
|
|
|
|
autoplayRing: HTMLElement
|
|
|
|
cancelButton: HTMLElement
|
2019-12-20 10:49:57 -06:00
|
|
|
suspendedMessage: HTMLElement
|
2019-12-17 09:49:33 -06:00
|
|
|
nextButton: HTMLElement
|
|
|
|
|
|
|
|
constructor (player: videojs.Player, options: any) {
|
|
|
|
super(player, options)
|
|
|
|
|
2019-12-20 10:49:57 -06:00
|
|
|
this.totalTicks = this.options_.timeout / this.interval
|
2019-12-17 09:49:33 -06:00
|
|
|
|
|
|
|
player.on('ended', (_: any) => {
|
2019-12-20 10:49:57 -06:00
|
|
|
if (!this.options_.condition()) return
|
2019-12-17 09:49:33 -06:00
|
|
|
|
|
|
|
player.addClass('vjs-upnext--showing')
|
|
|
|
this.showCard((canceled: boolean) => {
|
|
|
|
player.removeClass('vjs-upnext--showing')
|
|
|
|
this.container.style.display = 'none'
|
|
|
|
if (!canceled) {
|
2019-12-20 10:49:57 -06:00
|
|
|
this.options_.next()
|
2019-12-17 09:49:33 -06:00
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
player.on('playing', () => {
|
|
|
|
this.upNextEvents.trigger('playing')
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
createEl () {
|
|
|
|
const container = super.createEl('div', {
|
|
|
|
className: 'vjs-upnext-content',
|
|
|
|
innerHTML: getMainTemplate(this.options_)
|
|
|
|
})
|
|
|
|
|
|
|
|
this.container = container
|
|
|
|
container.style.display = 'none'
|
|
|
|
|
|
|
|
this.autoplayRing = container.getElementsByClassName('vjs-upnext-svg-autoplay-ring')[0]
|
|
|
|
this.title = container.getElementsByClassName('vjs-upnext-title')[0]
|
|
|
|
this.cancelButton = container.getElementsByClassName('vjs-upnext-cancel-button')[0]
|
2019-12-20 10:49:57 -06:00
|
|
|
this.suspendedMessage = container.getElementsByClassName('vjs-upnext-suspended')[0]
|
2019-12-17 09:49:33 -06:00
|
|
|
this.nextButton = container.getElementsByClassName('vjs-upnext-autoplay-icon')[0]
|
|
|
|
|
|
|
|
this.cancelButton.onclick = () => {
|
|
|
|
this.upNextEvents.trigger('cancel')
|
|
|
|
}
|
|
|
|
|
|
|
|
this.nextButton.onclick = () => {
|
|
|
|
this.upNextEvents.trigger('next')
|
|
|
|
}
|
|
|
|
|
|
|
|
return container
|
|
|
|
}
|
|
|
|
|
|
|
|
showCard (cb: Function) {
|
|
|
|
let timeout: any
|
|
|
|
|
2019-12-18 04:01:59 -06:00
|
|
|
this.autoplayRing.setAttribute('stroke-dasharray', '' + this.dashOffsetStart)
|
|
|
|
this.autoplayRing.setAttribute('stroke-dashoffset', '' + -this.dashOffsetStart)
|
2019-12-17 09:49:33 -06:00
|
|
|
|
2019-12-20 10:49:57 -06:00
|
|
|
this.title.innerHTML = this.options_.getTitle()
|
2019-12-17 09:49:33 -06:00
|
|
|
|
|
|
|
this.upNextEvents.one('cancel', () => {
|
|
|
|
clearTimeout(timeout)
|
|
|
|
cb(true)
|
|
|
|
})
|
|
|
|
|
|
|
|
this.upNextEvents.one('playing', () => {
|
|
|
|
clearTimeout(timeout)
|
|
|
|
cb(true)
|
|
|
|
})
|
|
|
|
|
|
|
|
this.upNextEvents.one('next', () => {
|
|
|
|
clearTimeout(timeout)
|
|
|
|
cb(false)
|
|
|
|
})
|
|
|
|
|
2019-12-20 10:49:57 -06:00
|
|
|
const goToPercent = (percent: number) => {
|
|
|
|
const newOffset = Math.max(-this.dashOffsetTotal, - this.dashOffsetStart - percent * this.dashOffsetTotal / 2 / 100)
|
|
|
|
this.autoplayRing.setAttribute('stroke-dashoffset', '' + newOffset)
|
|
|
|
}
|
2019-12-17 09:49:33 -06:00
|
|
|
|
2019-12-20 10:49:57 -06:00
|
|
|
const tick = () => {
|
|
|
|
goToPercent((this.ticks++) * 100 / this.totalTicks)
|
|
|
|
}
|
|
|
|
|
|
|
|
const update = () => {
|
|
|
|
if (this.options_.suspended()) {
|
|
|
|
this.suspendedMessage.innerText = this.options_.suspendedText
|
|
|
|
goToPercent(0)
|
|
|
|
this.ticks = 0
|
|
|
|
timeout = setTimeout(update.bind(this), 300) // checks once supsended can be a bit longer
|
|
|
|
} else if (this.ticks >= this.totalTicks) {
|
2019-12-17 09:49:33 -06:00
|
|
|
clearTimeout(timeout)
|
|
|
|
cb(false)
|
|
|
|
} else {
|
2019-12-20 10:49:57 -06:00
|
|
|
this.suspendedMessage.innerText = ''
|
|
|
|
tick()
|
2019-12-17 09:49:33 -06:00
|
|
|
timeout = setTimeout(update.bind(this), this.interval)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.container.style.display = 'block'
|
|
|
|
timeout = setTimeout(update.bind(this), this.interval)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// @ts-ignore-end
|
|
|
|
|
|
|
|
videojs.registerComponent('EndCard', EndCard)
|
|
|
|
|
|
|
|
const Plugin: VideoJSComponentInterface = videojs.getPlugin('plugin')
|
|
|
|
class UpNextPlugin extends Plugin {
|
|
|
|
constructor (player: videojs.Player, options: any = {}) {
|
|
|
|
const settings = {
|
|
|
|
next: options.next,
|
|
|
|
getTitle: options.getTitle,
|
|
|
|
timeout: options.timeout || 5000,
|
|
|
|
cancelText: options.cancelText || 'Cancel',
|
|
|
|
headText: options.headText || 'Up Next',
|
2019-12-20 10:49:57 -06:00
|
|
|
suspendedText: options.suspendedText || 'Autoplay is suspended',
|
|
|
|
condition: options.condition,
|
|
|
|
suspended: options.suspended
|
2019-12-17 09:49:33 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
super(player, settings)
|
|
|
|
|
|
|
|
this.player.ready(() => {
|
|
|
|
player.addClass('vjs-upnext')
|
|
|
|
})
|
|
|
|
|
|
|
|
player.addChild('EndCard', settings)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
videojs.registerPlugin('upnext', UpNextPlugin)
|
|
|
|
export { UpNextPlugin }
|