feat: extract player controls timeline keyboard

This commit is contained in:
2026-05-10 18:34:21 +02:00
parent 99874974ad
commit ff1579345b
3 changed files with 324 additions and 0 deletions

169
static/player/controls.ts Normal file
View File

@@ -0,0 +1,169 @@
import { state } from './state'
export const formatTime = (seconds: number): string => {
if (!Number.isFinite(seconds) || seconds < 0) return '00:00'
const mins = Math.floor(seconds / 60)
const secs = Math.floor(seconds % 60)
return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`
}
export const showControls = (): void => {
state.container.classList.add('show-controls')
window.clearTimeout(state.playerControlsTimeout)
state.playerControlsTimeout = window.setTimeout(() => {
if (!state.isScrubbing && !state.video.paused) {
state.container.classList.remove('show-controls')
}
}, 2000)
}
export const seekBy = (delta: number): void => {
if (state.video.duration <= 0) return
state.video.currentTime = Math.max(0, Math.min(state.video.duration, state.video.currentTime + delta))
showControls()
}
export const togglePlayPause = (): void => {
if (state.video.paused) {
state.video.play()
} else {
state.video.pause()
}
}
export const toggleMute = (): void => {
if (state.video.muted || state.video.volume === 0) {
const restored = state.lastKnownVolume > 0 ? state.lastKnownVolume : 1
state.video.muted = false
state.video.volume = restored
} else {
state.lastKnownVolume = state.video.volume > 0 ? state.video.volume : state.lastKnownVolume
state.video.muted = true
}
}
export const setVolume = (value: number): void => {
state.video.volume = Math.max(0, Math.min(1, value))
state.video.muted = value === 0
if (value > 0) state.lastKnownVolume = value
}
export const toggleFullscreen = (): void => {
if (document.fullscreenElement) {
document.exitFullscreen()
return
}
state.container.requestFullscreen?.()
}
export const syncVolumeUI = (): void => {
const { volumeRange, volumeUnderline, iconVolume, iconMuted } = getControls()
const value = state.video.muted ? 0 : Math.round(state.video.volume * 100)
if (volumeRange) {
volumeRange.value = String(value)
volumeRange.style.setProperty('--volume-percent', `${value}%`)
}
if (volumeUnderline) volumeUnderline.style.height = `${value}%`
updateMuteIcons(state.video.muted || state.video.volume === 0)
}
interface Controls {
playPause: HTMLButtonElement | null
muteBtn: HTMLButtonElement | null
volumePanel: HTMLElement | null
volumeRange: HTMLInputElement | null
volumeUnderline: HTMLElement | null
backwardBtn: HTMLButtonElement | null
forwardBtn: HTMLButtonElement | null
fullscreenBtn: HTMLButtonElement | null
iconPlay: SVGElement | null
iconPause: SVGElement | null
iconVolume: SVGElement | null
iconMuted: SVGElement | null
skipSegmentBtn: HTMLButtonElement | null
subtitleText: HTMLElement | null
autoplayBtn: HTMLInputElement | null
}
let controlsCache: Controls | null = null
const getControls = (): Controls => {
if (controlsCache) return controlsCache
const c = state.container
controlsCache = {
playPause: c.querySelector('[data-play-pause]'),
muteBtn: c.querySelector('[data-mute]'),
volumePanel: c.querySelector('[data-volume-panel]'),
volumeRange: c.querySelector('[data-volume-range]'),
volumeUnderline: c.querySelector('[data-volume-underline]'),
backwardBtn: c.querySelector('[data-backward]'),
forwardBtn: c.querySelector('[data-forward]'),
fullscreenBtn: c.querySelector('[data-fullscreen]'),
iconPlay: c.querySelector('[data-icon-play]'),
iconPause: c.querySelector('[data-icon-pause]'),
iconVolume: c.querySelector('[data-icon-volume]'),
iconMuted: c.querySelector('[data-icon-muted]'),
skipSegmentBtn: c.querySelector('[data-skip]'),
subtitleText: c.querySelector('[data-subtitle-text]'),
autoplayBtn: document.querySelector('[data-autoplay]'),
}
return controlsCache
}
const updatePlayPauseIcons = (isPlaying: boolean): void => {
const { iconPlay, iconPause } = getControls()
iconPlay?.classList.toggle('hidden', isPlaying)
iconPause?.classList.toggle('hidden', !isPlaying)
}
const updateMuteIcons = (isMuted: boolean): void => {
const { iconVolume, iconMuted } = getControls()
iconVolume?.classList.toggle('hidden', isMuted)
iconMuted?.classList.toggle('hidden', !isMuted)
}
export const setupControls = (): void => {
const {
playPause, muteBtn, volumePanel, volumeRange,
backwardBtn, forwardBtn, fullscreenBtn, skipSegmentBtn,
} = getControls()
playPause?.addEventListener('click', () => { togglePlayPause(); showControls() })
state.video.addEventListener('click', () => { togglePlayPause(); showControls() })
muteBtn?.addEventListener('click', () => { toggleMute(); showControls() })
volumeRange?.addEventListener('input', () => {
const value = Number(volumeRange.value) / 100
setVolume(value)
showControls()
})
volumeRange?.addEventListener('pointerdown', () => volumePanel?.classList.add('is-dragging'))
window.addEventListener('pointerup', () => volumePanel?.classList.remove('is-dragging'))
backwardBtn?.addEventListener('click', () => seekBy(-10))
forwardBtn?.addEventListener('click', () => seekBy(10))
fullscreenBtn?.addEventListener('click', () => { toggleFullscreen(); showControls() })
skipSegmentBtn?.addEventListener('click', () => {
if (!state.activeSkipSegment) return
state.video.currentTime = state.activeSkipSegment.end + 0.01
showControls()
})
document.addEventListener('fullscreenchange', () => {
state.isFullscreen = !!document.fullscreenElement
state.container.classList.toggle('fullscreen', state.isFullscreen)
if (state.isFullscreen) showControls()
})
state.video.addEventListener('play', () => { updatePlayPauseIcons(true); showControls() })
state.video.addEventListener('pause', () => { updatePlayPauseIcons(false); showControls() })
state.video.addEventListener('volumechange', syncVolumeUI)
state.container.addEventListener('mousemove', showControls)
updatePlayPauseIcons(false)
syncVolumeUI()
}

58
static/player/keyboard.ts Normal file
View File

@@ -0,0 +1,58 @@
import { state } from './state'
import { displayTimeFromAbsolute, absoluteTimeFromDisplay, absoluteTimeFromRatio, getBounds } from './timeline'
import { showControls, toggleMute, togglePlayPause, toggleFullscreen, seekBy, setVolume, formatTime } from './controls'
export const setupKeyboard = (): void => {
document.addEventListener('keydown', (e) => {
const target = e.target as HTMLElement
if (target.tagName === 'INPUT' || target.tagName === 'TEXTAREA' || target.isContentEditable) return
switch (e.code) {
case 'Space':
case 'KeyK':
e.preventDefault()
togglePlayPause()
showControls()
break
case 'ArrowLeft':
case 'KeyJ':
e.preventDefault()
seekBy(-10)
break
case 'ArrowRight':
case 'KeyL':
e.preventDefault()
seekBy(10)
break
case 'ArrowUp':
e.preventDefault()
setVolume(state.video.volume + 0.05)
showControls()
break
case 'ArrowDown':
e.preventDefault()
setVolume(state.video.volume - 0.05)
showControls()
break
case 'KeyM':
e.preventDefault()
toggleMute()
showControls()
break
case 'KeyF':
e.preventDefault()
toggleFullscreen()
showControls()
break
default:
if (/^\d$/.test(e.key)) {
const b = getBounds()
if (b.duration > 0) {
e.preventDefault()
state.video.currentTime = absoluteTimeFromRatio(parseInt(e.key, 10) / 10)
showControls()
}
}
}
})
}

97
static/player/timeline.ts Normal file
View File

@@ -0,0 +1,97 @@
import { TimelineBounds } from './types'
import { state } from './state'
const formatTime = (seconds: number): string => {
if (!Number.isFinite(seconds) || seconds < 0) return '00:00'
const mins = Math.floor(seconds / 60)
const secs = Math.floor(seconds % 60)
return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`
}
let cachedBounds: TimelineBounds = { start: 0, end: 0, duration: 0 }
export const timelineBounds = (): TimelineBounds => {
const duration = Number.isFinite(state.video.duration) && state.video.duration > 0 ? state.video.duration : 0
let start = 0
if (state.video.seekable.length > 0) {
const seekableStart = state.video.seekable.start(0)
if (Number.isFinite(seekableStart) && seekableStart > 0) start = seekableStart
}
if (duration > start) {
return { start, end: duration, duration: duration - start }
}
if (state.video.seekable.length > 0) {
const seekableEnd = state.video.seekable.end(state.video.seekable.length - 1)
if (Number.isFinite(seekableEnd) && seekableEnd > start) {
return { start, end: seekableEnd, duration: seekableEnd - start }
}
}
return { start: 0, end: duration, duration }
}
export const invalidateBounds = (): void => {
cachedBounds = timelineBounds()
}
export const getBounds = (): TimelineBounds => cachedBounds
export const displayTimeFromAbsolute = (absoluteTime: number): number => {
const b = getBounds()
if (!Number.isFinite(absoluteTime) || b.duration <= 0) return 0
return Math.max(b.start, Math.min(b.end, absoluteTime)) - b.start
}
export const absoluteTimeFromDisplay = (displayTime: number): number => {
const b = getBounds()
if (!Number.isFinite(displayTime) || b.duration <= 0) return 0
return b.start + Math.max(0, Math.min(b.duration, displayTime))
}
export const absoluteTimeFromRatio = (ratio: number): number => {
const b = getBounds()
if (!Number.isFinite(ratio) || b.duration <= 0) return 0
return b.start + Math.max(0, Math.min(1, ratio)) * b.duration
}
export const getBufferedEnd = (): number => {
const currentTime = state.video.currentTime
let end = 0
for (let i = 0; i < state.video.buffered.length; i++) {
if (state.video.buffered.start(i) <= currentTime && state.video.buffered.end(i) >= currentTime) {
end = state.video.buffered.end(i)
break
}
}
if (end === 0) {
for (let i = 0; i < state.video.buffered.length; i++) {
if (state.video.buffered.end(i) > currentTime) {
end = Math.max(end, state.video.buffered.end(i))
}
}
}
return end
}
export const updateTimeline = (currentTime: number): void => {
const { progress, scrubber, timeDisplay, durationDisplay, buffered } = getTimelineEls()
const b = getBounds()
if (b.duration <= 0) {
progress.style.width = '0%'
buffered.style.width = '0%'
scrubber.style.left = '0%'
timeDisplay.textContent = '00:00'
durationDisplay.textContent = '00:00'
return
}
const pct = (displayTimeFromAbsolute(currentTime) / b.duration) * 100
progress.style.width = `${pct}%`
scrubber.style.left = `${pct}%`
timeDisplay.textContent = formatTime(displayTimeFromAbsolute(currentTime))
durationDisplay.textContent = formatTime(b.duration)
const bufferedEnd = getBufferedEnd()
const bufferedPct = (displayTimeFromAbsolute(bufferedEnd) / b.duration) * 100
buffered.style.width = `${bufferedPct}%`
}