Tatoeba Visual Linker

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

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name           Tatoeba Visual Linker
// @namespace      http://userscripts.org/users/61020
// @description    Put a sentence into your "shopping cart" to easily link it to another sentence later.
// @include        http://*.tatoeba.org/*
// @include        https://*.tatoeba.org/*
// @match          http://*.tatoeba.org/*
// @match          https://*.tatoeba.org/*
// @grant          GM_getValue
// @grant          GM_setValue
// @require        http://code.jquery.com/jquery-latest.min.js
// @version 0.0.1.20150428074750
// ==/UserScript==

  // The Icon graphics included in this script can be found for free at http://www.famfamfam.com/lab/icons/silk/
  console.log('initializing');
  textfield_key = '';
  not_translation = '';
  server_error = '';
  cart_delete = '';
  server_connect = '';
  cart_add = '';
  cart_remove = '';
  cart_error = '';
  textfield_add = '';
  not_translation_color = '#F15F74';
  not_translation_svg = '';
  default_shopping = {
  };
  default_shopping = JSON.stringify(default_shopping);
  //BEGIN USER STATS
  shopping = GM_getValue('shopping');
  shopping = shopping || default_shopping;
  shopping = JSON.parse(shopping);
  console.log('shopping: ' + JSON.stringify(shopping));
  automatically_numeric = GM_getValue('automatically_numeric');
  automatically_numeric = automatically_numeric || false;
  setup = false;
  if (window.location.href.split('/') [4] == 'user' && window.location.href.split('/') [5] == 'profile' && window.location.href.split('/') [6] == $('.menuSection').attr('href').split('/') [4]) {
    setup = true;
    if ($('.userscriptSettings').is('*')) {
      settings = $('.userscriptSettings');
    } 
    else {
      settings = $('<div class="module profileSummary userscriptSettings"><h2>userscripts</h2></div>');
      $('.profileSummary').after(settings);
    }
    settings.append('<h3>Visual Linker</h3>');
    contentdiv = $('<form id="visuallinker"></form>');
    settings.append(contentdiv);
    contentdiv.append('<table>');
    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>');
    contentdiv.append('</table>');
    $('#shopping').click(function () {
      if (confirm('Really delete the whole shopping cart?')) {
        shopping = JSON.parse(default_shopping);
        GM_setValue('shopping', JSON.stringify(shopping));
        console.log('shopping: ' + JSON.stringify(shopping));
      }
    });
  } 
  else {
    interface_lang = $('#languageSelection option[selected="selected"]').val();
    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>');
    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>');
    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>');
    empty_cart_button.click(function (event) {
      event.preventDefault();
      if (confirm('Really delete the whole shopping cart?')) {
        shopping = JSON.parse(default_shopping);
        GM_setValue('shopping', JSON.stringify(shopping));
        console.log('shopping: ' + JSON.stringify(shopping));
        $('.sentences_set').each(function () {
          showcart($(this));
        });
      }
    });
    $('.sentences_set ul.menu .option.addToList').before(empty_cart_button);
    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>');
    numeric.click(function (event) {
      if (!$(this).parentsUntil('.sentences_set').parent().find('.cart #cart_0').is('*')) {
        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>');
        $(add_numeric).find('input').keypress(function (event) {
          if (event.which == 13) {
            event.preventDefault();
            id = $(this).val();
            console.log(id);
            $(add_numeric).find('.link.button img').attr('src', 'http://flags.tatoeba.org/img/loading-small.gif');
            if (typeof (shopping[id]) == 'undefined') {
              imtheget = $.get('http://tatoeba.org/sentences/show/' + id, function (data) {
                if ($(data).find('.sentences_set').is('*')) {
                  console.log(data);
                  lang = $(data).find('.sentences_set .mainSentence .languageFlag').attr('src');
                  cont = $(data).find('.sentences_set .mainSentence .sentenceContent').removeAttr('href').html();
                  audio = $(data).find('.sentences_set .mainSentence .audioButton').attr('href');
                  add_to_cart(id, {
                    lang: lang,
                    cont: cont,
                    audio: audio
                  });
                } 
                else {
                  $(add_numeric).find('.link.button img').attr('src', server_error);
                }
              }
              ).error(function () {
                $(add_numeric).find('.link.button img').attr('src', server_error);
              });
            } 
            else {
              $(add_numeric).find('.link.button img').attr('src', cart_error);
            }
          }
        });
        $(add_numeric).find('.link.button img').click(function () {
          id = $(add_numeric).find('input').val();
          console.log(id);
          $(add_numeric).find('.link.button img').attr('src', 'http://flags.tatoeba.org/img/loading-small.gif');
          if (typeof (shopping[id]) == 'undefined') {
            imtheget = $.get('http://tatoeba.org/sentences/show/' + id, function (data) {
              if ($(data).find('.sentences_set').is('*')) {
                console.log(data);
                lang = $(data).find('.sentences_set .mainSentence .languageFlag').attr('src');
                cont = $(data).find('.sentences_set .mainSentence .sentenceContent').removeAttr('href').html();
                audio = $(data).find('.sentences_set .mainSentence .audioButton').attr('href');
                add_to_cart(id, {
                  lang: lang,
                  cont: cont,
                  audio: audio
                });
              } 
              else {
                $(add_numeric).find('.link.button img').attr('src', server_error);
              }
            }
            ).error(function () {
              $(add_numeric).find('.link.button img').attr('src', server_error);
            });
          } 
          else {
            $(add_numeric).find('.link.button img').attr('src', cart_error);
          }
        });
        $(this).parentsUntil('.sentences_set').parent().find('.cart').append(add_numeric);
      } 
      else {
        $(this).parentsUntil('.sentences_set').parent().find('#cart_0').remove();
      }
      event.preventDefault();
    });
    numeric.dblclick(function (event) {
      event.preventDefault();
      automatically_numeric = !automatically_numeric;
      GM_setValue('automatically_numeric', automatically_numeric);
      console.log('automatically_numeric: ' + automatically_numeric);
      $(numeric).find('img').attr('src', (automatically_numeric ? textfield_key : textfield_add));
      $(this).parentsUntil('.sentences_set').parent().find('#cart_0').remove();
      $(numeric).click();
    });
    $('.sentences_set ul.menu .option.addToList').before(numeric);
    function showcart(sentences_set) {
      if ($(sentences_set).find('.cart').is('*')) {
        cart = $(sentences_set).find('.cart');
        $(cart).empty();
      } 
      else {
        cart = $('<div class="translations cart"></div>').css({
          'border-top': '1px dashed #CCCCCC',
          'margin-top':'10px',
        });
        $(sentences_set).find('.translations').after(cart);
      }
      chief_sentence = $(sentences_set).attr('id').split('_').reverse() [0];
      directs = $(sentences_set).find('.directTranslation').map(function () {
        return $(this).attr('id').split('_') [1];
      });
      indirects = $(sentences_set).find('.indirectTranslation').map(function () {
        return $(this).attr('id').split('_') [1];
      });
      $.each(shopping, function (id, value) {
        lang = shopping[id]['lang'];
        cont = shopping[id]['cont'];
        audio = shopping[id]['audio'];
        isself = (id == chief_sentence);
        isdirect = ($.inArray(id, directs) >= 0 ? true : false);
        isindirect = ($.inArray(id, indirects) >= 0 ? true : false);
        sentence_in_basket = $('<div id="cart_' + id + '" class="sentence' + (isself ? ' mainSentence' : (isdirect ? ' directTranslation' : ' indirectTranslation')) + '"></div>');
        sentence_in_basket.data({
          id: id,
          lang: lang,
          cont: cont,
          audio: audio
        });
        
        if (!isself) {
          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>');
        }
        else{
          sentence_in_basket.append('<a href="/sentences/show/' + id + '" class="infoIcon"><div></div></a>');
        }
        sentence_in_basket.append(cart_button_remove.clone());
        sentence_in_basket.append('<img width="30" height="20" alt="' + lang + '" class="languageFlag" src="' + lang + '">');
        if (!isself) {
          if (!isdirect) {
            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));
          } 
          else {
            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));
          }
        }
        else{
          
        }
        if (!$(sentences_set).parent().is('.sentenceInList')) {
          audio = audio | false;
          audiotrue = (audio ? audio.split('.').reverse() [0] == 'mp3' : false);
          audioURL = audio+'';
          sentence_in_basket.append('<a onclick="return false;" class="audioButton ' + (audiotrue ? 'audioAvailable' : 'audioUnavailable') + '" href="' + audioURL + '"></a>').click(function () {
            // this is copied from sentences.playaudio.js
            $('#audioPlayer').html('<object data="' + audioURL + '" type="audio/mpeg" data="' + audioURL + '" width="0" height="0">' +
            '<param name="src" value="' + audioURL + '" />' +
            '<object ' +
            'type="application/x-shockwave-flash" ' +
            'data="http://static.tatoeba.org/dewplayer-mini.swf?autostart=1&amp;mp3=' + audioURL + '" ' +
            'width="0" ' +
            'height="0" ' +
            '>' +
            '<param name="movie" value="http://static.tatoeba.org/dewplayer-mini.swf?autostart=1&amp;mp3=' + audioURL + '" />' +
            '</object>' +
            '</object>'
            );
          });
        }
        sentence_in_basket.append($('<a href="/sentences/show/' + id + '" class="sentenceContent"></div>').append($(cont).removeClass('editableSentence')));
        cart.append(sentence_in_basket);
      });
    }
    function add_to_cart(id, object) {
      // reload the shopping cart so we can use it across tabs
      shopping = GM_getValue('shopping');
      shopping = shopping || default_shopping;
      shopping = JSON.parse(shopping);
      shopping[id] = {
        lang: lang,
        cont: cont,
        audio: audio
      };
      GM_setValue('shopping', JSON.stringify(shopping));
      console.log('shopping: ' + JSON.stringify(shopping));
      $('.sentences_set').each(function () {
        showcart($(this));
        if (automatically_numeric) {
          $(this).find('.numeric').click();
        }
      });
    }
    function remove_from_cart(id) {
      // reload the shopping cart so we can use it across tabs
      shopping = GM_getValue('shopping');
      shopping = shopping || default_shopping;
      shopping = JSON.parse(shopping);
      console.log(id);
      delete shopping[id];
      GM_setValue('shopping', JSON.stringify(shopping));
      console.log('shopping: ' + JSON.stringify(shopping));
      $('.sentences_set').each(function () {
        showcart($(this));
        if (automatically_numeric) {
          $(this).find('.numeric').click();
        }
      });
    }
    function refresh_cart() {
      // reload the shopping cart so we can use it across tabs
      compare_shopping = shopping;
      shopping = GM_getValue('shopping');
      shopping = shopping || default_shopping;
      shopping = JSON.parse(shopping);
      if(JSON.stringify(compare_shopping)!=JSON.stringify(shopping)){
        console.log('updating shopping cart');
        $('.sentences_set').each(function () {
          showcart($(this));
          if (automatically_numeric) {
            $(this).find('.numeric').click();
          }
        });  
      } else {
        //console.log('no changes to shopping cart');
      }
    }
    window.onfocus = function(){refresh_cart();};
    $('.sentences_set').each(function () {
      showcart($(this));
      if (automatically_numeric) {
        $(this).find('.numeric').click();
      }
    });
    
    $('.sentences_set .translations:not(.cart) .sentence, .sentences_set > .mainSentence').each(function () {
      id = $(this).find('.languageFlag').attr('id').split('_').reverse() [0];
      lang = $(this).find('.languageFlag').attr('src');
      cont = $(this).find('.sentenceContent').removeAttr('href').html();
      audio = $(this).find('.audioButton').attr('href');
      incart = typeof (shopping[id]) == 'object';
      $(this).data({
        id: id,
        lang: lang,
        cont: cont,
        audio: audio
      });
      $(this).addClass('id' + id);
      $(this).find('.languageFlag').before((incart ? cart_button_remove.clone()  : cart_button_add.clone()));
    });
    $('.sentences_set').on('click', '.cartButton', function (event) {
      data = $(this).parentsUntil('sentence').data();
      id = data['id'];
      lang = data['lang'];
      cont = data['cont'];
      audio = data['audio'];
      incart = typeof (shopping[id]) == 'object';
      if (incart) {
        remove_from_cart(id);
        $('.sentences_set .sentence.id' + id + ' .cartButton').replaceWith(cart_button_add.clone());
      } 
      else {
        add_to_cart(id, {
          lang: lang,
          cont: cont,
          audio: audio
        });
        $('.sentences_set .sentence.id' + id + ' .cartButton').replaceWith(cart_button_remove.clone());
      }
    });
    //Below is an adaption of original code from Tatoeba, as greasemonkey cannot (to my knowledge) interact with the code from the page itself
    /**
		 * Tatoeba Project, free collaborative creation of multilingual corpuses project
		 * Copyright (C) 2011  HO Ngoc Phuong Trang <[email protected]>
		 *
		 * This program is free software: you can redistribute it and/or modify
		 * it under the terms of the GNU Affero General Public License as published by
		 * the Free Software Foundation, either version 3 of the License, or
		 * (at your option) any later version.
		 *
		 * This program is distributed in the hope that it will be useful,
		 * but WITHOUT ANY WARRANTY; without even the implied warranty of
		 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
		 * GNU Affero General Public License for more details.
		 *
		 * You should have received a copy of the GNU Affero General Public License
		 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
		 */
    //http://tatoeba.org/js/links.add_and_delete.js
    $('.cart .link').click(function (event) { //modified by Jakob
      event.preventDefault(); //modified by Jakob
      var sentenceId = $(this).data('sentenceId');
      var translationId = $(this).data('translationId');
      var rootUrl = 'http://tatoeba.org/';
      var image = $(this);
      var action = null;
      if ($(this).hasClass('add')) {
        var action = 'add';
        var newAction = 'delete';
        var removeClass = 'indirectTranslation';
        var addClass = 'directTranslation';
        var newType = 'direct';
      } else if ($(this).hasClass('delete')) {
        var action = 'delete';
        var newAction = 'add';
        var removeClass = 'directTranslation';
        var addClass = 'indirectTranslation';
        var newType = 'indirect';
      }
      if (action != null) {
        // Show the loading gif...
        $(this).html('<img src=\'/img/loading-small.gif\' alt=\'loading\'>'
        );
        // Send request...
        $.get(rootUrl + interface_lang + '/links/' + action + '/' + sentenceId + '/' + translationId, function (data) {
          var elementId = '#translation_' + translationId + '_' + sentenceId;
          var cartId = '#cart_' + sentenceId;
          // Update the link or unlink image
          image.html(data);
          image.removeClass(action);
          image.addClass(newAction);
          // update the class of the sentence and the arrow
          $(elementId).removeClass(removeClass);
          $(elementId).addClass(addClass);
          $(elementId + ' .show img').attr('src', '/img/' + newType + '_translation.png'
          );
          $(elementId + ' .link').html(data);
          $(image).parent().removeClass(removeClass);
          $(image).parent().addClass(addClass);
          $(image).parent().find(' .show img').attr('src', '/img/' + newType + '_translation.png'
          );
          $(image).html(data);
        }
        );
      }
    });
  }