您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
bilibili倍速快捷键+记忆(可自定义设置快捷键,自定义一个额外倍速按钮,设置记忆模式)
当前为
// ==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) })();