您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
RT
此脚本不应直接安装。它是供其他脚本使用的外部库,要使用该库请加入元指令 // @require https://update.gf.qytechs.cn/scripts/438161/1509124/DoubanSearch.js
// ==UserScript== // @name DoubanSearch // @namespace http://tampermonkey.net/ // @version 1.0 // @description RT // @author Pidanmeng // @grant GM_openInTab // @match * // @include * // @grant none // ==/UserScript== //豆瓣电影搜索助手 // 配置参数 let ev = null; var text = window.getSelection().toString().trim(); document.addEventListener('mouseup', function (e) { ev = e;//划词鼠标结束位置 }); const SettingOptions = { defaultsearchengine: "db", // 默认搜索引擎 searchPattern: "automatic", // 搜索模式 selectPattern: "select", // 划词模式 selectKey: "Ctrl", // 划词键 selectIconPosition: "right", // 划词图标位置 }; // 图标 const Images = { IconBase64: '', DbSvg: '<svg t="1634541962091" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2799" width="16" height="16"><path d="M701.44 519.168l-382.976 0 0-191.488 382.976 0 0 191.488zM877.568 69.632q34.816 0 59.392 24.064t24.576 59.904l0 731.136q0 34.816-24.576 59.392t-59.392 24.576l-732.16 0q-34.816 0-58.88-24.576t-24.064-59.392l0-731.136q0-35.84 24.064-59.904t58.88-24.064l732.16 0zM187.392 197.632l648.192 0 0-63.488-648.192 0 0 63.488zM253.952 263.168l0 318.464 512 0 0-318.464-512 0zM857.088 774.144l-176.128 0 62.464-111.616-70.656-53.248q-5.12 12.288-9.216 23.552t-9.728 23.552-11.776 23.552q-17.408 29.696-31.744 57.856t-19.456 36.352l-158.72 0q-4.096-8.192-18.432-36.352t-31.744-57.856q-7.168-11.264-12.288-23.552t-9.216-23.552-9.216-23.552l-70.656 53.248 62.464 111.616-176.128 0 0 65.536 690.176 0 0-65.536z" p-id="2800" fill="#4da64d"></path></svg>', ImdbSvg: '<svg t="1634544643843" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3906" width="16" height="16"><path d="M864 64H160C107 64 64 107 64 160v704c0 53 43 96 96 96h704c53 0 96-43 96-96V160c0-53-43-96-96-96zM106.6 458.4H106c0.2-0.2 0.4-0.6 0.6-0.8zM258 639.6H192V384h66z m226.4 0h-57.4v-172.8l-23.2 172.8h-41.2l-24.4-169v169h-58V384h85.6c6.6 39.6 12 79.8 17.4 119.8l15.2-119.8h86z m22.8 0V384h49.2c35.2 0 89.4-3.2 98 41.8 3.4 15.2 2.8 32.6 2.8 48.8 0 177 22.2 165.2-150 165z m321.8-58.4c0 31.4-4.8 61.8-44.4 61.8-18 0-30.4-6-41.8-19.6l-3.8 16.2h-59.6V384h63.4v83.4c12-13 24-18.4 41.8-18.4 42.8 0 44.4 25.6 44.4 60.2zM594 459.8c0-19.4 3.2-32-20.6-32v167.4c24.4 0.6 20.6-17.4 20.6-36.8z m171 52.2c0-10.8 2.2-25.4-12.4-25.4-12 0-9.8 17.8-9.8 25.4 0 1.2-2.2 79.2 2.2 89.4 1.6 3.2 4.4 4.8 7.6 4.8 15.6 0 12.4-18 12.4-28.8z" p-id="3907" fill="#f4ea2a"></path></svg>', MovieIconBase64: '', BookIconBase64: '', VideoDefaultImg: '', } // 链接 const Urls = { DbMoviePageUrl: 'https://movie.douban.com', ImDbVideoPageUrl: 'https://www.imdb.com', DbVideoInfoPageUrl: 'https://movie.douban.com/subject/{subjectId}', // 豆瓣视频详情页面 DbBookPageUrl: 'https://book.douban.com', DbBookInfoPageUrl: 'https://book.douban.com/subject/{subjectId}', // 豆瓣读书详情页面 ImdbVideoInfoPageUrl: 'https://www.imdb.com/title/{imdbId}', // imdb 视频详情页面 DbVideoSearchResultPageUrl: 'https://www.douban.com/search?cat=1002&q={title}', // 豆瓣电影搜索结果页面 DbBookSearchResultPageUrl: 'https://www.douban.com/search?cat=1001&q={title}', // 豆瓣读书搜索结果页面 MovieSearchResultPageUrl: 'https://movie.douban.com/subject_search?search_text={title}', BookSearchResultPageUrl: 'https://book.douban.com/subject_search?search_text={title}', DbVideoSearchApiUrl: 'https://www.douban.com/j/search?q={title}&cat=1002', // 豆瓣电影官方搜索接口 DbBookSearchApiUrl: 'https://www.douban.com/j/search?q={title}&cat=1001', // 豆瓣读书官方搜索接口 IframePageHost: 'https://yyy.rth1.me', // 取词遮罩地址 使用的热铁盒网页托管 ImgHandleUrl: 'https://images.weserv.nl/?url={url}', // 图片处理接口 可以缓存、修改图片尺寸等 本脚本用来防止图片跨域 DbImdbApiUrl: 'https://movie.querydata.org/api?id={subjectId}', // 可以获取豆瓣-imdb-烂番茄信息的聚合接口 api来自https://github.com/iiiiiii1/douban-imdb-api } const Logger = { debug: console.debug, log: console.log, info: console.info, warn: console.warn, error: console.error } const Utils = { /** * 字符串模板格式化 * @param {string} formatStr - 字符串模板 * @returns {string} 格式化后的字符串 * @example * Utils.StringFormat("ab{0}c{1}ed",1,"q") output "ab1cqed" */ StringFormat: function (formatStr) { let args = arguments; return formatStr.replace(/\{(\d+)\}/g, function (m, i) { i = parseInt(i); return args[i + 1]; }); }, /** * 日期格式化 * @param {Date} date - 日期 * @param {string} formatStr - 格式化模板 * @returns {string} 格式化日期后的字符串 * @example * Utils.DateFormat(new Date(),"yyyy-MM-dd") output "2020-03-23" * @example * Utils.DateFormat(new Date(),"yyyy/MM/dd hh:mm:ss") output "2020/03/23 10:30:05" */ DateFormat: function (date, formatStr) { let o = { "M+": date.getMonth() + 1, //月份 "d+": date.getDate(), //日 "h+": date.getHours(), //小时 "m+": date.getMinutes(), //分 "s+": date.getSeconds(), //秒 "q+": Math.floor((date.getMonth() + 3) / 3), //季度 "S": date.getMilliseconds() //毫秒 }; if (/(y+)/.test(formatStr)) { formatStr = formatStr.replace(RegExp.$1, (date.getFullYear() + "").substr(4 - RegExp.$1.length)); } for (let k in o) { if (new RegExp("(" + k + ")").test(formatStr)) { formatStr = formatStr.replace( RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length))); } } return formatStr; }, // 获取配置参数 GetSettingOptions: function () { let optionsJson = GM_getValue("db-search-options") || ""; if (optionsJson != "") { let optionsData = JSON.parse(optionsJson); for (let key in SettingOptions) { if (SettingOptions.hasOwnProperty(key) && optionsData.hasOwnProperty(key)) { SettingOptions[key] = optionsData[key]; } } } return SettingOptions; }, // 设置配置参数 SetSettingOptions: function () { let optionsJson = JSON.stringify(SettingOptions); GM_setValue("db-search-options", optionsJson); }, /** * 人员格式化 * @param {Array} personList - 人员数组 * @param {Number} len - 需要的数量 * @returns {String} 格式化后的字符串 * @example * Utils.FmtDbPerson([{name: '张 zhang'},{name: '王 wang'},{name: '李 li'}],2) output "张/王" * Utils.FmtDbPerson([{data: [{name: '张'}]},{data: [{name: '王'}]},{data: [{name: '李'}]}],2) output "张/王" */ FmtDbPerson: function (personList, len) { if (personList && personList.length > 0) { if (len && personList.length > len) { personList = personList.slice(0, len); } let nameArr = []; personList.forEach((item) => { if (item.hasOwnProperty('data')) { // db2使用 nameArr.push(item.data[0].name); } else { nameArr.push(item.name.split(' ')[0]); // db使用 } }) return nameArr.join(' / '); } return ''; }, /** * 解析视频列表 dom 元素 * @param {Document} htmlDoc - 视频列表dom * @param {String} selector - 选择器 * @returns {[VideoInfo]} 视频列表数组 * @example * Utils.ParseVideoListDom(document.getElementById('a'),'li') output [{videoInfo},{videoInfo},...] */ ParseVideoListDom: function (htmlDoc, selector) { let videoList = []; let bookList = []; if (htmlDoc) { let liDoc = $(htmlDoc).find(selector); liDoc.each((index, item) => { let titleADom = $(item).find('.title a'); let title = titleADom.html() || ''; let onclickStr = titleADom.attr("onclick"); // moreurl(this,{i: '14', query: 'hh', from: 'dou_search_movie', sid: 25741647, qcat: '1002'}) let id = 0; if (onclickStr) { id = onclickStr.split(',')[4].replace(/[^0-9]/ig, ''); } let image = $(item).find('img').attr("src"); let cast = ($(item).find('.subject-cast').html() || '').replace('/ / ', '/').replace(/原名:(.*?)\//g, ''); let socreDoc = $(item).find('.rating_nums'); let score = socreDoc.html() || ''; let ratingCount = ''; if (score && socreDoc.next() && socreDoc.next().html()) { let count = socreDoc.next().html().replace(/[^0-9]/ig, ''); if (count) { ratingCount = `${count} 人参与评分`; } } let videoInfo = { subjectId: id, image: Urls.ImgHandleUrl.replace('{url}', encodeURIComponent(image)), url: Urls.DbVideoInfoPageUrl.replace('{subjectId}', id), description: $(item).find('p').html() || '', title, score, ratingCount, cast: cast, } videoList.push(videoInfo); let bookInfo = { subjectId: id, image: Urls.ImgHandleUrl.replace('{url}', encodeURIComponent(image)), url: Urls.DbBookInfoPageUrl.replace('{subjectId}', id), description: $(item).find('p').html() || '', title, score, ratingCount, cast: cast, } bookList.push(bookInfo); }); } return videoList; }, /** * 解析String 为 dom 元素 * @param {String} text - 文档字符 * @returns {Document} 文档对象 * @example * Utils.ParseDomFromString('<div>a</div>') output Document */ ParseDomFromString: function (text) { if (!text) { return new Document(); } let parser = new DOMParser(); return parser.parseFromString(text, "text/html"); }, //查找第5次出现/索引 findStrIndex: function(str, cha, num) { var x = str.indexOf(cha); for (var i = 0; i < num-1; i++) { x = str.indexOf(cha, x + 1); } return x; }, } // 随机因子 用于元素属性后缀 以防止属性名称重复 const randomCode = Utils.DateFormat(new Date(), "yyMM").toString() + (Math.floor(Math.random() * 900000) + 100000).toString(); // 豆瓣搜索引擎1 爬虫模式 const DoubanMovieSearchByDom = { code: "db", codeText: "豆瓣", SearchVideoList: function (title) { return new Promise((resolve, reject) => { if (!title) { resolve([]); } GM_xmlhttpRequest({ method: "GET", url: Urls.DbVideoSearchResultPageUrl.replace('{title}', encodeURIComponent(title)), onload: function (response) { let videoList = []; if (response.status == 200) { let htmlDoc = Utils.ParseDomFromString(response.responseText); videoList = Utils.ParseVideoListDom(htmlDoc, '.result-list .result'); } else { Logger.warn('查询失败,title:' + title, response.statusText); } resolve(videoList); } }); }) }, SearchVideoInfo: function (subjectId) { let self = this; return new Promise((resolve, reject) => { if (!subjectId) { resolve(null); } let url = Urls.DbVideoInfoPageUrl.replace('{subjectId}', subjectId); GM_xmlhttpRequest({ method: "GET", url: url, onload: function (response) { if (response.status == 200) { let doubanInfo = {}; let htmlDoc = Utils.ParseDomFromString(response.responseText); if (htmlDoc) { let ldJsonDb = $(htmlDoc).find('script[type="application/ld+json"]'); let ldJsonDbOthers = $(htmlDoc).find('meta[name="keywords"]'); //中文标题 let ldJsonDbScri = $(htmlDoc).find('span[property="v:summary"]'); let ldJsonDbAttrs = $(htmlDoc).find('span[class="attrs"]'); let ldJsonDbDirector = ldJsonDbAttrs[0].innerText; if(ldJsonDbAttrs[2]) { if(ldJsonDbAttrs[2].innerText.split('/').length > 7) { ldJsonDbAttrs = ldJsonDbAttrs[2].innerText.slice(0, Utils.findStrIndex(ldJsonDbAttrs[2].innerText, "/", 6)) + "/ 更多…" } else { ldJsonDbAttrs = ldJsonDbAttrs[2].innerText; } } else { ldJsonDbAttrs =ldJsonDbAttrs[1].innerText; } if (ldJsonDb && ldJsonDb.length > 0 && ldJsonDb[0].innerText) { try { doubanInfo = JSON.parse(ldJsonDb[0].innerText.replace(/\r\n/g, '').replace(/\n/g, '')); //console.log(htmlDoc); let videoInfo = { //title: doubanInfo.name, title: ldJsonDbOthers[0].content.split(',')[0], image: Urls.ImgHandleUrl.replace('{url}', encodeURIComponent(doubanInfo.image)), url: Urls.DbMoviePageUrl + doubanInfo.url, //description: doubanInfo.description, description: ldJsonDbScri[0].innerText.replace(/\r\n/g, '').replace(/\n/g, '').trim(), //director: Utils.FmtDbPerson(doubanInfo.director, 2), director: ldJsonDbDirector, //actor: Utils.FmtDbPerson(doubanInfo.actor, 6), actor: ldJsonDbAttrs, score: doubanInfo.aggregateRating.ratingValue, ratingCount: doubanInfo.aggregateRating.ratingCount, genre: doubanInfo.genre.join(' / '), time: doubanInfo.datePublished, } // 获取imdbId let imdb_anchor = $(htmlDoc).find('#info span.pl:contains("IMDb")'); if (imdb_anchor && imdb_anchor.length > 0) { let imdbId = imdb_anchor[0].nextSibling.nodeValue.trim(); if (imdbId) { videoInfo.imdbId = imdbId; videoInfo.imdbUrl = Urls.ImdbVideoInfoPageUrl.replace('{imdbId}', imdbId); self.SearchImdbRating(imdbId); } } resolve(videoInfo); } catch (e) { Logger.log('解析失败', ldJsonDb[0].innerText, e) } } } } else { Logger.warn(response.statusText); } resolve(null); } }); }); }, // 查询imdb评分 直接替换dom 不是太好的解决方案 SearchImdbRating: function (imdbId) { GM_xmlhttpRequest({ method: "GET", url: Urls.ImdbVideoInfoPageUrl.replace('{imdbId}', imdbId), onload: function (response) { if (response.status == 200) { let imdbHtmlDoc = Utils.ParseDomFromString(response.responseText); let ldJsonImdb = $(imdbHtmlDoc).find('head > script[type="application/ld+json"]'); if (ldJsonImdb && ldJsonImdb.length > 0 && ldJsonImdb[0].innerText) { let select = Utils.StringFormat('.videoInfo{0} #imdbScore{0}_' + imdbId, randomCode); let imdbDom = $(select); try { let imdbInfo = JSON.parse(ldJsonImdb[0].innerText.replace(/\r\n/g, '').replace(/\n/g, '')); let imdbScore = ''; let imdbRatingCount = ''; if (imdbInfo && imdbInfo.aggregateRating) { let imdbRating = imdbInfo.aggregateRating.ratingValue; //imdbRatingCount = imdbInfo.aggregateRating.ratingCount || 0; imdbRatingCount = imdbInfo.aggregateRating.ratingCount || ''; imdbScore = imdbRating ? imdbRating.toFixed(1) : ''; } // 替换评分 if (imdbDom) { if (imdbScore) { //imdbDom.removeClass(Utils.StringFormat('loading{0}', randomCode)); imdbDom.html(imdbScore); //imdbDom.html(imdbScore + " " + imdbRatingCount); //imdbDom.parents('a').attr('title', `${imdbRatingCount}`); imdbDom.next().html("(" + imdbRatingCount + ")"); //console.log(imdbDom.next().text()); } } } catch (e) { Logger.log('解析失败', ldJsonImdb[0].innerText, e); if (imdbDom) { imdbDom.parents('a').remove(); } } } } } }) } }; //豆瓣搜索引擎2 使用api 接口评分不太准确 有imdb评分 const DoubanSearchByApi = { code: "db2", codeText: "豆瓣2", SearchVideoList: function (title) { return new Promise((resolve, reject) => { if (!title) { return resolve([]); } GM_xmlhttpRequest({ method: "GET", url: Urls.DbVideoSearchApiUrl.replace('{title}', encodeURIComponent(title)), onload: function (response) { let videoList = []; if (response.status == 200) { let responseText = response.responseText; let fmtText = responseText.replace(/\r\n/g, '').replace(/\n/g, '').replace(/\\/g, ''); let match = fmtText.match(/{"items":(.*?),"total":(\d+),"limit":(\d+),"more":(.*?)}/); if (match && match[1]) { let htmlDoc = Utils.ParseDomFromString(match[1]); videoList = Utils.ParseVideoListDom(htmlDoc, '.result'); } } else { Logger.warn(response.statusText); } resolve(videoList); } }); }) }, SearchVideoInfo: function (subjectId) { return new Promise((resolve, reject) => { if (!subjectId) { resolve(null); } let url = Urls.DbImdbApiUrl.replace('{subjectId}', subjectId); GM_xmlhttpRequest({ method: "GET", url: url, onload: function (response) { if (response.status == 200) { let doubanInfo = JSON.parse(response.responseText); if (doubanInfo && doubanInfo.data) { let data = doubanInfo.data[0]; let videoInfo = { title: data.name, image: data.poster, url: Urls.DbVideoInfoPageUrl.replace('{subjectId}', doubanInfo.doubanId), //description: data.description.slice(0, 100) + "…", description: data.description, director: Utils.FmtDbPerson(doubanInfo.director, 2), actor: Utils.FmtDbPerson(doubanInfo.actor, 6), score: doubanInfo.doubanRating, ratingCount: doubanInfo.doubanVotes, imdbScore: doubanInfo.imdbRating, imdbRatingCount: doubanInfo.imdbVotes, imdbUrl: Urls.ImdbVideoInfoPageUrl.replace('{imdbId}', doubanInfo.imdbId), genre: data.genre, time: Utils.DateFormat(new Date(doubanInfo.dateReleased), "yyyy-MM-dd"), } resolve(videoInfo); } } else { Logger.warn(response.statusText); } resolve(null); } }); }); }, }; // 豆瓣读书搜索引擎1 爬虫模式 const DoubanBookSearchByDom = { code: "db", codeText: "豆瓣", SearchBookList: function (title) { return new Promise((resolve, reject) => { if (!title) { resolve([]); } GM_xmlhttpRequest({ method: "GET", url: Urls.DbBookSearchResultPageUrl.replace('{title}', encodeURIComponent(title)), onload: function (response) { let bookList = []; if (response.status == 200) { let htmlDoc = Utils.ParseDomFromString(response.responseText); bookList = Utils.ParseVideoListDom(htmlDoc, '.result-list .result'); } else { Logger.warn('查询失败,title:' + title, response.statusText); } resolve(bookList); } }); }) }, SearchBookInfo: function (subjectId) { let self = this; return new Promise((resolve, reject) => { if (!subjectId) { resolve(null); } let url = Urls.DbBookInfoPageUrl.replace('{subjectId}', subjectId); GM_xmlhttpRequest({ method: "GET", url: url, onload: function (response) { if (response.status == 200) { let doubanInfo = {}; let htmlDoc = Utils.ParseDomFromString(response.responseText); if (htmlDoc) { let ldJsonDb = $(htmlDoc).find('script[type="application/ld+json"]'); let ldJsonDbImg = $(htmlDoc).find('meta[property="og:image"]'); let ldJsonDbScri = $(htmlDoc).find('meta[property="og:description"]'); let ldJsonDbAuth = $(htmlDoc).find('meta[property="book:author"]'); let ldJsonDbOthers = $(htmlDoc).find('meta[name="keywords"]'); let ldJsonDbScore = $(htmlDoc).find('strong[property="v:average"]'); let ldJsonDbCount = $(htmlDoc).find('span[property="v:votes"]'); if (ldJsonDb && ldJsonDb.length > 0) { try { doubanInfo = JSON.parse(ldJsonDb[0].innerText.replace(/\r\n/g, '').replace(/\n/g, '')); let bookInfo = { title: doubanInfo.name, image: Urls.ImgHandleUrl.replace('{url}', encodeURIComponent(ldJsonDbImg[0].content)), url: doubanInfo.url, //description: ldJsonDbScri[0].content.slice(0, 100) + "…", description: ldJsonDbScri[0].content, author: ldJsonDbAuth[0].content, score: ldJsonDbScore[0].innerText, ratingCount: ldJsonDbCount[0].innerText, //genre: doubanInfo.genre.join(' / '), publisher: ldJsonDbOthers[0].content.split(',简介')[0].split(',')[ldJsonDbOthers[0].content.split(',简介')[0].split(',').length-2].split(',')[0], orignal: '', translator: '', time: ldJsonDbOthers[0].content.split(',简介')[0].split(',')[ldJsonDbOthers[0].content.split(',简介')[0].split(',').length-1], } resolve(bookInfo); } catch (e) { Logger.log('解析失败', ldJsonDb[0].innerText, e) } } } } else { Logger.warn(response.statusText); } resolve(null); } }); }); }, }; const SearchMovie = { searchEngineList: {}, // 搜索引擎实例列表 searchEngine: "", // 当前搜索引擎。 db:豆瓣 searchEngineObj: {}, // 当前搜索引擎实例 searchText: "", // 被搜索内容 searchVideoList: [], // 当前搜索视频列表 searchVideoInfo: null, // 当前搜索视频内容 searchSelectTitle: '', // 列表选中的视频标题 Execute: function (h_onloadfn) { this.ResetSearchResult(); let title = this.searchText; this.searchEngineObj.SearchVideoList(title).then((videoList) => { this.searchVideoList = videoList; if (SettingOptions.searchPattern == 'automatic') { if (videoList && videoList.length > 0) { let subjectId = videoList[0].subjectId // 如果在列表中选择过视频,切换引擎后重新选中该视频 if (this.searchSelectTitle) { try { videoList.forEach((v, i) => { if (v && v.title == this.searchSelectTitle) { subjectId = v.subjectId; throw new Error('EndForEach'); } }) } catch (e) { if (e.message != 'EndForEach') { Logger.error(e); } } } this.searchEngineObj.SearchVideoInfo(subjectId).then((result) => { this.searchVideoInfo = result; h_onloadfn(); }); } else { h_onloadfn(); } } else if (SettingOptions.searchPattern == 'manual') { h_onloadfn(); } }); }, UpdateVideoInfo: function (subjectId, h_onloadfn) { this.searchVideoInfo = null; this.searchEngineObj.SearchVideoInfo(subjectId).then((result) => { this.searchVideoInfo = result; h_onloadfn(); }) }, Update: function () { this.ResetSearchResult(); this.searchEngineObj = this.searchEngineList[this.searchEngine]; }, Clear: function () { this.searchEngine = ""; this.searchText = ""; this.searchVideoList = []; this.searchVideoInfo = null; }, ResetSearchResult: function () { this.searchVideoInfo = null; this.searchVideoList = []; }, //注册(不可用)搜索引擎接口并执行搜索引擎的初始化接口 RegisterEngine: function () { /** * 搜索引擎必须提供以下接口 code:"", // 代号 codeText:"", // 代号描述 SearchVideoList: function (title) {}, // 返回视频列表 SearchVideoInfo: function (subjectId) {}, // 返回视频信息 init:function(){}, // 可选,初始化接口,在脚本创建时立即执行 */ const searchEngineListObj = {}; searchEngineListObj[DoubanMovieSearchByDom.code] = DoubanMovieSearchByDom; searchEngineListObj[DoubanSearchByApi.code] = DoubanSearchByApi; this.searchEngineList = searchEngineListObj; for (let key in this.searchEngineList) { if (this.searchEngineList.hasOwnProperty(key) && this.searchEngineList[key].hasOwnProperty("init")) { this.searchEngineList[key].init(); } } } }; const SearchBook = { searchEngineList: {}, // 搜索引擎实例列表 searchEngine: "", // 当前搜索引擎。 db:豆瓣 searchEngineObj: {}, // 当前搜索引擎实例 searchText: "", // 被搜索内容 searchBookList: [], // 当前搜索视频列表 searchBookInfo: null, // 当前搜索视频内容 searchSelectTitle: '', // 列表选中的视频标题 Execute: function (h_onloadfn) { this.ResetSearchResult(); let title = this.searchText; this.searchEngineObj.SearchBookList(title).then((bookList) => { this.searchBookList = bookList; if (SettingOptions.searchPattern == 'automatic') { if (bookList && bookList.length > 0) { let subjectId = bookList[0].subjectId // 如果在列表中选择过视频,切换引擎后重新选中该视频 if (this.searchSelectTitle) { try { bookList.forEach((v, i) => { if (v && v.title == this.searchSelectTitle) { subjectId = v.subjectId; throw new Error('EndForEach'); } }) } catch (e) { if (e.message != 'EndForEach') { Logger.error(e); } } } this.searchEngineObj.SearchBookInfo(subjectId).then((result) => { this.searchBookInfo = result; h_onloadfn(); }); } else { h_onloadfn(); } } else if (SettingOptions.searchPattern == 'manual') { h_onloadfn(); } }); }, UpdateBookInfo: function (subjectId, h_onloadfn) { this.searchBookInfo = null; this.searchEngineObj.SearchBookInfo(subjectId).then((result) => { this.searchBookInfo = result; h_onloadfn(); }) }, Update: function () { this.ResetSearchResult(); this.searchEngineObj = this.searchEngineList[this.searchEngine]; }, Clear: function () { this.searchEngine = ""; this.searchText = ""; this.searchBookList = []; this.searchBookInfo = null; }, ResetSearchResult: function () { this.searchBookInfo = null; this.searchBookList = []; }, //注册(不可用)搜索引擎接口并执行搜索引擎的初始化接口 RegisterEngine: function () { /** * 搜索引擎必须提供以下接口 code:"", // 代号 codeText:"", // 代号描述 SearchVideoList: function (title) {}, // 返回视频列表 SearchVideoInfo: function (subjectId) {}, // 返回视频信息 init:function(){}, // 可选,初始化接口,在脚本创建时立即执行 */ const searchEngineListObj = {}; searchEngineListObj[DoubanBookSearchByDom.code] = DoubanBookSearchByDom; this.searchEngineList = searchEngineListObj; for (let key in this.searchEngineList) { if (this.searchEngineList.hasOwnProperty(key) && this.searchEngineList[key].hasOwnProperty("init")) { this.searchEngineList[key].init(); } } } }; // 面板 const Panel = { popBoxEl: {}, Create: function (title, placement, isShowArrow, content, shownFn) { let self = this; $(self.popBoxEl).jPopBox({ title: title, className: 'JPopBox-tip-white', placement: placement, trigger: 'none', isTipHover: true, isShowArrow: isShowArrow, content: function () { return Utils.StringFormat('<div id="panelBody{0}">{1}</div>', randomCode, content); } }); $(self.popBoxEl).on("shown.jPopBox", function () { let $panel = $("div.JPopBox-tip-white"); typeof shownFn === 'function' && shownFn($panel); }); $(self.popBoxEl).jPopBox('show'); }, Update: function (Fn) { let $panel = $("div.JPopBox-tip-white"); Fn($panel); }, Destroy: function () { $(this.popBoxEl).jPopBox("destroy"); }, CreateStyle: function () { let s = ""; s += Utils.StringFormat("#panelBody{0}>div input,#panelBody{0}>div select{padding:3px;margin:0;background:#fff;font-size:14px;border:1px solid #a9a9a9;color:black;width:auto;height:25px;display:inline-block;position: static;}#panelBody{0} a{text-decoration:none;border-bottom:0;color:#494949}", randomCode); s += Utils.StringFormat("#panelBody{0} .head{0}{display:flex;align-items:center;height:30px}#panelBody{0} .logo{0}{width:75px;margin:0;vertical-align:bottom}#panelBody{0} .text{0}{margin:10px 0 0 10px;color:#388e8e;font-size:10px;cursor:pointer;opacity:0.6}#panelBody{0} .listBtn{0}{margin:10px 0 0 1px;color:#999;font-size:10px;cursor:pointer}", randomCode); s += Utils.StringFormat("#panelBody{0} .content{0} .noData{0}{margin:30px 40%}#panelBody{0} .content{0} .loading{0}{border:5px solid #f3f3f3;border-radius:50%;border-top:5px solid #3498db;width:30px;height:30px;animation:db_search_turn{0} 2s linear infinite;margin:20px;margin-left:45%}", randomCode); //s += Utils.StringFormat("#panelBody{0} .score{0}{position:absolute;top:5px;right:5px;display:flex;align-items:center;column-gap:10px}#panelBody{0} .score{0}>a{display:flex;align-items:center}#panelBody{0} .score{0} svg{background:0}#panelBody{0} .score{0} span{font-size:30px}#panelBody{0} .score{0} span.loading{0}{border:3px solid #f3f3f3;border-radius:50%;border-top:3px solid #3498db;width:20px;height:20px;animation:db_search_turn{0} 2s linear infinite;margin:5px;display:inline-block}", randomCode); s += Utils.StringFormat("#panelBody{0} .info{0} .title{0}{font-size:18px;font-weight:bold;margin:5px 0}#panelBody{0} .score{0}{position:absolute;right:15px;display:flex;align-items:center;column-gap:5px}#panelBody{0} .score{0}>a{display:flex;align-items:center}#panelBody{0} .score{0} svg{background:0;padding-bottom:1px;padding-right:3px}#panelBody{0} .score{0} .iscore{0}{font-size:18px}#panelBody{0} .score{0} .count{0}{font-size:8px;padding-top:3px;padding-left:2px;}#panelBody{0} .score{0} span.loading{0}{border:3px solid #f3f3f3;border-radius:50%;border-top:3px solid #3498db;width:20px;height:20px;animation:db_search_turn{0} 2s linear infinite;margin:5px;display:inline-block}#panelBody{0} .info{0} .left{0}{float:left;margin:3px 5px 0 0;display:block;position:relative;width:120px;height:168px;background-repeat:no-repeat;background-position:50%;background-image:url({1});background-size:100%}#panelBody{0} .info{0} .left{0}>img{width:120px;height:168px}#panelBody{0} .info{0} .right{0}{min-height:175px;padding-left:132px}#panelBody{0} .info{0} .right{0} .item{0}>span{color:#666}", randomCode, Images.VideoDefaultImg); //s += Utils.StringFormat("#panelBody{0} .list{0}{overflow:auto;height:200px;margin-top:5px}#panelBody{0} .list{0}::-webkit-scrollbar{display:none}#panelBody{0} .list{0} .listItem{0}{margin-top:5px;height:70px;position:relative}#panelBody{0} .listItem{0} .left{0}{float:left;margin-right:5px;display:block;position:relative;width:48px;height:68px;background-repeat:no-repeat;background-position:50%;background-image:url({1});background-size:100%}#panelBody{0} .listItem{0} .left{0}>img{width:48px;height:68px}#panelBody{0} .listItem{0} .right{0} .title{0}{width:320px;font-size:18px;margin-bottom:9px;font-weight:bold;white-space:nowrap;text-overflow:ellipsis;overflow:hidden;word-break:break-all}#panelBody{0} .listItem{0} .right{0} .score{0}{position:absolute;right:5px;top:0;font-size:25px}#panelBody{0} .listItem{0} .right{0} .info{0}{white-space:nowrap;text-overflow:ellipsis;overflow:hidden;word-break:break-all;font-size:12px;border:0}", randomCode, Images.VideoDefaultImg); s += Utils.StringFormat("#panelBody{0} .list{0}{overflow:auto;height:200px;margin-top:5px}#panelBody{0} .list{0}::-webkit-scrollbar{display:none}#panelBody{0} .list{0} .listItem{0}{margin-top:5px;height:70px;position:relative;padding-top:5px;padding-bottom:5px;}#panelBody{0} .listItem{0} .left{0}{float:left;margin-right:5px;display:block;position:relative;width:48px;height:68px;background-repeat:no-repeat;background-position:50%;background-image:url({1});background-size:100%}#panelBody{0} .listItem{0} .left{0}>img{width:48px;height:68px}#panelBody{0} .listItem{0} .right{0}{padding-left:60px}#panelBody{0} .listItem{0} .right{0} .title{0}{width:320px;font-size:13px;margin-bottom:9px;/*font-weight:bold;*/white-space:nowrap;text-overflow:ellipsis;overflow:hidden;word-break:break-all;color:#3377aa}#panelBody{0} .listItem{0} .right{0} .score{0}{position:absolute;right:5px;top:0px;padding-top:5px;font-size:13px;color:#e09015}#panelBody{0} .listItem{0} .right{0} .info{0}{white-space:nowrap;text-overflow:ellipsis;overflow:hidden;word-break:break-all;font-size:12px;border:0;color:#9d9d9d}", randomCode, Images.VideoDefaultImg); return s; } }; // 电影搜索面板 const MovieSearchPanel = { Create: function (popBoxEl) { var text = window.getSelection().toString().trim(); let self = this; let searchEngineOptionsHtml = ""; for (let k in SearchMovie.searchEngineList) { if (SearchMovie.searchEngineList.hasOwnProperty(k)) { let v = SearchMovie.searchEngineList[k].codeText; let selectOption = ""; if (SearchMovie.searchEngine == k) { selectOption = 'selected="selected"'; } searchEngineOptionsHtml += Utils.StringFormat('<option value="{0}" {2}>{1}</option>', k, v, selectOption); } } let wordSearchPanelHtml = ''; //let headHtml = Utils.StringFormat(`<div class="head{0}"><a href="" class="link22" target="_blank"><img class="logo{0}" src="" title="" /></a><select class="select22">{1}</select><div id="showSearchList_{0}" class="listBtn{0}">展示搜索列表</div></div>`, randomCode, searchEngineOptionsHtml); //let headHtml = Utils.StringFormat(`<div class="head{0}"><a href="${Urls.MovieSearchResultPageUrl.replace('{title}', encodeURIComponent(text))}" class="link22" target="_blank"><img id="doubanLogo_{0}" class="logo{0}" src="${Images.MovieIconBase64}" title="豆瓣电影" /></a><div id="showSearchText_{0}" class="text{0}">${text}</div><div id="showSearchList_{0}" class="listBtn{0}">展示搜索列表</div></div>`, randomCode, searchEngineOptionsHtml); let headHtml = Utils.StringFormat(`<div class="head{0}"><img id="doubanLogo_{0}" class="logo{0}" src="${Images.MovieIconBase64}" title="豆瓣电影" /><div id="showSearchText_{0}" class="text{0}">${text}</div><div id="showSearchList_{0}" class="listBtn{0}">展示搜索列表</div></div>`, randomCode, searchEngineOptionsHtml); wordSearchPanelHtml += headHtml; wordSearchPanelHtml += Utils.StringFormat('<div class="content{0}">', randomCode); if (SettingOptions.searchPattern == 'automatic') { wordSearchPanelHtml += self.GetVideoInfoHtml(); } else if (SettingOptions.searchPattern == 'manual') { wordSearchPanelHtml += self.GetVideoListHtml(); } wordSearchPanelHtml += '</div>'; Panel.popBoxEl = popBoxEl; Panel.Create("", "auto bottom", false, wordSearchPanelHtml, function ($panel) { console.log('pageX:' + ev.pageX + ' pageY:' + ev.pageY) console.log('clientX:' + ev.clientX + ' clientY:' + ev.clientY) //设置搜索结果面板鼠标位置 /*if(ev.pageX < 945) { $panel.css({ left: ev.pageX + 'px', top: ev.pageY + 12 + 'px' }); } else { $panel.css({ left: 945 + 'px', top: ev.pageY + 12 + 'px' }); }*/ if(ev.pageX < 945 && ev.clientY < 475) { // if(ev.pageX < 945 && ev.pageY < 485) { $panel.css({ left: ev.pageX + 'px', top: ev.pageY + 12 + 'px' }); // } else if(ev.pageX < 945 && ev.clientY > 485) { } else if(ev.pageX < 945 && ev.pageY > 475) { $panel.css({ left: ev.pageX + 'px', // top: ev.pageY - 300 - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + 'px' top: ev.pageY - 265 + 'px' }); } else if(ev.pageX > 945 && ev.clientY < 475) { // } else if(ev.pageX > 945 && ev.pageY < 485) { $panel.css({ left: 945 + 'px', top: ev.pageY + 12 + 'px' }); // } else if(ev.pageX > 945 && ev.clientY > 485) { } else if(ev.pageX > 945 && ev.pageY > 475) { $panel.css({ left: 945 + 'px', // top: ev.pageY - 300 - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + 'px' top: ev.pageY - 265 + 'px' }); /*} else { $panel.css({ left: 945 + 'px', top: ev.pageY + 12 + 'px' });*/ } // 搜索引擎 $panel.find(Utils.StringFormat("#panelBody{0} div:eq(0) select:eq(0)", randomCode)).change(function (e) { SearchMovie.searchEngine = $(this).find("option:selected").val(); self.Loading($panel); SearchMovie.Update(); SearchMovie.Execute(function () { if (SettingOptions.searchPattern == 'automatic') { self.Update(); } else if (SettingOptions.searchPattern == 'manual') { self.ShowSearchList(); } }); }); // 搜索列表 $panel.find(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).click(function (e) { self.ShowSearchList(); }); $panel.find(Utils.StringFormat("#panelBody{0} .logo{0}", randomCode)).click(function () { GM_openInTab(Urls.MovieSearchResultPageUrl.replace('{title}', encodeURIComponent(text)), { active: false }); //console.log("ok") }); if (SettingOptions.searchPattern == 'manual') { // 列表点击事件 $panel.find(Utils.StringFormat("#panelBody{0} .listItem{0}", randomCode)).click(function () { let subjectId = $(this).attr('data-id'); SearchMovie.searchSelectTitle = $(this).attr('data-name'); self.Loading($panel); SearchMovie.UpdateVideoInfo(subjectId, function () { self.Update(); }); $(Utils.StringFormat("#panelBody{0} #doubanLogo_{0}", randomCode)).hide();//logo隐藏 $(Utils.StringFormat("#panelBody{0} #showSearchText_{0}", randomCode)).hide();//划词隐藏 }); } if (SettingOptions.searchPattern == 'manual' || !SearchMovie.searchVideoList || SearchMovie.searchVideoList.length == 0) { $(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).hide(); } }); }, Update: function () { let self = this; Panel.Update(function ($panel) { let html = self.GetVideoInfoHtml(); $panel.find(Utils.StringFormat("#panelBody{0} .content{0}", randomCode)).html("").html(html); }); if (SearchMovie.searchVideoList && SearchMovie.searchVideoList.length > 0) { $(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).css("display", "inline"); $(Utils.StringFormat("#panelBody{0} #doubanLogo_{0}", randomCode)).css("display", "none");//logo隐藏 $(Utils.StringFormat("#panelBody{0} #showSearchText_{0}", randomCode)).css("display", "none");//划词隐藏 } //点title后台打开电影读书页面 //console.log(self.GetUrl()) var url = self.GetUrl(); let $panel = $("div.JPopBox-tip-white"); $panel.find(Utils.StringFormat("#panelBody{0} .title{0}", randomCode)).click(function () { GM_openInTab(url, { active: false }); //console.log("ok") }); $panel.find(Utils.StringFormat("#panelBody{0} .left{0}", randomCode)).click(function () { GM_openInTab(url, { active: false }); //console.log("ok") }); }, ShowSearchList: function () { let self = this; let html = self.GetVideoListHtml(); let $panel = $("div.JPopBox-tip-white"); $panel.find(Utils.StringFormat("#panelBody{0} .content{0}", randomCode)).html("").html(html); $panel.find(Utils.StringFormat("#panelBody{0} .listItem{0}", randomCode)).click(function () { let subjectId = $(this).attr('data-id'); SearchMovie.searchSelectTitle = $(this).attr('data-name'); self.Loading($panel); SearchMovie.UpdateVideoInfo(subjectId, function () { self.Update(); }); }); $(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).hide(); $(Utils.StringFormat("#panelBody{0} #doubanLogo_{0}", randomCode)).css("display", "inline");//列表logo显示 $(Utils.StringFormat("#panelBody{0} #showSearchText_{0}", randomCode)).css("display", "inline");//列表划词显示 }, GetVideoListHtml: function () { let htmlArr = []; let videoList = SearchMovie.searchVideoList; if (videoList && videoList.length > 0) { let itemTemplate = ` <div class="listItem{0}" data-id="{4}" data-name="{2}"> <div class="left{0}"><img src="{1}" onerror="javascript:this.src='${Images.VideoDefaultImg}'"></div> <div class="right{0}"> <div class="title{0}">{2}</div> <div class="score{0}" title="{7}">{3}</div> <div class="info{0}">{6}</div> <div class="info{0}">{5}</div> </div> </div>`; htmlArr.push(Utils.StringFormat('<div class="list{0}">', randomCode)); videoList.forEach((item) => { let videoItem = Utils.StringFormat(itemTemplate, randomCode, item.image, item.title, item.score, item.subjectId, item.description, item.cast, item.ratingCount); htmlArr.push(videoItem); }) htmlArr.push('</div>'); } else { htmlArr.push(Utils.StringFormat('<div class="noData{0}">未搜索到内容</div>', randomCode)); } return htmlArr.join(''); }, GetVideoInfoHtml: function () { let videoInfoHtml = ''; let videoInfo = SearchMovie.searchVideoInfo; if (videoInfo) { let templateArr = []; templateArr.push('<div class="videoInfo{0}">'); if (videoInfo.score || videoInfo.imdbScore || videoInfo.imdbId) { // 评分 templateArr.push('<div class="score{0}">'); if (videoInfo.score) { //templateArr.push(`<a href="{3}" target="_blank" title="{13} 人参与评分">${Images.DbSvg}<span>{5}</span></a>`); //templateArr.push(`<a href="{3}" target="_blank" title="">${Images.DbSvg}<span>{5}</span></a>`); templateArr.push(`<a href="{3}" target="_blank" class="link22" title="">${Images.DbSvg}<span class="iscore{0}">{5}</span><span class="count{0}">({13})</span></a>`); //templateArr.push(`<a href="javascript:;" onclick="widnow.open(\'{3}\')" class="link22" title="{13} 人参与评分">${Images.DbSvg}<span class="score22">{5}</span></a>`); //templateArr.push(`<a href="javascript:GM_openInTab(\'{3}\', {active: false})" class="link22" title="{13} 人参与评分">${Images.DbSvg}<span class="score22">{5}</span></a>`); } if (videoInfo.imdbScore || videoInfo.imdbId) { // imdb 有评分直接展示 没有评分有id的情况展示loading 豆瓣2不会传imdbId 用来区分 /*templateArr.push(`<a href="{11}" target="_blank" class="link22" title="{14} 人参与评分">${Images.ImdbSvg} <span class="score22" id="imdbScore{0}_{12}">{10}</span> </a>`);*/ /*templateArr.push(`<a href="{11}" target="_blank" class="link22" title="">${Images.ImdbSvg} <span class="score22" id="imdbScore{0}_{12}">{10}</span> </a>`);*/ templateArr.push(`<a href="{11}" target="_blank" class="link22" title="">${Images.ImdbSvg} <span class="iscore{0}" id="imdbScore{0}_{12}">{10}</span> <span class="count{0}">{14}</span> </a>`); /*templateArr.push(`<a href="javascript:;" onclick="window.open(\'{3}\'')" class="link22" title="{14} 人参与评分">${Images.ImdbSvg} <span class="${videoInfo.imdbScore ? 'score22' : 'loading{0}'}" id="imdbScore{0}_{12}">{10}</span> </a>`); */ /*templateArr.push(`<a href="javascript:GM_openInTab(\'{3}\', {active: false})" class="link22" title="{14} 人参与评分">${Images.ImdbSvg} <span class="score22" id="imdbScore{0}_{12}">{10}</span> </a>`); */ } templateArr.push('</div>'); } templateArr.push(` <div class="info{0}"> <!--<div class="title{0}"><a href="{3}" class="link22" target="_blank">{1}</a></div>--> <!--<div class="title{0}"><a href="" target="_blank">{1}</a></div>--> <div class="title{0}">{1}</div> <div class="left{0}"><img src="{2}" onerror="javascript:this.src='${Images.VideoDefaultImg}'"/></div> <div class="right{0}">`); /*templateArr.push(` <div class="info{0}"> <div class="title{0}"><a href="javascript:;" onclick="openlink(\'{3}\')" class="link22" target="_blank">{1}</a></div> <div class="left{0}"><img src="{2}" onerror="javascript:this.src='${Images.VideoDefaultImg}'"/></div> <div class="right{0}">`);*/ if (videoInfo.director) { templateArr.push('<div class="item{0}"><span class="item22">导演</span>:{9}</div>'); } if (videoInfo.actor) { templateArr.push('<div class="item{0}"><span class="item22">主演</span>:{8}</div>'); } if (videoInfo.genre) { templateArr.push('<div class="item{0}"><span class="item22">类型</span>:{7}</div>'); } if (videoInfo.time) { templateArr.push('<div class="item{0}"><span class="item22">时间</span>:{6}</div>'); } if (videoInfo.description) { templateArr.push('<div class="item{0}"><span class="item22">简介</span>:{4}</div>'); } templateArr.push(`</div></div>`); /*videoInfoHtml = Utils.StringFormat(templateArr.join(''), randomCode, videoInfo.title.slice(0, 35), videoInfo.image, videoInfo.url, videoInfo.description, videoInfo.score, videoInfo.time, videoInfo.genre, videoInfo.actor, videoInfo.director, videoInfo.imdbScore || '', videoInfo.imdbUrl, videoInfo.imdbId || 0, videoInfo.ratingCount || 0, videoInfo.imdbRatingCount || 0);*/ videoInfoHtml = Utils.StringFormat(templateArr.join(''), randomCode, videoInfo.title, videoInfo.image, videoInfo.url, videoInfo.description, videoInfo.score, videoInfo.time, videoInfo.genre, videoInfo.actor, videoInfo.director, videoInfo.imdbScore || '', videoInfo.imdbUrl, videoInfo.imdbId || 0, videoInfo.ratingCount || 0, videoInfo.imdbRatingCount || ''); } else { videoInfoHtml = Utils.StringFormat('<div class="noData{0}">未搜索到内容</div>', randomCode); } return videoInfoHtml; }, Loading: function (panel) { panel.find(Utils.StringFormat("#panelBody{0} .content{0}", randomCode)).html("").html(Utils.StringFormat('<div class="loading{0}"></div>', randomCode)); }, GetUrl: function () { //返回电影读书页面 let videoInfo = SearchMovie.searchVideoInfo; var url = videoInfo.url; return url; } }; // 读书搜索面板 const BookSearchPanel = { Create: function (popBoxEl) { var text = window.getSelection().toString().trim(); let self = this; let searchEngineOptionsHtml = ""; for (let k in SearchBook.searchEngineList) { if (SearchBook.searchEngineList.hasOwnProperty(k)) { let v = SearchBook.searchEngineList[k].codeText; let selectOption = ""; if (SearchBook.searchEngine == k) { selectOption = 'selected="selected"'; } searchEngineOptionsHtml += Utils.StringFormat('<option value="{0}" {2}>{1}</option>', k, v, selectOption); } } let wordSearchPanelHtml = ''; //let headHtml = Utils.StringFormat(`<div class="head{0}"><a href="" class="link22" target="_blank"><img class="logo{0}" src="" title="" /></a><select class="select22">{1}</select><div id="showSearchList_{0}" class="listBtn{0}">展示搜索列表</div></div>`, randomCode, searchEngineOptionsHtml); //let headHtml = Utils.StringFormat(`<div class="head{0}"><a href="${Urls.BookSearchResultPageUrl.replace('{title}', encodeURIComponent(text))}" class="link22" target="_blank"><img id="doubanLogo_{0}" class="logo{0}" src="${Images.BookIconBase64}" title="豆瓣读书" /></a><div id="showSearchText_{0}" class="text{0}">${text}</div><div id="showSearchList_{0}" class="listBtn{0}">展示搜索列表</div></div>`, randomCode, searchEngineOptionsHtml); let headHtml = Utils.StringFormat(`<div class="head{0}"><img id="doubanLogo_{0}" class="logo{0}" src="${Images.BookIconBase64}" title="豆瓣读书" /><div id="showSearchText_{0}" class="text{0}">${text}</div><div id="showSearchList_{0}" class="listBtn{0}">展示搜索列表</div></div>`, randomCode, searchEngineOptionsHtml); wordSearchPanelHtml += headHtml; wordSearchPanelHtml += Utils.StringFormat('<div class="content{0}">', randomCode); if (SettingOptions.searchPattern == 'automatic') { wordSearchPanelHtml += self.GetBookInfoHtml(); } else if (SettingOptions.searchPattern == 'manual') { wordSearchPanelHtml += self.GetBookListHtml(); } wordSearchPanelHtml += '</div>'; Panel.popBoxEl = popBoxEl; Panel.Create("", "auto bottom", false, wordSearchPanelHtml, function ($panel) { /*if(ev.pageX < 945) { $panel.css({ left: ev.pageX + 'px', top: ev.pageY + 12 + 'px' }); } else { $panel.css({ left: 945 + 'px', top: ev.pageY + 12 + 'px' }); }*/ if(ev.pageX < 945 && ev.clientY < 475) { // if(ev.pageX < 945 && ev.pageY < 485) { $panel.css({ left: ev.pageX + 'px', top: ev.pageY + 12 + 'px' }); // } else if(ev.pageX < 945 && ev.clientY > 485) { } else if(ev.pageX < 945 && ev.pageY > 475) { $panel.css({ left: ev.pageX + 'px', // top: ev.pageY - 300 - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + 'px' top: ev.pageY - 265 + 'px' }); } else if(ev.pageX > 945 && ev.clientY < 475) { // } else if(ev.pageX > 945 && ev.pageY < 485) { $panel.css({ left: 945 + 'px', top: ev.pageY + 12 + 'px' }); // } else if(ev.pageX > 945 && ev.clientY > 485) { } else if(ev.pageX > 945 && ev.pageY > 475) { $panel.css({ left: 945 + 'px', // top: ev.pageY - 300 - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + document.documentElement.scrollTop + 'px' // top: ev.clientY - 255 + 'px' top: ev.pageY - 265 + 'px' }); /*} else { $panel.css({ left: 945 + 'px', top: ev.pageY + 12 + 'px' });*/ } // 搜索引擎 $panel.find(Utils.StringFormat("#panelBody{0} div:eq(0) select:eq(0)", randomCode)).change(function (e) { SearchBook.searchEngine = $(this).find("option:selected").val(); self.Loading($panel); SearchBook.Update(); SearchBook.Execute(function () { if (SettingOptions.searchPattern == 'automatic') { self.Update(); } else if (SettingOptions.searchPattern == 'manual') { self.ShowSearchList(); } }); }); // 搜索列表 $panel.find(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).click(function (e) { self.ShowSearchList(); }); $panel.find(Utils.StringFormat("#panelBody{0} .logo{0}", randomCode)).click(function () { GM_openInTab(Urls.BookSearchResultPageUrl.replace('{title}', encodeURIComponent(text)), { active: false }); //console.log("ok") }); if (SettingOptions.searchPattern == 'manual') { // 列表点击事件 $panel.find(Utils.StringFormat("#panelBody{0} .listItem{0}", randomCode)).click(function () { let subjectId = $(this).attr('data-id'); SearchBook.searchSelectTitle = $(this).attr('data-name'); self.Loading($panel); SearchBook.UpdateBookInfo(subjectId, function () { self.Update(); }); $(Utils.StringFormat("#panelBody{0} #doubanLogo_{0}", randomCode)).hide();//logo隐藏 $(Utils.StringFormat("#panelBody{0} #showSearchText_{0}", randomCode)).hide();//划词隐藏 }); } if (SettingOptions.searchPattern == 'manual' || !SearchBook.searchBookList || SearchBook.searchBookList.length == 0) { $(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).hide(); } }); }, Update: function () { let self = this; Panel.Update(function ($panel) { let html = self.GetBookInfoHtml(); $panel.find(Utils.StringFormat("#panelBody{0} .content{0}", randomCode)).html("").html(html); }); if (SearchBook.searchBookList && SearchBook.searchBookList.length > 0) { $(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).css("display", "inline"); $(Utils.StringFormat("#panelBody{0} #doubanLogo_{0}", randomCode)).css("display", "none");//列表logo显示 $(Utils.StringFormat("#panelBody{0} #showSearchText_{0}", randomCode)).css("display", "none");//列表划词显示 } //点title后台打开电影读书页面 //console.log(self.GetUrl()) var url = self.GetUrl(); let $panel = $("div.JPopBox-tip-white"); $panel.find(Utils.StringFormat("#panelBody{0} .title{0}", randomCode)).click(function () { GM_openInTab(url, { active: false }); //console.log("ok") }); $panel.find(Utils.StringFormat("#panelBody{0} .left{0}", randomCode)).click(function () { GM_openInTab(url, { active: false }); //console.log("ok") }); }, ShowSearchList: function () { let self = this; let html = self.GetBookListHtml(); let $panel = $("div.JPopBox-tip-white"); $panel.find(Utils.StringFormat("#panelBody{0} .content{0}", randomCode)).html("").html(html); $panel.find(Utils.StringFormat("#panelBody{0} .listItem{0}", randomCode)).click(function () { let subjectId = $(this).attr('data-id'); SearchBook.searchSelectTitle = $(this).attr('data-name'); self.Loading($panel); SearchBook.UpdateBookInfo(subjectId, function () { self.Update(); }); }); $(Utils.StringFormat("#panelBody{0} #showSearchList_{0}", randomCode)).hide(); $(Utils.StringFormat("#panelBody{0} #doubanLogo_{0}", randomCode)).css("display", "inline");//列表logo显示 $(Utils.StringFormat("#panelBody{0} #showSearchText_{0}", randomCode)).css("display", "inlinee");//列表划词显示 }, GetBookListHtml: function () { let htmlArr = []; let bookList = SearchBook.searchBookList; if (bookList && bookList.length > 0) { let itemTemplate = ` <div class="listItem{0}" data-id="{4}" data-name="{2}"> <div class="left{0}"><img src="{1}" onerror="javascript:this.src='${Images.VideoDefaultImg}'"></div> <div class="right{0}"> <div class="title{0}">{2}</div> <div class="score{0}" title="{7}">{3}</div> <div class="info{0}">{6}</div> <div class="info{0}">{5}</div> </div> </div>`; htmlArr.push(Utils.StringFormat('<div class="list{0}">', randomCode)); bookList.forEach((item) => { let bookItem = Utils.StringFormat(itemTemplate, randomCode, item.image, item.title, item.score, item.subjectId, item.description, item.cast, item.ratingCount); htmlArr.push(bookItem); }) htmlArr.push('</div>'); } else { htmlArr.push(Utils.StringFormat('<div class="noData{0}">未搜索到内容</div>', randomCode)); } return htmlArr.join(''); }, GetBookInfoHtml: function () { let bookInfoHtml = ''; let bookInfo = SearchBook.searchBookInfo; if (bookInfo) { let templateArr = []; templateArr.push('<div class="videoInfo{0}">'); if (bookInfo.score) { // 评分 templateArr.push('<div class="score{0}">'); if (bookInfo.score) { //templateArr.push(`<a href="{3}" target="_blank" class="link22" title="{11} 人参与评分">${Images.DbSvg}<span class="score22">{6}</span></a>`); //templateArr.push(`<a href="{3}" target="_blank" class="link22" title="">${Images.DbSvg}<span class="score22">{6}</span></a>`); templateArr.push(`<a href="{3}" target="_blank" class="link22" title="">${Images.DbSvg}<span class="iscore{0}">{6}</span><span class="count{0}">({11})</span></a>`); //templateArr.push(`<a href="javascript:;" onclick="GM_openInTab(\'{3}\', loadInBackground)" class="link22" title="{11} 人参与评分">${Images.DbSvg}<span class="score22">{6}</span></a>`); } templateArr.push('</div>'); } templateArr.push(` <div class="info{0}"> <!--<div class="title{0}"><a href="{3}" class="link22" target="_blank">{1}</a></div>--> <div class="title{0}">{1}</div> <div class="left{0}"><img src="{2}" onerror="javascript:this.src='${Images.VideoDefaultImg}'"/></div> <div class="right{0}">`); if (bookInfo.author) { templateArr.push('<div class="item{0}"><span class="item22">作者</span>:{5}</div>'); } if (bookInfo.publisher) { templateArr.push('<div class="item{0}"><span class="item22">出版社</span>:{7}</div>'); } if (bookInfo.orignal) { templateArr.push('<div class="item{0}"><span class="item22">原作名</span>:{8}</div>'); } if (bookInfo.time) { templateArr.push('<div class="item{0}"><span class="item22">出版年</span>:{10}</div>'); } if (bookInfo.description) { templateArr.push('<div class="item{0}"><span class="item22">简介</span>:{4}</div>'); } templateArr.push(`</div></div>`); /*bookInfoHtml = Utils.StringFormat(templateArr.join(''), randomCode, bookInfo.title.slice(0, 35), bookInfo.image, bookInfo.url, bookInfo.description, bookInfo.author, bookInfo.score, bookInfo.publisher, bookInfo.orignal, bookInfo.translator, bookInfo.time, bookInfo.ratingCount || 0);*/ bookInfoHtml = Utils.StringFormat(templateArr.join(''), randomCode, bookInfo.title, bookInfo.image, bookInfo.url, bookInfo.description, bookInfo.author, bookInfo.score, bookInfo.publisher, bookInfo.orignal, bookInfo.translator, bookInfo.time, bookInfo.ratingCount || ''); } else { bookInfoHtml = Utils.StringFormat('<div class="noData{0}">未搜索到内容</div>', randomCode); } return bookInfoHtml; }, Loading: function (panel) { panel.find(Utils.StringFormat("#panelBody{0} .content{0}", randomCode)).html("").html(Utils.StringFormat('<div class="loading{0}"></div>', randomCode)); }, GetUrl: function () { //返回电影读书页面 let bookInfo = SearchBook.searchBookInfo; var url = bookInfo.url; return url; } }; // 选词 代码来自扩展 uBlock Origin https://github.com/gorhill/uBlock/blob/master/src/js/scriptlets/epicker.js const DoubanPickerTool = { sessionId: '', iframeHost: '', textFilterCandidates: [], targetElements: [], pickerRoot: null, initDoubanPicker: function (iframeHost) { if (!iframeHost) { return; } this.iframeHost = iframeHost; // 主页面监听message事件,接收子组件的值 let self = this; window.addEventListener('message', function (e) { if (e.origin == self.iframeHost) { self.onDialogMessage(e.data) } }, false); }, getElementBoundingClientRect: function (elem) { let rect = typeof elem.getBoundingClientRect === 'function' ? elem.getBoundingClientRect() : { height: 0, left: 0, top: 0, width: 0 }; if (rect.width !== 0 && rect.height !== 0) { return rect; } let left = rect.left, right = rect.right, top = rect.top, bottom = rect.bottom; for (const child of elem.children) { rect = this.getElementBoundingClientRect(child); if (rect.width === 0 || rect.height === 0) { continue; } if (rect.left < left) { left = rect.left; } if (rect.right > right) { right = rect.right; } if (rect.top < top) { top = rect.top; } if (rect.bottom > bottom) { bottom = rect.bottom; } } return { height: bottom - top, left, top, width: right - left }; }, highlightElements: function (elems, force) { if ( (force !== true) && (elems.length === this.targetElements.length) && (elems.length === 0 || elems[0] === this.targetElements[0]) ) { return; } this.targetElements = []; const ow = self.innerWidth; const oh = self.innerHeight; const islands = []; for (const elem of elems) { if (elem === this.pickerRoot) { continue; } this.targetElements.push(elem); const rect = this.getElementBoundingClientRect(elem); if ( rect.left > ow || rect.top > oh || rect.left + rect.width < 0 || rect.top + rect.height < 0 ) { continue; } islands.push( `M${rect.left} ${rect.top}h${rect.width}v${rect.height}h-${rect.width}z` ); } this.sendMessageToIframe({ ocean: `M0 0h${ow}v${oh}h-${ow}z`, islands: islands.join(''), what: "svgPaths" }); }, textFilterFromElement: function (elem) { if (elem === null) { return 0; } if (elem.nodeType !== 1) { return 0; } if (elem.nodeName === "HTML" || elem.nodeName === "BODY") { return 0; } this.textFilterCandidates = this.getNodeText(elem); return 1; }, getNodeText: function (elem) { let temp = [] if (elem) { const forFn = function (ele) { if (ele.childNodes.length > 0 && ele.nodeName != 'A') { let children = Array.from(ele.childNodes); children.forEach((c) => { forFn(c); }) } else { let text = ele.textContent; if (ele.nodeName == 'INPUT') { text = ele.value; } else if (ele.nodeName == 'A') { text = ele.innerText; } if (text && text.trim()) { temp.push(text.trim()); } } } forFn(elem); } return temp; }, filtersFrom: function (x, y) { this.textFilterCandidates.length = 0 let elem = null; if (typeof x === 'number') { elem = this.elementFromPoint(x, y); } else if (x instanceof HTMLElement) { elem = x; x = undefined; } this.textFilterFromElement(elem); return this.textFilterCandidates.length; }, showDialog: function () { this.sendMessageToIframe({ what: 'showDialog', text: this.textFilterCandidates }); }, elementFromPoint: function (x, y) { let lastX, lastY; if (x !== undefined) { lastX = x; lastY = y; } else if (lastX !== undefined) { x = lastX; y = lastY; } else { return null; } if (!this.pickerRoot) { return null; } this.pickerRoot.style.pointerEvents = 'none'; let elems = document.elementsFromPoint(x, y); elems = elems.filter(ele => ele.name != 'myFrame') || []; let elem = elems[0]; this.pickerRoot.style.pointerEvents = ''; return elem; }, highlightElementAtPoint: function (mx, my) { const elem = this.elementFromPoint(mx, my); this.highlightElements(elem ? [elem] : []); }, filterElementAtPoint: function (mx, my) { if (this.filtersFrom(mx, my) === 0) { return; } this.showDialog(); }, onKeyPressed: function (ev) { // Esc if (ev.key === 'Escape' || ev.which === 27) { ev.stopPropagation(); ev.preventDefault(); this.quitPicker(); return; } }, onViewportChanged: function () { this.highlightElements(this.targetElements, true); }, startPicker: function () { this.pickerRoot.focus(); this.pickerRoot.addEventListener('scroll', this.onViewportChanged, { passive: true }); this.pickerRoot.addEventListener('resize', this.onViewportChanged, { passive: true }); this.pickerRoot.addEventListener('keydown', this.onKeyPressed, true); }, quitPicker: function () { this.pickerRoot.removeEventListener('scroll', this.onViewportChanged, { passive: true }); this.pickerRoot.removeEventListener('resize', this.onViewportChanged, { passive: true }); this.pickerRoot.removeEventListener('keydown', this.onKeyPressed, true); if (this.pickerRoot === null) { return; } let parent = this.pickerRoot.parentElement; this.pickerRoot.remove(); this.pickerRoot = null; parent.focus(); }, onDialogMessage: function (msg) { switch (msg.what) { case 'start': this.startPicker(); if (this.targetElements.length === 0) { this.highlightElements([document.body], true); } break; case 'quitPicker': this.quitPicker(); break; case 'highlightElementAtPoint': this.highlightElementAtPoint(msg.mx, msg.my); break; case 'filterElementAtPoint': this.filterElementAtPoint(msg.mx, msg.my); break; case 'togglePreview': if (msg.state === false) { this.highlightElements(this.targetElements, true); } break; default: break; } }, sendMessageToIframe: function (msg) { this.pickerRoot.contentWindow.postMessage(msg, this.iframeHost); }, showPicker: function () { const self = this; if (this.pickerRoot) { return; } // loading let loadingRoot = document.createElement('div'); loadingRoot.innerHTML = ` <div style='z-index:2147483647;background: #000; opacity: 0.3; position: fixed; top: 0px; left: 0px; width: 100%; height: 100%;'></div> <div style="z-index:2147483647;border: 5px solid #f3f3f3; border-radius: 50%; border-top: 5px solid #3498db; width:4vw; height:4vw;min-width:30px;min-height:30px;max-width:50px;max-height:50px;animation: db_search_turn${randomCode} 2s linear infinite; margin: 20px; margin-left: 48%; top: 45%; position: fixed"></div>`; document.documentElement.append(loadingRoot); // iframe const pickerRoot = document.createElement('iframe'); pickerRoot.setAttribute('name', 'myFrame'); pickerRoot.setAttribute('src', this.iframeHost + '/picker.html'); pickerRoot.style = `color-scheme:initial;box-shadow:none !important;display:block !important;height:100vh !important;left:0px !important;max-height:none !important;max-width:none !important;min-height:unset !important;min-width:unset !important;opacity:1 !important;pointer-events:auto !important;position:fixed !important;top: 0px !important;visibility:visible !important;width:100% !important;z-index:2147483647 !important;background:transparent !important;border-width:0px !important;border-style:initial !important;border-color:initial !important;border-image:initial !important;border-radius:0px !important;margin:0px !important;outline:0px !important;padding: 0px !important`; pickerRoot.onload = function () { setTimeout(() => { loadingRoot.remove(); self.sendMessageToIframe({ what: 'connectionAccepted' }) }, 500); }; this.pickerRoot = pickerRoot; document.documentElement.append(pickerRoot); }, } //设置面板 const SettingPanel = { config: [{ title: "", name: "", type: "", attrName: "", item: [{ code: "", text: "" }] }], Create: function (popBoxEl) { let self = this; let settingHtml = []; this.InitConfig(); settingHtml.push(Utils.StringFormat('<div class="setting{0}">', randomCode)); for (let index = 0; index < this.config.length; index++) { let configItem = this.config[index]; settingHtml.push(Utils.StringFormat(`<div class="configItem{0}"><div class="title{0}">{1}</div><div class="right{0}">`, randomCode, configItem.title)); for (let itemIndex = 0; itemIndex < configItem.item.length; itemIndex++) { let itemObj = configItem.item[itemIndex]; settingHtml.push(Utils.StringFormat('<label><input type="radio" name="search{3}{0}" value="{1}">{2}</label>', randomCode, itemObj.code, itemObj.text, configItem.name)); } settingHtml.push('</div></div>'); } settingHtml.push(Utils.StringFormat(`<div class="bottom{0}"><button id="saveBtn{0}">保存</button><span id="saveStatus{0}" class="msg{0}">设置已保存。</span></div></div>`, randomCode)); let settingHtmlStr = settingHtml.join(""); Panel.popBoxEl = popBoxEl; Panel.Create("设置", "auto bottom", false, settingHtmlStr, function ($panel) { $panel.css({ position: "fixed", top: "120px" }); self.Update(); //保存设置 $panel.find(Utils.StringFormat("#panelBody{0} #saveBtn{0}", randomCode)).click(function (e) { self.config.forEach((item) => { let selecter = `#panelBody{0} input[name='search${item.name}{0}']:checked`; let value = $panel.find(Utils.StringFormat(selecter, randomCode)).val(); SettingOptions[item.attrName] = value; }); Utils.SetSettingOptions(); $panel.find(Utils.StringFormat("#panelBody{0} #saveStatus{0}", randomCode)).fadeIn(function () { setTimeout(function () { $panel.find(Utils.StringFormat("#panelBody{0} #saveStatus{0}", randomCode)).fadeOut(); }, 1500); }); }); }); }, Update: function () { let self = this; Utils.GetSettingOptions(); Panel.Update(function ($panel) { self.config.forEach((item) => { let selecter = `#panelBody{0} input[name='search${item.name}{0}'][value='{1}']`; $panel.find(Utils.StringFormat(selecter, randomCode, SettingOptions[item.attrName])).prop("checked", true); }); }); }, InitConfig: function () { this.config = []; let engineConfigObj = { title: "默认搜索引擎", name: "Engine", attrName: "defaultsearchengine", item: [] }; /*for (let k in Search.searchEngineList) { if (Search.searchEngineList.hasOwnProperty(k)) { let v = Search.searchEngineList[k].codeText; engineConfigObj.item.push({ code: k, text: v }); } }*/ this.config.push(engineConfigObj); let patternConfigObj = { title: "搜索模式", name: "Pattern", attrName: "searchPattern", item: [{ code: "automatic", text: "自动" }, { code: "manual", text: "手动" }] }; this.config.push(patternConfigObj); let selectConfigObj = { title: "划词模式", name: "Select", attrName: "selectPattern", item: [{ code: "select", text: "划词" }, { code: "hold", text: "划词键 + 划词" }] }; this.config.push(selectConfigObj); let keyConfigObj = { title: "划词键", name: "Key", attrName: "selectKey", item: [{ code: "Ctrl", text: "Ctrl" }, { code: "Alt", text: "Alt" }] }; this.config.push(keyConfigObj); let positionConfigObj = { title: "图标位置", name: "Position", attrName: "selectIconPosition", item: [{ code: "right", text: "右" }, { code: "left", text: "左" }, { code: "top", text: "上" }] }; this.config.push(positionConfigObj); }, CreateStyle: function () { let s = ""; s += Utils.StringFormat("#panelBody{0} .setting{0} .configItem{0}{display:flex;padding:5px}#panelBody{0} .configItem{0} .title{0}{width:90px;font-size: 14px}#panelBody{0} .configItem{0} .right{0}{margin-left:10px}#panelBody{0} .configItem{0} .right{0} label{cursor:pointer;display:inline-block;min-width:60px}#panelBody{0} .configItem{0} .right{0} input{cursor:pointer;vertical-align:middle;margin-top:-2px;margin-bottom:1px}", randomCode); s += Utils.StringFormat("#panelBody{0} .setting{0} button{border:0.5px solid black;margin:5px;padding:1px 5px;border-radius:revert;font:revert}#panelBody{0} .setting{0} .bottom{0} .msg{0}{display: none; margin-left:5px;background-color:#fff1a8;padding:3px}", randomCode); return s; } }; //主程序 const WebSearchlate = function () { const $doc = $(document); const $body = $("html body"); const createHtml = function () { const wordSearchIconHtml = Utils.StringFormat('<div id="wordSearch{0}" class="wordSearch{0}"><div class="wordSearchIcon{0}"></div></div>', randomCode); $body.append(Utils.StringFormat('<div id="webSearch{0}">', randomCode) + wordSearchIconHtml + '</div>'); }; const createStyle = function () { //尽可能避开csp认证 GM_xmlhttpRequest({ method: "get", url: "https://cdn.jsdelivr.net/gh/zyufstudio/jQuery@master/jPopBox/dist/jPopBox.min.css", onload: function (r) { GM_addStyle(r.responseText + ".JPopBox-tip-white{width: 482px;max-width: 550px;min-width: 450px}"); } }); let s = Utils.StringFormat("@keyframes db_search_turn{0}{0%{transform:rotate(0deg)}25%{transform:rotate(90deg)}50%{transform:rotate(180deg)}75%{transform:rotate(270deg)}100%{transform:rotate(360deg)}}", randomCode); s += Utils.StringFormat(".wordSearch{0}{background-color: rgb(245, 245, 245);box-sizing: content-box;cursor: pointer;z-index: 2147483647;border-width: 1px;border-style: solid;border-color: rgb(220, 220, 220);border-image: initial;border-radius: 5px;padding: 0.5px;position: absolute;display: none} .wordSearch{0}.animate{animation: db_search_turn{0} 5s linear infinite}", randomCode); s += Utils.StringFormat(".wordSearchIcon{0}{background-image: url({1});background-size: 25px;height: 25px;width: 25px}", randomCode, Images.IconBase64); s += Panel.CreateStyle(); s += SettingPanel.CreateStyle(); GM_addStyle(s); }; const RegMenu = function () { GM_registerMenuCommand("设置", function () { if (DoubanPickerTool.pickerRoot) { DoubanPickerTool.quitPicker(); } $("div#wordSearch" + randomCode).hide(); SearchMovie.Clear(); SearchBook.Clear(); Panel.Destroy(); SettingPanel.Create($body); }); GM_registerMenuCommand("进入取词模式", function () { if (DoubanPickerTool.pickerRoot) { return; } $("div#wordSearch" + randomCode).hide(); SearchMovie.Clear(); SearchBook.Clear(); Panel.Destroy(); DoubanPickerTool.showPicker(); }); }; this.init = function () { SearchMovie.RegisterEngine(); SearchBook.RegisterEngine(); createStyle(); createHtml(); RegMenu(); Utils.GetSettingOptions(); DoubanPickerTool.initDoubanPicker(Urls.IframePageHost) }; };
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址