Tatoeba Visual Linker

Put a sentence into your "shopping cart" to easily link it to another sentence later.

  1. // ==UserScript==
  2. // @name Tatoeba Visual Linker
  3. // @namespace http://userscripts.org/users/61020
  4. // @description Put a sentence into your "shopping cart" to easily link it to another sentence later.
  5. // @include http://*.tatoeba.org/*
  6. // @include https://*.tatoeba.org/*
  7. // @match http://*.tatoeba.org/*
  8. // @match https://*.tatoeba.org/*
  9. // @grant GM_getValue
  10. // @grant GM_setValue
  11. // @require http://code.jquery.com/jquery-latest.min.js
  12. // @version 0.0.1.20150428074750
  13. // ==/UserScript==
  14.  
  15. // The Icon graphics included in this script can be found for free at http://www.famfamfam.com/lab/icons/silk/
  16. console.log('initializing');
  17. textfield_key = '';
  18. not_translation = '';
  19. server_error = '';
  20. cart_delete = '';
  21. server_connect = '';
  22. cart_add = '';
  23. cart_remove = '';
  24. cart_error = '';
  25. textfield_add = '';
  26. not_translation_color = '#F15F74';
  27. not_translation_svg = '';
  28. default_shopping = {
  29. };
  30. default_shopping = JSON.stringify(default_shopping);
  31. //BEGIN USER STATS
  32. shopping = GM_getValue('shopping');
  33. shopping = shopping || default_shopping;
  34. shopping = JSON.parse(shopping);
  35. console.log('shopping: ' + JSON.stringify(shopping));
  36. automatically_numeric = GM_getValue('automatically_numeric');
  37. automatically_numeric = automatically_numeric || false;
  38. setup = false;
  39. if (window.location.href.split('/') [4] == 'user' && window.location.href.split('/') [5] == 'profile' && window.location.href.split('/') [6] == $('.menuSection').attr('href').split('/') [4]) {
  40. setup = true;
  41. if ($('.userscriptSettings').is('*')) {
  42. settings = $('.userscriptSettings');
  43. }
  44. else {
  45. settings = $('<div class="module profileSummary userscriptSettings"><h2>userscripts</h2></div>');
  46. $('.profileSummary').after(settings);
  47. }
  48. settings.append('<h3>Visual Linker</h3>');
  49. contentdiv = $('<form id="visuallinker"></form>');
  50. settings.append(contentdiv);
  51. contentdiv.append('<table>');
  52. contentdiv.append('<tr><td><label for="delete" class="field">delete list</label></td><td><input type="button" id="shopping" value="delete list" ' + (shopping == default_shopping ? 'diabled="disabled"' : '') + '"></td></tr>');
  53. contentdiv.append('</table>');
  54. $('#shopping').click(function () {
  55. if (confirm('Really delete the whole shopping cart?')) {
  56. shopping = JSON.parse(default_shopping);
  57. GM_setValue('shopping', JSON.stringify(shopping));
  58. console.log('shopping: ' + JSON.stringify(shopping));
  59. }
  60. });
  61. }
  62. else {
  63. interface_lang = $('#languageSelection option[selected="selected"]').val();
  64. cart_button_add = $('<a class="audioButton cartButton notincart" style="background-image:url(' + cart_add + '); background-repeat: no-repeat;background-position: center center; float: right; margin: 0.5em 0 0 0;" title="Add this sentence to the \'shopping cart\'."></a>');
  65. cart_button_remove = $('<a class="audioButton cartButton incart" style="background-image:url(' + cart_delete + '); background-repeat: no-repeat;background-position: center center; float: right; margin: 0.5em 0 0 0;" title="Remove this sentence from the \'shopping cart\'."></a>');
  66. empty_cart_button = $('<li class="option tocart"><a title="Click to remove all items from the whole shopping cart."><img width="16" height="16" src="' + cart_remove + '"></a></li>');
  67. empty_cart_button.click(function (event) {
  68. event.preventDefault();
  69. if (confirm('Really delete the whole shopping cart?')) {
  70. shopping = JSON.parse(default_shopping);
  71. GM_setValue('shopping', JSON.stringify(shopping));
  72. console.log('shopping: ' + JSON.stringify(shopping));
  73. $('.sentences_set').each(function () {
  74. showcart($(this));
  75. });
  76. }
  77. });
  78. $('.sentences_set ul.menu .option.addToList').before(empty_cart_button);
  79. numeric = $('<li class="option numeric"><a title="Click to toggle sentence number input field. (Double-click to always show the input field.)"><img width="16" height="16" src="' + (automatically_numeric ? textfield_key : textfield_add) + '"></a></li>');
  80. numeric.click(function (event) {
  81. if (!$(this).parentsUntil('.sentences_set').parent().find('.cart #cart_0').is('*')) {
  82. add_numeric = $('<div class="sentence indirectTranslation" id="cart_0"><a class="translationIcon direct" style="background:none !important;" title="Not yet linked to main sentence (click to go to this sentence)"><div style="background:none !important;"></div></a><!-- a style="background-image:url(' + cart_delete + '); background-repeat: no-repeat;background-position: center center;" class="audioButton cartButton" alt="0" title="Remove this sentence from the \'shopping cart\'."></a--><a class="link button" title="Fetch sentence from server by number. Type the sentence\'s number into the textfield and hit the [enter] key."><img src="' + server_connect + '"></a><!--img width="30" height="20" src="http://flags.tatoeba.org/img/flags/unknown.png" class="languageFlag" alt="unknown" title="The language of the sentence cannot be known yet." --><a class="sentenceContent text"><input type="number" min="1" size="10" title="Fetch sentence from server by number. Type the sentence\'s number into the textfield and hit the [enter] key."></a></div>');
  83. $(add_numeric).find('input').keypress(function (event) {
  84. if (event.which == 13) {
  85. event.preventDefault();
  86. id = $(this).val();
  87. console.log(id);
  88. $(add_numeric).find('.link.button img').attr('src', 'http://flags.tatoeba.org/img/loading-small.gif');
  89. if (typeof (shopping[id]) == 'undefined') {
  90. imtheget = $.get('http://tatoeba.org/sentences/show/' + id, function (data) {
  91. if ($(data).find('.sentences_set').is('*')) {
  92. console.log(data);
  93. lang = $(data).find('.sentences_set .mainSentence .languageFlag').attr('src');
  94. cont = $(data).find('.sentences_set .mainSentence .sentenceContent').removeAttr('href').html();
  95. audio = $(data).find('.sentences_set .mainSentence .audioButton').attr('href');
  96. add_to_cart(id, {
  97. lang: lang,
  98. cont: cont,
  99. audio: audio
  100. });
  101. }
  102. else {
  103. $(add_numeric).find('.link.button img').attr('src', server_error);
  104. }
  105. }
  106. ).error(function () {
  107. $(add_numeric).find('.link.button img').attr('src', server_error);
  108. });
  109. }
  110. else {
  111. $(add_numeric).find('.link.button img').attr('src', cart_error);
  112. }
  113. }
  114. });
  115. $(add_numeric).find('.link.button img').click(function () {
  116. id = $(add_numeric).find('input').val();
  117. console.log(id);
  118. $(add_numeric).find('.link.button img').attr('src', 'http://flags.tatoeba.org/img/loading-small.gif');
  119. if (typeof (shopping[id]) == 'undefined') {
  120. imtheget = $.get('http://tatoeba.org/sentences/show/' + id, function (data) {
  121. if ($(data).find('.sentences_set').is('*')) {
  122. console.log(data);
  123. lang = $(data).find('.sentences_set .mainSentence .languageFlag').attr('src');
  124. cont = $(data).find('.sentences_set .mainSentence .sentenceContent').removeAttr('href').html();
  125. audio = $(data).find('.sentences_set .mainSentence .audioButton').attr('href');
  126. add_to_cart(id, {
  127. lang: lang,
  128. cont: cont,
  129. audio: audio
  130. });
  131. }
  132. else {
  133. $(add_numeric).find('.link.button img').attr('src', server_error);
  134. }
  135. }
  136. ).error(function () {
  137. $(add_numeric).find('.link.button img').attr('src', server_error);
  138. });
  139. }
  140. else {
  141. $(add_numeric).find('.link.button img').attr('src', cart_error);
  142. }
  143. });
  144. $(this).parentsUntil('.sentences_set').parent().find('.cart').append(add_numeric);
  145. }
  146. else {
  147. $(this).parentsUntil('.sentences_set').parent().find('#cart_0').remove();
  148. }
  149. event.preventDefault();
  150. });
  151. numeric.dblclick(function (event) {
  152. event.preventDefault();
  153. automatically_numeric = !automatically_numeric;
  154. GM_setValue('automatically_numeric', automatically_numeric);
  155. console.log('automatically_numeric: ' + automatically_numeric);
  156. $(numeric).find('img').attr('src', (automatically_numeric ? textfield_key : textfield_add));
  157. $(this).parentsUntil('.sentences_set').parent().find('#cart_0').remove();
  158. $(numeric).click();
  159. });
  160. $('.sentences_set ul.menu .option.addToList').before(numeric);
  161. function showcart(sentences_set) {
  162. if ($(sentences_set).find('.cart').is('*')) {
  163. cart = $(sentences_set).find('.cart');
  164. $(cart).empty();
  165. }
  166. else {
  167. cart = $('<div class="translations cart"></div>').css({
  168. 'border-top': '1px dashed #CCCCCC',
  169. 'margin-top':'10px',
  170. });
  171. $(sentences_set).find('.translations').after(cart);
  172. }
  173. chief_sentence = $(sentences_set).attr('id').split('_').reverse() [0];
  174. directs = $(sentences_set).find('.directTranslation').map(function () {
  175. return $(this).attr('id').split('_') [1];
  176. });
  177. indirects = $(sentences_set).find('.indirectTranslation').map(function () {
  178. return $(this).attr('id').split('_') [1];
  179. });
  180. $.each(shopping, function (id, value) {
  181. lang = shopping[id]['lang'];
  182. cont = shopping[id]['cont'];
  183. audio = shopping[id]['audio'];
  184. isself = (id == chief_sentence);
  185. isdirect = ($.inArray(id, directs) >= 0 ? true : false);
  186. isindirect = ($.inArray(id, indirects) >= 0 ? true : false);
  187. sentence_in_basket = $('<div id="cart_' + id + '" class="sentence' + (isself ? ' mainSentence' : (isdirect ? ' directTranslation' : ' indirectTranslation')) + '"></div>');
  188. sentence_in_basket.data({
  189. id: id,
  190. lang: lang,
  191. cont: cont,
  192. audio: audio
  193. });
  194. if (!isself) {
  195. sentence_in_basket.append('<a class="translationIcon '+(isdirect ? 'direct' : 'indirect')+'" href="/sentences/show/' + id + '" title="' + (isdirect ? 'Already' : (isindirect ? 'Indirectly' : 'Not yet')) + ' linked to main sentence (click to go to this sentence)"><div '+(!isdirect && !isindirect ? 'style="background:none !important;"' : '')+'></div></a>');
  196. }
  197. else{
  198. sentence_in_basket.append('<a href="/sentences/show/' + id + '" class="infoIcon"><div></div></a>');
  199. }
  200. sentence_in_basket.append(cart_button_remove.clone());
  201. sentence_in_basket.append('<img width="30" height="20" alt="' + lang + '" class="languageFlag" src="' + lang + '">');
  202. if (!isself) {
  203. if (!isdirect) {
  204. sentence_in_basket.append($('<a class="add link button" href="/' + interface_lang + '/links/add/' + chief_sentence + '/' + id + '" title="Link this sentence to the main sentence."><img width="16" height="16" src="http://flags.tatoeba.org/img/link.svg?1421599964"></a>').data('sentenceId', chief_sentence).data('translationId', id));
  205. }
  206. else {
  207. sentence_in_basket.append($('<a class="delete link button" href="/' + interface_lang + '/links/delete/' + chief_sentence + '/' + id + '" title="Unlink this sentence from the main sentence."><img width="16" height="16" src="http://flags.tatoeba.org/img/unlink.svg?1421599964"></a>').data('sentenceId', chief_sentence).data('translationId', id));
  208. }
  209. }
  210. else{
  211. }
  212. if (!$(sentences_set).parent().is('.sentenceInList')) {
  213. audio = audio | false;
  214. audiotrue = (audio ? audio.split('.').reverse() [0] == 'mp3' : false);
  215. audioURL = audio+'';
  216. sentence_in_basket.append('<a onclick="return false;" class="audioButton ' + (audiotrue ? 'audioAvailable' : 'audioUnavailable') + '" href="' + audioURL + '"></a>').click(function () {
  217. // this is copied from sentences.playaudio.js
  218. $('#audioPlayer').html('<object data="' + audioURL + '" type="audio/mpeg" data="' + audioURL + '" width="0" height="0">' +
  219. '<param name="src" value="' + audioURL + '" />' +
  220. '<object ' +
  221. 'type="application/x-shockwave-flash" ' +
  222. 'data="http://static.tatoeba.org/dewplayer-mini.swf?autostart=1&amp;mp3=' + audioURL + '" ' +
  223. 'width="0" ' +
  224. 'height="0" ' +
  225. '>' +
  226. '<param name="movie" value="http://static.tatoeba.org/dewplayer-mini.swf?autostart=1&amp;mp3=' + audioURL + '" />' +
  227. '</object>' +
  228. '</object>'
  229. );
  230. });
  231. }
  232. sentence_in_basket.append($('<a href="/sentences/show/' + id + '" class="sentenceContent"></div>').append($(cont).removeClass('editableSentence')));
  233. cart.append(sentence_in_basket);
  234. });
  235. }
  236. function add_to_cart(id, object) {
  237. // reload the shopping cart so we can use it across tabs
  238. shopping = GM_getValue('shopping');
  239. shopping = shopping || default_shopping;
  240. shopping = JSON.parse(shopping);
  241. shopping[id] = {
  242. lang: lang,
  243. cont: cont,
  244. audio: audio
  245. };
  246. GM_setValue('shopping', JSON.stringify(shopping));
  247. console.log('shopping: ' + JSON.stringify(shopping));
  248. $('.sentences_set').each(function () {
  249. showcart($(this));
  250. if (automatically_numeric) {
  251. $(this).find('.numeric').click();
  252. }
  253. });
  254. }
  255. function remove_from_cart(id) {
  256. // reload the shopping cart so we can use it across tabs
  257. shopping = GM_getValue('shopping');
  258. shopping = shopping || default_shopping;
  259. shopping = JSON.parse(shopping);
  260. console.log(id);
  261. delete shopping[id];
  262. GM_setValue('shopping', JSON.stringify(shopping));
  263. console.log('shopping: ' + JSON.stringify(shopping));
  264. $('.sentences_set').each(function () {
  265. showcart($(this));
  266. if (automatically_numeric) {
  267. $(this).find('.numeric').click();
  268. }
  269. });
  270. }
  271. function refresh_cart() {
  272. // reload the shopping cart so we can use it across tabs
  273. compare_shopping = shopping;
  274. shopping = GM_getValue('shopping');
  275. shopping = shopping || default_shopping;
  276. shopping = JSON.parse(shopping);
  277. if(JSON.stringify(compare_shopping)!=JSON.stringify(shopping)){
  278. console.log('updating shopping cart');
  279. $('.sentences_set').each(function () {
  280. showcart($(this));
  281. if (automatically_numeric) {
  282. $(this).find('.numeric').click();
  283. }
  284. });
  285. } else {
  286. //console.log('no changes to shopping cart');
  287. }
  288. }
  289. window.onfocus = function(){refresh_cart();};
  290. $('.sentences_set').each(function () {
  291. showcart($(this));
  292. if (automatically_numeric) {
  293. $(this).find('.numeric').click();
  294. }
  295. });
  296. $('.sentences_set .translations:not(.cart) .sentence, .sentences_set > .mainSentence').each(function () {
  297. id = $(this).find('.languageFlag').attr('id').split('_').reverse() [0];
  298. lang = $(this).find('.languageFlag').attr('src');
  299. cont = $(this).find('.sentenceContent').removeAttr('href').html();
  300. audio = $(this).find('.audioButton').attr('href');
  301. incart = typeof (shopping[id]) == 'object';
  302. $(this).data({
  303. id: id,
  304. lang: lang,
  305. cont: cont,
  306. audio: audio
  307. });
  308. $(this).addClass('id' + id);
  309. $(this).find('.languageFlag').before((incart ? cart_button_remove.clone() : cart_button_add.clone()));
  310. });
  311. $('.sentences_set').on('click', '.cartButton', function (event) {
  312. data = $(this).parentsUntil('sentence').data();
  313. id = data['id'];
  314. lang = data['lang'];
  315. cont = data['cont'];
  316. audio = data['audio'];
  317. incart = typeof (shopping[id]) == 'object';
  318. if (incart) {
  319. remove_from_cart(id);
  320. $('.sentences_set .sentence.id' + id + ' .cartButton').replaceWith(cart_button_add.clone());
  321. }
  322. else {
  323. add_to_cart(id, {
  324. lang: lang,
  325. cont: cont,
  326. audio: audio
  327. });
  328. $('.sentences_set .sentence.id' + id + ' .cartButton').replaceWith(cart_button_remove.clone());
  329. }
  330. });
  331. //Below is an adaption of original code from Tatoeba, as greasemonkey cannot (to my knowledge) interact with the code from the page itself
  332. /**
  333. * Tatoeba Project, free collaborative creation of multilingual corpuses project
  334. * Copyright (C) 2011 HO Ngoc Phuong Trang <tranglich@gmail.com>
  335. *
  336. * This program is free software: you can redistribute it and/or modify
  337. * it under the terms of the GNU Affero General Public License as published by
  338. * the Free Software Foundation, either version 3 of the License, or
  339. * (at your option) any later version.
  340. *
  341. * This program is distributed in the hope that it will be useful,
  342. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  343. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  344. * GNU Affero General Public License for more details.
  345. *
  346. * You should have received a copy of the GNU Affero General Public License
  347. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  348. */
  349. //http://tatoeba.org/js/links.add_and_delete.js
  350. $('.cart .link').click(function (event) { //modified by Jakob
  351. event.preventDefault(); //modified by Jakob
  352. var sentenceId = $(this).data('sentenceId');
  353. var translationId = $(this).data('translationId');
  354. var rootUrl = 'http://tatoeba.org/';
  355. var image = $(this);
  356. var action = null;
  357. if ($(this).hasClass('add')) {
  358. var action = 'add';
  359. var newAction = 'delete';
  360. var removeClass = 'indirectTranslation';
  361. var addClass = 'directTranslation';
  362. var newType = 'direct';
  363. } else if ($(this).hasClass('delete')) {
  364. var action = 'delete';
  365. var newAction = 'add';
  366. var removeClass = 'directTranslation';
  367. var addClass = 'indirectTranslation';
  368. var newType = 'indirect';
  369. }
  370. if (action != null) {
  371. // Show the loading gif...
  372. $(this).html('<img src=\'/img/loading-small.gif\' alt=\'loading\'>'
  373. );
  374. // Send request...
  375. $.get(rootUrl + interface_lang + '/links/' + action + '/' + sentenceId + '/' + translationId, function (data) {
  376. var elementId = '#translation_' + translationId + '_' + sentenceId;
  377. var cartId = '#cart_' + sentenceId;
  378. // Update the link or unlink image
  379. image.html(data);
  380. image.removeClass(action);
  381. image.addClass(newAction);
  382. // update the class of the sentence and the arrow
  383. $(elementId).removeClass(removeClass);
  384. $(elementId).addClass(addClass);
  385. $(elementId + ' .show img').attr('src', '/img/' + newType + '_translation.png'
  386. );
  387. $(elementId + ' .link').html(data);
  388. $(image).parent().removeClass(removeClass);
  389. $(image).parent().addClass(addClass);
  390. $(image).parent().find(' .show img').attr('src', '/img/' + newType + '_translation.png'
  391. );
  392. $(image).html(data);
  393. }
  394. );
  395. }
  396. });
  397. }

QingJ © 2025

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