bilibili倍速快捷键(增强)+记忆

bilibili倍速快捷键+记忆(可自定义设置快捷键,自定义一个额外倍速按钮,设置记忆模式)

目前為 2023-02-19 提交的版本,檢視 最新版本

// ==UserScript==
// @name         bilibili倍速快捷键(增强)+记忆
// @namespace    tonyu_balabala_03e6ea
// @version      1.0
// @description  bilibili倍速快捷键+记忆(可自定义设置快捷键,自定义一个额外倍速按钮,设置记忆模式)
// @author       Tony
// @icon        
// @license       MIT
// @match        *://*.bilibili.com/*
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// ==/UserScript==

(function() {
    'use strict';
    let hasObserve = false
    let count = 0
    let defaultSpd = 1
    let callbackTimer
    let spdMemoryMode = GM_getValue('tony_spd_memory_mode', '2')
    let spdKCSettings = []
    let spdKCSettingKeys = ['tony_spd2_KC', 'tony_spd1p5_KC', 'tony_spd1p25_KC', 'tony_spd1_KC', 'tony_spd0p75_KC', 'tony_spd0p5_KC', 'tony_spdcus_KC']
    let spdKCSettingDefaultVals = [66, 86, 67, 71, 72, 74, 75]
    let spdKNames = []
    let spdKNameKeys = ['tony_spd2_KC_name', 'tony_spd1p5_KC_name', 'tony_spd1p25_KC_name', 'tony_spd1_KC_name', 'tony_spd0p75_KC_name', 'tony_spd0p5_KC_name', 'tony_spdcus_KC_name']
    let spdKNameDefaultVals = ['b', 'v', 'c', 'g', 'h', 'j', 'k']
    // spd2_KC_enable, spd1p5_KC_enable, spd1p25_KC_enable, spd1_KC_enable, spd0p75_KC_enable, spd0p5_KC_enable
    let spdEnableSettings = []
    let spdEnableSettingKeys = ['tony_spd2_KC_enable', 'tony_spd1p5_KC_enable', 'tony_spd1p25_KC_enable', 'tony_spd1_KC_enable', 'tony_spd0p75_KC_enable', 'tony_spd0p5_KC_enable', 'tony_spdcus_KC_enable']
    let spdEnableSettingDefaultVals = [true, true, false, false, false, false, true]
    for(let i =0;i<spdKCSettingKeys.length;i++) {
        spdKCSettings.push(GM_getValue(spdKCSettingKeys[i], spdKCSettingDefaultVals[i]))
    }
    for(let i =0;i<spdKNameKeys.length;i++) {
        spdKNames.push(GM_getValue(spdKNameKeys[i], spdKNameDefaultVals[i]))
    }
    for(let i= 0;i<spdEnableSettingKeys.length;i++) {
        spdEnableSettings.push(GM_getValue(spdEnableSettingKeys[i], spdEnableSettingDefaultVals[i]))
    }
    let cusSpd = GM_getValue('tony_spd_cus', 3)
    let speedList = [2, 1.5, 1.25, 1, 0.75, 0.5]
    speedList.push(cusSpd)
    let spd = GM_getValue('tony_spd', 1)
    let bvdos = getVdos()
    if(bvdos.length>0) defaultSpd = bvdos[0].playbackRate
    function getVdos() {
        let vdos = document.getElementsByTagName('bwp-video')
        if(vdos.length === 0) {
            vdos = document.getElementsByTagName('video')
        }
        return vdos
    }
    function getBeisuList() {
        let beisuList = document.getElementsByClassName("bilibili-player-video-btn-speed-menu-list")
        if(beisuList.length==0) {
            beisuList = document.querySelectorAll(".squirtle-speed-select-list>.squirtle-select-item ")
        }
        if(beisuList.length==0) {
            beisuList = document.querySelectorAll(".edu-player-speed-list>.edu-player-speed-item")
        }
        if(beisuList.length==0) {
            beisuList = document.querySelectorAll(".bpx-player-ctrl-playbackrate-menu>.bpx-player-ctrl-playbackrate-menu-item")
        }
        return beisuList
    }
    function changeSpeed(beisuList, spd) {
        switch(spd) {
            case 2:
                beisuList[0].click()
                break
            case 1.5:
                beisuList[1].click()
                break
            case 1.25:
                beisuList[2].click()
                break
            case 1:
                beisuList[3].click()
                break
            case 0.75:
                beisuList[4].click()
                break
            case 0.5:
                beisuList[5].click()
                break
            case cusSpd:
                beisuList[6].click()
                break
            default:
                break
        }
    }
    function setPlaybackRate1(beisuList, vdo1) {
        let intervalCount = 0
        let myInterval
        myInterval = setInterval(() => {
            intervalCount++
            if(intervalCount>40 || vdo1.playbackRate === 1) {
                // 有时候直接点击当前倍速按钮并没有效果,通过先点击1倍速按钮,再点击其他倍速按钮来实现。
                let temSpd = spd
                changeSpeed(beisuList, 1)
                changeSpeed(beisuList, temSpd)
                intervalCount = 0
                clearInterval(myInterval)
            }
        },100)
    }
    function initObserver() {
        let vdos = getVdos()
        let config = {attributes: true, attributeFilter:['src']}
//         let listWrapper = document.getElementsByClassName('list-wrapper')[0]
//         let vdoTitle = document.getElementsByClassName('video-title')[0]
//         let config1 = {attributes: true, attributeFilter:['class'], subtree: true}
//         let config2 = {childList: true}
        function callback(mutationsList, observer) {
            clearTimeout(callbackTimer)
            callbackTimer = setTimeout(function() {
                let vdo1s = getVdos()
                if(vdo1s.length>0) {
                    let vdo1 = vdo1s[0]
                    let beisuList = getBeisuList()
                    if(beisuList.length>0) {
                        function vdoPlayHandler() {
                            changeSpeed(beisuList, spd)
                            vdo1.removeEventListener('play',vdoPlayHandler)
                        }
                        vdo1.addEventListener('play', vdoPlayHandler)
                    }
                }
            },100)
        }
        if(vdos.length>0) {
            const observer = new MutationObserver(callback)
            observer.observe(vdos[0], config)
            hasObserve = true
        }
//         if(listWrapper) {
//             const observer1 = new MutationObserver(callback)
//             observer1.observe(listWrapper, config1)
//             hasObserve = true
//         } else if(vdoTitle) {
//             const observer2 = new MutationObserver(callback)
//             observer2.observe(vdoTitle, config2)
//             hasObserve = true
//         }
        count++
        if(count<=20 && !hasObserve) setTimeout(initObserver, 1000)
    }
    //initObserver()

    // 全局记忆
    if(spdMemoryMode==='2') defaultSpd = spd
    else {
        spd = defaultSpd
        GM_setValue('tony_spd',defaultSpd)
    }
    let cusBeisuMenu
    let initSpdCnt = 0
    let hasAddVdoListener = false
    let hasAddMenuListener = false
    let oldSrc = ''
    let vdoPlayTimer
    function initSpeedAndListener() {
        let beisuList = getBeisuList()
        initSpdCnt++
        let vdos
        vdos = getVdos()
        if(spdMemoryMode === '1' || spdMemoryMode === '2' || spdMemoryMode ==='3') {
            // 尝试持续监听视频play来判断切换
            if(vdos.length>0 && !hasAddVdoListener) {
                vdos[0].addEventListener('play', function() {
                    clearTimeout(vdoPlayTimer)
                    vdoPlayTimer = setTimeout(function() {
                        //视频开始播放
                        if(oldSrc !== vdos[0].src) {
                            let bsList = getBeisuList()
                            if(bsList.length>0) {
                                // 不知什么原因,部分场景下设置为其他倍速,比如2倍速,切换视频后显示依然是2倍速,但是视频确实1倍速,通过再次点击2倍速来纠正
                                // 每次初始倍速为1
                                if(spdMemoryMode === '3') spd = 1
                                // 默认 当前页面记忆倍速(页面刷新丢失记忆) 或 全局记忆
                                changeSpeed(bsList, spd)
                                setPlaybackRate1(bsList, vdos[0])
                            }
                            oldSrc = vdos[0].src
                        }
                    },100)
                })
                vdos[0].addEventListener('canplay', function() {
                    //if(beisuList.length>0)
                })
                hasAddVdoListener = true
            }
        } else if(vdos.length>0 && !hasAddVdoListener) {
            hasAddVdoListener = true
        }
        if(beisuList.length>0 && !hasAddMenuListener) {
            cusBeisuMenu = beisuList[0].cloneNode(true)
            let settingMenu = document.createElement('li')
            settingMenu.setAttribute('style', 'text-align: -webkit-match-parent;position: relative;height: 36px;line-height: 36px;cursor: pointer;margin:0;padding:0;')
            settingMenu.innerText = '设置'
            settingMenu.addEventListener('click',showSetting)
            cusBeisuMenu.setAttribute('data-value', cusSpd)
            cusBeisuMenu.innerText = cusSpd%1>0?(cusSpd+'x'):(cusSpd+'.0x')
            beisuList[0].parentNode.appendChild(cusBeisuMenu)
            beisuList[0].parentNode.insertBefore(settingMenu, beisuList[0])
            beisuList = getBeisuList()
            function getMenuClickHandler(i) {
                return function() {
                    spd = speedList[i]
                    //console.log('spd', spd)
                    GM_setValue('tony_spd',spd)
                }
            }
            for(let i = 0;i<beisuList.length;i++) {
                beisuList[i].addEventListener('click', getMenuClickHandler(i))
            }
            hasAddMenuListener = true
            changeSpeed(beisuList, spd)
        }
        if(initSpdCnt<=40 && (!hasAddMenuListener || !hasAddVdoListener)) {
            setTimeout(initSpeedAndListener,500)
        }
        if(hasAddMenuListener && hasAddVdoListener) {
            setPlaybackRate1(beisuList, vdos[0])
        }
    }
    initSpeedAndListener()

    // 节流标志
    let flag = false
    document.addEventListener('keydown', function(e){
        if(flag) return
        flag = true
        setTimeout(() => {
            flag = false
        },100)
        let beisuList = getBeisuList()
        for(let i=0;i<spdKCSettings.length;i++) {
            if(e.keyCode == spdKCSettings[i] && beisuList.length!==0) {
                if(spdEnableSettings[i]) {
                    if(spd === speedList[i]) {
                        beisuList[3].click()
                    } else {
                        beisuList[i].click()
                    }
                }
                break
            }
        }
    });
    let hasSettingShow = false
    function showSetting() {
        if(hasSettingShow) return
        let settingBox = document.createElement("div");
        settingBox.setAttribute("style", `position: fixed !important;
        top: 10px !important;
        left: 50% !important;
        z-index:2147483647 !important;
        `)
        let contentBox = document.createElement('div')
        contentBox.setAttribute("style", `box-shadow: 0 0 10px rgba(100, 100, 100, 0.2) !important;
        max-height: calc(100vh - 60px) !important;
        max-width: 80vw !important;
        overflow: auto !important;
        background-color: rgb(235, 235, 235) !important;
        text-align: center !important;
        font-size: 13px !important;
        border-radius: 16px !important;
        white-space: nowrap;
        padding: 10px 16px 10px 16px !important;
        margin-left: -50% !important;
        margin-right: 50% !important;
        z-index:2147483647 !important;
        box-sizing: content-box !important;
        border: 2px solid rgb(100 100 100 / 10%) !important;
        `)
        if('backdropFilter' in document.documentElement.style) {
            contentBox.style.backgroundColor = 'rgba(235, 235, 235, 0.8)'
            contentBox.style.backdropFilter = 'saturate(50%) blur(14px)'
            //contentBox.style.transform = 'translateZ(0)'
        }
        if(getComputedStyle) {
            let fontFml = getComputedStyle(document.body, null)['font-family']
            if(fontFml) contentBox.style.fontFamily = fontFml
        }
//         let logoDiv = document.createElement('div')
//         logoDiv.setAttribute("style", 'margin-bottom: 6px !important;')
//         contentBox.appendChild(logoDiv)
        let settingContent = document.createElement('div')
        settingContent.setAttribute('style', 'font-weight: 600 !important;')

        let MemorySettingTitle = document.createElement('div')
        MemorySettingTitle.setAttribute('style', 'margin-top:10px;')
        MemorySettingTitle.innerText = '倍速记忆模式'
        settingContent.appendChild(MemorySettingTitle)

        let spdMemoryModeSelector = document.createElement('select')
        spdMemoryModeSelector.setAttribute('style', 'margin-top:10px;padding:3px 10px;border:2px solid rgb(168,168,168);border-radius:10px;')
        let spdMemoryModeOption1 = document.createElement('option')
        spdMemoryModeOption1.value = '1'
        spdMemoryModeOption1.innerText = '仅当前页面记忆倍速'
        spdMemoryModeOption1.selected = spdMemoryMode==='1'
        let spdMemoryModeOption2 = document.createElement('option')
        spdMemoryModeOption2.value = '2'
        spdMemoryModeOption2.innerText = '全局记忆倍速'
        spdMemoryModeOption2.selected = spdMemoryMode==='2'
        let spdMemoryModeOption3 = document.createElement('option')
        spdMemoryModeOption3.value = '3'
        spdMemoryModeOption3.innerText = '每次初始倍速为1(不记忆)'
        spdMemoryModeOption3.selected = spdMemoryMode==='3'
        let spdMemoryModeOption4 = document.createElement('option')
        spdMemoryModeOption4.value = '4'
        spdMemoryModeOption4.innerText = '什么都不做'
        spdMemoryModeOption4.selected = spdMemoryMode==='4'
        spdMemoryModeSelector.appendChild(spdMemoryModeOption1)
        spdMemoryModeSelector.appendChild(spdMemoryModeOption2)
        spdMemoryModeSelector.appendChild(spdMemoryModeOption3)
        spdMemoryModeSelector.appendChild(spdMemoryModeOption4)
        spdMemoryModeSelector.addEventListener('change', function() {
            spdMemoryMode = spdMemoryModeSelector.value
            GM_setValue('tony_spd_memory_mode', spdMemoryMode)
        })
        settingContent.appendChild(spdMemoryModeSelector)

        let cusSpdTitle = document.createElement('div')
        cusSpdTitle.setAttribute('style', 'margin-top:10px;')
        cusSpdTitle.innerText = '自定义倍速键设置(0~4)'
        settingContent.appendChild(cusSpdTitle)

        let cusSpdSettingBox = document.createElement('div')
        cusSpdSettingBox.setAttribute('style', 'margin-top:10px;')
        let cusSpdSettingBtnBox = document.createElement('div')
        cusSpdSettingBtnBox.setAttribute('style', 'display:none;margin-top:8px;')
        let cusSpdSettingBtnOK = document.createElement('button')
        let cusSpdSettingBtnCancel = document.createElement('button')
        let cusSpdInput = document.createElement('input')
        cusSpdInput.setAttribute('type', 'number')
        cusSpdInput.setAttribute('style', 'max-width:100px;padding:5px 10px;border:2px solid rgb(168,168,168);border-radius:10px;')
        cusSpdInput.value = cusSpd
        let oldCusSpd
        cusSpdInput.addEventListener('focus', function() {
            oldCusSpd = cusSpd
            cusSpdSettingBtnBox.style.display = 'block'
        })

        cusSpdSettingBtnOK.innerText = '确认'
        cusSpdSettingBtnOK.setAttribute('style', 'padding:8px 10px;border:0;color:white;background-color:rgb(25,207,20);border-radius:28px;')
        cusSpdSettingBtnOK.addEventListener('click', function() {
            let temCusSpd = parseFloat(cusSpdInput.value)
            if(!temCusSpd || temCusSpd<=0 || temCusSpd>4) {
                cusSpdInput.value = cusSpd
                cusSpdSettingBtnBox.style.display = 'none'
                return
            }
            cusSpd = temCusSpd
            GM_setValue('tony_spd_cus', cusSpd)
            speedList[speedList.length-1] = cusSpd
            if(cusBeisuMenu) {
                cusBeisuMenu.setAttribute('data-value', cusSpd)
                cusBeisuMenu.innerText = cusSpd%1>0?(cusSpd+'x'):(cusSpd+'.0x')
            }
            speedSettingBtns[speedSettingBtns.length-1].innerText = cusSpd
            if(speedSettingBtns[speedSettingBtns.length-1].style.backgroundColor === 'rgb(50, 130, 236)') speedSettingBtns[speedSettingBtns.length-1].click()
            cusSpdSettingBtnBox.style.display = 'none'
        })

        cusSpdSettingBtnCancel.innerText = '取消'
        cusSpdSettingBtnCancel.setAttribute('style', 'margin-right:6px;padding:8px 10px;border:0;color:white;background-color:rgb(243,97,128);border-radius:28px;')
        cusSpdSettingBtnCancel.addEventListener('click', function() {
            cusSpdInput.value = cusSpd
            cusSpdSettingBtnBox.style.display = 'none'
        })
        cusSpdSettingBox.appendChild(cusSpdInput)
        cusSpdSettingBtnBox.appendChild(cusSpdSettingBtnCancel)
        cusSpdSettingBtnBox.appendChild(cusSpdSettingBtnOK)
        cusSpdSettingBox.appendChild(cusSpdSettingBtnBox)
        settingContent.appendChild(cusSpdSettingBox)

        let KCSettingTitle = document.createElement('div')
        KCSettingTitle.setAttribute('style', 'margin-top:10px;')
        KCSettingTitle.innerText = '倍速快捷键设置'
        settingContent.appendChild(KCSettingTitle)

        let KCBox = document.createElement('div')
        KCBox.setAttribute('style', 'margin-top:10px;')

        let settingBtnBox = document.createElement('div')
        settingBtnBox.setAttribute('style', 'margin-top:10px;')
        let speedSettingBtns = []
        for(let i = 0; i < speedList.length; i++) {
            let speedSettingBtn = document.createElement('button')
            speedSettingBtns.push(speedSettingBtn)
            speedSettingBtn.setAttribute('style', 'margin:0 3px;padding:8px 10px;min-width:38px;border:0;color:white;background-color:rgb(100,100,100);border-radius:28px;')
            speedSettingBtn.innerText = speedList[i]
            speedSettingBtn.addEventListener('click', function() {
                for(let j = 0; j<speedList.length;j++) {
                    if(j===i) speedSettingBtns[j].style.backgroundColor = 'rgb(50,130,236)'
                    else speedSettingBtns[j].style.backgroundColor = 'rgb(100,100,100)'
                }

                let KCSwitch = document.createElement('input')
                KCSwitch.setAttribute('style', 'vertical-align: middle;')
                let switchTip = document.createElement('span')
                switchTip.setAttribute('style', 'vertical-align: middle;')
                KCSwitch.setAttribute('type', 'checkbox')
                KCSwitch.checked = spdEnableSettings[i]
                switchTip.innerText = speedList[i]+'倍速快捷键 '+(spdEnableSettings[i]?'启用':'关闭')
                KCSwitch.addEventListener('change', function() {
                    spdEnableSettings[i] = KCSwitch.checked
                    switchTip.innerText = speedList[i]+'倍速快捷键 '+(spdEnableSettings[i]?'启用':'关闭')
                    GM_setValue(spdEnableSettingKeys[i], KCSwitch.checked)
                })

                let KCSettingBox = document.createElement('div')
                KCSettingBox.setAttribute('style', 'margin-top:10px;')

                let KNameTitle = document.createElement('span')
                KNameTitle.innerText = speedList[i]+'倍速快捷键(点击修改→):'
                KNameTitle.setAttribute('style', 'margin-right:6px;')
                let KCSettingBtnBox = document.createElement('div')
                KCSettingBtnBox.setAttribute('style', 'display:none;margin-top:8px;')
                let KCSettingBtnOK = document.createElement('button')
                let KCSettingBtnCancel = document.createElement('button')
                let KName = document.createElement('button')
                KName.innerText = spdKNames[i]
                KName.setAttribute('style', 'padding:8px 16px;border-radius:28px;color:rgb(5,107,0);background-color:rgb(240,240,240);border:1px solid rgb(100,100,100);')
                let clickSpdKC = spdKCSettings[i]
                let clickSpdKName = spdKNames[i]
                function keyupHandler(e) {
                    e.stopPropagation()
                    clickSpdKC = e.keyCode
                    clickSpdKName = e.key
                    KName.innerText = clickSpdKName
                }
                function kNameClickHandler() {
                    clickSpdKC = spdKCSettings[i]
                    clickSpdKName = spdKNames[i]
                    KName.innerText = '请按键ヾ(•ω•`)o'
                    KName.addEventListener('keydown', keyupHandler)
                    KName.removeEventListener('click', kNameClickHandler)
                    KCSettingBtnBox.style.display = 'block'
                    KName.focus()
                }
                KName.addEventListener('click', kNameClickHandler)

                KCSettingBtnOK.innerText = '确认'
                KCSettingBtnOK.setAttribute('style', 'padding:8px 10px;border:0;color:white;background-color:rgb(25,207,20);border-radius:28px;')
                KCSettingBtnOK.addEventListener('click', function() {
                    spdKCSettings[i] = clickSpdKC
                    spdKNames[i] = clickSpdKName
                    GM_setValue(spdKCSettingKeys[i], clickSpdKC)
                    GM_setValue(spdKNameKeys[i], clickSpdKName)
                    KName.innerText = spdKNames[i]
                    KName.removeEventListener('keydown', keyupHandler)
                    KName.addEventListener('click', kNameClickHandler)
                    KCSettingBtnBox.style.display = 'none'
                })

                KCSettingBtnCancel.innerText = '取消'
                KCSettingBtnCancel.setAttribute('style', 'margin-right:6px;padding:8px 10px;border:0;color:white;background-color:rgb(243,97,128);border-radius:28px;')
                KCSettingBtnCancel.addEventListener('click', function() {
                    KName.innerText = spdKNames[i]
//                     let clickSpdKC = spdKCSettings[i]
//                     let clickSpdKName = spdKNames[i]
                    KName.removeEventListener('keydown', keyupHandler)
                    KName.addEventListener('click', kNameClickHandler)
                    KCSettingBtnBox.style.display = 'none'
                })
                KCSettingBox.appendChild(KNameTitle)
                KCSettingBox.appendChild(KName)
                KCSettingBtnBox.appendChild(KCSettingBtnCancel)
                KCSettingBtnBox.appendChild(KCSettingBtnOK)
                KCSettingBox.appendChild(KCSettingBtnBox)

                KCBox.innerHTML = ''
                KCBox.appendChild(switchTip)
                KCBox.appendChild(KCSwitch)
                KCBox.appendChild(KCSettingBox)

            })
            settingBtnBox.appendChild(speedSettingBtn)
        }
        settingContent.appendChild(settingBtnBox)

        settingContent.appendChild(KCBox)

        let hideBtnBox = document.createElement('div')
        hideBtnBox.setAttribute('style', 'margin-top:10px;')
        let hideBtn = document.createElement('button')
        hideBtn.innerText = '关闭面板'
        hideBtn.setAttribute('style', 'padding:8px 10px;border:0;color:white;background-color:rgb(50,130,236);border-radius:28px;')
        hideBtn.addEventListener('click', function() {
            document.documentElement.removeChild(settingBox)
            hasSettingShow = false
        })
        hideBtnBox.appendChild(hideBtn)
        settingContent.appendChild(hideBtnBox)

        let settingTip = document.createElement('div')
        settingTip.setAttribute('style', 'margin-top:10px;')
        settingTip.innerText = '设置完成后刷新页面,以确保生效'
        settingContent.appendChild(settingTip)

        contentBox.appendChild(settingContent)
        settingBox.appendChild(contentBox)
        document.documentElement.appendChild(settingBox)
        hasSettingShow = true
        speedSettingBtns[0].click()
    }
    GM_registerMenuCommand('设置',showSetting)
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址