Tatoeba Visual Linker

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

// ==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 <tranglich@gmail.com>
		 *
		 * 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);
        }
        );
      }
    });
  }