🏠 Home 

DuckDuckGo Extended [fork]

Extends DDG by adding a customizable list of additional search engines for making fast searches from other engines.


Install this script?
  1. // ==UserScript==
  2. // @name DuckDuckGo Extended [fork]
  3. // @description Extends DDG by adding a customizable list of additional search engines for making fast searches from other engines.
  4. // @namespace userscripts.org/users/439657
  5. // @homepage http://userscripts-mirror.org/scripts/show/129505
  6. // @icon http://s3.amazonaws.com/uso_ss/icon/129505/large.png?1368599692
  7. // @OLDupdateURL https://userscripts.org/scripts/source/129505.meta.js
  8. // @OLDdownloadURL https://userscripts.org/scripts/source/129505.user.js
  9. // @match *://duckduckgo.com/*
  10. // @exclude *://duckduckgo.com/post2.html
  11. // @match http://mycroftproject.com/*
  12. // @grant GM_addStyle
  13. // @grant GM_getValue
  14. // @grant GM_setValue
  15. // @grant GM_xmlhttpRequest
  16. // @version 2.0.4-joey3
  17. // @author tumpio
  18. // ==/UserScript==
  19. var ddg_e = {
  20. list: document.createElement("ol"),
  21. engines: [],
  22. default: "Google==http://www.google.com/search?q={searchTerms}\
  23. ;;Images==http://www.bing.com/images/search?q={searchTerms}&FORM=BIFD\
  24. ;;Wiki==http://en.wikipedia.org/w/index.php?title=Special%3ASearch&profile=default&search={searchTerms}\
  25. ;;Maps==http://maps.google.com/maps?q={searchTerms}\
  26. ;;YouTube==http://www.youtube.com/r###lts?search_query={searchTerms}&aq=f\
  27. ;;IMDb==http://www.imdb.com/find?s=all&q={searchTerms}\
  28. ;;Music==http://www.musicsmasher.net/#{searchTerms}\
  29. ;;Facebook==http://www.facebook.com/search.php?q={searchTerms}\
  30. ;;Google+==https://plus.google.com/s/{searchTerms}\
  31. ;;twitter==https://twitter.com/#!/search/realtime/{searchTerms}\
  32. ;;Amazon==http://www.amazon.com/gp/search?ie=UTF8&keywords={searchTerms}\
  33. ;;Torrents==http://scrapetorrent.com/Search/index.php?search={searchTerms}&sort=seed&fz=&zs=&cat=\
  34. ;;filesTube==http://www.filestube.com/search.html?q={searchTerms}\
  35. ;;Translate==http://translate.google.com/translate_t#auto|en|{searchTerms}",
  36. style: "#ddg_extented { float:left; width:100%; position:relative; top:0px; z-index:999 }\
  37. #ddg_extented ol { clear:left; float:right; list-style:none; position:relative; right:50%; text-align:center; margin:0; padding:0 }\
  38. #ddg_extented li { display:inline; list-style:none; position:relative; left:50%; box-shadow:0 2px 5px 0 #888; margin:0; padding:0; background:#b60002 }\
  39. #ddg_extented li:first-child { border-bottom-left-radius: 10px }\
  40. #ddg_extented li:last-child { border-bottom-right-radius: 10px }\
  41. #ddg_extented li a:link,li a:visited { text-decoration:none; color:#fff; font-size:1.0em; font-weight:700; padding:0 9px }\
  42. #ddg_extented li a:hover { color:#91C5EE }\
  43. #ddg_extented li:hover { box-shadow:0 3px 3px 0 #888 }\
  44. #ddg_extented li.disabled a {pointer-events: none }\
  45. body #header_wrapper #header #header_content_wrapper #header_content #header_button_wrapper #header_button #header_button_menu_wrapper #header_button_menu li.disabled {display: none!important;}\
  46. #ddg_e_save a { background: #88FF61!important }\
  47. #ddg_e_cancel a { background: #FF8861!important }",
  48. get: function() {
  49. var e = GM_getValue("engines", this.
  50. default).split(";;");
  51. var a = [];
  52. for (var i = 0; i < e.length; i++)
  53. a.push(e[i].split("=="));
  54. this.length = a.length;
  55. return a;
  56. },
  57. set: function() {
  58. var e = "";
  59. for (var i = 0; i < this.engines.length; i++) {
  60. e += ";;" + this.engines[i].textContent + "==" + this.engines[i].firstChild.getAttribute("data-engine");
  61. }
  62. GM_setValue("engines", e.substr(2));
  63. },
  64. newList: function() {
  65. var l = "";
  66. var e = this.get();
  67. for (var i = 0; i < e.length; i++)
  68. l += '<li draggable="true" value=' + (i + 1) + '"><a data-engine="' + e[i][1] + '"href="#' + e[i][1] + '">' + e[i][0] + "</a></li>";
  69. this.list.innerHTML = l;
  70. this.engines = this.list.getElementsByTagName("li");
  71. // Joey added this so that clicks work again.
  72. // I don't know how they worked before.
  73. // Were they handled by DDG itself?
  74. // Or is it somewhere in this script that broke in Greasemonkey?
  75. for (var i = 0; i < this.engines.length; i++)
  76. this.engines[i].firstChild.onclick = onHashChange;
  77. },
  78. append: function(h) {
  79. var e = document.createElement("div");
  80. var b = h.style.background;
  81. if (b !== "")
  82. this.style += "#ddg_extented li { background:" + b + "!important }";
  83. GM_addStyle(this.style);
  84. e.setAttribute("id", "ddg_extented");
  85. e.appendChild(this.list);
  86. this.newList();
  87. h.appendChild(e);
  88. }
  89. };
  90. var options = {
  91. size: 7,
  92. list: [],
  93. strings: [
  94. ["Find new", "Find and add new search engines from MyCroft"],
  95. ["Reorder", "Drag and drop search engines"],
  96. ["Rename", "Click to rename search engines"],
  97. ["Remove", "Click to remove search engines"],
  98. ["Restore all", "Restores all default search engines"],
  99. ["Save", "Saves modified search engines"],
  100. ["Cancel", "Cancels all recent modifications"]
  101. ],
  102. create: function(m) {
  103. m.innerHTML += '<li class="header_button_menu_header">DDG Extended</li>';
  104. for (var i = 0, li; i < this.size; i++) {
  105. li = document.createElement("li");
  106. li.className = "enabled";
  107. li.innerHTML = '<a tabindex="-1" title="' + this.strings[i][1] + '">' + this.strings[i][0] + "</a>";
  108. this.list.push(li);
  109. m.appendChild(this.list[i]);
  110. }
  111. this.list[5].className = "disabled";
  112. this.list[6].className = "disabled";
  113. this.list[5].id = "ddg_e_save";
  114. this.list[6].id = "ddg_e_cancel";
  115. this.list[0].addEventListener("click", newEngine, false);
  116. this.list[1].addEventListener("click", enableDrag, false);
  117. this.list[2].addEventListener("click", enableRename, false);
  118. this.list[3].addEventListener("click", enableRemove, false);
  119. this.list[4].addEventListener("click", restoreEngines, false);
  120. this.list[5].addEventListener("click", saveActions, false);
  121. this.list[6].addEventListener("click", cancelActions, false);
  122. },
  123. toggleClass: function() {
  124. for (var i = 0; i < this.list.length; i++) {
  125. if (this.list[i].className === "disabled")
  126. this.list[i].className = "enabled";
  127. else
  128. this.list[i].className = "disabled";
  129. }
  130. }
  131. };
  132. var mycroft = {
  133. plugins: null,
  134. addLinks: function(p) {
  135. if (p) {
  136. this.plugins = document.evaluate('//a[@href="/jsreq.html"]',
  137. p, null, XPathR###lt.ORDERED_NODE_SNAPSHOT_TYPE, null);
  138. var reviews = document.evaluate('//a[.="[Review]"]',
  139. p, null, XPathR###lt.ORDERED_NODE_SNAPSHOT_TYPE, null);
  140. var addLink = document.createElement("a");
  141. addLink.setAttribute("href", "javascript:void(0)");
  142. addLink.setAttribute("style", "margin-left:5px; color:#000099");
  143. addLink.innerHTML = "[Add to DDG]";
  144. for (var i = 0, tmp; i < reviews.snapshotLength; i++) {
  145. tmp = addLink.cloneNode(true);
  146. tmp.setAttribute("data-ind", i);
  147. tmp.addEventListener("click", this.addNewEngine, false);
  148. reviews.snapshotItem(i).parentNode.insertBefore(tmp, reviews.snapshotItem(i).nextSibling);
  149. }
  150. }
  151. },
  152. addNewEngine: function() {
  153. var current = GM_getValue("engines", ddg_e.
  154. default);
  155. var i = this.getAttribute("data-ind");
  156. var name = mycroft.plugins.snapshotItem(i).innerHTML.split(" (")[0].split(" -")[0];
  157. var newEngine = mycroft.plugins.snapshotItem(i).getAttribute("onClick").split("'")[1];
  158. var newName = prompt("This engine will be added to DDG Extended.\nGive a name or cancel.", name);
  159. if (newName && newName.length > 0) {
  160. this.innerHTML = "[Added]";
  161. this.removeEventListener("click", this.addNewEngine, false);
  162. this.style.color = "#009900";
  163. this.removeAttribute("href");
  164. GM_xmlhttpRequest({
  165. method: "GET",
  166. url: "http://mycroftproject.com/externalos.php/" + newEngine + ".xml",
  167. onload: function(response) {
  168. var responseXML = null;
  169. // Inject responseXML into existing Object (only appropriate for XML content).
  170. if (!response.responseXML) {
  171. responseXML = new DOMParser()
  172. .parseFromString(response.responseText, "text/xml");
  173. }
  174. var engine = responseXML.getElementsByTagName("Url");
  175. if (engine.length > 0 && engine[0].getAttribute("template"))
  176. GM_setValue("engines", current + ";;" + newName + "==" + engine[0].getAttribute("template"));
  177. }
  178. });
  179. }
  180. }
  181. };
  182. // FUNCTIONS
  183. function onHashChange() {
  184. if (window.location.hash.indexOf("{searchTerms}") !== -1)
  185. window.location.hash = "";
  186. window.addEventListener("hashchange", function() {
  187. if (window.location.hash.indexOf("{searchTerms}") !== -1)
  188. window.location.href = window.location.hash.substr(1).replace("{searchTerms}", escape(document.getElementById("search_form_input").value));
  189. }, false);
  190. }
  191. function onColorChange(h, c) {
  192. if (c !== null) {
  193. c.addEventListener("change", function() {
  194. for (var i = 0; i < ddg_e.engines.length; i++)
  195. ddg_e.engines[i].style.background = h.style.background;
  196. }, false);
  197. }
  198. }
  199. function restoreEngines() {
  200. if (confirm("Do you want to restore the default search engines?")) {
  201. GM_setValue("engines", ddg_e.
  202. default);
  203. ddg_e.newList();
  204. }
  205. }
  206. function saveActions() {
  207. disableEvents();
  208. ddg_e.set();
  209. }
  210. function cancelActions() {
  211. disableEvents();
  212. ddg_e.newList();
  213. }
  214. function enableDrag() {
  215. options.toggleClass();
  216. for (var i = 0; i < ddg_e.engines.length; i++) {
  217. ddg_e.engines[i].style.cursor = "move";
  218. ddg_e.engines[i].className = "disabled";
  219. ddg_e.engines[i].addEventListener("dragstart", handleDragStart, false);
  220. ddg_e.engines[i].addEventListener("dragover", handleDragOver, false);
  221. ddg_e.engines[i].addEventListener("drop", handleDrop, false);
  222. }
  223. }
  224. function enableRemove() {
  225. options.toggleClass();
  226. for (var i = 0; i < ddg_e.engines.length; i++) {
  227. ddg_e.engines[i].style.cursor = "crosshair";
  228. ddg_e.engines[i].className = "disabled";
  229. ddg_e.engines[i].addEventListener("click", remove, false);
  230. }
  231. }
  232. function enableRename() {
  233. options.toggleClass();
  234. for (var i = 0; i < ddg_e.engines.length; i++) {
  235. ddg_e.engines[i].style.cursor = "text";
  236. ddg_e.engines[i].className = "disabled";
  237. ddg_e.engines[i].addEventListener("click", rename, false);
  238. }
  239. }
  240. function disableEvents() {
  241. options.toggleClass();
  242. for (var i = 0; i < ddg_e.engines.length; i++) {
  243. ddg_e.engines[i].style.cursor = "auto";
  244. ddg_e.engines[i].removeAttribute("class");
  245. ddg_e.engines[i].removeEventListener("dragstart", handleDragStart, false);
  246. ddg_e.engines[i].removeEventListener("dragover", handleDragOver, false);
  247. ddg_e.engines[i].removeEventListener("drop", handleDrop, false);
  248. ddg_e.engines[i].removeEventListener("click", remove, false);
  249. ddg_e.engines[i].removeEventListener("click", rename, false);
  250. }
  251. }
  252. function remove() {
  253. this.parentNode.removeChild(this);
  254. }
  255. function rename() {
  256. var n = prompt("Rename search engine", this.textContent);
  257. if (n && n.length > 0 && n.length < 20)
  258. this.firstChild.innerHTML = n;
  259. }
  260. function newEngine() {
  261. var n = prompt("Search and add new search engines from MyCroft.\nYou can search by name and url.", "");
  262. if (n && n.length > 0)
  263. window.location.href = "http://mycroftproject.com/search-engines.html?name=" + n;
  264. }
  265. /* Pure javascript and html5 drag-and-drop for an ordered list
  266. * source: https://gist.github.com/robophilosopher/7520460
  267. */
  268. var dragSrcEl,drop_index,numLis,count,token1,token2;
  269. function handleDragStart(e) {
  270. dragSrcEl = this;
  271. e.dataTransfer.effectAllowed = "move";
  272. e.dataTransfer.setData("text/html", this.innerHTML);
  273. }
  274. function handleDragOver(e) {
  275. if (e.preventDefault) e.preventDefault();
  276. return false;
  277. }
  278. function handleDrop(e) {
  279. if (e.stopPropagation) e.stopPropagation();
  280. if (e.preventDefault) e.preventDefault();
  281. drop_index = this.value;
  282. numLis = parseInt(drop_index - dragSrcEl.value);
  283. count = Math.abs(numLis);
  284. if (dragSrcEl != this) {
  285. // swap data when premises are adjacent
  286. if (Math.abs(numLis) == 1) {
  287. dragSrcEl.innerHTML = this.innerHTML;
  288. this.innerHTML = e.dataTransfer.getData("text/html");
  289. }
  290. // propagate data from non-adjacent drops
  291. if (Math.abs(numLis) > 1) {
  292. token1 = this.innerHTML;
  293. this.innerHTML = e.dataTransfer.getData("text/html");
  294. // bring premise up list
  295. if (numLis < -1) {
  296. for (var i = drop_index, counter = 0; counter < count; counter++, i++) {
  297. token2 = ddg_e.engines[i].innerHTML;
  298. ddg_e.engines[i].innerHTML = token1;
  299. token1 = token2;
  300. }
  301. }
  302. // bring premise down list
  303. if (numLis > 1) {
  304. for (var i = drop_index - 1, counter = 0; counter < count; counter++, i--) {
  305. token2 = ddg_e.engines[i - 1].innerHTML;
  306. ddg_e.engines[i - 1].innerHTML = token1;
  307. token1 = token2;
  308. }
  309. }
  310. }
  311. }
  312. return false;
  313. }
  314. // START
  315. if (window.location.href.indexOf("http://mycroftproject.com/") !== -1) {
  316. mycroft.addLinks(document.getElementById("plugins"));
  317. } else {
  318. var header = document.getElementById("header");
  319. if (header) {
  320. ddg_e.append(header);
  321. options.create(document.getElementById("header_button_menu"));
  322. onColorChange(header, document.getElementById("kj"));
  323. onHashChange();
  324. }
  325. }