]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blob - client/src/assets/player/shared/mobile/peertube-mobile-plugin.ts
Fix missing p2p media loader config for hls
[github/Chocobozzz/PeerTube.git] / client / src / assets / player / shared / mobile / peertube-mobile-plugin.ts
1 import { PeerTubeMobileButtons } from './peertube-mobile-buttons'
2 import videojs from 'video.js'
3 import debug from 'debug'
4
5 const logger = debug('peertube:player:mobile')
6
7 const Plugin = videojs.getPlugin('plugin')
8
9 class PeerTubeMobilePlugin extends Plugin {
10 private static readonly DOUBLE_TAP_DELAY_MS = 250
11 private static readonly SET_CURRENT_TIME_DELAY = 1000
12
13 private peerTubeMobileButtons: PeerTubeMobileButtons
14
15 private seekAmount = 0
16
17 private lastTapEvent: TouchEvent
18 private tapTimeout: ReturnType<typeof setTimeout>
19 private newActiveState: boolean
20
21 private setCurrentTimeTimeout: ReturnType<typeof setTimeout>
22
23 constructor (player: videojs.Player, options: videojs.PlayerOptions) {
24 super(player, options)
25
26 this.peerTubeMobileButtons = player.addChild('PeerTubeMobileButtons', { reportTouchActivity: false }) as PeerTubeMobileButtons
27
28 if (videojs.browser.IS_ANDROID && screen.orientation) {
29 this.handleFullscreenRotation()
30 }
31
32 if (!this.player.options_.userActions) this.player.options_.userActions = {};
33
34 // FIXME: typings
35 (this.player.options_.userActions as any).click = false
36 this.player.options_.userActions.doubleClick = false
37
38 this.player.one('play', () => {
39 this.initTouchStartEvents()
40 })
41 }
42
43 private handleFullscreenRotation () {
44 this.player.on('fullscreenchange', () => {
45 if (!this.player.isFullscreen() || this.isPortraitVideo()) return
46
47 screen.orientation.lock('landscape')
48 .catch(err => console.error('Cannot lock screen to landscape.', err))
49 })
50 }
51
52 private isPortraitVideo () {
53 return this.player.videoWidth() < this.player.videoHeight()
54 }
55
56 private initTouchStartEvents () {
57 const handleTouchStart = (event: TouchEvent) => {
58 if (this.tapTimeout) {
59 clearTimeout(this.tapTimeout)
60 this.tapTimeout = undefined
61 }
62
63 if (this.lastTapEvent && event.timeStamp - this.lastTapEvent.timeStamp < PeerTubeMobilePlugin.DOUBLE_TAP_DELAY_MS) {
64 logger('Detected double tap')
65
66 this.lastTapEvent = undefined
67 this.onDoubleTap(event)
68 return
69 }
70
71 this.newActiveState = !this.player.userActive()
72
73 this.tapTimeout = setTimeout(() => {
74 logger('No double tap detected, set user active state to %s.', this.newActiveState)
75
76 this.player.userActive(this.newActiveState)
77 }, PeerTubeMobilePlugin.DOUBLE_TAP_DELAY_MS)
78
79 this.lastTapEvent = event
80 }
81
82 this.player.on('touchstart', (event: TouchEvent) => {
83 // Only enable user active on player touch, we listen event on peertube mobile buttons to disable it
84 if (this.player.userActive()) return
85
86 handleTouchStart(event)
87 })
88
89 this.peerTubeMobileButtons.el().addEventListener('touchstart', (event: TouchEvent) => {
90 // Prevent mousemove/click events firing on the player, that conflict with our user active logic
91 event.preventDefault()
92
93 handleTouchStart(event)
94 }, { passive: false })
95 }
96
97 private onDoubleTap (event: TouchEvent) {
98 const playerWidth = this.player.currentWidth()
99
100 const rect = this.findPlayerTarget((event.target as HTMLElement)).getBoundingClientRect()
101 const offsetX = event.targetTouches[0].pageX - rect.left
102
103 logger('Calculating double tap zone (player width: %d, offset X: %d)', playerWidth, offsetX)
104
105 if (offsetX > 0.66 * playerWidth) {
106 if (this.seekAmount < 0) this.seekAmount = 0
107
108 this.seekAmount += 10
109
110 logger('Will forward %d seconds', this.seekAmount)
111 } else if (offsetX < 0.33 * playerWidth) {
112 if (this.seekAmount > 0) this.seekAmount = 0
113
114 this.seekAmount -= 10
115 logger('Will rewind %d seconds', this.seekAmount)
116 }
117
118 this.peerTubeMobileButtons.displayFastSeek(this.seekAmount)
119
120 this.scheduleSetCurrentTime()
121 }
122
123 private findPlayerTarget (target: HTMLElement): HTMLElement {
124 if (target.classList.contains('video-js')) return target
125
126 return this.findPlayerTarget(target.parentElement)
127 }
128
129 private scheduleSetCurrentTime () {
130 this.player.pause()
131 this.player.addClass('vjs-fast-seeking')
132
133 if (this.setCurrentTimeTimeout) clearTimeout(this.setCurrentTimeTimeout)
134
135 this.setCurrentTimeTimeout = setTimeout(() => {
136 let newTime = this.player.currentTime() + this.seekAmount
137 this.seekAmount = 0
138
139 newTime = Math.max(0, newTime)
140 newTime = Math.min(this.player.duration(), newTime)
141
142 this.player.currentTime(newTime)
143 this.seekAmount = 0
144 this.peerTubeMobileButtons.displayFastSeek(0)
145
146 this.player.removeClass('vjs-fast-seeking')
147 this.player.userActive(false)
148
149 this.player.play()
150 }, PeerTubeMobilePlugin.SET_CURRENT_TIME_DELAY)
151 }
152 }
153
154 videojs.registerPlugin('peertubeMobile', PeerTubeMobilePlugin)
155 export { PeerTubeMobilePlugin }