Adds hotkeys and more to invidious player.
// ==UserScript==
// @name Enhanced Invidious Player
// @version 1.4.0
// @author NotYou
// @description Adds hotkeys and more to invidious player.
// @icon 
// @namespace -
// @match *://inv.nadeko.net/watch*
// @match *://yewtu.be/watch*
// @match *://invidious.nerdvpn.de/watch*
// @match *://invidious.f5.si/watch*
// @match *://inv.nadekonw7plitnjuawu6ytjsl7jlglk2t6pyq6eftptmiv3dvqndwvyd.onion/watch*
// @match *://nadekoohummkxncchcsylr3eku36ze4waq4kdrhcqupckc3pe5qq.b32.i2p/watch*
// @license GPL-3.0-or-later
// @run-at document-end
// @grant none
// ==/UserScript==
!function() {
class Keybinds {
constructor(element) {
this.element = element
this.keybinds = []
this.element.addEventListener('keydown', ev => {
this.keybinds.forEach(keybind => {
const confimationCallback = keybind[0]
const keybindCallback = keybind[1]
if (confimationCallback(ev)) {
keybindCallback(element, ev)
}
})
})
}
bind(confimationCallback, keybindCallback) {
this.keybinds.push([confimationCallback, keybindCallback])
return this
}
}
class Constants {
static VOLUME_STEP = 0.021
static SKIP_STEP = 15
}
class Main {
static setupKeybinds($video, $contents) {
const keybinds = new Keybinds($video)
.bind(ev => ev.code === 'KeyR', video => {
video.loop = video.loop ? false : true
})
.bind(ev => ev.code === 'KeyT', video => {
$contents.style.width = $contents.style.width === '' ? '100%' : ''
})
.bind(ev => ev.code === 'KeyC', (video, ev) => {
let data = ''
if (ev.ctrlKey) {
data = video.currentSrc
} else {
data = JSON.parse(document.querySelector('#player_data').textContent).title
}
navigator.clipboard.writeText(data)
})
.bind(ev => ev.code === 'ArrowRight', (video, ev) => {
if (ev.ctrlKey) {
if (video.currentTime + Constants.SKIP_STEP < video.duration) {
video.currentTime += Constants.SKIP_STEP
} else {
video.currentTime = video.duration
}
}
})
.bind(ev => ev.code === 'ArrowLeft', (video, ev) => {
if (ev.ctrlKey) {
if (video.currentTime - Constants.SKIP_STEP > 0) {
video.currentTime -= Constants.SKIP_STEP
} else {
video.currentTime = 0
}
}
})
.bind(ev => ev.code === 'Numpad0' || ev.code === 'Numpad1', video => {
video.playbackRate = 1
})
.bind(ev => ev.code === 'Numpad2', video => {
video.playbackRate = 2
})
.bind(ev => ev.code === 'Numpad3', video => {
video.playbackRate = 2.5
})
.bind(ev => ev.code === 'Numpad4', video => {
video.playbackRate = 3
})
const scriptsDisabled = Boolean(document.querySelector('#player'))
if (scriptsDisabled) {
const $player = document.querySelector('#player')
keybinds.bind(ev => ev.code === 'KeyM', video => {
video.muted = video.muted ? true : false
}).bind(ev => ev.code === 'KeyF', () => {
if ($player.requestFullscreen) {
$player.requestFullscreen()
} else if ($player.webkitRequestFullscreen) {
$player.webkitRequestFullscreen()
} else if ($player.msRequestFullscreen) {
$player.msRequestFullscreen()
}
})
}
}
static setupMouseControls($video) {
const isFullscreenEnabled = () => document.fullscreenElement !== null
$video.addEventListener('mouseenter', () => {
if (isFullscreenEnabled()) {
return
}
$video.focus()
document.documentElement.style.overflow = 'hidden'
})
$video.addEventListener('mouseleave', () => {
if (isFullscreenEnabled()) {
return
}
document.documentElement.style.overflow = ''
})
$video.addEventListener('blur', () => {
document.documentElement.style.overflow = ''
})
$video.addEventListener('wheel', ev => {
const wentDown = ev.deltaY >= 0
if (wentDown) {
if ($video.volume - Constants.VOLUME_STEP <= 0) {
$video.volume = 0
} else {
$video.volume -= Constants.VOLUME_STEP
}
} else {
if ($video.muted) {
$video.muted = false
}
if ($video.volume + Constants.VOLUME_STEP >= 1) {
$video.volume = 1
} else {
$video.volume += Constants.VOLUME_STEP
}
}
})
}
static init() {
const $video = document.querySelector('video')
const $contents = $video.parentNode.parentNode.parentNode
this.setupKeybinds($video, $contents)
this.setupMouseControls($video)
}
}
Main.init()
}()
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址