Popup Search

Popup search box and translate button (etc) for selected texts

  1. // ==UserScript==
  2. // @name Popup Search
  3. // @name:zh Popup Search: 快捷搜索
  4. // @author lkytal
  5. // @namespace Lkytal
  6. // @version 4.3.1
  7. // @icon https://github.com/lkytal/GM/raw/master/icons/search.png
  8. // @homepage https://lkytal.github.io/
  9. // @homepageURL https://lkytal.github.io/GM
  10. // @description Popup search box and translate button (etc) for selected texts
  11. // @description:zh 为选中文字弹出搜索和翻译的快捷按钮
  12. // @license AGPL
  13. // @include *
  14. // @exclude */test/*.html*
  15. // @exclude http://acid3.acidtests.org/*
  16. // @exclude http://www.acfun.tv/*
  17. // @exclude http://www.sf-express.com/*
  18. // @exclude http://furk.net/*
  19. // @grant GM_xmlhttpRequest
  20. // @grant GM_addStyle
  21. // @grant GM_openInTab
  22. // @grant GM_setClipboard
  23. // @grant GM_download
  24. // @grant GM_getValue
  25. // @grant GM_setValue
  26. // @grant GM_registerMenuCommand
  27. // @grant GM_info
  28. // @run-at document-end
  29. // @require https://ajax.aspnetcdn.com/ajax/jQuery/jquery-3.3.1.min.js
  30. // @connect google.com
  31. // @connect translate.google.cn
  32. // @charset UTF-8
  33. // @supportURL https://github.com/lkytal/GM/issues
  34. // ==/UserScript==
  35.  
  36. "use strict";
  37. ;
  38. var CopyText,
  39. GetOpt,
  40. InTextBox,
  41. OnEngine,
  42. OpenSet,
  43. PopupInit,
  44. PopupLoad,
  45. ReadOpt,
  46. SaveOpt,
  47. SettingWin,
  48. ShowBar,
  49. TimeOutHide,
  50. UpdateLog,
  51. UpdateNotified,
  52. addAdditionalCSS,
  53. addCSS,
  54. ajaxError,
  55. doRequest,
  56. eventFromTextbox,
  57. fixPos,
  58. getLastRange,
  59. get_selection_offsets,
  60. isChrome,
  61. log,
  62. needPrefix,
  63. onTranslate,
  64. parseTranslationGoogle,
  65. popData,
  66. hasProp = {}.hasOwnProperty;
  67.  
  68. window.$ = window.jQuery = jQuery.noConflict(true);
  69.  
  70. popData = {
  71. count: 0,
  72. mouseIn: 0,
  73. bTrans: 0,
  74. additionalCSSLoaded: 0,
  75. codeVersion: 8,
  76. text: "",
  77. mousedownEvent: null,
  78. icons: {
  79. baiduIcon: "",
  80. bingIcon: "",
  81. translateIcon: "",
  82. googleIcon: "",
  83. linkIcon: "",
  84. taobaoIcon: "",
  85. yahooIcon: "",
  86. wikiIcon: "",
  87. jdIcon: "",
  88. eBayIcon: "",
  89. doubanIcon: "",
  90. amazonIcon: "",
  91. youkuIcon: "",
  92. youtubeIcon: "",
  93. tiebaIcon: "",
  94. inSiteIcon: '',
  95. settingIcon: "",
  96. tipDown: "",
  97. tipUp: "",
  98. pending: "",
  99. copyIcon: ""
  100. },
  101. optionList: [{
  102. id: "Trans_st",
  103. text: "显示翻译按钮 / Display 'Translate' button",
  104. defaultValue: 1
  105. }, {
  106. id: "copy_icon_st",
  107. text: "显示复制按钮 / Display 'Copy' button",
  108. defaultValue: 1
  109. }, {
  110. id: "AutoCopy_st",
  111. text: "自动复制选中文字 / Auto copy selections",
  112. defaultValue: 0
  113. }, {
  114. id: "Fade_st",
  115. text: "超时自动隐藏 / Hide after timeout",
  116. defaultValue: 1
  117. }, {
  118. id: "Tab_st",
  119. text: "新标签页打开 / Open in new tabs",
  120. defaultValue: 1
  121. }, {
  122. id: "Focus_st",
  123. text: "前台标签页打开 / Force foreground tabs",
  124. defaultValue: 1
  125. }, {
  126. id: "Iframe_st",
  127. text: "在Iframe中显示/ Activate in iframes",
  128. defaultValue: 0
  129. }, {
  130. id: "Dis_st",
  131. text: "显示于文字上方 / Display above selection",
  132. defaultValue: 1
  133. }, {
  134. id: "Ctrl_st",
  135. text: "仅按下Ctrl时显示 / Only when ctrl pressed",
  136. defaultValue: 0
  137. }, {
  138. id: "userEngine_st",
  139. text: "自定义引擎 / Enable Customize",
  140. defaultValue: 0
  141. }],
  142. userEngines: [],
  143. defaultEngines: [{
  144. id: "UserEngine",
  145. title: "Example of User Engine",
  146. description: "自定义引擎示例 / Example of user engine",
  147. src: "http://lkytal.qiniudn.com/ic.ico",
  148. href: "https://www.google.com/search?newwindow=1&safe=off&q=${text}"
  149. }, {
  150. id: "UserEngine2",
  151. title: "Example Engine use dataURL",
  152. description: "DataURL引擎示例 / Example Engine use dataURL",
  153. src: "",
  154. href: "https://www.google.com/search?q=${text}%20site:${domain}"
  155. }]
  156. };
  157.  
  158. popData.engines = [{
  159. id: "Open_st",
  160. title: "Open As Url",
  161. description: "选中文本视作链接打开 / Open selection as url",
  162. defaultState: 0,
  163. src: popData.icons.linkIcon,
  164. href: '${rawText}'
  165. }, {
  166. id: "Site_st",
  167. title: "Search Current Website",
  168. description: "在当前网站内搜索 / Search in current website",
  169. defaultState: 1,
  170. src: popData.icons.inSiteIcon,
  171. href: 'https://www.google.com/search?newwindow=1&safe=off&q=${text}%20site:${domain}'
  172. }, {
  173. id: "Bing_st",
  174. title: "Search with Bing",
  175. description: "必应搜索 / Search with Bing",
  176. defaultState: 1,
  177. src: popData.icons.bingIcon,
  178. href: 'https://bing.com/search?q=${text}&form=MOZSBR'
  179. }, {
  180. id: "Baidu_st",
  181. title: "百度搜索",
  182. description: "百度搜索 / Search with Baidu",
  183. defaultState: 1,
  184. src: popData.icons.baiduIcon,
  185. href: 'https://www.baidu.com/s?wd=${text}&ie=utf-8'
  186. }, {
  187. id: "Google_st",
  188. title: "谷歌搜索",
  189. description: "谷歌搜索 / Search with Google",
  190. defaultState: 1,
  191. src: popData.icons.googleIcon,
  192. href: 'https://www.google.com/search?newwindow=1&safe=off&q=${text}'
  193. }, {
  194. id: "Yahoo_st",
  195. title: "Search with Yahoo",
  196. description: "雅虎搜索 / Search with Yahoo",
  197. defaultState: 0,
  198. src: popData.icons.yahooIcon,
  199. href: 'https://search.yahoo.com/search;?p=${text}&ei=UTF-8'
  200. }, {
  201. id: "Taobao_st",
  202. title: "搜索淘宝",
  203. description: "搜索淘宝 / Search with Taobao",
  204. defaultState: 1,
  205. src: popData.icons.taobaoIcon,
  206. href: 'https://s.taobao.com/search?q=${text}'
  207. }, {
  208. id: "Tieba_st",
  209. title: "Search in Tieba",
  210. description: "搜索贴吧 / Search with Tieba",
  211. defaultState: 0,
  212. src: popData.icons.tiebaIcon,
  213. href: 'http://tieba.baidu.com/f/search/res?ie=utf-8&qw=${text}'
  214. }, {
  215. id: "youtube_st",
  216. title: "Search with Youtube",
  217. description: "搜索Youtube / Search with Youtube",
  218. defaultState: 1,
  219. src: popData.icons.youtubeIcon,
  220. href: 'https://www.youtube.com/results?search_query=${text}'
  221. }, {
  222. id: "youku_st",
  223. title: "Search with Youku",
  224. description: "搜索优酷 / Search with Youku",
  225. defaultState: 0,
  226. src: popData.icons.youkuIcon,
  227. href: 'http://www.soku.com/search_video/q_${text}'
  228. }, {
  229. id: "amazon_st",
  230. title: "Search with Amazon",
  231. description: "搜索亚马逊/ Search with Amazon",
  232. defaultState: 1,
  233. src: popData.icons.amazonIcon,
  234. href: 'https://www.amazon.com/s/field-keywords=${text}'
  235. }, {
  236. id: "eBay_st",
  237. title: "Search with eBay",
  238. description: "搜索eBay / Search with eBay",
  239. defaultState: 0,
  240. src: popData.icons.eBayIcon,
  241. href: 'http://www.ebay.com/sch/i.html?_nkw=${text}&_sacat=0'
  242. }, {
  243. id: "douban_st",
  244. title: "豆瓣电影",
  245. description: "搜索豆瓣电影 / Search with Douban Movie",
  246. defaultState: 0,
  247. src: popData.icons.doubanIcon,
  248. href: 'https://movie.douban.com/subject_search?search_text=${text}'
  249. }, {
  250. id: "jd_st",
  251. title: "Search with JD",
  252. description: "搜索京东 / Search with JD",
  253. defaultState: 0,
  254. src: popData.icons.jdIcon,
  255. href: 'https://search.jd.com/Search?keyword=${text}&enc=utf-8'
  256. }, {
  257. id: "Wiki_st",
  258. title: "Search with Wiki",
  259. description: "搜索维基百科 / Search with Wikipedia",
  260. defaultState: 0,
  261. src: popData.icons.wikiIcon,
  262. href: 'https://wikipedia.org/wiki/${text}'
  263. }];
  264.  
  265. log = function (msg) {
  266. popData.count += 1;
  267. return console.log("Popup Msg " + popData.count + " : " + msg);
  268. };
  269.  
  270. isChrome = function () {
  271. return navigator.userAgent.indexOf("Chrome") > -1;
  272. };
  273.  
  274. fixPos = function (sel, e) {
  275. var eventLeft, eventTop, fix, m_left, offsetLeft, offsetTop, offsets;
  276. offsets = get_selection_offsets(sel);
  277. offsetTop = offsets[0];
  278. offsetLeft = offsets[1];
  279. if (e != null) {
  280. eventTop = e.pageY;
  281. eventLeft = e.pageX;
  282. if (Math.abs(offsetTop - eventTop) > 50) {
  283. //log offsetTop + " : " + offsetLeft + " <==> " + eventTop + " : " + eventLeft
  284. offsetTop = eventTop - 8;
  285. }
  286. if (Math.abs(offsetLeft - eventLeft) > 50) {
  287. //translate
  288. offsetLeft = eventLeft + 10;
  289. }
  290. } else {
  291. $('#showUpBody').css('margin-left', '60px');
  292. }
  293. if (GetOpt('Dis_st')) {
  294. //UpSide
  295. offsetTop = offsetTop - 2 - $('#ShowUpBox').height();
  296. if (offsetTop - document.documentElement.scrollTop < 40) {
  297. offsetTop = document.documentElement.scrollTop + 40;
  298. }
  299. } else {
  300. offsetTop += 1.1 * offsets[2];
  301. }
  302. m_left = $('#ShowUpBox').width();
  303. fix = 0;
  304. if (offsetLeft - m_left < 4) {
  305. fix = 4 - offsetLeft + m_left;
  306. }
  307. $('#ShowUpBox').css("top", offsetTop + "px").css("left", offsetLeft - m_left + fix + "px");
  308. return $('#popupTip').css('margin-left', m_left - 20 - fix);
  309. };
  310.  
  311. TimeOutHide = function () {
  312. if (popData.mouseIn === 0 && GetOpt("Fade_st") && !popData.bTrans) {
  313. return $('#ShowUpBox').fadeOut(300);
  314. }
  315. };
  316.  
  317. OnEngine = function (e) {
  318. if (isChrome()) {
  319. //log "chrome"
  320. GM_openInTab($(this).data('link'), {
  321. active: GetOpt("Focus_st") === 1
  322. });
  323. } else {
  324. GM_openInTab($(this).data('link'), !GetOpt("Focus_st"));
  325. }
  326. $('#ShowUpBox').fadeOut(200);
  327. return false;
  328. };
  329.  
  330. PopupInit = function () {
  331. var $DivBox, $icon, EngineList, engine, j, k, l, len, len1, len2, ref, ref1, ref2;
  332. $('#ShowUpBox').remove();
  333. EngineList = "<a id='transBtn'> <img title='翻译' src='" + popData.icons.translateIcon + "' /> </a> <a id='copy_btn'> <img title='Copy' src='" + popData.icons.copyIcon + "' /> </a>";
  334. ref = popData.engines;
  335. for (j = 0, len = ref.length; j < len; j++) {
  336. engine = ref[j];
  337. EngineList += "<a id='" + engine.id + "Icon' class='engine'><img title='" + engine.title + "' src='" + engine.src + "' /></a>";
  338. }
  339. if (GetOpt("userEngine_st")) {
  340. ref1 = popData.userEngines;
  341. for (k = 0, len1 = ref1.length; k < len1; k++) {
  342. engine = ref1[k];
  343. EngineList += "<a id='" + engine.id + "Icon' class='userEngine'><img title='" + engine.title + "' src='" + engine.src + "' /></a>";
  344. }
  345. }
  346. $('body').append("<span id=\"ShowUpBox\"> <span id=\"showUpBody\"> <span id=\"popupWrapper\"> " + EngineList + " </span> <span id=\"transPanel\" /> </span> </span>");
  347. $DivBox = $('#ShowUpBox');
  348. $DivBox.hide();
  349. $DivBox.hover(function () {
  350. $(this).fadeTo(150, 1);
  351. return popData.mouseIn = 1;
  352. }, function () {
  353. if (popData.bTrans === 0 && $(this).is(":visible") && $(this).attr("opacity") === 1) {
  354. $(this).fadeTo(300, 0.7);
  355. clearTimeout(popData.timer);
  356. popData.timer = setTimeout(TimeOutHide, 6000);
  357. }
  358. return popData.mouseIn = 0;
  359. });
  360. $('#showUpBody').on("mouseup", function (event) {
  361. if (event.which === 3) {
  362. CopyText();
  363. $('#ShowUpBox').fadeOut(200);
  364. return false;
  365. } else if (event.which === 2) {
  366. GM_openInTab(document.defaultView.getSelection().toString());
  367. return false;
  368. }
  369. });
  370. $('#showUpBody').on("contextmenu", function (event) {
  371. return false;
  372. });
  373. $('#transPanel').on("mouseup contextmenu", function (event) {
  374. return event.stopPropagation();
  375. });
  376. $DivBox.on("click dblclick mousedown mouseup contextmenu", function (event) {
  377. return event.stopPropagation();
  378. });
  379. ref2 = popData.engines;
  380. for (l = 0, len2 = ref2.length; l < len2; l++) {
  381. engine = ref2[l];
  382. $icon = $("#" + engine.id + "Icon");
  383. if (!GetOpt(engine.id)) {
  384. $icon.hide();
  385. }
  386. }
  387. $DivBox.find('.engine, .userEngine').on('click', OnEngine);
  388. $('#transBtn').on("click", onTranslate);
  389. if (!GetOpt('Trans_st')) {
  390. $('#transBtn').hide();
  391. }
  392. $('#copy_btn').on("click", function () {
  393. CopyText(popData.rawText);
  394. return $('#ShowUpBox').fadeOut(200);
  395. });
  396. if (!GetOpt('copy_icon_st')) {
  397. $('#copy_btn').hide();
  398. }
  399. if (GetOpt('Tab_st')) {
  400. $DivBox.find('a').attr('target', '_blank');
  401. } else {
  402. $DivBox.find('a').attr('target', '_self');
  403. }
  404. if (GetOpt('Dis_st')) {
  405. popData.tip = popData.tipUp;
  406. return $DivBox.append('<span id="popupTip" class="tipUp"></span>');
  407. } else {
  408. popData.tip = popData.tipDown;
  409. return $DivBox.prepend('<span id="popupTip" class="tipDown"></span>');
  410. }
  411. };
  412.  
  413. ajaxError = function (res) {
  414. return $('#transPanel').empty().append("<p>Translate Error:<br /> " + res.statusText + " </p>").show();
  415. };
  416.  
  417. onTranslate = function (event) {
  418. event.preventDefault();
  419. popData.bTrans = 1;
  420. $("#transPanel").empty().append("<div style='padding:10px;'><img src='" + popData.icons.pending + "' /></div>").show();
  421. $('#popupWrapper').hide();
  422. fixPos(document.defaultView.getSelection());
  423. return doRequest(0, 2000);
  424. };
  425.  
  426. doRequest = function (i, wait) {
  427. var ErrHandle;
  428. ErrHandle = function () {
  429. return doRequest(i + 1, wait + 2000);
  430. };
  431. if (i >= 2) {
  432. ErrHandle = ajaxError;
  433. }
  434. return GM_xmlhttpRequest({
  435. method: 'POST',
  436. url: 'https://translate.google.cn/translate_a/single',
  437. data: "client=gtx&dj=1&q=" + popData.text + "&sl=auto&tl=auto&ie=UTF-8&oe=UTF-8&source=icon&dt=t&dt=bd",
  438. headers: {
  439. 'Content-Type': 'application/x-www-form-urlencoded'
  440. },
  441. timeout: wait,
  442. onload: parseTranslationGoogle,
  443. onerror: ErrHandle,
  444. ontimeout: ErrHandle
  445. });
  446. };
  447.  
  448. parseTranslationGoogle = function (responseDetails) {
  449. var PickMeaning, RLine, RTxt, Result, j, len, line, ref, sentence;
  450. if (!popData.bTrans) {
  451. return;
  452. }
  453. try {
  454. RTxt = JSON.parse(responseDetails.responseText);
  455. } catch (error) {
  456. return ajaxError(responseDetails);
  457. }
  458. RLine = function () {
  459. var j, len, ref, results;
  460. ref = RTxt.sentences;
  461. results = [];
  462. for (j = 0, len = ref.length; j < len; j++) {
  463. sentence = ref[j];
  464. results.push(sentence.trans);
  465. }
  466. return results;
  467. }().toString();
  468. PickMeaning = function (list) {
  469. var i, item, j, len, results;
  470. results = [];
  471. for (i = j = 0, len = list.length; j < len; i = ++j) {
  472. item = list[i];
  473. if (item.score > 0.005 || i < 3) {
  474. results.push(item.word);
  475. }
  476. }
  477. return results;
  478. };
  479. if (RTxt.dict != null) {
  480. ref = RTxt.dict;
  481. for (j = 0, len = ref.length; j < len; j++) {
  482. line = ref[j];
  483. RLine += "<br>" + (line.pos + " : " + PickMeaning(line.entry));
  484. }
  485. }
  486. Result = "<div id=\"tranRst\" style=\"font-size:13px;overflow:auto;padding:5px 12px;\"> <div style=\"line-height:200%;font-size:13px;\"> " + RLine + " </div> </div>";
  487. $('#transPanel').empty().append(Result).show();
  488. fixPos(document.defaultView.getSelection());
  489. };
  490.  
  491. $(document).on("mousedown", function (event) {
  492. popData.mousedownEvent = event;
  493. if (popData.bTrans === 1) {
  494. PopupInit();
  495. }
  496. return $('#ShowUpBox').fadeOut(200);
  497. });
  498.  
  499. $(document).on("mouseup", function (event) {
  500. if (event.which !== 1) {
  501. return;
  502. }
  503. if (GetOpt('Ctrl_st') && !event.ctrlKey) {
  504. return;
  505. }
  506. return ShowBar(event);
  507. });
  508.  
  509. eventFromTextbox = function (eventList) {
  510. var event, j, len;
  511. for (j = 0, len = eventList.length; j < len; j++) {
  512. event = eventList[j];
  513. if (event != null) {
  514. if ($(event.target).is('textarea, input, *[contenteditable="true"]')) {
  515. //console.log $(event.target)
  516. return true;
  517. }
  518. }
  519. }
  520. return false;
  521. };
  522.  
  523. InTextBox = function (selection) {
  524. var area, j, len, ref;
  525. if (isChrome()) {
  526. return false;
  527. }
  528. ref = $('textarea, input, *[contenteditable="true"]', document);
  529. for (j = 0, len = ref.length; j < len; j++) {
  530. area = ref[j];
  531. if (selection.containsNode(area, true)) {
  532. return true;
  533. }
  534. }
  535. return false;
  536. };
  537.  
  538. ShowBar = function (event) {
  539. var engine, j, k, len, len1, paraList, ref, ref1, sel, setHref;
  540. sel = document.defaultView.getSelection();
  541. if (InTextBox(sel) || eventFromTextbox([event, popData.mousedownEvent])) {
  542. return;
  543. }
  544. popData.rawText = sel.toString();
  545. popData.text = encodeURIComponent(popData.rawText.trim());
  546. if (popData.rawText === '') {
  547. return;
  548. }
  549. if (GetOpt("AutoCopy_st")) {
  550. CopyText(popData.rawText);
  551. }
  552. $('#transPanel').empty().hide();
  553. paraList = {
  554. "\\${rawText}": popData.rawText,
  555. "\\${text}": popData.text,
  556. "\\${domain}": document.domain,
  557. "\\${url}": location.href
  558. };
  559. setHref = function (engine) {
  560. var $engine, href, para, value;
  561. //log engine.id + " : " + engine.href
  562. href = engine.href;
  563. for (para in paraList) {
  564. if (!hasProp.call(paraList, para)) continue;
  565. value = paraList[para];
  566. href = href.replace(RegExp("" + para, "g"), value);
  567. }
  568. $engine = $("#" + engine.id + "Icon");
  569. return $engine.data('link', href);
  570. };
  571. ref = popData.engines;
  572. for (j = 0, len = ref.length; j < len; j++) {
  573. engine = ref[j];
  574. setHref(engine);
  575. }
  576. ref1 = popData.userEngines;
  577. for (k = 0, len1 = ref1.length; k < len1; k++) {
  578. engine = ref1[k];
  579. setHref(engine);
  580. }
  581. if (needPrefix(popData.rawText)) {
  582. $('#Open_stIcon').data('link', "http://" + popData.rawText.trim());
  583. }
  584. popData.mouseIn = 0;
  585. popData.bTrans = 0;
  586. clearTimeout(popData.timer);
  587. popData.timer = setTimeout(TimeOutHide, 6000);
  588. fixPos(sel, event);
  589. return $('#ShowUpBox').css('opacity', 0.9).fadeIn(200);
  590. };
  591.  
  592. needPrefix = function (url) {
  593. var j, len, prefix, urlPrefixes;
  594. urlPrefixes = ['http://', 'https://', 'ftp://', 'file://', 'thunder://', 'ed2k://'];
  595. for (j = 0, len = urlPrefixes.length; j < len; j++) {
  596. prefix = urlPrefixes[j];
  597. if (url.indexOf(prefix) === 0) {
  598. return 0;
  599. }
  600. }
  601. return 1;
  602. };
  603.  
  604. CopyText = function (selText) {
  605. if (selText == null) {
  606. selText = document.defaultView.getSelection().toString();
  607. }
  608. if ((typeof GM_info !== "undefined" && GM_info !== null ? GM_info.scriptHandler : void 0) === "Violentmonkey") {
  609. return document.execCommand('copy');
  610. }
  611. try {
  612. return GM_setClipboard(selText, "text");
  613. } catch (error) {
  614. return document.execCommand('copy');
  615. }
  616. };
  617.  
  618. GetOpt = function (id) {
  619. var ref;
  620. return GM_getValue(id, (ref = popData.optionList.find(function (item) {
  621. return item.id === id;
  622. })) != null ? ref.defaultValue : void 0);
  623. };
  624.  
  625. SaveOpt = function (id) {
  626. return GM_setValue(id, $("#" + id + " > input").prop("checked") + 0);
  627. };
  628.  
  629. ReadOpt = function (id) {
  630. return $("#" + id + " > input").prop("checked", GetOpt(id));
  631. };
  632.  
  633. OpenSet = function () {
  634. if ($('#popup_setting_bg').length === 0) {
  635. SettingWin();
  636. }
  637. return $('#popup_setting_bg').fadeIn(400);
  638. };
  639.  
  640. SettingWin = function () {
  641. var chsJSON, engJSON, engine, engineOptionList, generateEngineOption, generateOption, item, j, len, option, optionList, ref;
  642. addAdditionalCSS();
  643. $('#popup_setting_bg').remove();
  644. generateOption = function (option) {
  645. return "<span id='" + option.id + "' class='setting_item'> <img src=" + popData.icons.settingIcon + " /> <span class='text'>" + option.text + "</span> <input class='tgl tgl-flat' id='" + option.id + "_checkbox' type='checkbox'> <label class='tgl-btn' for='" + option.id + "_checkbox'></label> </span>";
  646. };
  647. optionList = function () {
  648. var j, len, ref, results;
  649. ref = popData.optionList;
  650. results = [];
  651. for (j = 0, len = ref.length; j < len; j++) {
  652. option = ref[j];
  653. results.push(generateOption(option));
  654. }
  655. return results;
  656. }().join(' ');
  657. generateEngineOption = function (engine) {
  658. return "<span id='" + engine.id + "' class='setting_item'> <img src=" + engine.src + " /> <span class='text'>" + engine.description + "</span> <input class='tgl tgl-flat' id='" + engine.id + "_checkbox' type='checkbox'> <label class='tgl-btn' for='" + engine.id + "_checkbox'></label> </span>";
  659. };
  660. engineOptionList = function () {
  661. var j, len, ref, results;
  662. ref = popData.engines;
  663. results = [];
  664. for (j = 0, len = ref.length; j < len; j++) {
  665. engine = ref[j];
  666. results.push(generateEngineOption(engine));
  667. }
  668. return results;
  669. }().join(' ');
  670. engJSON = '[\n {\n id: "UserEngine",\n title: "Example Engine",\n description: "Example of user-defined engine",\n src: "http://lkytal.qiniudn.com/ic.ico",\n href: "https://www.google.com/search?q=${text}"\n }\n]';
  671. chsJSON = '[\n {\n id: "UserEngine",\n title: "Example Engine",\n description: "自定义引擎示例",\n src: "http://lkytal.qiniudn.com/ic.ico",\n href: "https://www.google.com/search?q=${text}"\n }\n]';
  672. $("body").append("<div id='popup_setting_bg'> <div id='popup_setting_win'> <div id='popup_title'>PopUp Search Setting</div> <div id='popup_content'> <div id='tabs_box'> <div id='popup_tab1' class='popup_tab popup_selected'>\u9009\u9879 / General</div> <div id='popup_tab2' class='popup_tab'>\u641C\u7D22\u5F15\u64CE / Engines</div> <div id='popup_tab3' class='popup_tab'>\u81EA\u5B9A\u4E49 / Customize</div> <div id='popup_tab4' class='popup_tab'>\u5173\u4E8E / About</div> </div> <div id='page_box'> <div id='option_box'> <div id='popup_tab1Page'> " + optionList + " </div> <div id='popup_tab2Page'> " + engineOptionList + " </div> <div id='popup_tab3Page'> <div id='editTitle'> <div><b>\u8BF7\u9605\u8BFB\u5E2E\u52A9 / Read HELP first</b></div> <span id='popHelp'><u>Help</u></span> <span id='popReset'><u>Reset</u></span> </div> <textarea id='popup_engines'></textarea> </div> <div id='popup_tab4Page'> <h3>Authored by Lkytal</h3> <p> You can redistribute it under <a href='http://creativecommons.org/licenses/by-nc-sa/4.0/'> Creative Commons Attribution-NonCommercial-ShareAlike Licence </a> </p> <p class='contact-line'> Source Code at<br> Git OSChina <a class='tab-text' href='https://git.oschina.net/coldfire/GM'> https://git.oschina.net/coldfire/GM </a> <br /> Github <a class='tab-text' href='https://github.com/lkytal/GM'> https://github.com/lkytal/GM </a> </p> <p> Contact:<br> Github <a class='tab-text' href='https://github.com/lkytal'> https://github.com/lkytal/ </a> <br> Greasy fork <a class='tab-text' href='https://greasyfork.org/en/users/152-lkytal'> https://greasyfork.org/en/users/152-lkytal </a> </p> </div> </div> <div id='btnArea'> <div id='popup_save' class='setting_btn'>Save</div> <div id='popup_close' class='setting_btn'>Close</div> </div> </div> </div> </div> <div id='popup_help_bg'> <div id='popup_help_win'> <div id='popup_help_content'> <div id='helpLang'> <span id='engHead' class='popup_head_selected'>English</span> <span id='chsHead'>\u4E2D\u6587</span> </div> <div id='help_box'> <div id='engContent'> The content of custom engine should be in standard JSON format, in forms of following: <pre> " + engJSON + " </pre> The 'id' should be unique for every entry and must NOT contain any space character, the 'title' and the 'description' can be any text you like, the 'src' indicates the icon of every item, should be the URL to an image or be a <a href='http://dataurl.net/#about'>DataURL</a>. The 'href' is the link to be open upon click, you may have noticed the '${text}' variable, which will be replaced by selected text. Available variables are listed below: <ul> <li>${text} : will be replaced by the selected text (Url encoded, should use this by default)</li> <li>${rawText} : will be replaced by the original selected text</li> <li>${domain} : will be replaced by the domain of current URL</li> <li>${url} : will be replaced by the current web page's URL</li> </ul> Note: You can't modify built-in engines directly, however, you can disable them and add your own. </div> <div id='chsContent'> \u81EA\u5B9A\u4E49\u5F15\u64CE\u5185\u5BB9\u5E94\u5F53\u662F\u5408\u6CD5\u7684JSON\u683C\u5F0F, \u5982\u4E0B <pre> " + chsJSON + " </pre> \u6BCF\u4E00\u9879\u7684 id \u5FC5\u987B\u5404\u4E0D\u76F8\u540C\u4E14\u4E0D\u80FD\u542B\u6709\u7A7A\u683C, title \u548C description \u53EF\u4EE5\u968F\u610F\u586B\u5199, src \u662F\u8BE5\u9879\u7684\u56FE\u6807, \u53EF\u4EE5\u662F\u6307\u5411\u56FE\u6807\u7684 url \u6216\u8005\u662F\u4E00\u4E2A <a href='http://dataurl.net/#about'>DataURL</a>. href \u662F\u5F15\u64CE\u7684 url \u94FE\u63A5, \u5176\u4E2D\u53EF\u4EE5\u5305\u542B\u8BF8\u5982 ${text} \u8FD9\u6837\u7684\u53D8\u91CF, \u53D8\u91CF\u7684\u5927\u5C0F\u5199\u5FC5\u987B\u6B63\u786E, \u53EF\u7528\u7684\u53D8\u91CF\u6709: <ul> <li>${text} : \u4EE3\u8868\u9009\u4E2D\u6587\u5B57, \u7ECF\u8FC7 url encoding, \u4E00\u822C\u5E94\u5F53\u4F7F\u7528\u6B64\u9879</li> <li>${rawText} : \u4EE3\u8868\u672A\u7ECF encoding \u7684\u539F\u59CB\u9009\u4E2D\u6587\u5B57</li> <li>${domain} : \u4EE3\u8868\u5F53\u524D\u9875\u9762\u7684\u57DF\u540D</li> <li>${url} : \u4EE3\u8868\u5F53\u524D\u9875\u9762\u7684 url \u5730\u5740</li> </ul> \u6CE8\u610F: \u5185\u7F6E\u5F15\u64CE\u65E0\u6CD5\u76F4\u63A5\u4FEE\u6539, \u4F60\u53EF\u4EE5\u7981\u7528\u5B83\u4EEC\u7136\u540E\u6DFB\u52A0\u4F60\u81EA\u5B9A\u4E49\u7684\u5F15\u64CE </div> </div> <div id='help_btnArea'> <div id='popup_help_close' class='setting_btn'>Close</div> </div> </div> </div> </div> </div>");
  673. $("#popup_setting_bg, #popup_help_bg").hide();
  674. $("#tabs_box > .popup_tab").on("click", function (e) {
  675. $("#tabs_box > .popup_tab").removeClass("popup_selected");
  676. $(this).addClass("popup_selected");
  677. $("#option_box > div").hide();
  678. return $("#" + $(this).attr("id") + "Page").show();
  679. });
  680. $("#option_box > div").hide();
  681. $("#tabs_box > .popup_tab.popup_selected").click();
  682. $("#chsContent").hide();
  683. $("#engHead").on('click', function (event) {
  684. $("#engHead").addClass("popup_head_selected");
  685. $("#chsHead").removeClass("popup_head_selected");
  686. $("#engContent").show();
  687. return $("#chsContent").hide();
  688. });
  689. $("#chsHead").on('click', function (event) {
  690. $("#engHead").removeClass("popup_head_selected");
  691. $("#chsHead").addClass("popup_head_selected");
  692. $("#engContent").hide();
  693. return $("#chsContent").show();
  694. });
  695. ref = $("#popup_setting_win .setting_item");
  696. for (j = 0, len = ref.length; j < len; j++) {
  697. item = ref[j];
  698. if (item != null) {
  699. ReadOpt(item.id);
  700. }
  701. }
  702. $("#popup_engines").val(GM_getValue("engineString", popData.defaultEngineString));
  703. $("#popReset").click(function () {
  704. if (confirm("Reset?")) {
  705. return $("#popup_engines").val(popData.defaultEngineString);
  706. }
  707. });
  708. $("#popHelp").click(function () {
  709. return $("#popup_help_bg").fadeIn();
  710. });
  711. if (!GetOpt("userEngine_st")) {
  712. $("#popup_tab3").hide();
  713. }
  714. $("#popup_save").click(function () {
  715. var k, len1, ref1, userEngineString;
  716. ref1 = $("#popup_setting_win .setting_item");
  717. for (k = 0, len1 = ref1.length; k < len1; k++) {
  718. item = ref1[k];
  719. if (item != null) {
  720. SaveOpt(item.id);
  721. }
  722. }
  723. userEngineString = $("#popup_engines").val();
  724. if (userEngineString !== "") {
  725. try {
  726. popData.userEngines = JSON.parse(userEngineString);
  727. GM_setValue("engineString", userEngineString);
  728. } catch (error) {
  729. alert("搜索列表错误!请检查\nEngine config Error!");
  730. log(userEngineString);
  731. }
  732. }
  733. return $("#popup_setting_bg").fadeOut(300, function () {
  734. $("#popup_setting_bg").remove(); //force rebuild setting window
  735. return PopupInit(); //rebuild toolbar
  736. });
  737. });
  738. $("#popup_close, #popup_setting_bg").click(function () {
  739. return $("#popup_setting_bg").fadeOut(300, function () {
  740. return $("#popup_setting_bg").remove();
  741. });
  742. });
  743. $("#popup_help_bg, #popup_help_close").on("click", function (e) {
  744. return $("#popup_help_bg").fadeOut();
  745. });
  746. return $('#popup_setting_win, #popup_help_win, #popup_help_bg').click(function (e) {
  747. return e.stopPropagation();
  748. });
  749. };
  750.  
  751. UpdateLog = function () {
  752. addAdditionalCSS();
  753. $("body").append("<div id='popup_update_bg'> <div id='popup_update_win'> <div id='update_header'> Popup Search Updated (ver 4.3.0) </div> <div id='popup_update_content'> <div id='update_texts'> <p> <h3>此版本引入的重要改变:</h3> <p> 增加了复制按钮 </p> <p> 自定义引擎功能开放, 点击 'Open setting' 可以打开设置并启用该功能. 在自定义前请点击 'Help' 按钮以阅读帮助文档. </p> <p> 注意: 启用自定义引擎后, 重新打开设置窗口才会生效. </p> </p> <p> <h3>What's new in this version:</h3> <p> Provide 'Copy' button </p> <p> You can customize your own engines now. Click 'Open setting' to check it out. Remember to read 'HELP' before modify. </p> <p> Note: After you enabled customization, reopen setting window to take effect. </p> </p> </div> <div id='update_btnArea'> <div id='popup_update_open' class='setting_btn'>Open Setting</div> <div id='popup_update_close' class='setting_btn'>Close</div> </div> </div> </div> </div>");
  754. $('#popup_update_open').on('click', function (event) {
  755. UpdateNotified();
  756. $("#popup_update_bg").hide();
  757. OpenSet();
  758. });
  759. return $('#popup_update_close').on('click', function (event) {
  760. UpdateNotified();
  761. $("#popup_update_bg").hide();
  762. });
  763. };
  764.  
  765. UpdateNotified = function () {
  766. return GM_setValue("UpdateAlert", popData.codeVersion);
  767. };
  768.  
  769. PopupLoad = function () {
  770. var engine, j, k, len, len1, option, popupMenu, ref, ref1, setDefault, userEngineString;
  771. if (window.self !== window.top || window.frameElement) {
  772. if (!GM_getValue("Iframe_st", 0)) {
  773. return;
  774. }
  775. }
  776. addCSS();
  777. if (GM_getValue("UpdateAlert", 0) < popData.codeVersion) {
  778. setDefault = function (key, defaultValue) {
  779. return GM_setValue(key, GM_getValue(key, defaultValue));
  780. };
  781. ref = popData.optionList;
  782. for (j = 0, len = ref.length; j < len; j++) {
  783. option = ref[j];
  784. setDefault(option.id, option.defaultValue);
  785. }
  786. ref1 = popData.engines;
  787. for (k = 0, len1 = ref1.length; k < len1; k++) {
  788. engine = ref1[k];
  789. setDefault(engine.id, engine.defaultState);
  790. }
  791. UpdateLog();
  792. }
  793. popData.defaultEngineString = JSON.stringify(popData.defaultEngines, null, 4);
  794. if (GetOpt("userEngine_st")) {
  795. userEngineString = GM_getValue("engineString", popData.defaultEngineString);
  796. try {
  797. popData.userEngines = JSON.parse(userEngineString);
  798. } catch (error) {
  799. //alert "User Engine Syntax Error"
  800. console.error(userEngineString);
  801. }
  802. }
  803. PopupInit();
  804. GM_registerMenuCommand("Popup Search Setting / 设置", OpenSet, 'p');
  805. if (GM_getValue("PopupMenu", 0)) {
  806. popupMenu = document.body.appendChild(document.createElement("menu"));
  807. popupMenu.outerHTML = '<menu id="userscript-popup" type="context"><menuitem id="PopupSet" label="Popup Search设置"></menuitem></menu>';
  808. document.querySelector("#PopupSet").addEventListener("click", OpenSet, false);
  809. return $(document).on("contextmenu", function () {
  810. return document.body.setAttribute("contextmenu", "userscript-popup");
  811. });
  812. }
  813. };
  814.  
  815. setTimeout(PopupLoad, 100);
  816.  
  817. addCSS = function () {
  818. return GM_addStyle("#ShowUpBox { all: unset; width: auto; height: auto; position: absolute; z-index: 10240; color: black; display: inline-block; line-height: 0; vertical-align: baseline; box-sizing: content-box; } #showUpBody { min-width: 20px; max-width: 750px; min-height: 20px; max-height: 500px; display: block; border:solid 2px rgb(144,144,144); border-radius:1px; background:rgba(252, 252, 252, 1); } #popupWrapper { all: unset; margin: 3px 2px 3.8px 2px; display:block; line-height: 0; font-size:0; } #transPanel { line-height: normal; width: auto; font-size: 16px; overflow: auto; display: none; } #popupWrapper > a { all: unset; margin: 0px 2px; } #popupWrapper img { all: unset; margin: 0px; height: 24px; width: 24px; border-radius: 0px; padding: 0px; display: inline-block; transition-duration: 0.1s; -moz-transition-duration: 0.1s; -webkit-transition-duration: 0.1s; } #popupWrapper img:hover { margin: -1px -1px; height: 26px; width: 26px; } #popupTip { display: inline-block; clear: both; height: 9px; width: 9px; } .tipUp { background: url(" + popData.icons.tipUp + ") 0px 0px no-repeat transparent; margin-top: -2px; margin-bottom: 0px; } .tipDown { background: url(" + popData.icons.tipDown + ") 0px 0px no-repeat transparent; margin-top: 0px; margin-bottom: -2px; } #ShowUpBox a { text-decoration: none; display: inline-block; }");
  819. };
  820.  
  821. addAdditionalCSS = function () {
  822. if (popData.additionalCSSLoaded === 1) {
  823. return;
  824. }
  825. popData.additionalCSSLoaded = 1;
  826. return GM_addStyle("#popup_setting_bg { all: unset; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.2); position: fixed; left: 0px; top: 0px; z-index:102400; font-family: \"Hiragino Sans GB\", \"Microsoft Yahei\", Arial, sans-serif; display: -webkit-flex; display: flex; justify-content: center; align-items: center; } #popup_setting_win { all: unset; width: 760px; height: 90%; box-shadow: 0 0 10px #222; box-sizing: content-box !important; background: rgba(255, 255, 255, 0.98); overflow: hidden; display: -webkit-flex; display: -moz-flex; display: flex; flex-direction: column; } #popup_title { font-size:24px; font-weight: bold; text-align: center; padding: 15px; background: #16A085; color: white; flex-shrink: 0; height: 40px; } #popup_content { flex-grow: 1; flex-shrink: 1; height: calc(100% - 70px); padding: 0px; display: -webkit-flex; display: -moz-flex; display: flex; justify-content: space-between; align-items: stretch; } #tabs_box { display: -webkit-flex; display: -moz-flex; display: flex; flex-direction: column; flex-basis: 25%; flex-shrink: 0; background: #EEE; } .popup_tab { width: 100%; background: #EEE; padding: 15px; font-weight: bold; text-align: center; box-sizing: border-box; cursor: pointer; user-select: none; -moz-user-select: none; -webkit-user-select: none; } .popup_tab:hover { background: #ccc; } .popup_selected { border-right: none; background: #FFF; } .popup_selected:hover { background: #FFF; } #page_box { padding: 20px 30px; flex-grow: 1; flex-shrink: 1; max-height: 100%; display: -webkit-flex; display: -moz-flex; display: flex; flex-direction: column; } #option_box { display: -webkit-flex; display: -moz-flex; display: flex; flex-direction: column; align-items: stretch; flex-grow: 1; flex-shrink: 1; overflow-y: auto; } #option_box > div { scroll-behavior: smooth; flex-grow: 1; display: -webkit-flex; display: -moz-flex; display: flex; flex-direction: column; align-items: stretch; } #popup_engines { flex-grow: 1; border: solid 2px #ddd; text-overflow: clip; white-space: pre; overflow-x: auto; overflow-y: auto; word-wrap: break-word; resize: none; } #editTitle { padding: 0px 0px 15px 0px; display: -webkit-flex; display: -moz-flex; display: flex; justify-content: space-between; } #editTitle div { flex-grow: 1; } #editTitle span { margin-left: 20px; color : #1ABC9C; cursor: pointer; } #btnArea { display: -webkit-flex; display: -moz-flex; display: flex; justify-content: flex-end; margin-top: 20px; flex-shrink: 0; } .setting_btn { display: inline-block; font-size: 16px; text-align: center; mix-width: 50px; padding: 4px 10px 4px 10px; border-radius: 2px; margin: 0px 0px 0px 20px; background: #1ABC9C; color: #fff; cursor: pointer; user-select: none; -moz-user-select: none; -webkit-user-select: none; } .setting_btn:hover { text-shadow: 0px 0px 2px #FFF; } .setting_item { min-height: 28px; font-size: 14px; margin: 5px 0px 10px 0px; display: -webkit-flex; display: -moz-flex; display: flex; align-items: center; } .setting_item > img { width: 24px; height: auto; margin-right: 7px; } .setting_item .text{ flex-grow: 1; font-size: 16px; } #popup_help_bg { all: unset; width: 100%; height: 100%; position: fixed; top: 0; left: 0; background: transparent; display: -webkit-flex; display: flex; justify-content: center; align-items: center; z-index: 10240000; } #popup_help_win { all: unset; width: 650px; height: 80%; box-shadow: 0 0 10px #222; box-sizing: content-box !important; background: rgba(255, 255, 255, 0.98); padding: 20px; display: -webkit-flex; display: flex; flex-direction: column; align-items: stretch; /*overflow: hidden;*/ } #popup_help_content { max-height: 100%; flex-grow: 1; display: -webkit-flex; display: flex; flex-direction: column; align-items: stretch; } #helpLang { flex-grow: 0; flex-shrink: 0; display: -webkit-flex; display: flex; border-bottom: solid 1px #ccc; } #helpLang span { padding: 8px 30px; margin-bottom: -1px; cursor: pointer; user-select: none; -moz-user-select: none; -webkit-user-select: none; } #helpLang span.popup_head_selected { border-top: solid 1px #ccc; border-left: solid 1px #ccc; border-right: solid 1px #ccc; border-bottom: solid 3px #FFF; } #help_box { overflow-y: auto; flex-grow: 1; flex-shrink: 1; margin-top: 15px; } #help_box > div { word-wrap: break-word; } #help_btnArea { flex-grow: 0; flex-shrink: 0; display: -webkit-flex; display: flex; justify-content: flex-end; margin-top: 20px; } #popup_update_bg { all: unset; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.2); position: fixed; left: 0px; top: 0px; z-index: 1024000; font-family: \"Hiragino Sans GB\", \"Microsoft Yahei\", Arial, sans-serif; display: -webkit-flex; display: flex; justify-content: center; align-items: center; } #popup_update_win { all: unset; width: 50%; height: 80%; box-shadow:0 0 10px #222; box-sizing: content-box !important; background: rgba(255, 255, 255, 0.98); overflow: hidden; font-size: 16px; display: -webkit-flex; display: -moz-flex; display: flex; flex-direction: column; } #update_header { font-size: 24px; font-weight: bold; text-align: center; padding: 15px; background: #16A085; color: white; flex-shrink: 0; height: 40px; } #popup_update_content { flex-grow: 1; flex-shrink: 1; height: calc(100% - 70px); overflow: auto; padding: 15px; display: -webkit-flex; display: -moz-flex; display: flex; flex-direction: column; justify-content: space-between; align-items: stretch; } #update_texts{ flex-grow: 1; flex-shrink: 1; } #update_btnArea { flex-grow: 0; flex-shrink: 0; display: -webkit-flex; display: flex; justify-content: flex-end; margin-top: 20px; } .hidden { display: none; } .tgl { display: none; } .tgl, .tgl:after, .tgl:before, .tgl *, .tgl *:after, .tgl *:before, .tgl+.tgl-btn { -webkit-box-sizing: border-box; -moz-box-sizing: border-box; box-sizing: border-box; } .tgl::-moz-selection, .tgl:after::-moz-selection, .tgl:before::-moz-selection, .tgl *::-moz-selection, .tgl *:after::-moz-selection, .tgl *:before::-moz-selection, .tgl+.tgl-btn::-moz-selection { background: none; } .tgl::selection, .tgl:after::selection, .tgl:before::selection, .tgl *::selection, .tgl *:after::selection, .tgl *:before::selection, .tgl+.tgl-btn::selection { background: none; } .tgl+.tgl-btn { outline: 0; display: inline-block; width: 4em; height: 2em; position: relative; cursor: pointer; } .tgl+.tgl-btn:after, .tgl+.tgl-btn:before { position: relative; display: inline-block; content: \"\"; width: 50%; height: 100%; } .tgl+.tgl-btn:after { left: 0; } .tgl+.tgl-btn:before { display: none; } .tgl:checked+.tgl-btn:after { left: 50%; } .tgl-flat+.tgl-btn { padding: 2px; -webkit-transition: all .2s ease; transition: all .2s ease; background: #fff; border: 4px solid #f2f2f2; border-radius: 2em; } .tgl-flat+.tgl-btn:after { -webkit-transition: all .2s ease; transition: all .2s ease; background: #f2f2f2; content: \"\"; border-radius: 1em; } .tgl-flat:checked+.tgl-btn { border: 4px solid #1ABC9C; } .tgl-flat:checked+.tgl-btn:after { left: 50%; background: #1ABC9C; }");
  827. };
  828.  
  829. getLastRange = function (selection) {
  830. var j, rangeNum, ref;
  831. for (rangeNum = j = ref = selection.rangeCount - 1; ref <= 0 ? j <= 0 : j >= 0; rangeNum = ref <= 0 ? ++j : --j) {
  832. if (!selection.getRangeAt(rangeNum).collapsed) {
  833. return selection.getRangeAt(rangeNum);
  834. }
  835. }
  836. return selection.getRangeAt(selection.rangeCount - 1);
  837. };
  838.  
  839. get_selection_offsets = function (selection) {
  840. var $test_span, Rect, lastRange, newRange;
  841. $test_span = $('<span style="display:inline;">x</span>');
  842. // "x" because it must have a height
  843. lastRange = getLastRange(selection);
  844. newRange = document.createRange();
  845. newRange.setStart(lastRange.endContainer, lastRange.endOffset);
  846. newRange.insertNode($test_span[0]);
  847. Rect = $test_span[0].getBoundingClientRect();
  848. $test_span.remove();
  849. return [Rect.top + window.scrollY, Rect.left + window.scrollX, 0, 0];
  850. };