您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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() }()