Greasy Fork

Greasy Fork is available in English.

HTML5 视频音频默认音量

避免被一些默认 100% 音量的视频/音频吓一跳(或社死)!且支持各网站分别记住音量...

当前为 2025-11-24 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         HTML5 Video Audio Default Volume
// @name:zh-CN   HTML5 视频音频默认音量
// @name:zh-TW   HTML5 視訊音訊預設音量
// @name:ru      Громкость аудио-видео в формате HTML5 по умолчанию
// @version      1.0.4
// @author       X.I.U
// @description  Avoid being startled by some video/audio with default 100% volume! And support each website to remember the volume separately...
// @description:zh-CN  避免被一些默认 100% 音量的视频/音频吓一跳(或社死)!且支持各网站分别记住音量...
// @description:zh-TW  避免被一些預設 100% 音量的視訊/音訊嚇一跳(或社死)!且支援各網站分別記住音量...
// @description:ru  Больше не пугайтесь при просмотре видео или прослушивании аудио со стандартной громкостью 100%! Так ещё каждый веб-сайт запоминает громкость отдельно...
// @match        *://*/*
// @icon         
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        GM_openInTab
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_notification
// @license      GPL-3.0 License
// @run-at       document-start
// @namespace    https://github.com/XIU2/UserScript
// ==/UserScript==

(function() {
    'use strict';
    var menu_ID = [];
    registerMenuCommand(); // 注册脚本菜单
    function registerMenuCommand() {
        if (self != top) return
        // 如果菜单ID数组多于菜单数组,说明不是首次添加菜单,需要卸载所有脚本菜单
        if (menu_ID.length > 0){for (let i=0;i<menu_ID.length;i++){GM_unregisterMenuCommand(menu_ID[i]);}}
        menu_ID[0] = GM_registerMenuCommand('#️⃣ 修改全局默认音量 [ ' + GM_getValue('menu_defaultVolume', 30) + '% ]', function(){customDefaultVolume()});
        let nowVolume = ' (跟随全局)'
        if (localStorage.getItem('html5_xiu_currentVolume')) nowVolume = ' [ ' + parseInt(localStorage.getItem('html5_xiu_currentVolume')) + '% ]'
        menu_ID[1] = GM_registerMenuCommand('🔁 忘记当前网站音量' + nowVolume, function(){resetCurrentVolume()});
        // 强制当前网站使用全局音量(针对部分不支持调节音量的网站)
        if (menu_forcedToEnable('check')) { // 当前网站是否已存在强制列表中
            menu_ID[2] = GM_registerMenuCommand('✅ 已强制当前网站使用全局音量 (针对不支持调节音量的)', function(){menu_forcedToEnable('del')});
            menu_ID[4] = GM_registerMenuCommand('#️⃣ 修改当前网站默认音量 (针对不支持调节音量的)', function(){customCurrentDefaultVolume()});
        } else {
            menu_ID[2] = GM_registerMenuCommand('❌ 未强制当前网站使用全局音量 (针对不支持调节音量的)', function(){menu_forcedToEnable('add')});
        }
        menu_ID[3] = GM_registerMenuCommand('💬 反馈 & 建议', function () {GM_openInTab('https://github.com/XIU2/UserScript#xiu2userscript', {active: true,insert: true,setParent: true}); GM_openInTab('http://greasyfork.icu/zh-CN/scripts/438400/feedback', {active: true,insert: true,setParent: true});});
    }

    insPage();
    currentPage();

    // 强制当前网站使用全局音量(针对部分不支持调节音量的网站)
    function menu_forcedToEnable(type) {
        switch(type) {
            case 'check':
                if(check()) return true
                return false
                break;
            case 'add':
                add();
                break;
            case 'del':
                del();
                break;
        }

        function check() { // 存在返回真,不存在返回假
            let websiteList = GM_getValue('menu_forcedToEnable',[]); // 读取网站列表
            if (websiteList.indexOf(location.host) === -1) return false // 不存在返回假
            return true
        }

        function add() {
            if (check()) return
            let websiteList = GM_getValue('menu_forcedToEnable',[]); // 读取网站列表
            websiteList.push(location.host); // 追加网站域名
            GM_setValue('menu_forcedToEnable', websiteList); // 写入配置
            location.reload(); // 刷新网页
        }

        function del() {
            if (!check()) return
            let websiteList = GM_getValue('menu_forcedToEnable',[]), // 读取网站列表
            index = websiteList.indexOf(location.host);
            websiteList.splice(index, 1); // 删除网站域名
            GM_setValue('menu_forcedToEnable', websiteList); // 写入配置
            location.reload(); // 刷新网页
        }
    }


    // 网页本身的 Video Audio 标签
    function currentPage() {
        document.querySelectorAll('video, audio').forEach(function(_this){
            //console.log('网页本身:', _this)
            isFirstEvent(_this)
        })
    }


    // 后续动态插入的 Video Audio 标签(插入事件)
    function insPage() {
        const callback = (mutationsList, observer) => {
            for (const mutation of mutationsList) {
                for (const target of mutation.addedNodes) {
                    if (target.nodeType != 1) break
                    //console.log('所有插入:', target)
                    if (target.tagName === 'VIDEO' || target.tagName === 'AUDIO') {
                        //console.log('后续插入1:', target)
                        isFirstEvent(target)
                    } else if (target.tagName === 'DIV') {
                        target.querySelectorAll('video, audio').forEach(function(_this){
                            //console.log('后续插入2:', _this)
                            isFirstEvent(_this)
                        })
                    }
                }
            }
        };
        const observer = new MutationObserver(callback);
        observer.observe(document, { childList: true, subtree: true });
    }


    // 音量变化事件(记住音量)
    function volumeChangeEvent(event) {
        if (event.target.muted) { // 判断是否静音
            localStorage.setItem('html5_xiu_currentVolume', 0)
            registerMenuCommand(); // 修改脚本菜单
        } else if (localStorage.getItem('html5_xiu_currentVolume') || ((event.target.volume * 100) !== GM_getValue('menu_defaultVolume', 30))) {
            localStorage.setItem('html5_xiu_currentVolume', event.target.volume * 100)
            registerMenuCommand(); // 修改脚本菜单
        }
    }


    // 判断该视频/音频元素是否已监听事件
    function isFirstEvent(target) {
        if (!menu_forcedToEnable('check')) { // 如果未强制当前网站使用全局音量(针对部分不支持调节音量的网站)
            if (!target.controls) return; // 如果视频/音频已经有了自己的控件(即没有使用 HTML5 默认的控件),则退出
        }
        modifyVolume(target);
        // 如果没有该属性,则代表是还未监听事件
        if (target.dataset.html5VolumeXiu != 'true') {
            target.dataset.html5VolumeXiu = 'true'
            target.addEventListener('volumechange', volumeChangeEvent);
        }
    }


    // 修改视频音量
    function modifyVolume(_this) {
        let nowVolume = parseFloat(localStorage.getItem('html5_xiu_currentVolume')); // 先看看 localStorage 有没有(即用户是否手动调整过音量)
        if (!nowVolume && nowVolume !== 0) nowVolume = GM_getValue('menu_defaultVolume', 30); // 如果 localStorage 没有,那就从脚本配置中获取
        if (!((typeof nowVolume === 'number') && nowVolume <= 100)) nowVolume = 30; // 如果获取到的音量数值不是数字,或大于 100,则重置为 30
        //console.log(_this, _this.volume, nowVolume, nowVolume / 100)
        _this.volume = nowVolume / 100; // 设置音量为 0.00~1.00 范围
        //console.log(_this.volume)
    }


    // 修改全局默认音量
    function customDefaultVolume() {
        let newValue = parseFloat(prompt('修改全局默认音量,不影响各网站记住的音量,修改后当前网页立即生效~\n范围:0~100 (即 0%~100%,不需要加 % 百分号)\n默认:30', GM_getValue('menu_defaultVolume', 30)));
        if (!Number.isNaN(newValue) && newValue >= 0 && newValue <= 100) {GM_setValue('menu_defaultVolume', newValue);}
        currentPage(); // 重置当前网页的音量
        registerMenuCommand(); // 重新注册菜单(刷新菜单上的音量值)
    }


    // 修改当前网站默认音量 (针对不支持调节音量的网站)
    function customCurrentDefaultVolume() {
        let newValue = parseFloat(prompt('修改当前网站默认音量 (针对不支持调节音量的网站),修改后立即生效~\n范围:0~100 (即 0%~100%,不需要加 % 百分号)\n默认:全局默认音量', localStorage.getItem('html5_xiu_currentVolume') || GM_getValue('menu_defaultVolume', 30)));
        if (!Number.isNaN(newValue) && newValue >= 0 && newValue <= 100) {localStorage.setItem('html5_xiu_currentVolume', newValue);}
        currentPage(); // 重置当前网页的音量
        registerMenuCommand(); // 重新注册菜单(刷新菜单上的音量值)
    }


    // 忘记当前网站音量
    function resetCurrentVolume() {
        if (localStorage.getItem('html5_xiu_currentVolume')) localStorage.removeItem('html5_xiu_currentVolume') // 清理 localStorage
        currentPage(); // 重置当前网页的音量
        registerMenuCommand(); // 重新注册菜单(刷新菜单上的音量值)
    }
})();