Popup Search

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

当前为 2017-09-23 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Popup Search
  3. // @author lkytal
  4. // @namespace Lkytal
  5. // @homepage https://lkytal.github.io/
  6. // @description Popup search box and translate button (etc) for selected texts
  7. // @include *
  8. // @exclude */test/*.html*
  9. // @exclude http://acid3.acidtests.org/*
  10. // @exclude http://www.acfun.tv/*
  11. // @exclude http://www.sf-express.com/*
  12. // @exclude http://furk.net/*
  13. // @require https://cdn.bootcss.com/jquery/3.2.1/jquery.min.js
  14. // @version 4.2.2
  15. // @icon http://lkytal.qiniudn.com/search.png
  16. // @grant GM_xmlhttpRequest
  17. // @grant GM_addStyle
  18. // @grant unsafeWindow
  19. // @grant GM_openInTab
  20. // @grant GM_setClipboard
  21. // @grant GM_download
  22. // @grant GM_getValue
  23. // @grant GM_setValue
  24. // @grant GM_registerMenuCommand
  25. // @charset UTF-8
  26. // @homepageURL https://git.oschina.net/coldfire/GM
  27. // ==/UserScript==
  28.  
  29. "use strict";;
  30. var CopyText, GetOpt, InTextBox, OnEngine, OpenSet, PopupInit, PopupLoad, ReadOpt, SaveOpt, SettingWin, ShowBar, TimeOutHide, UpdateLog, UpdateNotified, addAdditionalCSS, addCSS, ajaxError, doRequest, eventFromTextbox, fixPos, getLastRange, get_selection_offsets, isChrome, log, needPrefix, onTranslate, parseTranslationGoogle, popData,
  31. hasProp = {}.hasOwnProperty;
  32.  
  33. window.$ = this.$ = this.jQuery = jQuery.noConflict(true);
  34.  
  35. popData = {
  36. count: 0,
  37. mouseIn: 0,
  38. bTrans: 0,
  39. additionalCSSLoaded: 0,
  40. codeVersion: 8,
  41. text: "",
  42. mousedownEvent: null,
  43. icons: {
  44. baiduIcon: "",
  45. bingIcon: "",
  46. translateIcon: "",
  47. googleIcon: "",
  48. linkIcon: "",
  49. taobaoIcon: "",
  50. yahooIcon: "",
  51. wikiIcon: "",
  52. jdIcon: "",
  53. eBayIcon: "",
  54. doubanIcon: "",
  55. amazonIcon: "",
  56. youkuIcon: "",
  57. youtubeIcon: "",
  58. tiebaIcon: "",
  59. inSiteIcon: '',
  60. settingIcon: "",
  61. tipDown: "",
  62. tipUp: "",
  63. pending: ""
  64. },
  65. optionList: [
  66. {
  67. id: "Fade_st",
  68. text: "超时自动隐藏 / Hide after timeout",
  69. defaultValue: 1
  70. }, {
  71. id: "Dis_st",
  72. text: "显示于文字上方 / Display above selection",
  73. defaultValue: 1
  74. }, {
  75. id: "Tab_st",
  76. text: "新标签页打开 / Open in new tabs",
  77. defaultValue: 1
  78. }, {
  79. id: "Focus_st",
  80. text: "前台标签页打开 / Force foreground tabs",
  81. defaultValue: 1
  82. }, {
  83. id: "Iframe_st",
  84. text: "在Iframe中显示/ Load in iframes",
  85. defaultValue: 0
  86. }, {
  87. id: "Copy_st",
  88. text: "自动复制选中文字 / Auto copy selections",
  89. defaultValue: 0
  90. }, {
  91. id: "Ctrl_st",
  92. text: "仅按下Ctrl时显示 / Only when ctrl pressed",
  93. defaultValue: 0
  94. }, {
  95. id: "userEngine_st",
  96. text: "自定义引擎 / Enable Customize",
  97. defaultValue: 0
  98. }
  99. ],
  100. userEngines: [],
  101. defaultEngines: [
  102. {
  103. id: "UserEngine",
  104. title: "Example of User Engine",
  105. description: "自定义引擎示例 / Example of user engine",
  106. src: "http://lkytal.qiniudn.com/ic.ico",
  107. href: "https://www.google.com/search?newwindow=1&safe=off&q=${text}"
  108. }, {
  109. id: "UserEngine2",
  110. title: "Example Engine use dataURL",
  111. description: "DataURL引擎示例 / Example Engine use dataURL",
  112. src: "",
  113. href: "https://www.google.com/search?q=${text}%20site:${domain}"
  114. }
  115. ]
  116. };
  117.  
  118. popData.engines = [
  119. {
  120. id: "Open_st",
  121. title: "Open As Url",
  122. description: "选中文本视作链接打开 / Open selection as url",
  123. defaultState: 0,
  124. src: popData.icons.linkIcon,
  125. href: '${rawText}'
  126. }, {
  127. id: "Site_st",
  128. title: "Search Current Website",
  129. description: "在当前网站内搜索 / Search in current website",
  130. defaultState: 1,
  131. src: popData.icons.inSiteIcon,
  132. href: 'https://www.google.com/search?newwindow=1&safe=off&q=${text}%20site:${domain}'
  133. }, {
  134. id: "Bing_st",
  135. title: "Search with Bing",
  136. description: "必应搜索 / Search with Bing",
  137. defaultState: 1,
  138. src: popData.icons.bingIcon,
  139. href: 'https://bing.com/search?q=${text}&form=MOZSBR'
  140. }, {
  141. id: "Baidu_st",
  142. title: "Search with Baidu",
  143. description: "百度搜索 / Search with Baidu",
  144. defaultState: 1,
  145. src: popData.icons.baiduIcon,
  146. href: 'https://www.baidu.com/s?wd=${text}&ie=utf-8'
  147. }, {
  148. id: "Google_st",
  149. title: "Search with Google",
  150. description: "谷歌搜索 / Search with Google",
  151. defaultState: 1,
  152. src: popData.icons.googleIcon,
  153. href: 'https://www.google.com/search?newwindow=1&safe=off&q=${text}'
  154. }, {
  155. id: "Yahoo_st",
  156. title: "Search with Yahoo",
  157. description: "雅虎搜索 / Search with Yahoo",
  158. defaultState: 0,
  159. src: popData.icons.yahooIcon,
  160. href: 'https://search.yahoo.com/search;?p=${text}&ei=UTF-8'
  161. }, {
  162. id: "Taobao_st",
  163. title: "Search with Taobao",
  164. description: "搜索淘宝 / Search with Taobao",
  165. defaultState: 1,
  166. src: popData.icons.taobaoIcon,
  167. href: 'https://s.taobao.com/search?q=${text}'
  168. }, {
  169. id: "Tieba_st",
  170. title: "Search in Tieba",
  171. description: "搜索贴吧 / Search with Tieba",
  172. defaultState: 0,
  173. src: popData.icons.tiebaIcon,
  174. href: 'http://tieba.baidu.com/f/search/res?ie=utf-8&qw=${text}'
  175. }, {
  176. id: "youtube_st",
  177. title: "Search with Youtube",
  178. description: "搜索Youtube / Search with Youtube",
  179. defaultState: 1,
  180. src: popData.icons.youtubeIcon,
  181. href: 'https://www.youtube.com/results?search_query=${text}'
  182. }, {
  183. id: "youku_st",
  184. title: "Search with Youku",
  185. description: "搜索优酷 / Search with Youku",
  186. defaultState: 0,
  187. src: popData.icons.youkuIcon,
  188. href: 'http://www.soku.com/search_video/q_${text}'
  189. }, {
  190. id: "amazon_st",
  191. title: "Search with Amazon",
  192. description: "搜索亚马逊/ Search with Amazon",
  193. defaultState: 1,
  194. src: popData.icons.amazonIcon,
  195. href: 'https://www.amazon.com/s/field-keywords=${text}'
  196. }, {
  197. id: "eBay_st",
  198. title: "Search with eBay",
  199. description: "搜索eBay / Search with eBay",
  200. defaultState: 0,
  201. src: popData.icons.eBayIcon,
  202. href: 'http://www.ebay.com/sch/i.html?_nkw=${text}&_sacat=0'
  203. }, {
  204. id: "douban_st",
  205. title: "Search with Douban",
  206. description: "搜索豆瓣电影 / Search with Douban Movie",
  207. defaultState: 0,
  208. src: popData.icons.doubanIcon,
  209. href: 'https://movie.douban.com/subject_search?search_text=${text}'
  210. }, {
  211. id: "jd_st",
  212. title: "Search with JD",
  213. description: "搜索京东 / Search with JD",
  214. defaultState: 0,
  215. src: popData.icons.jdIcon,
  216. href: 'https://search.jd.com/Search?keyword=${text}&enc=utf-8'
  217. }, {
  218. id: "Wiki_st",
  219. title: "Search with Wiki",
  220. description: "搜索维基百科 / Search with Wikipedia",
  221. defaultState: 0,
  222. src: popData.icons.wikiIcon,
  223. href: 'https://wikipedia.org/wiki/${text}'
  224. }
  225. ];
  226.  
  227. log = function(msg) {
  228. popData.count += 1;
  229. return console.log("Popup Msg " + popData.count + " : " + msg);
  230. };
  231.  
  232. isChrome = function() {
  233. return navigator.userAgent.indexOf("Chrome") > -1;
  234. };
  235.  
  236. fixPos = function(sel, e) {
  237. var eventLeft, eventTop, fix, m_left, offsetLeft, offsetTop, offsets;
  238. offsets = get_selection_offsets(sel);
  239. offsetTop = offsets[0];
  240. offsetLeft = offsets[1];
  241. if (e != null) {
  242. eventTop = e.pageY;
  243. eventLeft = e.pageX;
  244. if (Math.abs(offsetTop - eventTop) > 50) {
  245. offsetTop = eventTop - 8;
  246. }
  247. if (Math.abs(offsetLeft - eventLeft) > 50) {
  248. offsetLeft = eventLeft + 10;
  249. }
  250. } else {
  251. $('#showUpBody').css('margin-left', '60px');
  252. }
  253. if (GetOpt('Dis_st')) {
  254. offsetTop = offsetTop - 2 - $('#ShowUpBox').height();
  255. if ((offsetTop - document.documentElement.scrollTop) < 40) {
  256. offsetTop = document.documentElement.scrollTop + 40;
  257. }
  258. } else {
  259. offsetTop += 1.1 * offsets[2];
  260. }
  261. m_left = $('#ShowUpBox').width();
  262. fix = 0;
  263. if (offsetLeft - m_left < 4) {
  264. fix = 4 - offsetLeft + m_left;
  265. }
  266. $('#ShowUpBox').css("top", offsetTop + "px").css("left", (offsetLeft - m_left + fix) + "px");
  267. return $('#popupTip').css('margin-left', m_left - 20 - fix);
  268. };
  269.  
  270. TimeOutHide = function() {
  271. if (popData.mouseIn === 0 && GetOpt("Fade_st") && !popData.bTrans) {
  272. return $('#ShowUpBox').fadeOut(600);
  273. }
  274. };
  275.  
  276. OnEngine = function(e) {
  277. if (isChrome()) {
  278. GM_openInTab($(this).attr('href'), {
  279. active: GetOpt("Focus_st") === 1
  280. });
  281. } else {
  282. GM_openInTab($(this).attr('href'), !GetOpt("Focus_st"));
  283. }
  284. $('#ShowUpBox').fadeOut(200);
  285. return false;
  286. };
  287.  
  288. PopupInit = function() {
  289. var $DivBox, $icon, EngineList, engine, j, k, l, len, len1, len2, ref, ref1, ref2;
  290. $('#ShowUpBox').remove();
  291. EngineList = "<a id='transBtn'><img title='Translate' src='" + popData.icons.translateIcon + "' /></a>";
  292. ref = popData.engines;
  293. for (j = 0, len = ref.length; j < len; j++) {
  294. engine = ref[j];
  295. EngineList += "<a id='" + engine.id + "Icon' class='engine' href=''><img title='" + engine.title + "' src='" + engine.src + "' /></a>";
  296. }
  297. if (GetOpt("userEngine_st")) {
  298. ref1 = popData.userEngines;
  299. for (k = 0, len1 = ref1.length; k < len1; k++) {
  300. engine = ref1[k];
  301. EngineList += "<a id='" + engine.id + "Icon' class='userEngine' href=''><img title='" + engine.title + "' src='" + engine.src + "' /></a>";
  302. }
  303. }
  304. $('body').append("<span id=\"ShowUpBox\"> <span id=\"showUpBody\"> <span id=\"popupWrapper\"> " + EngineList + " </span> <span id=\"transPanel\" /> </span> </span>");
  305. $DivBox = $('#ShowUpBox');
  306. $DivBox.hide();
  307. $DivBox.hover(function() {
  308. $(this).fadeTo(150, 1);
  309. return popData.mouseIn = 1;
  310. }, function() {
  311. if (popData.bTrans === 0 && $(this).is(":visible") && $(this).attr("opacity") === 1) {
  312. $(this).fadeTo(300, 0.7);
  313. clearTimeout(popData.timer);
  314. popData.timer = setTimeout(TimeOutHide, 6000);
  315. }
  316. return popData.mouseIn = 0;
  317. });
  318. $('#showUpBody').on("mouseup", function(event) {
  319. if (event.which === 3) {
  320. CopyText();
  321. $('#ShowUpBox').fadeOut(200);
  322. return false;
  323. } else if (event.which === 2) {
  324. GM_openInTab(document.defaultView.getSelection().toString());
  325. return false;
  326. }
  327. });
  328. $('#showUpBody').on("contextmenu", function(event) {
  329. return false;
  330. });
  331. $('#transPanel').on("mouseup contextmenu", function(event) {
  332. return event.stopPropagation();
  333. });
  334. $DivBox.on("click dblclick mousedown mouseup contextmenu", function(event) {
  335. return event.stopPropagation();
  336. });
  337. ref2 = popData.engines;
  338. for (l = 0, len2 = ref2.length; l < len2; l++) {
  339. engine = ref2[l];
  340. $icon = $("#" + engine.id + "Icon");
  341. if (!GetOpt(engine.id)) {
  342. $icon.hide();
  343. }
  344. }
  345. $('#ShowUpBox').find('.engine, .userEngine').on('click', OnEngine);
  346. $('#transBtn').on("click", onTranslate);
  347. if (GetOpt('Tab_st')) {
  348. $DivBox.find('a').attr('target', '_blank');
  349. } else {
  350. $DivBox.find('a').attr('target', '_self');
  351. }
  352. if (GetOpt('Dis_st')) {
  353. popData.tip = popData.tipUp;
  354. return $DivBox.append('<span id="popupTip" class="tipUp"></span>');
  355. } else {
  356. popData.tip = popData.tipDown;
  357. return $DivBox.prepend('<span id="popupTip" class="tipDown"></span>');
  358. }
  359. };
  360.  
  361. ajaxError = function(res) {
  362. return $('#transPanel').empty().append("<p>Translate Error:<br /> " + res.statusText + " </p>").show();
  363. };
  364.  
  365. onTranslate = function(event) {
  366. event.preventDefault();
  367. popData.bTrans = 1;
  368. $("#transPanel").empty().append("<div style='padding:10px;'><img src='" + popData.icons.pending + "' /></div>").show();
  369. $('#popupWrapper').hide();
  370. fixPos(document.defaultView.getSelection());
  371. return doRequest(0, 1500);
  372. };
  373.  
  374. doRequest = function(i, wait) {
  375. var ErrHandle;
  376. ErrHandle = function() {
  377. return doRequest(i + 1, wait + 2000);
  378. };
  379. if (i >= 2) {
  380. ErrHandle = ajaxError;
  381. }
  382. return GM_xmlhttpRequest({
  383. method: 'POST',
  384. url: 'https://translate.google.cn/translate_a/single',
  385. data: "client=gtx&dj=1&q=" + popData.text + "&sl=auto&tl=auto&ie=UTF-8&oe=UTF-8&source=icon&dt=t&dt=bd",
  386. headers: {
  387. 'Content-Type': 'application/x-www-form-urlencoded'
  388. },
  389. timeout: wait,
  390. onload: parseTranslationGoogle,
  391. onerror: ErrHandle,
  392. ontimeout: ErrHandle
  393. });
  394. };
  395.  
  396. parseTranslationGoogle = function(responseDetails) {
  397. var PickMeaning, RLine, RTxt, Result, line, sentence;
  398. if (!popData.bTrans) {
  399. return;
  400. }
  401. try {
  402. RTxt = JSON.parse(responseDetails.responseText);
  403. } catch (error) {
  404. return ajaxError(responseDetails);
  405. }
  406. RLine = (function() {
  407. var j, len, ref, results;
  408. ref = RTxt.sentences;
  409. results = [];
  410. for (j = 0, len = ref.length; j < len; j++) {
  411. sentence = ref[j];
  412. results.push(sentence.trans);
  413. }
  414. return results;
  415. })();
  416. PickMeaning = function(list) {
  417. var i, item, j, len, results;
  418. results = [];
  419. for (i = j = 0, len = list.length; j < len; i = ++j) {
  420. item = list[i];
  421. if (item.score > 0.005 || i < 3) {
  422. results.push(item.word);
  423. }
  424. }
  425. return results;
  426. };
  427. if (RTxt.dict != null) {
  428. RLine += (function() {
  429. var j, len, ref, results;
  430. ref = RTxt.dict;
  431. results = [];
  432. for (j = 0, len = ref.length; j < len; j++) {
  433. line = ref[j];
  434. results.push("<br>" + (line.pos + " : " + (PickMeaning(line.entry))));
  435. }
  436. return results;
  437. })();
  438. }
  439. Result = "<div id=\"tranRst\" style=\"font-size:13px;overflow:auto;padding:5px 12px;\"> <div style=\"line-height:200%;font-size:13px;\"> " + RLine + " </div> </div>";
  440. $('#transPanel').empty().append(Result).show();
  441. fixPos(document.defaultView.getSelection());
  442. };
  443.  
  444. $(document).on("mousedown", function(event) {
  445. popData.mousedownEvent = event;
  446. if (popData.bTrans === 1) {
  447. PopupInit();
  448. }
  449. return $('#ShowUpBox').hide();
  450. });
  451.  
  452. $(document).on("mouseup", function(event) {
  453. if (event.which !== 1) {
  454. return;
  455. }
  456. if (GetOpt('Ctrl_st') && !event.ctrlKey) {
  457. return;
  458. }
  459. return ShowBar(event);
  460. });
  461.  
  462. eventFromTextbox = function(eventList) {
  463. var event, j, len;
  464. for (j = 0, len = eventList.length; j < len; j++) {
  465. event = eventList[j];
  466. if (event != null) {
  467. if ($(event.target).is('textarea, input, *[contenteditable="true"]')) {
  468. return true;
  469. }
  470. }
  471. }
  472. return false;
  473. };
  474.  
  475. InTextBox = function(selection) {
  476. var area, j, len, ref;
  477. if (isChrome()) {
  478. return false;
  479. }
  480. ref = $('textarea, input, *[contenteditable="true"]', document);
  481. for (j = 0, len = ref.length; j < len; j++) {
  482. area = ref[j];
  483. if (selection.containsNode(area, true)) {
  484. return true;
  485. }
  486. }
  487. return false;
  488. };
  489.  
  490. ShowBar = function(event) {
  491. var engine, j, k, len, len1, paraList, ref, ref1, sel, setHref;
  492. sel = document.defaultView.getSelection();
  493. if (InTextBox(sel) || eventFromTextbox([event, popData.mousedownEvent])) {
  494. return;
  495. }
  496. popData.rawText = sel.toString();
  497. popData.text = encodeURIComponent(popData.rawText);
  498. if (popData.rawText === '') {
  499. return;
  500. }
  501. if (GetOpt("Copy_st")) {
  502. CopyText(popData.rawText);
  503. }
  504. $('#transPanel').empty().hide();
  505. paraList = {
  506. "\\${rawText}": popData.rawText,
  507. "\\${text}": popData.text,
  508. "\\${domain}": document.domain,
  509. "\\${url}": location.href
  510. };
  511. setHref = function(engine) {
  512. var $engine, href, para, value;
  513. href = engine.href;
  514. for (para in paraList) {
  515. if (!hasProp.call(paraList, para)) continue;
  516. value = paraList[para];
  517. href = href.replace(RegExp("" + para, "g"), value);
  518. }
  519. $engine = $("#" + engine.id + "Icon");
  520. return $engine.attr("href", href);
  521. };
  522. ref = popData.engines;
  523. for (j = 0, len = ref.length; j < len; j++) {
  524. engine = ref[j];
  525. setHref(engine);
  526. }
  527. ref1 = popData.userEngines;
  528. for (k = 0, len1 = ref1.length; k < len1; k++) {
  529. engine = ref1[k];
  530. setHref(engine);
  531. }
  532. if (needPrefix(popData.rawText)) {
  533. $('#Open_stIcon').attr('href', "http://" + popData.rawText);
  534. }
  535. popData.mouseIn = 0;
  536. popData.bTrans = 0;
  537. clearTimeout(popData.timer);
  538. popData.timer = setTimeout(TimeOutHide, 6000);
  539. fixPos(sel, event);
  540. return $('#ShowUpBox').css('opacity', 0.9).fadeIn(150);
  541. };
  542.  
  543. needPrefix = function(url) {
  544. var j, len, prefix, urlPrefixes;
  545. urlPrefixes = ['http://', 'https://', 'ftp://', 'file://', 'thunder://', 'ed2k://'];
  546. for (j = 0, len = urlPrefixes.length; j < len; j++) {
  547. prefix = urlPrefixes[j];
  548. if (url.indexOf(prefix) === 0) {
  549. return 0;
  550. }
  551. }
  552. return 1;
  553. };
  554.  
  555. CopyText = function(selText) {
  556. if (selText == null) {
  557. selText = document.defaultView.getSelection().toString();
  558. }
  559. try {
  560. return GM_setClipboard(selText, "text");
  561. } catch (error) {
  562. alert("ERROR: Auto copy not supported and will be disabled now");
  563. return SaveOpt("Copy_st", 0);
  564. }
  565. };
  566.  
  567. GetOpt = function(id) {
  568. return GM_getValue(id);
  569. };
  570.  
  571. SaveOpt = function(id) {
  572. return GM_setValue(id, $("#" + id + " > input").prop("checked") + 0);
  573. };
  574.  
  575. ReadOpt = function(id) {
  576. return $("#" + id + " > input").prop("checked", GM_getValue(id));
  577. };
  578.  
  579. OpenSet = function() {
  580. if ($('#popup_setting_bg').length === 0) {
  581. SettingWin();
  582. }
  583. return $('#popup_setting_bg').fadeIn(400);
  584. };
  585.  
  586. SettingWin = function() {
  587. var chsJSON, engJSON, engine, engineOptionList, generateEngineOption, generateOption, item, j, len, option, optionList, ref;
  588. addAdditionalCSS();
  589. $('#popup_setting_bg').remove();
  590. generateOption = function(option) {
  591. 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>";
  592. };
  593. optionList = ((function() {
  594. var j, len, ref, results;
  595. ref = popData.optionList;
  596. results = [];
  597. for (j = 0, len = ref.length; j < len; j++) {
  598. option = ref[j];
  599. results.push(generateOption(option));
  600. }
  601. return results;
  602. })()).join(' ');
  603. generateEngineOption = function(engine) {
  604. 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>";
  605. };
  606. engineOptionList = ((function() {
  607. var j, len, ref, results;
  608. ref = popData.engines;
  609. results = [];
  610. for (j = 0, len = ref.length; j < len; j++) {
  611. engine = ref[j];
  612. results.push(generateEngineOption(engine));
  613. }
  614. return results;
  615. })()).join(' ');
  616. 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]';
  617. 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]';
  618. $("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'>选项 / General</div> <div id='popup_tab2' class='popup_tab'>搜索引擎 / Engines</div> <div id='popup_tab3' class='popup_tab'>自定义 / Customize</div> <div id='popup_tab4' class='popup_tab'>关于 / 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>请阅读帮助 / 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'>中文</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 entry, should be the url to an image or 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 bellow: <ul> <li>${text} : will be replaced by selected text (Url encoded, should use this in default)</li> <li>${rawText} : will be replaced by untouched selected text</li> <li>${domain} : will be replaced by current url\'s domain</li> <li>${url} : will be replaced by current webpage\'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'> 自定义引擎内容应当是合法的JSON格式, 如下 <pre> " + chsJSON + " </pre> 每一项的 id 必须各不相同且不能含有空格, title 和 description 可以随意填写, src 是该项的图标, 可以是指向图标的 url 或者是一个 <a href='http://dataurl.net/#about'>DataURL</a>. href 是引擎的 url 链接, 其中可以包含诸如 ${text} 这样的变量, 变量的大小写必须正确, 可用的变量有: <ul> <li>${text} : 代表选中文字, 经过 url encoding, 一般应当使用此项</li> <li>${rawText} : 代表未经 encoding 的原始选中文字</li> <li>${domain} : 代表当前页面的域名</li> <li>${url} : 代表当前页面的 url 地址</li> </ul> 注意: 内置引擎无法直接修改, 你可以禁用它们然后添加你自定义的引擎 </div> </div> <div id='help_btnArea'> <div id='popup_help_close' class='setting_btn'>Close</div> </div> </div> </div> </div> </div>");
  619. $("#popup_setting_bg, #popup_help_bg").hide();
  620. $("#tabs_box > .popup_tab").on("click", function(e) {
  621. $("#tabs_box > .popup_tab").removeClass("popup_selected");
  622. $(this).addClass("popup_selected");
  623. $("#option_box > div").hide();
  624. return $("#" + $(this).attr("id") + "Page").show();
  625. });
  626. $("#option_box > div").hide();
  627. $("#tabs_box > .popup_tab.popup_selected").click();
  628. $("#chsContent").hide();
  629. $("#engHead").on('click', function(event) {
  630. $("#engHead").addClass("popup_head_selected");
  631. $("#chsHead").removeClass("popup_head_selected");
  632. $("#engContent").show();
  633. return $("#chsContent").hide();
  634. });
  635. $("#chsHead").on('click', function(event) {
  636. $("#engHead").removeClass("popup_head_selected");
  637. $("#chsHead").addClass("popup_head_selected");
  638. $("#engContent").hide();
  639. return $("#chsContent").show();
  640. });
  641. ref = $("#popup_setting_win .setting_item");
  642. for (j = 0, len = ref.length; j < len; j++) {
  643. item = ref[j];
  644. if (item != null) {
  645. ReadOpt(item.id);
  646. }
  647. }
  648. $("#popup_engines").val(GM_getValue("engineString", popData.defaultEngineString));
  649. $("#popReset").click(function() {
  650. if (confirm("Reset?")) {
  651. return $("#popup_engines").val(popData.defaultEngineString);
  652. }
  653. });
  654. $("#popHelp").click(function() {
  655. return $("#popup_help_bg").fadeIn();
  656. });
  657. if (!GetOpt("userEngine_st")) {
  658. $("#popup_tab3").hide();
  659. }
  660. $("#popup_save").click(function() {
  661. var k, len1, ref1, userEngineString;
  662. ref1 = $("#popup_setting_win .setting_item");
  663. for (k = 0, len1 = ref1.length; k < len1; k++) {
  664. item = ref1[k];
  665. if (item != null) {
  666. SaveOpt(item.id);
  667. }
  668. }
  669. userEngineString = $("#popup_engines").val();
  670. if (userEngineString !== "") {
  671. try {
  672. popData.userEngines = JSON.parse(userEngineString);
  673. GM_setValue("engineString", userEngineString);
  674. } catch (error) {
  675. alert("搜索列表错误!请检查\nEngine config Error!");
  676. log(userEngineString);
  677. }
  678. }
  679. return $("#popup_setting_bg").fadeOut(300, function() {
  680. $("#popup_setting_bg").remove();
  681. return PopupInit();
  682. });
  683. });
  684. $("#popup_close, #popup_setting_bg").click(function() {
  685. return $("#popup_setting_bg").fadeOut(300, function() {
  686. return $("#popup_setting_bg").remove();
  687. });
  688. });
  689. $("#popup_help_bg, #popup_help_close").on("click", function(e) {
  690. return $("#popup_help_bg").fadeOut();
  691. });
  692. return $('#popup_setting_win, #popup_help_win, #popup_help_bg').click(function(e) {
  693. return e.stopPropagation();
  694. });
  695. };
  696.  
  697. UpdateLog = function() {
  698. addAdditionalCSS();
  699. $("body").append("<div id='popup_update_bg'> <div id='popup_update_win'> <div id='update_header'> Popup Search Updated (ver 4.1.0) </div> <div id='popup_update_content'> <div id='update_texts'> <p> <h3>此版本引入的重要改变:</h3> <p> 自定义引擎功能开放, 点击 'Open setting' 可以打开设置并启用该功能. 在自定义前请点击 'Help' 按钮以阅读帮助文档. </p> <p> 注意: 启用自定义引擎后, 重新打开设置窗口才会生效. </p> </p> <p> <h3>What's new in this version:</h3> <p> You can customize your own engines now. Click 'Open setting' to check it out. Remember to read 'HELP' before edit. </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>");
  700. $('#popup_update_open').on('click', function(event) {
  701. UpdateNotified();
  702. $("#popup_update_bg").hide();
  703. OpenSet();
  704. });
  705. return $('#popup_update_close').on('click', function(event) {
  706. UpdateNotified();
  707. $("#popup_update_bg").hide();
  708. });
  709. };
  710.  
  711. UpdateNotified = function() {
  712. return GM_setValue("UpdateAlert", popData.codeVersion);
  713. };
  714.  
  715. PopupLoad = function() {
  716. var engine, j, k, len, len1, option, popupMenu, ref, ref1, setDefault, userEngineString;
  717. if (window.self !== window.top || window.frameElement) {
  718. if (!GM_getValue("Iframe_st", 0)) {
  719. return;
  720. }
  721. }
  722. addCSS();
  723. if (GM_getValue("UpdateAlert", 0) < popData.codeVersion) {
  724. setDefault = function(key, defaultValue) {
  725. return GM_setValue(key, GM_getValue(key, defaultValue));
  726. };
  727. ref = popData.optionList;
  728. for (j = 0, len = ref.length; j < len; j++) {
  729. option = ref[j];
  730. setDefault(option.id, option.defaultValue);
  731. }
  732. ref1 = popData.engines;
  733. for (k = 0, len1 = ref1.length; k < len1; k++) {
  734. engine = ref1[k];
  735. setDefault(engine.id, engine.defaultState);
  736. }
  737. UpdateLog();
  738. }
  739. popData.defaultEngineString = JSON.stringify(popData.defaultEngines, null, 4);
  740. if (GetOpt("userEngine_st")) {
  741. userEngineString = GM_getValue("engineString", popData.defaultEngineString);
  742. try {
  743. popData.userEngines = JSON.parse(userEngineString);
  744. } catch (error) {
  745. log(userEngineString);
  746. }
  747. }
  748. PopupInit();
  749. GM_registerMenuCommand("Popup Search Setting / 设置", OpenSet, 'p');
  750. if (GM_getValue("PopupMenu", 0)) {
  751. popupMenu = document.body.appendChild(document.createElement("menu"));
  752. popupMenu.outerHTML = '<menu id="userscript-popup" type="context"><menuitem id="PopupSet" label="Popup Search设置"></menuitem></menu>';
  753. document.querySelector("#PopupSet").addEventListener("click", OpenSet, false);
  754. return $(document).on("contextmenu", function() {
  755. return document.body.setAttribute("contextmenu", "userscript-popup");
  756. });
  757. }
  758. };
  759.  
  760. setTimeout(PopupLoad, 100);
  761.  
  762. addCSS = function() {
  763. 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; }");
  764. };
  765.  
  766. addAdditionalCSS = function() {
  767. if (popData.additionalCSSLoaded === 1) {
  768. return;
  769. }
  770. popData.additionalCSSLoaded = 1;
  771. 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; }");
  772. };
  773.  
  774. getLastRange = function(selection) {
  775. var j, rangeNum, ref;
  776. for (rangeNum = j = ref = selection.rangeCount - 1; ref <= 0 ? j <= 0 : j >= 0; rangeNum = ref <= 0 ? ++j : --j) {
  777. if (!selection.getRangeAt(rangeNum).collapsed) {
  778. return selection.getRangeAt(rangeNum);
  779. }
  780. }
  781. return selection.getRangeAt(selection.rangeCount - 1);
  782. };
  783.  
  784. get_selection_offsets = function(selection) {
  785. var $test_span, Rect, lastRange, newRange;
  786. $test_span = $('<span style="display:inline;">x</span>');
  787. lastRange = getLastRange(selection);
  788. newRange = document.createRange();
  789. newRange.setStart(lastRange.endContainer, lastRange.endOffset);
  790. newRange.insertNode($test_span[0]);
  791. Rect = $test_span[0].getBoundingClientRect();
  792. $test_span.remove();
  793. return [Rect.top + window.scrollY, Rect.left + window.scrollX, 0, 0];
  794. };