selection-search

划词搜索

  1. // ==UserScript==
  2. // @name selection-search
  3. // @version 10000
  4. // @author logan
  5. // @description 划词搜索
  6. // @homepage https://github.com/loganoxo/selection-search
  7. // @homepageURL https://github.com/loganoxo/selection-search
  8. // @supportURL https://github.com/loganoxo/selection-search/issues
  9. // @icon 
  10. // @match *://*/*
  11. // @exclude *://192.168.*.*
  12. // @exclude *://10.*.*.*
  13. // @exclude *://172.16.*.*
  14. // @exclude *://127.0.0.*
  15. // @exclude *://localhost
  16. // @exclude *://*.local
  17. // @grant GM.openInTab
  18. // @grant GM.setClipboard
  19. // @grant GM.getValue
  20. // @grant GM.setValue
  21. // @grant GM.xmlHttpRequest
  22. // @run-at document-end
  23. // @license MIT
  24. // @namespace https://gf.qytechs.cn/users/1317201
  25. // ==/UserScript==
  26.  
  27. (async function () {
  28. "use strict";
  29.  
  30. ///////////////////////////////////////////////////////////////////
  31. // 配置
  32. ///////////////////////////////////////////////////////////////////
  33.  
  34. //=========================定义网站数据=======================================
  35. function SiteInfo(_name, _url, _homepage, _icon) {
  36. this.name = _name;
  37. this.url = _url;
  38. this.home = _homepage;
  39. this.icon = _icon;
  40.  
  41. this.callSiteInformation = function (_enable = true) {
  42. return {
  43. name: _name,
  44. url: _url,
  45. home: _homepage,
  46. icon: _icon,
  47. enable: _enable,
  48. };
  49. };
  50. // this.callSiteInformationNoHomepage = function (_enable = true) {
  51. // return {
  52. // name: _name,
  53. // url: _url,
  54. // icon: _icon,
  55. // enable: _enable,
  56. // };
  57. // };
  58. }
  59.  
  60. // 百度
  61. const Baidu_ICO = "";
  62. const Baidu = new SiteInfo(
  63. "百度",
  64. "https://www.baidu.com/s?wd=%s&ie=utf-8",
  65. "https://www.baidu.com/",
  66. Baidu_ICO,
  67. );
  68.  
  69. // 谷歌
  70. const Google_ICO = "";
  71. const Google = new SiteInfo(
  72. "谷歌",
  73. "https://www.google.com/search?q=%s&ie=utf-8&oe=utf-8",
  74. "https://www.google.com/",
  75. Google_ICO,
  76. );
  77.  
  78. // 必应
  79. const Bing_ICO = "";
  80. const Bing = new SiteInfo(
  81. "必应",
  82. "https://cn.bing.com/search?q=%s",
  83. "https://cn.bing.com/",
  84. Bing_ICO,
  85. );
  86.  
  87. // github
  88. const GitHub_ICO = "";
  89. const GitHub = new SiteInfo(
  90. "GitHub",
  91. "https://github.com/search?q=%s",
  92. "https://github.com/",
  93. GitHub_ICO,
  94. );
  95.  
  96. // 哔哩哔哩
  97. const Bilibili_ICO = "";
  98. const Bilibili = new SiteInfo(
  99. "哔哩哔哩",
  100. "https://search.bilibili.com/all?keyword=%s",
  101. "https://www.bilibili.com/",
  102. Bilibili_ICO,
  103. );
  104.  
  105. //youtube
  106. const YOUTUBE_ICO =
  107. "";
  108. const YOUTUBE = new SiteInfo(
  109. "YOUTUBE",
  110. "https://www.youtube.com/results?search_query=%s",
  111. "https://www.youtube.com/",
  112. YOUTUBE_ICO);
  113.  
  114. //V2EX
  115. const V2EX_ICO =
  116. "";
  117. const V2EX = new SiteInfo(
  118. "V2EX",
  119. "https://www.google.com/search?q=site:v2ex.com/t%20%s",
  120. "https://www.v2ex.com/",
  121. V2EX_ICO);
  122.  
  123. // 直达
  124. const ZHIDA_ICO =
  125. "";
  126. const Zhida = new SiteInfo("直达", "%s", "https://www.baidu.com/", ZHIDA_ICO);
  127.  
  128. //复制
  129. const FUZHI_ICO =
  130. "";
  131. const Fuzhi = new SiteInfo("复制", "%s", "https://www.baidu.com/", FUZHI_ICO);
  132.  
  133. // 知乎
  134. const Zhihu = new SiteInfo(
  135. "知乎",
  136. "https://www.zhihu.com/search?q=%s",
  137. "https://www.zhihu.com/",
  138. "https://static.zhihu.com/heifetz/favicon.ico",
  139. );
  140.  
  141. //fsou
  142. const FSOU = new SiteInfo(
  143. "FSOU",
  144. "https://fsoufsou.com/search?q=%s&tbn=all",
  145. "https://fsoufsou.com/",
  146. "https://fsoufsou.com/favicon.ico");
  147.  
  148.  
  149. const Baidufanyi = new SiteInfo(
  150. "百度翻译",
  151. "https://fanyi.baidu.com/#auto/zh/%s",
  152. "https://fanyi.baidu.com/",
  153. "https://fanyi-cdn.cdn.bcebos.com/webStatic/translation/img/favicon/favicon.ico",
  154. );
  155. const Baiduwangpan = new SiteInfo(
  156. "百度网盘",
  157. "https://pan.baidu.com/disk/home?#/search?key=%s",
  158. "https://pan.baidu.com/",
  159. "https://nd-static.bdstatic.com/m-static/v20-main/favicon-main.ico",
  160. );
  161. const Baidubaike = new SiteInfo(
  162. "百度百科",
  163. "https://baike.baidu.com/search/word?pic=1&sug=1&word=%s",
  164. "https://baike.baidu.com/",
  165. "https://baike.baidu.com/favicon.ico",
  166. );
  167. const Baiduzhidao = new SiteInfo(
  168. "百度知道",
  169. "https://zhidao.baidu.com/search?word=%s",
  170. "https://zhidao.baidu.com/",
  171. "https://www.baidu.com/favicon.ico?t=20171027",
  172. );
  173. const Baiduxinwen = new SiteInfo(
  174. "百度新闻",
  175. "https://www.baidu.com/s?rtt=1&bsst=1&cl=2&tn=news&rsv_dl=ns_pc&word=%s",
  176. "https://news.baidu.com/",
  177. "https://www.baidu.com/favicon.ico",
  178. );
  179. const Baiduwenku = new SiteInfo(
  180. "百度文库",
  181. "https://wenku.baidu.com/search?word=%s",
  182. "",
  183. "https://www.baidu.com/favicon.ico",
  184. );
  185. const Baidumap = new SiteInfo(
  186. "百度地图",
  187. "https://map.baidu.com/search?querytype=s&wd=%s",
  188. "",
  189. "https://map.baidu.com/favicon.ico",
  190. );
  191. const Baidutupian = new SiteInfo(
  192. "百度图片",
  193. "https://image.baidu.com/search/index?tn=baiduimage&ie=utf-8&word=%s",
  194. "",
  195. "https://www.baidu.com/favicon.ico",
  196. );
  197. const Baiduxueshu = new SiteInfo(
  198. "百度学术",
  199. "https://xueshu.baidu.com/s?wd=%s",
  200. "",
  201. "https://www.baidu.com/favicon.ico",
  202. );
  203. const Baidutieba = new SiteInfo(
  204. "贴吧",
  205. "https://tieba.baidu.com/f?kw=%s&ie=utf-8",
  206. "https://tieba.baidu.com/",
  207. "https://www.baidu.com/favicon.ico",
  208. );
  209.  
  210. const Googlefanyi = new SiteInfo(
  211. "谷歌翻译",
  212. "https://translate.google.com/?q=%s",
  213. "",
  214. "https://ssl.gstatic.com/translate/favicon.ico",
  215. );
  216. const Googlemap = new SiteInfo(
  217. "谷歌地图",
  218. "https://www.google.com/maps/search/%s",
  219. "https://www.google.com/maps/",
  220. "https://s2.loli.net/2022/08/17/SloXZzf9nC6LPbq.png",
  221. );
  222. const Googleearth = new SiteInfo(
  223. "谷歌地球",
  224. "https://earth.google.com/web/search/%s",
  225. "https://earth.google.com/web/",
  226. "https://s2.loli.net/2022/08/17/IOiPDl7YX3QnmsC.png",
  227. );
  228. const Googlexueshu = new SiteInfo(
  229. "谷歌学术",
  230. "https://scholar.google.com/scholar?hl=zh-CN&q=%s",
  231. "",
  232. "https://s2.loli.net/2022/08/17/4BaC1Acu2ebXJR9.png",
  233. );
  234. const Googlepic = new SiteInfo(
  235. "谷歌图片",
  236. "https://www.google.com/search?q=%s&tbm=isch",
  237. "https://www.google.com/imghp?hl=zh-CN",
  238. Google.icon,
  239. );
  240. const Googlenews = new SiteInfo(
  241. "谷歌新闻",
  242. "https://news.google.com/search?q=%s&hl=zh-CN&gl=CN&ceid=CN:zh-Hans",
  243. "https://news.google.com/topstories?hl=zh-CN&gl=CN&ceid=CN:zh-Hans",
  244. "https://s2.loli.net/2022/08/17/RTdZQMD2Aw8eobn.png",
  245. );
  246.  
  247. const StackOverflow = new SiteInfo(
  248. "StackOverflow",
  249. "https://stackoverflow.com/search?q=%s",
  250. "",
  251. "https://s2.loli.net/2022/08/16/mgMHa8UTekYIdV4.png",
  252. );
  253.  
  254. const Taobao = new SiteInfo(
  255. "淘宝",
  256. "https://s.taobao.com/search?q=%s",
  257. "https://www.taobao.com/",
  258. "https://www.taobao.com/favicon.ico",
  259. );
  260. const Jingdong = new SiteInfo(
  261. "京东",
  262. "https://search.jd.com/Search?keyword=%s&enc=utf-8",
  263. "https://www.jd.com/",
  264. "https://search.jd.com/favicon.ico",
  265. );
  266. const Tianmao = new SiteInfo(
  267. "天猫",
  268. "https://list.tmall.com/search_product.htm?q=%s",
  269. "https://www.tmall.com/",
  270. "https://www.tmall.com/favicon.ico",
  271. );
  272. const Maimai = new SiteInfo(
  273. "脉脉",
  274. "https://maimai.cn/web/search_center?type=gossip&query=%s&highlight=true",
  275. "https://maimai.cn/feed_list",
  276. "https://maimai.cn/favicon.ico",
  277. );
  278. const Weibo = new SiteInfo(
  279. "微博",
  280. "https://s.weibo.com/weibo/%s",
  281. "https://weibo.com/",
  282. "https://s.weibo.com/favicon.ico",
  283. );
  284.  
  285.  
  286. //=========================定义网站数据=======================================
  287.  
  288. const defaultConf = {
  289. //
  290. // 基本配置
  291. //
  292. showToolbar: true, // 显示划词工具条
  293. showFrequentEngines: true, // 显示常用搜索引擎
  294. showClassifiedEngines: true, // 显示分类搜索引擎
  295. showPlaceholder: false, // 显示使用方式提示信息(如搜索框placeholder)
  296. enableOnInput: true, // 是否在input/textarea上启用划词和快捷键
  297. autoCopyToClipboard: false, // 划词时自动复制到剪贴板(内容为文本格式)
  298. //
  299. // 常用搜索引擎列表
  300. // 属性:
  301. // - name 搜索引擎名称
  302. // - url 搜索引擎搜索url
  303. // - home 搜索引擎主页url
  304. // - icon 搜索引擎图标, base64编码
  305. // - enable 是否启用
  306. //
  307. frequentEngines: [
  308. Baidu.callSiteInformation(),
  309. Google.callSiteInformation(),
  310. FSOU.callSiteInformation(false),
  311. Bing.callSiteInformation(),
  312. Baidufanyi.callSiteInformation(false),
  313.  
  314. GitHub.callSiteInformation(),
  315.  
  316. Zhihu.callSiteInformation(false),
  317. Googlenews.callSiteInformation(false),
  318.  
  319. Bilibili.callSiteInformation(),
  320. Taobao.callSiteInformation(false),
  321. Jingdong.callSiteInformation(false),
  322. Tianmao.callSiteInformation(false),
  323. Baiduwangpan.callSiteInformation(false),
  324. Maimai.callSiteInformation(false),
  325.  
  326. YOUTUBE.callSiteInformation(),
  327. V2EX.callSiteInformation(),
  328.  
  329. Zhida.callSiteInformation(),
  330. Fuzhi.callSiteInformation()
  331. ],
  332. };
  333.  
  334. ///////////////////////////////////////////////////////////////////
  335. // css样式
  336. ///////////////////////////////////////////////////////////////////
  337.  
  338. const sheet = `
  339. /*
  340. 注意: 为了避免网页style对该工具的影响, 所有样式均进行初始化并加入!important,
  341. js中设置style时也要注意将其设置为important, 否则不能生效.
  342. */
  343. /* 划词工具条 */
  344. .qs-toolbar {
  345. /* 初始化所有style, 避免被网页本身的style影响 */
  346. // all: initial !important;
  347. margin: 0;
  348. position: absolute !important;
  349. display: flex !important;
  350. flex-direction: row;
  351. align-items: center;
  352. justify-items: center;
  353. gap: 8px;
  354. height: 50px !important;
  355. padding: 5px 4px !important;
  356. white-space: nowrap !important;
  357. border: 2px solid rgba(0, 196, 182, 0.9) !important;
  358. background-color: white !important;
  359. z-index: 10000 !important;
  360. border-radius: 15px;
  361. }
  362. .qs-toolbar-icon {
  363. // all: initial !important;
  364. display: block !important;
  365. margin: 0px !important;
  366. padding: 2px !important;
  367. width: 35px !important;
  368. height: 35px !important;
  369. // border: 1px solid #FFF !important;
  370. transition: all 0.3s ease;
  371. cursor: pointer !important;
  372. border-radius: 15px;
  373. }
  374. .qs-toolbar-icon:hover {
  375. // border: 1px solid #CCC !important;
  376. transform: scale(1.1, 1.1);
  377. }
  378. .engine-name {
  379. vertical-align: middle !important;
  380. // margin-left: 2px !important;
  381. font-size: 20px !important;
  382. font-family: arial,sans-serif !important;
  383. font-weight: 400 !important;
  384. color: #5F5F5F !important;
  385. height: 45px !important;
  386. width: 45px !important;
  387. border: 1px solid #262626 !important;
  388. cursor: pointer !important;
  389. display: flex;
  390. justify-content: center;
  391. align-items: center;
  392. background-color: white !important;
  393. transition: all 0.3s ease;
  394. padding: 2px !important;
  395. box-sizing: border-box;
  396. border-radius: 15px;
  397. outline: 1px solid rgba(0, 196, 182, 0.8) !important;
  398. }
  399. .engine-name:hover {
  400. outline: 3px solid #6DC5D1 !important;
  401. }
  402. `;
  403.  
  404. ///////////////////////////////////////////////////////////////////
  405. // 全局变量
  406. ///////////////////////////////////////////////////////////////////
  407. var shadowHost = null;
  408. var shadowRoot = null;
  409. var conf = await GM.getValue("qs-conf", defaultConf);
  410.  
  411. var qsPageLock = false; // 是否在当前页面锁定快搜所有功能, 锁定之后仅响应解锁快捷键
  412.  
  413. var qsToolbar = null; // 快搜划词工具条
  414. var qsMainBox = null; // 快搜主窗口
  415. var qsSettingBox = null; // 快搜设置窗口
  416. var selectionBak_qs = null;
  417. ///////////////////////////////////////////////////////////////////
  418. // 功能函数
  419. ///////////////////////////////////////////////////////////////////
  420.  
  421. // 获取可视窗口在文档(页面)中的绝对位置
  422. function getWindowPosition() {
  423. return {
  424. top: window.scrollY,
  425. bottom: window.scrollY + window.innerHeight,
  426. left: window.scrollX,
  427. right: window.scrollX + window.innerWidth,
  428. };
  429. }
  430.  
  431. // 获取选中文本
  432. function getSelection() {
  433. return window.getSelection().toString().trim();
  434. }
  435.  
  436. // 判断是否允许工具条
  437. function isAllowToolbar(event) {
  438. var target = event.target;
  439. if (!conf.showToolbar || qsPageLock) {
  440. return false;
  441. }
  442. if (
  443. !conf.enableOnInput &&
  444. (target.tagName == "INPUT" || target.tagName == "TEXTAREA")
  445. ) {
  446. return false;
  447. }
  448. if (
  449. (qsMainBox && qsMainBox.contains(target)) ||
  450. (qsSettingBox && qsSettingBox.contains(target))
  451. ) {
  452. return false;
  453. }
  454. return true;
  455. }
  456.  
  457. // 获取搜索引擎主页url
  458. function getEngineHome(engine) {
  459. if (engine.home) {
  460. return engine.home;
  461. } else {
  462. var url = new URL(engine.url);
  463. return `${url.protocol}//${url.hostname}/`;
  464. }
  465. }
  466.  
  467. // 打开url.
  468. // 当按下Cmd(Mac系统)/Ctrl(Windows/Linux系统), 则后台打开url.
  469. async function openUrl(url, event) {
  470. // console.log(`Quick Search: open url, url is ${url}`);
  471. if (!url) return;
  472. if (event.metaKey || event.ctrlKey) {
  473. await GM.openInTab(url, true);
  474. } else {
  475. await GM.openInTab(url, false);
  476. }
  477. }
  478.  
  479. // 打开engine搜索结果或engine主页.
  480. function openEngine(engine, query, event) {
  481. // console.log(`Quick Search: open engine, engine is ${engine.url}, query is ${query}`);
  482. if (!engine) return;
  483. if (engine.name === "直达") {
  484. var Expression =
  485. /http(s)?:\/\/([\w-]+\.)+[\w-]+(\/[\w- .\/?%&=]*)?/;
  486. var objExp = new RegExp(Expression);
  487. if (objExp.test(query) == false) {
  488. query = "https://" + query;
  489. }
  490. openUrl(query, event);
  491. return;
  492. }
  493. if (engine.name === "复制") {
  494. copyText(query);
  495. hideToolbar();
  496. return;
  497. }
  498. if (query) {
  499. var url = engine.url.replace("%s", encodeURIComponent(query));
  500. openUrl(url, event);
  501. } else {
  502. openUrl(getEngineHome(engine), event);
  503. }
  504. }
  505.  
  506. // 点击搜索引擎.
  507. // 当按下Alt, 则忽略查询词打开引擎主页, 否则正常搜索.
  508. function openEngineOnClick(engine, query, event) {
  509. if (event.altKey) {
  510. openEngine(engine, null, event);
  511. } else {
  512. openEngine(engine, query, event);
  513. }
  514. }
  515.  
  516. // 点击划词工具条搜索引擎.
  517. function openEngineOnClickToolbar(engine, event) {
  518. var query = getSelection();
  519. if (!query) {
  520. query = selectionBak_qs;
  521. }
  522. openEngineOnClick(engine, query, event);
  523. }
  524.  
  525. ///////////////////////////////////////////////////////////////////
  526. // 元素创建 与 元素事件响应
  527. ///////////////////////////////////////////////////////////////////
  528.  
  529. // 加载css样式
  530. function loadSheet() {
  531. var css = document.createElement("style");
  532. css.type = "text/css";
  533. css.id = "qs-css";
  534. css.textContent = sheet;
  535. shadowRoot.appendChild(css);
  536. }
  537.  
  538. //
  539. // 划词工具条
  540. //
  541.  
  542. // 创建划词工具条
  543. function createToolbar() {
  544. // 工具条container
  545. var toolbar = document.createElement("div");
  546. toolbar.id = "qs-toolbar";
  547. toolbar.className = "qs-toolbar";
  548. toolbar.style.setProperty("display", "none", "important");
  549. shadowRoot.appendChild(toolbar);
  550.  
  551. // 常用搜索引擎按钮
  552. conf.frequentEngines.forEach((engine, index) => {
  553. if (!engine.enable) return; // 此处return搭配forEach, 请勿改为其他形式循环
  554. var ele = document.createElement("div");
  555. ele.className = "engine-name";
  556. var icon = document.createElement("img");
  557. icon.id = "qs-toolbar-icon-" + index;
  558. icon.className = "qs-toolbar-icon";
  559. icon.src = engine.icon;
  560. icon.alt = engine.name;
  561. icon.title = engine.name;
  562. // icon.style.cssText = "display: none !important";
  563. icon.addEventListener(
  564. "click",
  565. function (e) {
  566. openEngineOnClickToolbar(engine, e);
  567. },
  568. true,
  569. );
  570. ele.appendChild(icon);
  571.  
  572. toolbar.appendChild(ele);
  573. });
  574.  
  575. qsToolbar = toolbar;
  576. }
  577.  
  578. // 划词工具条是否处于显示状态
  579. function isToolbarVisual() {
  580. return qsToolbar && qsToolbar.style.display !== "none";
  581. }
  582.  
  583. // 显示划词工具条
  584. function showToolbar(event) {
  585. ensureQuickSearchAlive();
  586. if (!qsToolbar || isToolbarVisual()) {
  587. return;
  588. }
  589.  
  590. var toolbar = qsToolbar;
  591.  
  592. toolbar.style.setProperty("top", "-10000px", "important");
  593. toolbar.style.setProperty("left", "-10000px", "important");
  594. toolbar.style.setProperty("display", "flex", "important");
  595.  
  596. var toolbarClientRect = toolbar.getBoundingClientRect();
  597. var toolbarWidth = toolbarClientRect.right - toolbarClientRect.left;
  598. var toolbarHeight = toolbarClientRect.bottom - toolbarClientRect.top;
  599.  
  600. var toolbarNewTop = event.pageY - 100;
  601. var toolbarNewLeft = event.pageX - toolbarWidth / 2;
  602. var windowPos = getWindowPosition();
  603. if (toolbarNewTop - 10 < windowPos.top) {
  604. toolbarNewTop = event.pageY + toolbarHeight + 15;
  605. }
  606. if (toolbarNewLeft < windowPos.left) {
  607. toolbarNewLeft = windowPos.left;
  608. } else if (toolbarNewLeft + toolbarWidth > windowPos.right) {
  609. toolbarNewLeft = windowPos.right - toolbarWidth;
  610. }
  611.  
  612. toolbar.style.setProperty("top", toolbarNewTop + "px", "important");
  613. toolbar.style.setProperty("left", toolbarNewLeft + "px", "important");
  614. }
  615.  
  616. // 隐藏划词工具条
  617. function hideToolbar() {
  618. if (qsToolbar) {
  619. qsToolbar.style.setProperty("display", "none", "important");
  620. }
  621. }
  622.  
  623. //复制
  624. var copyText = function (button, content, success) {
  625. if (!button) {
  626. return;
  627. }
  628.  
  629. if (typeof content == 'function') {
  630. success = content;
  631. content = null;
  632. }
  633.  
  634. success = success || function () {
  635. };
  636.  
  637. // 是否降级使用
  638. var isFallback = !navigator.clipboard;
  639.  
  640. if (typeof button == 'string' && !content) {
  641. if (content === false) {
  642. isFallback = true;
  643. }
  644. content = button;
  645. button = null;
  646. }
  647.  
  648. var eleTextarea = document.querySelector('#tempTextarea');
  649. if (!eleTextarea && isFallback) {
  650. eleTextarea = document.createElement('textarea');
  651. eleTextarea.style.width = 0;
  652. eleTextarea.style.position = 'fixed';
  653. eleTextarea.style.left = '-999px';
  654. eleTextarea.style.top = '10px';
  655. eleTextarea.setAttribute('readonly', 'readonly');
  656. document.body.appendChild(eleTextarea);
  657. }
  658.  
  659. var funCopy = function (text, callback) {
  660. callback = callback || function () {
  661. };
  662.  
  663. if (!isFallback) {
  664. navigator.clipboard.writeText(text).then(function () {
  665. callback();
  666. // 成功回调
  667. success(text);
  668. }, function () {
  669. // 禁止写入剪切板后使用兜底方法
  670. copyText(text, false);
  671. callback();
  672. // 成功回调
  673. success(text);
  674. });
  675.  
  676. return;
  677. }
  678.  
  679. eleTextarea.value = text;
  680. eleTextarea.select();
  681. document.execCommand('copy', true);
  682.  
  683. callback();
  684. // 成功回调
  685. success(text);
  686. };
  687.  
  688. funCopy(content);
  689. return;
  690. };
  691.  
  692. //
  693. // 创建以上所有东东
  694. //
  695.  
  696. function initQuickSearch() {
  697. selectionBak_qs = null;
  698. shadowHost = document.createElement("div");
  699. document.body.appendChild(shadowHost);
  700. shadowRoot = shadowHost.attachShadow({mode: 'closed'});
  701. loadSheet();
  702. if (conf.showToolbar) {
  703. createToolbar();
  704. }
  705. }
  706.  
  707. // 百度等网页会在不刷新页面的情况下改变网页内容, 导致quick search除了js脚本之外的东东全部没了.
  708. // 此函数用于确保quick search处于可用状态, 需在toolbar或mainbox等窗口每次显示时调用.
  709. function ensureQuickSearchAlive() {
  710. if (!shadowHost || !shadowRoot) {
  711. initQuickSearch();
  712. } else {
  713. var css = shadowRoot.getElementById("qs-css");
  714. if (!css) {
  715. initQuickSearch();
  716. }
  717. }
  718. }
  719.  
  720. initQuickSearch();
  721.  
  722. ///////////////////////////////////////////////////////////////////
  723. // 全局事件响应
  724. //
  725. // 我们将全局事件绑定在捕获阶段执行, 避免事件响应被网页自带的脚本拦截掉.
  726. ///////////////////////////////////////////////////////////////////
  727.  
  728. //
  729. // top window和iframe共用的事件处理逻辑
  730. //
  731.  
  732. document.addEventListener(
  733. "mousedown",
  734. function (e) {
  735. var target = e.target;
  736. // 隐藏工具条
  737. if (isToolbarVisual() && !qsToolbar.contains(target) && !shadowHost.contains(target)) {
  738. selectionBak_qs = null;
  739. hideToolbar();
  740. }
  741. },
  742. true,
  743. );
  744.  
  745. document.addEventListener(
  746. "mouseup",
  747. function (e) {
  748. var target = e.target;
  749. var selection = getSelection();
  750. if (!qsToolbar.contains(target) && !shadowHost.contains(target)) {
  751. selectionBak_qs = selection;
  752. }
  753. // 显示/隐藏工具条
  754. if (isAllowToolbar(e)) {
  755. if (selection && !isToolbarVisual()) {
  756. showToolbar(e);
  757. }
  758. if (!selection && isToolbarVisual() && !qsToolbar.contains(target) && !shadowHost.contains(target)) {
  759. selectionBak_qs = null;
  760. hideToolbar();
  761. }
  762. }
  763.  
  764. // 划词时自动复制文本到剪贴板
  765. if (
  766. conf.autoCopyToClipboard &&
  767. target.tagName !== "INPUT" &&
  768. target.tagName !== "TEXTAREA" &&
  769. !qsMainBox.contains(target) &&
  770. !qsSettingBox.contains(target)
  771. ) {
  772. if (selection) {
  773. GM.setClipboard(selection, "text/plain");
  774. }
  775. }
  776. },
  777. true,
  778. );
  779. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址