【工具】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%音量:按↑/↓键。

以上键位均可配置,其中整数倍数我喜欢更高的倍数所以我也额外令它可以配置了。配置菜单如下:
image

注意事项

注:目前测试了mooc、新闻联播网站、b站视频、本地视频,都可以,如果有不可以的网站请反馈给我。

注:用浏览器播放本地视频,需要将扩展插件设置为“允许访问文件URL”。

部分网站的备注如下:

  1. b站:视频的w是投币,要么改快捷键,要么全屏之后才可以调倍速;
  2. 有倍速检测的网站:自行判断是否适合采用本脚本,本脚本带来的风险概不负责,如国家中小学智慧教育平台部分视频有倍速检测
  3. 百度网盘:百度网盘会自己锁定倍速,调了它会立马改回去,所以本脚本不适用。

脚本代码

// ==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码表,常用键码值如下图:
![](https://img2023.cnblogs.com/blog/2208802/202307/2208802-20230730190159811-2549427.png

完整的表格可点击链接键码(KeyCode)对照表 - 一个工具箱查看:http://www.atoolbox.net/Tool.php?Id=815

最新配置方法如下:
如需修改脚本的快捷键,点击插件、找到对应脚本,然后点配置快捷键就可以了:
image

我提供的配置项在前面的脚本功能中介绍过了。

脚本原理

  原理究极无敌简单,就是修改video这个DOM元素的playbackRate或其他属性。
  唯一体现了一点点难度的,可能就只是调整之后弹出来的Toast轻提醒,如下:

在这里插入图片描述

  我今天上号发现安装量还挺不错的~!说明蛮多人都有按键播放需求。所以来补一个博客,引一下流(bushi)。

在这里插入图片描述

脚本建议

由于视频通常加载在页面的顶层框架,没有必要在其他层加载,为了防止脚本执行次数过多,建议编辑脚本并设置,使之仅仅运行在顶层框架上。
第一步,打开编辑面板:
image
第二步,设置-通用-仅在顶层页面(框架)运行-是:
image

脚本Q&A

2024/12/14更新
2025/10/15再次更新

脚本失效了怎么办?

请对照以下原因调整。

0 浏览器开发者模式没有打开

所有的油猴脚本执行都需要打开浏览器的开发者模式。如下图所示,下图为开启的状态:
image

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-任意视频倍速播放

posted @ 2023-07-30 19:06  shandianchengzi  阅读(7963)  评论(0)    收藏  举报