searx.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. /**
  2. * searx is free software: you can redistribute it and/or modify
  3. * it under the terms of the GNU Affero General Public License as published by
  4. * the Free Software Foundation, either version 3 of the License, or
  5. * (at your option) any later version.
  6. *
  7. * searx is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU Affero General Public License for more details.
  11. *
  12. * You should have received a copy of the GNU Affero General Public License
  13. * along with searx. If not, see < http://www.gnu.org/licenses/ >.
  14. *
  15. * (C) 2014 by Thomas Pointhuber, <thomas.pointhuber@gmx.at>
  16. */
  17. requirejs.config({
  18. baseUrl: './static/themes/oscar/js',
  19. paths: {
  20. app: '../app'
  21. }
  22. });
  23. ;/**
  24. * searx is free software: you can redistribute it and/or modify
  25. * it under the terms of the GNU Affero General Public License as published by
  26. * the Free Software Foundation, either version 3 of the License, or
  27. * (at your option) any later version.
  28. *
  29. * searx is distributed in the hope that it will be useful,
  30. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  31. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  32. * GNU Affero General Public License for more details.
  33. *
  34. * You should have received a copy of the GNU Affero General Public License
  35. * along with searx. If not, see < http://www.gnu.org/licenses/ >.
  36. *
  37. * (C) 2019 by Alexandre Flament
  38. */
  39. window.searx = (function(d) {
  40. 'use strict';
  41. // add data- properties
  42. var script = d.currentScript || (function() {
  43. var scripts = d.getElementsByTagName('script');
  44. return scripts[scripts.length - 1];
  45. })();
  46. return {
  47. autocompleter: script.getAttribute('data-autocompleter') === 'true',
  48. method: script.getAttribute('data-method')
  49. };
  50. })(document);
  51. ;/**
  52. * searx is free software: you can redistribute it and/or modify
  53. * it under the terms of the GNU Affero General Public License as published by
  54. * the Free Software Foundation, either version 3 of the License, or
  55. * (at your option) any later version.
  56. *
  57. * searx is distributed in the hope that it will be useful,
  58. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  59. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  60. * GNU Affero General Public License for more details.
  61. *
  62. * You should have received a copy of the GNU Affero General Public License
  63. * along with searx. If not, see < http://www.gnu.org/licenses/ >.
  64. *
  65. * (C) 2014 by Thomas Pointhuber, <thomas.pointhuber@gmx.at>
  66. */
  67. if(searx.autocompleter) {
  68. searx.searchResults = new Bloodhound({
  69. datumTokenizer: Bloodhound.tokenizers.obj.whitespace('value'),
  70. queryTokenizer: Bloodhound.tokenizers.whitespace,
  71. remote: './autocompleter?q=%QUERY'
  72. });
  73. searx.searchResults.initialize();
  74. }
  75. $(document).ready(function(){
  76. var original_search_value = '';
  77. if(searx.autocompleter) {
  78. $("#q").on('keydown', function(e) {
  79. if(e.which == 13) {
  80. original_search_value = $('#q').val();
  81. }
  82. });
  83. $('#q').typeahead(null, {
  84. name: 'search-results',
  85. displayKey: function(result) {
  86. return result;
  87. },
  88. source: searx.searchResults.ttAdapter()
  89. });
  90. $('#q').bind('typeahead:selected', function(ev, suggestion) {
  91. if(original_search_value) {
  92. $('#q').val(original_search_value);
  93. }
  94. $("#search_form").submit();
  95. });
  96. }
  97. });
  98. ;/**
  99. * searx is free software: you can redistribute it and/or modify
  100. * it under the terms of the GNU Affero General Public License as published by
  101. * the Free Software Foundation, either version 3 of the License, or
  102. * (at your option) any later version.
  103. *
  104. * searx is distributed in the hope that it will be useful,
  105. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  106. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  107. * GNU Affero General Public License for more details.
  108. *
  109. * You should have received a copy of the GNU Affero General Public License
  110. * along with searx. If not, see < http://www.gnu.org/licenses/ >.
  111. *
  112. * (C) 2014 by Thomas Pointhuber, <thomas.pointhuber@gmx.at>
  113. */
  114. $(document).ready(function(){
  115. /**
  116. * focus element if class="autofocus" and id="q"
  117. */
  118. $('#q.autofocus').focus();
  119. /**
  120. * select full content on click if class="select-all-on-click"
  121. */
  122. $(".select-all-on-click").click(function () {
  123. $(this).select();
  124. });
  125. /**
  126. * change text during btn-collapse click if possible
  127. */
  128. $('.btn-collapse').click(function() {
  129. var btnTextCollapsed = $(this).data('btn-text-collapsed');
  130. var btnTextNotCollapsed = $(this).data('btn-text-not-collapsed');
  131. if(btnTextCollapsed !== '' && btnTextNotCollapsed !== '') {
  132. if($(this).hasClass('collapsed')) {
  133. new_html = $(this).html().replace(btnTextCollapsed, btnTextNotCollapsed);
  134. } else {
  135. new_html = $(this).html().replace(btnTextNotCollapsed, btnTextCollapsed);
  136. }
  137. $(this).html(new_html);
  138. }
  139. });
  140. /**
  141. * change text during btn-toggle click if possible
  142. */
  143. $('.btn-toggle .btn').click(function() {
  144. var btnClass = 'btn-' + $(this).data('btn-class');
  145. var btnLabelDefault = $(this).data('btn-label-default');
  146. var btnLabelToggled = $(this).data('btn-label-toggled');
  147. if(btnLabelToggled !== '') {
  148. if($(this).hasClass('btn-default')) {
  149. new_html = $(this).html().replace(btnLabelDefault, btnLabelToggled);
  150. } else {
  151. new_html = $(this).html().replace(btnLabelToggled, btnLabelDefault);
  152. }
  153. $(this).html(new_html);
  154. }
  155. $(this).toggleClass(btnClass);
  156. $(this).toggleClass('btn-default');
  157. });
  158. /**
  159. * change text during btn-toggle click if possible
  160. */
  161. $('.media-loader').click(function() {
  162. var target = $(this).data('target');
  163. var iframe_load = $(target + ' > iframe');
  164. var srctest = iframe_load.attr('src');
  165. if(srctest === undefined || srctest === false){
  166. iframe_load.attr('src', iframe_load.data('src'));
  167. }
  168. });
  169. /**
  170. * Select or deselect every categories on double clic
  171. */
  172. $(".btn-sm").dblclick(function() {
  173. var btnClass = 'btn-' + $(this).data('btn-class'); // primary
  174. if($(this).hasClass('btn-default')) {
  175. $(".btn-sm > input").attr('checked', 'checked');
  176. $(".btn-sm > input").prop("checked", true);
  177. $(".btn-sm").addClass(btnClass);
  178. $(".btn-sm").addClass('active');
  179. $(".btn-sm").removeClass('btn-default');
  180. } else {
  181. $(".btn-sm > input").attr('checked', '');
  182. $(".btn-sm > input").removeAttr('checked');
  183. $(".btn-sm > input").checked = false;
  184. $(".btn-sm").removeClass(btnClass);
  185. $(".btn-sm").removeClass('active');
  186. $(".btn-sm").addClass('btn-default');
  187. }
  188. });
  189. $(".nav-tabs").click(function(a) {
  190. var tabs = $(a.target).parents("ul");
  191. tabs.children().attr("aria-selected", "false");
  192. $(a.target).parent().attr("aria-selected", "true");
  193. });
  194. });
  195. ;$(document).ready(function() {
  196. // Hide infobox toggle if shrunk size already fits all content.
  197. $('.infobox').each(function() {
  198. var infobox_body = $(this).find('.infobox_body');
  199. var total_height = infobox_body.prop('scrollHeight') + infobox_body.find('img.infobox_part').height();
  200. var max_height = infobox_body.css('max-height').replace('px', '');
  201. if (total_height <= max_height) {
  202. $(this).find('.infobox_toggle').hide();
  203. }
  204. });
  205. });
  206. ;/**
  207. * searx is free software: you can redistribute it and/or modify
  208. * it under the terms of the GNU Affero General Public License as published by
  209. * the Free Software Foundation, either version 3 of the License, or
  210. * (at your option) any later version.
  211. *
  212. * searx is distributed in the hope that it will be useful,
  213. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  214. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  215. * GNU Affero General Public License for more details.
  216. *
  217. * You should have received a copy of the GNU Affero General Public License
  218. * along with searx. If not, see < http://www.gnu.org/licenses/ >.
  219. *
  220. * (C) 2014 by Thomas Pointhuber, <thomas.pointhuber@gmx.at>
  221. */
  222. $(document).ready(function(){
  223. $(".searx_overpass_request").on( "click", function( event ) {
  224. var overpass_url = "https://overpass-api.de/api/interpreter?data=";
  225. var query_start = overpass_url + "[out:json][timeout:25];(";
  226. var query_end = ");out meta;";
  227. var osm_id = $(this).data('osm-id');
  228. var osm_type = $(this).data('osm-type');
  229. var result_table = $(this).data('result-table');
  230. var result_table_loadicon = "#" + $(this).data('result-table-loadicon');
  231. // tags which can be ignored
  232. var osm_ignore_tags = [ "addr:city", "addr:country", "addr:housenumber", "addr:postcode", "addr:street" ];
  233. if(osm_id && osm_type && result_table) {
  234. result_table = "#" + result_table;
  235. var query = null;
  236. switch(osm_type) {
  237. case 'node':
  238. query = query_start + "node(" + osm_id + ");" + query_end;
  239. break;
  240. case 'way':
  241. query = query_start + "way(" + osm_id + ");" + query_end;
  242. break;
  243. case 'relation':
  244. query = query_start + "relation(" + osm_id + ");" + query_end;
  245. break;
  246. default:
  247. break;
  248. }
  249. if(query) {
  250. //alert(query);
  251. var ajaxRequest = $.ajax( query )
  252. .done(function( html) {
  253. if(html && html.elements && html.elements[0]) {
  254. var element = html.elements[0];
  255. var newHtml = $(result_table).html();
  256. for (var row in element.tags) {
  257. if(element.tags.name === null || osm_ignore_tags.indexOf(row) == -1) {
  258. newHtml += "<tr><td>" + row + "</td><td>";
  259. switch(row) {
  260. case "phone":
  261. case "fax":
  262. newHtml += "<a href=\"tel:" + element.tags[row].replace(/ /g,'') + "\">" + element.tags[row] + "</a>";
  263. break;
  264. case "email":
  265. newHtml += "<a href=\"mailto:" + element.tags[row] + "\">" + element.tags[row] + "</a>";
  266. break;
  267. case "website":
  268. case "url":
  269. newHtml += "<a href=\"" + element.tags[row] + "\">" + element.tags[row] + "</a>";
  270. break;
  271. case "wikidata":
  272. newHtml += "<a href=\"https://www.wikidata.org/wiki/" + element.tags[row] + "\">" + element.tags[row] + "</a>";
  273. break;
  274. case "wikipedia":
  275. if(element.tags[row].indexOf(":") != -1) {
  276. newHtml += "<a href=\"https://" + element.tags[row].substring(0,element.tags[row].indexOf(":")) + ".wikipedia.org/wiki/" + element.tags[row].substring(element.tags[row].indexOf(":")+1) + "\">" + element.tags[row] + "</a>";
  277. break;
  278. }
  279. /* jshint ignore:start */
  280. default:
  281. /* jshint ignore:end */
  282. newHtml += element.tags[row];
  283. break;
  284. }
  285. newHtml += "</td></tr>";
  286. }
  287. }
  288. $(result_table).html(newHtml);
  289. $(result_table).removeClass('hidden');
  290. $(result_table_loadicon).addClass('hidden');
  291. }
  292. })
  293. .fail(function() {
  294. $(result_table_loadicon).html($(result_table_loadicon).html() + "<p class=\"text-muted\">"+could_not_load+"</p>");
  295. });
  296. }
  297. }
  298. // this event occour only once per element
  299. $( this ).off( event );
  300. });
  301. $(".searx_init_map").on( "click", function( event ) {
  302. var leaflet_target = $(this).data('leaflet-target');
  303. var map_lon = $(this).data('map-lon');
  304. var map_lat = $(this).data('map-lat');
  305. var map_zoom = $(this).data('map-zoom');
  306. var map_boundingbox = $(this).data('map-boundingbox');
  307. var map_geojson = $(this).data('map-geojson');
  308. require(['leaflet-0.7.3.min'], function(leaflet) {
  309. if(map_boundingbox) {
  310. southWest = L.latLng(map_boundingbox[0], map_boundingbox[2]);
  311. northEast = L.latLng(map_boundingbox[1], map_boundingbox[3]);
  312. map_bounds = L.latLngBounds(southWest, northEast);
  313. }
  314. // TODO hack
  315. // change default imagePath
  316. L.Icon.Default.imagePath = "./static/themes/oscar/img/map";
  317. // init map
  318. var map = L.map(leaflet_target);
  319. // create the tile layer with correct attribution
  320. var osmMapnikUrl='https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png';
  321. var osmMapnikAttrib='Map data © <a href="https://openstreetmap.org">OpenStreetMap</a> contributors';
  322. var osmMapnik = new L.TileLayer(osmMapnikUrl, {minZoom: 1, maxZoom: 19, attribution: osmMapnikAttrib});
  323. var osmWikimediaUrl='https://maps.wikimedia.org/osm-intl/{z}/{x}/{y}.png';
  324. var osmWikimediaAttrib = 'Wikimedia maps beta | Maps data © <a href="https://openstreetmap.org">OpenStreetMap</a> contributors';
  325. var osmWikimedia = new L.TileLayer(osmWikimediaUrl, {minZoom: 1, maxZoom: 19, attribution: osmWikimediaAttrib});
  326. // init map view
  327. if(map_bounds) {
  328. // TODO hack: https://github.com/Leaflet/Leaflet/issues/2021
  329. setTimeout(function () {
  330. map.fitBounds(map_bounds, {
  331. maxZoom:17
  332. });
  333. }, 0);
  334. } else if (map_lon && map_lat) {
  335. if(map_zoom)
  336. map.setView(new L.LatLng(map_lat, map_lon),map_zoom);
  337. else
  338. map.setView(new L.LatLng(map_lat, map_lon),8);
  339. }
  340. map.addLayer(osmMapnik);
  341. var baseLayers = {
  342. "OSM Mapnik": osmMapnik/*,
  343. "OSM Wikimedia": osmWikimedia*/
  344. };
  345. L.control.layers(baseLayers).addTo(map);
  346. if(map_geojson)
  347. L.geoJson(map_geojson).addTo(map);
  348. /*else if(map_bounds)
  349. L.rectangle(map_bounds, {color: "#ff7800", weight: 3, fill:false}).addTo(map);*/
  350. });
  351. // this event occour only once per element
  352. $( this ).off( event );
  353. });
  354. });
  355. ;$(document).ready(function(){
  356. $("#allow-all-engines").click(function() {
  357. $(".onoffswitch-checkbox").each(function() { this.checked = false;});
  358. });
  359. $("#disable-all-engines").click(function() {
  360. $(".onoffswitch-checkbox").each(function() { this.checked = true;});
  361. });
  362. });