WykopBingo!

Graj znaleziskami w Bingo na Wykopie

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

You will need to install an extension such as Tampermonkey to install this script.

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name        WykopBingo!
// @namespace   Wykop scripts
// @description Graj znaleziskami w Bingo na Wykopie
// @include     *://www.wykop.pl/link/*
// @version     0.86a
// @grant       GM_xmlhttpRequest
// @author      Orlin
// ==/UserScript==

// linijka poniżej na potrzeby skryptozakładki, pamiętaj tylko o usunięciu pozostałych komentarzy zaczynających się od //
//javascript:function GM_xmlhttpRequest(a){ alert('Wersja skryptozakładkowa na razie nie potrafi łączyć się z serwerem, więc nie prześle poniższego żądania:\n'+JSON.stringify(a));}; 
/**
 * Wartości pól planszy do gry w WykopBingo!
 */
var boardSquares = [
  'IDIOTA NA DRODZE',
  'IMIGRANCI',
  'JAK XYZ TRAKTUJE KLIENTÓW',
  'PLANETA PODOBNA DO ZIEMI',
  'SKRADZIONO SAMOCHÓD / ROWER',
  'MARIUSZ MAX KOLONKO',
  'AFERA NA WYKOPIE',
  '"POLSKIE OBOZY" W ZACHODNICH MEDIACH',
  'ISLAM',
  'ARTYKUŁ O TESLI',
  'PRZEŁOMOWY WYNALAZEK POLAKÓW',
  'KOTY',
  '[W]',
  'ELON MUSK',
  'KORWIN',
  'A.M.A.',
  'VIKTOR ORBAN',
  'BATERIE NOWEJ GENERACJI',
  'GRAFEN',
  'KWOTA WOLNA OD PODATKU',
  'LEK NA RAKA',
  'KONTROWER-\nSYJNY POMYSŁ RZĄDU/UE',
  'NOWE ŹRÓDŁO ENERGII',
  'OKRADŁ GO ZUS / KOMORNIK',
  'AMERYKAŃSCY NAUKOWCY ODKRYLI'
];

var zalogowany = false;
/*===========================================================================================================================*/

/**
 * Dodaje przycisk gry WykopBingo! do przycisków w formularzu komentarza
 */
function addBingoButton()
{
    var bb = document.createElement('a');
    bb.className = 'button bingoButton';
    /*    bb.innerHTML='▦';*/
    bb.innerHTML = '⌧';
    bb.title = 'WykopBingo!';
    bb.onclick = function () {
      playBingo(this);
    };
/*    var bbb = JSON.parse(JSON.stringify(bb));*/
    var bbb = bb.cloneNode(true);
    bbb.onclick = function () {
      playBingo(this);
    };

  var survButt = document.getElementsByClassName('button openAddSurveyOverlay');
  if(survButt.length > 0)
  {
    survButt[0].parentNode.insertBefore(bb, survButt[0].nextSibling);
    zalogowany = true;
  }
  else
  {
    zalogowany = false;
  }
  var navbar = document.getElementsByClassName('nav fix-b-border');
  if(navbar.length > 0)
  {
    navbar[0].insertBefore(bbb, navbar[0].firstChild);
  }
  else
  {
    alert('Skrypt(ozakładka) działa poprawnie tylko na stronach ze znaleziskami @ wykop.pl.');
  }
}
/*===========================================================================================================================*/

/**
 * Tworzy/pokazuje okienko z planszą gry
 * @param {DOM element reference} buttonNodeRef Żeby po kliknięciu 'Publikuj', komentarz trafił do pola tekstowego, przy którym kliknięto wcześniej przycisk WykopBingo!
 */
function playBingo(buttonNodeRef)
{
  var bingoButtonTst = document.getElementById('bingoTableWrap');
  var overlayDiv;
  
  if(bingoButtonTst !== null)
  {
    bingoButtonTst.style.display = 'block';
    document.getElementById('bingoPublishButton').onclick = function(){
                   generateComment(buttonNodeRef);
                   document.getElementById('bingoTableWrap').style.display = 'none';
                   var ovrls = document.getElementsByClassName('overlay');
                   for(i=0; i<ovrls.length; ++i)
                   {
                     ovrls[i].parentNode.removeChild(ovrls[i]);
                   }
    };
    overlayDiv = document.createElement('div');
    overlayDiv.className = 'overlay';
    overlayDiv.style.display = 'block';
    document.body.insertBefore(overlayDiv, document.body.firstChild);
    resetGame(true);
    fillBoard();
    return;
  }
  var closeBGC, rowBGC, wrapperBGC;
  if(document.body.className.indexOf('night') >= 0)
  {
    /* nocny */
    closeBGC = '000';
    wrapperBGC = '333';
    rowBGC = 'rgba(81,81,81,0.5)';
    
  }
  else
  {
    /* dzienny */
    closeBGC = 'ccc';
    wrapperBGC = '777';
    rowBGC = 'rgba(208,208,208,0.5)';
  }
  
  addCss('.bingoTab\n{\n\tborder-collapse:collapse;\n\tborder-spacing:0;\n}\n\n#bingoTableWrap\n{\n\tdisplay: table;\n\tz-index: 999999;\n\tleft: 25%;\n\ttop: 30px;\n\tposition: fixed;\n\tbackground-color: #'+wrapperBGC+';\n}\n\n#bingoTableWrap > a\n{\n\tcursor: pointer;\n\tposition: absolute;\n\tright: 0px;\n\ttop: -20px;\n\tbackground-color: #'+closeBGC+';\n}\n\n.bingoTab\n{\n\tbackground-size: contain;\n\tbackground-repeat: no-repeat;\n\tbackground-position: center;\n}\n\n.bingoTab tr, .bingoTab tr:hover, .bingoTab tr:nth-child(even)\n{\n\tbackground-color: '+rowBGC+';\n}\n\n.bingoTab td:hover\n{\n\tbackground-color: #888888;\n}\n\n.bingoTab td\n{\n\tposition: relative;\n\tfont-family:Arial, sans-serif;\n\tfont-weight: bold;\n\tfont-size:10px;\n\tpadding:1px;\n\tborder-style:solid;\n\tborder-width:1px;\n\toverflow:hidden;\n\tword-break:normal;\n\ttext-align:center;\n\twidth:80px;\n\theight:80px;\n\tmax-width:80px;\n\tmax-height:80px;\n\tmin-width:80px;\n\tmin-height:80px;\n\tcursor:crosshair;\n\tvertical-align: middle;\n}\n\n.bingoLine\n{\n\tbackground-color: rgba(0,255,0,0.5);\n}\n\n.bingoCrossed\n{\n\tposition: absolute;\n\tdisplay:none; \n\ttop: 40%;\n\twidth: 100%;\n\tfont-color: red;\n\tcolor: red;\n\tfont-weight: bold;\n\tfont-size:68px;\n\n}\n\n.bingoNotCrossed\n{\n\tdisplay:none; \n}\n\n.bingoProofLinks\n{\n    position: absolute;\n\tright: 0;\n    bottom: 0;\n\ttext-align:right;\n\tz-index: 2;\n}\n\n.bingoProofLinksOff\n{\n\tdisplay:none;\n}\n\n#bingoCtxMenu\n{\n\tposition: absolute;\n\tdisplay:none;\n\tbackground-color: lightgrey;\n\tborder: 1px dashed blue;\n\tz-index: 5;\n}\n\n#bingoCtxMenu\n{\n\tpadding: 5px 10px;\n\tposition: fixed;\n\tdisplay:none;\n\tbackground-color: lightgrey;\n\tborder: 1px dashed blue;\n\tz-index: 5;\n}\n\n#bingoCtxMenu a\n{\n\tdisplay:block;\n\tcursor:pointer;\n\tcolor: blue;\n}\n\n#bingoCtxMenu a:hover\n{\n\ttext-decoration: underline;\n}\n\n#bingoCtxMenu a:before\n{\n\tcontent:"• ";\n}\n\n.bingoButtons\n{\n\tcolor: #000;\n\tmargin: 5px\n}\n\n.bingoButtons :focus\n{\n\tcolor: #000;\n\tmargin: 5px\n}\n\n#bingoLinksEditor\n{\n\tposition: fixed;\n\tleft: 10%;\n\ttop:50px;\n\tcolor: #000;\n\tbackground-color: #aaa;\n\tz-index:5;\n\twidth: 75%;\n\tmin-width: 640px;\n}\n\n#bingoLinksEditor > a\n{\n\tcursor: pointer;\n\tposition: absolute;\n\tright: 0px;\n\ttop: -20px;\n\tbackground-color: #'+closeBGC+';\n}\n\n#bingoLinksEditor table\n{\n\tbackground-color: #aaa;\n}\n\n#bingoLinksEditor thead\n{\n\tbackground-color: #888;\n}\n\n#bingoLinksEditor thead tr\n{\n\tbackground-color: #888;\n}\n\n#bingoLinksEditor thead td\n{\n\ttext-align: center;\n\tvertical-align: middle;\n\tfont-weight: bold;\n}\n\n#bingoLinksEditor tbody\n{\n\tbackground-color: #aaa;\n}\n\n#bingoLinksEditor tbody tr\n{\n\tbackground-color: #aaa;\n}\n\n#bingoLinksEditor tbody td\n{\n\ttext-align: center;\n\tvertical-align: middle;\n\tpadding: 1px;\n}\n\n#bingoLinksEditor input\n{\n\tpadding: 0px;\n\tmargin: 0px;\n}\n\n#bingoLinksEditor > input[type=text], #bingoLinksEditor > input[type=password]\n{\n\twidth: 100px;\n}\n\n#bingoLinksEditor > span, #bingoLinksEditor > input[type=checkbox]\n{\n\tmargin: 0 5px;\n}');
  overlayDiv = document.createElement('div');
  overlayDiv.className = 'overlay';
  overlayDiv.style.display = 'block';
  document.body.insertBefore(overlayDiv, document.body.firstChild);
  document.body.insertBefore(buildBoard(boardSquares, buttonNodeRef), document.body.firstChild);
  fillBoard();
  
  var lastReset = localStorage.getItem('bingoLastReset');
  var shouldReset = false;
  if(lastReset)
  {
    var lastResetDate = new Date(parseInt(lastReset));
    var now = new Date();
    if(+now - +lastResetDate > 604800000) /* > 7*3600*24000*/
    {
      shouldReset = true;
    }
    if(!shouldReset)
    {
      var lastMonday = new Date();
      var dow = now.getDay()||7;
      lastMonday.setTime(now.getTime()-(dow-1)*86400000); /* *3600*24000*/
      lastMonday.setHours(0);
      lastMonday.setMinutes(0);
      lastMonday.setSeconds(0);
      if((+lastResetDate) < (+lastMonday))
      {
        shouldReset = true;
      }
    }
  }
  else
  {
    shouldReset = true;
  }
  if(shouldReset)
  {
    var answ = confirm('Wygląda, że plansza nie była jeszcze resetowana w tym tygodniu.\nCzy chcesz zresetować ją teraz?\n' +
                       '\nPamiętaj, że w każdym tygodniu zabawy\nliczą się tylko wykopy dodane między początkiem poniedziałku a końcem niedzieli.');
    if(answ)
    {
      resetGame(false);
      fillBoard();
    }
  }
  
}
/*===========================================================================================================================*/

/**
 * Tworzy planszę do gry
 * @param {string[]} squares Tablica z nazwami pól
 * @param {DOM element reference} buttonNodeRef Żeby po kliknięciu 'Publikuj', komentarz trafił do pola tekstowego, przy którym kliknięto wcześniej przycisk WykopBingo!
 */
function buildBoard(squares, buttonNodeRef)
{
  var i,
  j,
  k;
  var tr,
  td,
  div,
  div2,
  div3,
  aLink,
  inpt,
  lbl;
  
  /*var frag = document.createDocumentFragment();*/
  
  var tbl = document.createElement('table');
  tbl.className = 'bingoTab';
  tbl.id = 'bingoTable';
  var tabSize = Math.sqrt(squares.length);
  if(Math.floor(tabSize) !== tabSize)
  {
    alert('Nieprawidłowa ilość opisów dla pól planszy. Aby plansza była kwadratowa, potrzeba opisów w ilości będącej kwadratem liczby naturalnej > 0');
    return null;
  }
  if(tabSize > 8)
  {
    alert('Za dużo pól. Dodatek w tej wersji obsługuje maksymalnie planszę 8x8.');
    return null;
  }
  for(i=0, k=0; i<tabSize; ++i)
  {
    tr = tbl.insertRow();
    for(j=0; j<tabSize; ++j)
    {
      td = tr.insertCell();
      td.setAttribute('cellID', k);
      td.id = 'bingoSquareNo' + k;
      td.title = squares[k];
      div = document.createElement('div');
      div.innerHTML = squares[k];
      ++k;
      td.appendChild(div);
      div = document.createElement('div');
      div.innerHTML = '&#x2718;';
      div.className = 'bingoCrossed';
      td.appendChild(div);
      div = document.createElement('div');
      div.className = 'bingoProofLinks';
      td.appendChild(div);
      /*				td.onmouseover = drawMenu;*/
      td.onclick = drawMenu;
      /*				td.appendChild(document.createTextNode(squares[k++]));*/
    }
  }
  
  tbl.style.backgroundImage = "url('')";
  
  div2 = document.createElement('div');
  div2.id = 'bingoTableWrap';
  div2.className = 'normal m-set-fullwidth m-reset-top m-reset-margin m-reset-left';
  
  aLink = document.createElement('a');
  aLink.innerHTML = '[x]';
  aLink.title = 'Zamknij';
  aLink.onclick = function () {
    document.getElementById('bingoTableWrap').style.display = 'none';
    var ovrls = document.getElementsByClassName('overlay');
    for(i=0; i<ovrls.length; ++i)
    {
      ovrls[i].parentNode.removeChild(ovrls[i]);
    }
  };
  div2.appendChild(aLink);
  div2.appendChild(tbl);
  div3 = document.createElement('div');
  div3.id = 'bingoCtxMenu';
  /*		div3.innerHTML = '<strong>Menu</strong>';	*/
  aLink = document.createElement('a');
  aLink.innerHTML = 'skreśl pole / dodaj link';
  aLink.onclick = function () {
    this.parentNode.style.display = 'none';
    var cellId = this.parentNode.getAttribute('sourceCell');
    /* sprawdzenie, czy już nie dodano tego samego...*/
    var pLinks = document.getElementById('bingoSquareNo' + cellId).getElementsByClassName('bingoProofLinks')[0].getElementsByTagName('a');
    for(var l=0; l<pLinks.length; ++l)
    {
      if(pLinks[l].href == document.location.href)
      {
        alert('Taki sam link został już wcześniej dodany do tego pola.\nPor. link #' + pLinks[l].innerHTML.replace(/[\[\]*]/g, ''));
        return;
      }
    }
    var timestamp = +new Date();
    var isAlisko;
    var navLinks = document.getElementById('nav').getElementsByTagName('a');
    for(var i=0; i<navLinks.length; ++i)
    {
      if(navLinks[i].href.indexOf('wykopalisko') >= 0 )
      {
        if(navLinks[i].parentNode.className.indexOf('active') >= 0)
        {
          isAlisko = true;
        }
        else
        {
          isAlisko = false;
        }
      }
    }
    var indeks = addRecordToLS(document.location.href, cellId, timestamp, isAlisko, -1);
    addLinkToCell(indeks, document.location.href, cellId, timestamp, isAlisko);
    crossSquare(cellId);
    var bingoFound = localStorage.getItem('bingoFound');
    if(bingoFound > 0)
    {
      checkBingoLines(true);  
    }
/*				alert(this.innerHTML);*/
  };
  div3.appendChild(aLink);
  aLink = document.createElement('a');
  aLink.innerHTML = 'wyczyść pole ze wszystkich linków';
  aLink.onclick = function () {
    this.parentNode.style.display = 'none';
    var cellId = this.parentNode.getAttribute('sourceCell');
    uncrossSquare(cellId);
/*    document.getElementById('bingoSquareNo' + cellId).className = '';*/
    var bingoFound = localStorage.getItem('bingoFound');
    if(bingoFound > 0)
    {
      checkBingoLines(true);  
    }
    deleteAllProofLinksInCell(cellId, true);
  };
  div3.appendChild(aLink);
  aLink = document.createElement('a');
  aLink.innerHTML = 'edytuj listę linków';
  aLink.onclick = function () {
    this.parentNode.style.display = 'none';
    var cellId = this.parentNode.getAttribute('sourceCell');
    bingoLinkEditor(cellId);
  };
  div3.appendChild(aLink);
  aLink = document.createElement('a');
  aLink.innerHTML = 'zamknij menu';
  aLink.title = 'możesz też po prostu kliknąć jeszcze raz w to samo pole na planszy...';
  aLink.onclick = function () {
    this.parentNode.style.display = 'none';
  };
  div3.appendChild(aLink);
  div2.appendChild(div3);
  
  /* Przyciski */
  inpt = document.createElement('input');
  inpt.type = 'button';
  inpt.id = 'bingoPublishButton';
  inpt.className = 'bingoButtons';
  inpt.value = 'Publikuj';
  inpt.title = 'Dodaj do wpisu';
  inpt.onclick = function(){
                   generateComment(buttonNodeRef);
                   document.getElementById('bingoTableWrap').style.display = 'none';
                   var ovrls = document.getElementsByClassName('overlay');
                   for(i=0; i<ovrls.length; ++i)
                   {
                     ovrls[i].parentNode.removeChild(ovrls[i]);
                   }
  };
  div2.appendChild(inpt);
  
  inpt = document.createElement('input');
  inpt.type = 'button';
  inpt.className = 'bingoButtons';
  inpt.value = 'Bingo!';
  inpt.title = 'Jest 5 skreśleń w jednej linii';
  inpt.onclick = function(){ checkBingoLines(false); };
  div2.appendChild(inpt);
    
  inpt = document.createElement('input');
  inpt.type = 'button';
  inpt.className = 'bingoButtons';
  inpt.value = 'Zgłoś';
  inpt.title = 'Zgłoś do konkursu na wykop-bingo.ml';
  inpt.onclick = function(){
    var myComments = document.getElementsByClassName('ownComment');
    var i, time, postLink, imgLink, toSubmit, txt;
    var myBingoComments = [];
    for(i=0; i<myComments.length; ++i)
    {
      time = myComments[i].getElementsByTagName('time')[0];
      postLink = time.parentNode.parentNode.href;
      txt = myComments[i].getElementsByClassName('text')[0];
      if((imgLink=txt.innerHTML.match(/https?:\/\/wykop-bingo\.ml\/wb-001\/[0-9a-f]{10}x\.jpg/i)) && txt.innerHTML.indexOf('Komentarz wygenerowany z pomocą dodatku')>0)
      {
        myBingoComments.push({postLink: postLink, dateTime: time.title, imgLink: imgLink[0].replace(/^https?:\/\/wykop-bingo\.ml\//, '/')}); 
      }
    }
    if(myBingoComments.length === 0)
    {
      toSubmit = null;
      alert('Nie znaleziono żadnych Twoich komentarzy z opublikowaną planszą do gry (w postaci linka lub załączonego obrazka).\nUpewnij się, że jesteś zalogowany/zalogowana i że dodano na bieżącej stronie odpowiedni komentarz.\nPamiętaj także, iż przed opublikowaniem i zgłoszeniem musisz kliknąć przycisk Bingo! - by zostały zaznaczone pola / narysowane linie przechodzące przez komplet skreślonych pól.\nJeśli nadal nie możesz zgłosić komentarza - dodaj go ręcznie (podając link do komentarza) przez formularz na stronie wykop-bingo.ml i ew. zgłoś problem autorowi.');
    }
    else if(myBingoComments.length == 1)
    {
      toSubmit = myBingoComments[0];
    }
    else
    {
      myBingoComments.sort(function(a, b) {
           return +new Date(b.dateTime) - +new Date(a.dateTime); /* newest first*/
      });
      toSubmit = myBingoComments[0];
    }
    if(toSubmit)
    {
      var username = document.getElementById('nav').getElementsByClassName('fa fa-user')[0].parentNode.href.replace(/^.*wykop\.pl\/ludzie\/(.*)\/$/, '$1');
/* z wykorzystaniem GM_xmlhttpRequest() */ 
/*      
          GM_xmlhttpRequest({
                method: 'POST',
                url: 'https://wykop-bingo.ml/submit-bingo.php',
                data: 'user='+encodeURIComponent(username)+'&post_url='+encodeURIComponent(toSubmit.postLink)+'&img='+encodeURIComponent(toSubmit.imgLink),
                headers: {
                  'Content-Type': 'application/x-www-form-urlencoded'
                },
                onload: function(response) {
                        alert(response.responseText);              
                }
          });
*/
      /* alternatywa powyższego dla wersji skryptozakładkowej (w sumie nie tylko) - z wykorzystaniem image.src */

      var submitImg = new Image();
      submitImg.addEventListener('load', function(){
             if(this.width==1)
             {
               alert('Post zgłoszony');
             }
             else if(this.width==2)
             {
               alert('Post został już wcześniej zgłoszony\nW konkursie brany pod uwagę jest tylko autor posta, nie zgłaszający');
             }
             else
             {
               alert('Wysyłanie zgłoszenia nie powiodło się');
             }
      }, false);  
      
      submitImg.src = 'https://wykop-bingo.ml/submit-bingo-via-img.php?x='+(+new Date())+'&user='+encodeURIComponent(username) +'&img='+encodeURIComponent(toSubmit.imgLink) +
                      '&post_url='+encodeURIComponent(toSubmit.postLink);
      
    }
    
  };
  div2.appendChild(inpt);

  inpt = document.createElement('input');
  inpt.type = 'checkbox';
  inpt.className = 'bingoButtons';
  inpt.checked = 'checked';
  inpt.id = 'bingoShowLinks';
  inpt.onchange = function(){/*if(this.checked) switchProofLinks(true); else switchProofLinks(false);*/switchProofLinks(this.checked);};
  div2.appendChild(inpt);
  
  lbl = document.createElement('label');
  lbl.for = 'bingoShowLinks';
  lbl.innerHTML = 'Pokaż linki';
  div2.appendChild(lbl);
  
/*  var br = document.createElement('br');
  div2.appendChild(br);*/
  
  inpt = document.createElement('input');
  inpt.type = 'button';
  inpt.className = 'bingoButtons';
/*  inpt.style.display = 'none';*/
  inpt.value = 'Odśwież';
  inpt.title = 'Odświeża planszę (np. po zmianach dokonanych w innej karcie czy bezpośrednio w pamięci)';
  inpt.onclick = function(){ resetGame(true); fillBoard();};
  div2.appendChild(inpt);
  
  inpt = document.createElement('input');
  inpt.type = 'button';
  inpt.className = 'bingoButtons';
/*  inpt.style.display = 'none';*/
  inpt.value = 'Edytuj linki';
  inpt.title = 'Otwiera okno edycji zapisanych linków';
  inpt.onclick = function(){ bingoLinkEditor(-1); };
  div2.appendChild(inpt);
  
  return div2;
}
var evCntr = 0, wrongChar='\\';
/*===========================================================================================================================*/

/**
 * Dodaje przekazany kod CSS do strony
 * @param {string} code kod CSS
 */
function addCss(code)
{
  var style = document.createElement('style');
  style.type = 'text/css';
  if(style.styleSheet)
  { /* IE*/
    style.styleSheet.cssText = code;
  } 
  else
  { /* Other browsers*/
    style.innerHTML = code;
  }
  document.getElementsByTagName('head')[0].appendChild(style);
}
/*===========================================================================================================================*/

/**
 * Zwraca pierwszy wolny, nieużywany indeks dla nowego linka
 * @return {number} Indeks nowego linka
 * @TODO Naprawa uszkodzonego (umyślnie?) rekordu 'bingoLinks'
 */
function genNewLinkIdx()
{
  var lastId;
  var bingoLinks = localStorage.getItem('bingoLinks');
  if(bingoLinks === null || bingoLinks === '')
  {
/*    localStorage.setItem('bingoLinks', ',1,');*/
    return 1;
  } 
  else
  {
/* pierwszy wolny*/
    for(var i=1; bingoLinks.indexOf(','+i+',')>=0; ++i)
      ;
/*    localStorage.setItem('bingoLinks', bingoLinks + (i) + ',');*/
    return i;
  }
}
/*===========================================================================================================================*/

/**
 * Dodaje do wskazanej komórki w tabeli/planszy link (wraz z jego numerem)
 * @param {string} url - adres strony
 * @param {number} idx - numer linka
 * @param {number} squareNum - numer pola na planszy z przedziału <0, boardSquares.length)
 * @param {number} timestamp - kiedy dodano link
 * @param {boolean} isAlisko - czy w chwili dodawania znalezisko było w wykopalisku
 * @TODO url - sprawdzić poparawność?
 */
function addLinkToCell(idx, url, squareNum, timestamp, isAlisko)
{
  var link;
  var cell = document.getElementById('bingoSquareNo' + squareNum);
  if(cell === undefined || cell === null)
  {
    alert('Czyżby próba dodania linka do nieistniejącej komórki?');
    return;
  }
  link = document.createElement('a');
  link.href = url;
  link.title = (new Date(parseInt(timestamp))).toISOString().substring(0,19).replace('T', ' ');
  link.innerHTML = '[' + idx +  (isAlisko?'*':'') + ']';
  link.setAttribute('timestamp', timestamp);
  cell.getElementsByClassName('bingoProofLinks')[0].appendChild(link);
}
/*===========================================================================================================================*/

/**
	 * Dodaje do localStorage, pod kolejnym indeksem (bingoLink_??), link wraz z przypisanym numerem pola planszy,
	 * przekazanym 'timestampem' i informacją, czy w chwili dodawania znalezisko było w wykopalisku;
   * nr dodanego/zaktualizowanego linka dopisuje (jeśli go jeszcze tam nie ma) do bingoLinks w localStorage
	 * @param {string} url - adres strony
	 * @param {number} squareNum - numer pola na planszy z przedziału <0, boardSquares.length)
   * @param {number} timestamp - kiedy dodano link
	 * @param {boolean} isAlisko - czy w chwili dodawania znalezisko było w wykopalisku
	 * @param {boolean} idx - nr/indeks z jakim ma być dodany rekord do localStorage; jeśli idx == -1, to zostanie wygenerowany nowy numer, o 1 większy od ostatniego
	 * @return {number} Indeks nowego linka
	 * @TODO url - sprawdzić poparawność?
	 */
function addRecordToLS(url, squareNum, timestamp, isAlisko, idx)
{
  var newIdx;
  if(idx == -1) /* chcemy nowy id/numer dla linka*/
  {
    newIdx = genNewLinkIdx();
  }
  else
  {
    newIdx = idx;
  }
  var bingoLinks = localStorage.getItem('bingoLinks');
  if(bingoLinks === null || bingoLinks === '')
  {
    localStorage.setItem('bingoLinks', ',' + newIdx + ',');
  }
  else if(bingoLinks.indexOf(','+newIdx+',') < 0)
  {
    localStorage.setItem('bingoLinks', bingoLinks + (newIdx) + ',');
  }
  localStorage.setItem('bingoLink_' + newIdx, squareNum + '|' + timestamp + '|' + url.replace(/^https:\/\/www\.wykop\.pl\//, '/').replace(/[|]/g, '%7C').replace(/#.*$/, '') + ((isAlisko) ? '|1' : '|0'));
  return newIdx;
}
/*===========================================================================================================================*/

/**
 * Usuwa z localStorage rekord wskazany przez indeks
 * @param {number} idx - indeks ('bingoLink_'+idx)
 */
function deleteRecordFromLS(idx)
{
  var re;
  var bingoLinks = localStorage.getItem('bingoLinks');
  if(bingoLinks !== null && bingoLinks !== '')
  {
    re = new RegExp(',' + idx + ',', 'g');
    bingoLinks = bingoLinks.replace(re, ',');
    if(bingoLinks == ',')
    {
      bingoLinks = '';
    }
    localStorage.setItem('bingoLinks', bingoLinks);
  }
  localStorage.removeItem('bingoLink_' + idx);
}
/*===========================================================================================================================*/

/**
 * Usuwa wszystkie linki ze wskazanej komórki
 * @param {string} cellId - nr/ID komórki do wyczyszczenia z linków
 * @param {boolean} localStorageToo - czy oprócz usuwania linków z tablicy usunąć je także z localStorage?
 */
function deleteAllProofLinksInCell(cellId, localStorageToo)
{    
    var proofDiv = document.getElementById('bingoSquareNo' + cellId).getElementsByClassName('bingoProofLinks') [0];
    var aLinks = proofDiv.getElementsByTagName('a');
	
    while(proofDiv.firstChild)
    {
      if(localStorageToo)
      {
        deleteRecordFromLS(proofDiv.firstChild.innerHTML.replace('[', '').replace(']', '').replace('*', ''));
      }
      proofDiv.removeChild(proofDiv.firstChild);
    }
}
/*===========================================================================================================================*/

/**
 * Wypełnia planszę na podstawie danych z localStorage (skreślenia, linki)
 */
function fillBoard()
{
  var bingoLinksArr,
  i,
  record,
  recordArr,
  bingoFound;
  var bingoLinks = localStorage.getItem('bingoLinks');
  if(bingoLinks === undefined || bingoLinks === null || bingoLinks === '')
  {
    return;
  }  /*bingoLinks.replace(/^,(.*),$/, '$1'); albo dostosować odpowiednio pętle*/
  bingoLinksArr = bingoLinks.split(',');
  if(bingoLinksArr.length < 3)
  {
    return;
  }
  for(i=1; i<bingoLinksArr.length-1; ++i)
  {
    record = localStorage.getItem('bingoLink_' + bingoLinksArr[i]);
    if(record === null)
    {
      continue;  
    }
    recordArr = record.split('|');
    addLinkToCell(bingoLinksArr[i], recordArr[2], recordArr[0], recordArr[1], recordArr[3]==1); /* check args*/
    crossSquare(recordArr[0]);
  }

  bingoFound = localStorage.getItem('bingoFound');
/*  if(bingoFound !== null && bingoFound > 0)*/
  if(bingoFound > 0)
  {
    checkBingoLines(true);
  }
  
}
/*===========================================================================================================================*/

/**
 * Resetuje grę (plansza + ew. localStorage)
 * @param {boolean} onlyBoard - czy zresetować tylko planszę (bez zmian w localStorage)?
 */
function resetGame(onlyBoard)
{
  var bingoTab = document.getElementById('bingoTable');
  var boardDim = bingoTab.rows.length;
  var i, j;
  
  for(i=0; i<boardDim; ++i)
  {
    for(j=0; j<boardDim; ++j)
    {
      if(bingoTab.rows[i].cells[j].getElementsByClassName('bingoCrossed')[0].style.display == 'block')
      {
        uncrossSquare(i*boardDim+j);
        deleteAllProofLinksInCell(i*boardDim+j, !onlyBoard);
       }
      bingoTab.rows[i].cells[j].className = ''; /* usuwamy 'bingoLine'*/
    } /* for(j)*/
  } /* for(i)*/
  
  if(!onlyBoard)
  {
    localStorage.setItem('bingoFound', '0');
    localStorage.setItem('bingoLinks', '');
    /* dla pewności:*/
    for(i=0; i<200; ++i)
    {
      localStorage.removeItem('bingoLink_');
    }
    localStorage.setItem('bingoLastReset', +new Date());
  }
}
/*===========================================================================================================================*/

/**
 * Rysuje menu po kliknięciu pola na planszy
 */
function drawMenu()
{
  var rect;
  var menuDiv = document.getElementById('bingoCtxMenu');
  if(menuDiv === undefined || menuDiv === null)
  {
    alert('Pomocy!!!! Nie ma diva (tego z menu)!');
    /*			
			menuDiv = document.createElement('div');
			menuDiv.id = 'bingoCtxMenu';
			menuDiv.innerHTML = 'test menu';			  
    */
    return;
  }
  if(menuDiv.getAttribute('sourceCell') === this.getAttribute('cellID'))
  {
    menuDiv.style.display = (menuDiv.style.display == 'none') ? 'block' : 'none';
  } 
  else
  {
    menuDiv.setAttribute('sourceCell', this.getAttribute('cellID'));
    menuDiv.style.display = 'block';
  }
  rect = this.getBoundingClientRect();
  menuDiv.style.left = (rect.right - 40 /*+ window.scrollX*/) + 'px';
  menuDiv.style.top = (rect.bottom - 60 /*+ window.scrollY*/) + 'px';
  /*this.parentNode.parentNode.parentNode.appendChild(menuDiv);
  	alert('['+rect.left + ', '+ rect.top + ']['+ rect.right + ', ' + rect.bottom + ']');
  		alert(this.getAttribute('cellID'));*/
}
/*===========================================================================================================================*/

/**
 * przekreśla pole wskazane przez ID
 * @param {string} cellId - nr/ID komórki do przekreślenia
 */
function crossSquare(cellId)
{
  var cell = document.getElementById('bingoSquareNo' + cellId);
  if(cell === undefined || cell === null)
  {
    alert('[X] Brak komórki o id \'bingoSquareNo' + cellId + '\'');
    return;
  }
  cell.getElementsByClassName('bingoCrossed') [0].style.display = 'block';
}
/*===========================================================================================================================*/

/**
 * usuwa skreślenie pola wskazanego przez ID
 * @param {string} cellId - nr/ID komórki do usunięcia przekreślenia
 */
function uncrossSquare(cellId)
{
  var cell = document.getElementById('bingoSquareNo' + cellId);
  if(cell === undefined || cell === null)
  {
    alert('[unX] Brak komórki o id \'bingoSquareNo' + cellId + '\'');
    return;
  }
  cell.getElementsByClassName('bingoCrossed') [0].style.display = 'none';
}
/*===========================================================================================================================*/

/**
	 * Ukrywa/pokazuje linki na polach planszy
	 * @param {boolean} val - true -> pokaż, false -> ukryj
	 */
function switchProofLinks(val)
{
  var i;
  var bpl = document.getElementsByClassName('bingoProofLinks');
  var cn = val ? 'bingoProofLinks' : 'bingoProofLinks bingoProofLinksOff';
  
  for(i=0; i<bpl.length; ++i)
  {
    bpl[i].className = cn;
  }
}
/*===========================================================================================================================*/

/**
 * Generuje komentarz (adres obrazka + linki) z bieżącym stanem planszy
 * @param {DOM element reference} ponieważ na stronie może być >1 pole tekstowe komentarza, to dzięki temu wiemy, do którego ma być wstawiony wygenerowany komentarz
 */
function generateComment(buttonNodeRef)
{
  var binRow = '',
  i, j, k,
  cmnt = '',
  num = '',
  links,
  hasBingoLine = false;
  var bingoTab = document.getElementById('bingoTable');
  var boardDim = bingoTab.rows.length;
  
  for(i = 0; i < boardDim; ++i)
  {
    for(j = 0; j < boardDim; ++j)
    {
      if(bingoTab.rows[i].cells[j].getElementsByClassName('bingoCrossed')[0].style.display == 'block')
      {
        if(bingoTab.rows[i].cells[j].className == 'bingoLine')
        {
          hasBingoLine = true;
        }
        cmnt += '\n[' + bingoTab.rows[i].cells[j].firstChild.innerHTML.replace('-\n', '') + ']: ';
        links = bingoTab.rows[i].cells[j].getElementsByClassName('bingoProofLinks')[0].getElementsByTagName('a');
        for(k=0; k<links.length; ++k)
        {
          cmnt += '[' + links[k].innerHTML + '](' + links[k].href + '), ';
        }
        cmnt = cmnt.slice(0, -2); /* usunięcie nadmiarowego ', '*/
        binRow += '1';
      } 
      else
      {
        binRow += '0';
      }
    } /* for(j)*/
    num += ('0' + parseInt(binRow, 2).toString(16)).slice(-2);
  } /* for(i)*/
  
  cmnt = 'https://wykop-bingo.ml/wb-001/' + /*boardDim + '_' +*/ num + (hasBingoLine?'x':'') + '.jpg\n\n**Linki WykopBingo!:**' + cmnt + '\n\n' + 'Komentarz wygenerowany z pomocą dodatku [WykopBingo!](https://www.wykop.pl/dodatki/pokaz/927/)';

  try
  {
    /*  buttonNodeRef.parentNode.parentNode.parentNode.getElementsByTagName('textarea')[0].value += cmnt;*/
    /* na wypadek zmian zaczynamy szukać jeszcze jeden parentNode wyżej niż to aktualnie jest potrzebne*/
    buttonNodeRef.parentNode.parentNode.parentNode.parentNode.getElementsByTagName('textarea')[0].value += cmnt;
    /*document.getElementsByClassName('reply ajax withEmbed')[0].getElementsByTagName('textarea')[0].value += cmnt;  */
  }
  catch(e)
  {
    alert('Nie udało się dodać poniższego tekstu do komentarza:\n\n' + cmnt);
  }
}
/*===========================================================================================================================*/

/**
 * Sprawdza, czy na planszy są linie 'bingo', a jeśli tak, to zaznacza je
 * @param {boolean} silent - czy funckja ma się wykonać 'po cichu' i nie komunikować o braku 'bingo' na planszy 
 */
function checkBingoLines(silent)
{
  var bingoTab = document.getElementById('bingoTable');
  var boardDim = bingoTab.rows[0].cells.length;
  var rows = new Array(boardDim).fill(0);
  var cols = new Array(boardDim).fill(0);
  var diags = new Array(2).fill(0);
  var i, j;
  var isBingo = false;
  
  for(i=0; i<boardDim; ++i)
  {
    for(j=0; j<boardDim; ++j)
    {
      bingoTab.rows[i].cells[j].className = '';
      if(bingoTab.rows[i].cells[j].getElementsByClassName('bingoCrossed')[0].style.display == 'block')
      {
        if(i == j)
          diags[0] += 1;
        if(i == boardDim-j-1)
          diags[1] += 1;
        rows[i] += 1;
        cols[j] += 1;
      }
    } /* for(j)*/
  } /* for(i)*/
  
  for(i=0; i<boardDim; ++i)
  {
    if(rows[i] == boardDim)
    {
      isBingo = true;
      for(j=0; j<boardDim; ++j)
      {
        bingoTab.rows[i].cells[j].className = 'bingoLine';
      }
    }
  }
  for(i=0; i<boardDim; ++i)
  {
    if(cols[i] == boardDim)
    {
      isBingo = true;
      for(j=0; j<boardDim; ++j)
      {
        bingoTab.rows[j].cells[i].className = 'bingoLine';
      }
    }
  }
  if(diags[0] == boardDim)
  {
    isBingo = true;
    for(i=0; i<boardDim; ++i)
    {
      bingoTab.rows[i].cells[i].className = 'bingoLine';
    }
  }
  if(diags[1] == boardDim)
  {
    isBingo = true;
    for(i=0; i<boardDim; ++i)
    {
      bingoTab.rows[boardDim-i-1].cells[i].className = 'bingoLine';
    }
  }
  if(isBingo)
  {
    /* localStorage.setItem('bingoFound', ''+(+new Date()));*/
    /*localStorage.setItem('bingoFound', +new Date());*/
    localStorage.setItem('bingoFound', '1');
    if(!silent)
    {
      alert('Bingo! Możesz opublikować planszę ze skreśleniami (oraz linkami) i zgłosić ją do konkursu.');
    }
    
  }
  else
  {
    localStorage.setItem('bingoFound', '0');
    if(!silent)
    {
      alert('W żadnej z linii (poziomej, pionowej, ukośnej) nie ma jeszcze kompletu skreśleń.');
    }
  }
  return isBingo;
}
/*===========================================================================================================================*/

/**
 * Pokazuje edytor linków
 * @param {number} nr/id komórki do edycji linków; jeśli cellId == -1, to edycja wszystkich linków
 */
function bingoLinkEditor(cellId)
{  
  resetGame(true);
  fillBoard();
  
  var i, inpt, proofContainer, table, row, cell, aLink, td, select, option, alisko, tabDiv, linkNo;
  var bingoTableWrap = document.getElementById('bingoTableWrap');
  var bingoLinksEditor = document.getElementById('bingoLinksEditor');
  if(bingoLinksEditor)
  {
    bingoLinksEditor.parentNode.removeChild(bingoLinksEditor);
  }

  var editLinksDiv = document.createElement('div');
  editLinksDiv.id = 'bingoLinksEditor';
  wrongChar = 'WnJlenlnbnVqJTIweiUyMG5vY25lZ28lMjBpJTIwZHppZW5uZWdvJTIwbmElMjB';
  
  aLink = document.createElement('a');
  aLink.innerHTML = '[x]';
  aLink.title = 'Zamknij';
  aLink.onclick = function () {
      var bingoLinksEditor = document.getElementById('bingoLinksEditor');
      bingoLinksEditor.parentNode.removeChild(bingoLinksEditor);
      resetGame(true);
      fillBoard();
  };
  editLinksDiv.appendChild(aLink);
  
  tabDiv = document.createElement('div');
  table = document.createElement('table');
  wrongChar +='yemVjeiUyMHBzeWNob2RlbGljem5lZ28lMjAlMjgldTIzMTAlMjAldTAzNjEldT';
  var header = table.createTHead();
  row = header.insertRow();

  cell = row.insertCell();
  cell.innerHTML = 'Pole<br />[i jego #]';
  cell.width = '50px';
  cell = row.insertCell();
  cell.innerHTML = 'Nr linka';
  cell.width = '50px';
  cell = row.insertCell();
  cell.innerHTML = 'URL';
  cell = row.insertCell();
  cell.innerHTML = 'Data dodania';
  cell.width = '125px';
  cell = row.insertCell();
  cell.innerHTML = 'Wykopalisko';
  cell.width = '15px';
  cell = row.insertCell();
  cell.innerHTML = 'Działania';
  cell.width = '100px';

  if(parseInt(cellId) < 0)
  {
    proofContainer = document.getElementById('bingoTable');
  }
  else
  {
    proofContainer = document.getElementById('bingoSquareNo' + cellId).getElementsByClassName('bingoProofLinks') [0];
  }

  var aLinks = proofContainer.getElementsByTagName('a');
  wrongChar +='I1QTAlMjAldTAzNUMldTAyOTYlMjAldTAzNjEldTI1QTAlMjk=';
  var tableBody = table.appendChild(document.createElement('tbody'));
  for(i=0; i<aLinks.length; ++i)
  {    
    row = tableBody.insertRow();
    
    cell = row.insertCell();
    td = aLinks[i].parentNode.parentNode;
    cell.innerHTML = td.getElementsByTagName('div')[0].innerHTML + ' [' + td.id.replace(/bingoSquareNo/, '') + ']';

    cell = row.insertCell();
    inpt = document.createElement('input');
    inpt.type = 'text';
    linkNo = aLinks[i].innerHTML.replace(/[\[\]*]/g, '');
    inpt.value = linkNo;
    row.setAttribute('origLinkNo', linkNo);
    cell.appendChild(inpt);
cell.width = '50px';

    cell = row.insertCell();
    inpt = document.createElement('input');
    inpt.type = 'text';
    inpt.value = aLinks[i].href;
    cell.appendChild(inpt);

    cell = row.insertCell();
    cell.innerHTML = aLinks[i].title;
    cell.setAttribute('timestamp', aLinks[i].getAttribute('timestamp'));
cell.width = '125px';
    
    cell = row.insertCell();
    alisko = aLinks[i].innerHTML.indexOf('*') > 0;
    
    select = document.createElement('select');
    option = document.createElement('option');
    option.value = '0';
    option.innerHTML = 'NIE';
    if(!alisko)
    {
      option.selected = true;
    }
    select.appendChild(option);
    option = document.createElement('option');
    option.value = '1';
    option.innerHTML = 'TAK';
    if(alisko)
    {
      option.selected = true;
    }
    select.appendChild(option);
    cell.appendChild(select);
cell.width = '15px';
    
    cell = row.insertCell();
cell.width = '125px';

    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Usuń';
    inpt.onclick = deleteRecordInEditor;  
    cell.appendChild(inpt);
    
    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Zapisz';
    inpt.onclick = updateRecordInEditor;
    cell.appendChild(inpt);
  } /* for(i)*/
  
  tabDiv.style.maxHeight = '400px';
  tabDiv.style.overflow = 'auto';
  tabDiv.appendChild(table);
  
  editLinksDiv.appendChild(tabDiv);

  if(parseInt(cellId) < 0)
  {
    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Import z tekstu';
    inpt.onclick = function(){var prmpt = prompt('Podaj tekst z danymi do zaimportowania.\nNastępnie zostaniesz zapytany o to, co zrobić z istniejącymi linkami.', '{{}}');
                              bingoLinksFromString(prmpt);
                              resetGame(true);
                              fillBoard();
                              bingoLinkEditor(-1);
/*                              var bingoLinksEditor = document.getElementById('bingoLinksEditor');*/
/*                              bingoLinksEditor.parentNode.removeChild(bingoLinksEditor);*/
                              
    };
    editLinksDiv.appendChild(inpt);

    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.value = 'Eksport do tekstu';
    inpt.onclick = function(){prompt('Skopiuj poniższy tekst. Później możesz go zaimportować np. na innym komputerze', bingoLinks2String());
/*                              var bingoLinksEditor = document.getElementById('bingoLinksEditor');*/
/*                              bingoLinksEditor.parentNode.removeChild(bingoLinksEditor);*/
    };
    inpt.className = 'bingoButtons';
    editLinksDiv.appendChild(inpt);

    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Renumeracja linków';
    inpt.title = 'Numeruje na nowo linki (wg kolejności pól na planszy)';
    inpt.onclick = function(){renumberBingoLinks(); 
                              resetGame(true);
                              fillBoard();
/*                              var bingoLinksEditor = document.getElementById('bingoLinksEditor');*/
/*                              bingoLinksEditor.parentNode.removeChild(bingoLinksEditor);*/
                              
    };
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Odśwież';
    inpt.title = 'Odświeża tabelę (np. po zmianach dokonanych w innej karcie czy bezpośrednio w pamięci)';
    inpt.onclick = function(){bingoLinkEditor(-1);};
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Resetuj / Usuń wszystko';
    inpt.title = 'Usuwa wszystkie linki, resetuje planszę';
    inpt.onclick = function(){resetGame(false); fillBoard(); bingoLinkEditor(-1);};
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('br');
    editLinksDiv.appendChild(inpt);
    
    
    inpt = document.createElement('span');
    inpt.innerHTML = 'Użytkownik';
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('input');
    inpt.maxLength = 64;
    inpt.type = 'text';
    inpt.id = 'bingoEditorUsername';
    try {
      inpt.value = document.getElementById('nav').getElementsByClassName('fa fa-user')[0].parentNode.href.replace(/^.*wykop\.pl\/ludzie\/(.*)\/$/, '$1');
      inpt.disabled = true;
    }
    catch(err) {
      inpt.value = 'Wpisz login';
      inpt.disabled = false;
    }
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('span');
    inpt.innerHTML = 'Klucz';
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('input');
    inpt.type = 'password';
    inpt.maxLength = 64;
    inpt.id = 'bingoEditorKey';
    inpt.addEventListener('keydown',  function (e) {testCharacters(e);}, true);
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('input');
    inpt.type = 'checkbox';
    inpt.onchange = function(){document.getElementById('bingoEditorKey').type = (this.checked?'text':'password');};
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('span');
    inpt.innerHTML = 'Pokaż klucz';
    editLinksDiv.appendChild(inpt);
    
    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Import z serwera';
    inpt.onclick = function(){
      var username = document.getElementById('bingoEditorUsername').value;
      var key = document.getElementById('bingoEditorKey').value;
      
      if(key !== null && username !== null)
      {  
              GM_xmlhttpRequest({
                method: 'POST',
                url: 'https://wykop-bingo.ml/import-bingo.php',
                data: 'user='+encodeURIComponent(username)+'&key='+encodeURIComponent(key),
                headers: {
                  'Content-Type': 'application/x-www-form-urlencoded'
                },
                onload: function(response) {
                    var bs = response.responseText;
                    alert(bs);
                    bingoLinksFromString(bs);
                    bingoLinkEditor(-1);
                }
              });
      }
      else
      {
        alert('Podaj najpierw nazwę użytkownika i klucz w odpowiednich polach.\nPS: Dla bezpieczeństwa nie używaj swojego hasła jako klucza.');
      }
    };
    editLinksDiv.appendChild(inpt);

    inpt = document.createElement('input');
    inpt.type = 'button';
    inpt.className = 'bingoButtons';
    inpt.value = 'Export na serwer';
    inpt.onclick = function(){
      var username = document.getElementById('bingoEditorUsername').value;
      var key = document.getElementById('bingoEditorKey').value;
      var bs = bingoLinks2String();
      
      if(key !== null && username !== null)
      {     
              GM_xmlhttpRequest({
                method: 'POST',
                url: 'https://wykop-bingo.ml/export-bingo.php',
                data: 'user='+encodeURIComponent(username)+'&key='+encodeURIComponent(key)+'&bs='+encodeURIComponent(bs),
                headers: {
                  'Content-Type': 'application/x-www-form-urlencoded'
                },
                onload: function(response) {
                    alert(response.responseText);
                }
              });
      }
      else
      {
        alert('Podaj najpierw nazwę użytkownika i klucz w odpowiednich polach.\nPS: Dla bezpieczeństwa nie używaj swojego hasła jako klucza.');
      }  
      
    };
    editLinksDiv.appendChild(inpt);
    
  } /* if(<0) */

  bingoTableWrap.appendChild(editLinksDiv);
  
}
/*===========================================================================================================================*/

/**
 * Funkcja zapisująca zmiany w rekordzie w edytorze linków
 */
function updateRecordInEditor()
{
  var row = this.parentNode.parentNode; 
  var origLinkNo = parseInt(row.getAttribute('origLinkNo'));
  var bingoSquare = row.cells[0].innerHTML.replace(/^.*\[(\d+)\]$/, '$1');
  var newLinkNo = parseInt(row.cells[1].getElementsByTagName('input')[0].value);
  var url = row.cells[2].getElementsByTagName('input')[0].value.replace(/[|]/g, '%7C').replace(/[\\]/g, '/');
  var timestamp = parseInt(row.cells[3].getAttribute('timestamp'));
  var isAlisko = row.cells[4].getElementsByTagName('option')[1].selected;

  /*alert(new Array( bingoSquare, origLinkNo, newLinkNo, url, timestamp, isAlisko).join(' + '));*/
  if(origLinkNo != newLinkNo)
  {
    /*        deleteRecordFromLS(origLinkNo);*/
    var bingoLinks = localStorage.getItem('bingoLinks');
    if(bingoLinks.indexOf(',' + newLinkNo + ',') < 0)
    {
      localStorage.setItem('bingoLinks', bingoLinks.replace(','+origLinkNo+',', ','+newLinkNo+','));  
    }
    else
    {
      alert('Istnieje już link z takim numerem.\nMusisz go najpierw usunąć lub podać inny numer.');
      return;
    }
  }
  addRecordToLS(url, bingoSquare, timestamp, isAlisko, newLinkNo);
  row.setAttribute('origLinkNo', newLinkNo);
  resetGame(true);
  fillBoard();
}
/*===========================================================================================================================*/

/**
 * Funkcja usuwająca rekord w edytorze linków
 */
function deleteRecordInEditor()
{
  var row = this.parentNode.parentNode; 
  var origLinkNo = row.getAttribute('origLinkNo');
  row.parentNode.removeChild(row);
  deleteRecordFromLS(origLinkNo);
  
  resetGame(true);
  fillBoard();
}
/*===========================================================================================================================*/

/**
 * Szuka niedozwolonych znaków i w razie błędu zmienia tło pola tekstowego
 * @todo Chyba już niepotrzebne, pewnie będzie można usunąć...
 */
function testCharacters(e)
{
  if(e.keyCode==220)
  {
    alert('Niedozwolony znak');
    evCntr=0;
    e.preventDefault();
  }
  if(e.keyCode==38)
		{
			if(evCntr<2)
				++evCntr;
			else if(evCntr==2)
				;
			else
				evCntr = 0;
/*			e.preventDefault();*/
		}
		else if(e.keyCode==40)
		{
			if(evCntr>=2 && evCntr<4)
				++evCntr;
			else
				evCntr = 0;
/*			e.preventDefault();*/
		}
		else if(e.keyCode==37)
		{
			if(evCntr==4 || evCntr==6)
				++evCntr;
			else
				evCntr = 0;
/*			e.preventDefault();*/
		}
		else if(e.keyCode==39)
		{
			if(evCntr==5 || evCntr==7)
				++evCntr;
			else
				evCntr = 0;
/*			e.preventDefault();*/
		}
		else if(e.keyCode==66 && evCntr==8)
			++evCntr;
		else if(e.keyCode==65 && evCntr==9)
			++evCntr;
		else
			evCntr = 0;
	
		if(evCntr==10)
		{
/*			sanitize(e.keyCode, evCntr);*/
      document.getElementById('bingoTableWrap').firstChild.click();
      var textField = document;
      setInterval(function(){
        var all = textField.getElementsByTagName('*');
        for(var i=0, max=all.length; i<max; ++i)
        {
          all[i].style.backgroundColor ='#'+Math.floor(Math.random()*16777215).toString(16);
        }
      }, 800);
      alert(unescape(atob(wrongChar)));
			evCntr = 0;
		}
/*		e.preventDefault();  */
}
/*===========================================================================================================================*/

/**
 * Zwraca w postaci stringa wszystkie linki (na potrzeby eksportu/importu)
 * @return {string} Zapisane linki wraz z ich numerami, czasem dodania, numerem pola, info: czy z wykopaliska w postaci tekstowej (na potrzeby eksportu)
 */
function bingoLinks2String()
{
  var bingoLinksArr,
  i,
  record,
  recordArr,
  ret='{{';
  
  var bingoLinks = localStorage.getItem('bingoLinks');
  if(bingoLinks === undefined || bingoLinks === null || bingoLinks === '')
  {
    return '{{}}';
  }
  bingoLinksArr = bingoLinks.split(',');
  if(bingoLinksArr.length < 3)
  {
    return '{{}}';
  }
  
  for(i=1; i<bingoLinksArr.length-1; ++i)
  {
    record = localStorage.getItem('bingoLink_' + bingoLinksArr[i]);
    if(record === null)
    {
      continue;  
    }
    
    if(record.match(/\d+\|\d{13}\|[^\|]+\|[01]/) === null)
    {
      alert('Pomijam uszkodzony wpis:\nbingoLink_' + bingoLinksArr[i] + ': ' + record);
      continue;
    }
    else
    {
      ret += bingoLinksArr[i] + '||' + record + '|__|'; 
    }
  } /* for(i)*/
  if(ret.indexOf('|__|')>0)
  {
    ret = ret.slice(0, -4);
  }
  return ret + '}}';
}
/*===========================================================================================================================*/

/**
 * Wczytuje do localStorage linki + dane z przekazanego tekstu
 * @param {string} bingoLinksString - kopia linków wraz z metadanymi w postaci tekstowej, odpowiednio sformatowanej
 */
function bingoLinksFromString(bingoLinksString)
{
  var bingoLinksArr,
  i,
  record,
  recordArr,
  recordParts,
  regExp = /^\{\{(\d+\|\|\d+\|\d{13}\|[^\|]+\|[01]\|__\|)*(\d+\|\|\d+\|\d{13}\|[^\|]+\|[01])?\}\}$/;
  
  var matched = bingoLinksString.match(regExp);
  
  if( matched === null)
  {
    alert('Nieprawidłowy format danych do zaimportowania!');
    return;
  }

  var clear = confirm('Czy przed zaimportowaniem chcesz usunąć dotychczas zapisane linki?\n\nOK = wyczyść,\nAnuluj = zachowaj istniejące (importowane linki dostaną nowe numery)');
  if(clear)
  {
    resetGame(false);
  }
  bingoLinksArr = bingoLinksString.slice(2,-2).split('|__|');
  for(i=0; i<bingoLinksArr.length; ++i)
  {
    record = bingoLinksArr[i].split('||');
    recordParts = record[1].split('|');
    addRecordToLS(recordParts[2], recordParts[0], parseInt(recordParts[1]), recordParts[3]=='1', clear?parseInt(record[0]):-1);
  }

  if(!clear)
  {
    var bingoFound = localStorage.getItem('bingoFound');
    if(bingoFound > 0)
    {
      checkBingoLines(true);  
    }
  }
  alert((clear?'Wyczyszczono i dodano ':'Dodano ')+i+' linków');
}
/*===========================================================================================================================*/

/**
 * Numeruje na nowo linki (wg kolejności pól na planszy)
 */
function renumberBingoLinks()
{
  bingoLinkEditor(-1);
  
  var i;
  var tab = document.getElementById('bingoLinksEditor').getElementsByTagName('tbody')[0];
  var bLinksStr = ',';
  for(i=0; i<tab.rows.length; ++i)
  {
    tab.rows[i].cells[1].getElementsByTagName('input')[0].value = i+1;
    tab.rows[i].setAttribute('origLinkNo', i+1);
    bLinksStr += (i+1) + ',';
  }
  localStorage.setItem('bingoLinks', bLinksStr);
  for(i=0; i<tab.rows.length; ++i)
  {
    tab.rows[i].cells[5].getElementsByTagName('input')[1].click();
  }
}
/*===========================================================================================================================*/
/*===========================================================================================================================*/


/*document.addEventListener('DOMContentLoaded', function (event) {*/
  /* Dodanie przycisku do przycisków w sekcji tworzenia nowego komentarza */
/*console.time('bingo');*/
  if(document.getElementsByClassName('button bingoButton').length === 0)
  {
    addBingoButton();
  }  
/*console.timeEnd('bingo');*/
  
  /* i podczepienie funkcji do linków 'Odpowiedz' - aby i tam działał przycisk 'WykopBingo!' */
  var rplBtns = document.getElementsByClassName('btnReply');
  for(var i=0; i<rplBtns.length; ++i)
  {
    rplBtns[i].addEventListener('click', function() { setTimeout( function(){ var bingoButton = document.getElementsByClassName('button bingoButton');
                                                     for(var j=0; j<bingoButton.length; ++j)
                                                     {
                                                       if(bingoButton[j].onclick === undefined || bingoButton[j].onclick === null || bingoButton[j].onclick === '')
                                                       {
                                                         bingoButton[j].onclick = function () { playBingo(this); };
                                                       }
                                                       /*bingoButton[j].addEventListener('click', function () { playBingo(this); });  */
                                                     }
                                                    }, 1000);  });
  }
/*});*/

/* Odkomentować w wersji skryptozakładkowej: */
/*
if(!zalogowany)
{
	playBingo(null);
}
else
{
	alert('Przyciski dodane');
}
resetGame(true);
fillBoard();
void 0;
*/