// ==UserScript== // @name 哔哩哔哩新版首页排版调整和去广告(bilibili) // @namespace http://tampermonkey.net/ // @version 1.9.15 // @author Ling2Ling4 // @description 对新版B站首页的每行显示的视频数量进行调整, 同时删除所有广告, 并可设置屏蔽内容 (大尺寸屏幕每行将显示更多的视频) // @license MIT // @icon  // @match *://www.bilibili.com/* // @exclude *://www.bilibili.com/all* // @exclude *://www.bilibili.com/anime* // @exclude *://www.bilibili.com/pgc* // @exclude *://t.bilibili.com/* // @exclude *://www.bilibili.com/opus* // @exclude *://www.bilibili.com/live* // @exclude *://www.bilibili.com/article* // @exclude *://www.bilibili.com/upuser* // @exclude *://www.bilibili.com/match* // @exclude *://www.bilibili.com/platform* // @exclude *://www.bilibili.com/bangumi* // @exclude *://www.bilibili.com/cheese* // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @grant window.onurlchange // @noframes // @compatible chrome // @compatible edge // @compatible firefox // @downloadURL https://update.greasyfork.icu/scripts/477487/%E5%93%94%E5%93%A9%E5%93%94%E5%93%A9%E6%96%B0%E7%89%88%E9%A6%96%E9%A1%B5%E6%8E%92%E7%89%88%E8%B0%83%E6%95%B4%E5%92%8C%E5%8E%BB%E5%B9%BF%E5%91%8A%28bilibili%29.user.js // @updateURL https://update.greasyfork.icu/scripts/477487/%E5%93%94%E5%93%A9%E5%93%94%E5%93%A9%E6%96%B0%E7%89%88%E9%A6%96%E9%A1%B5%E6%8E%92%E7%89%88%E8%B0%83%E6%95%B4%E5%92%8C%E5%8E%BB%E5%B9%BF%E5%91%8A%28bilibili%29.meta.js // ==/UserScript== (() => { "use strict"; const info = { scriptUrl: "https://greasyfork.org/zh-CN/users/1196880-ling2ling4#user-script-list", apiUrl: "https://api.bilibili.com/x/web-interface/wbi/index/top/feed/rcmd", logoUrl: "https://i0.hdslb.com/bfs/archive/c8fd97a40bf79f03e7b76cbc87236f612caef7b2.png", imgDetails: "@672w_378h_1c_!web-home-common-cover", loadVideoNum: 10, marginTop1: 24, marginTop2: 24, clickMinInterval: 200, testNetInterval: 4, netSpeed: { value: 0, lastValue: 0 }, baseNetSpeed: 65, refreshLoopFnInfo: { interval: 40, svgNum: 12, rollNum: 20, rollNum2: 20, addNumMultiple: 0.8, maxAddNum: 25, }, isRemoveDelDom: !1, isScriptRefresh: !1, pageName: "", keyBase: "setting_", zoom: window.devicePixelRatio, otherAdCssDomId: "otherAdCss", homeClassList: { 标题: "h3 a", 作者: "bili-video-card__info--bottom", 分类: "floor-title", author: "bili-video-card__info--author", carousel: "recommended-swipe", vDom: ["container", "no-banner-container", "is-version8"], video: "feed-card", nav: "bili-header__bar", banner: "bili-header__banner", btn: "roll-btn", btn2: "flexible-roll-btn", specialCard: "floor-single-card", addVideo: "add-video", delUpBtn: "ll-del-up-btn", videoTime: "bili-video-card__stats__duration", }, homeSelector: { rbBtnAd: { selector: ".palette-button-wrap .adcard-content", type: "" }, rbCardAd: { selector: ".adcard", value: "adcard", type: "class" }, }, homeDelClassList: { 广告: [ "bili-video-card__info--ad", { value: "bili-video-card__mask", content: "广告" }, ], 推广: "bili-video-card__info--creative-ad", 特殊: "floor-single-card", 直播: "living", 番剧: "分类=番剧", 综艺: "分类=综艺", 课堂: "分类=课堂", 漫画: "分类=漫画", 国创: "分类=国创", 电影: "分类=电影", 纪录片: "分类=纪录片", 电视剧: "分类=电视剧", }, iconSvg: { blacklist: '', }, txt: { delUpBtnTT: "屏蔽该up的所有视频, 屏蔽后可在插件的屏蔽设置中修改" }, events: { rollSvg: !1, roll: !1, roll2: !1, resize: !1, wheel: !1 }, btnDoms: {}, errKeyArr: ["", 2, 3], errKeyInfo: { disNum: "setting_err_disNum", errNum: "setting_err_num", errTime: "setting_err_time", isTip: "setting_err_isTip", }, disErrTipNum: 3, errTipNum: 15, errTipInterval: 2, errNumReset: 5, videoInfo: { startTime: 500, pathInterval: 500, interval: 1e3, timeoutNum: 8, selector: { playerContainer: { value: "#bilibili-player .bpx-player-container", flag: "bpx-state-paused", }, likeBtn: { value: "#app .video-toolbar-container .video-like", flag: "on", }, curTime: { value: ".bpx-player-ctrl-time-current" }, ttTime: { value: ".bpx-player-ctrl-time-duration" }, }, classList: { videoPaused: "", likeBtn: "on" }, delDom: [ { name: "几乎全部广告", pre: ".", str: "ad-report", type: "ad", isAll: !0, }, { name: "横幅广告", pre: "#", str: "bannerAd" }, { name: "活动广告", pre: "#", str: "slide_ad", type: "ad" }, { name: "视频广告", pre: ".", str: "video-card-ad-small" }, { name: "右下广告", pre: "#", str: "right-bottom-banner" }, { name: "横幅通告", pre: ".", str: "bili-comments-header-renderer reply-notice", type: "ad", }, { name: "横幅通告", pre: "", value: "bili-comments-notice", selector: "bili-comments-notice", type: "ad", v: "v240713", shadowRootPath: [ [ { type: "ele", selector: "bili-comments", value: "bili-comments", }, { type: "ele", selector: "bili-comments-header-renderer", value: "bili-comments-header-renderer", }, ], ], }, { name: "游戏推荐", pre: ".", str: "video-page-game-card-small", type: "other", }, { name: "活动推广", pre: "#", str: "activity_vote", type: "other" }, { name: "直播推广", pre: ".", str: "pop-live-small-mode", type: "other", }, ], }, }, keyBase = info.keyBase, settings = { isCarousel: { value: !0, base: !0, key: keyBase + "isCarousel", txt: "是否显示轮播图 (修改后需要调整排列规则才能达到预期效果,轮播图占两个视频的宽度,所以每行显示视频数加减2即可)", type: "首页", valType: "boolean", compType: "radio", valueText: { true: "显示轮播图", false: "隐藏轮播图" }, }, delOtherAd: { value: ["rbBtnAd", "rbCardAd"], base: ["rbBtnAd", "rbCardAd"], key: keyBase + "delOtherAd", txt: "请勾选要删除的非视频类广告", type: "首页", valType: "array", compType: "checkbox", valueText: { rbBtnAd: "右下角按钮广告", rbCardAd: "右下角卡片广告" }, }, isRefreshLast: { value: !1, base: !1, key: keyBase + "isRefreshLast", txt: '是否在刷新视频时刷新掉最后的"直播"类视频 (新视频为普通视频)', type: "首页", valType: "boolean", compType: "radio", valueText: { true: "刷新掉'直播'类视频", false: "保留末尾的'直播'类视频", }, }, isAutoLayout: { value: !0, base: !0, key: keyBase + "isAutoLayout", txt: "是否根据缩放自动调整视频布局", type: "首页", valType: "boolean", compType: "radio", valueText: { true: "缩放时自动调整布局", false: "仅窗口变化时调整布局", }, }, isLoadOne: { value: !0, base: !0, key: keyBase + "isLoadOne", txt: "是否在进入网站时加载前三行的全部视频, 开启后前三行将不会出现预加载视频", type: "首页", valType: "boolean", compType: "radio", valueText: { true: "加载视口的全部视频", false: "保留视口的预加载视频", }, }, isDelUpBtn: { value: !0, base: !0, key: keyBase + "isDelUpBtn", txt: "是否在鼠标移入一个视频区域时显示屏蔽该up所有视频的按钮", type: "首页", valType: "boolean", compType: "radio", valueText: { true: "显示屏蔽up的按钮", false: "不显示屏蔽up的按钮" }, }, isReorderVideo: { value: !0, base: !0, valType: "boolean", key: keyBase + "isReorderVideo", txt: "是否按照排列规则调整布局 (每行能够显示更多视频)", type: "排列规则", compType: "radio", valueText: { true: "按照排列规则调整布局", false: "使用原来的视频排列", }, }, videoNumRule: { title: "排列规则", value: "0,1300,2\n1300,1800,3\n1800,3000,4\n3000,3700,5\n3700,6300,6", base: "0,1300,2\n1300,1800,3\n1800,3000,4\n3000,3700,5\n3700,6300,6", base2: "0,1300,4\n1300,1800,5\n1800,3000,6\n3000,3700,7\n3700,6300,8", key: keyBase + "videoNumRule", txt: "视频排列规则, 每条规则用 ; 分隔 或 换行书写. 未包含的尺寸将按照初始方式排列 (轮播图会影响视频的排列)\n 示例: 1300,1800,3; 1800,3000,4 表示浏览器宽度在1300~1800像素时每行显示3个视频(前两行), 1800~3000像素每行4个视频", type: "排列规则", valType: "string", compType: "textarea", compH: "120px", }, delClassArr: { value: "广告, 推广", base: "广告, 推广", key: keyBase + "delClassArr", title: "屏蔽设置", txt: "可根据需要自行修改, 可自定义, 每项用 , 分隔 或 换行书写\n例如: 广告, 推广, 直播, 作者==零泠丶\n----可选:\n广告、推广、特殊、直播、番剧、综艺、课堂、漫画、国创、电影、纪录片、电视剧 (注: '特殊'包含了它右边所有的)\n----自定义:\n1. 标题=xxx, 可屏蔽标题含xxx的视频, xxx部分支持&&运算符, 如: 标题=A&&B, 表示屏蔽标题同时含有A B内容的视频\n2. 作者=xxx, 可屏蔽作者名和发布日期中含xxx的视频, 若书写为\"作者==xxx\"则表示屏蔽作者xxx的视频", type: "屏蔽设置", valType: "string", compType: "textarea", compH: "100px", }, minVideoTime: { value: "", base: "", key: keyBase + "minVideoTime", title: "按时长屏蔽", txt: "请设置视频的屏蔽时长, 小于等于该时长的视频将被屏蔽\n示例:\n30 屏蔽小于等于30秒的视频\n12:30 屏蔽小于等于12分30秒的视频\n12分30秒 ...", type: "屏蔽设置", valType: "string", compType: "textarea", compH: "30px", }, video_isDelAd: { groupTitle2: "广告屏蔽设置", value: !0, base: !0, key: keyBase + "video_isDelAd", txt: "是否删除视频页的广告", type: "视频页", valType: "boolean", compType: "radio", valueText: { true: "删除广告", false: "保留广告" }, }, video_delSetting: { value: "部分广告", base: "部分广告", key: keyBase + "video_delSetting", txt: "可根据需要自行修改, 每项用 , 分隔 (该设置仅在删除广告时生效)\n例如: 部分广告, 活动推广\n----可选:\n全部广告: 所有直接广告和间接广告\n部分广告: 所有直接广告 (横幅广告,活动广告,视频广告,右下广告,横幅通告)\n横幅广告: 视频下方的横幅广告\n活动广告: 右侧视频列表上方的广告\n视频广告: 右侧视频列表上方的小视频广告\n右下广告: 右侧视频列表下方的广告\n横幅通告: 评论区上方的横幅通告\n游戏推荐: 视频右侧的视频的相关游戏推荐\n活动推广: 评论区上方的活动推广\n直播推广: 右侧视频列表下方的直播推广", type: "视频页", valType: "string", compType: "textarea", compH: "30px", }, video_isAutoLike: { groupTitle2: "自动点赞设置", value: !1, base: !1, key: keyBase + "video_isAutoLike", txt: "是否在播放视频后自动给视频点赞", type: "视频页", valType: "boolean", compType: "radio", valueText: { true: "播放后点赞", false: "关闭该功能" }, }, video_autoLike: { value: "50%", base: "50%", key: keyBase + "video_autoLike", txt: "视频播放到什么位置后自动给视频点赞, 可书写百分比, 也可书写分钟或秒 \n例如:\n80% 表示视频播放完80%后自动点赞\n2分钟 表示视频播放2分钟后自动点赞(视频需要大于2分钟)", type: "视频页", valType: "string", compType: "textarea", compH: "30px", }, isClearAd: { value: !0, base: !0, key: keyBase + "isClearAd", txt: "首页: 是否删除广告, 若不删除则会将所有广告移至视频列表的最后", type: "其他设置", valType: "boolean", compType: "radio", valueText: { true: "删除广告", false: "广告后移" }, }, isTrueEnd: { value: !1, base: !1, key: keyBase + "isTrueEnd", txt: "首页: 是否将广告移至预加载视频的后面, 关闭后广告将放置在预加载视频的前面 一般视频的后面 (仅在不删除广告时生效)", type: "其他设置", valType: "boolean", compType: "radio", valueText: { true: "放置页尾", false: "放置预加载视频前" }, }, isTestNetSpeed: { value: !0, base: !0, key: keyBase + "isTestNetSpeed", txt: "首页: 是否检测网速, 并根据网速调整'判断是否采用脚本刷新全部视频'的最大间隔时间,开启后在网速较低时不那么容易出现刷新按钮失效的bug", type: "其他设置", valType: "boolean", compType: "radio", valueText: { true: "测试网速", false: "不测试网速" }, }, addVideoNum: { value: 0, base: 0, valType: "number", key: keyBase + "addVideoNum", txt: "首页: 加载用于填充视口空白视频的视频数量, 0表示自动计算数量 (自动计算时可能会因为屏蔽了部分视频导致视口区域还留有空白视频)", type: "其他设置", compType: "textarea", compH: "30px", }, homeDelAdTime: { value: 0, base: 0, valType: "number", key: keyBase + "homeDelAdTime", txt: "首页: 点击换一换按钮并加载完新视频后 进行广告屏蔽的间隔时间, 0表示立即屏蔽 (毫秒, 1秒=1000毫秒)", type: "其他设置", compType: "textarea", compH: "30px", }, }; function getValue({ base, key, valType = "string", isReSet = !0, getValue = null, setValue = null, getVal = null, setVal = null, } = {}) { getValue && (getVal = getValue), setValue && (setVal = setValue); let val = getVal ? getVal(key) : localStorage.getItem(key); return ( void 0 !== base && null == val && ((val = base), isReSet && ("string" != typeof base && (base = JSON.stringify(base)), setVal ? setVal(key, base) : localStorage.setItem(key, base))), (valType = valType.toLowerCase()), "string" == typeof val ? "string" === valType ? val : "boolean" === valType || "number" === valType ? JSON.parse(val) : "object" === valType ? val ? JSON.parse(val) : {} : "array" === valType ? val ? JSON.parse(val) : [] : val : val ); } function getData() { for (const valName in settings) { const setting = settings[valName]; setting.value = getValue({ base: setting.base, key: setting.key, valType: setting.valType, getVal: GM_getValue, setVal: GM_setValue, }); } return settings; } const utils = { getW(isAutoLayout) { let width = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth; return ( console.log("浏览器实际宽度:", width * window.devicePixelRatio), console.log("浏览器像素宽度:", width * info.zoom), !isAutoLayout && (width *= window.devicePixelRatio), width ); }, getValue(key, defa = null) { let value = GM_getValue(key); return null == value ? defa : value; }, boolTxt: (val) => (val ? "是 (确定)" : "否 (取消)"), strToDom(str) { let tmpDom = document.createElement("div"); tmpDom.innerHTML = str; const dom = tmpDom.children[0]; return (tmpDom = null), dom; }, formatDate(timestamp, { delimiter = "-", isYear = !1, isAlign = !0 } = {}) { if (!timestamp) return -1; const date = new Date(timestamp), year = date.getFullYear(); let month = (date.getMonth() + 1).toString(), day = date.getDate().toString(); return ( isAlign && ((month = month.padStart(2, "0")), (day = day.padStart(2, "0"))), (isYear ? year + delimiter : "") + month + delimiter + day ); }, formatTime(time, { delimiter = ":", isAlign = !0 } = {}) { if (!time) return -1; let h = Math.floor(time / 60 / 60), m = Math.floor((time / 60) % 60), s = Math.floor(time % 60); return ( isAlign && ((h = h ? h.toString().padStart(2, "0") : ""), (m = m.toString().padStart(2, "0")), (s = s.toString().padStart(2, "0"))), (h ? h + delimiter : "") + m + delimiter + s ); }, async request(url, options = {}) { options.method = options.method || "GET"; try { const res = await fetch(url, options); if (!res.ok) throw new Error(`HTTP error! Status: ${res.status}`); return await res.json(); } catch (error) { throw error; } }, errHandle({ e = null, errTxt = "", logTxt = "", key = "" } = {}) { key && (key = "_" + key); let errNum = GM_getValue(info.errKeyInfo.errNum) || 0; if (errNum >= info.errTipNum) return; let disErrNum = GM_getValue(info.errKeyInfo.disNum + key) || 0; const curTime = Date.now(); let disS = (curTime - (GM_getValue(info.errKeyInfo.errTime + key) || curTime)) / 1e3; if (((disS = 0 === disS ? 5 : disS), disS < 5)) return; let flag = GM_getValue(info.errKeyInfo.isTip + key); (flag = void 0 === flag || flag), e && console.log(e), console.log(logTxt || errTxt), disS >= 60 * info.errTipInterval * 60 && ((flag = !0), GM_setValue(info.errKeyInfo.isTip + key, !0), GM_setValue(info.errKeyInfo.disNum + key, 0)), flag && disErrNum <= info.disErrTipNum && disS < (info.errTipInterval / 10) * 60 * 60 && (errNum++, disErrNum++, GM_setValue(info.errKeyInfo.errNum, errNum), GM_setValue(info.errKeyInfo.disNum + key, disErrNum), GM_setValue(info.errKeyInfo.errTime + key, curTime), alert(errTxt), disErrNum === info.disErrTipNum && GM_setValue(info.errKeyInfo.isTip + key, !1)); }, resetErrInfo() { const curTime = Date.now(); (curTime - info.errKeyArr.reduce((a, b) => { const t = +GM_getValue(info.errKeyInfo.errTime + b); return t < a ? t : a; }, curTime)) / 1e3 >= 24 * info.errNumReset * 60 * 60 && (GM_setValue(info.errKeyInfo.errNum, 0), info.errKeyArr.forEach((key) => { key && (key = "_" + key), GM_setValue(info.errKeyInfo.disNum + key, 0); })); }, getPathDom(classObj, pathArr, baseDom = document, isGetAll = !1) { if (!pathArr) return; const resultArr = []; let target; 0 === pathArr.length && pathArr.push([]), (pathArr = [...pathArr]); for (let i = 0; i < pathArr.length; i++) { const path = pathArr[i]; let dom = path.baseDom || baseDom || document, isContinue = !1; for (let j = 0; j < path.length; j++) { const item = path[j]; if (item.multiple) { let doms = dom.querySelectorAll(item.selector); if (0 === doms.length) return; doms = [].slice.call(doms); const addArr = path.slice(j + 1); doms.forEach((addBaseDom) => { const curAddArr = [...addArr]; (curAddArr.baseDom = addBaseDom.shadowRoot || addBaseDom), pathArr.push(curAddArr); }), (isContinue = !0); break; } if (((dom = dom.querySelector(item.selector)), !dom)) break; dom = dom.shadowRoot || dom; } if ( dom && !isContinue && ((target = dom.querySelector(classObj.selector)), target) ) { if (!isGetAll) return target; resultArr.push(target); } } return isGetAll ? resultArr : target; }, }; function textToTime(str) { if ((str += "").includes(":")) { const tArr = str.split(":"), len = tArr.length; let h = 0, m = 0, s = 0; 3 === len ? ((h = +tArr[0]), (m = +tArr[1]), (s = +tArr[2])) : 2 === len ? ((m = +tArr[0]), (s = +tArr[1])) : (s = +tArr[0]), (str = 60 * h * 60 + 60 * m + s); } return +str; } function timeStringToSeconds(timeString) { let seconds = 0; const hmsMatch = timeString.match( /^(\d{1,2})(?::(\d{1,2}))?(?::(\d{1,2}))?$/ ); if (hmsMatch) { const hours = Number(hmsMatch[1] || 0), minutes = Number(hmsMatch[2] || 0), secs = Number(hmsMatch[3] || 0); return ( (seconds = void 0 === hmsMatch[2] ? hours : void 0 === hmsMatch[3] ? 60 * hours + minutes : 3600 * hours + 60 * minutes + secs), seconds ); } const hmsChineseMatch1 = timeString.match( /(?:(\d+)小时)?(?:(\d+)分钟)?(?:(\d+)秒)?/ ); if (hmsChineseMatch1 && hmsChineseMatch1[0]) { return ( (seconds = 3600 * Number(hmsChineseMatch1[1] || 0) + 60 * Number(hmsChineseMatch1[2] || 0) + Number(hmsChineseMatch1[3] || 0)), seconds ); } const hmsChineseMatch2 = timeString.match( /(?:(\d+)时)?(?:(\d+)分)?(?:(\d+)秒)?/ ); if (hmsChineseMatch2 && hmsChineseMatch2[0]) { return ( (seconds = 3600 * Number(hmsChineseMatch2[1] || 0) + 60 * Number(hmsChineseMatch2[2] || 0) + Number(hmsChineseMatch2[3] || 0)), seconds ); } throw new Error("无效的时间格式"); } const videoObj = { info: { isDelAd: null, delSetting: null, base_isDelAd: null, base_delSetting: null, startTime: info.videoInfo.startTime, pathInterval: info.videoInfo.pathInterval, interval: info.videoInfo.interval, timeoutNum: info.videoInfo.timeoutNum, isVideoLiked: !1, likeTimer: null, }, doms: { ad: [], playerContainer: null, likeBtn: null, curTime: null, ttTime: null, }, delDom: info.videoInfo.delDom, initValue() { (this.info.isDelAd = settings.video_isDelAd.value), (this.info.delSetting = settings.video_delSetting.value), (this.info.base_isDelAd = settings.video_isDelAd.base), (this.info.base_delSetting = settings.video_delSetting.base); }, getAdInfo() { if (this.info.delSetting.includes("全部广告")) return this.delDom.filter( (item) => "ad" === item.type || "other" === item.type ); if ("部分广告" === this.info.delSetting) return this.delDom.filter( (item) => "ad" === item.type || "other" !== item.type ); { let arr = this.delDom.filter((item) => this.info.delSetting.includes(item.name) ); if (this.info.delSetting.includes("部分广告")) { const arr2 = this.delDom.filter((item) => { if (!arr.includes(item)) return "ad" === item.type || "other" !== item.type; }); arr = arr.concat(arr2); } return 0 === arr.length ? (utils.errHandle({ errTxt: `插件设置的广告屏蔽设置中 '${GM_getValue( "setting_video_delSetting" )}' 格式书写错误`, key: info.errKeyArr[2], }), -1) : arr; } }, getAd() { const ad = [], delAdInfo = this.getAdInfo(); -1 !== delAdInfo ? (delAdInfo.forEach((item) => { let adList = []; if ("." === item.pre) if (item.isAll) (adList = document.querySelectorAll(item.pre + item.str)), (adList = [].slice.call(adList)); else { const dom = document.querySelector(item.pre + item.str); dom && adList.push(dom); } else if (item.v) { const ele = utils.getPathDom(item, item.shadowRootPath); ele && adList.push(ele); } else { const ele = document.querySelector(item.pre + item.str); ele && adList.push(ele); } adList.length > 0 && adList.forEach((adDom) => { ad.push(adDom); }); }), (this.doms.ad = this.info.isDelAd ? ad.filter((item) => "none" !== item.style.display) : ad)) : (this.doms.ad = []); }, delAd() { this.doms.ad.forEach((item) => { "none" !== item.style.display && (item.style.display = "none"); }); }, showAd() { this.doms.ad.forEach((item) => { "none" === item.style.display && (item.style.display = ""); }); }, updateData() { (this.info.isDelAd = settings.video_isDelAd.value), (this.info.delSetting = settings.video_delSetting.value); }, autoLike() { if (settings.video_isAutoLike.value) { const selector = info.videoInfo.selector, doms = this.doms, getDom = () => { let playerContainer = doms.playerContainer; if (!playerContainer) { if ( ((playerContainer = document.querySelector( selector.playerContainer.value )), !playerContainer) ) return console.log("未获取到视频播放器的dom"), -1; doms.playerContainer = playerContainer; } return doms.likeBtn || ((doms.likeBtn = document.querySelector(selector.likeBtn.value)), doms.likeBtn) ? doms.curTime || ((doms.curTime = playerContainer.querySelector( selector.curTime.value )), doms.curTime) ? doms.ttTime || ((doms.ttTime = playerContainer.querySelector( selector.ttTime.value )), doms.ttTime) ? void 0 : (console.log("未获取到视频总时间的dom"), -1) : (console.log("未获取到当前播放时间的dom"), -1) : (console.log("未获取到点赞按钮的dom"), -1); }; let likeStart = !1; const clickHandle = () => { if (likeStart) return; if (this.info.isVideoLiked) return; const playerClass = selector.playerContainer.flag, likeBtnClass = selector.likeBtn.flag, timer = this.info.likeTimer; timer && clearInterval(timer), (likeStart = !0); let isDoms = !1, num = 0; this.info.likeTimer = setInterval(() => { let f, likeBtn = doms.likeBtn, isLiked = likeBtn && likeBtn.classList.contains(likeBtnClass); if (isLiked || this.info.isVideoLiked) return ( !isLiked && this.info.isVideoLiked && doms.likeBtn.click(), clearInterval(this.info.likeTimer), (this.info.likeTimer = null), void (this.info.isVideoLiked = !0) ); if ((isDoms || ((f = getDom()), num++), num > 5)) return ( clearInterval(this.info.likeTimer), void (this.info.likeTimer = null) ); if (-1 === f) return; isDoms = !0; const player = doms.playerContainer; if ( ((likeBtn = doms.likeBtn), (isLiked = likeBtn.classList.contains(likeBtnClass)), !player.classList.contains(playerClass) && !isLiked) ) { (() => { let isNeedLike = !1; const curTimeS = textToTime(doms.curTime.innerText), ttTimeS = textToTime(doms.ttTime.innerText), autoLikeStr = settings.video_autoLike.value; let targetTime = parseInt(autoLikeStr) || 0; return ( autoLikeStr.includes("分") || autoLikeStr.includes("min") ? (targetTime *= 60) : autoLikeStr.includes("秒") || autoLikeStr.includes("s") || (targetTime = (targetTime / 100) * ttTimeS), curTimeS + 1 >= targetTime && (isNeedLike = !0), !!isNeedLike && (doms.likeBtn.click(), console.log("给视频点赞"), !0) ); })() && (this.info.isVideoLiked = !0); } }, 1e3); }; document.addEventListener("click", clickHandle, !0); } }, init() { if ((this.initValue(), this.autoLike(), !this.info.isDelAd)) return; let i = 0; const timer = setInterval(() => { i++, this.getAd(), i > this.info.timeoutNum && clearInterval(timer), this.delAd(); }, this.info.interval); utils.resetErrInfo(); }, }; const baseCfg = { state: "", isEditing: !1, hasSelectedPage: !1, param: { id: "ll_edit_wrap", box: document.body, classBase: "ll_edit_", w: "500px", h: "", contentH: "450px", bg: "rgba(0, 0, 0, 0.15)", color: "#333", fontSize: "15px", fontFamily: "PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif", zIndex: 11e3, resetTt: "重置当前页的所有设置为默认值", isShowMenu: !1, isScrollStyle: !0, isResetBtn: !0, isOnlyResetCurPage: !0, showPage: void 0, isIntervalRun: !1, interval: 1e3, page: [], callback: { resetBefore: null, reset: null, confirmBefore: null, finished: null, interval: null, cancelBefore: null, cancelled: null, }, }, }, cfg = { version: "v1.2.2", isEditing: baseCfg.isEditing, hasSelectedPage: baseCfg.hasSelectedPage, timer: null, interval: 1e3, param: {}, tempParam: {}, allData: {}, oldData: {}, lastData: {}, baseData: {}, controls: {}, doms: { page: [] }, editText: {}, }; const css = function getCss() { const param = cfg.param, cBase = (param.page, param.classBase), baseStart = `#${param.id} .${cBase}`, fSize = param.fontSize ? param.fontSize : "14px", css = `#${ param.id } {\n position: fixed;\n left: 0;\n top: 0;\n width: 100%;\n height: 100%;\n z-index: ${ param.zIndex || 11e3 };\n background: ${ param.bg || "rgba(0, 0, 0, 0.12)" };\n display: none;\n}\n${baseStart}box {\n text-align: initial;\n letter-spacing: 1px;\n position: relative;\n width: ${ param.w || "450px" };\n ${ param.h ? "max-height:" + param.h : "" };\n margin: auto;\n color: ${ param.color || "#333" };\n background: #fff;\n font-size: ${fSize};\n line-height: normal;\n font-family: ${ param.fontFamily || "PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif" };\n border: 3px solid #dfedfe;\n border-radius: 10px;\n box-sizing: border-box;\n padding: 14px 8px 10px 15px;\n overflow: hidden;\n overflow-y: auto;\n}\n${baseStart}menu {\n font-weight: bold;\n font-size: ${ parseInt(fSize) + 1 }px;\n display: flex;\n flex-wrap: wrap;\n gap: 0 8px;\n}\n${baseStart}menu-item {\n margin-bottom: 8px;\n border: 1px solid #dfedfe;\n color: #9ecaff;\n background: #eef6ff;\n border-radius: 6px;\n padding: 6px 10px;\n cursor: pointer;\n}\n${baseStart}menu-item:hover {\n color: #65aaff;\n background: #dfedfe;\n border: 1px solid #dfedfe;\n}\n${baseStart}menu-item.active {\n color: #65aaff;\n background: #dfedfe;\n border: 1px solid #dfedfe;\n}\n${baseStart}page-box {\n max-height: ${ param.contentH || "" };\n padding-right: 7px;\n margin-bottom: 8px;\n overflow: hidden;\n overflow-y: auto;\n}\n${baseStart}page {\n display: none;\n}\n${baseStart}page.curPage {\n display: block;\n}\n${baseStart}comp {\n margin-bottom: 8px;\n}\n${baseStart}comp:last-child {\n margin-bottom: 2px;\n}\n${baseStart}tt {\n font-weight: bold;\n font-size: ${ parseInt(fSize) + 6 }px;\n margin-top: 4px;\n}\n${baseStart}tt2 {\n font-weight: bold;\n font-size: ${ parseInt(fSize) + 4 }px;\n margin-top: 3px;\n margin-bottom: 7px;\n}\n${baseStart}tt3 {\n font-weight: bold;\n font-size: ${ parseInt(fSize) + 2 }px;\n margin-top: 2px;\n margin-bottom: 6px;\n}\n${baseStart}desc {\n line-height: 1.5;\n}\n${baseStart}comp-tt {\n font-weight: bold;\n font-size: ${ parseInt(fSize) + 1 }px;\n line-height: 1.5;\n}\n${baseStart}comp-desc {\n line-height: 1.5;\n}\n${baseStart}rd-arr {\n line-height: 22px;\n}\n${baseStart}rd-arr label {\n margin-right: 6px;\n cursor: pointer;\n}\n${baseStart}rd-arr input {\n vertical-align: -2px;\n cursor: pointer;\n}\n${baseStart}rd-arr span {\n color: #666;\n margin-left: 2px;\n}\n#${ param.id } textarea {\n width: 100%;\n max-width: 100%;\n max-height: 300px;\n border-radius: 6px;\n line-height: normal;\n padding: 5px 7px;\n outline-color: #cee4ff;\n border: 1px solid #aaa;\n box-sizing: border-box;\n font-size: ${ parseInt(fSize) - 2 }px;\n font-family: PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif;\n /* 保留空格 */\n white-space: pre-wrap;\n /* 允许词内换行 */\n word-break: break-all;\n letter-spacing: 1px;\n overflow: hidden;\n overflow-y: auto;\n}\n#${ param.id } textarea::placeholder {\n color: #bbb;\n}\n${baseStart}ta-desc {\n margin-bottom: 3px;\n}\n${baseStart}btn-box {\n display: flex;\n justify-content: flex-end;\n}\n${baseStart}btn-box button {\n font-size: 16px;\n line-height: normal;\n color: #65aaff;\n background: #dfedfe;\n outline: none;\n border: none;\n border-radius: 6px;\n padding: 8px 16px;\n box-sizing: border-box;\n cursor: pointer;\n}\n${baseStart}btn-box .${cBase}reset-btn {\n position: absolute;\n left: 15px;\n bottom: 10px;\n color: #888;\n background: #f4f4f4;\n margin-right: 15px;\n}\n${baseStart}btn-box .${cBase}reset-btn:hover {\n color: #666;\n background: #eee;\n}\n${baseStart}btn-box .${cBase}cancel-btn {\n color: #888;\n background: #f4f4f4;\n margin-right: 15px;\n}\n${baseStart}btn-box .${cBase}cancel-btn:hover {\n color: #666;\n background: #eee;\n}\n${baseStart}btn-box .${cBase}confirm-btn {\n margin-right: 7px;\n}\n${baseStart}btn-box .${cBase}confirm-btn:hover {\n background: #cee4ff;\n}\n`; return param.isScrollStyle ? css + "\n.ll-scroll-style-1::-webkit-scrollbar,\n.ll-scroll-style-1 ::-webkit-scrollbar {\n width: 8px;\n}\n.ll-scroll-style-1-size-2::-webkit-scrollbar,\n.ll-scroll-style-1 .ll-scroll-style-1-size-2::-webkit-scrollbar {\n width: 10px;\n}\n.ll-scroll-style-1-size-3::-webkit-scrollbar,\n.ll-scroll-style-1 .ll-scroll-style-1-size-3::-webkit-scrollbar {\n width: 12px;\n}\n.ll-scroll-style-1::-webkit-scrollbar-thumb,\n.ll-scroll-style-1 ::-webkit-scrollbar-thumb {\n border-radius: 10px;\n -webkit-box-shadow: inset 0 0 8px rgba(0, 0, 0, 0.05);\n opacity: 0.2;\n background: #daedff;\n}\n.ll-scroll-style-1::-webkit-scrollbar-track,\n.ll-scroll-style-1 ::-webkit-scrollbar-track {\n -webkit-box-shadow: inset 0 0 8px rgba(0, 0, 0, 0.08);\n border-radius: 0;\n background: #fff;\n border-radius: 5px;\n}" : css; }; const editArea_html = function getHTML() { function getCompHTML({ info, active = "", id }) { let type = info.type; if ( ((type = { menuTitle: "mtt", title: "tt", title2: "tt2", title3: "tt3", desc: "ds", radio: "rd", checkbox: "cb", textarea: "ta", mtt: "mtt", tt: "tt", tt2: "tt2", tt3: "tt3", ds: "ds", rd: "rd", cb: "cb", ta: "ta", }[type]), (id = 0 === id ? "0" : id || ""), 0 === info.value && (info.value = "0"), !type) ) return console.log("不存在的组件类型"), !1; let title = "", desc = "", ctrlTt = ""; switch ( (["tt", "tt2", "tt3", "ds", "mtt"].includes(type) || ((title = info.title ? `
${info.title}
` : ""), (desc = info.desc ? `
${info.desc}
` : "")), type) ) { case "mtt": return ( (info.value = info.value || ""), info.value ? `
${info.value}
` : "" ); case "tt": case "tt2": case "tt3": return ( (info.value = info.value || ""), info.value ? `
${info.value}
` : "" ); case "ds": return ( (info.value = info.value || ""), info.value ? `
${info.value}
` : "" ); case "rd": const name = info.name || info.id + new Date().getTime(); (ctrlTt = info.ctrlTt || ""), ctrlTt && (ctrlTt = `title="${ctrlTt}"`); let radio = `
`; if (void 0 === info.value && info.radioList[0]) { const obj = info.radioList[0]; info.value = void 0 === obj.value ? obj.text : obj.value; } return ( info.radioList.forEach((item, i) => { void 0 === item.value && (info.radioList[i].value = item.text), void 0 === item.text && (info.radioList[i].text = item.value); const value = item.value; let tt = item.tt || ""; tt && (tt = `title="${tt}"`); let selected = ""; info.value + "" == item.value + "" && (selected = "checked"), (radio += ``); }), (radio += "
"), `
${title}${desc}${radio}
` ); case "cb": const name2 = info.name || new Date().getTime(); if ( ((ctrlTt = info.ctrlTt || ""), ctrlTt && (ctrlTt = `title="${ctrlTt}"`), void 0 === info.value && info.radioList[0]) ) { const obj = info.radioList[0]; info.value = void 0 === obj.value ? obj.text : obj.value; } let checkbox = `
`; return ( info.radioList.forEach((item, i) => { void 0 === item.value && (info.radioList[i].value = item.text), void 0 === item.text && (info.radioList[i].text = item.value); const value = item.value; let tt = item.tt || ""; tt && (tt = `title="${tt}"`); let selected = ""; info.value.includes(value) && (selected = "checked"), (checkbox += ``); }), (checkbox += "
"), `
${title}${desc}${checkbox}
` ); case "ta": const taH = `height:${info.height || "30px"};`, style = `style="${ info.width ? "width:" + info.width + ";" : "" }${taH}${ info.fontSize ? "font-size:" + info.fontSize + ";" : "" }${ info.fontFamily ? "font-family:" + info.fontFamily + ";" : "" }"`, textarea = ``; return `
${title}${desc}${textarea}
`; } } const param = cfg.param, page = param.page, cBase = param.classBase, isMenu = 1 !== page.length; let menu = `
`, pageHTML = `
`; page.forEach((curPage, index) => { let pgid = curPage.id || index; (pgid += ""), (cfg.allData[pgid] = {}), (cfg.baseData[pgid] = {}); let pageFlag = ""; if ( (cfg.hasSelectedPage || ((void 0 === param.showPage || pgid === param.showPage + "") && ((pageFlag = "curPage"), (cfg.hasSelectedPage = !0))), (pageHTML += `
`), curPage.components) ) { let compIndex = 0; if (isMenu || param.isShowMenu) { let curMenu = curPage.components.find( (item) => "menuTitle" === item.type ); curMenu || (curMenu = { type: "menuTitle", value: pgid }), (menu += getCompHTML({ info: curMenu, active: pageFlag ? "active" : "", })); } curPage.components.forEach((item) => { const cpid = item.id || compIndex; "menuTitle" !== item.type && (pageHTML += getCompHTML({ info: item, id: cpid })), ["title", "title2", "title3", "desc", "menuTitle"].includes( item.type ) || ((item.base = void 0 === item.base ? item.value : item.base), (cfg.allData[pgid][cpid] = item.value), (cfg.baseData[pgid][cpid] = item.base), compIndex++); }); } pageHTML += "
"; }), (pageHTML += "
"), isMenu || param.isShowMenu ? (menu += "
") : (menu = ""); const resetBtn = param.isResetBtn ? `` : "", btnBox = `
\n${resetBtn}\n\n\n
`; return `
\n${menu}\n${pageHTML}\n${btnBox}\n
`; }, baseParam = baseCfg.param, controls = cfg.controls, doms = cfg.doms; function createEditEle({ id = baseParam.id, box = baseParam.box, classBase = baseParam.classBase, w = baseParam.w, h = baseParam.h, contentH = baseParam.contentH, bg = baseParam.bg, color = baseParam.color, fontSize = baseParam.fontSize, fontFamily = baseParam.fontFamily, zIndex = baseParam.zIndex, resetTt = baseParam.resetTt, isShowMenu = baseParam.isShowMenu, isScrollStyle = baseParam.isScrollStyle, isResetBtn = baseParam.isResetBtn, isOnlyResetCurPage = baseParam.isOnlyResetCurPage, showPage = baseParam.showPage, isIntervalRun = baseParam.isIntervalRun, interval = baseParam.interval, page = [], callback = baseParam.callback, } = {}) { (cfg.state = baseCfg.state), (cfg.isEditing = baseCfg.isEditing), (cfg.hasSelectedPage = baseCfg.hasSelectedPage), (cfg.param = { ...baseParam }); const param = cfg.param; (box = box || document.body), (param.id = id), (param.box = box), (param.classBase = classBase), (param.w = w), (param.h = h), (param.contentH = contentH), (param.bg = bg), (param.color = color), (param.fontSize = fontSize), (param.fontFamily = fontFamily), (param.zIndex = zIndex), (param.resetTt = resetTt), (param.isShowMenu = isShowMenu), (param.isScrollStyle = isScrollStyle), (param.isResetBtn = isResetBtn), (param.isOnlyResetCurPage = isOnlyResetCurPage), (param.showPage = showPage), (param.isIntervalRun = isIntervalRun), (param.interval = interval), (param.page = page), (param.callback = callback), (cfg.interval = interval), (cfg.callback = callback); const html = editArea_html(); return ( box.querySelector(`#${param.classBase}${param.id}-css`) || (function addCss(cssText, box = document.body, id = "") { const style = document.createElement("style"); return ( id && (style.id = id), box.appendChild(style), (style.innerHTML = cssText), style ); })(css(), box, param.classBase + param.id + "-css"), (doms.wrap = (function createEle({ className = "", id = "", title = "", css, box = document.body, type = "div", } = {}) { const ele = document.createElement(type); return ( id && (ele.id = id), className && (ele.className = className), title && (ele.title = title), css && (ele.style.cssText = css), box.appendChild(ele), ele ); })({ className: id, id })), (doms.wrap.innerHTML = html), (function getDoms() { const param = cfg.param, cBase = param.classBase; (doms.box = doms.wrap.querySelector(`.${cBase}box`)), (doms.cancel = doms.box.querySelector(`.${cBase}cancel-btn`)), (doms.confirm = doms.box.querySelector(`.${cBase}confirm-btn`)); const isMenu = 1 !== param.page.length; (isMenu || param.isShowMenu) && ((doms.menu = doms.box.querySelector(`.${cBase}menu`)), (doms.menus = [].slice.call( doms.menu.querySelectorAll(`.${cBase}menu-item`) ))); const pages = [].slice.call(doms.box.querySelectorAll(`.${cBase}page`)); (doms.page = []), param.isResetBtn && (doms.reset = doms.box.querySelector(`.${cBase}reset-btn`)); pages.forEach((curPage, index) => { cfg.hasSelectedPage || (curPage.classList.add("curPage"), (isMenu || param.isShowMenu) && doms.menus[0].classList.add("active"), (cfg.hasSelectedPage = !0)); const page = {}, pgid = curPage.dataset.pgid; (page.pgid = curPage.pgid = pgid), (page.controls = [].slice.call( curPage.querySelectorAll(`.${cBase}ctrl`) )), (page.ele = curPage), doms.page.push(page), (isMenu || param.isShowMenu) && (doms.menus[index].settingsPage = curPage); const ctrls = {}; (controls[pgid] = ctrls), page.controls.forEach((item, i) => { const cpid = item.dataset.cpid, cType = item.dataset.type; let dom; (item.cpid = cpid), "rd" === cType || "cb" === cType ? ((dom = [].slice.call(item.querySelectorAll("input"))), (dom.compType = cType)) : "ta" === cType && ((dom = item.querySelector("textarea")), (dom.compType = cType), (dom.value = cfg.allData[pgid][cpid])), (ctrls[cpid] = dom); }); }); })(), cfg.timer && clearInterval(cfg.timer), (function bindEvents() { const param = cfg.param; function menuHandle(e) { const dom = e.target, cBase = param.classBase; if (dom.classList.contains(`${cBase}menu-item`)) { const old = doms.menu.querySelector(".active"); old.classList.remove("active"), old.settingsPage.classList.remove("curPage"), dom.classList.add("active"), dom.settingsPage.classList.add("curPage"); } } function cancelEdit(e) { const cBase = param.classBase; if ( (e.stopPropagation(), e.target.className !== `${cBase}wrap` && e.target.className !== `${cBase}cancel-btn`) ) return; const callback = cfg.callback; !1 !== runCallback(callback.cancelBefore) && (showEditArea(!1), setCompValue(cfg.oldData), param.isIntervalRun && (setCompValue(cfg.oldData), (cfg.allData = cfg.oldData)), runCallback(callback.cancelled)); } function confirmEdit() { const callback = cfg.callback, data = getAllData(); (cfg.allData = data), !1 !== runCallback(callback.confirmBefore, data) && (showEditArea(!1), (cfg.state = "finished"), runCallback(callback.finished, data), (cfg.state = "")); } function resetEdit() { const callback = cfg.callback, data = getAllData(); !1 !== runCallback(callback.resetBefore, data) && (!(function resetEditData(isOnlyPage = !1) { const param = cfg.param; if (param.isResetBtn) if (isOnlyPage) { const data = getAllData(), curMenu = doms.menu.querySelector(".active"); (data[curMenu.innerText] = cfg.baseData[curMenu.innerText]), setCompValue(data); } else setCompValue(cfg.baseData); })(param.isOnlyResetCurPage), runCallback(callback.reset, data)); } doms.menu && doms.menu.addEventListener("click", menuHandle), doms.wrap.addEventListener("click", cancelEdit), doms.cancel.addEventListener("click", cancelEdit), doms.confirm.addEventListener("click", confirmEdit), doms.reset && doms.reset.addEventListener("click", resetEdit); })(), (cfg.state = "created"), cfg ); } function getAllData() { function getCompItem(pgid, cpid) { if (!controls[pgid]) return; const ctrl = controls[pgid][cpid]; if (ctrl) { if (!Array.isArray(ctrl)) return ctrl.value; if ("rd" === ctrl.compType) { const result = ctrl.find((item) => item.checked).dataset.val; return "false" !== result && ("true" === result || result); } if ("cb" === ctrl.compType) { return ctrl .filter((item) => item.checked) .map((item) => { const value = item.dataset.val; return "false" !== value && ("true" === value || value); }); } } } const data = {}; if (0 === arguments.length) { for (const key in controls) { const page = controls[key]; data[key] = {}; for (const key2 in page) data[key][key2] = getCompItem(key, key2); } return data; } if (1 === arguments.length) { const ctrls = arguments[0]; for (const pgid in ctrls) { data[pgid] = {}; controls[pgid].forEach((cpid) => { data[pgid][cpid] = getCompItem(pgid, cpid); }); } return cfg.allData; } return getCompItem(arguments[0], arguments[1]); } function setCompValue() { function setCompItem(pgid, cpid, value) { if (!controls[pgid]) return; const ctrl = controls[pgid][cpid]; if (ctrl) if (Array.isArray(ctrl)) { if ("rd" === ctrl.compType) { const selected = ctrl.find((item) => item.checked); selected && (selected.checked = !1); const select = ctrl.find((item) => item.dataset.val === value + ""); select && (select.checked = !0); } else if ("cb" === ctrl.compType) { if ( (ctrl .filter((item) => item.checked) .forEach((item) => { item.checked = !1; }), Array.isArray(value)) ) value.forEach((val) => { const select = ctrl.find( (item) => item.dataset.val === val + "" ); select && (select.checked = !0); }); else { const select = ctrl.find( (item) => item.dataset.val === value + "" ); select && (select.checked = !0); } } } else ctrl.value = value; } if (1 === arguments.length) { const data = arguments[0]; for (const key in data) { const pageData = data[key]; for (const key2 in pageData) { setCompItem(key, key2, pageData[key2]); } } } else { setCompItem(arguments[0], arguments[1], arguments[2]); } } function showEditArea(isShow = !0, callback = null) { if ( (cfg.param.isIntervalRun && (cfg.timer && clearInterval(cfg.timer), (cfg.timer = setInterval(() => { const data = getAllData(), oldType = cfg.state; (cfg.state = "interval"), runCallback(cfg.callback.interval, data), (cfg.state = oldType), (cfg.lastData = data); }, cfg.interval))), (cfg.state = "created"), isShow) ) { if (((cfg.oldData = getAllData()), "function" == typeof callback)) { if (!1 === callback(cfg.oldData, cfg.oldData, cfg.baseData)) return; } cfg.state = "show"; } (cfg.isEditing = isShow), (doms.wrap.style.display = isShow ? "block" : "none"), isShow && !doms.box.style.top && (doms.box.style.top = window.innerHeight / 2 - doms.box.clientHeight / 2 + "px"), callback && (cfg.callback = callback); } function runCallback(callback, data) { let result; if (callback) { data || (data = getAllData()); const func = callback; Array.isArray(func) ? func.curFn ? ((result = func[curFn](data, cfg.oldData, cfg.baseData)), (func.curFn = null)) : func.forEach((fn) => { result = fn(data, cfg.oldData, cfg.baseData); }) : "function" == typeof callback && (result = func(data, cfg.oldData, cfg.baseData)); } return result; } function toPageObj({ settings, param = {}, otherPageName = "无分类" } = {}) { param = { ...param }; const pageArr = [], menuList = []; let isOtherType = !1; for (let key in settings) { const item = settings[key]; item.type ? menuList.includes(item.type) || menuList.push(item.type) : isOtherType || (isOtherType = !0); } return ( isOtherType && menuList.push(otherPageName), menuList.forEach((menuTt) => { const components = [], page = { id: menuTt, components }, arr = []; for (let key in settings) { const item = settings[key]; menuTt === otherPageName ? item.type || arr.push(item) : item.type === menuTt && arr.push(item); } arr.forEach((item) => { let desc = item.desc || item.txt || ""; desc && (desc = desc.replaceAll("\n", "
").trim()); let comp, base = item.base; if ( (Array.isArray(base) && (base = base.join(", ")), item.groupTitle1) ) { const comp = { id: item.key + "-gTt1", type: "title", value: item.groupTitle1, }; components.push(comp); } if (item.groupTitle2) { const comp = { id: item.key + "-gTt2", type: "title2", value: item.groupTitle2, }; components.push(comp); } if (item.groupTitle3) { const comp = { id: item.key + "-gTt3", type: "title3", value: item.groupTitle3, }; components.push(comp); } if (item.groupDesc) { const comp = { id: item.key + "-gDesc", type: "desc", value: item.groupDesc, }; components.push(comp); } if ( (["menuTitle", "title", "desc", "title2", "title3"].includes( item.compType ) ? ((comp = { ...item }), (comp.type = comp.compType), (comp.desc = desc)) : (comp = { id: item.key, type: item.compType, tt: item.tt || "", title: item.title || "", desc, descTt: item.descTt || "", name: item.key, value: item.value, base: item.base, }), "textarea" === comp.type) ) (comp.ph = base), (comp.width = item.compW), (comp.height = item.compH), (comp.ctrlTt = "默认: " + base); else if ("radio" === comp.type || "checkbox" === comp.type) { let str = "默认: "; if ("checkbox" === comp.type) { let arr = item.base; Array.isArray(arr) || (arr = arr.split(/,|,/)), arr.forEach((val, i) => { 0 !== i && (str += ", "), (val = val.trim()); let valTxt = item.valueText[val]; void 0 === valTxt && (valTxt = val), (str += valTxt); }); } else { let val = item.valueText[item.base]; void 0 === val && (val = item.base), (str += val); } comp.ctrlTt = str; } if (item.valueText) { comp.radioList = []; for (let key in item.valueText) { const rd = { text: item.valueText[key], value: key }; comp.radioList.push(rd); } } components.push(comp); }), pageArr.push(page); }), (param.page = pageArr), param ); } const intervalLoopFunc = function ({ fn, breakFn, num = 10, interval = 100, isStartRun = !0, successFn, finishedFn, } = {}) { if (fn) if (isStartRun && breakFn && breakFn()) fn(), successFn && successFn(), finishedFn && finishedFn(); else { let i = isStartRun ? 1 : 0; const timer = setInterval(() => { if (breakFn && breakFn()) return ( fn(), successFn && successFn(), finishedFn && finishedFn(), void clearInterval(timer) ); i++, i >= num && (finishedFn && finishedFn(), clearInterval(timer)); }, interval); } }; function getRandom(min, max) { return Math.floor(Math.random() * (max - min + 1) + min); } const icons = { base: { color: "#666", width: "100%", height: "80%", marginTop: "10%", html: "", }, lishi: { color: "#8a8a8a", width: "100%", height: "80%", marginTop: "10%", html: '', }, shoucang: { color: "#fe9850", width: "100%", height: "80%", marginTop: "10%", html: '', }, }; const classList = info.homeClassList, delClassList = info.homeDelClassList; let vDom, nav, banner, carousel, otherAdCssDom, base_videoNumRule, base_videoNumRule2, isClearAd, isTrueEnd, isAutoLayout, isLoadOne, isCarousel, videoNumRule, delClassArr; const apiUrl = info.apiUrl, imgDetails = info.imgDetails, queryNum = 0, marginTop1 = info.marginTop1, marginTop2 = info.marginTop2, zoom = info.zoom; let cssDom, cssText, oldCssText, w, isChange = !1, rowVideoNum = 3, videoNum = 0, newVideoNum = 0, firstAdIndex = 0, pageZoom = 1, loadNum = 0, fresh_idx = getRandom(0, 3e3), curDelNum = 0, scriptRefreshFlag = 0, refreshFlag = 0, isScriptRefresh = !1, hasDelUpBtn = !1, isCanMarkOldVideo = !0, isNeedMarkOldVideo = !1; window.location.pathname; function main(isLoads = !0) { !(function initValue() { (isChange = !1), (rowVideoNum = 3), (videoNum = 0), (newVideoNum = 0), (firstAdIndex = 0), (pageZoom = 1); })(), delOtherAd(), (isLoads && (loadNum++, loadNum > 3)) || (updateData(), (function home_getDoms() { if (vDom && nav && banner && carousel) return void (isCarousel || showCarousel(!1)); "string" == typeof classList.vDom ? (vDom = document.querySelector("." + classList.vDom)) : classList.vDom.forEach((item) => { !vDom && (vDom = document.querySelector("." + item)); }); if (!vDom) { const dom = document.querySelector("." + classList.video); if ((dom && (vDom = dom.parentElement), !vDom)) return; } (carousel = vDom.querySelector("." + classList.carousel)), (nav = document.querySelector("." + classList.nav)), (banner = document.querySelector("." + classList.banner)), (videoNumRule = (function home_getValue(key, defa = "") { let value = GM_getValue(key); return null == value ? defa : value; })( "setting_videoNumRule", carousel && isCarousel ? base_videoNumRule2 : base_videoNumRule )), "string" == typeof videoNumRule && (videoNumRule = getVideoNumRule(videoNumRule)); isCarousel || showCarousel(!1); })(), vDom || !isLoads ? ((loadNum = 0), (w = utils.getW(isAutoLayout)), zoomPage(), setStyle(), addDelUpBtnCss(), bindDelUpBtnEvent(), intervalLoopFunc({ breakFn: hasPreloadVideo, fn: function adFunc() { (videoNum = getVideoNum()), delAd(getAd(queryNum, delClassArr)), loadBlankVideo(() => { markOldVideo(), createDelUpBtn(); }); }, successFn: () => { intervalLoopFunc({ breakFn: hasSpecialVideo, fn: delAdFn }); }, interval: 50, num: 15, }), utils.resetErrInfo(), (function home_bindEvents() { let adArr, timer, timer2, timer3, rollBtn = info.btnDoms.rollBtn, btnSvg = info.btnDoms.btnSvg, rollBtn2 = info.btnDoms.rollBtn2; function bindEventBreakFn() { if ( (rollBtn || (rollBtn = document.querySelector( "button." + classList.btn )), btnSvg || (btnSvg = rollBtn && rollBtn.querySelector("svg")), rollBtn2 || (rollBtn2 = document.querySelector("." + classList.btn2)), (info.btnDoms.rollBtn = rollBtn), (info.btnDoms.btnSvg = btnSvg), (info.btnDoms.rollBtn2 = rollBtn2), rollBtn) ) return !0; } function bindBtnEvents() { let curRefreshFlag; const breakFn = () => !vDom.children[1].getAttribute("oldvideo"), fn = () => { getVideoNum(), (adArr = getAd( 3 * rowVideoNum + (carousel ? 2 : 0) + 3, delClassArr, getFirstAdIndex() )), delAd(adArr); }, refreshAfterFn = () => { refreshFlag === curRefreshFlag && (markOldVideo(), createDelUpBtn()); }, finishedFn = () => { (curRefreshFlag = refreshFlag), refreshVideo(refreshAfterFn); }, getAddNum = () => { if (!settings.isTestNetSpeed || 0 === info.netSpeed.value) return 0; let addNum = Math.ceil( (info.baseNetSpeed - info.netSpeed.value) * info.refreshLoopFnInfo.addNumMultiple ); return ( (!addNum || addNum < 0) && (addNum = 0), addNum > info.refreshLoopFnInfo.maxAddNum && (addNum = info.refreshLoopFnInfo.maxAddNum), addNum ); }; if (((btnSvg = null), btnSvg && !info.events.rollSvg)) (info.events.rollSvg = !0), btnSvg.addEventListener("transitionrun", () => { markOldVideo(); }), btnSvg.addEventListener("transitionend", () => { refreshFlag++, console.log("刷新视频"), markOldVideo(), info.isScriptRefresh ? refreshVideo(refreshAfterFn) : setTimeout(() => { intervalLoopFunc({ fn, breakFn, interval: info.refreshLoopFnInfo.interval, num: info.refreshLoopFnInfo.svgNum + getAddNum(), finishedFn, }); }, +settings.homeDelAdTime.value); }); else if (!info.events.roll) { info.events.roll = !0; const throttleRollClickHandle = (function throttle( fn, interval ) { let lastTime = 0; return function () { let nowTime = new Date().getTime(); const args = arguments; interval - (nowTime - lastTime) <= 0 && (fn.apply(this, args), (lastTime = nowTime)); }; })(() => { refreshFlag++, markOldVideo(), console.log("刷新视频"), info.isScriptRefresh ? refreshVideo(refreshAfterFn) : setTimeout(() => { intervalLoopFunc({ breakFn, fn, finishedFn, interval: info.refreshLoopFnInfo.interval, num: info.refreshLoopFnInfo.rollNum + getAddNum(), }); }, 100 + settings.homeDelAdTime.value); }, info.clickMinInterval); rollBtn.addEventListener("click", throttleRollClickHandle); } if (rollBtn2 && !info.events.roll2) { info.events.roll2 = !0; const breakFn2 = () => { const dom = vDom.children[1], isA = dom.querySelector("a"); return !dom.getAttribute("oldvideo") && isA; }, btnFn2 = () => { (videoNum = getVideoNum()), (adArr = getAd(queryNum, delClassArr, 1, !!isTrueEnd)), delAd(adArr, !0), loadBlankVideo(refreshAfterFn); }, finishedFn2 = () => {}; rollBtn2.addEventListener("click", () => { refreshFlag++, markOldVideo(), console.log("刷新全部视频"), isScriptRefresh ? refreshVideo(refreshAfterFn) : setTimeout(() => { intervalLoopFunc({ breakFn: breakFn2, fn: btnFn2, finishedFn: finishedFn2, interval: info.refreshLoopFnInfo.interval, num: info.refreshLoopFnInfo.rollNum2 + getAddNum(), }); }, 200 + settings.homeDelAdTime.value); }); } } function bindBtnEventFn() { intervalLoopFunc({ breakFn: bindEventBreakFn, fn: bindBtnEvents, successFn: () => { console.log("已获取刷新按钮"); }, interval: 300, }); } bindBtnEventFn(), info.events.resize || ((info.events.resize = !0), window.addEventListener("resize", () => { timer && clearTimeout(timer), (timer = setTimeout(() => { const newW = utils.getW(isAutoLayout); newW > w && delAdFn(), (w = newW), zoomPage(), setStyle(); }, 400)); })); info.events.wheel || ((info.events.wheel = !0), window.addEventListener("wheel", () => { rollBtn2 || bindBtnEventFn(), timer2 && clearTimeout(timer2), timer3 && clearTimeout(timer3), (timer2 = setTimeout(() => { delAdFn(timer3); }, 600)), (timer3 = setTimeout(() => { delAdFn(); }, 1500)); })); })(), setTimeout(() => { settings.isTestNetSpeed && testNetworkSpeed(5), setInterval(() => { settings.isTestNetSpeed && testNetworkSpeed(5); }, 6e4 * info.testNetInterval); }, 4e3), (document.testNetworkSpeed = testNetworkSpeed)) : setTimeout(() => { main(); }, 500)); } function zoomPage() { if (document.getBoxObjectFor) return; const rootDom = document.documentElement; if (!settings.isReorderVideo.value) return (pageZoom = 1), void (rootDom.style.zoom = 1); let rate = rootDom.scrollWidth / getMainW(); function getMainW() { let navW = nav ? nav.scrollWidth : 0; return navW > (banner ? banner.scrollWidth : 0) ? navW : rootDom.clientWidth; } !document.body.style.overflow && (document.body.style.overflow = "hidden auto"), rate > 1 ? ((pageZoom *= 1 / rate), (rootDom.style.zoom = pageZoom)) : ((pageZoom = 1), (rootDom.style.zoom = 1), (rate = rootDom.scrollWidth / getMainW()), rate > 1 && ((pageZoom *= 1 / rate), (rootDom.style.zoom = pageZoom))); } function getAd(queryNum, delClassArr, startIndex = 1, isAll = !1, vList) { curDelNum = 0; const arr = []; delClassArr.forEach(() => { arr.push([]); }), vList || (vList = vDom.children); const vLen = vList.length; let len = newVideoNum < vLen ? newVideoNum : vLen; (len = isAll || len > vLen ? vLen : len), (queryNum = queryNum || len), (queryNum += startIndex) > len && (queryNum = len); for (let i = startIndex; i < queryNum; i++) { const vItem = vList[i]; if (vItem) { if (!isAll && !vItem.querySelector("a")) break; for (let j = 0; j < delClassArr.length; j++) if (isChecked(vItem, delClassArr[j])) { (vItem.isNeedDelete = !0), arr[j].push(vItem); break; } } } return arr; } function delAd(adArr, isDel = !1, isDelJudge = !0) { function delItem(item) { curDelNum++; const h3 = item.querySelector("h3") || item.querySelector(".title"); isClearAd || isDel ? (console.log("删除视频:", h3 && h3.innerText, item), vDom.appendChild(item), (item.style.display = "none"), isRemove && setTimeout(() => { item.remove(); }, 1e3), newVideoNum--, videoNum--) : (console.log("后移视频:", h3 && h3.innerText, item), isTrueEnd ? (newVideoNum--, videoNum--, (item.style.display = "block"), vDom.appendChild(item)) : ((item.style.display = "block"), vDom.insertBefore(item, vDom.children[newVideoNum]))); } function isDeletable() { if (!isNeedDelJudege) return !0; let len = info.loadVideoNum + 1; const vList = vDom.children; for (let i = firstAdIndex + 1; i < len; i++) { const item = vList[i]; if (!item) return !0; if (item.getAttribute("oldvideo")) return !1; } return (isNeedDelJudege = !1), !0; } let isNeedDelJudege = isDelJudge; curDelNum = 0; const isRemove = info.isRemoveDelDom; for (let i = adArr.length - 1; i >= 0; i--) { adArr[i].forEach((item) => { isNeedDelJudege ? ((isCanMarkOldVideo = !1), intervalLoopFunc({ fn: () => { delItem(item); }, breakFn: isDeletable, finishedFn: () => { (isCanMarkOldVideo = !0), isNeedMarkOldVideo && markOldVideo(); }, interval: 50, num: 10, })) : delItem(item); }); } } function markOldVideo(len) { if (!isCanMarkOldVideo) return void (isNeedMarkOldVideo = !0); (isNeedMarkOldVideo = !1), (len = len || getFillVideoNum() || info.loadVideoNum); const vList = vDom.children; for (let i = 1; i < len + 1; i++) { const item = vList[i]; if (!item || !item.querySelector("a")) break; item.setAttribute("oldvideo", "true"); } } function delAdFn(timer = null) { if ((getVideoNum(), newVideoNum > videoNum)) { return ( delAd( getAd( queryNum, delClassArr, isTrueEnd ? videoNum : getFirstAdIndex() ), !1, !1 ), (videoNum = newVideoNum), timer && clearTimeout(timer), createDelUpBtn(), !0 ); } } function setStyle() { if (!settings.isReorderVideo.value) return cssDom && cssDom.remove(), (cssDom = null), void (oldCssText = ""); if ( ((isChange = !1), videoNumRule.forEach((item) => { !(function setVideoNum(vRule) { const min = (vRule = vRule.map((i) => +i.trim()))[0] / (isAutoLayout ? zoom : 1), max = vRule[1] / (isAutoLayout ? zoom : 1), num = vRule[2]; (0 === min || min) && max && num ? (w >= min && w < max && ((cssText = carousel ? `.container {grid-template-columns: repeat(${ num + 2 },1fr) !important}\n.container>div:nth-child(n){margin-top:${marginTop2}px !important}\n.container>div:nth-child(-n+${ 3 * num + 2 + 1 }){margin-top:${marginTop1}px !important;display:block !important}\n.container>div:first-child{display:${ isCarousel ? "block" : "none" } !important}\n.container>div:nth-child(-n+${ 2 * (num + 1) - 1 }){margin-top:0px !important}` : `.container {grid-template-columns: repeat(${num},1fr) !important}\n.container>div:nth-child(n){margin-top:${marginTop2}px !important}\n.container>div:nth-child(${ 3 * num + 1 }){display:block !important}\n `), (isChange = !0), (rowVideoNum = num)), isChange || ((cssText = ""), (rowVideoNum = carousel ? 5 : 3))) : utils.errHandle({ errTxt: `插件设置的视频排列规则设置中 '${vRule.join( "" )}' 格式书写错误`, key: info.errKeyArr[1], }); })(item); }), isChange) ) { let isCssDom = !!cssDom; isCssDom || ((cssDom = document.createElement("style")), cssDom.setAttribute("type", "text/css")), oldCssText !== cssText && (cssDom.innerHTML = cssText), (oldCssText = cssText), !isCssDom && document.head.appendChild(cssDom); } else !isChange && cssDom && ((oldCssText = ""), (cssDom.innerHTML = "")); } function getVideoNum() { const vArr = vDom.children, len = vArr.length; let i; for (i = 1; i < len; i++) { if (!vArr[i].querySelector("a")) return (newVideoNum = i), i; } return (newVideoNum = i), i; } function getFirstAdIndex() { const vArr = vDom.children, len = vArr.length; firstAdIndex = 0; for (let i = 1; i < len; i++) { const vItem = vArr[i]; for (let j = 0; j < delClassArr.length; j++) if (isChecked(vItem, delClassArr[j])) return (firstAdIndex = i), i; if (!vItem.querySelector("a")) return 0; } return 0; } function hasPreloadVideo() { const vArr = vDom.children, lastDom = vArr[vArr.length - 1]; return !(!lastDom || lastDom.querySelector("a")); } function hasSpecialVideo(range = 0) { const vClass = info.homeClassList.specialCard; let len = vDom.children.length; range && range < len && (len = range); for (let i = 1; i < len; i++) { const item = vDom.children[i]; if (item.classList.contains(vClass)) return !!item.querySelector("a") && i; } return !1; } function isChecked(vEle, delStr) { function custom(txt, type, selector) { const dom = vEle.querySelector(selector); if (!dom) return; const domTxt = dom.innerText; txt = txt.replace(type, ""); let f = !1; if ("作者==" !== type) { { const txtArr = txt.split("&&"); if (!txtArr[0]) return; txtArr.forEach((item) => { f = f || domTxt.includes(item); }); } flag = f; } else flag = domTxt === txt; } let flag = !1; if (settings.minVideoTime.value) { const minTime = timeStringToSeconds(settings.minVideoTime.value), timeDom = vEle.querySelector("." + info.homeClassList.videoTime); let time = 99999; if ( (timeDom && (time = timeStringToSeconds(timeDom.innerText)), time <= minTime) ) return !0; } const map = classList; if ( ((delStr = delStr.trim()), (delStr = delClassList[delStr] || delStr).includes("标题=")) ) custom(delStr, "标题=", map.标题); else if (delStr.includes("作者==")) custom(delStr, "作者==", "." + map.author); else if (delStr.includes("作者=")) custom(delStr, "作者=", "." + map.作者); else if (delStr.includes("分类=")) custom(delStr, "分类=", "." + map.分类); else { if (delStr.includes("json=")) { delStr = delStr.replace("json=", "").replaceAll("$1$", ","); try { delStr = JSON.parse(delStr); } catch (e) { console.log(e), utils.errHandle({ errTxt: `插件设置的屏蔽设置中 '${delStr}' 格式书写错误, 请在<>中按照json语法进行书写`, e, }); } } try { Array.isArray(delStr) || (delStr = [delStr]), (flag = delStr.some((item) => { if ("object" == typeof item) { const target = vEle.querySelector("." + item.value); if (target) { if (item.content) return target.innerText === item.content; if (item.include) return target.innerText.includes(item.include); } return !1; } return ( vEle.classList.contains(item) || vEle.querySelector("." + item) ); })); } catch (e) { console.log(e), utils.errHandle({ errTxt: `插件设置的屏蔽设置中 '${delStr}' 格式书写错误, 自定义格式应以 '标题=' 或 '作者=' 开头`, e, }); } } return flag; } function createVideoDom(data) { function formatNum(num) { return num > 1e8 ? (num / 1e8).toFixed(1) + "亿" : num > 1e4 ? (num / 1e4).toFixed(1) + "万" : "" + num; } function videoDom(item, reportStr) { if (item.business_info) return -1; const like = Math.floor(item.stat.like / 1e4), likeHTML = like > 1 ? `
${like}万点赞
` : "", upIconHTML = likeHTML ? "" : '\x3c!--[--\x3e\n \n \n \x3c!--]--\x3e\n', html = `
\n
\n
\n
\n \n \n \n \n \n
\n
\n
\n
\n
\n \n \n ${formatNum( item.stat.view )}\n \n \n \n \n \n ${formatNum( item.stat.danmaku )}
${utils.formatTime( item.duration )}\n
\n
\n
\n
\n \n
\n
`; return utils.strToDom(html); } const videoList = []; let num = 10; return ( data.forEach((item, index) => { num++; const dom = videoDom(item, `4-${index + 1}-${num}`); -1 !== dom && videoList.push(dom); }), videoList ); } function getFillVideoNum() { return isCarousel ? 3 * rowVideoNum + 2 : 3 * rowVideoNum; } async function requestVideos(num, isDelAd = !0) { if (num <= 0) return null; const params = new URLSearchParams({ web_location: 1430650, y_num: 4, fresh_type: 4, feed_version: "V8", fresh_idx_1h: 1, fetch_row: 4, fresh_idx, brush: 1, homepage_ver: 1, ps: num, last_y_num: 5, screen: window.innerWidth + "-" + window.innerHeight, wts: Math.floor(new Date().getTime() / 1e3), }); fresh_idx += getRandom(1, 10); const data = await utils.request(`${apiUrl}?${params}`); if (data.data.item) { let vList = createVideoDom(data.data.item); if (isDelAd) { getVideoNum(); delAd(getAd(num, delClassArr, 0, !1, vList), !1, !1), getVideoNum(); } return (vList = vList.filter((i) => !i.isNeedDelete)), vList; } return null; } async function loadBlankVideo(callBack = null) { if (!isLoadOne) return void (callBack && callBack()); let specialIndex; try { specialIndex = hasSpecialVideo(); const newNum = settings.addVideoNum.value || (() => { const dis = specialIndex ? 0 : -1; return isCarousel ? 3 * rowVideoNum + 2 - (newVideoNum - 1) + dis : 3 * rowVideoNum - (newVideoNum - 1) + dis; })(), vList = await requestVideos(newNum); if (((specialIndex = hasSpecialVideo()), vList)) { const insertIndex = specialIndex || newVideoNum; for (let i = vList.length - 1; i >= 0; i--) vDom.insertBefore(vList[i], vDom.children[insertIndex]); } callBack && callBack(); } catch (e) { console.log(e), setTimeout(() => { document.documentElement.scrollTo(0, 400), setTimeout(() => { document.documentElement.scrollTo(0, 0), setTimeout(() => { delAdFn(), callBack && callBack(); }, 800); }, 20); }, 1e3); } } async function refreshVideo(callBack = null) { if (!isLoadOne) return void (callBack && callBack()); scriptRefreshFlag++; const curFlag = scriptRefreshFlag; let vList, target, num = 0, isDelSpecialDom = !1, hasSpecialDom = !1, isOldVideos = !1; const vClass = info.homeClassList.addVideo, specialClass = info.homeClassList.specialCard, fillNum = getFillVideoNum(), removeArr = []; let specialDom; const vNum = getVideoNum(); for (let i = 1; i < vNum; i++) { const item = vDom.children[i]; if (!item || !item.querySelector("a")) break; if (i <= fillNum + 2) { if (item.classList.contains(specialClass)) { (specialDom = item), (hasSpecialDom = !0), (num += fillNum - i), settings.isRefreshLast.value && ((isDelSpecialDom = !0), num++); break; } } else { if ( (!isOldVideos && item.getAttribute("oldvideo") && (isOldVideos = !0), !isOldVideos) ) break; if (item.classList.contains(specialClass)) { (specialDom = item), (hasSpecialDom = !0), settings.isRefreshLast.value && (isDelSpecialDom = !0); break; } } } if (vDom.children[1].getAttribute("oldvideo")) { isScriptRefresh && (info.isScriptRefresh = !0), (isScriptRefresh = !0), (num = fillNum - 1), isDelSpecialDom && num++, console.log(`加载${num}个视频`); for (let i = fillNum; i >= 1; i--) { const item = vDom.children[i]; item && item.querySelector("a") && !item.classList.contains(specialClass) && removeArr.push(item); } target = vDom.children[1]; } else { if (((isScriptRefresh = !1), (info.isScriptRefresh = !1), isOldVideos)) { (num = 0), hasSpecialDom ? ((target = specialDom), num--, isDelSpecialDom && num++) : (target = vDom.children[info.loadVideoNum + 1]); for (let i = fillNum + 1; i < vNum; i++) { const item = vDom.children[i]; if (!item || item.isNeedDelete || !item.getAttribute("oldvideo")) break; item.classList.contains(vClass) ? removeArr.push(item) : item.classList.contains(specialClass) || removeArr.push(item); } } else hasSpecialDom ? (target = specialDom) : ((vNum <= fillNum + 1 || vNum <= fillNum + 1 + 2) && (num += fillNum + 1 - vNum), (target = vDom.children[info.loadVideoNum + 1])); const endIndex = isOldVideos ? fillNum : fillNum + 2; for (let i = 1; i <= endIndex; i++) { const item = vDom.children[i]; if (!item) break; if (item.classList.contains(specialClass)) break; (item.classList.contains(vClass) || item.getAttribute("oldvideo")) && (num++, removeArr.push(item)); } } try { if ( ((vList = await requestVideos(num)), vList && curFlag === scriptRefreshFlag) ) { if ((vList.length != num && (num = vList.length), target)) for (let i = 0; i < num; i++) vDom.insertBefore(vList[i], target); removeArr.forEach((i) => i.remove()), isDelSpecialDom && specialDom.remove(); } callBack && callBack(); } catch (e) { console.log("视频请求失败"), console.log(e), callBack && callBack(); } } function testNetworkSpeed(iterations) { let totalSpeed = 0; function runTest(interval = 0) { return new Promise((resolve, reject) => { let startTime, endTime; setTimeout(() => { let urlWithTimestamp = info.logoUrl + "?t=" + new Date().getTime(); (startTime = performance.now()), fetch(urlWithTimestamp) .then((response) => { if (!response.ok) throw new Error("Network request failed"); endTime = performance.now(); let duration = (endTime - startTime) / 1e3, speed = ( response.headers.get("content-length") / 1024 / duration ).toFixed(2); (totalSpeed += parseFloat(speed)), resolve(); }) .catch((error) => { console.error("Error:", error), reject(error); }); }, interval); }); } let promises = [], interval = 0; for (let i = 0; i < iterations; i++) promises.push(runTest(interval)), (interval += 1e3); Promise.all(promises) .then(() => { const speedStr = (totalSpeed / iterations).toFixed(2), netSpeed = +speedStr, avgSpeed = 0.6 * netSpeed + 0.4 * (info.netSpeed.value || netSpeed); (info.netSpeed.value = avgSpeed), console.log( `当前网速: ${speedStr}KB/s, 平均网速: ${avgSpeed.toFixed(2)}KB/s` ); }) .catch((error) => { console.error("Error during testing:", error); }); } function getDelClassArr(value) { const obj = (function extractBracesContent(str) { const matches = [], regex = /<([^>]+)>/g; let cleanString = str; return ( (cleanString = cleanString.replace( regex, (match, group) => (matches.push(group), "") )), { newStr: cleanString.trim(), matches } ); })(value); value = obj.newStr; return ( obj.matches.forEach((item) => { (item = "json=" + item.replaceAll(",", "$1$")), (value = value + "," + item); }), (value = value .replaceAll(";", ",") .replaceAll(";", ",") .replaceAll(",", ",") .replaceAll(",\n", "\n") .replaceAll("\n", ",\n")).split(",") ); } function getVideoNumRule(value) { return (value = (value = value .replaceAll(";", ";") .replaceAll(";\n", ";") .replaceAll("\n", ";\n")).split(";")).map((item) => item.split(/,|,/)); } function updateData() { (isClearAd = settings.isClearAd.value), (isTrueEnd = settings.isTrueEnd.value), (isAutoLayout = settings.isAutoLayout.value), (isLoadOne = settings.isLoadOne.value), (isCarousel = settings.isCarousel.value), (videoNumRule = getVideoNumRule(settings.videoNumRule.value)), (delClassArr = getDelClassArr(settings.delClassArr.value)), (base_videoNumRule = getVideoNumRule(settings.videoNumRule.base)), (base_videoNumRule2 = getVideoNumRule(settings.videoNumRule.base2)); } function showCarousel(isShow = !0) { carousel || (carousel = vDom.querySelector("." + classList.carousel)), carousel && (carousel.style.display = isShow ? "block" : "none"), isShow || (carousel = !1); } function createDelUpBtn() { if (!settings.isDelUpBtn.value) return; const vList = vDom.children, len = vList.length, upClass = info.homeClassList["作者"], svg = (function getIconHTML({ name = "", svg, color, width, height, marginTop, css = "", } = {}) { let icon; if ( (svg ? ((icon = { ...icons.base }), (icon.html = svg)) : (icon = icons[name]), icon) ) return ( (css += `\nfill:${color || icon.color || icons.base.fill};\nwidth:${ width || icon.width || icons.base.width };\nheight:${ height || icon.height || icons.base.height };\nmargin-top:${ marginTop || icon.marginTop || icons.base.marginTop };`), icon.html.replace("svgStyleFlag", css) ); })({ svg: info.iconSvg.blacklist, color: "rgb(148, 153, 160)", width: "12px", height: "12px", marginTop: "1px", }), createBtn = () => { const btn = document.createElement("span"); return ( (btn.className = info.homeClassList.delUpBtn), (btn.title = info.txt.delUpBtnTT), (btn.innerHTML = svg), btn ); }; for (let i = 1; i < len; i++) { const item = vList[i]; if (item.hasDelUpBtn) continue; if (!item.querySelector("a")) break; const up = item.querySelector(`.${upClass}`); up && (up.appendChild(createBtn()), (item.hasDelUpBtn = !0)); } } function addDelUpBtnCss() { if (!settings.isDelUpBtn.value) return; if (hasDelUpBtn) return; const style = document.createElement("style"); document.head.appendChild(style); const selector = "." + [].slice.call(vDom.classList).join("."); style.innerHTML = `${selector} > div:hover .${info.homeClassList.delUpBtn} {\n opacity: 1;\n}\n.${info.homeClassList.delUpBtn} {\n opacity: 0;\n display: flex;\n margin-left: 8px;\n cursor: pointer;\n transition: opacity 0.5s;\n}`; } function bindDelUpBtnEvent() { if (!settings.isDelUpBtn.value) return; if (hasDelUpBtn) return; const btnClass = info.homeClassList.delUpBtn; vDom.addEventListener("click", (e) => { let target = e.target, flag = target.classList.contains(btnClass); if ( (flag || ((target = target.parentElement), (flag = target.classList.contains(btnClass)), flag || ((target = target.parentElement), (flag = target.classList.contains(btnClass)))), flag) ) { if ( !((dom) => { const author = dom.parentElement.querySelector( `.${info.homeClassList.author}` ); if (!author) return !1; const str = `\n作者==${author.innerText}`, newDelText = settings.delClassArr.value + str; return ( (delClassArr = getDelClassArr(newDelText)), GM_setValue(settings.delClassArr.key, newDelText), !0 ); })(target) ) return; delAd(getAd(queryNum, delClassArr), !1, !1); } }), (hasDelUpBtn = !0); } function delOtherAd() { let cssText, isDel = !0, delAdList = settings.delOtherAd.value; 0 === delAdList.length && (isDel = !1), (delAdList = delAdList.map((item) => info.homeSelector[item].selector)), (cssText = isDel ? `${delAdList.join(",")}{display:none}` : ""), otherAdCssDom || ((otherAdCssDom = document.createElement("style")), (otherAdCssDom.id = info.otherAdCssDomId), document.body.appendChild(otherAdCssDom)), (otherAdCssDom.innerHTML = cssText); } function setValue({ value, base, key, verification = null, getValue = null, setValue = null, getVal = null, setVal = null, } = {}) { getValue && (getVal = getValue), setValue && (setVal = setValue); let f = !1; try { (getVal !== GM_getValue && setVal !== GM_setValue) || (f = !0); } catch (e) {} let newVal = value, oldVal = getVal ? getVal(key) : localStorage.getItem(key); return ( void 0 !== base && null == oldVal && ((oldVal = base), "string" == typeof base || f || (base = JSON.stringify(base)), setVal ? setVal(key, base) : localStorage.setItem(key, base)), null !== newVal && ("function" != typeof verification || ((newVal = verification(newVal, oldVal, base)), null !== newVal)) && newVal !== oldVal && ("string" == typeof newVal || f || (newVal = JSON.stringify(newVal)), setVal ? setVal(key, newVal) : localStorage.setItem(key, newVal), !0) ); } function showSettings() { let showPage; "首页" === info.pageName ? (showPage = "基础设置") : "视频" === info.pageName && (showPage = "视频页"), (info.settingsArea = (function createEdit({ settings, param = {}, oldEditCfg, updateDataFn, isNewEdit = !0, isSyncOtherPage = !0, otherPageName = "无分类", } = {}) { let oldSettings, curSettings; updateDataFn && isSyncOtherPage && ((oldSettings = JSON.stringify(settings)), (settings = updateDataFn() || settings), (curSettings = JSON.stringify(settings))); const editInfo = { settings, param, otherPageName }; if (oldEditCfg) { if (isNewEdit) return ( oldEditCfg.doms.wrap.remove(), createEditEle(toPageObj(editInfo)) ); isSyncOtherPage && updateDataFn && oldSettings !== curSettings && (oldEditCfg.doms.wrap.remove(), (oldEditCfg = createEditEle(toPageObj(editInfo)))), isSyncOtherPage && !updateDataFn && (oldEditCfg.doms.wrap.remove(), (oldEditCfg = createEditEle(toPageObj(editInfo)))); } else oldEditCfg = createEditEle(toPageObj(editInfo)); return oldEditCfg; })({ settings, param: { resetTt: "重置当前页的所有设置为默认值", isOnlyResetCurPage: !0, showPage, }, oldEditCfg: info.settingsArea, updateDataFn: getData, })), updateData(), videoObj.updateData(); const callback = { resetBefore: () => confirm("是否重置当前页的所有设置为默认值?"), confirmBefore: () => {}, finished: (data) => { console.log(data); if ( !(function isValueChange(type = "auto") { const param = cfg.param, curData = getAllData(), curDataStr = JSON.stringify(curData); let oldDataStr; return ( "auto" === type && ("interval" === cfg.state && param.isIntervalRun && (type = "interval_current"), "finished" === cfg.state && (type = "auto")), (oldDataStr = "interval_current" === type ? JSON.stringify(cfg.lastData) : "base_current" === type ? JSON.stringify(cfg.baseData) : JSON.stringify(cfg.oldData)), "{}" !== oldDataStr && curDataStr !== oldDataStr ); })() ) return; const changeObj = {}; for (const key in settings) changeObj[key] = !1; for (const pageName in data) { const page = data[pageName]; for (const key in page) { const value = page[key]; let verifyFn; const flag = key.replace(info.keyBase, ""), item = settings[flag]; switch (key) { case settings.homeDelAdTime.key: verifyFn = (newVal) => (newVal = +newVal) >= 0 ? newVal : settings.homeDelAdTime.base; break; case settings.addVideoNum.key: verifyFn = (newVal) => (newVal = +newVal) >= 0 ? newVal : settings.addVideoNum.base; break; case settings.videoNumRule.key: verifyFn = (newVal) => newVal.replaceAll(" ", "").toLowerCase(); break; case settings.delClassArr.key: verifyFn = (newVal) => newVal.replaceAll(" ", ""); break; case settings.minVideoTime.key: verifyFn = (newVal, oldVal) => { if ("" === newVal) return ""; try { timeStringToSeconds(newVal); return newVal; } catch (e) { return oldVal; } }; break; case settings.video_delSetting.key: verifyFn = (newVal) => newVal.replaceAll(" ", ""); } if (!item) return void console.log("设置的数据对应的对象获取失败"); changeObj[flag] = setValue({ value, base: item.base, key, verification: verifyFn, getValue: GM_getValue, setValue: GM_setValue, }); } } getData(); const changeArr = (function getTrueArr(obj) { const arr = []; for (const key in obj) obj[key] && arr.push(key); return arr; })(changeObj); if ("视频" === info.pageName) (changeArr.includes("video_delSetting") || changeArr.includes("video_isDelAd")) && (videoObj.updateData(), videoObj.getAd(), settings.video_isDelAd.value ? videoObj.delAd() : videoObj.showAd()), changeArr.includes("video_isAutoLike") && videoObj.autoLike(); else if ("首页" === info.pageName) { (function hasCommonItems(arr1, arr2) { for (let i = 0; i < arr1.length; i++) if (arr2.includes(arr1[i])) return !0; return !1; })(changeArr, [ "isCarousel", "isAutoLayout", "isClearAd", "isTrueEnd", "isLoadOne", ]) ? main(!1) : (updateData(), changeArr.includes("isReorderVideo") && setStyle(), changeArr.includes("videoNumRule") && (function rearrange(rule) { (videoNumRule = getVideoNumRule(rule)), zoomPage(), setStyle(); })(settings.videoNumRule.value), (changeArr.includes("delClassArr") || changeArr.includes("minVideoTime")) && (function updateVideo(delClass) { (delClassArr = getDelClassArr(delClass)), delAd(getAd(queryNum, delClassArr), !1, !1); })(settings.delClassArr.value), changeArr.includes("isDelUpBtn") && settings.isDelUpBtn.value && (createDelUpBtn(), addDelUpBtnCss(), bindDelUpBtnEvent()), changeArr.includes("delOtherAd") && delOtherAd()); } }, }; showEditArea(!0, callback); } let pageName = (function getBiliPageType() { const url = window.location.href, hostname = window.location.hostname, pathname = window.location.pathname; return "www.bilibili.com" !== hostname || ("/" !== pathname && "/index.html" !== pathname) ? url.includes("www.bilibili.com/video") || url.includes("www.bilibili.com/list") ? "视频" : "search.bilibili.com" === hostname ? "搜索" : "space.bilibili.com" === hostname ? "主页" : url.includes("t.bilibili.com") || url.includes("www.bilibili.com/opus") ? "动态" : url.includes("www.bilibili.com/read") ? "专栏" : url.includes("www.bilibili.com/bangumi") ? "番剧" : "live.bilibili.com" === hostname ? "直播" : "message.bilibili.com" === hostname && "/" === pathname ? "消息" : "其他" : "首页"; })(window.location); if ( ((info.pageName = pageName), ("首页" !== pageName && "视频" !== pageName) || (getData(), GM_registerMenuCommand("设置", () => { showSettings(); }), GM_registerMenuCommand("脚本主页", () => { !(function openUrl(url) { let tempALink = document.createElement("a"); tempALink.setAttribute("target", "_blank"), tempALink.setAttribute("id", "openWin"), tempALink.setAttribute("href", url), document.body.appendChild(tempALink), document.getElementById("openWin").click(), document.body.removeChild(tempALink), tempALink.remove(); })(info.scriptUrl); })), "首页" === pageName) ) main(); else if ("视频" === pageName) { setTimeout(() => { videoObj.init(); }, videoObj.info.startTime); let url = window.location.href; const pathInterval = videoObj.info.pathInterval; setInterval(() => { url !== window.location.href && ((url = window.location.href), setTimeout(() => { videoObj.init(); }, videoObj.info.startTime)); }, pathInterval); } })();