【工具】JS脚本|网页任意视频倍速播放(包括MOOC、本地视频、其他的视频)
实际发布时间:2022-12-14 22:54:52。
csdn禁止浏览器脚本相关博客了,就只能重新发到这儿了。
2024/12/14更新:更新了常见问题Q&A,可以配合食用。
只要浏览器播放视频时用的是<video>
,而且当前页面只有这一个视频,就可以用这个脚本。
这是我很久以前为了倍速看MOOC视频,所以写的一个脚本,一直没有专门写博客去介绍它。刚刚,我想倍速看新闻联播,才发现忘记写博客了。
浏览器视频倍速播放的原理非常容易,我也是看了一下网页源代码才发现的。自从发现了这一点之后,我连本地视频都特别喜欢直接用浏览器播放。
脚本链接
Greasy Fork 链接:https://greasyfork.org/zh-CN/scripts/449828-任意视频倍速播放。
注:如果你的浏览器有可以运行 JS 脚本的扩展插件,那么点击即可安装。
脚本功能
PC端,任意浏览器视频倍速播放,按键调速。
其实除了倍速播放,还有调进度、调音量的功能。这是因为我觉得挺方便,就一块儿加上了。
功能清单
① +/-0.25倍速:按w/s键;
② 切换整数倍速:按1/2/3/4键;
③ +/-0.5s:按→/←键;
④ +/-10%音量:按↑/↓键。
以上键位均可配置,其中整数倍数我喜欢更高的倍数所以我也额外令它可以配置了。配置菜单如下:
注意事项
注:目前测试了mooc、新闻联播网站、b站视频、本地视频,都可以,如果有不可以的网站请反馈给我。
注:用浏览器播放本地视频,需要将扩展插件设置为“允许访问文件URL”。
部分网站的备注如下:
- b站:视频的w是投币,要么改快捷键,要么全屏之后才可以调倍速;
- 有倍速检测的网站:自行判断是否适合采用本脚本,本脚本带来的风险概不负责,如国家中小学智慧教育平台部分视频有倍速检测
- 百度网盘:百度网盘会自己锁定倍速,调了它会立马改回去,所以本脚本不适用。
脚本代码
// ==UserScript==
// @name 任意视频倍速播放
// @namespace http://tampermonkey.net/
// @version 0.5.3
// @description 任意浏览器视频倍速播放,按键调速。
// @author shandianchengzi
// @include *
// @icon 
// @license GPL3
// @grant GM_registerMenuCommand
// @grant GM_getValue
// @grant GM_setValue
// ==/UserScript==
// 默认按键配置
const DEFAULT_KEYS = {
speedUp: 'w',
speedDown: 's',
forward: 'ArrowRight',
backward: 'ArrowLeft',
volumeUp: 'ArrowUp',
volumeDown: 'ArrowDown',
normalSpeed: '1',
doubleSpeed: '2',
tripleSpeed: '3',
quadrupleSpeed: '4',
speedList: {
normalSpeed: 1.0,
doubleSpeed: 2.0,
tripleSpeed: 3.0,
quadrupleSpeed: 4.0
}
};
const JS_name = "VIDEOSPEED_keys";
// 获取配置
function getConfig() {
const savedKeys = GM_getValue(JS_name);
return savedKeys ? {...DEFAULT_KEYS, ...savedKeys} : DEFAULT_KEYS;
}
// 轻提醒
function Toast(msg, duration) {
duration = isNaN(duration) ? 3000 : duration;
var m = document.createElement('div');
m.innerHTML = msg;
m.style.cssText = "font-family:siyuan;max-width:60%;min-width: 150px;padding:0 14px;height: 40px;color: rgb(255, 255, 255);line-height: 40px;text-align: center;border-radius: 4px;position: fixed;top: 50%;left: 50%;transform: translate(-50%, -50%);z-index: 999999;background: rgba(0, 0, 0,.7);font-size: 16px;";
document.body.appendChild(m);
setTimeout(function() {
var d = 0.5;
m.style.webkitTransition = '-webkit-transform ' + d + 's ease-in, opacity ' + d + 's ease-in';
m.style.opacity = '0';
setTimeout(function() {
document.body.removeChild(m)
}, d * 1000);
}, duration);
}
// 配置快捷键
function configureShortcuts() {
const keys = getConfig();
const configWindow = document.createElement('div');
configWindow.style.position = 'fixed';
configWindow.style.top = '50%';
configWindow.style.left = '50%';
configWindow.style.transform = 'translate(-50%, -50%)';
configWindow.style.backgroundColor = '#fff';
configWindow.style.padding = '20px';
configWindow.style.border = '1px solid #ddd';
configWindow.style.borderRadius = '8px';
configWindow.style.boxShadow = '0 6px 12px rgba(0, 0, 0, 0.3)';
configWindow.style.zIndex = '10000';
configWindow.style.width = '500px';
configWindow.style.maxHeight = '80vh';
configWindow.style.overflowY = 'auto';
configWindow.style.fontFamily = 'Arial, sans-serif';
// 创建表单内容
let formHTML = `
<h3 style="margin-top: 0; font-size: 18px; color: #333; border-bottom: 1px solid #eee; padding-bottom: 10px;">
配置快捷键
</h3>
<div style="display: grid; grid-template-columns: 1fr 1fr; gap: 10px;">
`;
// 添加快捷键输入字段
const keyLabels = {
speedUp: '加速键 (默认: w)',
speedDown: '减速键 (默认: s)',
forward: '前进5秒键 (默认: →)',
backward: '后退5秒键 (默认: ←)',
volumeUp: '增加音量键 (默认: ↑)',
volumeDown: '降低音量键 (默认: ↓)'
};
const speedLabels = {
normalSpeed: '倍速键 (默认: 1)',
doubleSpeed: '倍速键 (默认: 2)',
tripleSpeed: '倍速键 (默认: 3)',
quadrupleSpeed: '倍速键 (默认: 4)'
};
for (const [key, label] of Object.entries(keyLabels)) {
formHTML += `
<label style="font-size: 14px; display: block; margin-bottom: 5px;">
${label}
<input type="text" id="${key}" value="${keys[key] || ''}"
style="width: 100%; padding: 5px; margin-top: 2px; border: 1px solid #ccc; border-radius: 4px;">
</label>
`;
}
for (const [key, label] of Object.entries(speedLabels)) {
formHTML += `
<label style="font-size: 14px; display: block; margin-bottom: 5px;">
<input type="text" id="${key}_value" value="${keys['speedList'][key] || ''}" style="width: 10%; padding: 5px; margin-top: 2px; border: 1px solid #ccc; border-radius: 4px;">
${label}
<input type="text" id="${key}" value="${keys[key] || ''}"
style="width: 100%; padding: 5px; margin-top: 2px; border: 1px solid #ccc; border-radius: 4px;">
</label>
`;
}
formHTML += `
</div>
<p>倍速调节范围:0.1~16<br>具体的键位与名称的对应关系可以查询:https://developer.mozilla.org/zh-CN/docs/Web/API/KeyboardEvent/code</p>
<div style="text-align: right; margin-top: 20px; border-top: 1px solid #eee; padding-top: 15px;">
<button id="saveShortcuts" style="padding: 8px 16px; margin-right: 10px; background-color: #4CAF50;
color: white; border: none; border-radius: 4px; cursor: pointer;">保存</button>
<button id="cancelShortcuts" style="padding: 8px 16px; background-color: #f44336;
color: white; border: none; border-radius: 4px; cursor: pointer;">取消</button>
</div>
`;
configWindow.innerHTML = formHTML;
document.body.appendChild(configWindow);
// 添加事件监听器
document.getElementById('saveShortcuts').addEventListener('click', () => {
const newKeys = {...keys};
const newSpeedList = {...keys.speedList};
// 更新普通按键
for (const key of Object.keys(keyLabels)) {
newKeys[key] = document.getElementById(key).value.trim() || DEFAULT_KEYS[key];
}
// 更新速度按键和对应的速度值
for (const key of Object.keys(speedLabels)) {
newKeys[key] = document.getElementById(key).value.trim() || DEFAULT_KEYS[key];
const speedValue = parseFloat(document.getElementById(`${key}_value`).value);
newSpeedList[key] = isNaN(speedValue) ? DEFAULT_KEYS.speedList[key] : speedValue;
}
newKeys.speedList = newSpeedList;
GM_setValue(JS_name, newKeys);
document.body.removeChild(configWindow);
Toast('快捷键配置已保存,刷新页面后生效', 2000);
});
document.getElementById('cancelShortcuts').addEventListener('click', () => {
document.body.removeChild(configWindow);
});
// 添加ESC键关闭功能
const closeOnEsc = (e) => {
if (e.key === 'Escape') {
document.body.removeChild(configWindow);
document.removeEventListener('keydown', closeOnEsc);
}
};
document.addEventListener('keydown', closeOnEsc);
}
// 显示当前配置
function showCurrentConfig() {
const keys = getConfig();
const configText = `
当前快捷键配置:
加速: ${keys.speedUp}
减速: ${keys.speedDown}
前进5秒: ${keys.forward}
后退5秒: ${keys.backward}
增加音量: ${keys.volumeUp}
降低音量: ${keys.volumeDown}
${keys.speedList.normalSpeed}倍速: ${keys.normalSpeed}
${keys.speedList.doubleSpeed}倍速: ${keys.doubleSpeed}
${keys.speedList.tripleSpeed}倍速: ${keys.tripleSpeed}
${keys.speedList.quadrupleSpeed}倍速: ${keys.quadrupleSpeed}
`.trim();
alert(configText);
}
// 重置为默认配置
function resetToDefault() {
if (confirm('确定要重置为默认快捷键配置吗?')) {
GM_setValue(JS_name, DEFAULT_KEYS);
Toast('已重置为默认配置,刷新页面后生效', 2000);
}
}
async function mainFunc(){
const keys = getConfig();
document.body.onkeydown = function(ev) {
var e = ev || event;
// 避免短时间重复触发同一按键
this.lastKey = this.lastKey || '';
this.lastTime = this.lastTime || 0;
if(this.lastKey === e.key) {
if((new Date()).getTime() - this.lastTime < 100) {
return;
}
}
this.lastKey = e.key;
this.lastTime = (new Date()).getTime();
let video = document.getElementsByTagName('video')[0];
if(video){
switch(e.key){
case keys.speedUp: // 加速
video.playbackRate += 0.25;
Toast(video.playbackRate.toFixed(2), 100);
break;
case keys.speedDown: // 减速
video.playbackRate -= 0.25;
Toast(video.playbackRate.toFixed(2), 100);
break;
case keys.forward: // 前进5秒
video.currentTime += 5;
break;
case keys.backward: // 后退5秒
video.currentTime -= 5;
break;
case keys.volumeUp: // 增加音量
video.volume = Math.min(video.volume + 0.1, 1);
Toast(video.volume.toFixed(1), 100);
break;
case keys.volumeDown: // 降低音量
video.volume = Math.max(video.volume - 0.1, 0);
Toast(video.volume.toFixed(1), 100);
break;
case keys.normalSpeed: // 正常速度
video.playbackRate = keys.speedList.normalSpeed;
Toast(video.playbackRate.toFixed(1), 100);
break;
case keys.doubleSpeed: // 2倍速
video.playbackRate = keys.speedList.doubleSpeed;
Toast(video.playbackRate.toFixed(1), 100);
break;
case keys.tripleSpeed: // 3倍速
video.playbackRate = keys.speedList.tripleSpeed;
Toast(video.playbackRate.toFixed(1), 100);
break;
case keys.quadrupleSpeed: // 4倍速
video.playbackRate = keys.speedList.quadrupleSpeed;
Toast(video.playbackRate.toFixed(1), 100);
break;
default:
return;
}
}
}
}
// 注册菜单命令
GM_registerMenuCommand('配置快捷键', configureShortcuts);
GM_registerMenuCommand('查看当前配置', showCurrentConfig);
GM_registerMenuCommand('重置为默认配置', resetToDefault);
(function() {
'use strict';
window.onhashchange=mainFunc;
mainFunc();
// 这个代码不管用,会导致所有的都运行不了。建议编辑脚本,设置-通用-仅在顶层页面(框架)运行-是
// if(window.top === window.self){
// }
})();
2025/10/15更新掉了,不再用keyCode了,所以以下内容废弃:
注意,如需修改脚本的快捷键,别用ASCII码表。
这个脚本的代码和按键的对应关系是用KeyCode
码表,常用键码值如下图:
对照表 - 一个工具箱
查看:http://www.atoolbox.net/Tool.php?Id=815。
最新配置方法如下:
如需修改脚本的快捷键,点击插件、找到对应脚本,然后点配置快捷键就可以了:
我提供的配置项在前面的脚本功能中介绍过了。
脚本原理
原理究极无敌简单,就是修改video
这个DOM元素的playbackRate
或其他属性。
唯一体现了一点点难度的,可能就只是调整之后弹出来的Toast
轻提醒,如下:
我今天上号发现安装量还挺不错的~!说明蛮多人都有按键播放需求。所以来补一个博客,引一下流(bushi)。
脚本建议
由于视频通常加载在页面的顶层框架,没有必要在其他层加载,为了防止脚本执行次数过多,建议编辑脚本并设置,使之仅仅运行在顶层框架上。
第一步,打开编辑面板:
第二步,设置-通用-仅在顶层页面(框架)运行-是:
脚本Q&A
2024/12/14更新
2025/10/15再次更新
脚本失效了怎么办?
请对照以下原因调整。
0 浏览器开发者模式没有打开
所有的油猴脚本执行都需要打开浏览器的开发者模式。如下图所示,下图为开启的状态:
1 版本原因
我有时的更新会导致脚本不可用,如果真的导致了不可用,请立即在Github Issue中提出,或者通过博客园发送私信反馈给我,我会快速修复或者回退,在Greasyfork留言我可能查看不及时。
你现在可以考虑更新到最新版本试一试问题是否仍然存在,如果仍然存在,你可以考虑退回到0.4或以前的版本,对于这种功能简单的脚本,越原始往往意味着越稳定。
2 网站原因
大部分网站我都测试过,除了百度网盘我明确测试了它具备反倍速修改的功能、并且目前尚未绕过之外,其他的网站如b站、爱奇艺都是可以改倍速的。若有其他不支持的网站,请直接在Github Issue中提出,如果有空我会帮忙测试是否真的不可用,如果真的不可用我会找时间做兼容适配。
3 热键占用
其他插件或脚本占用了wasd这几个按键则无法调倍速。
4 环境原因
若你的运行环境不在电脑上,而是在平板或手机上。我现在还没有测试过平板的功能支持性,所以可能平板上会有不兼容的情况。
抽时间我会对我的安卓和苹果平板都做一下测试并兼容,改好后我会更新这里的说明。
手机我很少用来运行脚本,因此不考虑做手机视频倍速的兼容适配。
5 窗口锚点
如果你的运行环境在电脑上,请确保你的窗口锚点落在浏览器窗口内,意思是确定你当前操作的界面是你要调速的界面。为了确保这一点,你可以在使用脚本功能之前,关闭画中画,点击视频暂停再点击播放,再按下wasd键进行调速。
6 运行失败
若你的运行环境在电脑上,可以打开浏览器的开发者工具并切换到控制台选项卡,按下wasd这些键盘后控制台中应该会显示test输出,如果显示了说明脚本被正常执行了,如果没有显示则会出现相应的报错,将报错截图通过Github Issue发给我,或者通过博客园发送私信。
其他问题
如果有其他问题,请在Github Issue中提出,或者通过博客园发送私信,这样我会有邮件提醒。
如果后续代码有更新,我会优先更新 Github 里的代码,所以最新代码请直接查看 Github 仓库。转载或修改请标明出处:https://greasyfork.org/zh-CN/scripts/449828-任意视频倍速播放。