// ==UserScript== // @name bilibili关灯及快捷操作 // @namespace hhh2000 // @version 1.0.0.9 // @description bilibili关灯及快捷操作(把被新版B站藏起来的关灯按钮揪出来,在关闭弹幕按钮左边,还可以用快捷键,默认'A')、非全屏滚轮音量控制、弹幕控制快捷操作等 // @author hhh2000 // @include http*://www.bilibili.com/* // @include http*://www.bilibili.com/video/av* // @include http*://www.bilibili.com/video/BV* // @include http*://www.bilibili.com/watchlater/* // @include http*://www.bilibili.com/medialist/play/* // @include http*://www.bilibili.com/bangumi/play/ep* // @include http*://www.bilibili.com/bangumi/play/ss* // @include http*://bangumi.bilibili.com/anime/*/play* // @include http*://bangumi.bilibili.com/movie/* // @include http*://www.bilibili.com/blackboard/* // @include http*://t.bilibili.com/* // @include http*://space.bilibili.com/* // @include http*://www.bilibili.com/video/online.html* // @include http*://www.bilibili.com/account/history // @include http*://api.bilibili.com/* // @include http*://search.bilibili.com/* // @include http*://space.bilibili.com/39638/favlist/* // @exclude http*://www.bilibili.com/watchroom/* // @require https://cdn.staticfile.org/jquery/1.12.4/jquery.min.js // @run-at document-end // @grant GM_setValue // @grant GM_getValue // @grant GM_removeValue // @grant GM_listValues // @grant none // @compatible chrome /* globals jQuery, $, waitForkeyElements */ /* eslint-disable no-multi-spaces, dot-notation */ // @downloadURL none // ==/UserScript== 'use strict'; var hhh_lightoff_main = { init() { var //globle ver, ver2, fps, h5Player, curr_focus, is_in_biliplayer, is_show_hint, $tip, move_frames, biliinfo = { 'is_login': 0, 'vip_status': 0, 'lightoff_state': false } app_page_parameters = { player_setting_opacity: 0, player_setting_area: 0, player_setting_fontsize: 0, }; // sort_arr = [], // real_to_draw = []; const [LIGHT_DEFAULT, LIGHTON_LIKE] = ['default', 'lightOnWhenLike'] //BILI_3_X_VIDEO (一般视频 | 旧版 | 默认) //BILI_3_X_VIDEO_V1(一般视频 | 新版) //BILI_3_X_MOVIE (电影电视剧和番剧) const [ BILI_2_X_V2 , BILI_2_X_V3 , BILI_2_X , BILI_3_X_VIDEO, BILI_3_X_MOVIE , BILI_3_X_VIDEO_V1 , BILI_4_X_V1, ALL] = ['bili_2.x.v2', 'bili_2.x.v3', 'bili_2.x', 'bili_3.x' , 'bili_3.x.movie', 'bili_3.x.video.v1', 'bili_4.x.v1', 'all']; const [NO_NEED_LOAD] = ['no need to load']; var //切换番剧和一般视频class bb = {}, bb_type = undefined, bb_config = { bb_class_data: { //其实这样不方便调试,但方便测试 'player':{[BILI_2_X_V2]:'.player'}, //main //$('meta[property="og:image"]').attr('content') 'coverImg':{[ALL]:'meta[property="og:image"]'}, //封面 'playTipWrap':{[BILI_2_X] :'.bilibili-player-dm-tip-wrap', [BILI_3_X_VIDEO]:'.bpx-player-video-perch'}, //paly/pause 'fullScreen':{[BILI_2_X] :'.bilibili-player-video-btn-fullscreen', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-btn.bpx-player-ctrl-full', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-btn.bpx-player-ctrl-full'}, //全屏 'webFullScreen':{[BILI_2_X]:'.bilibili-player-video-web-fullscreen', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-btn.bpx-player-ctrl-web', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-btn.bpx-player-ctrl-web'}, //网页全屏 'wideScreen':{[BILI_2_X] :'.bilibili-player-video-btn-widescreen', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-btn.bpx-player-ctrl-wide', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-btn.bpx-player-ctrl-wide'}, //宽屏 //一健三联 toolbar-left 'like' :{[BILI_2_X_V2]:'.ops .like' , [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .video-like' , [BILI_3_X_MOVIE]:'.toolbar_like_info__Eog_Q', [BILI_3_X_VIDEO]:'.ops .like', [NO_NEED_LOAD]: true}, //点赞 'likeon' :{[BILI_2_X_V2]:'.ops .like.on', [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .video-like.on', [BILI_3_X_MOVIE]:'.toolbar_like_info__Eog_Q.toolbar_active__QT3OV', [BILI_3_X_VIDEO]:'.ops .like.on'}, //已点赞 'coin' :{[BILI_2_X_V2]:'.ops .coin' , [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .video-coin' , [BILI_3_X_MOVIE]:'.toolbar_coin_info__5hnd9', [BILI_3_X_VIDEO]:'.ops .coin'}, //硬币 'coinon' :{[BILI_2_X_V2]:'.ops .coin.on', [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .video-coin.on', [BILI_3_X_MOVIE]:'.toolbar_coin_info__5hnd9.toolbar_active__QT3OV', [BILI_3_X_VIDEO]:'.ops .coin.on'}, //已投币 // 'like' :{[BILI_2_X_V2]:'.ops .like' , [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .like' , [BILI_3_X_MOVIE]:'.like-info', [BILI_3_X_VIDEO]:'.ops .like', [NO_NEED_LOAD]: true}, //点赞 // 'likeon' :{[BILI_2_X_V2]:'.ops .like.on', [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .like.on', [BILI_3_X_MOVIE]:'.like-info.active', [BILI_3_X_VIDEO]:'.ops .like.on'}, //已点赞 // 'coin' :{[BILI_2_X_V2]:'.ops .coin' , [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .coin' , [BILI_3_X_MOVIE]:'.coin-info', [BILI_3_X_VIDEO]:'.ops .coin'}, //硬币 // 'coinon' :{[BILI_2_X_V2]:'.ops .coin.on', [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .coin.on', [BILI_3_X_MOVIE]:'.coin-info.active', [BILI_3_X_VIDEO]:'.ops .coin.on'}, //已投币 'collect':{[BILI_2_X_V2]:'.ops .collect', [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .video-fav', [BILI_3_X_MOVIE]:'', [BILI_3_X_VIDEO]:'.ops .collect'}, //收藏 'collecton':{[BILI_2_X_V2]:'.ops .collect.on', [BILI_3_X_VIDEO_V1]:'.video-toolbar-left .video-fav.on', [BILI_3_X_MOVIE]:'', [BILI_3_X_VIDEO]:'.ops .collect.on'}, //已收藏 'biliDlgM':{[BILI_2_X]:'.bili-dialog-m', [BILI_3_X_MOVIE]:'.coin-dialog-mask', [BILI_3_X_VIDEO]:'.bili-dialog-m', [BILI_3_X_VIDEO_V1]:'.bili-dialog-m'}, //投币、收藏框 'coinDlgCoin':{[BILI_2_X]:'.bili-dialog-m .coin-operated-m', [BILI_3_X_MOVIE]:'.coin-dialog-mask .coin-operated', [BILI_3_X_VIDEO]:'.bili-dialog-m .coin-operated-m', [BILI_3_X_VIDEO_V1]:'.bili-dialog-m .coin-operated-m-exp'}, //投币div 'coinDlgCloseBtn':{[BILI_2_X]:'.bili-dialog-m .close', [BILI_3_X_MOVIE]:'.coin-dialog-mask .icon-close', [BILI_3_X_VIDEO]:'.bili-dialog-m .close'}, //投币确定按钮 'coinDlgOkBtn':{[BILI_2_X]:'.bili-dialog-m .bi-btn', [BILI_3_X_MOVIE]:'.coin-dialog-mask .coin-btn', [BILI_3_X_VIDEO]:'.bili-dialog-m .bi-btn'}, //投币确定按钮 //弹幕 'danmukuTopClose':{[BILI_2_X]:'.bilibili-player-block-filter-type[data-name=ctlbar_danmuku_top_close]', [BILI_3_X_VIDEO]:'.bpx-player-block-filter-type.bpx-player-block-typeTop.bpx-player-active'}, //顶部弹幕 'danmukuTop' :{[BILI_2_X]:'.bilibili-player-block-filter-type[ftype=top]', [BILI_3_X_VIDEO]:'.bpx-player-block-filter-type.bpx-player-block-typeTop'}, //顶部弹幕 'danmukuBottomClose':{[BILI_2_X]:'.bilibili-player-block-filter-type[data-name=ctlbar_danmuku_bottom_close]', [BILI_3_X_VIDEO]:'.bpx-player-block-filter-type.bpx-player-block-typeBottom.bpx-player-active'}, //底部弹幕 'danmukuBottom' :{[BILI_2_X]:'.bilibili-player-block-filter-type[ftype=bottom]', [BILI_3_X_VIDEO]:'.bpx-player-block-filter-type.bpx-player-block-typeBottom'}, //底部弹幕 'progressVal' :{[ALL]:'.bui-progress-val'}, //读数 'progressWrap' :{[ALL]:'.bui-progress-wrap'}, //进度条 'progressBar' :{[ALL]:'.bui-progress-bar'}, //百分数 'settingOpacity' :{[BILI_2_X]:'.bilibili-player-setting-opacity', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-ui-opacity'}, //弹幕透明度 'settingFontsize':{[BILI_2_X]:'.bilibili-player-setting-fontsize', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-ui-fontsize'}, //弹幕字号 'settingArea' :{[BILI_2_X]:'.bilibili-player-setting-area', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-ui-area'}, //显示区域 'settingFs' :{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-left-fs input', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-left-fs input'}, //弹幕随屏幕缩放 //音量 'volumeHint' :{[BILI_2_X]:'.bilibili-player-volumeHint', [BILI_3_X_VIDEO]:'.bpx-player-volume-hint'}, //音量显示 'volumeHintText':{[BILI_2_X]:'.bilibili-player-volumeHint-text', [BILI_3_X_VIDEO]:'.bpx-player-volume-hint-text'}, //音量显示百分比读数 'volumeHintIcon':{[BILI_2_X]:'.bilibili-player-volumeHint-icon', [BILI_3_X_VIDEO]:'.bpx-player-volume-hint-icon'}, //音量显示图标 // 'volumebarWrp' :{[BILI_2_X]:'.bilibili-player-video-volumebar-wrp', [BILI_3_X_MOVIE]:'.squirtle-volume-bar-wrap', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-volume-box'}, //音量条 // 'volumeNum' :{[BILI_2_X]:'.bilibili-player-video-volume-num', [BILI_3_X_MOVIE]:'.squirtle-volume-num', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-volume-number'}, //音量读数 'volumebarWrp' :{[BILI_2_X]:'.bilibili-player-video-volumebar-wrp', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-btn.bpx-player-ctrl-volume', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-volume-box'}, //音量条 'volumeNum' :{[BILI_2_X]:'.bilibili-player-video-volume-num', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-volume-box', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-volume-number'}, //音量读数 //视频速度 'videoSpeedActive':{[BILI_2_X]:'.bilibili-player-video-btn-speed-menu-list.bilibili-player-active', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-playbackrate-menu-item.bpx-state-active', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-playbackrate-menu-item.bpx-state-active'}, //视频速度 'videoSpeed':{[BILI_2_X]:'.bilibili-player-video-btn-speed-menu-list', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-playbackrate-menu-item', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-playbackrate-menu-item'}, //视频速度 'videoSpeedName':{[BILI_2_X]:'.bilibili-player-video-btn-speed-name', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-playbackrate-result', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-playbackrate-result'}, //视频速度 //弹幕设置等 'switchBody':{[ALL]:'.bui-switch-body'}, //系统关灯css设置 'switchDot':{[ALL]:'.bui-switch-dot'}, //系统弹幕设置按钮wrap进度条拖动点 'switchInput':{[BILI_3_X_VIDEO_V1]:'.bui-danmaku-switch-input', [BILI_3_X_MOVIE]:'.bui-danmaku-switch-input', [ALL]:'.bui-switch-input'}, //弹幕设置switch按钮 'switchLabel':{[BILI_3_X_VIDEO_V1]:'.bui-danmaku-switch-label', [BILI_3_X_MOVIE]:'.bui-danmaku-switch-label', [ALL]:'.bui-switch-label'}, //弹幕设置switchLabel 'danmaku':{[BILI_2_X]:'.bilibili-player-video-danmaku', [BILI_3_X_VIDEO]:'.bpx-player-row-dm-wrap'}, //弹幕 'danmakuRoot':{[BILI_2_X]:'.bilibili-player-video-danmaku-root', [BILI_3_X_VIDEO]:'.bpx-player-dm-root'}, //系统弹幕设置条 'danmakuSwitch':{[BILI_2_X]:'.bilibili-player-video-danmaku-switch', [BILI_3_X_VIDEO]:'.bpx-player-dm-switch'}, //关闭弹幕按钮 'dm':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting'}, //系统弹幕设置按钮 'dmWrap':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-wrap', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-wrap'}, //系统弹幕设置wrap 'dmBox':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-box', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-box'}, //系统弹幕设置box 'dmLeft':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-left', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-left'}, 'dmLeftMore':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-left-more', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-left-more'}, 'dmLeftMoreText':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-left-more-text', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-left-more-text'}, 'dmLeftBlock':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-left-block', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-left-block'}, 'dmLeftBlockTitle':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-left-block-title', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-left-block-title'}, 'dmLeftFlagTitle':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-left-flag-title', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-left-flag-title'}, 'dmRightMore':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-right-more', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-right-more'}, 'dmRightMoreText':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-right-more-text', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-right-more-text'}, 'dmRightSeparator':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-right-separator', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-right-separator'}, 'dmRightReset':{[BILI_2_X]:'.bilibili-player-video-danmaku-setting-right-reset', [BILI_3_X_VIDEO]:'.bpx-player-dm-setting-right-reset'}, 'videoWrap':{[BILI_2_X]:'.bilibili-player-video-wrap', [BILI_3_X_VIDEO]:'.bpx-player-video-area'}, //播放wrap 'videoContextMenu':{[BILI_2_X]:'.bilibili-player-video-wrap', [BILI_3_X_VIDEO]:'.bpx-player-video-perch'}, //播放contextmenu 'video':{[BILI_2_X]:'.bilibili-player-video', [BILI_3_X_VIDEO]:'.bpx-player-video-wrap'}, //播放 'videoTopMask':{[BILI_2_X]:'.bilibili-player-video-top-mask', [BILI_3_X_VIDEO]:'.bpx-player-top-mask'}, //全屏时鼠标悬停时产生的顶端mask //系统设置等 'playArea':{[BILI_2_X]:'.bilibili-player-area'}, //哔哩哔哩播放器 'playVideo':{[BILI_2_X]:'.bilibili-player-video-btn'}, //系统设置 'playVideoControlWrap':{[BILI_2_X]:'.bilibili-player-video-control-wrap', [BILI_3_X_VIDEO]:'.bpx-player-control-wrap'}, //系统控制面板 'playSetting':{[BILI_2_X]:'.bilibili-player-video-btn-setting'}, //系统播放设置 'playSettingWrap':{[BILI_2_X]:'.bilibili-player-video-btn-setting-wrap'}, //系统播放设置wrap bpx-player-ctrl-btn bpx-player-ctrl-setting 'playSettingAutoplay':{[BILI_2_X]:'.bilibili-player-video-btn-setting-left-autoplay input', [BILI_3_X_MOVIE]: '.bpx-player-ctrl-setting-autoplay', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-setting-autoplay input'}, //自动播放 'playSettingRepeatInput':{[BILI_2_X]:'.bilibili-player-video-btn-setting-left-repeat input', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-setting-loop', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-setting-loop input'}, //洗脑循环按钮 'playSettingLightoff':{[BILI_2_X]:'.bilibili-player-video-btn-setting-right-others-content-lightoff input', [BILI_3_X_MOVIE]:'.bpx-player-ctrl-setting-checkbox.bpx-player-ctrl-setting-lightoff', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-setting-checkbox.bpx-player-ctrl-setting-lightoff input'}, //关灯按钮 // 'playSettingAutoplay':{[BILI_2_X]:'.bilibili-player-video-btn-setting-left-autoplay input', [BILI_3_X_MOVIE]: '.squirtle-setting-autoplay', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-setting-autoplay'}, //自动播放 // 'playSettingRepeatInput':{[BILI_2_X]:'.bilibili-player-video-btn-setting-left-repeat input', [BILI_3_X_MOVIE]:'.squirtle-setting-loop', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-setting-loop'}, //洗脑循环按钮 // 'playSettingLightoff':{[BILI_2_X]:'.bilibili-player-video-btn-setting-right-others-content-lightoff input', [BILI_3_X_MOVIE]:'.squirtle-single-setting-other-choice.squirtle-lightoff', [BILI_3_X_VIDEO]:'.bpx-player-ctrl-setting-checkbox.bpx-player-ctrl-setting-lightoff input'}, //关灯按钮 //'bpxStateLightOff':{[BILI_3_X_VIDEO]:'.bpx-state-light-off'}, //关灯bpx 'playJumpElectric':{[BILI_2_X]: '.bilibili-player-electric-panel-jump', [BILI_3_X_VIDEO]: '.bpx-player-electric-jump'}, //2.69.4版本B站取消充电鸣谢,2.72又加回来了 'playTooltipArea' :{[BILI_3_X_VIDEO]: '.bpx-player-tooltip-area'}, 'playTooltipItem' :{[BILI_3_X_VIDEO]: '.bpx-player-tooltip-item'}, 'playTooltipTitle':{[BILI_3_X_VIDEO]: '.bpx-player-tooltip-title'}, 'playCtrlSubtitle':{[BILI_3_X_VIDEO]: '.bpx-player-ctrl-subtitle .bpx-common-svg-icon'}, //高能进度条 'playPBP':{[BILI_3_X_VIDEO]: '.bpx-player-pbp'}, //稍后再看 'playWatchLater':{[BILI_3_X_VIDEO]: '.ops-watch-later'}, //视频选集/连播列表 //'videoList':{[BILI_3_X_VIDEO]: '.list-box,.video-section-list'}, //右键菜单 'playerContextMenu':{[BILI_2_X]:'.bilibili-player-context-menu-container.black.bilibili-player-context-menu-origin', [BILI_3_X_VIDEO]:'.bpx-player-contextmenu.bpx-player-black'}, //右键菜单 'hotkeyPanel':{[BILI_2_X]:'.bilibili-player-hotkey-panel-container', [BILI_3_X_VIDEO]:'.bpx-player-hotkey-panel'}, //快捷键说明面板 'hotkeyPanelClose':{[BILI_2_X]:'.bilibili-player-hotkey-panel-close', [BILI_3_X_VIDEO]:'.bpx-player-hotkey-panel-close'}, //快捷键说明面板关闭按钮 //'videoInfo':{[BILI_2_X]:'.bilibili-player-video-info', [BILI_3_X_VIDEO]:'.bpx-player-info'}, //视频统计信息 'videoInfoClose':{[BILI_2_X]:'.bilibili-player-video-info-close', [BILI_3_X_VIDEO]:'.bpx-player-info-close'}, //视频统计信息关闭按钮 'videoInfoContainer':{[BILI_2_X]:'.bilibili-player-video-info-container', [BILI_3_X_VIDEO]:'.bpx-player-info-container'}, //视频统计信息 'videoInfoShow':{[BILI_2_X]:'.bilibili-player-video-info-container active', [BILI_3_X_VIDEO]:'.bpx-player-info-container'}, //视频统计信息面板显示 'DOMNodeInsertedVideoInfoShow':{[BILI_2_X]:'.bilibili-player-video-info-container active', [BILI_3_X_VIDEO]:'.info-line'}, // }, set_bb(_bb_type) { bb_type = _bb_type; for(var k in this.bb_class_data){ //var class_str = this.bb_class_data[k][bb_type] || this.bb_class_data[k][[BILI_2_X]] || this.bb_class_data[k][[ALL]]; let cc = this.bb_class_data[k]; var class_str = cc[bb_type] || cc[Object.keys(cc).find((t)=>{ return bb_type.indexOf(t) !== -1 })] || cc[[ALL]]; bb[k] = class_str; } }, }; function deep_clone(obj){ return JSON.parse(JSON.stringify(obj)) } class keybindingKeyString{ mergeSpace(str) { return str.replace(/\s+/g, ' ') } delSpace(str) { return str.replace(/\s+/g, '') } uppercaseFirst(str) { return str[0].toUpperCase() + str.substr(1) } getkeyCode(k) { var keyCodes = { 300: '滚↑轮', 301: '滚↓轮', 302: '鼠标左键', 303: '鼠标右键', 304: '鼠标中键', 305: '鼠标左前侧键', 306: '鼠标左后侧键', 307: '鼠标右前侧键', 308: '鼠标右后侧键', 309: '鼠标中前侧键', 310: '鼠标中后侧键', 0: "", 3: "break", 8: "Backspace", 9: "Tab", 12: "Clear", 13: "Enter", 16: "Shift", 17: "Ctrl", 18: "Alt", 19: "PauseBreak", 20: "CapsLock", 27: "Escape", 32: "Space", 33: "PageUp", 34: "PageDown", 35: "End", 36: "Home", 37: "←", //LeftArrow ↑ ↓ ← → 38: "↑", //UpArrow 39: "→", //RightArrow 40: "↓", //DownArrow 45: "Insert", 46: "Delete", 48: "0", 49: "1", 50: "2", 51: "3", 52: "4", 53: "5", 54: "6", 55: "7", 56: "8", 57: "9", 65: "A", 66: "B", 67: "C", 68: "D", 69: "E", 70: "F", 71: "G", 72: "H", 73: "I", 74: "J", 75: "K", 76: "L", 77: "M", 78: "N", 79: "O", 80: "P", 81: "Q", 82: "R", 83: "S", 84: "T", 85: "U", 86: "V", 87: "W", 88: "X", 89: "Y", 90: "Z", 93: "ContextMenu", 96: "NumPad0", 97: "NumPad1", 98: "NumPad2", 99: "NumPad3", 100: "NumPad4", 101: "NumPad5", 102: "NumPad6", 103: "NumPad7", 104: "NumPad8", 105: "NumPad9", 106: "NumPad_Multiply", 107: "NumPad_Add", 108: "NumPad_Separator", 109: "NumPad_Subtract", 110: "NumPad_Decimal", 111: "NumPad_Divide", 112: "F1", 113: "F2", 114: "F3", 115: "F4", 116: "F5", 117: "F6", 118: "F7", 119: "F8", 120: "F9", 121: "F10", 122: "F11", 123: "F12", 124: "F13", 125: "F14", 126: "F15", 127: "F16", 128: "F17", 129: "F18", 130: "F19", 144: "NumLock", 145: "ScrollLock", 166: "BrowserBack", 167: "BrowserForward", 170: "BrowserSearch", 172: "BrowserHome", 173: "AudioVolumeMute", 174: "AudioVolumeDown", 175: "AudioVolumeUp", 176: "MediaTrackNext", 177: "MediaTrackPrevious", 178: "MediaStop", 179: "MediaPlayPause", 180: "LaunchMail", 181: "LaunchMediaPlayer", 183: "LaunchApp2", 186: ";", 187: "=", 188: ",", 189: "-", 190: ".", 191: "/", 192: "`", 193: "ABNT_C1", 194: "ABNT_C2", 219: "[", 220: "\\", 221: "]", 222: "'", 223: "OEM_8", 226: "OEM_102", 229: "KeyInComposition", }; return keyCodes[k]; } getKCode(e){ return e.ctrlKey<<18 | e.shiftKey<<17 | e.altKey<<16 | e.keyCode; } kcodeToKeys(kcode){ let keys = []; let i = 0; if(kcode>>18 & 1) keys[i++] = 'Ctrl'; if(kcode>>17 & 1) keys[i++] = 'Shift'; if(kcode>>16 & 1) keys[i++] = 'Alt'; let key = kcode & 0xFFFF; if(['ctrl', 'shift', 'alt'].indexOf(this.getkeyCode(key)?.toLowerCase()) === -1) keys[i++] = this.getkeyCode(key); return keys; } keysToStr(keys){ let keystr = ''; for (var index = 0; index < keys.length-1; index++) { keystr += `${keys[index]} + `; } keystr += keys[index]; return keystr; } strToKeys(str){ return str.split(/\s*\+\s*/) } stringifyKeyCode(e){ return this.keysToStr(this.kcodeToKeys(this.getKCode(e))) } is_ctl(str) { return ['ctrl', 'shift', 'alt'].indexOf(str.replace(/\s+/g,'').toLowerCase()) !== -1 } combine_same_keys(keystr_arr){ //log(keystr_arr) if(keystr_arr.length <= 1) return keystr_arr; //需要多于一个 //判断相似结构 let s0 = keystr_arr[0].split('+'); let len0 = s0.length; let str0__1 = s0.slice(0,-1).toString(); for (let i = 1; i < keystr_arr.length; i++) { const s = keystr_arr[i].split('+'); //是否需要合并 if(len0 <= 1 || this.is_ctl(s.slice(-1).toString()) === true || len0 !== s.length || str0__1 !== s.slice(0,-1).toString()){ str0__1 = -1; break; } } if(str0__1 !== -1){ //组合键组合 Ctrl+X / Ctrl+A -> Ctrl+X/A let arr = [this.keysToStr([...s0.slice(0,-1), s0.slice(-1)])]; for (let i = 1; i < keystr_arr.length; i++) { const s = keystr_arr[i].split('+'); arr.push(s.slice(-1)[0]); } keystr_arr = arr; }else if(len0 > 1){ //避免歧义 Ctrl+X / A -> Ctrl+X/(A) let arr = []; for (let i = 0; i < keystr_arr.length; i++) { const s = keystr_arr[i].split('+'); (s.length === 1 && this.is_ctl(s[0]) === false) ? arr.push('('+keystr_arr[i]+')') : arr.push(keystr_arr[i]); } keystr_arr = arr; } //滚轮组合 滚[↑↓]轮/滚[↑↓]轮 -> 滚轮 if(keystr_arr.toString().match(/滚[↑↓]轮,滚[↑↓]轮/)) keystr_arr = keystr_arr.toString().replace(/滚[↑↓]轮,滚[↑↓]轮/, '滚轮').split(','); return keystr_arr; } } const [ON, OFF] = [true, false]; var //config keycode = { 'Enter': 13, 'Ctrl': 17, 'Esc': 27, 'left': 37, 'right': 39, 'up': 38, 'down': 40, 'space': 32, '/': 191, }, kb = new keybindingKeyString(); config = { //一些主要开关设置 sets: {}, group: 'commonCheckbox', //key唯一 findGroupByKey(key){ for(let [group, {options}] of Object.entries(this.sets)){ if(!!options[key]) return group; } return null; }, getCheckboxSettingStatus(key) { let group = this.findGroupByKey(key); return group && this.sets[group]['options'][key]['status']; }, getCheckboxSettingFn(key) { let group = this.findGroupByKey(key); return group && this.sets[group]['options'][key]['fn']; }, getCheckboxSettingArgs(key, subkey) { let group = this.findGroupByKey(key); if(group) { if(subkey === undefined) return this.sets[group]['options'][key]['args']; else return this.sets[group]['options'][key]['args'][subkey]; } }, getCheckboxSettingTempArgs(key, subkey) { let group = this.findGroupByKey(key); if(group) { if(subkey === undefined) return this.sets[group]['options'][key]['tempargs']; else return this.sets[group]['options'][key]['tempargs'][subkey]; } }, setCheckboxSettingStatus(key, status) { let group = this.findGroupByKey(key); group && (this.sets[group]['options'][key]['status'] = status); }, setCheckboxSettingTips(key, tip) { let group = this.findGroupByKey(key); group && (this.sets[group]['options'][key]['tip'] = tip); }, setCheckboxSettingTempArgs(key, tempargs) { let group = this.findGroupByKey(key); group && (this.sets[group]['options'][key]['tempargs'] = deep_clone(tempargs)); }, setCheckboxSettingArgs(key, args) { let group = this.findGroupByKey(key); group && (this.sets[group]['options'][key]['args'] = deep_clone(args)); }, storageCheckboxSetting() { set_value('hhh_checkboxes', this.sets); }, defaultCheckboxSetting() { this.sets = JSON.parse(JSON.stringify(this.checkboxes)); }, setKeySettingKeystr(key, keystr) { //let group = this.findGroupByKey(key); this.sets['keyBinding']['options'][key]['keystr'] = keystr }, getKeySettingNonuse(key) { return this.sets['keyBinding']['options'][key]['nonuse'] }, setKeySettingNonuse(key, nonuse) { //log(key, nonuse) this.sets['keyBinding']['options'][key]['nonuse'] = nonuse }, getKeySettingKeycount(key) { return this.sets['keyBinding']['options'][key]['keycount'] }, setKeySettingKeycount(key, keycount) { //log(key, nonuse) this.sets['keyBinding']['options'][key]['keycount'] = keycount }, o2a(o, arr=[]){ for (let [key, val] of Object.entries(o)) { arr.push(key); if(typeof val === 'object'){ this.o2a(val, arr); } } return arr; }, //深度合并多个对象的方法 deepAssign(){ //判断对象是否是一个纯粹的对象 function isPlainObject(obj){ return typeof obj === 'object' && Object.prototype.toString.call(obj) === '[object Object]' } let len = arguments.length, target = arguments[0] if(!isPlainObject(target)){ target = {} } for (let i = 1; i < len; i++){ let source = arguments[i] if( isPlainObject( source ) ){ for( let s in source ){ if ( s === '__proto__' || target === source[s] ) { continue } if( isPlainObject( source[s] ) ){ target[s] = config.deepAssign(target[s], source[s]) }else{ target[s] = source[s] } } } } return target }, //根据dest_sets找src_sets中没有的 getOptionsDiff(src_sets, dest_sets){ // log('------1------') // log(src_sets) // log(dest_sets) // log('------2------') function getOptionsKey(o){ let options = {} for (let key of Object.keys(o)) { options[key] = Object.keys(o[key]['options']) //转换为数组 } return options } let diffOptions = [] let src_mainkeys = Object.keys(src_sets) for (let mainkey of Object.keys(dest_sets)) { //log(mainkey) if(src_mainkeys?.includes(mainkey) === false){ //log('--',mainkey) diffOptions.push([mainkey, '']) } } let src_options = getOptionsKey(src_sets) let dest_options = getOptionsKey(dest_sets) // log('------3------') // log(src_options) // log(dest_options) for (let key of Object.keys(dest_sets)) { let option = dest_sets[key]['options'] for(let k of Object.keys(option)){ if(src_options[key]?.includes(k) === false){ log(key,k) diffOptions.push([key, k]) } } } return diffOptions }, //是否有重复key compareDuplicateKey(sets, excludeType = ['key']){ //排除指定类型 let bucket = {} for (let key of Object.keys(sets)) { if(excludeType.includes(sets[key]['type']) === true) continue let option = sets[key]['options'] for(let k of Object.keys(option)){ if(bucket[k] === true) { log('checkboxes包含重复key!!!: ', k); return } bucket[k] = true } } }, getCheckboxSetting() { this.sets = get_value('hhh_checkboxes') || JSON.parse(JSON.stringify(this.checkboxes)) //是否有重复key this.compareDuplicateKey(this.checkboxes) if(JSON.stringify(this.o2a(this.sets)) !== JSON.stringify(this.o2a(this.checkboxes))){ //log('设置更新,设置为默认'); //log(JSON.stringify(this.o2a(this.sets))) //log(JSON.stringify(this.o2a(this.checkboxes))) //this.defaultCheckboxSetting(); log('设置更新,合并更新设置') for(let currDiff of this.getOptionsDiff(this.sets, this.checkboxes)){ log('增加设置项:', currDiff[0], currDiff[1]) } this.sets = this.deepAssign({}, this.checkboxes, this.sets) for(let currDiff of this.getOptionsDiff(this.checkboxes, this.sets)){ log('去除设置项:', currDiff[0], currDiff[1]) if(currDiff[1] !== '') delete this.sets[currDiff[0]].options[currDiff[1]] else delete this.sets[currDiff[0]] } } this.storageCheckboxSetting() }, findKeyBinding(curr_key){ let arr = []; for (let [k, {text, keystr, keycount, display, delimiter, nonuse}] of Object.entries(config.sets['keyBinding'].options)) { //config.sets中查找 let keys = keystr.split(/\s+/); keys.forEach((key, index) => { if(keycount !== 0 && curr_key === key){ arr.push({id:k, text: text, index: index, key: key, is_system_key: (k.indexOf('sys') === 0), nonuse: nonuse}); } }); } return arr; }, //options 里的 key 不能重复,不包含快捷键key checkboxes: { commonCheckbox: { options: { autoPlay: { text: '自动播放', status: OFF, tip: '' }, autoLightOff: { text: '自动关灯', status: OFF, tip: '' }, autoFullScreen: { text: '自动全屏', status: OFF, tip: '' }, autoWebFullScreen: { text: '自动网页全屏', status: OFF, tip: '' }, autoWideScreen: { text: '自动宽屏', status: OFF, tip: '' }, videoRepeat: { text: '自动洗脑循环', status: OFF, tip: '' }, lightOffWhenPlaying: { text: '播放时自动关灯', status: OFF, fn: 'lightOffWhenPlaying' }, lightOnWhenPause: { text: '暂停时自动开灯', status: OFF, fn: 'lightOnWhenPause' }, lightOnWhenLike: { text: '关灯状态时滚动屏幕控制开关灯', status: OFF, tip: '默认下滚超过0.3屏开灯,方便关灯时点赞、评论等', fn: 'lightOnWhenLike', args:{ screen_top: 0.3, args_opt: { tip: ['占屏比(0.0~1.0)'] } } }, autoTopQualityVideo: { text: '自动选择最高质量视频', status: OFF }, autoOpenDanmu: { text: '自动打开弹幕', status: ON, tip: '', fn: 'autoOpenDanmu' }, rememberVideoRepeat: { text: '记忆洗脑循环', status: ON, tip: '记住最后选择,优先级低于【自动洗脑循环】', show: OFF }, //优先级低于videoRepeat dblclickFullScreen: { text: '双击或中键、功能键全屏', status: ON, tip: '', fn: 'dblclickFullScreen' }, volumeControlWhenNonFullScreen: { text: '非全屏滚轮音量调节', status: ON, tip: '', fn___: 'wheel_volumeHint', args:{ screen_left: 0.3, screen_rght: 0.7, args_opt: { tip: ['鼠标有效范围(全屏0.0~1.0)'] } } }, //0~1全屏 volumeControlWhenPause: { text: '非全屏暂停时滚轮音量调节', status: ON, tip: '' }, danmuOpacityControl: { text: '弹幕透明度控制', status: ON, tip: 'Ctrl + 滚轮', fn___: 'wheel_opacity' }, //args:{ delta: 5, args_opt: { tip: ['调节幅度%'] } } }, //ctrl+滚轮 keyVideoSpeed: { text: '键盘调节视频速度', status: ON, tip: 'Ctrl + ↑↓', show:OFF }, removeVideoTopMask: { text: '去掉顶部mask', status: ON, tip: '', fn: 'removeVideoTopMask' }, jumpElectric: { text: '跳过充电鸣谢', status: ON, fn: 'jumpElectric' }, //2.69.4版本B站取消充电鸣谢,2.72又加回来了 //removeMostViewedListener: { text: '删除动态首页UP主动态提示', status: ON }, }, btn: '常用设置', type: 'checkbox' }, otherCheckbox: { options: { openHotKey: { text: '开启自定义快捷键', status: ON, tip: '', fn: 'set_hotkey' }, reloadDanmuku: { text: '快进快退恢复重载弹幕效果', status: OFF }, //快进快退恢复重载弹幕效果 3.13版本取消默认不重载效果,改回默认重载弹幕 loopPlayback: { text: '开启段落循环', status: OFF, tip: 'Ctrl+左键点击进度条设定段落循环 / 设定后L键切换', fn: 'loopPlayback' }, hideThreePopup: { text: '隐藏屏幕三连弹窗', status: OFF, tip: '', fn: 'hideThreePopup' }, hideScorePopup: { text: '隐藏屏幕打分弹窗', status: OFF, tip: '', fn: 'hideScorePopup' }, expandList: { text: '订阅合集列表自动展开', status: ON, tip: '', fn: 'expand_list', args:{ columns: 10, args_opt: { tip: ['列表展开长度(行)'] } } }, collectionFilter: { text: '收藏夹关键字过滤', status: ON, tip: '', fn: 'collection_filter' }, saveRecommendList: { text: '记忆首页推荐列表(暂无法设置)', status: ON, tip: '', fn: 'run_save_recommend_list', disable: ON }, onlinePreview: { text: '当前在线视频预览(暂无法设置)', status: ON, tip: '', fn: 'run_online_preview', disable: ON }, openVideoInNewTab: { text: '新窗口打开连播列表视频', status: OFF, tip: '点击右侧自动连播列表视频在新窗口打开', fn: 'run_rec_list_newtab' }, customPlayRate: { text: '开启自定义倍速播放', status: ON, tip: '长按 → 键', fn: 'customPlayRate', args:{ rate: 2.0, args_opt: { tip: ['播放倍速(倍),系统默认3倍'] } } }, dynamicSpace: { text: '合理化空格键', status: ON, tip: 'B站空格键不合理,恢复非视频焦点时的默认效果(翻页)' }, danmuAreaSwicth: { text: '弹幕显示区域切换设置', status: ON, tip: '弹幕显示区域大切换到小时重新加载弹幕' }, //showDynListContent: { text: '动态首页显示/隐藏简介', status: ON, tip: '可在当前页面显示/简介隐藏内容' }, favSetting: { text: '加长收藏夹列表显示', status: ON, tip: '加长' }, //__line__1: { text: '', type: 'line', status: OFF, tip: '', }, memoryProgress: { text: '记忆选集视频进度', status: ON, tip: '在选集列表里用颜色和进度条指示', fn: 'memory_multipart_progress' }, autoJumpMemoryProgress: { text: '选集视频自动跳转', status: OFF, level: 1 }, autoSubtitle: { text: '自动开启字幕', status: OFF, tip: '如果当前视频有字幕,自动开启' }, autoListPlay: { text: '选集/合集独立连播', status: ON, tip: '自动连播时选集/合集独立连播', fn: 'auto_list_play' }, circlePlay: { text: '循环播放', status: OFF, level: 1 }, }, btn: '其他设置', type: 'checkbox' }, otherCheckbox2: { options: { //不显示有明显变化的提示,关灯、关弹幕等,因为对有些人来说这些操作变化明显可见,提示反而多余且遮挡屏幕 hotKeyHint: { text: '快捷键屏幕提示', status: OFF, tip: '不显示有明显变化的提示,关灯、弹幕等' }, sortList: { text: '添加合集播放及弹幕数并支持排序', status: ON, tip: '网页刷新后生效' }, dynamicMainheader: { text: '动态显示隐藏BiliBili头部标题栏', status: OFF, tip: '', fn: 'dynamic_hide_mainheader' }, autoLike: { text: '自动点赞', status: OFF, tip: '可指定播放开始X秒后点赞(默认15秒),视频长度不足的播放完成后点赞', fn: 'auto_like', args:{ second: 15, args_opt: { tip: ['播放0~300秒后点赞【0代表打开页面即点赞】'] } } }, likeFollowed: { text: '只自动点赞已关注UP主视频', status: OFF, level: 1 }, }, btn: '其他设置Ⅱ', type: 'checkbox' }, keyBinding: { options: { 'lightOff': { text: '关灯/开灯', keystr: 'A', keycount: 1, keycode: 0, nonuse: OFF, }, //NumPad_Decimal 'onePerVolumeControl': { text: '非全屏音量调节${step}%', keystr: '滚↑轮 滚↓轮 NumPad_Decimal NumPad0', keycount: 4, delimiter: '、', nonuse: OFF, tempargs: [ {name: 'step', value: 1, opt: { step: 1, min: 0, max: '100', tip: ['调节幅度%'] }}, ], }, 'fivePerVolumeControl': { text: '非全屏音量调节${step}%', keystr: '↑ ↓', keycount: 2, nonuse: OFF, tempargs: [ {name: 'step', value: 5, opt: { step: 5, min: 0, max: '100', tip: ['调节幅度%'] }}, ], }, 'webFullscreen': { text: '网页全屏', keystr: 'W', keycount: 1, nonuse: OFF, }, 'widescreen': { text: '宽屏模式', keystr: 'Q', keycount: 1, nonuse: OFF, }, //nonuse test 'danmu': { text: '开关弹幕', keystr: 'D', keycount: 1, nonuse: OFF, }, 'danmuTopBottom': { text: '开关顶部/底部弹幕', keystr: 'T B', keycount: 2, nonuse: OFF, }, 'videoRepeat': { text: '洗脑循环', keystr: 'R', keycount: 1, nonuse: OFF, }, 'addsubDanmuOpacity': { text: '减增弹幕透明度${step}%', keystr: 'Z C', keycount: 2, nonuse: OFF, tempargs: [ {name: 'step', value: 10, opt: { step: 10, min: 0, max: '100', tip: ['调节幅度%'] }}, ], }, 'addsubDanmuFontsize': { text: '减增弹幕字号${step}%', keystr: 'V N', keycount: 2, nonuse: OFF, tempargs: [ {name: 'step', value: 10, opt: { step: 10, min: 0, max: '100', tip: ['调节幅度%'] }}, ], }, 'danmuArea': { text: '弹幕显示区域 1/4屏~不限', keystr: '1 2 3 4 5', keycount: 5, delimiter: '、', nonuse: OFF, }, 'wheelDanmuOpacity': { text: '增减弹幕透明度${step}%', keystr: 'Ctrl+滚↑轮 Ctrl+滚↓轮', keycount: 2, nonuse: OFF, tempargs: [ {name: 'step', value: 5, opt: { step: 5, min: 0, max: '100', tip: ['调节幅度%'] }}, ], }, 'fastForwardBackward30s': { text: '快进/快退${second}s', keystr: 'Ctrl+← Ctrl+→', keycount: 2, nonuse: OFF, tempargs: [ {name: 'second', value: 30, opt: { step: 10, min: 0, max: '600', tip: ['调节幅度-秒'] }}, ], }, 'frameByFrame': { text: '逐帧操作', keystr: 'Shift+← Shift+→', keycount: 2, keycode: 0, tip: '', keyshow: 'Shift + ←/→', nonuse: OFF, }, 'keyVideoSpeed': { text: '调节视频播放速度', keystr: 'Ctrl+↑ Ctrl+↓', keycount: 2, nonuse: OFF, }, 'keyVideoSpeed0dot1': { text: '0.1倍速调节视频播放速度', keystr: 'Ctrl+NumPad_Decimal Ctrl+NumPad0', keycount: 2, nonuse: OFF, }, 'ctrl_enterFullScreen': { text: '全屏', keystr: 'Ctrl+Enter', keycount: 1, nonuse: OFF, }, 'danmuZoomWithScreen': { text: '弹幕随屏幕缩放', keystr: 'S', keycount: 1, nonuse: OFF, }, 'like': { text: '点赞、投币、收藏、长按一键三连', keystr: 'Y U I O', keycount: 4, delimiter: '、', nonuse: OFF, }, 'setCustomPlayRate': { text: '调节自定义倍速速度', keystr: 'Shift+↑ Shift+↓', keycount: 2, nonuse: OFF, }, 'subtitle': { text: '开关字幕', keystr: 'E', keycount: 1, nonuse: OFF, }, 'watchLater': { text: '开关稍后再看', keystr: 'H', keycount: 1, nonuse: OFF, }, 'dblclickFullScreen': { text: '全屏', keystr: '双击/中键/功能键', keycount: 0, }, //keycount <= 0 不能更改 'loop': { text : '设定段落循环 / 设定后切换', keystr: 'Ctrl+鼠标左键 L', keycount: -2, }, 'dynamicProgress': { text : '快进/快退时显示醒目进度条', keystr: '← →', keycount: -2,}, 'dynamicSpace': { text : '合理空格键', keystr: 'Space', keycount: -1, }, 'dynamicEsc': { text : '优先取消右键菜单等', keystr: 'Escape', keycount: -1, }, 'dynamicEnter': { text : '有投币框时回车投币', keystr: 'Enter', keycount: -1, }, //系统热键,用sys前缀表示 'sysLike': { text:'单次点赞/取消点赞,长按一键三连', keystr:'Q', display: 'none', }, 'sysCoin': { text:'投币', keystr:'W', display: 'none', }, 'sysCollect': { text:'收藏', keystr:'E', display: 'none', }, 'sysThree': { text:'长按一键三连', keystr:'R', display: 'none', }, 'sysPlayPause': { text:'播放/暂停', keystr:'Space', display: 'none', }, 'sysFastForward': { text:'单次快进5s,长按倍速播放', keystr:'→', display: 'none', }, 'sysFastBackward': { text:'快退5s', keystr:'←', display: 'none', }, 'sysVolume+': { text:'音量增加10%', keystr:'↑', display: 'none', }, 'sysVolume-': { text:'音量降低10%', keystr:'↓', display: 'none', }, 'sysEscFullScreen': { text:'退出全屏', keystr:'Esc', display: 'none', }, 'sysPlayPause2': { text:'播放/暂停', keystr:'媒体键play/pause', display: 'none', }, 'sysFullScreen': { text:'全屏/退出全屏', keystr:'F', display: 'none', }, 'sysPrecPart': { text:'多P上一个', keystr:'[', display: 'none', }, 'sysNextpart': { text:'多P下一个', keystr:']', display: 'none', }, 'sysSendDM': { text:'发弹幕', keystr:'Enter', display: 'none', }, 'sysSwitchDM': { text:'开启/关闭弹幕', keystr:'D', display: 'none', }, 'sysSwitchMute': { text:'开启/关闭静音', keystr:'M', display: 'none', }, 'sysOneSpeed': { text:'一倍速(正常倍速)', keystr:'Shift+1', display: 'none', }, 'sysDoubleSpeed': { text:'二倍速', keystr:'Shift+2', display: 'none', }, }, btn: '快捷键设置', type: 'key' }, //参数 TODO // args: { // options: { // }, // btn: '参数', // type: 'args' // }, /* // quickDo: { // fullscreen: { value: 'f', text: '全屏', }, // webFullscreen: { value: 'w', text: '网页全屏', }, // widescreen: { value: 'q', text: '宽屏', }, // subSpeed: { value: '[', text: '减少速度', }, // addSpeed: { value: ']', text: '增加速度', }, // resetSpeed: { value: '\\', text: '重置速度', }, // danmu: { value: 'd', text: '弹幕', }, // playAndPause: { value: 'p', text: '暂停播放', }, // prevPart: { value: 'k', text: '上一P', }, // nextPart: { value: 'l', text: '下一P', }, // showDanmuInput: { value: 'enter', text: '发弹幕', }, // mirror: { value: 'j', text: '镜像', }, // danmuTop: { value: 't', text: '顶部弹幕', }, // danmuBottom: { value: 'b', text: '底部弹幕', }, // danmuScroll: { value: 's', text: '滚动弹幕', }, // danmuPrevent: { value: 'c', text: '防挡弹幕', }, // rotateRight: { value: 'o', text: '向右旋转', }, // rotateLeft: { value: 'i', text: '向左旋转', }, // lightOff: { value: 'y', text: '灯', }, // download: { value: 'z', text: '下载', }, // seek: { value: 'x', text: '空降', }, // mute: { value: 'm', text: '静音', }, // jump: { value: '', text: '跳转', }, // scroll2Top: { value: '', text: '回到顶部', }, // jumpContent: { value: '', text: '跳过鸣谢', }, // playerSetOnTop: { value: '', text: '播放器置顶', }, // setRepeatStart: { value: '', text: '循环起点', }, // setRepeatEnd: { value: '', text: '循环终点', }, // resetRepeat: { value: '', text: '清除循环点', }, // subVolume: { value: '', text: '减少音量', }, // addVolume: { value: '', text: '增加音量', }, // subProgress: { value: '', text: '快退', }, // addProgress: { value: '', text: '快进', }, // }, */ }, //快捷键 QDs: {}, //未使用 getQD(key) { return this.QDs[key]; }, //未使用 saveQD() { for (let [key, { value, text }] of Object.entries(this.hotKeyMenu)) { this.QDs[key] = {value: value, keyCode: value.charCodeAt(), text: text}; } }, //未使用 hotKeyMenu: { //只是右键菜单的数据,如需改动快捷键改run函数 'volumeControl': { value: '滚轮', text: '音量调节', }, 'lightOff': { value: 'A', text: '关灯/开灯', }, 'webFullscreen': { value: 'W', text: '网页全屏', }, 'widescreen': { value: 'Q', text: '宽屏模式', }, 'danmu': { value: 'D', text: '开关弹幕', }, 'danmuTopBottom': { value: 'T/B', text: '顶部/底部弹幕', }, 'videoRepeat': { value: 'R', text: '洗脑循环', }, 'addsubDanmuOpacity': { value: 'Z/C', text: '减增弹幕透明度10%', }, 'addsubDanmuFontsize': { value: 'V/N', text: '减增弹幕字号10%', }, 'danmuArea': { value: '1、2、3、4、5', text: '弹幕显示区域 1/4屏~不限', }, // 'quarterArea': { value: '1', text: '1/4屏', }, // 'halfArea': { value: '2', text: '半屏', }, // 'threeQuarterArea': { value: '3', text: '3/4屏', }, // 'nonOverArea': { value: '4', text: '不重叠', }, // 'fullArea': { value: '5', text: '不限', }, 'wheelDanmuOpacity': { value: 'Ctrl + 滚轮', text: '增减弹幕透明度5%', }, 'fastForwardBackward30s': { value: 'Ctrl + ←/→', text: '快进/快退30s', }, 'frameByFrame': { value: 'Shift + ←/→', text: '逐帧操作', }, 'keyVideoSpeed': { value: 'Ctrl + ↑/↓', text: '调节视频播放速度', }, 'dblclickFullScreen': { value: '双击/中键/功能键', text: '全屏', }, 'ctrl_enterFullScreen': { value: 'Ctrl+Enter', text: '全屏', }, 'danmuZoomWithScreen': { value: 'S', text: '弹幕是否随屏幕缩放', }, 'like': { value: 'Y、U、I、O', text: '点赞、投币、收藏、长按一键三连', }, 'loop': { value: 'Ctrl + 左键 / L', text: '设定段落循环 / 设定后切换', }, }, }; const assert = function(condition, message) { if (!condition) throw Error('Assert failed: ' + (message || 'Assertion failed')); } let log = console.log let dir = console.dir let err = console.error function set_value(key, value) { localStorage.setItem(key, JSON.stringify(value)) } function get_value(key, default_value) { return JSON.parse(localStorage.getItem(key)) || default_value } function set_value2(key, value){ localStorage.setItem(key, value) } function get_value2(key, default_value){ return localStorage.getItem(key) || default_value } function sec2str(t){ var d = Math.floor(t/86400), h = (''+Math.floor(t/3600) % 24).slice(-2), m = ('0'+Math.floor(t/60)%60).slice(-2), s = ('0' + t % 60).slice(-2); return (d>0?d+'天':'')+((d>0||h>0)?h+':':'')+m+':'+s; } // 对Date的扩展,将 Date 转化为指定格式的String // 月(M)、日(d)、小时(h)、分(m)、秒(s)、季度(q) 可以用 1-2 个占位符, // 年(y)可以用 1-4 个占位符,毫秒(S)只能用 1 个占位符(是 1-3 位的数字) // 例子: // (new Date()).Format("yyyy-MM-dd hh:mm:ss.S") ==> 2006-07-02 08:09:04.423 // (new Date()).Format("yyyy-M-d h:m:s.S") ==> 2006-7-2 8:9:4.18 Date.prototype.Format = function (fmt) { var o = { "M+": this.getMonth() + 1, //月份 "d+": this.getDate(), //日 "h+": this.getHours(), //小时 "m+": this.getMinutes(), //分 "s+": this.getSeconds(), //秒 "q+": Math.floor((this.getMonth() + 3) / 3), //季度 "S": this.getMilliseconds() //毫秒 }; if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length)); for (var k in o) if (new RegExp("(" + k + ")").test(fmt)) fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length))); return fmt; } function waitForNode(nodeSelector, callback, time=100) { //log('----n---'); if(--time < 0) {err('waitForNode 超时 '+nodeSelector); return false;} if(time < 0) return; var node = nodeSelector(); if (node) { callback(node); } else { time-=1; setTimeout(function() { waitForNode(nodeSelector, callback, time); }, 50); } } function waitForTrue(ifTrue, callback, time=100) { if(--time < 0) {err('waitForTrue 超时 '+ifTrue); return false;} const fn = waitForTrue; //let fn = arguments.callee; if (ifTrue()) { callback(); return true; } else { setTimeout(function() { fn(ifTrue, callback, time); }, 50); } } function waitForTrue_async(ifTrue, callback, time=50){ return new Promise((resolve, reject)=>{ let tid = setInterval(()=>{ if(--time < 0) { clearInterval(tid); reject('waitForTrue 超时 '+ifTrue) } if (ifTrue()) { clearInterval(tid); callback(); resolve() } },50) }) } function geth5Player(){ return $(`${bb['video']} video`)[0] || $(`${bb['video']} bwp-video`)[0] } function is_fullscreen() { //bpx-player-container bpx-state-paused //if(bb_type.indexOf(BILI_2_X) !== -1){ return bili_player.isFullScreen() } if(bb_type.indexOf(BILI_2_X) !== -1){ return $('#bilibiliPlayer').hasClass('mode-webfullscreen') || $('#bilibiliPlayer').hasClass('mode-fullscreen') } else if(bb_type.indexOf(BILI_3_X_VIDEO) !== -1){ return $('.bpx-player-container').attr('data-screen') === 'web' || $('.bpx-player-container').attr('data-screen') === 'full' } //normal wide web full } function toggle_control_top(show) { if(bb_type.indexOf(BILI_2_X) !== -1){ if($(bb['playArea']).hasClass('video-control-show') === true) $(bb['playVideoControlWrap']).mouseout(); else $(bb['playVideoControlWrap']).mousemove(); }else{ if($('.bpx-player-container').attr('data-ctrl-hidden') === "true"){ //控制栏隐藏状态 if(show !== 'hide'){ $('.bpx-player-container').attr('data-ctrl-hidden', "false"); $('.bpx-player-control-entity').attr('data-shadow-show','false'); } }else{ if(show !== 'show'){ $('.bpx-player-container').attr('data-ctrl-hidden', "true"); $('.bpx-player-control-entity').attr('data-shadow-show','true'); } } } } function fullscreen() { $(bb['fullScreen']).click() } function web_fullscreen() { $(bb['webFullScreen']).click() } function wide_screen() { $(bb['wideScreen']).click() } function subtitle() { $(bb['playCtrlSubtitle']).click() } function watchLater() { $(bb['playWatchLater']).click() } function is_watchlater_added() { return $(bb['playWatchLater'])[0] && $(bb['playWatchLater']+'.added').length > 0 } function auto_top_quality_video(status) { //自动选择最高质量视频 if(status !== ON || biliinfo.is_login !== true) return if(!biliinfo.vip_status) { //选大会员码率之下最高的 if($('.bpx-player-ctrl-quality-menu-item:contains(大会员)').length !== 0) $('.bpx-player-ctrl-quality-menu-item:contains(大会员):last').next().click() else $('.bpx-player-ctrl-quality-menu-item:first').click() }else{ $('.bpx-player-ctrl-quality-menu-item:first').click() } } function is_lightoff() { //if (bb_type.indexOf(BILI_2_X) !== -1) { return !bili_player.getPlayerState().lightOn } if (bb_type.indexOf(BILI_2_X) !== -1) { return $('#bilibiliPlayer').hasClass('mode-light-off') } else if(bb_type === BILI_3_X_MOVIE) { return $(bb['playSettingLightoff']).hasClass('active') } else if(bb_type.indexOf(BILI_3_X_VIDEO) !== -1) { return $('.bpx-docker-major').hasClass('bpx-state-light-off') } } function is_danmaku_show(){ return $(bb['danmakuSwitch']+' '+bb['switchInput']+':last')[0].checked; } function lightoff() { $(bb['playSettingLightoff']).click(); } function add_tip($tip, $node, args){ //log('addtip '+$tip.length) $node.mouseover(function(){ var e = this; var op = {}; let t = setTimeout(function() { $tip.appendTo(e); let $this = $(e); let $new_tip = $('.player-tooltips.tip.top-center.animation').css({top:0,left:0}); true === !!$this.data('tip') ? $.extend(true, op, $this.data('tip')) : $.extend(true, op, args); $new_tip.find('.tooltip').text(op['text']); let tip_new_top = $this.offset().top - $new_tip.offset().top - $new_tip.height(); //console.log('top:'+$this.offset().top+' tiptop: '+$new_tip.offset().top +' height: '+ $new_tip.height()); //console.log('left:'+$this.offset().left+' tipleft: '+$new_tip.offset().left +' width: '+ $this.width()+' tipwidth: '+ $new_tip.width()); //console.log('top:'+$this.css('top')+'left:'+$this.css('left')) let tip_new_left = $this.offset().left - $new_tip.offset().left + ($this.width() - $new_tip.width())/2; //console.log(tip_new_top+' '+tip_new_left); tip_new_left = tip_new_left <= 0 ? 0 : tip_new_left; let diff_full_top = is_fullscreen() ? (op['diff_full_top'] || 0) : 0; //全屏时位置不同 $new_tip.css({'top':tip_new_top + op['top'] + diff_full_top, 'left':tip_new_left + op['left']}).addClass('active'); }, op['millisec']); $(this).data('timeout', t); }). mouseout(function(e){ clearTimeout($(this).data('timeout')); $('.player-tooltips.tip.top-center.animation').removeClass('active'); $('.player-tooltips.tip.top-center.animation').remove(); }) } //关灯按钮样式 function lightoff_btn_css() { var body_brgb = 'rgb(160, 130, 110)'; var dot_crgb = 'rgb(230, 200, 180)'; var dot_brgb = 'rgb(50, 50, 50)'; var dark_rgb = 'rgb(77, 77, 77)'; if ($('#hhh_lightoff '+bb['switchInput'])[0].checked === false) { //关灯 $('#hhh_lightoff '+bb['switchBody']+':first').css('background-color', dark_rgb); $('#hhh_lightoff '+bb['switchBody']+':first>'+bb['switchDot']).css('color', dark_rgb); $(`#hhh_lightoff ${bb['switchLabel']}`).removeClass('checked'); } else { $('#hhh_lightoff '+bb['switchBody']+':first').css('background-color', body_brgb); $('#hhh_lightoff '+bb['switchBody']+':first>'+bb['switchDot']).css({'color': dot_crgb, 'background-color': dot_brgb}); $(`#hhh_lightoff ${bb['switchLabel']}`).addClass('checked'); } } //关灯按钮 function lightoff_btn(lightoff_entry_state = LIGHT_DEFAULT) { //lightoff_entry_state - 此状态标明谁执行的关灯操作 is_show_hint && (is_lightoff() ? showHint(parent, '#hhh_wordsHint', '开灯') : showHint(parent, '#hhh_wordsHint', '关灯')) lightoff() //$('#hhh_lightoff .bui-switch-input').click() let $light_input = $('#hhh_lightoff '+bb['switchInput']) if(is_lightoff() === $light_input[0].checked) { //checked==true开灯 false关灯 $light_input[0].checked = !$light_input[0].checked } lightoff_btn_css() //更新关灯tip,两种方式更新light_tip_text tip_update_3_X({ target: $('#hhh_lightoff'), tip_target: $('#hhh_tip'), gap: 6 }) //title参数为空,原tip自更新 //$('#hhh_lightoff .tooltip').text(light_tip_text) //2.x //记录关灯状态(排除下滚时产生的开关灯状态) if(lightoff_entry_state !== LIGHTON_LIKE) { biliinfo.lightoff_state = is_lightoff() lightOnWhenLike(config.getCheckboxSettingStatus('lightOnWhenLike'), config.getCheckboxSettingArgs('lightOnWhenLike', 'screen_top')) //log('lightoff_entry_state:',lightoff_entry_state,'is_lightoff:', is_lightoff(), 'lightoff_state:', biliinfo.lightoff_state) } } //关灯按钮初始化 function lightoff_init() { //插入关灯按钮 $(`${bb['danmakuSwitch']}:first`).clone().prependTo(`${bb['danmakuRoot']}:first`)[0].id = 'hhh_lightoff'; $('#hhh_lightoff '+bb['switchInput'])[0].checked = true; if(bb_type !== BILI_3_X_VIDEO_V1 && bb_type !== BILI_3_X_MOVIE) $(`#hhh_lightoff ${bb['switchDot']}`)[0].innerHTML = '灯'; else{ //开灯 $('#hhh_lightoff').find('.bui-danmaku-switch-on svg').replaceWith(` ` ); //关灯 $('#hhh_lightoff').find('.bui-danmaku-switch-off svg').replaceWith(` ` ); } //css lightoff_btn_css(); //add tip if(bb_type.indexOf(BILI_2_X) !== -1){ waitForTrue(()=> ($tip !== undefined), () => { // if(bb_type === BILI_3_X_MOVIE) $('.squirtle-controller-wrap').css({"display":"flex"}); // $(bb['dmWrap']).css({"display":"block"}); if(bb_type === BILI_2_X_V2) $('#hhh_lightoff').data('tip', {'text': '关灯', 'millisec':1, 'top':2, 'left':0, 'diff_full_top':-20}); else if(bb_type === BILI_3_X_VIDEO_V1) $('#hhh_lightoff').data('tip', {'text': '关灯', 'millisec':1, 'top':-4, 'left':0, 'diff_full_top':-14}); add_tip($tip, $('#hhh_lightoff'), {}); // $(bb['dmWrap']).css({"display":"none"}); // if(bb_type === BILI_3_X_MOVIE) $('.squirtle-controller-wrap').css({"display":"none"}); }) } else { //创建关灯tip tip_create_3_X({ target: $('#hhh_lightoff'), tip_target: $('#hhh_tip'), gap: 6, title: ()=>{ let $light_input = $('#hhh_lightoff '+bb['switchInput']); let light_tip_text = $light_input[0].checked === true? '关灯': '开灯'; light_tip_text += ' '+`(${config.sets.keyBinding.options.lightOff.keystr.toLowerCase()})`; return light_tip_text; } }); } } //模拟B站音量调节 //2.X版本可以直接调用系统函数window.player.volume(),但不能直接使用H5Player.volume //3.X版本去掉了window.player.volume(),但H5Player.volume功能发生变动,基本等价于window.player.volume() function volume(v, is_show_hint=true){ function volume_bar(v){ //未使用 if(v === undefined) return; v = v<0? 0: v>1? 1: v; $('.bilibili-player-video-volume-num').text(Math.round(v*100)); $('.bilibili-player-video-volumebar-wrp .bui-bar.bui-bar-normal')[0].style.transform = `scaleY(${v})`; $('.bilibili-player-video-volumebar-wrp .bui-thumb')[0].style.transform = `translateY(${-48*v}px)`; v === 0? $('.bilibili-player-video-btn.bilibili-player-video-btn-volume').addClass('video-state-volume-min'): $('.bilibili-player-video-btn.bilibili-player-video-btn-volume').removeClass('video-state-volume-min') } function volume_hint_bar(v){ if(v === undefined) return; assert(typeof v === "number", '| volume err: v is not number') //debug //v = v<0? 0: v>1? 1: v; v = +Math.max(Math.min(v,1),0).toFixed(3); var $volumeHintIcon = $(`#hhh_volumeHint ${bb['volumeHintIcon']}`); var volumeHintIconClassName = bb['volumeHintIcon'].substr(1); if(bb_type.indexOf(BILI_2_X) !== -1){ if(v <= 0) $volumeHintIcon.attr('class', `${volumeHintIconClassName} video-state-volume-min`); else if(v >= 1) $volumeHintIcon.attr('class', `${volumeHintIconClassName} video-state-volume-max`); else $volumeHintIcon.attr('class', volumeHintIconClassName); } else{ if(v <= 0){ $volumeHintIcon.find('svg:last').css('display','block'); $volumeHintIcon.find('svg:first').css('display','none'); } else{ $volumeHintIcon.find('svg:last').css('display','none'); $volumeHintIcon.find('svg:first').css('display','block'); } } if(v <= 0) showHint(this, '#hhh_volumeHint', '静音'); else showHint(this, '#hhh_volumeHint', Math.round(v*100)+'%'); } h5Player = geth5Player() // bug:调节清晰度会影响 h5Player 和window.player,需要重新赋值 assert(typeof h5Player.volume === "number", '| volume err: h5Player.volume is not number') //debug if(v === undefined) { /*log('v: '+h5Player.volume);*/ return +h5Player.volume;} v = +Math.max(Math.min(v,1),0).toFixed(3); assert(typeof v === "number", '| volume err: v is not number') //debug //log('volume: '+v+' '+h5Player); //v = v<0.01? 0: v>1? 1: v; //0.01防止误差 bb_type.indexOf(BILI_2_X) !== -1? window.player.volume(v): h5Player.volume = v; //h5Player.volume = v; if(is_show_hint === true) volume_hint_bar(+h5Player.volume); return +h5Player.volume; } //显示提示 function showHint(parent, selector_str, text, t=1e3){ $(bb['volumeHint']).css({"visibility":"visible"}) $(bb['volumeHint']).css('display', 'none'); //隐藏所有提示,避免提示重叠 $(`${selector_str}>${bb['volumeHintText']}`).text(text); //百分比显示 var Hint = $(selector_str); //显示及渐隐效果(抄bilibili^^) clearTimeout(parent.showHintTimer), Hint.stop().css("opacity", 1).show(), parent.showHintTimer = window.setTimeout((function() { Hint.animate({ opacity: 0 }, 300, (function() { $(this).hide() })) } ), t) } /* * 控制进度条 * .bilibili-player-setting-opacity 透明度 * .bilibili-player-setting-area 显示区域 * .bilibili-player-setting-speedplus 弹幕速度 等 * 利用系统mousedown事件 * 0 ~ 100 */ function set_progress(selector_str, percent, limit_left, limit_right){ function calc_bar_offset2(percent, bar_width, limit_left, limit_right){ //某种插值算法,未使用 let p = +percent; p = plimit_right? limit_right: p; //log(p+' - '+bar_width+' - '+(limit_right-limit_left)); let limits = limit_right - limit_left; let quo = Math.floor((p-limit_left)*bar_width/limits); let rem = (p-limit_left)*bar_width%limits; //log(quo+'****'+rem); return (bb_type === BILI_3_X_MOVIE)? Math.round((p-limit_left)/limits*bar_width): (rem>=(limits/2)? quo+1: quo); //百分比对应进度条位置 } function calc_bar_offset(percent, bar_width, limit_left, limit_right){ let p = Math.max(Math.min(+percent, limit_right), limit_left); //log(p+' - '+bar_width+' - '+(limit_right-limit_left)); let limits = limit_right - limit_left; let bar_offset = (p-limit_left) / limits * bar_width; return Math.round(bar_offset); //百分比对应进度条位置 } let selector = document.querySelector(selector_str); let e1 = new MouseEvent('mousedown'), e2 = new MouseEvent('mouseup'); if(bb_type === BILI_3_X_MOVIE) $('.squirtle-controller-wrap').css({"display":"flex"}); $(bb['dmWrap']).css({"display":"block"}); let selector_rect = selector.getClientRects(); //log($(`${selector_str} ${bb['progressWrap']}`).innerWidth()) let bar_offset = calc_bar_offset(percent, $(`${selector_str} ${bb['progressWrap']}`).innerWidth(), limit_left, limit_right); //log('wrapWIdth: '+ calc_bar_offset(percent, $(`${selector_str} ${bb['progressWrap']}`).innerWidth(), limit_left, limit_right)); for(let i=0;i<10;i++){ let dest_per = Math.max(Math.min(+percent, limit_right), limit_left); let clientX = selector_rect[0].left + bar_offset; e1.initMouseEvent('mousedown',1,1,window,1,0,0,clientX,0,0,0,0,0,0,null); e2.initMouseEvent('mouseup' ,1,1,window,1,0,0,clientX,0,0,0,0,0,0,null); selector.dispatchEvent(e1); selector.dispatchEvent(e2); let curr_per = +$(`${selector_str} ${bb['progressVal']}`).text().slice(0,-1); //log(curr_per+' - '+dest_per); if(curr_per !== dest_per){ curr_per < dest_per ? ++bar_offset : --bar_offset; //log(bar_offset); }else{ break; } } $(bb['dmWrap']).css({"display":"none"}); if(bb_type === BILI_3_X_MOVIE) $('.squirtle-controller-wrap').css({"display":"none"}); //激活设置,记忆进度条位置 if(bb_type === BILI_3_X_MOVIE){ $(bb['dm'])[0].dispatchEvent( new MouseEvent('mouseleave') ); //3.X }else{ $(bb['dm'])[0].dispatchEvent( new MouseEvent('mouseout') ); //2.X } return $(`${selector_str} ${bb['progressVal']}`).text(); } /* * 调节弹幕设置进度条 * 利用系统mousedown事件 * '正数': right, '负数': left, -100 ~ +100 */ function adjust_progress(selector_str, inc_percent, limit_left, limit_right){ var curr_percent = Number($(`${selector_str} ${bb['progressVal']}`).text().slice(0,-1)); return set_progress(selector_str, curr_percent + inc_percent, limit_left, limit_right); } /* / \ | | \ / clip-path: polygon(40% 0%, 25% 30%, 25% 70%, 40% 100%, 0% 100%, 0% 0%,60% 0%, 75% 30%, 75% 70%, 60% 100%, 100% 100%, 100% 0%);
11111111111111111111111
11111111111111111111111
11111111111111111111111
11111111111111111111111
11111111111111111111111
11111111111111111111111
11111111111111111111111
11111111111111111111111
11111111111111111111111
$('.bilibili-player-context-menu-container.active li:last').mouseenter(function(){ let hotkey_class = '.bilibili-player-hotkey-panel-container'.substr(1); $('#hhh_hotkey_panel').clone(true,true).replaceAll($(`.${hotkey_class}:last`)).attr({'id': '', 'class': hotkey_class, 'style': ''}); let $hotkey = $('.bilibili-player-hotkey-panel-container').addClass('active').css('z-index', $('.bilibili-player-video-control-wrap').css('z-index')); }).mouseleave(function(){ $('.bilibili-player-hotkey-panel-container').removeClass('active').css('display', 'none'); }) BV to AV // 作者:mcfx // 链接:https://www.zhihu.com/question/381784377/answer/1099438784 // 来源:知乎 // 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。 function bv2av() { const bv = document.getElementById('bv').value; document.getElementById('av').value = dec(bv); } function av2bv() { const av = +document.getElementById('av').value; document.getElementById('bv').value = enc(av); } const table = 'fZodR9XQDSUm21yCkr6zBqiveYah8bt4xsWpHnJE7jL5VG3guMTKNPAwcF' const tr = {} for (let i = 0; i < 58; i++) tr[table[i]] = i const s = [11, 10, 3, 8, 4, 6] const xor = 177451812 const add = 8728348608 function dec(x) { let r = 0 for (let i = 0; i < 6; i++) { r += tr[x[s[i]]] * 58 ** i } return (r - add) ^ xor } function enc(x) { x = (x ^ xor) + add r = 'BV1 4 1 7 '.split('') for (let i = 0; i < 6; i++) { r[s[i]] = table[parseInt(x / 58 ** i) % 58] } return r.join('') } // console.log(dec('BV17x411w7KC')) // console.log(dec('BV1Q541167Qg')) // console.log(dec('BV1mK4y1C7Bz')) // console.log(enc(170001)) // console.log(enc(455017605)) // console.log(enc(882584971)) // console.log('----------------------------'); // console.log(1, enc(1)) // console.log(93761707, dec(enc(93761707))) // console.log('BV1xE411x7Mn', enc(dec('BV1xE411x7Mn'))) // console.log(add === 100618342136696320, add); */ //自定义快捷键说明页面 function add_custom_hotkey(hotkey_panel_class, hotkey, is_append) { var $hotkey_panel = $(`.${hotkey_panel_class}`) var $hotkey_item = $hotkey_panel.find(`.${hotkey_panel_class}-item:first`) var key_name = bb_type.indexOf(BILI_2_X) !== -1 ? 'key' : 'name' if(is_append === false) $hotkey_panel.empty() for (let [key, { keystr, text, keycount, delimiter, display, tempargs }] of Object.entries(hotkey)) { if(display === 'none') continue if(keycount !== 0){ // keycount=0 原样显示 let keystr_arr = keystr.split(/\s+/) keystr_arr = kb.combine_same_keys(keystr_arr) keystr = keystr_arr.join(delimiter || '/') } $hotkey_item.clone().appendTo($hotkey_panel).css('display', 'block') let $hotkey_key = $hotkey_panel.find(`.${hotkey_panel_class}-${key_name}:last`) $hotkey_key.text(keystr) //value const argsvalue = tempargs?.[0].value if(!!argsvalue){ //替换形如${...} text = text.replace(/\$\{\w+\}/, argsvalue) } $hotkey_key.next().text(text) $hotkey_key.width(200) //$hotkey_key.next().width(185) } } //添加右键菜单自定义快捷键说明2.X function add_custom_hotkey_menu_2_X(custom_hotkey) { //主class名,去掉"." let hotkey_panel_class = 'bilibili-player-hotkey-panel', is_init = false; //设置右键 & 快捷键菜单 (function add_menu(){ $('#bilibiliPlayer')[0].addEventListener('DOMSubtreeModified', function fn_(e) { //typeof e.target.className === 'string' && e.target.className !== '' && log(e.target.className); let $target = $(e.target); if($target.hasClass('context-line context-menu-function hover') && $('.bilibili-player-hotkey-panel').length === 1){ if(is_init === false){ //log('-------添加快捷键菜单--------'); is_init = true; let $last_item = $('.bilibili-player-hotkey-panel').find('.bilibili-player-hotkey-panel-item:last'); $last_item.attr('id', 'hhh_last_system_hotkey_panel_item'); add_custom_hotkey(hotkey_panel_class, custom_hotkey, true); }else if($target[0].id !== '__sizzle__'){ //过滤掉第一个懒加载 let $last_system_itme = $('#hhh_last_system_hotkey_panel_item'); //log($target.find('a').text()); if($target.find('a').text() === '快捷键说明'){ //log('--------显示快捷键说明-------'); $last_system_itme.prevAll().css('display','block'); $last_system_itme.css('display','block'); $last_system_itme.nextAll().css('display','none'); }else if($target.find('a').text() === '快捷键说明(bilibili关灯)'){ //log('--------显示快捷键说明(bilibili关灯)-------'); $last_system_itme.prevAll().css('display','none'); $last_system_itme.css('display','none'); $last_system_itme.nextAll().css('display','block'); } let wheelEvent = new WheelEvent('mousewheel', { deltaY: -10000, deltaMode: 0 }); //打开时回到最上 $('.bilibili-player-hotkey-panel-wrap')[0].dispatchEvent(wheelEvent); $('.bilibili-player-hotkey-panel-wrap').find('.bscroll-vertical-scrollbar')[0].style.cssText+=";background:#f1f1f1!important"; //默认滚动条对比度不明显,浅化背景颜色,增强对比度 } } else if($target.hasClass('bilibili-player-context-menu-container black bilibili-player-context-menu-origin') && $target[0].id === '') { //还有一个id = __sizzle__ this.removeEventListener('DOMSubtreeModified', fn_); //this.removeEventListener('DOMSubtreeModified', arguments.callee); //log('--------添加右键菜单-------'); let $li_sys_hotkey = $target.find('a:contains("快捷键说明"):first').parent(); //返回li,XXX: 此时contains("快捷键说明")多于一个 let $a_hhh_hotkey = $target.find('a:contains("快捷键说明(bilibili关灯)")'); var $li_hhh_hotkey = ($a_hhh_hotkey.length && $a_hhh_hotkey.parent()) || $li_sys_hotkey.clone(false, false).insertAfter($li_sys_hotkey).css('display', '').find('a').text('快捷键说明(bilibili关灯)'); //插入自定义快捷键 add_menu(); } }); })(); } //添加右键菜单自定义快捷键说明3.X function add_custom_hotkey_menu_3_X(custom_hotkey) { //主class名,去掉"." let hotkey_panel_class = 'bpx-player-hotkey-panel-content'; //监听右键菜单,生成自定义快捷键说明页面,只执行一次 (function set_custom_hotkey(){ $('.bpx-player-container')[0].addEventListener('DOMNodeInserted', function(e) { //typeof e.target.className === 'string' && e.target.className !== '' && log(e.target.className); let $target = $(e.target); if($target.hasClass('bpx-player-hotkey-panel')){ //log('-------添加快捷键菜单--------'); let $last_item = $target.find('.bpx-player-hotkey-panel-content-item:last'); $last_item.attr('id', 'hhh_last_system_hotkey_panel_item'); add_custom_hotkey(hotkey_panel_class, custom_hotkey, true); } }); })(); //右键菜单弹出时添加项 (function add_menu(){ $('.bpx-player-container')[0].addEventListener('DOMSubtreeModified', function fn_(e) { //typeof e.target.className === 'string' && e.target.className !== '' && log(e.target.className); let $target = $(e.target); if($target.hasClass('bpx-player-contextmenu bpx-player-black') && $target[0].childNodes.length !== 0) { this.removeEventListener('DOMSubtreeModified', fn_); //this.removeEventListener('DOMSubtreeModified', arguments.callee); //log('--------添加右键菜单-------'); let $li_sys_hotkey = $target.find('li:contains("快捷键说明"):first'); var $li_hhh_hotkey = $li_sys_hotkey.clone(false, false).insertAfter($li_sys_hotkey).text('快捷键说明(bilibili关灯)'); //插入自定义快捷键 let $last_system_itme = $('#hhh_last_system_hotkey_panel_item'); $last_system_itme.nextAll().remove(); //重新加载自定义快捷键菜单 add_custom_hotkey(hotkey_panel_class, custom_hotkey, true); //log('--------显示快捷键说明-------'); $li_sys_hotkey.off('mousedown.sys_hotkey'); $li_sys_hotkey.on('mousedown.sys_hotkey', function(){ $last_system_itme.prevAll().css('display','block'); $last_system_itme.css('display','block'); $last_system_itme.nextAll().css('display','none'); $('.bpx-player-hotkey-panel').width(400) }); //log('--------显示快捷键说明(bilibili关灯)-------'); $li_hhh_hotkey.off('mousedown.hhh_hotkey'); $li_hhh_hotkey.on('mousedown.hhh_hotkey', function(){ $last_system_itme.prevAll().css('display','none'); $last_system_itme.css('display','none'); $last_system_itme.nextAll().css('display','block'); $('.bpx-player-hotkey-panel').width(440) }); let wheelEvent = new WheelEvent('mousewheel', { deltaY: -10000, deltaMode: 0 }); //打开时回到最上 $('.bpx-player-hotkey-panel-area')[0].dispatchEvent(wheelEvent); add_menu(); } }); })(); //模拟右键菜单消息,激活菜单 let evt = new MouseEvent('contextmenu', { clientX:-9999, clientY:-9999 }); $(bb['videoWrap'])[0].dispatchEvent(evt); //模拟点击菜单,激活热键菜单DOM evt = new MouseEvent('mousedown',{ bubbles:true }); $(`.bpx-player-contextmenu.bpx-player-black li:contains("快捷键说明")`)[0].dispatchEvent(evt); //关闭热键菜单 $('.bpx-player-hotkey-panel-close').click(); } //添加右键菜单自定义快捷键说明 function add_custom_hotkey_menu(hotKeyMenu) { if(bb_type.indexOf(BILI_2_X) !== -1) add_custom_hotkey_menu_2_X(hotKeyMenu); else add_custom_hotkey_menu_3_X(hotKeyMenu); } //取得版本号-未使用 //function get_ver() { // if(bb_type.indexOf(BILI_2_X) !== -1) ver = bili_player.getVersion().version; // else ver = $('.bpx-common-opacity-60').text().split('-')[0]; //} //取得视频FPS(Frames Per Second)、版本、biliinfo function get_video_fps_ver_biliinfo() { $(bb['videoWrap'])[0].addEventListener('DOMNodeInserted', function fn_(e) { //插入info面板时截取fps值 if($(e.target).hasClass(bb['DOMNodeInsertedVideoInfoShow'].substr(1))) { this.removeEventListener('DOMNodeInserted', fn_); //this.removeEventListener('DOMNodeInserted', arguments.callee); let $video_info_close = $(bb['videoInfoClose']); $video_info_close.click(); //模拟关闭统计信息面板,2.X版本一次即可关掉 $video_info_close.click(); //3.X版本执行两次才能关掉面板,保险起见3次 $video_info_close.click(); //估计是点击事件开始太快,系统未能处理,也可能3.X测试版自己的问题 //text中取得fps值 var get_title_text = function(title) { return $(bb['videoInfoContainer']).find(`.info-title:contains("${title}")`).next().text() } fps = Number(get_title_text('Resolution').match(/\d+\.\d+/)) || Number(get_title_text('FPS')) || 30 //是否登录、是否大会员 waitForTrue(()=> !!window?.__BiliUser__?.cache?.data?.isLogin, () => { biliinfo.is_login = window.__BiliUser__.cache.data.isLogin biliinfo.vip_status = biliinfo.is_login === true ? window.__BiliUser__.cache.data.vipStatus : undefined }) } }) //模拟右键菜单消息,激活菜单DOM let evt = new MouseEvent('contextmenu', { clientX:-9999, clientY:-9999 }); $(bb['videoContextMenu'])[0].dispatchEvent(evt); //模拟点击菜单,激活热键菜单DOM,版本号 if(bb_type.indexOf(BILI_2_X) !== -1){ let evt = new MouseEvent('click',{ bubbles:true }); $(`${bb['playerContextMenu']} a:contains("视频统计信息")`)[0].dispatchEvent(evt); ver2 = $(`${bb['playerContextMenu']} a:contains("更新历史")`).text().split(' ')[1].split('-')[0]; } else if(bb_type.indexOf(BILI_3_X_VIDEO) !== -1) { let evt = new MouseEvent('mousedown',{ bubbles:true }); $(`${bb['playerContextMenu']} li:contains("视频统计信息")`)[0].dispatchEvent(evt); ver2 = $(`${bb['playerContextMenu']} li:contains("更新历史")`).text().match(/\S+/g)[1].split('-')[0]; } } //笨办法,激活系统音量设置,复制volumeHint DOM function pick_volume_hint(){ var original_volume = h5Player.volume; //监视提取提示DOM $(bb['videoWrap'])[0].addEventListener('DOMNodeInserted', function fn_(e) { if($(e.target).hasClass(bb['volumeHint'].substr(1))) { this.removeEventListener('DOMNodeInserted', fn_); //this.removeEventListener('DOMNodeInserted', arguments.callee); volume(original_volume); //模拟鼠标拖动无法按1%精确控制音量,系统自身限制或者说bug //添加 volumeHint wordsHint opacityHint DOM $(bb['volumeHint']).clone().appendTo(bb['videoWrap']).attr('id','hhh_volumeHint'); $('#hhh_volumeHint').clone().appendTo(bb['videoWrap']).attr('id','hhh_wordsHint').css({'opacity': 0, 'display': 'none'}).find(bb['volumeHintIcon']).remove(); $('#hhh_wordsHint').clone().appendTo(bb['videoWrap']).attr('id','hhh_opacityHint'); if(bb_type.indexOf(BILI_2_X) !== -1){ $('#hhh_wordsHint').css({'width':'auto','margin-left':'0px','padding-left':'8px','padding-right':'15px','transform':'translate(-50%)'}) .find(bb['volumeHintText']).css({'width': 'auto', 'padding-left': '10px'}); $('#hhh_opacityHint').find(bb['volumeHintText']).css({'padding-right': '6px'}); } else{ $('#hhh_wordsHint, #hhh_opacityHint').find(bb['volumeHintText']).css({'padding': '0 10px'}); } //隐藏提示 $(bb['volumeHint']).css({"visibility":"hidden"}); } }); //激活系统音量设置,复制volumeHint DOM let evt = new KeyboardEvent('keydown', { keyCode:keycode['up'] }); window.dispatchEvent(evt); } //快进时显示醒目进度条 function dynamicProgress(dynamicHeight, staticHeight){ if(bb_type.indexOf(BILI_2_X) !== -1) { if (1||$('.bilibili-player-area').hasClass('progress-shadow-show') === true) { $('.bilibili-player-video-progress-shadow .bui-track-video-progress').css('cssText', `height:${dynamicHeight}px !important`); clearTimeout(document.showVideoProgress); document.showVideoProgress = window.setTimeout((function() { $('.bilibili-player-video-progress-shadow .bui-track-video-progress').css('cssText', `height:${staticHeight}px !important`); }), 2000); } } else { if($('.bpx-player-container').hasClass('bpx-state-no-cursor') === true || //3.x $('.bpx-player-control-entity').hasClass('bpx-state-no-cursor') === true) { //3.x.13 $('.bpx-player-shadow-progress-area').css({'height': `${dynamicHeight}px`}); //3.x.13 //$('.bpx-player-progress').css({'height': `${dynamicHeight/2}px`}); $('.squirtle-progress-bar').css({'height': `${dynamicHeight}px`}); //3.x clearTimeout(document.showVideoProgress); document.showVideoProgress = window.setTimeout((function() { $('.bpx-player-shadow-progress-area').css({'height': `${staticHeight}px`}); //$('.bpx-player-progress').css({'height': `${staticHeight/2}px`}); $('.squirtle-progress-bar').css({'height': `${staticHeight}px`}); }), 2000); } } } //修复选择历史弹幕时弹幕填装信息丢失问题 function fix_danmaku_info() { if(bb_type.indexOf(BILI_2_X) !== -1) { //log($('.danmaku-box').length); $('.danmaku-box')[0].addEventListener('DOMSubtreeModified', function(e) { if($(e.target).hasClass('player-auxiliary-danmaku-list bpui-component bpui-undefined bpui-selectable') && $('.bilibili-player-video-info').hasClass('bilibili-player-hide-dm')) { $('.bilibili-player-video-info').removeClass('bilibili-player-hide-dm'); //得到弹幕数 let danmaku_number = $('.player-auxiliary-danmaku-contaner .player-auxiliary-danmaku-list').height()/$('.danmaku-info-row:first').height(); $('.bilibili-player-video-info-danmaku-number').text(danmaku_number); } }); } else { $('.bpx-player-video-info')[0].addEventListener('DOMSubtreeModified', function(e) { if($(e.target).hasClass('bpx-player-video-info-dm') && $('.bpx-player-video-info').hasClass('bpx-player-hide-dm')) { $('.bpx-player-video-info').removeClass('bpx-player-hide-dm'); } }); } } //快捷键设置 function set_keybindings($to_div){ let html = `
` $(html).appendTo($to_div); let $main = $('.keybindings-editor'); let $scroll_element = $('.monaco-scrollable-element'); let row_count = 0; let row_line_height = 24; let oldkeystr = ''; let oldwheelstr = ''; //设置keybinding function set_keybinding_key($keybinding, keystrs, combine){ let keybinding_key = ``; let separator = ``; $keybinding.attr('key-str', keystrs).empty(); let delimiter = $keybinding.attr('key-delimiter'); let keycount = $keybinding.attr('key-count'); if(+keycount <= 0){ //原样输出 if(+keycount === 0) $(keybinding_key).clone().appendTo($keybinding).text(keystrs+'(暂不能更改)'); else $(keybinding_key).clone().appendTo($keybinding).text(keystrs.replace(/\s+/,' / ')+'(暂不能更改)'); return; } let keystr_arr = keystrs.split(/\s+/); //合并同类项 if(1&&combine === true) keystr_arr = kb.combine_same_keys(keystr_arr); //显示 keystr_arr.forEach((split, index, array) => { let keys = kb.strToKeys(split); for (var i = 0; i < keys.length; i++) { $(keybinding_key).clone().appendTo($keybinding).text(keys[i]); if(['ctrl', 'shift', 'alt'].indexOf(keys[i].toLowerCase()) !== -1) //是控制键 $(separator).clone().appendTo($keybinding).text('+'); } if(index < array.length-1) $(separator).clone().appendTo($keybinding).text(delimiter); }); } //滚动刷新 function scroll_updata($dom, newY){ let screen_height = $('.monaco-list').height(); let screen_line_count = parseInt(screen_height/row_line_height); let scale = screen_line_count / (row_count + parseInt(screen_line_count/2)); let limittop = (-row_count + parseInt(screen_line_count/1.75))*row_line_height; // list高度+半屏空屏缓冲 $scroll_element.find('.vertical').height(screen_height); $scroll_element.find('.slider').height(screen_height*scale); let newtop = +newY.top, newslider = +newY.slider; if(isNaN(newtop) === true) newtop = newslider / scale; newtop = newtop >= 0 ? 0 : newtop <= limittop ? limittop : newtop; $dom.find('.monaco-list-rows').css({top: `${newtop}px`}); //list $dom.find('.slider').css({top: `${-newtop * scale}px`}); //进度条 } //=========初始化==========// //#cd5c5c55 function set_status_svg(stroke){ return ` ` } //初始化list_row function init_list_row(){ function ret_list_row(index, text, key, keystr, keycount, delimiter, nonuse, top){ const parity = ['even', 'odd']; return `
${set_status_svg('#cd5c5c55')}
${text}
`; } let $div = $('
'); for (let [key, {text, tempargs, keystr, keycount, display, delimiter, nonuse}] of Object.entries(config.sets['keyBinding'].options)) { if(delimiter === undefined) delimiter = '/'; //默认是'/' // color: grey; // text-shadow: 1px 1px #fff; // background-color: silver; let list_row; if(display === 'none') list_row = ret_list_row(-1, text, key, keystr, keycount, delimiter, nonuse, -1*row_line_height); else list_row = ret_list_row(row_count++, text, key, keystr, keycount, delimiter, nonuse, (row_count-1)*row_line_height); let $list_row = $(list_row); /////////////////////////////////////////// //创建一个 args num input function create_num_input(key, arg, arg_index){ //创建inputDOM function create_num_input_$dom(key, arg){ let $container = $(`
`); $container.append(``) if(bb_type.indexOf(BILI_3_X_VIDEO) !== -1 && !!arg.opt) tip_create_3_X({ target: $container, tip_target: $('#hhh_tip'), color: '#00ffff', title: arg.opt.tip[0], gap: 1 }); return $container; } //设置inputDOM event function num_input_event_init($num_input_dom, arg_index){ $input = $num_input_dom.find('input') //---change--- $input.off('change.hhh_num_input'); $input.on ('change.hhh_num_input', function(e){ let $this = $(this) let key = $this.attr('key') let args = config.getCheckboxSettingTempArgs(key) let new_args = {} args[arg_index]['value'] = $this.val() config.setCheckboxSettingTempArgs(key, args); config.storageCheckboxSetting(); }); //---focus--- $input.off('focus.hhh_num_input'); $input.on ('focus.hhh_num_input', function(e){ //---keydown stopPropagation on--- $('body').off('keydown.hhh_num_input'); $('body').on ('keydown.hhh_num_input', function(e){ e.stopPropagation() }); $(this).css({width: "36px"}).prop('type', 'number'); }) //---dblclick--- //* 双击 number input 会透过input节点产生blur事件 */ bug? $input.off('dblclick.hhh_num_input'); $input.on ('dblclick.hhh_num_input', function(e){ e.stopImmediatePropagation() e.preventDefault() }) //---blur--- $input.off('blur.hhh_num_input'); $input.on ('blur.hhh_num_input', function(e){ //---keydown stopPropagation off--- $('body').off('keydown.hhh_num_input'); $(this).prop('type', 'tel'); this.style.width = "10px"; //让 scrollWidth 获取最小值,达到回缩的效果 this.style.width = (this.scrollWidth)+"px"; }) //---mouseenter、mouseleave--- $input.off('mouseenter.hhh_num_input'); $input.off('mouseleave.hhh_num_input'); $input.on ('mouseenter.hhh_num_input', function(e){ let $this = $(this); $this.css({width: "36px"}).prop('type', 'number'); let rect = this.getBoundingClientRect(); document.onmousemove = function(e){ let x = e.clientX; let y = e.clientY; if(x < rect.left || x > rect.right || y < rect.top || y > rect.bottom ){ $this.mouseleave(); } } }).on('mouseleave.hhh_num_input', function(e){ document.onmousemove = null; if(this !== document.activeElement){ $(this).prop('type', 'tel'); this.style.width = "10px"; //让 scrollWidth 获取最小值,达到回缩的效果 this.style.width = (this.scrollWidth)+"px"; } }) } let $num_input_dom = create_num_input_$dom(key, arg) num_input_event_init($num_input_dom, arg_index) return $num_input_dom } //插入到相应 list row if(!!tempargs){ let $text = $(`
`) let static_text = text.split(/\${.+?}/) let i; for(i = 0; i < tempargs.length; ++i){ $text[0].appendChild( document.createTextNode(static_text[i]) ) $text[0].appendChild( create_num_input(key, tempargs[i], i)[0] ) } $text[0].appendChild( document.createTextNode(static_text[i]) ) $list_row.find(`:contains(${text}):last`).replaceWith($text) } $list_row.appendTo($div); set_keybinding_key($list_row.find('.monaco-keybinding'), keystr, true); if(display === 'none') $list_row.hide(); } $div.find('.monaco-list-row').appendTo($('.monaco-list-rows')); //初始化input宽度 $('.monaco-list-rows').find('.hhh-bpx-player-dm-setting-number').mouseenter().mouseleave() //滚动条半屏空屏 if($('.monaco-list').height() < row_count*row_line_height){ //初始化滚动条 scroll_updata($scroll_element, {top: 0}) //滚轮滚动条 $scroll_element.on('mousewheel',function(e){ var wheelDelta = e.originalEvent.wheelDelta; let delta = wheelDelta >= 120 ? row_line_height*2 : wheelDelta <= -120 ? -row_line_height*2 : 0; let newtop = parseInt($(this).find('.monaco-list-rows').css('top').split('px')[0]) + delta; scroll_updata($(this), {top: newtop}); return false; }); let track = 'mouseover'; //拖动滚动条 $scroll_element.find('.slider').mousedown(function(e){ let $this = $(this); let doc = document; $this.addClass('active'); let fixed_offsetY = e.clientY - parseInt($this.css('top').split('px')[0]); //点击时鼠标坐标Y相对滚动条top的偏移 doc.onmousemove = function(e){ scroll_updata($scroll_element, {slider: parseInt(fixed_offsetY - e.clientY)}); }; doc.onmouseup = function(e){ //清除事件 doc.onmousemove = null; doc.onmouseup = null; $this.removeClass('active'); track === 'mouseleave' && $scroll_element.mouseleave(); return false; } return false; }); //点击滚动条轨道,滑块中点移到点击位置 $scroll_element.find('.vertical').mousedown(function(e){ let newslider = $(this).find('.slider').height()/2 - e.offsetY; //计算滑块中点相对于鼠标点击位置偏移 scroll_updata($scroll_element, {slider: newslider}); return false; }); //over & leave $scroll_element.mouseover(function(){ track = 'mouseover'; if($(this).find('.slider').hasClass('active') !== true){ $(this).find('.vertical').removeClass('invisible') .removeClass('fade') .addClass('visible'); } }).mouseleave(function(){ track = 'mouseleave'; if($(this).find('.slider').hasClass('active') !== true){ $(this).find('.vertical').removeClass('visible') .addClass('invisible') .addClass('fade'); } }); } //焦点集中在第一项 $('.monaco-list-row').removeClass('focused'); $('.monaco-list-row:first').addClass('focused'); } //=========事件==========// //list-row Event function set_listrow_event(open){ //input Event function set_input_event(open){ //=========== //input Event function set_widget(curr_keystr){ function set_input(keystr){ let delimiter = $('.defineKeybindingWidget .monaco-keybinding').attr('key-delimiter'); delimiter = delimiter === '/' ? ' / ' : delimiter; $('.defineKeybindingWidget input').val(keystr.replace(/\s+/g, `${delimiter}`)); } function set_existing(keystr){ let str = keystr.split(' ').slice(-1)[0]; let keyarr = config.findKeyBinding(str); let len = keyarr.length; let html = ''; if(len > 0){ //有重复 let len_html = ''; let sys_html = ''; let id_text = ''; //如果是系统快捷键 let i = 0; for (; i < keyarr.length-1; i++) { const o = keyarr[i]; if(o.is_system_key) id_text += `"${o.id}(系统命令)" / `; else id_text += `"${o.id}" / `; } if(keyarr[i].is_system_key) id_text += `"${keyarr[i].text}(系统命令)"`; else id_text += `"${keyarr[i].text}"`; //如果多于一个 if(len > 1) len_html = `等 ${len} 个命令`; html = `
已有命令 ${id_text} ${len_html}绑定此按键
` } let $existing = $main.find($('.existingText')); $existing.text('').append(html); } //设置 input keybinging existing let keycount = Math.abs(+$('.defineKeybindingWidget .monaco-keybinding').attr('key-count')); let keystr_temp = $('.defineKeybindingWidget .monaco-keybinding').attr('key-str-temp'); let keystr = curr_keystr; if(keystr_temp !== '' && keystr_temp.split(' ').length < keycount){ keystr = keystr_temp + ' ' + keystr; } set_input(keystr); set_keybinding_key($('.defineKeybindingWidget .monaco-keybinding'), keystr, true); set_existing(keystr); return keystr; } if (open !== ON){ $main.find('input').off('keydown.hhh_widget_input'); $main.find('input').off('keyup.hhh_widget_input'); $main.find('input').add('.defineKeybindingWidget').off('mousewheel.hhh_widget_input'); $main.find('input').off('blur.hhh_widget_input'); }else{ //---------------- //keydown && keyup $main.find('input').off('keyup.hhh_widget_input'); $main.find('input').on ('keyup.hhh_widget_input',function(e){ oldkeystr = ''; oldwheelstr = ''; }); $main.find('input').off('keydown.hhh_widget_input'); $main.find('input').on ('keydown.hhh_widget_input',function(e){ let curr_keystr = kb.delSpace(kb.stringifyKeyCode(e)); if(curr_keystr === oldkeystr) return false; if(curr_keystr === 'Enter'){ //确定keybinding $('.overlay-container').hide(); //更新list-row let index = $('.defineKeybindingWidget').attr('data-index'); let $keybinding = $('.monaco-list-rows .monaco-list-row').eq(index).find('.monaco-keybinding'); let keystr = $('.defineKeybindingWidget .monaco-keybinding').attr('key-str'); let keyid = $('.defineKeybindingWidget .monaco-keybinding').attr('key-id'); set_keybinding_key($keybinding, keystr, true); //log('确认: '+keyid+' - '+keystr); config.setKeySettingKeystr(keyid, keystr); config.storageCheckboxSetting(); }else if(curr_keystr === 'Escape'){ //取消keybinding $('.overlay-container').hide(); }else{ //储存keybinding let keystr = set_widget(curr_keystr); let curr_last_keystr = kb.getkeyCode(e.keyCode); if(['ctrl', 'shift', 'alt'].indexOf(curr_last_keystr.toLowerCase()) === -1){ // 非控制键暂存 $('.defineKeybindingWidget .monaco-keybinding').attr('key-str-temp', keystr); } } oldkeystr = curr_keystr; return false; }); //---------- //mousewheel $main.find('input').add('.defineKeybindingWidget').off('mousewheel.hhh_widget_input'); $main.find('input').add('.defineKeybindingWidget').on ('mousewheel.hhh_widget_input',function(e){ e.keyCode = e.originalEvent.wheelDelta >= 120 ? '300' : '301'; let curr_keystr = kb.delSpace(kb.stringifyKeyCode(e)); // // let id, index, key; // //e.ctrlKey = true; // let keybindings = config.findKeyBinding(kb.delSpace(kb.stringifyKeyCode(e))); // if(keybindings.length){ // id = keybindings[0]['id']; // index = keybindings[0]['index']; // key = keybindings[0]['key']; // }; // keybindings.forEach(element => { // log(element) // }); // if(curr_keystr === oldwheelstr) return false; let keystr = set_widget(curr_keystr); $('.defineKeybindingWidget .monaco-keybinding').attr('key-str-temp', keystr); oldwheelstr = curr_keystr; return false; }); //----- //blur $main.find('input').off('blur.hhh_widget_input'); $main.find('input').on ('blur.hhh_widget_input',function(){ $('.overlay-container').hide(); $main.find('input').off('keydown.hhh_widget_input'); $main.find('input').off('keyup.hhh_widget_input'); $main.find('input').add('.defineKeybindingWidget').off('mousewheel.hhh_widget_input'); }) } } function key_input_run(){ //log(+$('.monaco-list-row.focused .monaco-keybinding').attr('key-count')) if(+$('.monaco-list-row.focused .monaco-keybinding').attr('key-count') <= 0) //不可设置 return false; let bindWidth = $('.keybindings-editor').width(); let bindHeight = $('.keybindings-editor').height(); let inputWidth = $('.defineKeybindingWidget').outerWidth(); let inputHeight = $('.defineKeybindingWidget').outerHeight(); let index = $('.monaco-list-row.focused').attr('data-index'); $('.defineKeybindingWidget').css({left: `${(bindWidth-inputWidth)/2}px`, top: `${(bindHeight-inputHeight)/2}px`}) .attr({'data-index': index}); let key_id = $('.monaco-list-row.focused').find('.monaco-keybinding').attr('key-id'); let key_str = $('.monaco-list-row.focused').find('.monaco-keybinding').attr('key-str'); let key_count = $('.monaco-list-row.focused').find('.monaco-keybinding').attr('key-count'); let delimiter = $('.monaco-list-row.focused').find('.monaco-keybinding').attr('key-delimiter'); $('.defineKeybindingWidget .monaco-keybinding').attr({'key-id': key_id, 'key-str': key_str, 'key-count': key_count, 'key-delimiter': delimiter, 'key-str-temp': ''}); $('.defineKeybindingWidget .existingText').text(''); let key_count_str = +key_count===1 ? '' : `(${key_count}组)`; $('.defineKeybindingWidget .message').text(`先按所需的组合键${key_count_str},再按 Enter 键。`); $('.defineKeybindingWidget .monaco-keybinding').empty(); $('.overlay-container').show(); $('.defineKeybindingWidget input').val('').focus(); oldkeystr = ''; oldwheelstr = ''; set_input_event(ON); //set_widget(key_str); //可以在显示时初始化 } if (open !== ON){ $('body').off('keydown.hhh_widget_listrow'); $main.find('.monaco-list-rows').off('click.hhh_widget_listrow'); $main.find('.monaco-list-rows').off('dblclick.hhh_widget_listrow'); }else{ //-------- //keydown $('body').off('keydown.hhh_widget_listrow'); $('body').on ('keydown.hhh_widget_listrow',function(e){ if(config.group !== 'keyBinding') return; //选择执行 key event let kcode = kb.getkeyCode(e.keyCode); if(kcode === 'Enter'){ key_input_run(); e.stopImmediatePropagation() e.preventDefault() }else if(kcode === '↑' || kcode === '↓' ){ let $next_prev = $('.monaco-list-row.focused').next(); if(kcode === '↑'){ $next_prev = $('.monaco-list-row.focused').prev(); } if($next_prev.length !== 0 && $next_prev.css('display') !== 'none'){ $('.monaco-list-row.focused').removeClass('focused'); $next_prev.addClass('focused'); }else{ $next_prev = $('.monaco-list-row.focused'); } let rows_top = +Math.abs($scroll_element.find('.monaco-list-rows').css('top').split('px')[0]); let screen_height = $('.monaco-list').height(); let focus_row_top = +$next_prev.css('top').split('px')[0]; let focus_row_bottom = focus_row_top + $next_prev.height(); let focus_row_virtual_top = focus_row_bottom - screen_height; if(rows_top > focus_row_top) { scroll_updata($scroll_element, {top: -focus_row_top}); }else if(rows_top < focus_row_virtual_top) { scroll_updata($scroll_element, {top: -focus_row_virtual_top}); } e.stopImmediatePropagation() e.preventDefault() //log('curr_row_top: '+focus_row_bottom+' - rows_top: '+rows_top+' - focus_row_virtual_top: '+(focus_row_bottom-screen_height)) } }); //------------------ //click && dblclick $main.find('.monaco-list-rows').off('click.hhh_widget_listrow'); $main.find('.monaco-list-rows').off('dblclick.hhh_widget_listrow'); $main.find('.monaco-list-rows').on ('click.hhh_widget_listrow',function(e){ let $target = $(e.target); $('.monaco-list-row').removeClass('focused'); $target.parents(".monaco-list-row").addClass('focused'); }).on('dblclick.hhh_widget_listrow',function(){ key_input_run(); }); //------------------ //mouseenter & mouseleave $main.find('.monaco-list-row').off('click.hhh_widget_listrow'); $main.find('.monaco-list-row').off('mouseenter.hhh_widget_listrow'); $main.find('.monaco-list-row').off('mouseleave.hhh_widget_listrow'); $main.find('.monaco-list-row').on ('mouseenter.hhh_widget_listrow',function(e){ let key_id = $(this).find('.monaco-keybinding').attr('key-id') // 第三个 let keycount = config.getKeySettingKeycount(key_id) if(keycount > 0){ $(this).find('.monaco-table-td:first>.usable-sign').hide() $(this).find('.monaco-table-td:first>.usable-hover').show() } }).on('mouseleave.hhh_widget_listrow',function(e){ let key_id = $(this).find('.monaco-keybinding').attr('key-id') // 第三个 let keycount = config.getKeySettingKeycount(key_id) if(keycount > 0){ $(this).find('.monaco-table-td:first>.usable-hover').hide() if(config.getKeySettingNonuse(key_id) === ON){ $(this).find('.monaco-table-td:first>.usable-sign').show() }else{ $(this).find('.monaco-table-td:first>.usable-sign').hide() } } }).on('click.hhh_widget_listrow',function(e){ let $target = $(e.target) if((e.target.className === 'monaco-table-td' && $target.index() === 0) || $target.parents(".monaco-table-td").index() === 0){ // 第一个 let key_id = $(this).find('.monaco-keybinding').attr('key-id') // 第三个 let keycount = config.getKeySettingKeycount(key_id) if(keycount > 0){ $(this).find('.monaco-table-td:first>.usable-hover').hide() let nonuse = !config.getKeySettingNonuse(key_id) config.setKeySettingNonuse(key_id, nonuse) config.storageCheckboxSetting() if(nonuse === ON){ $(this).find('.monaco-table-td:first>.usable-sign').show() }else{ $(this).find('.monaco-table-td:first>.usable-sign').hide() } $(this).toggleClass('mask') return false } } }); } } init_list_row(); set_listrow_event(ON); // $('.bpx-player-dm-setting').mouseenter() // $('#hhh_custom').click() // $('#keyBinding').click() // $('#hhh_item>div:first').css({'height':'max-content','width':'max-content'}) // { // "key": "ctrl+h", // "command": "workbench.action.tasks.runTask", // "args": { // "task": "VS Code - Build", // "type": "npm" // } // } } //自定义快捷键设置 //ON - 启用热键,OFF - 关闭热键 function set_hotkey(open){ if (open !== ON){ $(document).off('keyup.hhh_lightoff') $(document).off('keydown.hhh_lightoff') $(document).off('mousewheel.hhh_lightoff') $(document).off('DOMMouseScroll.hhh_lightoff') }else{ var parent = document; var like_shake = false; var prev_time = 0; var old_index = app_page_parameters.player_setting_area; is_show_hint = config.getCheckboxSettingStatus('hotKeyHint'); function hot_run(e){ let id, text, index, key, status; let keybindings = config.findKeyBinding(kb.delSpace(kb.stringifyKeyCode(e))); if(keybindings.length){ id = keybindings[0]['id']; text = e.type === 'keyup' ? keybindings[0]['text'] + e.type : keybindings[0]['text']; index = keybindings[0]['index']; key = keybindings[0]['key']; nonuse = keybindings[0]['nonuse'] //控制执行当前热键 //log('①',id,nonuse) if(nonuse === ON) return //config.setKeySettingNonuse(id, OFF) }; keybindings.forEach(element => { //log(element) //log(text) }); //特例 if(keybindings.length && (e.type === 'mousewheel'||e.type === 'DOMMouseScroll') && text !== '非全屏音量调节${step}%'){ e.preventDefault() e.stopPropagation() } // log(id) // log(text) // log(text+id) // log('非全屏音量调节\${step}%'+id) // log((text+id) === '非全屏音量调节\${step}%'+id) if(text === '快进/快退时显示醒目进度条'){ //快进时显示醒目进度条 && 段落循环 //log( $('#hhh_loop_wrap')[0].hhh_loop_time_id ); // if(0&&$('#hhh_loop_wrap')[0].hhh_loop_time_id === true){ // return false; // }else{ if(config.getCheckboxSettingStatus('reloadDanmuku') === ON) { //恢复重载弹幕效果 e.stopPropagation(); h5Player.currentTime = index === 0 ? h5Player.currentTime - 5 : h5Player.currentTime + 5; h5Player.play(); setTimeout(function(){ $(bb['danmakuSwitch']).last().find('input').click(); $(bb['danmakuSwitch']).last().find('input').click(); },0); } dynamicProgress(12, 2); //} } else if(text === '合理空格键') { //空格键2.X版本不合理,改成和3.X版本一样,3.X版也o了 if( config.getCheckboxSettingStatus('dynamicSpace') === ON && !is_fullscreen()){ if(is_in_biliplayer === false) e.stopPropagation() } } else if(text === '优先取消右键菜单等') { //优先取消右键菜单等 Esc if($(bb['hotkeyPanel']).length === 1 && ($(bb['hotkeyPanel']).hasClass('active') === true || $(bb['hotkeyPanel']).css('display') !== 'none')){ //$(bb['hotkeyPanel']).removeClass('active').css('display', 'none'); $(bb['hotkeyPanel']).find(bb['hotkeyPanelClose']).click(); //慢 return false; } if($(bb['playerContextMenu']).hasClass('active') === true || $(bb['playerContextMenu']).hasClass('bpx-player-active') === true ){ let evt = new MouseEvent('contextmenu', { clientX:-9999, clientY:-9999 }); $(bb['videoWrap'])[0].dispatchEvent(evt); return false; } if($('#hhh_img').length >= 1 && $('#hhh_img').css('display') !== 'none'){ $('#hhh_img').css('display', 'none'); return false; } } else if(text === '有投币框时回车投币') { //有投币框时回车投币 Enter //log($(bb['coinDlgOkBtn']).length) // if($(bb['coinDlgOkBtn']).length === 1){ // $(bb['coinDlgOkBtn']).click(); // } } else if(text === '关灯/开灯') { //开关灯 $('#hhh_lightoff input').click(); } else if(text+id === '非全屏音量调节${step}%'+'onePerVolumeControl') { //非全屏音量调节 0~1 (b站默认"滚轮"(如果使用滚轮的话)操作某些情况会失效,一并处理全屏情况) //两个参数指定屏幕范围(按百分比),第三个参数表示滚动一下增加的音量百分比,参数四表示暂停时是否调节 let tempargs = config.getCheckboxSettingTempArgs(key); let delta = 1 //默认1% for(v of config.getCheckboxSettingTempArgs(id)){ if(v?.name === 'step') { delta = parseInt(v.value); break } } let args = config.getCheckboxSettingArgs('volumeControlWhenNonFullScreen'), isWheelVolume = config.getCheckboxSettingStatus('volumeControlWhenNonFullScreen'), screen_left = args.screen_left, screen_rght = args.screen_rght, isPauseVolume = config.getCheckboxSettingStatus('volumeControlWhenPause'); if(isWheelVolume === OFF) return; //缺省屏幕百分比参数,默认0.3~0.7 screen_left = (screen_left<0 || screen_left>1)? 0.3: screen_left; screen_rght = (screen_rght<0 || screen_rght>1)? 0.7: screen_rght; //缺省音量百分比,默认1 delta = (delta<1 || delta>100)? 1: delta; //非暂停(可选) && 鼠标在屏幕指定位置时处理 let isPauseStillRun = isPauseVolume || h5Player.paused === false; let Rect = $(bb['videoWrap'])[0].getBoundingClientRect(); let offsetX = e.originalEvent.x - Rect.x; let inLimit = offsetX > Rect.width*screen_left && offsetX < Rect.width*screen_rght; if(is_fullscreen() || (isPauseStillRun && (inLimit || (e.type !== 'mousewheel' && e.type !== 'DOMMouseScroll') ))) { //阻止页面滚动 && 阻止冒泡 if (e.type === 'mousewheel' || e.type === 'DOMMouseScroll') { e.preventDefault(); e.stopPropagation(); } let v = volume(); if(index === 0 || index === 2) { //向上滚动,减少音量 volume(+(v+(delta/100)).toFixed(3)); //+ string to number } else if(index === 1 || index === 3) { //向下滚动,增大音量 volume(+(v-(delta/100)).toFixed(3)); } } } else if(text+id === '非全屏音量调节${step}%'+'fivePerVolumeControl') { //音量控制,替换系统默认 let tempargs = config.getCheckboxSettingTempArgs(key); let delta = 0.05 //默认5% for(v of config.getCheckboxSettingTempArgs(id)){ if(v?.name === 'step') { delta = parseInt(v.value)/100; break } } let diff = index === 0 ? delta : -delta; function limitv(v,limit){ Math.ceil((0.15)*20)*5 log(v,parseInt(v/limit)*limit) return parseInt(v/limit)*limit } window.setTimeout((function() { //长按时保持DOM更新 //volume(volume()+diff); let vol = (volume() + diff).toFixed(2) volume(diff > 0 ? Math.floor(vol*20)/20 : Math.ceil(vol*20)/20) // 5%的倍数 }),10); return false; } else if(text === '全屏') { //全屏 fullscreen(); } else if(text === '网页全屏') { //网页全屏 B站改成点赞了 web_fullscreen(); e.stopPropagation(); } else if(text === '宽屏模式') { //宽屏模式 B站改成投币了 is_fullscreen() ? web_fullscreen() : wide_screen(); e.stopPropagation(); } else if(text === '开关顶部/底部弹幕') { //开关顶部弹幕 if(index === 0){ $(bb['danmukuTopClose']).length === 0 ? showHint(parent, '#hhh_wordsHint', '关闭顶部弹幕') : showHint(parent, '#hhh_wordsHint', '打开顶部弹幕'); dm_remember(function(){$(bb['danmukuTop']).click()}); }else{ $(bb['danmukuBottomClose']).length === 0 ? showHint(parent, '#hhh_wordsHint', '关闭底部弹幕') : showHint(parent, '#hhh_wordsHint', '打开底部弹幕'); dm_remember(function(){$(bb['danmukuBottom']).click()}); } } else if(text === '开关弹幕') { //开关弹幕 bilibili增加了关弹幕的快捷键,也是D $(bb['danmakuSwitch']).last().find('input').click(); is_show_hint && (is_danmaku_show() === true ? showHint(parent, '#hhh_wordsHint', '开弹幕') : showHint(parent, '#hhh_wordsHint', '关弹幕')); e.stopPropagation(); } else if(text === '洗脑循环') { //开关洗脑循环 B站改成一键三连了 $(bb['playSettingRepeatInput'])[0].checked ? showHint(parent, '#hhh_wordsHint', '关洗脑循环') : showHint(parent, '#hhh_wordsHint', '开洗脑循环'); $(bb['playSettingRepeatInput']).click(); e.stopPropagation(); } else if(text === '弹幕随屏幕缩放') { //弹幕随屏幕缩放 $(bb['settingFs'])[0].checked === false ? showHint(parent, '#hhh_wordsHint', '弹幕随屏幕缩放') : showHint(parent, '#hhh_wordsHint', '弹幕不随屏幕缩放'); dm_remember(function(){$(bb['settingFs']).click()}); } else if(text === '增减弹幕透明度${step}%' ) { //滚轮调节弹幕透明度(ctrl+滚轮),参数表示滚动一下增加的透明度百分比 if(config.getCheckboxSettingStatus('danmuOpacityControl') === OFF) return; //阻止页面滚动 && 阻止冒泡 //e.preventDefault(); //e.stopPropagation(); //缺省透明度百分比,默认5 let delta = 5 //默认5% for(v of config.getCheckboxSettingTempArgs(id)){ if(v?.name === 'step') { delta = parseInt(v.value); break } } //let delta = parseInt(config.getCheckboxSettingArgs('danmuOpacityControl').delta); delta = (delta<0 || delta>100)? 5: delta; var opacity; if(index === 0) { //向上滚动,增大透明度 opacity = adjust_progress(bb['settingOpacity'], delta, 10, 100); } else if(index === 1) { //向下滚动,减少透明度 opacity = adjust_progress(bb['settingOpacity'], -delta, 10, 100); } if(opacity !== undefined) showHint(document, '#hhh_opacityHint', '透 '+opacity); } else if(text === '减增弹幕透明度${step}%') { //+ -弹幕透明度 let inc_percent = 10 //默认10% for(v of config.getCheckboxSettingTempArgs(id)){ if(v?.name === 'step') { inc_percent = parseInt(v.value); break } } inc_percent = index === 0 ? -inc_percent : inc_percent; window.setTimeout((function() { //长按时保持DOM更新 dm_remember(function(){ let opacity = adjust_progress(bb['settingOpacity'], inc_percent, 10, 100); showHint(parent, '#hhh_opacityHint', '透 ' + opacity); }); }),10); } else if(text === '减增弹幕字号${step}%') { //+ -弹幕字号 let inc_percent = 10 //默认10% for(v of config.getCheckboxSettingTempArgs(id)){ if(v?.name === 'step') { inc_percent = parseInt(v.value); break } } inc_percent = index === 0 ? -inc_percent : inc_percent; //let inc_percent = index === 0 ? -10 : 10; window.setTimeout((function() { //长按时保持DOM更新 dm_remember(function(){ let opacity = adjust_progress(bb['settingFontsize'], inc_percent, 50, 170); showHint(parent, '#hhh_opacityHint', '字 ' + opacity); }); }),10); } else if(text === '弹幕显示区域 1/4屏~不限') { //弹幕显示区域 dm_remember(function(){ var area_text = {0:'1/4屏',25:'半屏',50:'3/4屏',75:'不重叠',100:'不限'}; var percent = index * 25; //((0~4)*25)% //弹幕显示区域大到小切换时清空弹幕 if(old_index > index){ setTimeout(function(){ $(bb['danmakuSwitch']).last().find('input').click(); $(bb['danmakuSwitch']).last().find('input').click(); },0); } old_index = index; set_progress(bb['settingArea'], percent, 0, 100); showHint(parent, '#hhh_wordsHint', area_text[percent]); }); } else if(text === '快进/快退${second}s') { //+ -30s ctrl+ left/right h5Player = geth5Player() $(bb['danmakuSwitch']).last().find('input').click().click() let second = 30 //默认30s for(let v of config.getCheckboxSettingTempArgs(id)){ if(v?.name === 'second') { second = parseInt(v.value); break } } h5Player.currentTime = index === 0 ? +h5Player.currentTime - second : +h5Player.currentTime + second; h5Player.play(); dynamicProgress(12, 2); } else if(text === '逐帧操作') { //逐帧 shift+ left/right h5Player.pause(); for(let move_frames = 1.01; move_frames>0.1;){ let move_frames_time = move_frames/fps; h5Player.currentTime = index === 0 ? +(h5Player.currentTime - move_frames_time).toFixed(3) : +(h5Player.currentTime + move_frames_time).toFixed(3); if(prev_time === h5Player.currentTime){ //如果未走帧 move_frames /= 2; //log('2.1: h5currTime: '+h5Player.currentTime+' - prev_time: '+prev_time+' - '+move_frames_time+' - '+move_frames); continue; } else{ //log('2.2: h5currTime: '+h5Player.currentTime+' - prev_time: '+prev_time+' - '+move_frames_time+' - '+move_frames); let total_frame = +(h5Player.duration*fps).toFixed(); let current_frame = +(h5Player.currentTime*fps).toFixed(); showHint(parent, '#hhh_wordsHint', `${current_frame}/${total_frame}`); prev_time = h5Player.currentTime; break; } } e.stopPropagation(); } else if(text === '调节视频播放速度') { //ctrl + ↑↓ if(config.getCheckboxSettingStatus('keyVideoSpeed') === OFF) return; window.setTimeout((function() { //长按时保持DOM更新 index === 0 ? video_select_speed(1) : video_select_speed(-1); }),10); } else if(text === '0.1倍速调节视频播放速度') { //ctrl + '0'&'.' window.setTimeout((function() { //长按时保持DOM更新 index === 0 ? video_speed_0dot1(0.1) : video_speed_0dot1(-0.1); }),10); } else if(text === '调节自定义倍速速度') { //shift + ↑↓ if(config.getCheckboxSettingStatus('customPlayRate') === OFF) return const args = config.getCheckboxSettingArgs('customPlayRate') const newrate = Math.max(0, (index === 0 ? +args.rate + 0.5 : +args.rate - 0.5)) const new_args = {rate: newrate} //log(newrate, args.rate, config.getCheckboxSettingArgs('customPlayRate')['rate']) config.setCheckboxSettingArgs('customPlayRate', {...args, ...new_args}) showHint(parent, '#hhh_wordsHint', `→${newrate.toFixed(1)}x`) customPlayRate(config.getCheckboxSettingStatus('customPlayRate'), +config.getCheckboxSettingArgs('customPlayRate')['rate']); } else if(text === '开关字幕') { //E subtitle() e.stopPropagation() } else if(text === '开关稍后再看') { //H is_watchlater_added() !== undefined && (is_watchlater_added() === false ? showHint(parent, '#hhh_wordsHint', '已加稍后再看') : showHint(parent, '#hhh_wordsHint', '已从稍后再看列表中移除')) //有延迟 watchLater() e.stopPropagation() } else if(text === '点赞、投币、收藏、长按一键三连') { //长按一键三连(keydown) - 对应系统快捷键 Q、R、W、E if(index === 0 || index === 3){ //点赞、三连 $(bb['like'])[0].dispatchEvent( new MouseEvent('mousedown') ); if(like_shake === false) like_shake = !!$('.van-icon-videodetails_like.shake').length; }else if(index === 1){ //投币 $(bb['coin']).click(); set_dialog('coin', 'hhh_test_coin') }else if(index === 2){ //收藏 $(bb['collect']).click(); } } else if(text === '点赞、投币、收藏、长按一键三连'+'keyup') { //点赞;长按一键三连(keyup) if(index === 0 || index === 1){ $(bb['like'])[0].dispatchEvent(new MouseEvent('mouseup')); if(index === 0 && like_shake === false) { let like_text = $(bb['like']).text().match(/.+/)[0]; let like_num = like_text.match(/\d+$/)? (+like_text.match(/\d+$/)[0])+1: like_text; if($(bb['likeon']).length === 1) showHint(parent, '#hhh_wordsHint', '取消点赞'); else showHint(parent, '#hhh_wordsHint', `点赞成功 ${like_num}`); $(bb['like']).click(); } like_shake = false; } //TEST } else if(e.type === 'keydown' && (e.keyCode === 'X'.charCodeAt() || e.keyCode === keycode['/'])) { //TEST 显示、隐藏video-control、高能进度条 if(bb_type.indexOf(BILI_2_X) !== -1){ if($(bb['playArea']).hasClass('video-control-show') === true) $(bb['playVideoControlWrap']).mouseout(); else $(bb['playVideoControlWrap']).mousemove(); }else{ if($('.bpx-player-container').attr('data-ctrl-hidden') === "true"){ $('.bpx-player-container').attr('data-ctrl-hidden', "false"); $('.bpx-player-control-entity').attr('data-shadow-show','false'); $(bb['playPBP']).addClass('show') //高能进度条 }else{ $('.bpx-player-container').attr('data-ctrl-hidden', "true"); $('.bpx-player-control-entity').attr('data-shadow-show','true'); $(bb['playPBP']).removeClass('show') } } } else if(e.type === 'keydown' && (e.keyCode === 'H'.charCodeAt())) { //TEST //if(bb_type===BILI_3_X_MOVIE) $('.bpx-player-dm-setting-wrap').css('display','block'); $('.bilibili-player-video-danmaku-setting-wrap').mouseover(); //$('.bilibili-player-area.video-state-blackside').attr('class', 'bilibili-player-area video-state-blackside video-state-pause video-control-show'); } else{ //console.dir(e.keyCode); } } //key function global_binding_key(){ $(document).off('keydown.hhh_lightoff keyup.hhh_lightoff'); $(document).on('keydown.hhh_lightoff keyup.hhh_lightoff',function(e){ //跳过输入框 if(!!e.target.type && e.target.type.toLowerCase().search(/text|textarea/) !== -1) return true if(e.target.className === 'ql-editor') return true //log('global_binding_key2:'+e.target.type) //处理键盘及鼠标 return hot_run(e); }); } //wheel function global_binding_wheel(){ $(bb['videoWrap']).off('mousewheel.hhh_lightoff'); $(bb['videoWrap']).on('mousewheel.hhh_lightoff',function(e){ //处理键盘及鼠标 e.keyCode = e.originalEvent.wheelDelta >= 120 ? '300' : '301'; //滚轮上滚和下滚 return hot_run(e); }) //firefox $(bb['videoWrap']).off('DOMMouseScroll.hhh_lightoff'); $(bb['videoWrap']).on('DOMMouseScroll.hhh_lightoff',function(e){ //处理键盘及鼠标 e.keyCode = e.detail < 0 ? '300' : '301' //滚轮上滚和下滚 return hot_run(e) }) } global_binding_key(); global_binding_wheel(); } } //添加 `); } } //初始化自定义设置 节点 事件 tip //setting - wrap - box | wrap - move - item function init_setting(run_default_setting_flag){ var $DSet = $(bb['dm']); var $wrap = $DSet.find('.bui-panel-wrap'); var $move = $DSet.find('.bui-panel-move'); var $item_0 = $DSet.find('.bui-panel-item:eq(0)'); var $item_1 = $DSet.find('.bui-panel-item:eq(1)'); var $item_2; var old_width = $item_0.width() + $item_1.width() + 5; var item_0_height = $item_0.height() + 36; //var item_2_width = 'auto'; // todo //var item_2_height = 100 + 18 * 22; // if($DSet.length!=1 || $wrap.length!=1 || $move.length!=1) log('init_setting失败'); if($item_0.length!=1 || $item_1.length!=1) log('init_setting失败2'); /*----------------------------------- * 初始化 *----------------------------------*/ //初始化【$item_0】高度 $item_0.css('height', item_0_height);$wrap $item_0[0].hhh_new_height = item_0_height; //初始化【$move】宽度 //$DSet.find('.bui-panel-move').css('width', old_width + item_2_width); //$move.css('width', '10000px'); //留出足够空间,省事 后面值会改变 /*----------------------------------- * 复制所需节点 *----------------------------------*/ //复制【item_0】【高级设置】To【自定义设置】 $(bb['dmLeftMore']).clone().appendTo(bb['dmLeft']).attr('id', 'hhh_custom') .css('top', '20px') .find(bb['dmLeftMoreText']).text('自定义设置'); //复制【item_1】 To 【item_2】 $item_2 = $item_1.clone().appendTo($move).attr('id', 'hhh_item') .css({'width':'auto', 'height':'auto'}) .removeClass('bui-panel-item-active') .find('div *').remove().end(); //复制【分割线】模板 var $separator = $item_1.find(`${bb['dmRightSeparator']}:first`).clone(); //复制【$title $hotkey父节点】模板 var $title_div = $item_0.find(bb['dmLeftBlock']).clone().attr('id', 'hhh_title_div') //.css({'display': 'flex', 'flex-direction': 'column'}) .empty().css({'margin-top': 6, 'margin-bottom': 2, 'height': 32+0}); //复制【title】模板 var $title = $item_0.find(bb['dmLeftBlockTitle']).clone().css({'color': '#12b3e8'}) .css({'display': 'inline-flex', 'line-height': '32px'}) .text('bilibili关灯自定义设置'); //复制【快捷键说明】模板 var $hotkey = $item_1.find(bb['dmRightMore']).clone().attr('id', 'hhh_item_2_hotkey') .css({'display': 'inline-flex', 'float': 'right'}) .find('span:first').remove().end() .find(bb['dmRightMoreText']).text('快捷键说明(点击切换)').end(); // var $div = $('
'); var $middle_div = $div.clone().attr('id','hhh_middle_div').css({'display':'flex', 'justify-content':'space-between'}) //.css({'widht':'400px'}) //.css('background', 'blue') ; //复制【封面】模板 var $cover = $item_1.find(bb['dmRightMore']).clone().attr('id', 'hhh_item_2_cover') //.css({'display': 'inline-flex', 'float': 'right', 'flex-direction': 'row-reverse'}) //.css({'float': 'right'}) .find('span:first').remove().end() .find(bb['dmRightMoreText']).text('显示封面(点击切换)').end(); //复制【input-checkbox父节点】模板 var $input_div = $item_0.find(bb['dmLeftBlock']).clone().attr('id', 'hhh_input_div') .empty().css({'margin-top': 4, 'margin-bottom': 2}) .css('padding-right', '10px') //.css('background', 'red') //.css('width', 'auto') ; //复制【更多其他】模板 var $group_div = $item_0.find(bb['dmLeftBlock']).clone().attr('id', 'hhh_group_div') .empty().css({'margin-top': 4, 'margin-bottom': 2}) //.css('background', 'yellow') .css('width', 123) //.css('height', 100) ; //复制【input-checkbox】模板 var $input = $item_0.find(bb['dmLeftFlagTitle']).clone().find('input').removeAttr('aria-label').prop('checked', false).end() .find('.bui-checkbox-name').text('bilibili关灯').end() .appendTo($div.clone()).parent().css({display: 'flex', 'margin-top': '6px'}) ; //复制【恢复默认设置】模板 var $reset = $item_1.find(bb['dmRightReset']).clone().css({'margin-top': 6, 'margin-bottom': 6, width: 'auto'}) .find('div').text('恢复默认设置(如有配置错误请尝试点击)').end() ; //复制【返回】模板 var $reback = $item_1.find(bb['dmRightMore']).clone().attr('id', 'hhh_item_2_more') .css({'margin-top': 2, 'margin-bottom': 5, 'margin-right': 10}) .find(bb['dmRightMoreText']) .text('返回').end(); /*----------------------------------- * 添加节点 *----------------------------------*/ let $item_2_div = $item_2.children(':first'); /*----------------------------------- * 上部div *----------------------------------*/ //添加title hotkey $title_div.appendTo($item_2_div); $title.clone().appendTo($title_div); $hotkey.clone().appendTo($($title_div)); //添加分割线 $separator.clone().appendTo($item_2_div); /*----------------------------------- * 中部div *----------------------------------*/ $middle_div.appendTo($item_2_div); $input_div.appendTo($middle_div); $group_div.appendTo($middle_div); //添加封面 $cover.appendTo($group_div); //test // $input_div.css('border','1px red solid'); //红色 input // $group_div.css('border','1px gold solid'); //黄色 group // $middle_div.css('border','1px green solid'); //绿色 // $item_2_div.css('border','1px cyan solid'); //青色 //$input_div.css('width','500px'); //$group_div.css('width','300px'); //$middle_div.css('width','atuo'); //// //$item_2_div.css('width','700px'); //log($input_div.width()+' '+$group_div.width()+' '+$middle_div.width()+' '+$item_2_div.width()); // $('.bpx-player-dm-setting').mouseenter() // $('#hhh_custom').click() // $('#keyBinding').click() // $('#hhh_item>div:first').css({'height':'max-content','width':'max-content'}) // $('.bui-panel-wrap').css({'height': $('#hhh_item').css('height'), 'width': $('#hhh_item').css('width')}); // $move.css('transform', `translateX(-${old_width}px)`); // $('#commonCheckbox').click() // $('.bui-panel-wrap').width(350) // $('.bui-panel-wrap').height(700) //设置复选框 let input_max_width = 0; let input_number_max_width = 0; let all_checkbox_width = 'auto'; //更新选择框设置(二级) function checkbox_args_update(key, status){ let $setting_input = $(`#${key} .hhh-bpx-player-dm-setting_container input`); $setting_input.prop('disabled', !status); if(status === ON) $setting_input.css({color: '#cccccc', border: '1px solid #00a1d699'}); else $setting_input.css({color: '#cccccc66', border: '1px solid #ffffff33'}); } //fn run function checkbox_go(key, status, fn, args){ //执行fn等 if(key === 'volumeControlWhenNonFullScreen'){ //let pause_status = config.getCheckboxSettingStatus('volumeControlWhenPause'); //eval(`${fn}(${status}, ${args.screen_left}, ${args.screen_rght}, ${args.delta}, ${pause_status})`); } else if(key === 'volumeControlWhenPause'){ //let non_full_status = config.getCheckboxSettingStatus('volumeControlWhenNonFullScreen'); //let fn = config.getCheckboxSettingFn('volumeControlWhenNonFullScreen'); //let args = config.getCheckboxSettingArgs('volumeControlWhenNonFullScreen'); //eval(`${fn}(${non_full_status}, ${args.screen_left}, ${args.screen_rght}, ${args.delta}, ${status})`); } else if(key === 'danmuOpacityControl'){ //eval(`${fn}(${status}, ${args.delta})`) } else if(key === 'hotKeyHint'){ is_show_hint = config.getCheckboxSettingStatus(key) } else if(key === 'expandList'){ eval(`${fn}(${status}, ${args.columns})`) } else if(key === 'customPlayRate'){ eval(`${fn}(${status}, ${args.rate})`) } else if(key === 'lightOnWhenLike'){ eval(`${fn}(${status}, ${args.screen_top})`) } else if(key === 'memoryProgress'){ eval(`${fn}(${status}, [${config.getCheckboxSettingStatus('sortList')}])`) } else if(key === 'autoLike'){ eval(`${fn}(${status}, [${config.getCheckboxSettingStatus('likeFollowed')}, ${args.second} ])`) } else if(!!fn){ // openHotKey | removeVideoTopMask 等 eval(`${fn}(${status})`) //new Function(`${fn}(${status})`)(); //与eval相比作用域不同 } } //选择框设置(二级) function checkbox_args_event(){ //---change--- $item_2.find('#hhh_input_div').off('change.hhh_checkbox_args'); $item_2.find('#hhh_input_div').on('change.hhh_checkbox_args', function(e){ let target = e.target; if(target.name === 'checkbox_setting'){ //选择框设置(二级) let key = $(target).attr('checkbox_id'); let arg = $(target).attr('arg'); let args = config.getCheckboxSettingArgs(key); let new_args = {}; if(key === 'customPlayRate'){ //自定义倍速 new_args = {rate: target.value} }else if(key === 'expandList'){ //订阅合集列表自动展开 new_args = {columns: target.value} }else if(key === 'danmuOpacityControl'){ //弹幕透明度控制 new_args = {delta: target.value} }else if(key === 'volumeControlWhenNonFullScreen' && arg === 'Left'){ //指定鼠标范围 Left new_args = {screen_left: target.value} }else if(key === 'volumeControlWhenNonFullScreen' && arg === 'Right'){ //指定鼠标范围 Right new_args = {screen_rght: target.value} }else if(key === 'lightOnWhenLike'){ //下滚时自动开灯,上滚关灯 new_args = {screen_top: target.value} }else if(key === 'autoLike'){ //XX秒后自动点赞 new_args = {second: target.value} } config.setCheckboxSettingArgs(key, {...args, ...new_args}); ///////// //log($(target)) //log(target.name+' - '+target.attributes.checkbox_id.value+' = '+$(target).attr('checkbox_id')); ///////// config.storageCheckboxSetting(); let fn = config.getCheckboxSettingFn(key); let status = config.getCheckboxSettingStatus(key); args = config.getCheckboxSettingArgs(key); checkbox_go(key, status, fn, args); } }); //---focus--- $('input[name=checkbox_setting]').off('focus.hhh_checkbox_args'); $('input[name=checkbox_setting]').on ('focus.hhh_checkbox_args', function(e){ $(this).css({width: "40px"}).prop('type', 'number'); }) //---blur--- $('input[name=checkbox_setting]').off('blur.hhh_checkbox_args'); $('input[name=checkbox_setting]').on ('blur.hhh_checkbox_args', function(e){ $(this).css({width: "25px"}).prop('type', 'tel'); }) //---mouseenter、mouseleave--- $('input[name=checkbox_setting]').off('mouseenter.hhh_checkbox_args'); $('input[name=checkbox_setting]').off('mouseleave.hhh_checkbox_args'); $('input[name=checkbox_setting]').on ('mouseenter.hhh_checkbox_args', function(e){ let $this = $(this); let key = $this.attr('checkbox_id'); //if(key === 'customPlayRate' || key === 'expandList' || key === 'volumeControlWhenNonFullScreen'){ //自定义倍速、订阅合集列表自动展开 $this.css({width: "40px"}).prop('type', 'number'); let rect = this.getBoundingClientRect(); document.onmousemove = function(e){ let x = e.clientX; let y = e.clientY; if(x < rect.left || x > rect.right || y < rect.top || y > rect.bottom ){ $this.mouseleave(); } } //} }).on('mouseleave.hhh_checkbox_args', function(e){ let $this = $(this); let key = $this.attr('checkbox_id'); //if(key === 'customPlayRate' || key === 'expandList' || key === 'volumeControlWhenNonFullScreen'){ //自定义倍速、订阅合集列表自动展开 document.onmousemove = null; if(this !== document.activeElement) $this.css({width: "25px"}).prop('type', 'tel'); //} }) //---keydown--- $('body').off('keydown.hhh_checkbox_args'); $('body').on ('keydown.hhh_checkbox_args', function(e){ if(config.group === 'commonCheckbox' || config.group === 'otherCheckbox') //选择执行 key event e.stopPropagation(); }); } //初始化选择框设置(二级) function checkbox_args_init(group){ //define css append_css($middle_div, 'hhh_style_dm_setting_number', ` .hhh-bpx-player-dm-setting_container { display: flex; } .hhh-bpx-player-dm-setting-number { width: 25px; border: 1px solid #00a1d699; border-radius: 5px; /* appearance: none; */ color: #cccccc; background-color: initial; margin-left: 5px; text-align: center; } ` ) //creat for (let [key, {text, status, args}] of Object.entries(config.sets[group].options)) { if(!args) continue let name = 'checkbox_setting'; let $container = $(`
`); if(key === 'customPlayRate'){ //自定义倍速 $container.append(``) }else if(key === 'expandList'){ //订阅合集列表自动展开 $container.append(``) }else if(key === 'danmuOpacityControl'){ //弹幕透明度控制 $container.append(``) }else if(key === 'volumeControlWhenNonFullScreen'){ //鼠标在指定屏幕范围内音量调节 $container.append(` `) }else if(key === 'lightOnWhenLike'){ //下滚时自动开灯,上滚关灯 $container.append(``) }else if(key === 'autoLike'){ //自动点赞 $container.append(``) } if($container.find(`input[name=${name}]`).length > 0){ $(`#${key}`).append($container); checkbox_args_update(key, status); if(bb_type.indexOf(BILI_3_X_VIDEO) !== -1) tip_create_3_X({ target: $container, tip_target: $('#hhh_tip'), color: '#00ffff', title: args.args_opt.tip[0], gap: 1 }); input_number_max_width = Math.max(input_number_max_width, $(`#${key}`).find('.hhh-bpx-player-dm-setting_container').outerWidth()); //记录最大宽度 //log(input_number_max_width); } } //add event checkbox_args_event(); } //设置【checkbox】点击 click function checkbox_event(){ $item_2.find('#hhh_input_div').off('click.hhh_checkbox'); $item_2.find('#hhh_input_div').on('click.hhh_checkbox', function(e){ let target = e.target; if(target.type === 'checkbox'){ //一级选择框设置 let key = bb_type.indexOf(BILI_2_X) !== -1 ? target.parentNode.id : target.parentNode.parentNode.parentNode.id; let status = target.checked; //log('checked: '+e.target.checked) config.setCheckboxSettingStatus(key, status); config.storageCheckboxSetting(); let fn = config.getCheckboxSettingFn(key); let args = config.getCheckboxSettingArgs(key); //设定参数设置 checkbox_args_update(key, status); checkbox_go(key, status, fn, args); } }) } //生成checkbox function checkbox_init(group){ let len = 0 //按行添加 for (let [key, {text, level, status, tip, disable}] of Object.entries(config.sets[group].options)) { //log(text+' - '+tip) ++len let $key = $input.clone().appendTo($input_div).attr('id', key).css('width', 'auto') .find('input').prop('checked', status).end() .find('.bui-checkbox-name').text(text).end() input_max_width = Math.max(input_max_width, $key.find('label').outerWidth()) //记录最大宽度 //按级别缩进 if(!!level) { //console.log('__level__',level) $key.css({'margin-left': `${level*18}px`}) //$key.addClass('bpx-player-dm-setting-right-separator') //$key.empty().css({'margin-top': '13px', 'border-top': '1px solid rgba(229,233,239,.1)' ,'height': '8px'}) //$key.css({'margin-bottom': '0px'}) } //set tips if(bb_type.indexOf(BILI_2_X) !== -1){ !!tip && add_tip($tip, $key.find('.bpx-player-dm-setting-left-flag-title'), {'text':tip, 'millisec':200, 'top':9, 'left':0}) }else{ !!tip && tip_create_3_X({ target: $key.find('.bpx-player-dm-setting-left-flag-title'), tip_target: $('#hhh_tip'), title: tip, gap: 6 }) } //禁用 if(disable === ON) { $key.css({'pointer-events': 'none', 'opacity': 0.3}) } } //填充空行 let one_input_height = +$input_div.children(':first').outerHeight() + (+$input_div.children(':first').css('margin-top').match(/\d+/)[0]); for(; len<=18; len++){ $div.clone().appendTo($input_div).css({height: `${one_input_height}px`}); } //add event checkbox_event(); } //初始化【group】和【checkbox】 function input_init(){ for(let [group, {options, btn, type}] of Object.entries(config.sets)){ $cover.clone().appendTo($group_div).attr('id', group) //复用封面DOM模板 .css({height: '22px'/*, border: '1px solid hsla(0,0%,100%,.2)'*/}) .find(bb['dmRightMoreText']).text(btn+'(点击)').end(); //$reset.clone().appendTo($group_div).attr('id', group) //复用封面DOM模板 // .css({height: '22px', display: 'flex'}).text(btn+'(点击)'); $(`#${group}`).click(function(){ config.group = group; // //if($('body')[0].hhh_body_key_event !== '快捷键设置') return; //$('body')[0].hhh_body_key_event = group; //取消激活 let $active = $group_div.find('.active'); let active_text = $active.find(bb['dmRightMoreText']).text(); $active.find(bb['dmRightMoreText']).text(active_text.split('***')[0]); $active.removeClass('active'); //激活 let text = $(this).find(bb['dmRightMoreText']).text(); $(this).find(bb['dmRightMoreText']).text(text+'***'); $(this).addClass('active'); $input_div.empty(); if(type == 'checkbox') { checkbox_init(group); $input_div.css('width', `${all_checkbox_width}`); $wrap.css({'height': $item_2.css('height'), 'width': $item_2.css('width')}); $move.css('transform', `translateX(-${old_width}px)`); $('#hhh_item_2_more').parent().css({'justify-content': 'space-between'}); //设置"返回"位置 checkbox_args_init(group); } else if(1&&type == 'key') { set_keybindings($input_div); $input_div.css('width', 'auto'); $('#hhh_item>div:first').css({'height':'max-content','width':'max-content'}) $wrap.css({'height': $item_2.css('height'), 'width': $item_2.css('width')}); $move.css('transform', `translateX(-${old_width}px)`); $('#hhh_item_2_more').parent().css({'justify-content': 'flex-end'}); //设置"返回"位置 } }) } } input_init(); //添加恢复默认设置 $reset.appendTo($item_2_div); //重设置【item_2 - 恢复默认设置】 click $reset.click(function(){ config.defaultCheckboxSetting(); config.storageCheckboxSetting(); for(let [group, {options, type}] of Object.entries(config.checkboxes)){ if(type === 'checkbox'){ for (let [key, { status, fn, args }] of Object.entries(options)) { //if(config.getCheckboxSettingStatus(key) === status) continue; $(`#${key}`).find('input').prop('checked', status); //设定参数设置 checkbox_args_update(key, status); checkbox_go(key, status, fn, args); } } } $(`#${config.group}`).click(); //更新 }) /*----------------------------------- * 下部div *----------------------------------*/ //添加分割线 $separator.clone().appendTo($item_2_div); //添加返回 //$reback.clone().appendTo($item_2_div); //TEST let $bottom_div = $div.clone(); $bottom_div.appendTo($item_2_div).css({display:'flex', 'justify-content':'space-between'}); $reback.clone().appendTo($bottom_div); //总在最上 let $always_top = $input.clone().appendTo($bottom_div).attr('id', 'hhh_always_top').css('width', 'auto') .css({'margin':'2px 0px 5px 20px'}) .find('input').prop('checked', false).end() .find('.bui-checkbox-name').text('总在最上').end(); let boserver; $always_top.click(function(e){ if(e.target.type === 'checkbox'){ let checkbox = e.target; if(e.target.checked === ON){ //鼠标离开时panel保持在最上 boserver?.disconnect(); boserver = new MutationObserver((mutations, observer) => { mutations.forEach(mutation => { const target = mutation.target; if(typeof target.className === 'string' && target.className !== ''){ //console.log(target.className +' | '+mutation.attributeName); $(target).show(); } }); }); boserver.observe($(bb['dmWrap'])[0], { attributes: true }); //鼠标是否在弹幕设置界面内 document.onmouseup = function(e){ let path = e.path; //log(path) let is_in_dm_setting_panel = false; $(path).each(function(){ if(this.className === 'bui-panel-wrap') { is_in_dm_setting_panel = true; return; } }) if(is_in_dm_setting_panel === false){ document.onmouseup = null; checkbox.checked = OFF; boserver?.disconnect(); $(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseleave')); } } }else{ document.onmouseup = null; boserver?.disconnect(); } } }) /*----------------------------------- * 添加事件等 *----------------------------------*/ //重设置【自定义设置按钮】 // $DSet.mouseenter(function(){ // $wrap.css('height', $DSet.find('.bui-panel-item-active').css('height')); // }) //重设置【item_0 - 自定义设置】 click $('#hhh_custom').click(function(){ $wrap.css({'height': $item_2.css('height'), 'width': $item_2.css('width')}); $move.css('transform', `translateX(-${old_width}px)`); $DSet.find('.bui-panel-item-active').removeClass('bui-panel-item-active'); $item_2.addClass('bui-panel-item-active'); $(`#${Object.keys(config.sets)[0]}`).click(); //设置第一项 }) //重设置【item_1 - 更多弹幕设置】 click $item_1.find(bb['dmRightMore']).click(function(){ $item_0.css('height', item_0_height); //会改变item_0长度 $wrap.css({'height': $item_0.css('height'), 'width': $item_0.css('width')}); }) //重设置【item_2 - 返回】 click $item_2.find('#hhh_item_2_more').click(function(){ $wrap.css({'height': $item_0.css('height'), 'width': $item_0.css('width')}); $move.css('transform', 'translateX(0px)'); $DSet.find('.bui-panel-item-active').removeClass('bui-panel-item-active'); $item_0.addClass('bui-panel-item-active'); }) //重设置【item_2 - 封面】 click $item_2.find('#hhh_item_2_cover').click(function(){ if($('#hhh_img').length === 1){ let $hhh_img = $('#hhh_img'); //let width = $(bb['video']).width()*0.9; //$('#hhh_img').css('width', width); $hhh_img.css('display') === 'none'? $hhh_img.css('display', 'flex'): $hhh_img.css('display', 'none'); } else { let imgurl = $(bb['coverImg']).attr('content').replace('http:','https:').replace(/@.*/,'') let $img = $(``); let $d = $div.clone().attr('id', 'hhh_img') .css({position: 'absolute', top:'50%', left:'50%', transform: 'translate(-50%,-50%)'}) .css({'z-index':`${$(bb['playTipWrap']).css('z-index')}`}) .css({'display':'flex'}) //.css('width',width) .css({'align-items':'center','justify-content':'center'}) .appendTo($(bb['videoWrap'])); $img.css({width:'100%', height:'auto'}).appendTo($d); //重设置【item_2 - 封面 - IMG】 $('#hhh_img').click(function(){ $(this).css('display','none'); window.open(imgurl); }) } }) //重设置【item_2 - 快捷键说明】 click let set_hotkey_panel_close = false; $item_2.find('#hhh_item_2_hotkey').click(function(){ let is_active = bb_type.indexOf(BILI_2_X) !== -1 ? $(bb['hotkeyPanel']).hasClass('active') : $(bb['hotkeyPanel']).css('display') !== 'none'; //2.X和3.X不一样?我来让你一样! if(is_active === false){ let evt = new MouseEvent('contextmenu', { clientX:-9999, clientY:-9999 }); $(bb['videoWrap'])[0].dispatchEvent(evt); evt = bb_type.indexOf(BILI_2_X) !== -1? new MouseEvent('click', { bubbles:true }): new MouseEvent('mousedown', { bubbles:true }); let node = bb_type.indexOf(BILI_2_X) !== -1? 'a': 'li'; $(bb['playerContextMenu']).find(`${node}:contains("快捷键说明(bilibili关灯)")`)[0].dispatchEvent(evt); let $last_system_itme = $('#hhh_last_system_hotkey_panel_item'); $last_system_itme.prevAll().css('display','none'); $last_system_itme.css('display','none'); $last_system_itme.nextAll().css('display','block'); //$(bb['playerContextMenu']).find(`li:contains("快捷键说明(bilibili关灯)")`).mousedown(); let $hotkey = $(bb['hotkeyPanel']); //let $hotkey = $(bb['hotkeyPanel']).css('z-index', $(bb['playVideoControlWrap']).css('z-index')-1); //同步显示 // let wrap_left = $(bb['dmWrap']).position().left; // let box_left = $(bb['dmBox']).position().left; // $(bb['dmWrap']).css('left', wrap_left + $item_0.width()); // $(bb['dmBox']).css('left', box_left - $item_0.width()); // $(bb['dmWrap']).css('height', $hotkey.innerHeight()); // $(bb['dmWrap']).css('width', $hotkey.innerWidth()-50); // $(bb['dmBox']).css('width', item_2_width); //$(bb['dmWrap']).css('background', 'blue').css('opacity', 0.8); //test //同步隐藏 // waitForTrue(()=> ($(bb['hotkeyPanel']).hasClass('active') === false && $(bb['hotkeyPanel']).css('display') === 'none') || $(bb['dmWrap']).css('display') === 'none', () => { // //$(bb['hotkeyPanel']).removeClass('active').css('display', 'none'); // $(bb['hotkeyPanelClose']).click(); // }); let new_left = $item_2.offset().left + $item_2.width(); let new_top = $hotkey.offset().top + ($item_2.offset().top + $item_2.height()) - ($hotkey.offset().top + $hotkey.innerHeight()); $hotkey.offset({left:new_left, top:new_top}); //$hotkey.offset({left:new_left, top:new_top}).find(bb['hotkeyPanelClose']).css('display', 'none'); if(set_hotkey_panel_close === false){ // set_hotkey_panel_close = true; $(bb['hotkeyPanelClose']).click(function(){ $(bb['hotkeyPanel']).css({left: '50%', top:'50%'}); }) } }else{ $(bb['hotkeyPanel']).css({left: '50%', top:'50%'}); $(bb['hotkeyPanelClose']).click(); // $(bb['dmWrap'])[0].style.cssText = ''; // $(bb['dmBox'])[0].style.cssText = ''; // $(bb['dmWrap']).css('display', 'block'); } }) let id = 0; //重设置弹幕设置触发按钮 $DSet.on('mouseenter', function(){ clearInterval(id); if ($(bb['dmWrap']).css('display') !== 'none') return; $item_0.css('height', $item_0[0].hhh_new_height); $wrap.css({'height': $item_0.css('height'), 'width': $item_0.css('width')}); $move.css('transform', 'translateX(0px)'); $move.css('width', '10000px'); //留出足够空间,省事 $DSet.find('.bui-panel-item-active').removeClass('bui-panel-item-active'); $item_0.addClass('bui-panel-item-active'); $(bb['dmWrap'])[0].style.cssText = ''; $(bb['dmBox'])[0].style.cssText = ''; let dm_right = $('.bpx-player-dm-setting')[0].getBoundingClientRect().right let dm_max_width = 724 //默认最大宽度 let dm_wrap_cssRight = parseInt($('.bpx-player-dm-setting-wrap').css('right')) // log('dm_right: '+dm_right) // log('dm-box_width: '+dm_max_width) // log('dm-wrap_cssRight: '+dm_wrap_cssRight) // log(dm_max_width-dm_right-10) // log($('.bpx-player-dm-setting-wrap').css('right')) if(dm_right-dm_max_width < dm_wrap_cssRight){ $('.bpx-player-dm-setting-wrap').css('right', dm_right-dm_max_width-10) } }).on('mouseleave', function(){ //解决mouseleave和bb['dmWrap']).css('display')不同步的问题 clearInterval(id); id = setInterval(() => { if ($(bb['dmWrap']).css('display') !== 'none') return; clearInterval(id); let $main = $('.keybindings-editor'); $main.find('.monaco-list-rows').off('click.hhh_widget_listrow'); $main.find('.monaco-list-rows').off('dblclick.hhh_widget_listrow'); $main.find('input').off('keydown.hhh_widget_input'); $main.find('input').off('keyup.hhh_widget_input'); $main.find('input').add('.defineKeybindingWidget').off('mousewheel.hhh_widget_input'); $main.find('input').off('blur.hhh_widget_input'); $('body').off('keydown.hhh_widget_listrow'); $('body').off('keydown.hhh_checkbox_args'); }, 200); }); // $DSet.on('mouseleave', function(){ // return false; // }) //执行所有默认设置 if(run_default_setting_flag === 'run_default_setting') $reset.click() //初始化【checkbox】 $(bb['dmWrap']).find('.bui-panel-wrap').css('visibility', 'hidden') //避免弹幕设置框闪影 $(bb['dmWrap']).find('.bui-panel-move').css('visibility', 'hidden') $(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseenter')) $(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseover')) $(bb['dmWrap']).css('visibility', 'hidden') Object.keys(config.sets).forEach((kname)=>{ if(config.sets[kname].type === 'checkbox') $(`#${kname}`).click() }) //点击所有checkbox,得到最大宽度 $(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseout')) $(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseleave')) setTimeout(() => { $(bb['dmWrap']).hide() $(bb['dmWrap']).css('visibility', 'visible') $(bb['dmWrap']).find('.bui-panel-wrap').css('visibility', 'visible') $(bb['dmWrap']).find('.bui-panel-move').css('visibility', 'visible') }, 0) all_checkbox_width = input_max_width + input_number_max_width + 'px' } //记忆弹幕操作 function dm_remember(setting_func){ // if(bb_type.indexOf(BILI_3_X_VIDEO) !== -1) $('.squirtle-controller-wrap').css({"display":"flex"}); // $(bb['dmWrap']).css({"display":"block"}); $(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseenter')); //$(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseover')); setting_func(); //$(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseout')); $(bb['dm'])[0].dispatchEvent(new MouseEvent('mouseleave')); // $(bb['dmWrap']).css({"display":"none"}); // if(bb_type.indexOf(BILI_3_X_VIDEO) !== -1) $('.squirtle-controller-wrap').css({"display":"none"}); } //因为遮挡弹幕,去掉全屏时鼠标悬停时产生的顶端mask function removeVideoTopMask(is_remove){ let top_mask_class = bb['videoTopMask'].substr(1); if(is_remove === ON) $(bb['videoTopMask']).attr('id', top_mask_class).removeClass(); else $(`#${top_mask_class}`).addClass(top_mask_class); } //1、播放关灯 2、暂停开灯 3、下滚时开灯 4、跳过充电 function lightOffWhenPlaying(status){ $(h5Player).off('play.hhh'); if(status === OFF) return $(h5Player).on('play.hhh', ()=> !is_lightoff() && lightoff_btn()) } function lightOnWhenPause(status){ $(h5Player).off('pause.hhh'); if(status === OFF) return $(h5Player).on('pause.hhh', ()=> is_lightoff() && lightoff_btn()) } function lightOnWhenLike(status, screen_top){ $(window).off('scroll.hhh_light_like'); if(status === OFF || biliinfo.lightoff_state === OFF) return $(window).on('scroll.hhh_light_like', function() { let $mini = $('.bpx-player-mini-warp') if($mini.length > 0 && $mini.css('display') !== 'none') return const video_top = $('.bpx-player-video-area')[0].getBoundingClientRect().top - $('#biliMainHeader').height()// + $('#biliMainHeader').height() const light_on_limit = $('.bpx-player-video-area')[0].getBoundingClientRect().height * screen_top // log(-video_top, light_on_limit, screen_top) if(-video_top >= light_on_limit) is_lightoff() && lightoff_btn(LIGHTON_LIKE) else !is_lightoff() && lightoff_btn(LIGHTON_LIKE) }) } function jumpElectric(status){ $(h5Player).off('ended.hhh'); if(status === OFF) return; $(h5Player).on('ended.hhh', ()=> $(bb['playJumpElectric']).click()); } //自动打开弹幕 function autoOpenDanmu(){ $('body')[0].addEventListener('DOMSubtreeModified', function fn_(e) { //typeof e.target.className === 'string' && e.target.className !== '' && log(e.target.className); //log($(bb['danmakuSwitch']).last().find('input')[0].checked) let $target = $(e.target); if($target.hasClass('bilibili-player-video-info-people-number') || bb_type.indexOf(BILI_3_X_VIDEO) !== -1) { this.removeEventListener('DOMSubtreeModified', fn_); //this.removeEventListener('DOMSubtreeModified', arguments.callee); let $danmaku = $(bb['danmakuSwitch']).last().find('input'); if($danmaku[0].checked === false) $danmaku.click(); let t = setInterval(() => { if($danmaku[0].checked === false) $danmaku.click(); }, 200); setTimeout(() => { clearInterval(t) }, 2000); } }); } //隐藏屏幕三连弹窗 function hideThreePopup(status){ let $popup = $('.bili-guide-all') if($popup.length === 0) return if(status === ON) $popup.hide() else $popup.show() } //隐藏屏幕打分弹窗 ** function hideScorePopup(status){ let $popup = $('.bili-score'); if($popup.length === 0) return if(status === ON) $popup.hide() else $popup.show() } //双击或中键全屏 function dblclickFullScreen(status){ $(bb['videoWrap']).off('mousedown.hhh'); if(status === OFF) return; $(bb['videoWrap']).on('mousedown.hhh', function(e){ if(e.button === 1 || e.button === 4 || e.button === 3) { // 1中键 if(e.button === 1) web_fullscreen(); else fullscreen(); e.preventDefault(); e.stopPropagation(); } }); //$(bb['playTipWrap']).dblclick(function(){ fullscreen() }); //2.66自带双击全屏 } //扩展播放倍速范围 function extend_video_speed(){ let $speedul = $(bb['videoSpeedActive']).parent(); let ratios = ['0.1x','0.5x','0.75x','1.0x','1.25x','1.5x','1.75x','2.0x','2.5x','3.0x','3.5x','4.0x','4.5x','5.0x']; let $li = $speedul.find('li:first').clone(true, true); $speedul.empty(); ratios.forEach(ratio => { $this = $li.clone().prependTo($speedul).text(ratio); $this.attr('data-value', `${$this.text().split('x')[0]}`); }); let speed_active = (bb_type === BILI_3_X_MOVIE)? 'active': (bb_type.indexOf(BILI_3_X_VIDEO) !== -1)? 'bpx-state-active': 'bilibili-player-active'; $speedul.find('li:contains("1.0x")').addClass(speed_active); $speedul.on('click', function(e){ let $li = $(e.target); let speed = $li.attr('data-value'); if(speed !== undefined){ h5Player.playbackRate = +speed; setTimeout(function(){ let speed_active = (bb_type === BILI_3_X_MOVIE)? 'active': (bb_type.indexOf(BILI_3_X_VIDEO) !== -1)? 'bpx-state-active': 'bilibili-player-active'; $(bb['videoSpeedActive']).removeClass(speed_active); $li.addClass(speed_active); //$(bb['videoSpeedName']).text($li.text()); },0) } }) } //区间舍入 function roundoff(rough_speed, direction, ratios){ if(direction > 0){ for(let i=0; i=0; i--){ let speed = parseFloat(ratios[i]) if(rough_speed >= speed) return speed } return parseFloat(ratios[0]) } } //调节视频倍速 function video_select_speed(level) { const $speedul = $(bb['videoSpeed']).parent() const data_x = [] $speedul.find('li').each(function(){ data_x.unshift($(this).text()) }) //类似['0.5x', '0.75x', '1.0x', '1.25x', '1.5x', '2.0x']; const value_x = data_x.map((v)=>parseFloat(v)) const v = roundoff(h5Player.playbackRate, level, data_x) const index = value_x.indexOf(v) const x = h5Player.playbackRate !== v ? data_x[index] : data_x[index + level] || data_x[index] //log(level, h5Player.playbackRate, v, x, data_x[index + level], data_x[index]) $(`${bb["videoSpeed"]}:contains("${x}")`).click() showHint(parent, '#hhh_wordsHint', x) } //roundoff function video_speed_0dot1(dot) { const rate = (h5Player.playbackRate + dot).toFixed(1) h5Player.playbackRate = rate showHint(parent, '#hhh_wordsHint', `${rate}x`) } //计算坐标 function tip_calc_xy_3_X(t){ let crects = t.target[0].getBoundingClientRect(), trects = t.tip_target[0].getBoundingClientRect(), tip_height = trects.height, tip_width = trects.width, width = crects.width, top = crects.top - tip_height, left = crects.left + (width - tip_width) / 2; return {x:left, y:top}; } //创建tip function tip_create_3_X(et){ function show_tip(t){ //显示及渐隐效果(抄bilibili^^) let o = t.tip_target, tip = t.target.data('tip'), color = t.color || 'white', text = (typeof tip === 'function' && tip()) || tip; o.find('div').css({color: color}).text(text); let xy = tip_calc_xy_3_X(t), target = t.target[0]; clearTimeout(target.showHintTimer), target.showHintTimer = window.setTimeout((function() { //log(target); o.stop(); o.css({visibility: 'visible'}); o.css({left: xy.x, top: xy.y}); o.animate({ opacity: 1, top: xy.y - t.gap, }, t.duration) } ), t.delay) } let $target = et.target, $tip = et.tip_target; $target.data('tip', et.title); $target.mouseenter(function(e){ show_tip({ name: 'hhh_tip', target: $target, tip_target: $tip, color: et.color, position: 2, delay: 200, duration: 300, gap: et.gap, type:'show', }); }).mouseleave(function(e){ clearTimeout($target[0].showHintTimer); $tip.stop().css({visibility: 'hidden', opacity: 0}); }) } //更新tip function tip_update_3_X(t){ if(t.title !== undefined) t.target.data('tip', t.title); tip = t.target.data('tip'), text = (typeof tip === 'function' && tip()) || tip; t.tip_target.find('div').text(text); let xy = tip_calc_xy_3_X(t); t.tip_target.css({left: xy.x, top: xy.y - t.gap}); } //初始化tip function tip_init() { if(bb_type.indexOf(BILI_2_X) !== -1){ $('.bilibili-player-video-danmaku-setting-left-ps').mouseover(); waitForTrue(()=> $('.player-tooltips.tip.top-center.animation').length === 1, () => { $tip = $('.player-tooltips.tip.top-center.animation').clone().removeClass('active'); $tip.find('.tooltip').css('background-color', 'rgba(0,0,0,0.9)'); $('.bilibili-player-video-danmaku-setting-left-ps').mouseout(); }); } else { if($('#hhh_tip').length <= 0) $(bb['playTooltipArea']).append(`
TIP
`); } } //自定义倍速播放 function customPlayRate(open=OFF, rate=2){ if(open === !ON){ $(document).off('keydown.hhh_rate'); $(document).off('keyup.hhh_rate'); let $three_rate = $('.bpx-player-three-playrate-hint'); if($three_rate.length > 0) $three_rate.contents()[1].nodeValue = `倍速播放中`; return false; }else{ let go_id = null; let keystrokes = 0; let old_rate; function kup(e){ if(e.keyCode === keycode['right']){ //right arrow down $('.bpx-player-three-playrate-hint').css('display','none'); clearTimeout(go_id); go_id = null; keystrokes = 0; h5Player.playbackRate = old_rate || h5Player.playbackRate; } } function kdown(e){ if(e.keyCode === keycode['right']){ //right arrow down if(e.ctrlKey || e.shiftKey || e.altKey) return; if(++keystrokes === 1) old_rate = h5Player.playbackRate; //记录第一次按键时播放速度 let $three_rate = $('.bpx-player-three-playrate-hint'); if($three_rate.length > 0){ function go(op){ function g(op, i){ op.now[i] += op.steps[i]; if(op.now[i] > op.end || op.now[i] < op.start){ op.steps[i] = -op.steps[i]; } let $r = $three_rate.find('g:first'); let $g = $r.children(`g:eq(${i})`); $g.attr('opacity', op.now[i]); } g(op,0); g(op,1); g(op,2); //恢复初始状态 if(1&&op.now[0] < op.start){ op.now[0] = op.now_init[0], op.now[1] = op.now_init[1], op.now[2] = op.now_init[2]; op.steps[0] = op.step, op.steps[1] = op.step, op.steps[2] = op.step; //console.table(op) } } //rate=倍速 function loop(rate=2){ //键盘连按大约1秒30次,间隔33毫秒 let op = (function(rate){ let start = 0.15; let now = [0.15, 0.35, 0.55]; let end = 0.75; let millisec = 33; let frame = 1000 / millisec; let cycle = end - start; let step = cycle / (frame/rate/1.33); return { start: start, now_init: JSON.parse(JSON.stringify(now)), now: JSON.parse(JSON.stringify(now)), end: end, millisec: millisec, step: step, steps: [step, step, step], } })(rate); go_id = setInterval(()=>{ go(op); }, op.millisec); } if(keystrokes === 2){ //第二次按键开始倍速播放 $three_rate.css('display',''); Hanzi_digit = ['〇','一','二','三','四','五','六','七','八','九','十'][rate] || rate; $three_rate.contents()[1].nodeValue = `${Hanzi_digit}倍速播放中`; h5Player.play(); h5Player.playbackRate = rate; //loop(rate); //... } if(keystrokes !== 1) return false; //第一次按键正常返回,执行默认操作(快进5秒) } } } $(document).off('keydown.hhh_rate').on('keydown.hhh_rate',kdown) $(document).off('keyup.hhh_rate').on('keyup.hhh_rate',kup) } } //标题抬头显示tags(视频属于哪个区) function add_tags_to_video_title(){ let tags = []; $('.firstchannel-tag a, .secondchannel-tag a').each(function(){ tags.push($(this).text()) }) $('.pubdate-ip.item').after(`( ${tags.join(' | ')} )`) } //读取cookie function get_cookie(name) { let c = document.cookie.split(';') let v = c.find(e => e.match(name+'=')) return v.split(name+'=')[1] } //设置cookie function set_cookie(name, value, day) { day = void 0 !== day ? day : 365 let d = new Date d.setTime(d.getTime() + 24 * day * 60 * 60 * 1e3) document.cookie = name + "=" + escape(value) + ";expires=" + d.toGMTString() + "; path=/; domain=.bilibili.com" } //添加返回旧版按钮 暂不使用红 function add_go_back(){ if(ver === BILI_3_X_VIDEO_V1){ let $sidenav = $('.fixed-sidenav-storage') let $go_back = $sidenav.append('
返回旧版
').children(':last') $go_back.click(()=>{ set_cookie("go_old_video", "1") //返回旧版 location.reload() }) } } //内测版首页添加返回新版按钮 function run_go_back_new_version(){ if(ver === BILI_4_X_V1){ //测试版 waitForTrue(()=> $('.palette-button-wrap .top-btn-wrap').length === 1, () => { let $last_primary = $('.palette-button-wrap .top-btn-wrap') let $go_back = $last_primary.after($last_primary.clone()).next() $go_back.find('svg').remove('svg') $go_back.find('.primary-btn').addClass('visible') $go_back.find('.primary-btn-text').css('width','34px').text('返回新版') $go_back.click(()=>{ set_cookie('i-wanna-go-back', '-1') //返回新版 set_cookie('i-wanna-go-feeds', '-1') //返回新版 set_cookie('buvid3', '') //返回新版 set_cookie('nostalgia_conf', '1') //返回新版 location.reload() }) }) } //if(ver !== BILI_3_X_VIDEO){ //旧版 // set_cookie('buvid3', '') //返回新版 // set_cookie('buvid4', '') //返回新版 // location.reload() //} } //动态隐藏biliMainHeader function dynamic_hide_mainheader(open){ function is_block(){ if ($('.center-search__bar.is-focus').length > 0) return true //是否弹出搜索历史 let is_block = false $('.v-popover').each(function(){ // console.log($(this).css('display')) if($(this).css('display') !== 'none'){ //是否弹出直播、漫画、消息、动态、收藏、历史等 is_block = true return false } }) return is_block } $('body').off('mousemove.hhh_dynamic_hide_mainheader') const $main_header = $('#biliMainHeader') if(open === OFF) { $main_header.css('position', '').show(); return } waitForTrue(()=>$main_header.length > 0, ()=>{ const height = $main_header[0].getBoundingClientRect().height $main_header.css('position', 'absolute').hide() $('body').on('mousemove.hhh_dynamic_hide_mainheader', (e)=>{ const x = e.clientX const y = e.clientY // log(e.clientX, e.clientY) const rect = $('.up-panel-container')[0].getBoundingClientRect() const up_left = rect.left const up_right = rect.right const up_top = rect.top const up_bottom = rect.bottom const is_in_up_box = x > up_left && x < up_right && y > up_top && y < up_bottom const is_pop_dm_setting_box = $('.bpx-player-dm-setting-wrap').css('display') !== 'none' //是否弹出弹幕设置 if(y > 0 && y < height && !is_in_up_box && !is_pop_dm_setting_box) { $main_header.show() } else if(y >= height && !is_block()) { $main_header.hide() } }) }) } //自动点赞 function auto_like(open, [following_status, second]){ // log('---auto_like---') const fn = auto_like function removeEventLists(timerun, timeend){ h5Player.removeEventListener('timeupdate', timerun) h5Player.removeEventListener('ended', timeend) } removeEventLists(fn.timerun, fn.timeend) function is_following_UP() { return $('.following').length > 0 } function is_following_UP_by_Joint() { //联合投稿 if($('.membersinfo-upcard-wrap').length === $('.add-follow-btn').length) return false //里面没有关注的UP主 return true } function is_like() { return $(bb['like']).hasClass('on') } //备注:$.ajax - url: `https://api.bilibili.com/x/web-interface/archive/has/like?bvid=${bvid}`, .data function like_click() { $(bb['like']).click() } if(open === OFF) return if(is_like() === true) return if(following_status && !is_following_UP() && !is_following_UP_by_Joint()) return if(second <= 0) { log('自动点赞,打开网页后'); like_click(); return } //进入视频页立即点赞 h5Player = geth5Player() let timerun = function(){ // log('---timerun---') const fn = timerun if(fn.time === undefined){ fn.time = Date.now() fn.interval = 0 } const now = Date.now() fn.interval += now - fn.time fn.time = now // log('second: ', fn.interval, second) if(fn.interval > second*1000){ removeEventLists(fn, timeend) log(`自动点赞,播放${second}秒后`) like_click() } } let timeend = function(){ // log('---timeend---') removeEventLists(timerun, timeend) log(`自动点赞,播放完毕后`) like_click() } fn.timerun = timerun fn.timeend = timeend h5Player.addEventListener('timeupdate', timerun) h5Player.addEventListener('ended', timeend) } //加载视频时执行,处理不刷新页面也重新加载视频的情况 function video_loaded_run(){ //记忆洗脑循环 if(config.getCheckboxSettingStatus('rememberVideoRepeat') === ON && config.getCheckboxSettingStatus('videoRepeat') !== ON) { log('记忆洗脑循环') //if(JSON.parse(get_value('hhh-rememberVideoRepeat', $(bb['playSettingRepeatInput'])[0].checked)) === true) $(bb['playSettingRepeatInput']).click(); //localStorage只能保存字符串 if(get_value('hhh-rememberVideoRepeat', $(bb['playSettingRepeatInput'])[0].checked) === true) $(bb['playSettingRepeatInput']).click(); //localStorage只能保存字符串 $(bb['playSettingRepeatInput']).click(function() { set_value('hhh-rememberVideoRepeat', $(bb['playSettingRepeatInput'])[0].checked); }); } //自动点赞 auto_like(config.getCheckboxSettingStatus('autoLike'), [config.getCheckboxSettingStatus('likeFollowed'), config.getCheckboxSettingArgs('autoLike')['second']]) } //收藏列表添加自动连播、洗脑循环按钮(不能同时为真) function add_favlist_autoplay_btn(){ if($('.action-list-container').length <= 0) return if($('.head-right .next-button').length <= 0){ $('head').append('') $('.action-list-body-top').append(`
自动连播
洗脑循环
`) let $next_button_autoplay = $('.head-right .next-button:contains(自动连播)') let $switch_button_autoplay = $next_button_autoplay.find('.switch-button') let $next_button_circle = $('.head-right .next-button:contains(洗脑循环)') let $switch_button_circle = $next_button_circle.find('.switch-button') let $自动切集 = $('.bpx-player-ctrl-setting-handoff .bui-radio-item:contains(自动切集)').find('input') let $播完暂停 = $('.bpx-player-ctrl-setting-handoff .bui-radio-item:contains(播完暂停)').find('input') let $洗脑循环 = $('.bpx-player-ctrl-setting-loop input') if($自动切集.prop('checked') === true) $switch_button_autoplay.addClass('on') if($洗脑循环.prop('checked') === true) $switch_button_circle.addClass('on') function on_circle(){ if($洗脑循环.prop('checked') === false) $洗脑循环.click() } function off_circle(){ if($洗脑循环.prop('checked') === true) $洗脑循环.click() } $next_button_autoplay.click(()=>{ if($switch_button_autoplay.hasClass('on')){ $switch_button_autoplay.removeClass('on') $播完暂停.click() }else{ $switch_button_autoplay.addClass('on') $自动切集.click() $switch_button_circle.removeClass('on') off_circle() } }) $switch_button_circle.click(()=>{ if($switch_button_circle.hasClass('on')){ $switch_button_circle.removeClass('on') off_circle() }else{ $switch_button_circle.addClass('on') on_circle() $switch_button_autoplay.removeClass('on') $播完暂停.click() } }) $播完暂停.click(()=>{ $switch_button_autoplay.removeClass('on') }) $自动切集.click(()=>{ $switch_button_autoplay.addClass('on') }) $洗脑循环.click(()=>{ if($洗脑循环.prop('checked')) $switch_button_circle.addClass('on') else $switch_button_circle.removeClass('on') }) } } //主程序,页面刷新完成时执行 function run(){ //防止重复加载 if ($('#hhh_lightoff').length === 1) { log('重复加载'); return } //bpx test function bpx_test(){ log('-------------------'); log('ver: '+ver2); log('fps: '+fps); log('------h5Player-----'); dir(h5Player); log('currentTime : '+h5Player.currentTime); log('duration : '+h5Player.duration); log('playbackRate : '+h5Player.playbackRate); log('volume : '+h5Player.volume); log('paused : '+h5Player.paused); log('pause() : '+'h5Player.pause()'); log('videoHeight : '+h5Player.videoHeight); log('videoWidth : '+h5Player.videoWidth); log('-------------------'); } /*----------------------------------- *初始化等 *----------------------------------*/ //取得h5 video & window.player h5Player = geth5Player() //保存一些视频参数初始值 app_page_parameters.player_setting_area = $(`.bpx-player-dm-setting-left-area ${bb['progressBar']}`).width()/25 //初始化tip tip_init(); //初始化关灯按钮 lightoff_init(); //激活系统弹幕设置,以此调用系统网页全屏等 $(bb['dm']).mouseenter().mouseleave(); //激活系统播放设置,以此调用系统关灯等 //去掉mouseleave(),否则如果太快执行mouseleave()无法激活关灯class,应该是mouseenter()未执行完就被mouseleave打断了 $(`${bb['playVideo']}${bb['playSetting']}`).mouseenter(); //避免显示设置页面 // waitForNode(() => document.querySelector(bb['playSettingWrap']), (node) => {; // $(node).css({"visibility":"hidden"}); //visible // }) //解决因为激活关灯class,导致全屏时滚轮操作(系统自带)无法调节音量的问题 // waitForTrue(()=> $(bb['playSettingWrap']).css('display') === 'block', () => { // $(bb['playSettingWrap']).css('display', 'none').css('visibility', 'visible'); // }); //取得视频fps get_video_fps_ver_biliinfo(); //取得版本号 //get_ver(); //激活系统提示添加音量等自定义提示DOM pick_volume_hint(); //添加自定义快捷键说明到右键菜单 //add_custom_hotkey_menu(config.hotKeyMenu); add_custom_hotkey_menu(config.sets.keyBinding.options); //初始化自定义设置 init_setting(); /*----------------------------------- *事件等 *----------------------------------*/ //点击关灯 $(`#hhh_lightoff ${bb['switchInput']}:first`).click(function(){ lightoff_btn() }); //判断当前鼠标点击焦点 $(document).mouseup(function(e){ if(e.originalEvent === undefined) return; let path = e.originalEvent.path; curr_focus = path[0]; is_in_biliplayer = false; //鼠标是否在播放界面内 $(path).each(function(){ if(this.id === 'bilibili-player') { is_in_biliplayer = true; return; } }) //隐藏封面 2.x && 3.x if(is_in_biliplayer == false || e.target.className === 'bilibili-player-dm-tip-wrap' || e.target.tagName.toLowerCase() === 'video'){ $('#hhh_img').css('display', 'none'); } }) //扩展播放倍速范围 extend_video_speed(); //修复选择历史弹幕时弹幕填装信息丢失问题 fix_danmaku_info(); //自动运行 if( config.getCheckboxSettingStatus('autoPlay') === ON && $(bb['playSettingAutoplay'])[0].checked === false) $(bb['playSettingAutoplay'] ).click();//开启自动播放 if( config.getCheckboxSettingStatus('autoLightOff') === ON ) lightoff_btn(); //自动关灯 if( config.getCheckboxSettingStatus('autoFullScreen') === ON ) fullscreen(); //自动全屏 if( config.getCheckboxSettingStatus('autoWebFullScreen') === ON ) web_fullscreen(); //自动网页全屏 if( config.getCheckboxSettingStatus('autoWideScreen') === ON ) wide_screen(); //自动宽屏 if( config.getCheckboxSettingStatus('videoRepeat') === ON ) $(bb['playSettingRepeatInput']).click(); //开启洗脑循环(系统默认关闭) if( config.getCheckboxSettingStatus('autoOpenDanmu') === ON ) autoOpenDanmu(); //自动显示弹幕 if( config.getCheckboxSettingStatus('autoSubtitle') === ON && $(bb['playCtrlSubtitle']).find('defs+g>g').length > 1 ) subtitle() //自动开启字幕,判断是否开启字幕,'defs+g>g'.length,关闭字幕:= 2,开启字幕:= 1 //双击或中键全屏 dblclickFullScreen(config.getCheckboxSettingStatus('dblclickFullScreen')) //非全屏滚轮音量调节 //两个参数指定屏幕范围(按百分比),第三个参数表示滚动一下增加的音量百分比,参数四表示暂停时是否调节 //let args = config.getCheckboxSettingArgs('volumeControlWhenNonFullScreen') //wheel_volumeHint(config.getCheckboxSettingStatus('volumeControlWhenNonFullScreen'), args.screen_left,args.screen_rght,args.delta, config.getCheckboxSettingStatus('volumeControlWhenPause')); //滚轮调节弹幕透明度(ctrl),参数表示滚动一下增加的透明度百分比,默认5 //wheel_opacity(config.getCheckboxSettingStatus('danmuOpacityControl'), 5) //因为遮挡弹幕,去掉全屏时鼠标悬停时产生的顶端mask removeVideoTopMask(config.getCheckboxSettingStatus('removeVideoTopMask')) //播放关灯,暂停开灯,跳过充电鸣谢,自动选择最高质量视频,选集/合集独立自动连播 lightOffWhenPlaying(config.getCheckboxSettingStatus('lightOffWhenPlaying')) lightOnWhenPause(config.getCheckboxSettingStatus('lightOnWhenPause')) jumpElectric(config.getCheckboxSettingStatus('jumpElectric')) auto_top_quality_video(config.getCheckboxSettingStatus('autoTopQualityVideo')) auto_list_play(config.getCheckboxSettingStatus('autoListPlay')) /*----------------------------------- *键盘控制 *----------------------------------*/ set_hotkey(config.getCheckboxSettingStatus('openHotKey')); /*设置界面 //viewBox="10 8.5 10 12" $('.bilibili-player-video-danmaku-setting').clone(false,false).insertBefore($('#hhh_lightoff')).attr('id', 'hhh_setting') $('#hhh_setting>span>svg')[0].attributes[1].value = '10 8.5 10 12' $('#hhh_setting>span>svg>path:first').remove() */ //段落循环 loopPlayback(config.getCheckboxSettingStatus('loopPlayback')); //自定义倍速播放 customPlayRate(config.getCheckboxSettingStatus('customPlayRate'), config.getCheckboxSettingArgs('customPlayRate')['rate']); //抬头显示tags(视频属于哪个区) add_tags_to_video_title() //返回旧版 //add_go_back() //动态隐藏biliMainHeader dynamic_hide_mainheader(config.getCheckboxSettingStatus('dynamicMainheader')) //收藏列表添加自动连播按钮 add_favlist_autoplay_btn() //TEST bpx_test() } //干净链接 function clear_url(){ const [path, query] = location.href.split('?') if(path.match(/BV\w+/)?.length === 1){ const new_query = query?.match(/p=\d+/)?.[0] const href = new_query === undefined ? path : `${path}?${new_query}` history.replaceState(null,null, href) } return location.href } const removeSpam = function (url) { const [pathname, qs] = url.split('?') const params = new URLSearchParams(qs) console.log(params) const newParams = new URLSearchParams() for (const [key, val] of params) { if (!SPAM_PARAMS.has(key)) { newParams.append(key, val) } } console.log(qs) console.log(newParams) const newQs = newParams.toString() console.log(newQs) if (newQs) { return `${pathname}?${newQs}` } return pathname } //段落循环 function loopPlayback(open){ /* bpx-player-progress-wrap hhh_loop_wrap bpx-player-progress bpx-player-progress-schedule-wrap //进度条 bpx-player-progress-schedule bpx-player-progress-schedule-buffer bpx-player-progress-schedule-current bpx-player-progress-point-wrap bpx-player-progress-thumb bpx-player-progress-move-indicator //移动指示器 bpx-player-progress-popup //预览 bpx-player-progress-preview bpx-player-progress-preview-image bpx-player-progress-preview-time //进度时间框 bpx-player-progress-pull-indicator bpx-player-progress-cursor bpx-player-progress-scaleplate hhh_loop_arrow_1 hhh_loop_arrow_2 hhh_loop_time_1 hhh_loop_time_2 */ let $progress_area = $('.bpx-player-progress-area'); let $progress_wrap = $('.bpx-player-progress-wrap'); let $progress = $('.bpx-player-progress'); let $indeicator = $('.bpx-player-progress-move-indicator'); let $popup = $('.bpx-player-progress-popup'); let $down = $('.bpx-player-progress-move-indicator-down'); let $up = $('.bpx-player-progress-move-indicator-up'); let $ctrl_time_current = $('.bpx-player-ctrl-time-current'); //监听视频窗口大小变化 let ro = new ResizeObserver( entries => { for (let entry of entries) { //更新loop位置 resize_updata(); } }); if(open !== ON) { $('#hhh_loop_wrap').off('mousedown.hhh_loop_wrap'); $('#hhh_loop_wrap').off('mouseup.hhh_loop_wrap'); $(document).off('keydown.hhh_document'); $(document).off('keyup.hhh_document'); $(document).off('mouseup.hhh_document'); $('#hhh_loop_time_1, #hhh_loop_time_2').off('mouseenter.hhh_loop_time'); $('#hhh_loop_time_1, #hhh_loop_time_2').off('mousedown.hhh_loop_time'); ro.unobserve($(bb['video'])[0]); if($('#hhh_loop_wrap').length > 0) { loop_state_switch(0); ctrl_play_time(); $('#hhh_loop_wrap')[0].hhh_loop_state_switch = false; $('#hhh_loop_time_1')[0].hhh_seconds = undefined; $('#hhh_loop_time_2')[0].hhh_seconds = undefined; } return false; } //if($('#hhh_loop_wrap').length > 0) return true; if(bb_type.indexOf(BILI_3_X_VIDEO) === -1) return false; //保存arrow原color if(!$progress_area.attr('old_arrow_color')) $progress_area.attr('old_arrow_color', $down.css('border-top-color')); let old_arrow_color = $progress_area.attr('old_arrow_color'); let new_arrow_color = '#FFA500'; let is_begin = true; // Observe one or multiple elements //squirtle-video-time-now 2.X bilibili-player-video-time-now 3.X ro.unobserve($(bb['video'])[0]); ro.observe($(bb['video'])[0]); //css $('#hhh_head_style').remove(); //test if($('#hhh_head_style').length<1){ $progress_area.append(``); } //更新位置 function resize_updata(){ let $p1 = $('#hhh_loop_time_1'); let $p2 = $('#hhh_loop_time_2'); let $bar = $('#hhh_loop_bar'); if($p1.length !== 1 || $p2.length !== 1 && $bar.length !== 1) return; let t1 = $p1[0].hhh_seconds; let t2 = $p2[0].hhh_seconds; let id1 = $p1.attr('hhh_loop_arrow_id'); let id2 = $p2.attr('hhh_loop_arrow_id'); let rect = $progress[0].getBoundingClientRect(); //let rect = $('.bilibili-player-video-progress')[0].getBoundingClientRect(); let newleft = t1/h5Player.duration*rect.width; $p1.add($('#'+id1)).css('left', newleft); newleft = t2/h5Player.duration*rect.width; $p2.add($('#'+id2)).css('left', newleft); //log($p1.css('left'),$p2.css('left')); //log('------'+newleft); t1 = $p1.css('left').slice(0,-2); t2 = $p2.css('left').slice(0,-2); let begin = Math.min(t1,t2); let end = Math.max(t1,t2); $bar.css({left:begin, width:end-begin}); } function show_loop_time(sign_show){ //$popup.find('.bpx-player-progress-preview-image').hide(); } function hide_loop_time(){ //$popup.find('.bpx-player-progress-preview-image').show(); } //更新 function loop_update($hhh_loop_time, o){ //function PrefixZero(num, n) { return (Array(n).join(0) + num).slice(-n) } //time = PrefixZero(Math.floor(time/60),2) + ':' + PrefixZero(time%60,2); let left = 0, seconds = 0, rect = $progress[0].getBoundingClientRect(); //rect2 = $progress_wrap[0].getBoundingClientRect(); //log(`rect_width: ${rect.width} - $progress.width(): ${$progress.width()}`); //log(`rect2_width: ${rect2.width} - $progress_wrap.width(): ${$progress_wrap.width()}`); //get second if(o.left !== undefined){ left = o.left - rect.x; if(left < 0 || left > rect.width) return $hhh_loop_time[0].hhh_seconds; //Position To Time seconds = Math.floor(left/rect.width*h5Player.duration); }else if(o.seconds !== undefined){ //Time To Position left = o.seconds/h5Player.duration*rect.width; if(left < 0 || left > rect.width) return $hhh_loop_time[0].hhh_seconds; seconds = o.seconds; }else{ log('参数错误',o); return;} //updata left $hhh_loop_time.css({left:left, display:'block'}); $("#"+$hhh_loop_time.attr('hhh_loop_arrow_id')).css({left:left, display:'block'}); //updata time $hhh_loop_time[0].hhh_seconds = seconds; let time = sec2str(seconds); $hhh_loop_time.find('.hhh-bpx-player-progress-preview-time').text(time); //updata (width + margin-left) //log($('.bpx-player-progress-preview-time')[0].getBoundingClientRect()) let width = $hhh_loop_time.find('.hhh-bpx-player-progress-preview-time')[0].getBoundingClientRect().width; $hhh_loop_time.find('.hhh-bpx-player-progress-preview').width(width); $hhh_loop_time.css('margin-left', `${-width/2}px`); //updata bar let s1 = $('#hhh_loop_time_1')[0].hhh_seconds; let s2 = $('#hhh_loop_time_2')[0].hhh_seconds; if(s1!==undefined && s2!==undefined){ let t1 = $('#hhh_loop_time_1').css('left').slice(0,-2); let t2 = $('#hhh_loop_time_2').css('left').slice(0,-2); let begin = Math.min(t1,t2); let end = Math.max(t1,t2); $('#hhh_loop_bar').css({display:'block', left:`${begin}px`, width:`${end-begin}px`}); } return seconds; } function ctrl_play_time(){ //time now function set_time(e) { //log('time-current') if($('#hhh_loop_wrap')[0].hhh_loop_state_switch && $('#hhh_loop_wrap')[0].hhh_loop_state_switch !== 0){ let t1 = $('#hhh_loop_time_1')[0].hhh_seconds; let t2 = $('#hhh_loop_time_2')[0].hhh_seconds; let begin = Math.min(t1,t2); let end = Math.max(t1,t2); //log(t1+' - '+t2); if(h5Player.currentTime > end){ //log(t1+' - '+t2); h5Player.currentTime = begin; } } } let s1 = $('#hhh_loop_time_1')[0].hhh_seconds; let s2 = $('#hhh_loop_time_2')[0].hhh_seconds; let is_bar = $('#hhh_loop_bar').css('display') !== 'none'; //log(s1+' - '+s2+' - '+is_bar) if(s1!==undefined && s2!==undefined && is_bar){ $ctrl_time_current[0].removeEventListener('DOMNodeInserted', set_time); $ctrl_time_current[0].addEventListener('DOMNodeInserted', set_time); }else{ $ctrl_time_current[0].removeEventListener('DOMNodeInserted', set_time); } } //loop显示状态切换 function loop_state_switch(state){ $('#hhh_loop_wrap')[0].hhh_loop_state_switch = state; let $p1 = $('#hhh_loop_time_1').add($('#hhh_loop_arrow_1')), $p2 = $('#hhh_loop_time_2').add($('#hhh_loop_arrow_2')), $bar = $('#hhh_loop_bar'); if(state === 0){ //no loop $p1.hide(), $p2.hide(), $bar.hide(); }else if(state === 1){ //loop 1 if($p1[0].hhh_seconds) $p1.show(); if($p2[0].hhh_seconds) $p2.show(); if($p1[0].hhh_seconds && $p2[0].hhh_seconds) $bar.show(); }else{ //loop 2 $p1.hide(), $p2.hide(); if($p1[0].hhh_seconds && $p2[0].hhh_seconds) $bar.show(); } } /*--------------------------- * 插入节点 *--------------------------*/ if($('#hhh_loop_wrap').length === 0){ let progwrapclass = $progress_wrap.attr('class'); //wrap append let $hhh_loop_wrap = $progress.wrap(`
`).parent(); //arrow append * 2 $indeicator.clone(true,true).prependTo($progress).attr('id','hhh_loop_arrow_1').hide() .css({visibility:'visible', opacity:1}) .find('.bpx-player-progress-move-indicator-up').remove().end() .find('.bpx-player-progress-move-indicator-down').css({'border-top-color':new_arrow_color}).end(); $('#hhh_loop_arrow_1').clone(true,true).prependTo($progress).attr('id','hhh_loop_arrow_2'); //preview-time append * 2 $popup.clone(true,true).prependTo($progress_area).attr({'id':'hhh_loop_time_1', 'hhh_loop_arrow_id':'hhh_loop_arrow_1'}) .addClass('hhh-bpx-player-progress-popup').removeClass('bpx-player-progress-popup').hide() .find('.bpx-player-progress-preview-image').remove().end() .find('.bpx-player-progress-hotspot').remove().end() .find('.bpx-player-progress-preview').addClass('hhh-bpx-player-progress-preview').removeClass('bpx-player-progress-preview').end() .find('.bpx-player-progress-preview-time').addClass('hhh-bpx-player-progress-preview-time').removeClass('bpx-player-progress-preview-time').end(); $('#hhh_loop_time_1').clone(true,true).prependTo($progress_area).attr({'id':'hhh_loop_time_2', 'hhh_loop_arrow_id':'hhh_loop_arrow_2'}); //schedule append $('.bpx-player-control-top').find('.bpx-player-progress-schedule-buffer:first').clone(true,true).appendTo($progress).css({transform:'scaleX(0.999999)'}) //配合其他transform,否则height不同,scaleX会改变height,bug? .css({background:'rgba(255,165,0,0.5)', display:'none'}) .attr('id','hhh_loop_bar'); } /*--------------------------- * 事件 *--------------------------*/ //设置起点和终点 $('#hhh_loop_wrap')[0].hhh_loop_time_id = false; $('#hhh_loop_wrap').off('mousedown.hhh_loop_wrap'); $('#hhh_loop_wrap').off('mouseup.hhh_loop_wrap'); $('#hhh_loop_wrap').on('mousedown.hhh_loop_wrap', function(e){ if(e.ctrlKey === true){ return false } }).on('mouseup.hhh_loop_wrap', function(e){ if(e.ctrlKey === true){ let $hhh_loop_time = is_begin? $('#hhh_loop_time_1'): $('#hhh_loop_time_2'); is_begin = !is_begin; loop_update($hhh_loop_time, {left:e.pageX}); //$('.bilibili-player-video-toast-wrp').css('z-index', 0); this.hhh_loop_time_id = $hhh_loop_time[0].id; loop_state_switch(1); ctrl_play_time(); return false; }else{ this.hhh_loop_time_id = false; } }); //键盘 down $(document).off('keydown.hhh_document'); $(document).on('keydown.hhh_document',function(e){ if(e.keyCode === keycode['Ctrl']) { //Ctrl if($progress_wrap.hasClass('bpx-state-active') === true){ //隐藏(img + up) $down.css('border-top-color', new_arrow_color); $up.hide(); show_loop_time(); } $('#hhh_loop_time_1, #hhh_loop_time_2').css({cursor: 'pointer'}) //代表可以点击time标签 }else if(e.keyCode === keycode['left'] || e.keyCode === keycode['right']){ //键盘调整开始结束时间 if($('#hhh_loop_wrap')[0].hhh_loop_time_id !== false && $('#hhh_loop_time_1').css('display') !== 'none'){ toggle_control_top('show'); let $hhh_loop_time = $('#'+$('#hhh_loop_wrap')[0].hhh_loop_time_id); let seconds = $hhh_loop_time[0].hhh_seconds; e.keyCode === keycode['right'] ? ++seconds : --seconds; loop_update($hhh_loop_time, {seconds:seconds}); return false; } }else if(e.keyCode === 'L'.charCodeAt()){ //切换 if($('#hhh_loop_time_1')[0].hhh_seconds !== undefined){ toggle_control_top('show'); $('#hhh_loop_wrap')[0].hhh_loop_state_switch = ++$('#hhh_loop_wrap')[0].hhh_loop_state_switch % 3; loop_state_switch($('#hhh_loop_wrap')[0].hhh_loop_state_switch); ctrl_play_time(); } } }); //键盘 up $(document).off('keyup.hhh_document'); $(document).on('keyup.hhh_document',function(e){ if(e.keyCode === 17) { //Ctrl $down.css('border-top-color', old_arrow_color); $up.show(); hide_loop_time(); $('#hhh_loop_time_1, #hhh_loop_time_2').css({cursor: 'w-resize'}) } else if(e.keyCode === keycode['left'] || e.keyCode === keycode['right']){ //段落循环 if($('#hhh_loop_wrap')[0].hhh_loop_time_id !== false){ //hide_loop_time(); return false; } } }); //判断当前鼠标点击焦点 $(document).off('mouseup.hhh_document'); $(document).on('mouseup.hhh_document', function(e){ if(!e.ctrlKey && $('#hhh_loop_wrap').length > 0 && $('#hhh_loop_wrap')[0].hhh_loop_time_id) $('#hhh_loop_wrap')[0].hhh_loop_time_id = false; }); //拖动确定范围 $('#hhh_loop_time_1, #hhh_loop_time_2').off('mouseenter.hhh_loop_time'); $('#hhh_loop_time_1, #hhh_loop_time_2').off('mousedown.hhh_loop_time'); $('#hhh_loop_time_1, #hhh_loop_time_2').on('mouseenter.hhh_loop_time', function(e){ $(this).attr('in_dom', true); }).on('mouseleave.hhh_loop_time', function(e){ $(this).attr('in_dom', false); }).on('mousedown.hhh_loop_time', function(e){ var $this = $(this); var doc = document; $this.css('z-index', 10000); //当前的在上面 if(e.ctrlKey){ loop_state_switch(0); ctrl_play_time(); }else{ loop_update($this, {left:e.clientX}); doc.onmousemove = function(e){ //log(e.clientX) loop_update($this, {left:e.clientX}); return false; }; doc.onmouseup = function(e){ //清除事件 doc.onmousemove = null; doc.onmouseup = null; //hide_loop_time(); //$this.find('.bilibili-player-video-progress-detail-container').show(); $this.css('z-index', 'auto'); $('#hhh_loop_wrap')[0].hhh_loop_time_id = $this[0].id; return false; } } }); } //增加当前在线视频预览 function run_online_preview(open=ON){ if(open !== ON) return false; if($('#hhh_online_list_style').length === 1) return true; function hide(parent, selector, show_or_hide){ var o = $(); if($.isArray(selector) === true){ $.each(selector, function(i,v){ o = o.add(v); }) }else{ o = $(selector); } //显示及渐隐效果(抄bilibili^^) clearTimeout(parent.showHintTimer), o.stop().show(), parent.showHintTimer = window.setTimeout((function() { o.animate({ opacity: 0 }, 600, (function() { $(this).hide() })) } ), 0) } function show(parent, selector, show_or_hide){ var o = $(); if($.isArray(selector) === true){ $.each(selector, function(i,v){ o = o.add(v); }) }else{ o = $(selector); } //显示及渐隐效果(抄bilibili^^) clearTimeout(parent.showHintTimer), parent.showHintTimer = window.setTimeout((function() { o.stop().show(), o.animate({ opacity: 1 }, 600, (function() { $(this).show() })) } ), 300) } if($('#hhh_online_list_style').length<1){ $('.online-list').append(``); } var bvs = {}, box_width = 0, bg_img_row = 0, bg_y_block_len = 0, bg_img_totle = 0; $('.online-list>.ebox>a>div').mouseenter(function(e){ this.state = 'enter'; let $lazy_img = $(this); let bvid = $lazy_img.parent().attr('href').match(/BV\w+$/); bvid = bvid && bvid[0]; this.bvid = bvid; if(bvs[bvid] === undefined){ bvs[bvid] = $lazy_img.parent().attr('title'); $.getJSON("https://api.bilibili.com/x/player/videoshot", { bvid: bvid, index: "1" }, function(json){ let bg_img = json.data.image[0]; box_width = $lazy_img.width(); bg_img_totle = json.data.index.length > (json.data.img_x_len*json.data.img_x_len)? (json.data.img_x_len*json.data.img_x_len): json.data.index.length; bg_img_row = bg_img_totle < json.data.img_x_len? bg_img_totle: json.data.img_x_len; bg_y_block_len = json.data.img_y_size/json.data.img_x_size*box_width; bvs[bvid] = {box_width:box_width, bg_img_totle:bg_img_totle, bg_img_row:bg_img_row, bg_y_block_len:bg_y_block_len}; //log(bg_img_row+' - '+bg_img_totle+' - '+bg_y_block_len); //log(bvs[bvid]); $lazy_img.css({'overflow':'hidden', 'position':'relative'}); $lazy_img.append(`
`); /*硬编码,随时调整*/ if($lazy_img[0].state === 'enter') show($lazy_img[0], [$lazy_img.find('.hhh-preview-bg'), $lazy_img.find('.hhh-preview-wrapper')], 'show'); }); }else{ show($lazy_img[0], [$lazy_img.find('.hhh-preview-bg'), $lazy_img.find('.hhh-preview-wrapper')], 'show'); } }).mouseleave(function(){ this.state = 'leave'; hide(this, [$(this).find('.hhh-preview-bg'), $(this).find('.hhh-preview-wrapper')], 'hide'); }).mousemove(function(e){ //log(this.bvid); let obv = bvs[this.bvid]; let amounts = Math.floor(e.offsetX/(obv.box_width/obv.bg_img_totle)); let x = Math.floor(amounts % obv.bg_img_row) * -obv.box_width; x = Math.min(Math.max(x,-obv.box_width*(obv.bg_img_totle-1)),0); let y = Math.floor(amounts/obv.bg_img_row) * -obv.bg_y_block_len; //log('---'+box_width+' - '+bg_img_totle); //log(e.offsetX+' - '+amounts+' - '+x+' - '+y+' - '+bg_y_block_len+' - '+Math.floor(amounts/bg_img_row)); $(this).find('.hhh-preview-bg').css({'background-position':`${x}px ${y}px`}); let w = Math.round(e.offsetX/obv.box_width*100); $(this).find('.hhh-preview-progress-bar').css({'width':`${w}%`}); }) } //动态首页直接显示隐藏content function run_content(open=ON){ if(open !== ON){ return false; //Test if($('[hhh_has_content]').length){ let $major = $('.bili-dyn-content__orig__major'); let $content = $major.find('.bili-dyn-card-video__desc'); let $expand = $major.find('.bili-dyn-card-video_expand___hhh'); $content.removeClass('bili-dyn-card-video__desc___new_expand') .removeClass('bili-dyn-card-video__desc___new_collapse'); $expand.add($content).off('click.hhh_content'); $expand.remove('.bili-dyn-card-video_expand___hhh'); $major.find('.bili-dyn-card-video').removeClass('bili-dyn-card-video___new'); $major.find('.bili-dyn-card-video__header').removeClass('bili-dyn-card-video__header___new'); $major.find('.bili-dyn-card-video__body').removeClass('bili-dyn-card-video__body___new'); $('[hhh_has_content]').removeAttr('hhh_has_content'); } return false; } if($('[hhh_has_content]').length < $('.bili-dyn-content__orig__major').length){ $('.bili-dyn-content__orig__major').each(function(){ let $major = $(this); let $hhh_has_content = $major.find('[hhh_has_content]'); if($hhh_has_content.length >= 1) return true; //$major.attr('hhh_has_content', true); //log($hhh_has_content.length) //隐藏弹幕,弹幕意义不大还遮挡窗口视频 $major.find($('.dyn-video-preview__danmaku')).hide() //隐藏"投稿视频",同样意义不大还遮挡窗口视频 if($major.find($('.bili-dyn-card-video__badge:contains(投稿视频)')).length === 1){ log($major.text()) $major.find($('.bili-dyn-card-video__badge:contains(投稿视频)')).hide() } //新版“bili-dyn-card-video__stat”去掉position: absolute,使用默认值 $major.find('.bili-dyn-card-video__stat').css({'position': 'static'}) //显示隐藏标题 let $title = $major.find('.bili-dyn-card-video__title'); let padding_bottom = '24px'; //根据标题行数确定 .bili-dyn-card-video__stat 位置 if($title.length >= 1){ let font_size = $title.css('font-size'); let line_height = +font_size.match(/\d+/)[0] + 5; //估算 let h = $title.removeClass('bili-dyn-card-video__title').css({'font-size':font_size}).height(); $title[0].style = ""; //log($title.text()+': '+line_height+' - '+h) $title.addClass('bili-dyn-card-video__title'); if(h > line_height*2){ $title.attr('title', $title.text()) } if(h > line_height){ padding_bottom = '10px'; } } //显示隐藏content let $content = $major.find('.bili-dyn-card-video__desc'); if($content.length >= 1){ //console.log($content.text().slice(0,10)+ ' : ' +$content.height()); let line_height = $content.css('line-height'); //log($content.height()); let font_size = $content.css('font-size'); let h = $content.removeClass('bili-dyn-card-video__desc').css({'line-height':line_height, 'font-size':font_size}).height(); $content[0].style = ""; line_height = line_height.match(/\d+/)[0]; //log('---line_h:'+line_height+' h: '+h+' title: '+$major.find('.title').text()); $content.addClass('bili-dyn-card-video__desc'); $content.attr('hhh_has_content', true); if(h > line_height*2){ // $major.find('.bili-dyn-card-video').height('auto'); // $major.find('.bili-dyn-card-video__body').css('display', 'block'); // $major.find('.bili-dyn-card-video__header').height(127); if($('#hhh_content_style').length<1){ $major.append(``); } $major.find('.bili-dyn-card-video').addClass('bili-dyn-card-video___new'); $major.find('.bili-dyn-card-video__header').addClass('bili-dyn-card-video__header___new'); $major.find('.bili-dyn-card-video__body').addClass('bili-dyn-card-video__body___new'); $content.addClass('bili-dyn-card-video__desc___new_collapse'); // $content.css({'padding-top':'8px'}); // console.log(line_height); // $content.css({'height': `${line_height}px`}); //let $expand = $(`
展开
`).insertAfter($content); let $expand = $(`
展开
`).insertAfter($content); $expand.add($content).off('click.hhh_content'); $expand.add($content).on('click.hhh_content', function(){ //$expand.add($content).click(function(){ if($expand.text() === '展开'){ //console.log('=='+$content.height()+'==='+$content.css('line-height').match(/\d+/)[0]); //$content.css({height:'auto', '-webkit-box-orient':'inline-axis'}); $content.removeClass('bili-dyn-card-video__desc___new_collapse').addClass('bili-dyn-card-video__desc___new_expand'); $expand.css({'padding-bottom': '0px'}); $expand.text('收起'); }else{ //$content.height(line_height); //$content.css({'-webkit-box-orient':'vertical'}); $content.removeClass('bili-dyn-card-video__desc___new_expand').addClass('bili-dyn-card-video__desc___new_collapse'); $expand.css({'padding-bottom': $expand.attr('padding-bottom')}); $expand.text('展开'); } return false }); } } }) } } //添加首页轮播 prev next slide (抄网易) function run_add_carousel_slide(){ if(bb_type !== BILI_3_X_VIDEO) return false waitForTrue(()=> $('.rcmd-box-wrap').length === 1, () => { if($('.focus-carousel_hhh').length > 0) return false; let style = ``; let $carousel = $('.focus-carousel'); $carousel.append(style); $carousel.addClass('focus-carousel_hhh'); let $prev = $(`
`).appendTo($carousel); let $next = $(`
`).appendTo($carousel); //轮播效果 $prev.on('click', function(){ if($carousel.find('.trigger span.on').index() === $carousel.find('.trigger span:first').index()) $carousel.find('.trigger span:last').click(); else $carousel.find('.trigger span.on').prev().click(); }); $next.on('click', function(){ if($carousel.find('.trigger span.on').index() === $carousel.find('.trigger span:last').index()) $carousel.find('.trigger span:first').click(); else $carousel.find('.trigger span.on').next().click(); }); }); } //新版首页添加观看列表 function run_add_online(){ let $box = $('.recommend-container__2-line'); if($box.length !== 1) return; let $last_card = $box.children('.bili-video-card:last'); let left = $last_card.position().left; let width = $last_card.outerWidth(); let $online_wrap = $('.bili-grid.short-margin.grid-anchor:last'); $online_wrap.append(`
`); $('#hhh_online').append($('#hhh_btn_num>button').clone().empty().css({width:`${width}px`, 'border-radius':'4px', background:'#f4f4f4', display:'flex', border:'1px solid #e7e7e7'}) .append(`观看列表`)); $('#hhh_online a').hover(function(){$(this).css('color','#00a1d6')}, function(){$(this).css('color','#212121')}); } //去掉首页插件提示 function run_adblock_remove(){ if(ver === BILI_3_X_VIDEO_V1){ $('.adblock-tips').remove() } } //记忆首页推荐列表 function run_save_recommend_list(open=ON){ if(open !== ON) return false; if ($('.rcmd-box-wrap').length === 1) save_recommend_list_old(); else if($('.bili-grid.short-margin.grid-anchor:first').length === 1) save_recommend_list_new(); //else err('记忆首页推荐列表错误 - 找不到dom'); } function save_recommend_list_old(){ if($('#hhh_btn_num').length !== 0) return true; let $box_wrap = $('.rcmd-box-wrap'); let $box = $('.rcmd-box'); let box_data = []; let $cbtn = $('.change-btn'); let h = $cbtn.outerHeight(); let w = $cbtn.outerWidth(); let curr_box_num = 0; //取得过滤器 let filter = get_value('hhh_rcmd_filter', {}) filter_list = [] fresh_rcmd_list = [] load_card_box() //list-box left $('#elevator').css({left:`calc(50% + ${w+3+20}px)`}); //calc(50% + 3px); let $empty_cbtn = $cbtn.clone().empty(); //num $box_wrap.append($empty_cbtn.clone().attr('id','hhh_btn_num').css({cursor:'default'}) .css({height:'21px', 'line-height':'21px', top:`${h+2}px`}) .css({padding:'0px', background:'#e2e2e2'}).append('1/1')) //left let cbtn_h = $('#hhh_btn_num').outerHeight() + 3; $box_wrap.append($empty_cbtn.clone().attr('id','hhh_btn_zuo').css({height:'auto', top:`${h+1+cbtn_h}px`}) .css({'border-bottom':'0px', 'border-bottom-left-radius':'0px', 'border-bottom-right-radius':'0px'}) .append('')) //right cbtn_h += $('#hhh_btn_zuo').outerHeight() - 1; $box_wrap.append($empty_cbtn.clone().attr('id','hhh_btn_you').css({height:'auto', top:`${h+1+cbtn_h}px`}) .css({'border-top':'0px', 'border-top-left-radius':'0px', 'border-top-right-radius':'0px'}) .append('')) //button cbtn_h += $('#hhh_btn_you').outerHeight() + 10; $box_wrap.append($empty_cbtn.clone().attr('id','hhh_btn_remove_filter').css({height:'auto', top:`${h+1+cbtn_h}px`}) .css({padding: '0px', border: '0px'}) .append('')) .find('button').css({padding: '2px'}) function card_filter($card, filter){ let BV = $card.find('a')[0].href.match(/BV\w+/)[0] //取得BV号 const [对钩, 叉叉] = ['bili-icon_caozuo_yitianjia', 'bili-icon_sousuo_yichu'] let filter_value = filter[BV] === ON ? 对钩 : 叉叉 //add filter $card.find('.hhh_filter').remove().end() .prepend( $card.find('.watch-later-video:first').clone() .addClass('hhh_filter').css({top:'8px', 'background-image':'none'}) .find('.wl-tips').css({left:'-4px', display:'none'}).end() .prepend($(``).clone()) ) if(filter_value === 对钩){ $card.find('.info-box').css({'background-image':'none', 'background-color':'black'}) $card.find('.info-box img').css({'opacity':0.2}) }else{ $card.find('.info-box').css({'background-image':'', 'background-color':''}) $card.find('.info-box img').css({'opacity':''}) } //event $card.find('.hhh_filter').off('mouseover.hhh_filter').on('mouseover.hhh_filter', function(){ let $bilifont = $(this).find('i') let $wl_tips = $(this).find('.wl-tips') if($bilifont.hasClass(`bilifont ${叉叉}`)){ $wl_tips.text('点击屏蔽') $wl_tips.css({left:`${($bilifont.width() - $wl_tips.width())/2}px`}) }else if($bilifont.hasClass(`bilifont ${对钩}`)){ $wl_tips.text('点击恢复') $wl_tips.css({left:`${($bilifont.width() - $wl_tips.width())/2}px`}) } $wl_tips.removeClass('van-watchlater-move-enter-active').addClass('van-watchlater-move-enter-active').show() }).off('mouseout.hhh_filter').on('mouseout.hhh_filter', function(){ $(this).find('.wl-tips').removeClass('van-watchlater-move-enter-active').hide() }).off('click.hhh_filter').on('click.hhh_filter', function(){ let $bilifont = $(this).find('i') let $wl_tips = $(this).find('.wl-tips') let $info_box = $(this).parent().find('.info-box') let BV = $info_box.find('a')[0].href.match(/BV\w+/)[0] //取得BV号 if($bilifont.hasClass(`bilifont ${叉叉}`)){ filter[BV] = ON set_value('hhh_rcmd_filter', filter) //log(get_value('hhh_rcmd_filter')) $wl_tips.text('已屏蔽') $wl_tips.css({left:`${($bilifont.width() - $wl_tips.width())/2}px`}) $bilifont.removeClass(`bilifont ${叉叉}`).addClass(`bilifont ${对钩}`) $info_box.css({'background-image':'none', 'background-color':'black'}) $info_box.find('img').css({'opacity':0.2}) }else if($bilifont.hasClass(`bilifont ${对钩}`)){ delete filter[BV] set_value('hhh_rcmd_filter', filter) //log(get_value('hhh_rcmd_filter')) $wl_tips.text('已恢复') $wl_tips.css({left:`${($bilifont.width() - $wl_tips.width())/2}px`}) $bilifont.removeClass(`bilifont ${对钩}`).addClass(`bilifont ${叉叉}`) $info_box.css({'background-image':'', 'background-color':''}) $info_box.find('img').css({'opacity':''}) } }) } function cbtn_event($cbtn){ $cbtn.click(function(){ //激活动画 let $i = $(this).find('i'); $i.addClass('active'); setTimeout(function(){ $i.removeClass('active') },500); //恢复前一个list curr_box_num = this.id === 'hhh_btn_zuo'? Math.max(curr_box_num-1, 0): Math.min(curr_box_num+1, box_data.length-1); $('#hhh_btn_num').text(`${curr_box_num+1}/${box_data.length}`); $box.find('.video-card-reco').each(function(i){ $(this).empty().append($(box_data[curr_box_num]).find('.video-card-reco:eq('+i+')>div').clone()); card_filter($(this), filter) }) }) } cbtn_event($('#hhh_btn_zuo')); //left cbtn_event($('#hhh_btn_you')); //right //refresh $cbtn.click((e)=>{ $box_wrap[0].addEventListener('DOMSubtreeModified', function fn_(e) { let eve_this = this; if(typeof e.target.className === 'string' && e.target.className === $box.attr('class')){ //最后一个class if($box.find('.video-card-reco').length === 10){ eve_this.removeEventListener('DOMSubtreeModified', fn_); load_card_box() } } }) }) //重置 $box_wrap.find('#hhh_btn_remove_filter').click((e)=>{ let box = new hhh_box() box.create({target: $('#hhh_btn_remove_filter button'), color: '#00b5e5', text: '是否清空屏蔽列表', gap: 8, enter_fn: ()=>{ const json_filter = JSON.parse(localStorage.getItem('hhh_rcmd_filter')) log(Object.keys(json_filter).length) log(json_filter) localStorage.removeItem('hhh_rcmd_filter') filter = {} $box.find('.video-card-reco').each(function(i){ card_filter($(this), filter) }) } }) }).mouseenter(function(){ const json_filter = JSON.parse(localStorage.getItem('hhh_rcmd_filter')) $(this).attr('title', `已屏蔽${Object.keys(json_filter).length}个`) }) function load_card_box(){ //hhh_rcmd_filter = {BVxxx: ON, BVxxx2: ON} //使用过滤器(BV号判断)过滤card数据,过滤后的数据 $box.find('.video-card-reco').each(function(i){ let BV = $(this).find('a')[0].href.match(/BV\w+/)[0] //取得BV号 if(filter[BV] === ON){ //对比BV号过滤 //log(BV) filter_list.push(i) } //filter[BV] = ON }) //set_value('hhh_rcmd_filter', filter) // filter_list = [] // filter_list.push(Math.floor(Math.random() * (9 - 0 + 1) ) + 0) // filter_list.push(Math.floor(Math.random() * (9 - 0 + 1) ) + 0) //log('filter_list', filter_list) function get_fresh_rcmd(){ let items $.ajax({ url: "https://api.bilibili.com/x/web-interface/index/top/rcmd?fresh_type=0&fresh_idx=0&fresh_idx_1h=0&homepage_ver=0", type: "GET", xhrFields: { withCredentials: true }, crossDomain: true, async: false, timeout: 1, success: function (data) { items = data.data.item //log(data.data); } }); return items } function set_card($card, item){ $card.find('.info-box a')[0].href = item.uri $card.find('.info-box a img')[0].src = item.pic.split(/https*:/)?.[1] + '@412w_232h_1c.webp' $card.find('.info-box a img')[0].alt = item.title $card.find('.info-box a .info .title')[0].title = item.title $card.find('.info-box a .info .title').text(item.title) $card.find('.info-box a .info .up').contents()[1].nodeValue = item.owner.name $card.find('.info-box a .info .play').text(item.stat.view >= 10000 ? (item.stat.view/10000).toFixed(1)+'万播放' : item.stat.view+'播放') } //如果发现需过滤BV,读取新rcmd填充 if(filter_list.length !== 0){ //ajax读取rcmd填充 while(filter_list.length > fresh_rcmd_list.length){ let new_rcmd_list = get_fresh_rcmd() new_rcmd_list = new_rcmd_list.filter((v)=>{ return filter[v.bvid] !== ON }) fresh_rcmd_list.push(...new_rcmd_list) } let $card_clone = $box.find('.video-card-reco:eq(0)').clone() for(let i of filter_list){ let $card = $('.rcmd-box .video-card-reco').eq(i) $card.empty().append( $card_clone.find('>*').clone() ) // 去掉watch-later-video van-watchlater black [click],由于稍后再看保存旧card内容,不去掉的话会错乱 let item = fresh_rcmd_list.shift() set_card($card, item) } } box_data.push($box.clone()); curr_box_num = box_data.length - 1; $('#hhh_btn_num').text(`${box_data.length}/${box_data.length}`); if(box_data.length >= 10){ $('#hhh_btn_num').css({width:'37px', right:'-41px'}) } $box.find('.video-card-reco').each(function(i){ card_filter($(this), filter) }) } } function save_recommend_list_new(){ if($('#hhh_btn_num').length !== 0) return true; let $box_wrap = $('.bili-grid.short-margin.grid-anchor:first'); let $box = $('.recommend-container__2-line'); let box_data = [$box.clone()]; let $rbtn = $('.roll-btn-wrap'); let $empty_rbtn = $rbtn.clone().empty(); let h = $rbtn.outerHeight(); let w = $rbtn.outerWidth(); let $left = $('.buttons.not-gray').find('button:first'); let $right = $('.buttons.not-gray').find('button:last'); let is_change_break = null; let curr_box_num = 0; //palette-button left waitForTrue(()=> $('.palette-button-wrap').length === 1, () => { $('.palette-button-wrap').css({left:`calc(100% + ${w+10}px)`}); //calc(100% + 0px); }); //num $box_wrap.append($rbtn.clone().attr('id','hhh_btn_num').css({top:`${h+2}px`}) .find('svg').remove().end() .find('span').remove().end() .find('button').css({cursor:'default', background:'#e2e2e2', 'align-items':'center', 'border-radius':'8px'}).removeClass('primary-btn').append('1/1').end()); //left let cbtn_h = $('#hhh_btn_num').outerHeight() + 3; $box_wrap.append($empty_rbtn.clone().attr('id','hhh_btn_zuo').css({top:`${h+1+cbtn_h}px`}).append($left) .find('button').addClass('primary-btn roll-btn').css({'border-bottom':'0px', 'border-bottom-left-radius':'0px', 'border-bottom-right-radius':'0px'}).end() .find('svg').css({'margin-bottom':'1px'}).end()); //right cbtn_h += $('#hhh_btn_zuo').outerHeight() - 1; $box_wrap.append($empty_rbtn.clone().attr('id','hhh_btn_you').css({top:`${h+1+cbtn_h}px`}).append($right) .find('button').addClass('primary-btn roll-btn').css({'border-top':'0px', 'border-top-left-radius':'0px', 'border-top-right-radius':'0px'}).end() .find('svg').css({'margin-bottom':'1px'}).end()); //left right - event function rbtn_event($rbtn){ $rbtn.click(function(){ //恢复前一个list curr_box_num = this.id === 'hhh_btn_zuo'? Math.max(curr_box_num-1, 0): Math.min(curr_box_num+1, box_data.length-1); $('#hhh_btn_num>button').text(`${curr_box_num+1}/${box_data.length}`); $box.children('.bili-video-card').each(function(i){ //log($(box_data[curr_box_num]).children(`.bili-video-card:eq(${i})`).attr('data-report')); $(this).empty().append($(box_data[curr_box_num]).children(`.bili-video-card:eq(${i})`).clone()); }) }) } rbtn_event($('#hhh_btn_zuo')); //left rbtn_event($('#hhh_btn_you')); //right //refresh - event $rbtn.click(()=>{ $box_wrap[0].addEventListener('DOMSubtreeModified', function fn_(e) { let eve_this = this; if(typeof e.target.className === 'string' && e.target.className === 'bili-video-card__image--wrap'){ clearTimeout(is_change_break); is_change_break = setTimeout(function() { //log("推荐页面刷新完毕"); eve_this.removeEventListener('DOMSubtreeModified', fn_); //eve_this.removeEventListener('DOMSubtreeModified', arguments.callee); box_data.push($box.clone()); curr_box_num = box_data.length - 1; $('#hhh_btn_num>button').text(`${box_data.length}/${box_data.length}`); //log(box_data.length) }, 50); } }) }); } //新窗口打开自动连播列表视频 function run_rec_list_newtab(open){ $('.rec-list .video-page-card').each(function(e){ if(open === ON){ let $a = $(this).find('.info a:first').attr('target',"_blank"); let $span = $(this).find('.info span'); $span.clone().prependTo($a).attr('hhh_new', true); $span.hide(); }else{ let $a = $(this).find('.info a:first').attr('target',""); $(this).find('.info span[hhh_new]').remove(); $(this).find('.info span').show(); } }) } //全屏时显示投币、收藏对话框等 function set_dialog(type, classname) { //log('set_dialog:'+classname) if(type === 'coin'){ //投币对话框 //log('coin') let is_Dlg_exist = classname.includes('teleport') === true //harmony-font header-v2 win webscreen-fix //let is_coin_over = classname.includes('header-v2 win webscreen-fix') === true if(is_Dlg_exist === true){ if($('.coin-operated-m-exp').length <= 0){ //log('off') $(document).off('keydown.hhh_coin') return false } //log('--set_dialog 1--') // TODO: 全屏时显示 // $('.bili-dialog-m').appendTo('.bpx-player-container'); // $('.bili-dialog-m').find('.coin-bottom').css('line-height', 1.5); if($(bb['biliDlgM']).length === 1){ //网页全屏时显示 let z_index = +$('#bilibili-player').css('z-index'); !isNaN(z_index) && $(bb['biliDlgM']).css('z-index', z_index+1); !isNaN(z_index) && $('.bili-msg').css('z-index', z_index+1); //硬币不足 提示 //键盘事件 $(document).off('keydown.hhh_coin') $(document).on('keydown.hhh_coin', function(e){ if(e.keyCode === keycode['left']){ $('.left-con').click(); }else if(e.keyCode === keycode['right']){ $('.right-con').click(); }else if(e.keyCode === keycode['Enter']){ let c_num = $('.mc-box.on').text() $(bb['coinDlgOkBtn']).click() //$(bb['coinDlgCloseBtn']).click() showHint(document, '#hhh_wordsHint', `投币成功 ${$('.video-coin-info').text()}+${c_num}`, 2e3) }else if(e.keyCode === keycode['Esc']){ $(bb['coinDlgCloseBtn']).click() } return false }) } }else if(classname === 'hhh_test_coin'){ //log('--set_dialog 2--'); if($(bb['coin']).filter('[title="对本稿件的投币枚数已用完"]').length > 0 && $('.van-message').length > 0){ showHint(document, '#hhh_wordsHint', '对本稿件的投币枚数已用完'); } } } } //添加合集列表关键字过滤 function list_filter2(){ let $list_card = $('.list-box li').add('.video-section-list .video-episode-card') $list_card.show() if($list_card.length < 20) return //action-list-inner const [MULTI_P, VIDEO_LIST, ACTION_LIST] = [1, 2, 3] const LIST_TYPE = $('.list-box').length > 0 ? MULTI_P : //视频选集 $('.video-section-list').length > 0 ? VIDEO_LIST : undefined //合集 if(LIST_TYPE === undefined) return let list_input_div = `
`; if($('#hhh_list_input').length === 0){ //插入 $('#multi_page, .base-video-sections-v1, .video-sections-v1').before($(list_input_div)) $('#hhh_list_input').off('input.hhh_list') $('#hhh_list_input').on('input.hhh_list',function(){ let val = $(this).val(); $list_card.each(function(i){ if(!val) { $(this).show() return; } if($(this).text().toUpperCase().indexOf(val.toUpperCase()) !== -1) { $(this).show() }else{ $(this).hide() } }) //回到首行位置 $('.cur-list, .video-sections-content-list').scrollTop(0) //更新过滤计数 if(!!val) $('#hhh_list_input').next().text(`过滤${$('.list-box li:visible').add('.video-section-list .video-episode-card:visible').length}个`) else $('#hhh_list_input').next().text('') //console.log($('.list-box li:visible').add('.video-section-list .video-episode-card:visible').length, val) }) } } //添加合集列表关键字过滤 function list_filter(){ let $list_card = $('.list-box li').add('.video-section-list .video-episode-card').add('.action-list-inner .action-list-item-wrap') $list_card.show() if($list_card.length < 1) return //action-list-inner const [MULTI_P, VIDEO_LIST, ACTION_LIST] = [1, 2, 3] const LIST_TYPE = $('.list-box').length > 0 ? MULTI_P : //视频选集 $('.video-section-list').length > 0 ? VIDEO_LIST : //合集 $('.action-list-inner').length > 0 ? ACTION_LIST : undefined //收藏 if(LIST_TYPE === undefined) return let list_input_div = `
`; if($('#hhh_list_input').length === 0){ //插入 $('#multi_page, .base-video-sections-v1, .video-sections-v1, .action-list-container').before($(list_input_div)) $('#hhh_list_input').off('input.hhh_list') $('#hhh_list_input').on('input.hhh_list',function(){ let val = $(this).val() let $list_card = $('.list-box li').add('.video-section-list .video-episode-card').add('.action-list-inner .action-list-item-wrap') $list_card.each(function(i){ if(!val) { $(this).show() return } log(i,$(this).text().toUpperCase(), '-----------------',val.toUpperCase()) if($(this).text().toUpperCase().indexOf(val.toUpperCase()) !== -1) { $(this).show() }else{ $(this).hide() } }) //回到首行位置 $('.cur-list, .video-sections-content-list, #playlist-video-action-list').scrollTop(0) //更新过滤计数 if(!!val) $('#hhh_list_input').next().text(`过滤${$('.list-box li:visible').add('.video-section-list .video-episode-card:visible').add('.action-list-inner .action-list-item-wrap:visible').length}个`) else $('#hhh_list_input').next().text('') //console.log($('.list-box li:visible').add('.video-section-list .video-episode-card:visible').length, val) }) } } //添加到收藏夹关键字过滤(video wrap) collection-m-exp coin-operated-m-exp function collection_filter(open=ON){ if($('.collection-m-exp').length <= 0) return false $(document).off('keydown.hhh_coin') if(open !== ON){ $('.bili-dialog-m .title')[0].style['padding-left'] = ''; $('#hhh_collection_input').off('input.hhh_collection'); $('#hhh_collection_input').parent().remove(); return false; } let collection_div = `
`; if($('#hhh_collection_input').length === 0){ $(collection_div).prependTo($('.bili-dialog-m .title').css('padding-left', '150px')) // log('collection_filter') // $('#hhh_collection_input')[0].dispatchEvent( new MouseEvent('mousedown') ) // $('.bpx-player-video-wrap')[0].dispatchEvent( new MouseEvent('click') ) setTimeout(() => { //$('.group-list').find(':contains(排行榜)').parent()[0].dispatchEvent( new MouseEvent('click') ) $('#hhh_collection_input').focus() //$('.group-list').find(':contains(排行榜)').parent().click() }, 0); $('#hhh_collection_input').off('input.hhh_collection') $('#hhh_collection_input').on('input.hhh_collection',function(){ let val = $(this).val(); $('.content li').each(function(i){ if(!val) { $(this).css('display','block'); return; } if($(this).find('.fav-title').text().toUpperCase().indexOf(val.toUpperCase()) !== -1) { $(this).css('display','block'); }else{ $(this).css('display','none'); } }) }) if(is_fullscreen() === true){ let z_index = +$('#bilibili-player').css('z-index') $(bb['biliDlgM']).css('z-index', z_index+1) } } } //添加到收藏夹关键字过滤 collection-m-exp coin-operated-m-exp function collection_filter2(open=ON){ if(open !== ON){ $('#hhh_collection_input').off('input.hhh_collection') $('#hhh_collection_input').parent().remove() return false } let collection_div = `
`; if($('#hhh_collection_input').length === 0){ let $fav_list = $('#fav-createdList-container .fav-list') let $fav_item = $('#fav-createdList-container .fav-item.cur') $fav_list.before($(collection_div)) setTimeout(() => { //$('.group-list').find(':contains(排行榜)').parent()[0].dispatchEvent( new MouseEvent('click') ) $('#hhh_collection_input').focus() //$('.group-list').find(':contains(排行榜)').parent().click() }, 0); $('#hhh_collection_input').off('input.hhh_collection') $('#hhh_collection_input').on('input.hhh_collection',function(){ let val = $(this).val() $fav_list.find($('.fav-item')).each(function(i){ if(!val) { $(this).css('display','block'); return; } if($(this).find('.text').text().toUpperCase().indexOf(val.toUpperCase()) !== -1) { $(this).css('display','block'); }else{ $(this).css('display','none'); } }) }) } return true } //调整收藏夹长度 & 添加收藏夹关键字过滤 function run_fav(open){ if(open === ON){ let $fav_list_li = $('#fav-createdList-container .fav-list li:first') let length = $fav_list_li.height() $('#fav-createdList-container').css('max-height', `${length*15}px`) collection_filter2() }else{ //let max_height = parseInt($('#fav-createdList-container').css('max-height')); //$('#fav-createdList-container').css('max-height', `${max_height/2}px`); } } //未使用 function run_history(){ $.get("https://api.bilibili.com/x/web-goblin/history/search", { pn:"1", keyword: "12", business: "all" }, function(json){ console.log(json) }); } //自动(合集/选集)连播/循环 function auto_list_play(open=ON){ //视频选集/合集(选集是单个BV号,合集是多个BV号) const [MULTI_P, VIDEO_LIST] = [1,2] const LIST_TYPE = $('.list-box').length > 0 ? MULTI_P : //视频选集 $('.video-section-list').length > 0 ? VIDEO_LIST : undefined //合集 if(LIST_TYPE === undefined) return h5Player = geth5Player() let $switch_button = $('.next-button:contains(自动连播)') h5Player.removeEventListener('ended', $switch_button[0].hhh_fn_ended) if(open === OFF) return //视频结束时调用 const fn_ended = function(e){ const list_total = $('.video-section-list .video-episode-card').add($('.list-box li')).length-1 const curr_list = $('.list-box li.on').add($('.video-episode-card__info-playing').parent()).index() if(list_total === curr_list){ //最后一个 $('.bpx-player-ending-related-item-cancel').click() //取消系统连播 if(config.getCheckboxSettingStatus('circlePlay') === ON){ //循环 $('.list-box li:first .clickitem').add($('.video-episode-card:first')).click() } } } h5Player.addEventListener('ended', fn_ended) $switch_button[0].hhh_fn_ended = fn_ended } //订阅合集列表展开 function expand_list(open=ON, columns=10){ if(open !== ON) return false let $first_right = $('.first-line-right:first') if ($first_right.length <= 0) { /*log('不是订阅合集');*/ return false } //第一次执行 let old_height = '152px'; let old_max_height = '152px'; if($('#hhh_expand_list').length !== 1) { if($first_right.parent()[0].tagName.toLowerCase() !== 'span'){ $('').appendTo($first_right.parent()).append($first_right) } $first_right.clone().appendTo($first_right.parent()) .attr('id', 'hhh_expand_list') .css({'margin-top': '5px'}) .find('.txt').text(`展开列表`) old_height = $('.video-sections-content-list').css('height'); old_max_height = $('.video-sections-content-list').css('max-height'); } //点击列表展开 let $vcard = $('.video-episode-card:last'); let vcard_height = $vcard.outerHeight() + parseInt($vcard.css('margin-top')) + parseInt($vcard.css('margin-bottom')); let $switch_button = $('#hhh_expand_list .switch-button'); if($switch_button.hasClass('on') === true) $switch_button.toggleClass('on'); //初始不展开状况 $('#hhh_expand_list .next-button').off('click.hhh_expand_list'); $('#hhh_expand_list .next-button').on('click.hhh_expand_list', function(){ $switch_button.toggleClass('on'); if($switch_button.hasClass('on') === true){ let expand_height = 0; $('.video-sections-item').each(function(){ expand_height += $(this).height() }); //当前展开高度 let limit_height = vcard_height * columns; //限高 expand_height = expand_height > limit_height ? limit_height : 'auto'; $('.video-sections-content-list').css({'height': expand_height, 'max-height': 'none'}); //展开columns个的高度 } else { $('.video-sections-content-list').css({'height': old_height, 'max-height': old_max_height}); } }); $('#hhh_expand_list .next-button').click(); //默认展开 } //简单的对话框 class hhh_box{ $box_container = null option = { id: 'hhh_box', target: null, tip_target: null, enter_fn: ()=>{}, cancel_fn: ()=>{}, color: 'white', gap: 2, z_index: 1, } //计算坐标 calc_xy(t){ let crects = t.target[0].getBoundingClientRect(), trects = t.box_target[0].getBoundingClientRect(), box_height = trects.height, box_width = trects.width, width = crects.width, top = crects.top - box_height, left = crects.left + (width - box_width) / 2; //log(crects, trects) // + document.scrollingElement.scrollTop return {x:left, y:top}; } create(option_) { if(this.$box_container !== null) { console.error('已创建对话框'); return 0 } //add box let box = `
确定
取消
` let o = {...this.option, ...option_} this.$box_container = $(box).appendTo($('body')) let $box = this.$box_container.find('#hhh_box') o.box_target = $box //add css let csstext = ` .hhh-button-area { display: flex; min-width: 40px; transform: scale(0.8); color: #fff; border: 1px solid hsla(0,0%,100%,.2); justify-content: center; width: 100%; height: 100%; border-radius: 2px; box-sizing: border-box; transition: all .2s } .hhh-button-area:hover { --bpx-fn-color: #00a1d6; color: var(--bpx-fn-color,#00a1d6); border-color: var(--bpx-fn-color,#00a1d6) } .hhh-button { width: 40px; height: 22px; line-height: 22px; margin-top: 9px; display: inline-flex; vertical-align: middle; cursor: pointer } ` let style_id = 'hhh_box_style' $(`#${style_id}`).remove(); if($(`#${style_id}`).length <= 0){ this.$box_container.prepend(``); } $box.find('#hhh_content').css({color: o.color}).text(o.text) let xy = this.calc_xy(o) $box.css({left: xy.x, top: xy.y - o.gap, 'z-index': o.z_index}) //event $box.find('#hhh_enter').click(()=>{ o.enter_fn(); this.destroy() }) $box.find('#hhh_cancel').click(()=>{ o.cancel_fn(); this.destroy() }) } destroy() { this.$box_container.empty() this.$box_container.remove() this.$box_container = null } } //时间to秒 function timestr_to_second(d){ d = d?.trim()?.split(/:/)?.reverse() return d?.reduce((t,v,i)=>t+parseInt(v)*(60**i),0) } //格式化数字(播放量等),感谢ChatGPT function formatPlayCount(playCount) { if (playCount >= 10000 && playCount < 100000000) { // 如果播放量在一万以上,但不到一亿,以万为单位显示,保留一位小数 return (playCount / 10000).toFixed(1) + '万'; } else if (playCount >= 100000000) { // 如果播放量在一亿以上,以亿为单位显示,保留一位小数 return (playCount / 100000000).toFixed(1) + '亿'; } else { // 如果播放量在一万以下,直接显示整数 return Math.round(playCount).toString(); } } //观看进度 function memory_multipart_progress(open, [status]){ // log('=====memory_multipart_progress=====') // (function(){ // if(!window.localStorage) { // console.log('浏览器不支持localStorage'); // } // var size = 0; // for(item in window.localStorage) { // if(window.localStorage.hasOwnProperty(item)) { // if(item === 'hhh_checkboxes'){ // console.log(item) // console.log(window.localStorage.getItem(item)) // } // size += window.localStorage.getItem(item).length; // } // } // console.log('当前localStorage使用容量为' + (size / 1024).toFixed(2) + 'KB'); // })() // 删除所有 hhh_BV // (function(){ // if(!window.localStorage) { // console.log('浏览器不支持localStorage'); // } // var size = 0; // for(item in window.localStorage) { // if(window.localStorage.hasOwnProperty(item)) { // if(item.match(/hhh_BV/)){ // localStorage.removeItem(item) // console.log(item) // } // //size += window.localStorage.getItem(item).length; // } // } // console.log('当前localStorage使用容量为' + (size / 1024).toFixed(2) + 'KB'); // })() const [MULTI_P, VIDEO_LIST] = [1,2] const LIST_TYPE = $('.list-box').length > 0 ? MULTI_P : //视频选集 $('.video-section-list').length > 0 ? VIDEO_LIST : //合集 undefined function get_bv() { return $('meta[itemprop=url]').attr('content')?.match(/BV\w+/)?.[0] } function get_sid() { return $('.first-line-title').attr('href')?.match(/sid=\w+/)?.[0] } //合集ID function get_progress_id() { return 'hhh_' + (LIST_TYPE === MULTI_P ? get_bv() : get_sid())} function get_part() { return +$('.cur-page')?.text()?.match(/(\d+)\/\d+/)?.[1] } function get_total() { return +$('.cur-page')?.text()?.match(/(\d+)\/(\d+)/)?.[2] } function get_draw_part(){ return LIST_TYPE === VIDEO_LIST ? +$('.video-episode-card__info').index($('.video-episode-card__info-playing')) + 1 : +$('.list-box li.on').index() + 1 } let run_count = 0 //换P激活 $(bb['video'])[0].removeEventListener('DOMSubtreeModified', $(bb['video'])[0].hhh_fn_change_part) if(open === ON){ const curr_bv = get_bv() function fn_change_part(e) { h5Player = geth5Player() if($(h5Player).attr('src')?.match(/blob:http.:\/\/www.bilibili.com\//)){ // log('------换P------', get_bv()) if(get_bv() === curr_bv){ run(open, run_count++) } } } $(bb['video'])[0].addEventListener('DOMSubtreeModified', fn_change_part) $(bb['video'])[0].hhh_fn_change_part = fn_change_part } function run(open, run_count){ // log('-----memory_multipart_progress run-----') $wrap = $(`#hhh_mysectionns_head`) const [MULTI_P, VIDEO_LIST] = [1,2] const LIST_TYPE = $('.list-box').length > 0 ? MULTI_P : //视频选集 $('.video-section-list').length > 0 ? VIDEO_LIST : undefined //合集 function bulid_real_to_draw(){ let real_to_draw = Array.from(new Array(1+get_total()).keys()) //默认real to real $('.hhh_episode_view').each((i,v)=>{ real_to_draw[+$(v).attr('real_idx')] = i+1 }) return real_to_draw } function get_obj(){ const progress_id = get_progress_id() const obj = get_value(progress_id) return obj } function save_obj(obj){ set_value(get_progress_id(), obj) } function update_obj(ended){ //log('---------update_obj----------') const progress_id = get_progress_id() const part = get_part() const currentTime = h5Player.currentTime // log('---------update_obj----------',progress_id, currentTime, part) const date = new Date() const obj = get_value(progress_id) || {} obj.last_part = part+'p' obj.currentTime = currentTime obj.watch_date = date.Format("MM-dd") obj.watch_time = date.Format("hh:mm") obj.progress_list ??= {} obj.progress_list[part] = ended || currentTime //看完的视频赋值 -1 save_obj(obj) // log(progress_id, obj) // log(progress_id, get_obj()) return obj } function counter_obj(){ const obj = get_obj() || {} obj.counter = obj?.counter === undefined ? 1 : ++obj.counter save_obj(obj) } function init(open){ //log('---------init----------') //$('.hhh-multi-page-progress').width(0) h5Player = geth5Player() const id = 'hhh_memory_multipart_progress' $(`#${id}`).remove() $('.hhh-multi-page-progress').remove() $('.hhh-multi-page-progress-border-right').remove() $('.hhh-viewed').remove() //log($(bb['video'])[0].hhh_fn_timeupdate,$(bb['video'])[0].hhh_fn_ended,$(bb['video'])[0].hhh_fn_change_part) h5Player.removeEventListener('timeupdate', $(bb['video'])[0].hhh_fn_timeupdate) h5Player.removeEventListener('ended', $(bb['video'])[0].hhh_fn_ended) //去掉所有 √ $('.video-episode-card__info-duration, .list-box .duration').each(function(i){ $duration = $(this) if($duration.text().match(/√/)) $duration.text($duration.text().match(/([\d:]+)/)?.[1]) //去掉 √ //test $duration.parent().css('backgraound-color', 'aqua') }) //初始化sort_arr和real_to_draw // sort_arr = [] // real_to_draw = [] // init_real_to_draw() // log(sort_arr) // log(real_to_draw) if(LIST_TYPE === undefined) return false if(open === OFF) return false //多video-sections-item small-mode,每个item添加视频数量 if($('.video-section-title p:first>span').length <= 0){ $('.video-sections-item').each((i, item)=>{ let $item = $(item) let n = $item.find('.video-episode-card').length // console.log('n: ',n) $item.find('.video-section-title p').append(` (${n})`) }) } if($('.video-episode-card__info-duration, .list-box .duration').length <= 0) return false // console.log($('.video-episode-card')[0].getBoundingClientRect().y) if($(`#${id}`).length <= 0){ //添加到mysectionns && 更新hhh_total_time已看时长百分比 $(`
`).appendTo($wrap) //已看时长百分比 if(!!get_obj()?.progress_list){ //log(get_obj()) let $duration = $('.video-section-list .video-episode-card__info-duration, .list-box .duration') let lis = get_obj().progress_list let passed_second = 0 let real_to_draw2 = bulid_real_to_draw() for(let [part, currentTime] of Object.entries(lis)){ //累计已看秒数 if(currentTime === -1){ //看完,currentTime=视频时长 currentTime = timestr_to_second($duration.eq(`${real_to_draw2[part]-1}`).text()?.match(/([\d:]+)/)?.[1]) } passed_second += currentTime } let percent = (passed_second/$('#hhh_total_time').attr('total-second')*100).toFixed(0) let new_text = `${$('#hhh_total_time').text().split('|')?.[0]}|(已看${percent}%)` $('#hhh_total_time').text(new_text) } } //鼠标悬浮显示百分比 $('.list-box, .video-sections-content-list').off('mouseover.hhh_title').on('mouseover.hhh_title',function(e){ if(e.originalEvent === undefined) return let path = e.originalEvent.path //判断是否为title for (let i = 0; i < path.length; i++) { const element = path[i] if(element.className === 'router-link-active' || element.className === 'video-episode-card'){ //读width确定百分比 const width = $(element).find('.hhh-multi-page-progress')[0]?.style.width const centstr = width !== undefined ? ' ('+parseInt($(element).find('.hhh-multi-page-progress')[0]?.style.width)+'%)' : '' //log($(element).find('.hhh-multi-page-progress')[0].style.width) if(LIST_TYPE === MULTI_P){ element.title = $(element).find('.part').text() + centstr }else if(LIST_TYPE === VIDEO_LIST){ const $title = $(element).find('.video-episode-card__info-title') $title[0].title = $title.text().trim() + centstr } } } }) //多video-sections-item,每个item添加视频数量 if($('.video-section-title .title:first>span').length <= 0){ $('.video-sections-item').each((i, item)=>{ let $item = $(item) let n = $item.find('.video-episode-card').length // console.log('n: ',n) $item.find('.video-section-title .title').append(` (${n})`) }) } counter_obj() // function jump_last_video(a){ // if(!$(a).attr('part') || $(a).attr('currentTime')) return false //无数据 // //log($(a).attr('part'), $(a).attr('currentTime')) // $(`.list-box li:eq(${parseInt($(a).attr('part'))-1}) .clickitem`).click() // h5Player.currentTime = $(a).attr('currentTime') // } const $multipart_progress = $(`#${id}`) //log($multipart_progress) $multipart_progress[0].restart = true //timeupdate const fn_timeupdate = function(e){ // log('---------fn_timeupdate----------') // log(h5Player.currentTime) let fn = fn_timeupdate fn_timeupdate.limit = 1000 //ms if($multipart_progress[0].restart === true) { fn.interval = 0 fn.interval_update = 0 fn.time = Date.now() $multipart_progress[0].restart = false } const now = Date.now() fn.interval += now - fn.time fn.interval_update += now - fn.time fn.time = now // log('---fn_timeupdate obj---',h5Player.currentTime,fn.interval,fn.limit) if((fn.interval === 0 || fn.interval > fn.limit) && h5Player.currentTime !== 0 ){ //默认每秒更新一次 //log(h5Player.duration, h5Player.currentTime) //新进度 < 记忆进度 60s更新 const obj = get_value(get_progress_id()) const part = get_part() const time = obj?.progress_list?.[part] if(time === undefined || (time !== -1 && time < h5Player.currentTime)){ fn.limit_update = 1000 } else{ fn.limit_update = 60000 } //60s fn.interval = 0 if(fn.interval_update > fn.limit_update){ update_obj() update_progress(part) update_title($multipart_progress) fn.interval_update = 0 } draw_progress_red(get_draw_part()) } } h5Player.addEventListener('timeupdate', fn_timeupdate) $(bb['video'])[0].hhh_fn_timeupdate = fn_timeupdate //end const fn_ended = function(e){ //log('---ended') $multipart_progress[0].restart = true update_obj(-1) update_progress() update_title($multipart_progress) } h5Player.addEventListener('ended', fn_ended) $(bb['video'])[0].hhh_fn_ended = fn_ended //点击跳转最后观看 //$multipart_progress.find('a').off('click.hhh_progress').on('click.hhh_progress', function(){ jump_last_video(this) }) //自动跳转最后观看 - 可能存在问题 //if(run_count <= 0 && $(bb['video'])[0].hhh_last_bv === get_bv() && config.getCheckboxSettingStatus('autoJumpMemoryProgress') === ON){ if($(bb['video'])[0].hhh_last_bv !== get_bv() && config.getCheckboxSettingStatus('autoJumpMemoryProgress') === ON){ log('自动跳转最后观看:',run_count,$(bb['video'])[0].hhh_last_bv,get_bv()) $(bb['video'])[0].hhh_last_bv = get_bv() waitForTrue( ()=> $('.hhh-video-seek').length >= 1, ()=> $('.hhh-video-seek').click() ) } //换P自动跳转最后观看时间 if(config.getCheckboxSettingStatus('autoJumpMemoryProgress') === ON){ log('换P自动跳转最后观看时间: ',run_count) const obj = get_obj() const part = get_part() const real_to_draw2 = bulid_real_to_draw() if(!!obj?.progress_list?.[part]){ let currentTime = obj?.progress_list?.[part] currentTime = currentTime < 0 ? 0 : currentTime $(`.list-box li:eq(${real_to_draw2[part]-1}) .clickitem`).click() //log(currentTime) //h5Player = geth5Player() h5Player.currentTime = currentTime } } return $multipart_progress } //根据进度画右侧红线 function draw_progress_red(part){ // log('---------draw_progress_red----------') // log(part,get_part(),real_to_draw[get_part()]) // log(sort_arr) // log(real_to_draw) // part = real_to_draw[part] let $list = $(`.list-box li:eq(${part-1})`) //视频选集 let $duration = $list.find('.duration') let height = '36px' if($('.video-section-list').length > 0){ //合集 $list = $(`.video-section-list .video-episode-card__info:eq(${part-1})`) $duration = $list.find('.video-episode-card__info-duration') height = '30px' } if($list.find('.hhh-multi-page-progress-border-right').length <= 0){ $list.find('.hhh-multi-page-progress').clone().appendTo($list.children('a').add($list)) .removeClass('hhh-multi-page-progress').addClass('hhh-multi-page-progress-border-right') } const duration = timestr_to_second($duration.text()?.match(/([\d:]+)/)?.[1]) const cent = (h5Player.currentTime/duration)*100 $list.find('.hhh-multi-page-progress-border-right').width(`${cent}%`).height(`${height}`) //右侧红线 } //title function update_title($multipart_progress){ //log('---------update_title----------') const obj = get_obj() //log(obj) if(!obj || !obj.watch_date || !obj.watch_time || (obj.currentTime !== 0 && !obj.currentTime) || !obj.last_part) return //log(get_progress_id(), obj.counter, obj.currentTime, sec2str(parseInt(obj.currentTime)), obj) //log(obj.last_part, real_to_draw[obj.last_part], real_to_draw) const format_date = obj.watch_date === new Date().Format("MM-dd") ? '今天' : obj.watch_date const real_to_draw2 = bulid_real_to_draw() const duration = LIST_TYPE === MULTI_P ? $(`.list-box .duration:eq(${parseInt(obj.last_part)-1})`).text().split(/√/)?.[0] : $(`.video-episode-card__info-duration:eq(${real_to_draw2[parseInt(obj.last_part)]-1})`).text().split(/√/)?.[0] const cent = parseInt(obj.currentTime / timestr_to_second(duration) * 100) //log(obj.currentTime, parseInt(obj.last_part), cent) const html = `看到${obj.last_part} ${sec2str(parseInt(obj.currentTime))} (${cent}%) ` //if(obj.counter <= 1) $multipart_progress.html('') //$multipart_progress.html('') //log($multipart_progress.html() === '') if(1 && $multipart_progress.html() === '') { //log('--------1--------') $multipart_progress.html(`${html}`) // add tip * 3 tip_create_3_X({ target: $('.hhh-video-seek'), tip_target: $('#hhh_tip'), gap: 6, title: ()=>{ return `${format_date} ${obj.watch_time}观看` } }) tip_create_3_X({ target: $multipart_progress.find('button'), tip_target: $('#hhh_tip'), gap: 6, title: '清空所有进度,重置为初始状态' }) tip_create_3_X({ target: $('.hhh-viewed'), tip_target: $('#hhh_tip'), gap: 6, title: '已观看数量,点击循环定位(右键向后)' }) //get last part function get_last_part(){ return parseInt($('.hhh-video-seek').attr('part'))-1 } //part click function part_click(part){ const real_to_draw2 = bulid_real_to_draw() part = real_to_draw2[part] if(LIST_TYPE === MULTI_P){ //多P $(`.list-box li:eq(${part}) .clickitem`).click() }else if(listyp === VIDEO_LIST){ //合集 $(`.video-section-list .video-episode-card:eq(${part})`).click() } } // get one_height function get_one_height(){ let $first_list if(LIST_TYPE === MULTI_P){ //多P $first_list = $('.list-box li:first') }else if(LIST_TYPE === VIDEO_LIST){ //合集 video-episode-card $first_list = $('.video-section-list .video-episode-card:first') } const margin_top = parseInt($first_list.css('margin-top')) const margin_bottom = parseInt($first_list.css('margin-bottom')) const one_height = $first_list.outerHeight() + Math.min(margin_top, margin_bottom) return one_height } //重置 $('#hhh_memory_multipart_progress button').off('click.hhh_progress_button').on('click.hhh_progress_button', ()=>{ let box = new hhh_box() box.create({target: $('#hhh_memory_multipart_progress button'), color: '#00b5e5', text: '是否重置', gap: 8, enter_fn: ()=>{ localStorage.removeItem(get_progress_id()) $('.hhh-multi-page-progress').remove() $('.hhh-multi-page-progress-border-right').remove() $('.hhh-viewed').remove() $('#hhh_total_time').text($('#hhh_total_time').text().split('|')[0]) const $list = $('.list-box .duration').add($('.video-section-list .video-episode-card__info-duration')) $list.each(function(){ const $duration = $(this) if($duration.text().match(/√/)) $duration.text($duration.text().match(/([\d:]+)/)?.[1]) //去掉 √ }) } }) }) //循环定位 $('.hhh-viewed').off('contextmenu').on('contextmenu', (e)=>{ e.stopImmediatePropagation(); e.preventDefault() }) $('.hhh-viewed').off('mouseup.hhh_progress_viewed').on('mouseup.hhh_progress_viewed', function(e){ const index = e.button === 2 ? -1 : 1 //前进或后退 const one_height = get_one_height() //const total_view = $('.list-box .hhh-multi-page-progress').length // 8 const total_view = LIST_TYPE === MULTI_P ? $('.list-box .hhh-multi-page-progress').length // 8 : $('.video-section-list .hhh-multi-page-progress').length const curr_view = parseInt($('.hhh-viewed').text()?.match(/\d+/)?.[0]) // 1 - 8 const is_single_num = !parseInt($('.hhh-viewed').text()?.match(/\d+.(\d+)/)?.[1]) // 判断显示类型 let next_view = (curr_view + index) || total_view // 0 -> length, length+1 -> 1 next_view = next_view > total_view ? 1 : next_view if(is_single_num && index === -1) next_view = total_view // (num)的形式,循环定位第一或最后 const curr_view_index = LIST_TYPE === MULTI_P ? $('.list-box li:has(.hhh-multi-page-progress)').eq(next_view - 1).index() : $('.video-section-list .video-episode-card:has(.hhh-multi-page-progress)').eq(next_view - 1).index() //log(one_height,total_view,curr_view,curr_view_index) $('.cur-list, .video-sections-content-list').scrollTop(4 + one_height * curr_view_index) $('.hhh-viewed').text(`(${next_view}/${total_view})`) //已观看数量 }) //初始化已观看数量 const total_view = LIST_TYPE === MULTI_P ? $('.list-box .hhh-multi-page-progress').length : $('.video-section-list .hhh-multi-page-progress').length if($('.hhh-viewed').attr('count') !== total_view){ $('.hhh-viewed').attr('count', total_view) $('.hhh-viewed').text(`(${total_view})`) //已观看数量 } //双击跳转到观看进度 $('.list-box, .video-sections-content-list').off('dblclick.hhh_video_list').on('dblclick.hhh_video_list', function(e){ const obj = get_obj() let part = get_part() //console.log('双击跳转观看进度', part) if(!!obj?.progress_list?.[part]){ let currentTime = obj?.progress_list?.[part] currentTime = currentTime < 0 ? 0 : currentTime const real_to_draw2 = bulid_real_to_draw() part = real_to_draw2[part] LIST_TYPE === MULTI_P ? $(`.list-box li:eq(${part-1}) .clickitem`).click() : $(`.video-section-list .video-episode-card:eq(${part-1})`).click() h5Player.currentTime = currentTime } }) //跳转最后1P 不使用 function jump_last_video(selector){ //log($(selector).attr('part'), $(selector).attr('currentTime')) if(!$(selector).attr('part') || !$(selector).attr('currentTime')) return false //无数据 part_click(get_last_part()) //$(`.list-box li:eq(${parseInt($(selector).attr('part'))-1}) .clickitem`).click() h5Player.currentTime = $(selector).attr('currentTime') h5Player.play() const margin_top = parseInt($('.list-box li:first').css('margin-top')) const margin_bottom = parseInt($('.list-box li:first').css('margin-bottom')) const one_height = $('.list-box li:first').outerHeight() + Math.max(margin_top, margin_bottom) $('.list-box').scrollTop(one_height * (parseInt($(selector).attr('part'))-4)) } //event * 3 $('.hhh-video-seek').off('click.hhh_progress_seek').on('click.hhh_progress_seek', function(){ const part = parseInt($('.hhh-video-seek').attr('part')) const real_to_draw2 = bulid_real_to_draw() $('.cur-list, .video-sections-content-list').scrollTop(4 + get_one_height() * (real_to_draw2[part]-1)) //log('---jump_last_video click---') //jump_last_video(this) }) }else{ //log('--------2--------') // const html = `看到${obj.last_part} ${sec2str(parseInt(obj.currentTime))} (${cent}%) // // ` let $video_seek = $multipart_progress.find('.hhh-video-seek') $video_seek.attr('part', obj.last_part) $video_seek.attr('currentTime', obj.currentTime) $video_seek.text(`看到${obj.last_part} ${sec2str(parseInt(obj.currentTime))} (${cent}%) `) //初始化已观看数量 const total_view = LIST_TYPE === MULTI_P ? $('.list-box .hhh-multi-page-progress').length : $('.video-section-list .hhh-multi-page-progress').length if($('.hhh-viewed').text().match(/^(\d+)$/) && $('.hhh-viewed').attr('count') !== total_view){ $('.hhh-viewed').attr('count', total_view) $('.hhh-viewed').text(`(${total_view})`) //已观看数量 } } } //progress function update_progress(part_index){ // log('---------update_progress----------') if($('#hhh_style_progress').length <= 0){ append_css($('#multi_page,.video-sections-content-list'), 'hhh_style_progress', ` .hhh-video-seek, .hhh-viewed { color: inherit; } .hhh-multi-page-li { position: relative; } .hhh-multi-page-progress { position: absolute; pointer-events:none; left: 0px; top: 0px; background-color: rgb(0 255 255 / 0.12); border-right: none; /*height: 36|30px;*/ } .hhh-multi-page-progress-border-right { position: absolute; pointer-events:none; left: 0px; top: 0px; background-color: #00000000; border-right: #ffbbbb 1px solid; /*height: 36|30px;*/ } ` ) } //用颜色标明进度百分比 const obj = get_obj() if(!!obj?.progress_list){ let real_to_draw2 = bulid_real_to_draw() // log(real_to_draw2) // log(real_to_draw) let slice = part_index === undefined ? obj.progress_list : {[part_index]: obj.progress_list[part_index]} for(let [part, currentTime] of Object.entries(slice)){ if(currentTime === 0) continue //画进度条和百分比 function draw_progress(part, currentTime){ let $list = $('.list-box').find('li').eq(`${part-1}`) //视频选集 let $duration = $list.find('.duration') let height = '36px' if($('.video-section-list').length > 0){ //合集 $list = $('.video-section-list').find('.video-episode-card__info').eq(`${part-1}`) $duration = $list.find('.video-episode-card__info-duration') height = '30px' } if($list.find('.hhh-multi-page-progress').length <= 0){ $list.css('position', 'relative') LIST_TYPE === MULTI_P ? $list.children('a').append(`
`) : $list.append(`
`) $list.find('.hhh-multi-page-progress').css('height', `${height}`) } //-1代表已看完,加 √ if(currentTime === -1){ $list.find('.hhh-multi-page-progress').width(`100%`) if(!$duration.text().match(/√/)) $duration.text($duration.text()+' √') }else{ const cent = (currentTime / timestr_to_second($duration.text())) * 100 $list.find('.hhh-multi-page-progress').width(`${cent}%`) if($duration.text().match(/√/)) $duration.text($duration.text().match(/([\d:]+)/)?.[1]) //去掉 √ } } // log(part, real_to_draw[part], currentTime) draw_progress(real_to_draw2[part], currentTime) // draw_progress(part, currentTime) } } } function init_sort(open){ if(open === OFF){ $('#hhh_listsort').off('click.hhh_listsort') $('#hhh_listsort').remove() $('#hhh_style_listsort').remove() $('.hhh_episode_view').remove() return false } //list中插入播放量和弹幕数 if(LIST_TYPE === VIDEO_LIST && $('.hhh_episode_view').length <= 0){ //得到视频列表信息 function get_season(bv){ //同步 let season = [] $.ajax({ url: `https://api.bilibili.com/x/web-interface/view?bvid=${bv}`, type: "GET", xhrFields: { withCredentials: true }, crossDomain: true, async: false, timeout: 1, success: function (data) { data.data.ugc_season.sections.forEach(e => { season = [...season, ...e.episodes] }) } }); return season } //插入播放量和弹幕数 & 将排序用内容保存到数组 waitForTrue(()=>$('.video-info-detail .view.item:first path').length > 0, ()=>{ // log('-----init sort-----') $('.base-video-sections-v1').width('100%') let season = get_season(get_bv()) //定义view dm let $view = $('.video-info-detail .view.item:first').clone() let $dm = $('.video-info-detail .dm.item:first').clone() $view.add($dm).find('svg').css({'width': '12px', 'height': '12px'}) $view.find('svg')[0].setAttribute('viewBox', '-2 -2 20 20') $dm.find('svg')[0].setAttribute('viewBox', '-2 -2 20 20') //定义episode let $episode = $('
') $episode.css({'color': 'var(--text3)', 'font-size': '12px'}) $episode.append($view.clone(), $dm.clone()) //找到最多播放量,最多弹幕,代表最宽宽度 let [max_view, max_dm] = [0, 0] let [max_view_index, max_dm_index] = [0, 0] $('.video-section-list .video-episode-card__info-title').each(function(index){ let title = $(this).text().trim() let {view:view_count, danmaku:dm_count} = season[index]?.arc?.stat if(max_view < view_count) { max_view = view_count; max_view_index = index} if(max_dm < dm_count) { max_dm = dm_count; max_dm_index = index} }) // log(max_view, max_dm) // log(max_view_index, max_dm_index) //插入最多播放量,最多弹幕节点,计算width let $title = $('.video-section-list .video-episode-card__info-title:first') let {view:view_count} = season[max_view_index].arc.stat let {danmaku:dm_count} = season[max_dm_index].arc.stat $episode.find('.view')[0].lastChild.nodeValue = ` ${formatPlayCount(view_count)} ` $episode.find('.dm')[0].lastChild.nodeValue = ` ${formatPlayCount(dm_count)} ` $title.after($episode.clone()) let head_text_arr = ['重置', 'Title', 'View', 'DM', 'TM', 'TM_str', 'View_fmt', 'DM_fmt'] //加入排序HEAD function add_listsort(){ //Sort Head CSS $('#hhh_style_listsort').remove() if($('#hhh_style_listsort').length <= 0){ append_css($('.video-sections-head'), 'hhh_style_listsort', ` .hhh-listsort-head { background-repeat: no-repeat; background-position: center right; padding: 4px 0px 4px 0px; white-space: normal; cursor: pointer; border: 1px solid #cdcdcd; border-right-width: 0px; background-color: #99bfe6; /*#99bfe6*/ background-image: url(); font-weight: bold; text-shadow: 0 1px 0 rgb(204 204 204 / 70%); opacity: 0.7; user-select: none; white-space: pre; /*background-color: #9fbfdf;*/ /*background-image: url(); ascending*/ /*background-image: url(); descending*/ } .hhh-listsort-head-asc { /*opacity: 0.91;*/ background-image: url(); } .hhh-listsort-head-desc { /*opacity: 0.91;*/ background-image: url(); } ` ) } let x1 = $('.video-episode-card__info')[0].getBoundingClientRect().x let x2 = $('.hhh_episode_view .view')[0].getBoundingClientRect().x let x3 = $('.hhh_episode_view .dm')[0].getBoundingClientRect().x let x4 = $('.video-episode-card__info-duration')[0].getBoundingClientRect().x let x5 = $('.video-episode-card__info-duration')[0].getBoundingClientRect().x + $('.video-episode-card__info-duration')[0].getBoundingClientRect().width + 12 let w1 = x2 - x1 let w2 = x3 - x2 let w3 = x4 - x3 let w4 = x5 - x4 let wt = w1 + w2 + w3 + w4 if(w2 === 0 || w3 === 0){ let w2_5 = (w2 + w3)/2; w2 = w2_5; w3 = w2_5; } // console.log(x1,x2,x3,x4,x5,'|',w1,w2,w3,w4,wt) // console.log(w1/wt,w2/wt,w3/wt,w4/wt) let [Index, Title, View, DM, TM] = ['重置', 'Title', 'View', 'DM', 'TM'] $('#hhh_listsort').remove() $('.border-bottom-line:first').before(`
${Index} ${Title} ${View} ${DM} ${TM}
`) return [w1,w2,w3,w4] } //修改bilibili setSize函数,修改对应语句,重置right-container宽度,适配插入的节点listsort(主要是点击量和弹幕量) function reset_setSize(){ //n = 1680 < innerWidth ? 411 : 350 let reg = /(n\s*=\s*\d+\s*<\s*innerWidth\s*\?\s*)(\d+)\s*:\s*(\d+)/ let new_w = $('.base-video-sections-v1').width() + 50 window.setSize = eval(`(${setSize.toString().replace(reg, `$1${new_w}:${new_w}`)})`) window.dispatchEvent(new Event('resize')) } //$('.base-video-sections-v1').css('visibility', 'hidden') waitForTrue(()=>{ if($('.hhh_episode_view:first').height()<=16 && $('.hhh_episode_view:last').height()<=16){ add_listsort() reset_setSize() return true }else{ let w = +$('.base-video-sections-v1').width() $('.base-video-sections-v1').width(`${w+50}`) return false } }, ()=>{ // console.time('main') requestAnimationFrame(()=>true) $('.hhh_episode_view').remove() $('.video-section-list .video-episode-card__info-title').each(function(index){ let $this = $(this) let {view:view_count, danmaku:dm_count} = season[index].arc.stat //[index(从1开始), title, view, dm, tm_str, tm, view_fmt, dm_fmt] //sort_arr.push([index+1, title, view_count, dm_count, $this.next().text().trim(), tm, formatPlayCount(view_count), formatPlayCount(dm_count)]) $episode.find('.view')[0].lastChild.nodeValue = ` ${formatPlayCount(view_count)} ` $episode.find('.dm')[0].lastChild.nodeValue = ` ${formatPlayCount(dm_count)} ` $episode.attr('view', view_count) $episode.attr('dm', dm_count) $episode.attr('real_idx', index+1) $this.after($episode.clone()) }) //video-section-list section-0 section-1 section-2 ... function store(i){ let $video_list = $(`.video-section-list:eq(${i})`) let $video_cards = $video_list.find('.video-episode-card') let video_cards_array = Array.from($video_cards) return video_cards_array } function bulid_sort_arr(cards){ // console.time('bulid_sort_arr') let sort_arr = [] $(cards).each((i,v)=>{ let $v = $(v) let Title = $v.find('.video-episode-card__info-title').attr('title') let $episode = $v.find('.hhh_episode_view') let [real_idx, View, DM] = [+$episode.attr('real_idx'), +$episode.attr('view'), +$episode.attr('dm')] let TM = timestr_to_second($v.find('.video-episode-card__info-duration').text()) // console.log(TM,$v.find('.video-episode-card__info-duration').text()) // log(real_idx, Title, View, DM, TM) sort_arr.push([real_idx, Title, View, DM, TM]) }) // console.timeEnd('bulid_sort_arr') return sort_arr } function restore(new_sort_arr, sort_arr, video_cards_array, i){ //建立排序后和真实顺序的对应关系,范围:1~length let real_to_draw = [] new_sort_arr.forEach(function(v,i){ real_to_draw[v[0]] = i+1 }) let new_video_cards_array = [] sort_arr.forEach(function(v,i){ new_video_cards_array[real_to_draw[v[0]]-1] = video_cards_array[i] }) let video_list = $('.video-section-list')[`${i}`] video_list.innerHTML = '' new_video_cards_array.forEach(function(v){ video_list.appendChild(v) }) } //sort function arr_sort(_sort_arr, order, i){ //ascending, descending let new_sort_arr if(order === 'ascending') { new_sort_arr = [..._sort_arr].sort(function(a, b) { if(typeof a[i] === 'string'){ return a[i].localeCompare(b[i]) }else{ return a[i] - b[i] } }) }else if(order === 'descending') { new_sort_arr = [..._sort_arr].sort(function(b, a) { if(typeof a[i] === 'string'){ return a[i].localeCompare(b[i]) }else{ return a[i] - b[i] } }) }else{err('[arr_sort] Out of range')} return new_sort_arr } //list_sort function list_sort(sort_order, i){ //[index, Title, View, DM, tm_str, TM, view_fmt, dm_fmt] = [0, 1, 2, 3, 4, 5, 6, 7] //let head_text_arr = ['重置', 'Title', 'View', 'DM', 'TM_str', 'TM', 'View_fmt', 'DM_fmt'] // log('------------------',$('.video-section-list').length) // log(sort_order, i, head_text_arr.indexOf('TM'), bulid_sort_arr(store(0))) $('.video-section-list').each((index)=>{ let cards = store(index) let sort_arr = bulid_sort_arr(cards) let new_sort_arr = arr_sort(sort_arr, sort_order, i) restore(new_sort_arr, sort_arr, cards, index) }) } $('#hhh_listsort').off('click.hhh_listsort') $('#hhh_listsort').on('click.hhh_listsort', function(e){ $('.hhh-listsort-head').css('opacity', 0.7) let $head = $(e.target) let sort = $head.attr('sort') let text = $head.text().trim() $head.css('opacity', 0.91) if(sort === 'none'){ $head.attr('sort', 'descending') .removeClass('hhh-listsort-head-asc') .addClass('hhh-listsort-head-desc') }else if(sort === 'ascending'){ $head.attr('sort', 'descending') .removeClass('hhh-listsort-head-asc') .addClass('hhh-listsort-head-desc') }else if(sort === 'descending'){ $head.attr('sort', 'ascending') .removeClass('hhh-listsort-head-desc') .addClass('hhh-listsort-head-asc') }else{err('[hhh_listsort click] Out of range')} if(text === '重置'){ list_sort('ascending', head_text_arr.indexOf(text)) $('.hhh-listsort-head').removeClass('hhh-listsort-head-asc') .removeClass('hhh-listsort-head-desc') }else{ list_sort($head.attr('sort'), head_text_arr.indexOf(text)) // head_text = Title, View, DM, TM } }) requestAnimationFrame(()=>false) // console.timeEnd('main') }) }) } } const $multipart_progress = init(open) if($multipart_progress === OFF) return false //update_obj() update_progress() update_title($multipart_progress) init_sort(status) } run(open, run_count++) } //给列表添加总时长 & 标注进度等 function add_to_video_sections_head(){ // log('======add_to_video_sections_head======') //统计list秒数 function get_list_second($items, index){ //all let second = 0 let begin = index === undefined ? 0 : index let end = index === undefined ? $items.length - 1 : index //log(index, begin, end) for(; begin <= end; ++begin){ second += timestr_to_second($items.eq(`${begin}`).text()?.match(/([\d:]+)/)?.[1]) } return second } function add_total_time($wrap, second){ if(!$wrap) return false const id = 'hhh_total_time' if($(`#${id}`).length <= 0){ $(`
`).appendTo($wrap) } let hour = ((second / (60*60/2)).toFixed(0)/2).toFixed(1) //0.5的倍数 $(`#${id}`).text(`总时长: ${sec2str(second)}(${hour}小时)`) .attr('total-second', second) } function init(){ let $items = $('.video-section-list .video-episode-card__info-duration') //合集 let $left = $('.second-line_left') if($items.length <= 0) { $items = $('.list-box .duration') //选集 $left = $('.head-left') } const id = 'hhh_mysectionns_head' if($items.length > 0){ if($(`#${id}`).length <= 0){ $left.wrap(`
`).parent() } } return [$(`#${id}`), $items] } let [$wrap, $items] = init() add_total_time($wrap, get_list_second($items)) //支持视频选集/合集(选集是单个BV号,合集是多个BV号) memory_multipart_progress(config.getCheckboxSettingStatus('memoryProgress'), [config.getCheckboxSettingStatus('sortList')]) } //搜索界面自动展开更多筛选 function auto_more_conditions(){ waitForTrue(()=>{ if($('.more-conditions.ov_hidden').length > 0) $('.vui_button.vui_button--active-shrink.i_button_more').click() return $('.more-conditions.ov_hidden').length <= 0 }, ()=>{}) } //新版评论图片,点击图片返回 function return_comment(){ let $image_wrap = $('.show-image-wrap') let $image_content = $('.image-content') if($image_content[0].id !== 'hhh_show_image_wrap'){ $image_content[0].id = 'hhh_show_image_wrap' //点击图片关闭图片:相对于右上角X $image_content.click(()=>{ $('.operation-btn-icon.close-container').click() }) //点击空白位置前后轮播图片:相当于左右箭头 $image_wrap.click((e)=>{ let wrap = $image_wrap[0].getBoundingClientRect() let image = $image_content[0].getBoundingClientRect() let clientX = e.clientX if(clientX >= wrap.left && clientX <= image.left) { $('.operation-btn-icon.last-image').click(); return false } if(clientX <= wrap.right && clientX >= image.right) { $('.operation-btn-icon.next-image').click(); return false } }) //设置不透明度,防止遮挡 $('.preview-list').css('opacity', '0.5') } } //https://api.bilibili.com/x/web-interface/archive/relation?aid=896486345&bvid=BV1CA4y1S7fS //得到投币数量 //初始化 function init() { // var box = document.getElementById("hhh"); // //配置选项 // var config = { childList: true, subtree: true }; // var observer = new MutationObserver(function (mutationsList, observer) { // for (var mutation of mutationsList) { // const target = mutation.target; // //typeof target.className === 'string' && target.className !== '' && log(target.className); // if (mutation.type == 'childList') { // //console.log('子元素被修改'); // } // else if (mutation.type == 'attributes') { // //console.log(mutation.attributeName + '属性被修改'); // } // } // }); // //开始观测 // observer.observe(document.body, config); //console.clear(); //读取设置 设置信息 && 快捷键信息 config.getCheckboxSetting() // GM_setValue('zw_test', a); // console.log(GM_getValue('zw_test')); // console.log(GM_getValue('zw_test').name); let is_card_load = true let is_content_load = null let is_rec_list_load = null let is_homepage_load = true let is_fav_list_load = true let is_history_break = null let is_run_3X_break = true let run_load = true let run_done = false let dm_t let clickitem_name let class_name_obj = {} let old_url new MutationObserver((mutations, observer) => { mutations.forEach(mutation => { function get_ver_3x_(){ let ver = null; let a = 0; setTimeout(()=>a=1, 1000); let r = waitForTrue(()=> $('.bpx-player-loading-panel-text').length>0 && a===1, ()=>{ ver = $('.bpx-player-loading-panel-text').text().match(/(\d+).(\d+).(\d+)-(\w+)/); return true; }); //log(r); return ver; } function get_ver_3x(){ let ver_reg = $('.bpx-player-loading-panel-text').length > 0? $('.bpx-player-loading-panel-text').text().match(/(\d+).(\d+).(\d+)-(\w+)/): null; let ver; if(ver_reg === null) ver = null; else if(ver_reg[1] === '3' && ver_reg[2] === '13') ver = BILI_3_X_VIDEO; else ver = BILI_3_X_MOVIE; return ver_reg; } function get_ver_(){ let ver_reg = $('.bpx-player-loading-panel-text').length > 0? $('.bpx-player-loading-panel-text').text().match(/(\d+).(\d+).(\d+)-(\w+)/): null; let ver; if(ver_reg === null) ver = null; else if(ver_reg[1] === '3' && ver_reg[2] === '13') ver = BILI_3_X_VIDEO; else ver = BILI_3_X_MOVIE; return ver; } function show_class_name(class_name_obj){ //显示查找 className个数 function show_appoint_num(min, max){ for(let [k,v] of Object.entries(class_name_obj)){ if(max === -1){ if(v >= min) log(k,v) }else{ if(v >= min && v <= max) log(k) } } } for(let i=0; i<=10; i++){ if(i<10){ log(`--------------[${i}]----------------`); show_appoint_num(i,i) } else if(i===10){ log(`--------------[${i}+]----------------`); show_appoint_num(i,-1) } } log('--------------End----------------') return for(let [k,v] of Object.entries(class_name_obj)){ log(k+' - '+v) let arr=[]; //for (let [k, v] of Object.entries(bb)) { !!v && v[0] === '.' && $(v).length !== -1 && arr.push(v+' - '+$(v).length) && 0 && log(v+' - '+$(v).length); } for(let i=0; i<=10; i++){ if(i<10){ log(`--------------[${i}]----------------`); arr.forEach(function(v){ +v.slice(-1) === i && v.slice(-2,-1) === ' ' && log(v); }) } else if(i===10){ log(`--------------[${i}+]----------------`); arr.forEach(function(v){ $.isNumeric(parseInt(v.slice(-2,-1))) && log(v); }) } } log('--------------End----------------') } log(class_name_obj) return; //for (let [k, v] of Object.entries(bb)) { !!v && v[0] === '.' && $(v).length !== length && log(v+' - '+$(v).length); } if(!!ver) { bb = {}; bb_config.set_bb(ver) } let arr=[]; for (let [k, v] of Object.entries(bb)) { !!v && v[0] === '.' && $(v).length !== -1 && arr.push(v+' - '+$(v).length) && 0 && log(v+' - '+$(v).length); } for(let i=0; i<=10; i++){ if(i<10){ log(`--------------[${i}]----------------`); arr.forEach(function(v){ +v.slice(-1) === i && v.slice(-2,-1) === ' ' && log(v); }) } else if(i===10){ log(`--------------[${i}+]----------------`); arr.forEach(function(v){ $.isNumeric(parseInt(v.slice(-2,-1))) && log(v); }) } } log('--------------End----------------') //log(arr) } const target = mutation.target //class_name_obj[target.className] = class_name_obj?.[target.className] === undefined ? 1 : ++class_name_obj[target.className] // typeof target.className === 'string' && target.className !== '' && log(target.className) // let $item_active = $('.list-box .on, .video-episode-card__info-playing') // let name = $item_active.text() // log(name) //typeof target.className === 'string' && target.className !== '' && log($('.rec-list .video-page-card').length); //干净链接 if(old_url !== window.location.href) { old_url = clear_url() } //const stage = mutation.previousSibling && target.getAttribute('stage') if($('.bilibili-player-video-wrap').length === 1){ //2.X // if($('#app').hasClass('app-v1')){ // log("2.X“V3版”加载完毕"); // bb_config.set_bb(BILI_3_X_VIDEO_V1); // run(); // }else{ // log("2.X“V2版”加载完毕"); // bb_config.set_bb(BILI_2_X_V2); // run(); // } } else if(1&&typeof target.className === 'string' && (target.className === 'bui-select-list-wrap#####' || target.className === 'rec-list####') ) { //bug?? bpx-player-control-top 会激活重新加载视频 clearTimeout(is_rec_list_load) is_rec_list_load = setTimeout(function() { log('订阅合集列表加载完毕'); expand_list(config.getCheckboxSettingStatus('expandList'), config.getCheckboxSettingArgs('expandList', 'columns')); log("连播列表加载"); run_rec_list_newtab(config.getCheckboxSettingStatus('openVideoInNewTab')); add_to_video_sections_head() //列表加入总时长等 list_filter() //合集列表关键字过滤 //列表加入总时长 //add_total_time() }, 200); //} else if(1&&typeof target.className === 'string' && (target.className === 'bpx-player-control-top###' || target.className === 'rec-list###')) { //|| target.className.includes('header-v2 win webscreen-fix') === true } else if(1&&typeof target.className === 'string' && (target.className.includes('teleport') === true || target.className.includes('header-v2 win webscreen-fix') === true || target.className.includes('main-container') === true || target.className.includes('video-toolbar-container') === true)) { log("投币快捷设置或添加到收藏夹关键字过滤") //log(target.className) collection_filter(config.getCheckboxSettingStatus('collectionFilter')) set_dialog('coin', target.className) //} else if(!bb_type && typeof target.className === 'string' && target.className === 'bpx-player-loading-panel-text') { //3.X pbp-tip bpx-player-loading-panel-text //视频界面 } else if(typeof target.className === 'string' && ((target.className === 'bui-select-list-wrap' && run_load === true) || target.className === 'rec-list')) { //3.X pbp-tip bpx-player-loading-panel-text function get_ver_____(){ let ver_reg = $('.bpx-player-loading-panel-text').length > 0? $('.bpx-player-loading-panel-text').text().match(/(\d+).(\d+).(\d+)-(\w+)/): null; let ver; if(ver_reg === null) { ver = null; ver_reg = []} else if($('#app').find('.l-con').length > 0) ver = BILI_3_X_VIDEO; else if($('#app').find('.plp-l').length > 0) ver = BILI_3_X_MOVIE; //todo plp-l else if($('#app.app-v1').length > 0) ver = BILI_3_X_VIDEO_V1; //todo plp-l else ver = BILI_2_X; //log(ver_reg[0]+' | '+ver) //log($('#app').find('.l-con').length+' | '+$('#app').find('.plp-l').length) return [ver, ver_reg[0]]; } function get_ver(){ let ver_reg = $('.bpx-player-loading-panel-text').length > 0? $('.bpx-player-loading-panel-text').text().match(/(\d+).(\d+).(\d+)-(\w+)/): null //log(ver_reg) ver_reg ??= $('.bpx-player-info-log .info-title').text().match(/[\d.]+/) //log(ver_reg) let ver if(ver_reg === null) { ver = null; ver_reg = [] } if($('#app').find('.l-con').length > 0) ver = BILI_3_X_VIDEO; else if($('#__next').find('.plp-l').length > 0) ver = BILI_3_X_MOVIE; //todo plp-l else if($('#app').length > 0) ver = BILI_3_X_VIDEO_V1; //todo plp-l else ver = BILI_2_X; //log(ver_reg[0]+' | '+ver) //log($('#app').find('.l-con').length+' | '+$('#app').find('.plp-l').length) return [ver, ver_reg[0]]; } function show_bb_config(ver){ //显示查找 className DOM个数 //for (let [k, v] of Object.entries(bb)) { !!v && v[0] === '.' && $(v).length !== length && log(v+' - '+$(v).length); } if(!!ver) { bb = {}; bb_config.set_bb(ver) } let arr=[]; for (let [k, v] of Object.entries(bb)) { !!v && v[0] === '.' && $(v).length !== -1 && arr.push(v+' - '+$(v).length) && 0 && log(v+' - '+$(v).length); } for(let i=0; i<=10; i++){ if(i<10){ log(`--------------[${i}]----------------`); arr.forEach(function(v){ +v.slice(-1) === i && v.slice(-2,-1) === ' ' && log(v); }) } else if(i===10){ log(`--------------[${i}+]----------------`); arr.forEach(function(v){ $.isNumeric(parseInt(v.slice(-2,-1))) && log(v); }) } } log('--------------End----------------') //log(arr) } function isrun_3_x_movie() { let exclude_k = ['danmukuTop', 'danmukuTopClose', 'danmukuBottom', 'danmukuBottomClose', 'biliDlgM', 'coinDlgCoin', 'coinDlgCloseBtn', 'coinDlgOkBtn', 'playPBP', 'playTipWrap2', 'playCtrlSubtitle', 'playWatchLater', 'likeon' , 'coinon', 'switchDot', 'like', 'coin' ]; let exclude_v = ['.like-info.active', '.coin-info.active', '.collect-info.active', '.coin-dialog-mask .coin-btn', '.bpx-player-volume-hint', '.bili-dialog-m', '.bpx-player-volume-hint-text', '.bpx-player-volume-hint-icon', '.bpx-player-hotkey-panel', '.coin-dialog-mask', '.bpx-player-electric-jump', '.bpx-player-hotkey-panel-close', '.bpx-player-info-close', '.bpx-player-info-container', '.info-line','.bpx-player-top-mask']; for(let [k, v] of Object.entries(bb)){ //if(!!k && exclude_k.indexOf(k) !== -1) log(k,v); //test if(!!k && exclude_k.indexOf(k) !== -1) continue; if(!!v && exclude_v.indexOf(v) === -1 && $(v).length === 0) log( k, exclude_v.indexOf(v) +' - '+$(v).length+' - '+v ); //test if(!!v && exclude_v.indexOf(v) === -1 && $(v).length === 0) return false; } return true; } function isrun_3_x_video() { let exclude_k = ['danmukuTop', 'danmukuTopClose', 'danmukuBottom', 'danmukuBottomClose', 'biliDlgM', 'coinDlgCoin', 'coinDlgCloseBtn', 'coinDlgOkBtn', 'playPBP', 'playTipWrap2', 'playCtrlSubtitle', 'playWatchLater', ]; let exclude_v = ['.ops .like.on', '.ops .coin.on', '.ops .collect.on', '.coin-dialog-mask .coin-btn', '.bpx-player-volume-hint', '.bpx-player-electric-jump', '.bpx-player-volume-hint-text', '.bpx-player-volume-hint-icon', '.bpx-player-hotkey-panel', '.bili-dialog-m', '.coin-dialog-mask', '.bpx-player-hotkey-panel-close', '.bpx-player-info-close', '.bpx-player-info-container', '.info-line','.bpx-player-top-mask', '.bili-dialog-m .bi-btn']; for(let [k, v] of Object.entries(bb)){ //if(!!k && exclude_k.indexOf(k) !== -1) log(exclude_k.indexOf(k) +' - '+k+' - '+v); if(!!k && exclude_k.indexOf(k) !== -1) continue; //if(!!v && exclude_v.indexOf(v) === -1 && $(v).length === 0) log(exclude_v.indexOf(v) +' - '+$(v).length+' - '+v); if(!!v && exclude_v.indexOf(v) === -1 && $(v).length === 0) return false; } return true; } function isrun_3_x_video_v1() { let exclude_k = ['danmukuTop', 'danmukuTopClose', 'danmukuBottom', 'danmukuBottomClose', 'biliDlgM', 'coinDlgCoin', 'coinDlgCloseBtn', 'coinDlgOkBtn', 'playPBP', 'playTipWrap2', 'playCtrlSubtitle', 'playWatchLater', 'playTipWrap', ]; let exclude_v = ['.video-toolbar-left .video-like.on', '.video-toolbar-left .video-coin.on', '.video-toolbar-left .video-fav.on', '.coin-dialog-mask .coin-btn', '.bpx-player-volume-hint', '.bpx-player-electric-jump', '.bpx-player-volume-hint-text', '.bpx-player-volume-hint-icon', '.bpx-player-hotkey-panel', '.bili-dialog-m', '.coin-dialog-mask', '.bpx-player-hotkey-panel-close', '.bpx-player-info-close', '.bpx-player-info-container', '.info-line','.bpx-player-top-mask', '.bili-dialog-m .bi-btn']; for(let [k, v] of Object.entries(bb)){ //if(!!k && exclude_k.indexOf(k) !== -1) log(exclude_k.indexOf(k) +' - '+k+' - '+v); if(!!k && exclude_k.indexOf(k) !== -1) continue; if(!!v && exclude_v.indexOf(v) === -1 && $(v).length === 0) log(exclude_v.indexOf(v) +' - '+$(v).length+' - '+v); if(!!v && exclude_v.indexOf(v) === -1 && $(v).length === 0) return false; } return true; } //clearTimeout(run_load); run_load = false //log('-------run_load----------') //log(target.className) //log($('.bpx-player-loading-panel-text').length) clearTimeout(run_done) run_done = setTimeout(function() { //log(`-------run_load2----------`) //log($('.bpx-player-loading-panel-text').text()) let ver_string [ver, ver_string] = get_ver(); log(`[${ver_string}][${ver}]加载完毕`); //if(!ver || !ver_string) { if(!ver) { clearTimeout(run_load); return true; } // log('---bb_config.set_bb(ver);---') bb_config.set_bb(ver) //////////////////////// //////////////////////// // show_bb_config() //setTimeout(()=>show_class_name(class_name_obj),2000) //show_class_name(class_name_obj) //return; //////////////////////// //////////////////////// function run_(){ run() video_loaded_run() //视频加载时执行 log('订阅合集列表加载完毕'); expand_list(config.getCheckboxSettingStatus('expandList'), config.getCheckboxSettingArgs('expandList', 'columns')) log("连播列表加载"); run_rec_list_newtab(config.getCheckboxSettingStatus('openVideoInNewTab')) list_filter() //合集列表关键字过滤 add_to_video_sections_head() //列表加入总时长等 //log('add_to_video_sections_head: ',target.className) //setTimeout(()=>run_load = true, 500) } if(ver === BILI_3_X_VIDEO) waitForTrue(isrun_3_x_video, run_); else if(ver === BILI_3_X_VIDEO_V1) waitForTrue(isrun_3_x_video_v1, run_); else waitForTrue(isrun_3_x_movie, run_); // log('订阅合集列表加载完毕'); expand_list(config.getCheckboxSettingStatus('expandList'), config.getCheckboxSettingArgs('expandList', 'columns')); // log("连播列表加载"); run_rec_list_newtab(config.getCheckboxSettingStatus('openVideoInNewTab')); // add_to_video_sections_head() //列表加入总时长等 }, 800); //动态首页 } else if(typeof target.className === 'string' && target.className==='bili-dyn-list__items'){ clearTimeout(is_content_load); //log(!!config.getCheckboxSettingStatus('showDynListContent') || 1) is_content_load = setTimeout(function() { log("动态首页加载完毕"); run_content(!!config.getCheckboxSettingStatus('showDynListContent') || true); }, 200) //当前在线 } else if(typeof target.className === 'string' && target.className === 'avatar' && target.baseURI.indexOf('www.bilibili.com/video/online.html') !== -1){ log("当前在线加载完毕"); run_online_preview(config.getCheckboxSettingStatus('onlinePreview')) //首页 } else if(typeof target.className === 'string' && window.location.href.match(/https?:\/\/www.bilibili.com\/?$/) && is_homepage_load === true){ is_homepage_load = false waitForTrue(()=> $('.rcmd-box-wrap').length === 1 || $('.bili-grid.short-margin.grid-anchor:first').length === 1 || $('.recommended-swipe.grid-anchor:first').length === 1, () => { if($('#app').length > 0) ver = BILI_3_X_VIDEO else if($('#i_cecream main.bili-layout').length > 0) ver = BILI_3_X_VIDEO_V1 else if($('#i_cecream main.bili-feed4-layout').length > 0) ver = BILI_4_X_V1 else ver = null bb_config.set_bb(ver) log(`[${ver}]-首页-加载完毕`) run_save_recommend_list(config.getCheckboxSettingStatus('saveRecommendList')) run_add_carousel_slide() run_add_online() run_adblock_remove() run_go_back_new_version() }) //第一时间执行 } else if(0&&typeof target.className === 'string' && window.location.href.match(/https?:\/\/www.bilibili.com\/.+/) && is_homepage_load === true){ //收藏夹 } else if(typeof target.className === 'string' && target.className==='be-scrollbar fav-list-container ps'){ // log(config.getCheckboxSettingStatus('favSetting')) // log(config.sets) clearTimeout(is_fav_list_load) is_fav_list_load = setTimeout(function() { log("收藏夹设置完毕") run_fav(config.getCheckboxSettingStatus('favSetting')); }, 200) //三连和选择弹窗 } else if(typeof target.className === 'string' && target.className.indexOf('bili-guide-all bili-guide') !== -1){ hideThreePopup(config.getCheckboxSettingStatus('hideThreePopup')); //打分弹窗 } else if(typeof target.className === 'string' && target.className.indexOf('bili-score bili-no-event') !== -1){ hideScorePopup(config.getCheckboxSettingStatus('hideScorePopup')); //全屏时视频左上角显示弹幕数和时长 // } else if(typeof target.className === 'string' && target.className.indexOf('bpx-player-top-left-title') !== -1){ // log('---title---'+$('.bpx-player-top-left-title').text()+' ### '+$('.bpx-player-video-info-dm').text()) // log($('.bpx-player-top-left-title').text().match(/已装填.*?条弹幕/).length) // if($('.bpx-player-top-left-title').text().match(/已装填.*?条弹幕/).length <= 0){ // $('.bpx-player-video-info-dm').text($('.bpx-player-video-info-dm').text()) // } // waitForTrue(()=>parseInt($('.bpx-player-video-info-dm').text().match(/\d+/)?.[0]) > 0, ()=>{ // log('title-2: '+parseInt($('.bpx-player-video-info-dm').text().match(/\d+/)?.[0])) // }) //log('2: '+$('.bpx-player-video-info-dm').text()) //全屏时视频左上角显示弹幕数和时长 } else if(typeof target.className === 'string' && target.className.indexOf('bpx-player-video-info-dm') !== -1){ //log('----dm----') //$('.bpx-player-top-wrap').css({visibility: '', opacity: ''}) //log(name+'|'+dm_count) //log(dm_count, clickitem_name, name, $item_active.length) //if((clickitem_name === undefined || ($item_active.length > 0 && name !== clickitem_name)) && dm_count >= 0) { //第一次执行或切换播放列表 // if((clickitem_name === undefined && dm_count >= 0) || // ($item_active.length > 0 )) { //第一次执行或切换播放列表 // log('#'+clickitem_name+'#|#'+name+'#') // clickitem_name = name //log('duration: '+$('.bpx-player-ctrl-time-duration').text()) waitForTrue(()=>bb_type !== undefined,()=>{ if(bb_type !== BILI_3_X_MOVIE){ setTimeout(()=>{waitForTrue(()=> $('.bpx-player-top-left-title').text() !== '' && $('.bpx-player-ctrl-time-duration').text() !== '00:00', ()=>{ let title = $('.bpx-player-top-left-title').text().match(/(.+?)\s*\|\s*已装填.+条弹幕/)?.[1] || $('.bpx-player-top-left-title').text() //log('show title: '+$('.bpx-player-top-left-title').text(),$('.bpx-player-ctrl-time-duration').text()) $('.bpx-player-top-left-title').text(title+' | '+$('.bpx-player-video-info-dm').text() +' | '+$('.bpx-player-ctrl-time-duration').text()) let $item_active = $('.list-box .on, .video-episode-card__info-playing') let name = $item_active.text() let dm_count = parseInt($('.bpx-player-video-info-dm').text().match(/\d+/)?.[0]) //if((clickitem_name === undefined && dm_count >= 0) || ($item_active.length > 0 )) { //第一次执行或切换播放列表 //log('1') //log(clickitem_name+'|'+name) if(clickitem_name !== name){ //log('2') clickitem_name = name $('.bpx-player-top-wrap').css({visibility: 'visible', opacity: 1}) clearTimeout(dm_t) dm_t = setTimeout(()=>{ $('.bpx-player-top-wrap').css({visibility: '', opacity: ''}) }, 4000) } })},0) } }) // } //全屏时视频左上角显示弹幕数(弹幕数为0时,不会激活‘bpx-player-video-info-dm’,但始终会激活online) } else if(typeof target.className === 'string' && target.className.indexOf('van-qrcode') !== -1){ //log('van-qrcode') // let $item_active = $('.list-box .on, .video-episode-card__info-playing') // let name = $item_active.text() // log(name) waitForTrue(()=>$('.bpx-player-top-left-title').text() !== '', ()=>{ setTimeout(()=>{ if($('.bpx-player-video-info-dm').text().match(/\d+/)?.[0] === '0') $('.bpx-player-video-info-dm').text($('.bpx-player-video-info-dm').text()) }, 0) //激活dm }) //全屏时视频左上角显示弹幕数(弹幕数为0时,不会激活‘bpx-player-video-info-dm’,但始终会激活online) // } else if(typeof target.className === 'string' && target.className.indexOf('bpx-player-video-info-online') !== -1){ // log('online') // let $item_active = $('.list-box .on, .video-episode-card__info-playing') // let name = $item_active.text() // log(name) // waitForTrue(()=>$('.bpx-player-top-left-title').text() !== '', ()=>{ // log('2: '+$('.bpx-player-video-info-dm').text()) // log('3: '+$('.bpx-player-video-info-online').text()) // setTimeout(()=>{ // if($('.bpx-player-video-info-dm').text().match(/\d+/)?.[0] === '0') $('.bpx-player-video-info-dm').text($('.bpx-player-video-info-dm').text()) // //$('.bpx-player-video-info-dm').text('***') // }, 800) //激活dm // }) //搜索界面自动展开更多筛选 } else if(typeof target.className === 'string' && target.className.indexOf('header-entry-avatar') !== -1){ auto_more_conditions() //新版评论图片,任意点击返回 } else if(typeof target.className === 'string' && target.className.indexOf('show-image-wrap') !== -1){ return_comment() //新版评论图片,任意点击返回 } else if(typeof target.className === 'string' && target.className.indexOf('toolbar-left-item-wrap') !== -1){ log('video-like: ',$('.video-like').hasClass('on')) $('.video-like')[0].addEventListener('DOMSubtreeModified', function(e) { log('video-like2: ',$('.video-like').hasClass('on')) }); // } else if(typeof target.className === 'string' && target.className==='history-list'){ // clearTimeout(is_history_break); // is_history_break = setTimeout(function() { log("历史记录加载完毕"); run_history()}, 200); //clearTimeout(is_rcmd_box_break); //is_rcmd_box_break = setTimeout(function() { log("推荐列表加载完毕"); run_recommend_box()}, 50); // } else if(is_card_load === false && typeof target.className === 'string' && target.className === 'card' && $(target).find('.video-container .content').length){ // is_card_load = true; // log("2.X动态首页加载完毕"); // run_card(); //搞笑,刚写完,2.70.7就有了 // removeMostViewedListener(config.getCheckboxSettingStatus('removeMostViewedListener')); } }); }).observe(document, { childList: true, subtree: true, //attributes: true, }); } init() } } hhh_lightoff_main.init()