Цей скрипт не слід встановлювати безпосередньо. Це - бібліотека для інших скриптів для включення в мета директиву // @require https://update.greasyfork.org/scripts/438161/1009060/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) {
//设置搜索结果面板鼠标位置
/*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 < 485) {
$panel.css({
left: ev.pageX + 'px',
top: ev.pageY + 12 + 'px'
});
} else if(ev.pageX < 945 && ev.clientY > 485) {
$panel.css({
left: ev.pageX + 'px',
top: 750 - 255 + document.documentElement.scrollTop + 'px'
});
} else if(ev.pageX > 945 && ev.clientY < 485) {
$panel.css({
left: 945 + 'px',
top: ev.pageY + 12 + 'px'
});
} else if(ev.pageX > 945 && ev.clientY > 485) {
$panel.css({
left: 945 + 'px',
top: 750 - 255 + document.documentElement.scrollTop + '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 < 485) {
$panel.css({
left: ev.pageX + 'px',
top: ev.pageY + 12 + 'px'
});
} else if(ev.pageX < 945 && ev.clientY > 485) {
$panel.css({
left: ev.pageX + 'px',
top: 750 - 255 + document.documentElement.scrollTop + 'px'
});
} else if(ev.pageX > 945 && ev.clientY < 485) {
$panel.css({
left: 945 + 'px',
top: ev.pageY + 12 + 'px'
});
} else if(ev.pageX > 945 && ev.clientY > 485) {
$panel.css({
left: 945 + 'px',
top: 750 - 255 + document.documentElement.scrollTop + '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)
};
};