Difference between revisions of "MediaWiki:JKey.js"

From Biowikifarm Metawiki
Jump to: navigation, search
(adjust URLs to upload.wikimedia.org and commons.wikimedia.org to https links to avoid mixed content warnings)
m
Line 1: Line 1:
/*** TEMP NOTE: HistorySubkeyHeading not yet implemented ***/
+
/*** TEMP NOTE: current class flags: jkey-hidekeymetadata (TEST), jkey-nocontrols (OK) jkey-autostart (FAIL nested keys) jkey-simplified (NEEDS TESTING) ***/
/*** TEMP NOTE: current class flags: jkey-nocontrols (OK) jkey-autostart (FAIL nested keys) jkey-simplified (NEEDS TESTING) ***/
+
/**
 
+
* @description jKey.js - Copyright (c) 2009-2012 Stephan Opitz, Andreas Plank & G. Hagedorn, JKI Berlin Dahlem
/* Copyright (c) 2009, Stephan Opitz, JKI Berlin Dahlem
+
* This program is free software; you can redistribute it and/or modify it under the terms of the EUPL v.1.1
This program is free software; you can redistribute it and/or modify it under the terms of the EUPL v.1.1 or (at your option) the GNU General Public License as published by the Free Software Foundation; either GPL v.3 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 General Public License (http://www.gnu.org/licenses/) for more details. */
+
* or (at your option) the GNU General Public License as published by the Free Software Foundation; either
 
+
* GPL v.3 or (at your option) any later version. This program is distributed in the hope that it will be
/*global $j, wgContentLanguage, wgPageName, wgUserName, importScript, window, document, location, console, alert, Image */ /* = settings for JSLint */
+
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
 
+
* PARTICULAR PURPOSE. See the GNU General Public License (http://www.gnu.org/licenses/) for more details.
"use strict"; // set ECMAScript 5 Strict Mode
+
* @requires: $.imglinkBuilder
 
+
  * @requires: $.linkBuilder
$j.jI18n = { // resource string dictionary
+
  * @requires: $.random
  en: {
+
  * @requires: $.toggleAllCollapsible
    iconStart1st : "https://upload.wikimedia.org/wikipedia/commons/thumb/4/49/View-playback_Gion_simple.svg/20px-View-playback_Gion_simple.svg.png",
+
  * TODO add documentation
    iconStartNew : "https://upload.wikimedia.org/wikipedia/commons/thumb/0/05/View-refresh_Gion_simple.svg/20px-View-refresh_Gion_simple.svg.png",
+
    iconOverview : "https://upload.wikimedia.org/wikipedia/commons/thumb/2/22/View-pause_Gion_simple.svg/20px-View-pause_Gion_simple.svg.png",
+
    iconResume  : "https://upload.wikimedia.org/wikipedia/commons/thumb/4/49/View-playback_Gion_simple.svg/20px-View-playback_Gion_simple.svg.png",
+
    iconCloseWindow : "https://upload.wikimedia.org/wikipedia/commons/8/87/Close_icon_default.jpg",
+
    iconCloseWindowHover  : "https://upload.wikimedia.org/wikipedia/commons/d/d0/Close_icon_hover.jpg",
+
    historyActiveOn : "https://upload.wikimedia.org/wikipedia/commons/thumb/9/94/Symbol_support_vote.svg/20px-Symbol_support_vote.svg.png",
+
    historyActiveOff  : "https://upload.wikimedia.org/wikipedia/commons/thumb/3/37/Symbol_partial_support_vote.svg/20px-Symbol_partial_support_vote.svg.png",
+
    playerStart1st  : "Step-by-step identification",
+
    playerStartNew  : "Start new identification",
+
    playerOverview  : "Key overview (printable)",
+
    playerResume  : "Resume",
+
    coupletContinue : " Continue ",
+
    editorEdit  : "Edit Key",
+
    editorSave  : "Save",
+
    certaintyLabel  : "Flag decision above as uncertain",
+
    certaintyHint : "(check, then make your next decision)",
+
    tryAllAlternatives  : "Undecided: Try all alternatives",
+
    mainResultMsg : "You identified: ",
+
    historyHeading  : "Previous decisions",
+
    historyConfirmable  : "Confirmable decisions:",
+
    historyNested : "Alternative",
+
    historyResult : "Result: ",
+
    historyConfirm  : "confirm",
+
    historyRevise : "revise",
+
    historyUncertainFlag  : "(uncertain)",
+
    toolTipIsActivePath : "Currently active identification path (multiple alternatives are being followed)",
+
    imageMetadataLink : "(Information about Creator, License and Copyright)",
+
    collapseCaption : " (show less) ",
+
    expandCaption : " (more...) ",
+
    expandAll : "Show all extras",
+
    toolTipClose  : "Click to close",
+
    toolTipImageZooming : "Images can be enlarged by clicking on it",
+
    zoomNotPossible : "(The enlargement function is currently not available for this image)"
+
  },
+
  de: {
+
    playerStart1st  : "Interaktive Bestimmung",
+
    playerStartNew  : "Neue Bestimmung",
+
    playerOverview  : "Übersichtsdarstellung (druckbar)",
+
    playerResume  : "Bestimmung fortsetzen",
+
    coupletContinue : " Weiter ",
+
    editorEdit  : "Bearbeiten",
+
    editorSave  : "Speichern",
+
    certaintyLabel  : "Markiere obige Entscheidung als unsicher",
+
    certaintyHint : "(markieren, dann nächste Entscheidung fällen)",
+
    tryAllAlternatives  : "Nicht entscheidbar: Verfolge alle Alternativen",
+
    mainResultMsg : "Ergebnis: ",
+
    historyHeading  : "Bisherige Entscheidungen",
+
    historyConfirmable  : "Entscheidungen in Überprüfung:",
+
    historyResult : "Ergebnis: ",
+
    historyConfirm  : "bestätigen",
+
    historyRevise : "überarbeiten",
+
    historyUncertainFlag  : "(unsicher)",
+
    toolTipIsActivePath : "Derzeit aktiver Bestimmungsweg (mehrere Alternativen werden verfolgt)",
+
    imageMetadataLink : "(Informationen zu Autor, Lizenz und Copyright)",
+
    collapseCaption : " (weniger anzeigen) ",
+
    expandCaption : " (mehr...) ",
+
    expandAll : "Alle Zusatzinformationen zeigen",
+
    toolTipClose  : "Zum Schließen klicken",
+
    toolTipImageZooming : "Bilder können durch Anklicken vergrößert betrachtet werden",
+
    zoomNotPossible : "(Die Vergrößerungsfunktion ist zur Zeit für dieses Bild nicht verfügbar)"
+
  },
+
  it: {
+
    playerStart1st  : "Esegui passo-dopo-passo",
+
    playerStartNew  : "Nuova identificazione",
+
    playerOverview  : "Sintesi completa (stampabile)",
+
    playerResume  : "Ricomincia l’identificazione",
+
    coupletContinue : " Continua ",
+
    editorEdit  : "Modifica",
+
    editorSave  : "Salva",
+
    certaintyLabel  : "Segna scelta come insicura", //REVISE
+
    certaintyHint : "(click, then make your next decision)", //TRANSLATE
+
    mainResultMsg : "Il risultato dell'identificazione è: ",
+
    historyHeading  : "Scelte precedenti",
+
    historyConfirmable  : "Scelta confermabile:",
+
    historyResult : "Risultato: ",
+
    historyConfirm  : "conferma",
+
    historyRevise : "correggi",
+
    historyUncertainFlag  : "(incerta)",
+
    toolTipIsActivePath : "Percorso di identificazione attualmente attivo (vengono seguite alternative multiple)",
+
    imageMetadataLink : "(Informazione sull'Autore, Licenza e Copyright)",
+
    collapseCaption : " (mostra di meno) ",
+
    expandCaption : " (più...) ",
+
    expandAll : "Mostra tutti informazione", //REVISE
+
    toolTipClose  : "Clicca per chiudere",
+
    toolTipImageZooming : "Le immagini possono essere ingrandite cliccandoci sopra",
+
    zoomNotPossible : "(Al momento non è possibilie ingrandire questa immagine)"
+
  }
+
};
+
 
+
/*
+
* Description: Get resource string (text, image URLs) for a given language, based on a string-key
+
*  If no resource is defined in a given language for a resource key, the resource for "en" will be returned, if this is missing as well an error message.
+
  * resourceKey: key for the resource (string)
+
 
  */
 
  */
$j.resource = function (resourceKey) {
+
/* Settings for JSLint: */
    var lang = wgContentLanguage.split("-")[0]; // language: "pt-BR", "de-formal", etc.
+
/*jslint sloppy:false, maxerr:100, indent:2 */
    return ($j.jI18n[lang] && $j.jI18n[lang][resourceKey] ?
+
/*global $, jQuery, document, console, window, alert, wgPageName, wgServer, wgScript, wgUserName, jedtInit */
            $j.jI18n[lang][resourceKey] :
+
            ($j.jI18n.en[resourceKey]) ? $j.jI18n.en[resourceKey] : "MISSING RESOURCE");
+
  };
+
  
/*
+
// set ECMAScript 5 Strict Mode. Done globally, not within each function. Later the whole jKey should be wrapped as a gadget providing scope for this
* Descriptions: Create html string for link with image and/or text content
+
"use strict";
* attributes: string of combined other attributes of link element; must use ' as inner quotes, and \" inside event functions
+
* imgResourceKey, txtResourceKey: resource keys; txtContent: direct string
+
* href: URI (for linkBuilder only)
+
*/
+
$j.linkBuilder = function (txtResourceKey, txtContent, href, attributes) {
+
  return (txtResourceKey.length ? "<a href='" + href + "' " + (attributes.length ? attributes : "") + ">" + $j.resource(txtResourceKey) + "</a>" : (txtContent.length ? "<a href='" + href + "' " + (attributes.length ? attributes : "") + ">" + txtContent + "</a>" : ""));
+
};
+
$j.imglinkBuilder = function (imgResourceKey, txtResourceKey, attributes) {
+
  return (imgResourceKey.length ? "<a href='#'" + (attributes.length ? " " + attributes : "") + "><img src='" + $j.resource(imgResourceKey) + "' /></a>&nbsp;" : "") + $j.linkBuilder(txtResourceKey, "", "#", attributes);
+
};
+
 
+
$j.random = function (min, max) { // NO CHECKS: if(min>max) {return -1;}  if(min==max) {return min;}
+
  return (min + parseInt(Math.random() * (max - min + 1), 10));
+
};
+
 
+
///////////////////////
+
// Highlight targets //
+
///////////////////////
+
 
+
/* Description: Highlight all targets of page-internal links; generic function but
+
*  especially useful in long internally linked tables like identification keys (see Template:Key_Start)
+
*/
+
 
+
// Highlight a single element that is target of the link-object caller (e.g. <a href=...>)
+
function highlightTarget(caller) {
+
  var target = $j(caller.hash.replace(/\./g, "\\.")); // hash could be 'a.34', jquery needs 'a\.34'
+
  if (target.length) {
+
    var tStyle = target.get(0).style,
+
      resetString = "resetHighlight(\"" + caller.hash + "\",\"" + tStyle.backgroundColor + "\",\"" + tStyle.textDecoration + "\")";
+
    tStyle.backgroundColor = "#EAEAEA";
+
    tStyle.textDecoration = "blink";
+
    window.setTimeout(resetString, 2000);
+
  }
+
}
+
 
+
// Stop highlighting
+
function resetHighlight(hash, backColor, txtDeco) {
+
  if (hash) { // reset
+
    var tStyle = $j(hash.replace(/\./g, "\\.")).get(0).style;
+
    tStyle.backgroundColor = backColor;
+
    tStyle.textDecoration = (txtDeco === "") ? "none" : txtDeco;
+
  }
+
}
+
 
+
// Add onclick events to all page-internal links
+
function initTargetHighlighting() {
+
  for (var i=0, max=document.links.length; i < max; i++) {
+
    var lnk = document.links[i];
+
    if ((lnk.pathname === location.pathname) && lnk.hash.length > 1) { // page internal link; exluding single "#"
+
      lnk.onclick = function() {
+
        highlightTarget(this);
+
      };
+
    }
+
  }
+
}
+
 
+
/////////////////////
+
// Collapse Tables //
+
/////////////////////
+
 
+
/* Description: Allows tables to be collapsed, showing only the header row.
+
* Original from Wikipedia; rewritten for jquery
+
*/
+
 
+
/* Description expand or collapse table
+
* caller: collapse/expand link inside a table.
+
* shallExpand: optional boolean; if absent visibility will be toggled
+
*/
+
function toggleCollapse(caller, shallExpand) {
+
  var jButton = $j(caller),
+
      jTable  = jButton.closest("table");
+
  if (jTable.length && jButton.length) {
+
    if (shallExpand===null) {
+
    /* If action not passed as parameter, then test: which text currently on button?
+
      Note: Opera 10 has problems with encodings: an "&Acute;" character gets somewhere into
+
      and comparison is always false: replace white spaces/&nbsp; + non-words (\W)
+
      */
+
      var tempCurrent = jButton.html().replace(/(&nbsp;| )/gi,'').replace(/\W+/gi,'');
+
      var tempExpand  = $j.resource("expandCaption").replace(/(&nbsp;| )/gi,'').replace(/\W+/gi,'');
+
      shallExpand = (tempCurrent==tempExpand);
+
    }
+
    jButton.html($j.resource( shallExpand ? "collapseCaption" : "expandCaption" ));
+
    jTable.find("tr:not(tr:first)").toggle(shallExpand);
+
    // show/hide all in set, tr of nested tables are included
+
  }
+
  return false;
+
}
+
 
+
function toggleAllCollapsible(shallExpand) { // all collapsible tables on wiki page
+
  $j("table span.collapseButton a").each(function() {toggleCollapse(this, shallExpand);} );
+
  $j("span.toggleAllExtras input[type=checkbox]").each(function() {this.checked = shallExpand;} ); // in case of multiple keys
+
}
+
 
+
function initCollapseButtons() {
+
  var autoCollapse = 2, // CONSTANT
+
    idx = 0,
+
    linkstring = $j.linkBuilder("collapseCaption", "", "#", "onclick='return toggleCollapse(this,null);'");
+
  var eachTable = function() { // is closure relative to idx, linkstring
+
    var jTable = $j(this),
+
      jHeader = jTable.find("tr th"); // add collapse button only if header row present
+
    if (jHeader.length) {
+
      // init "more..."-links
+
      var jButton = $j(linkstring);
+
      jButton.get(0).style.color = jHeader.get(0).style.color;
+
      jHeader.append($j('<span class="collapseButton noprint"/>').append(jButton));
+
      // do collapse
+
      if (jTable.hasClass("collapsed") ||
+
        (idx >= autoCollapse && jTable.hasClass("autocollapse")) ||
+
        // also collapse inner if innercollapse and is inside outercollapse
+
        (jTable.hasClass("innercollapse") && jTable.closest(".outercollapse").length)) {
+
        toggleCollapse(jButton.get(0),null);
+
      }
+
      idx++;
+
    }
+
  };
+
  $j("table.collapsible").each(eachTable);
+
}
+
  
 
////////////////////////
 
////////////////////////
Line 237: Line 24:
 
////////////////////////
 
////////////////////////
  
/*
+
/**
  * Description: Exception constructor, together with toString method
+
  * @description: Exception constructor, providing toString method
  * errorCode: string identifying errors
+
  * @param {string} errorCode  identifying errors
  * variables: may used for detailed information
+
  * @param {object} variables optional for detailed information
 +
* @returns {jException}
 
  */
 
  */
function JKeyException(errorCode, variables) {
+
function jException(errorCode, variables) {
 
   this.errorCode = errorCode;
 
   this.errorCode = errorCode;
 
   this.variables = variables;
 
   this.variables = variables;
 
}
 
}
JKeyException.prototype.toString = function() {
+
/**
   var message="JKey Exception " + this.errorCode + "\n\n";
+
* @augments jException()
   for (var key in this.variables) {
+
* @returns {String}
     message += " " + key + ": " + this.variables[key] + "\n";
+
*/
 +
jException.prototype.toString = function () {
 +
   var message = "JKey Exception " + this.errorCode + "\n\n",
 +
  key;
 +
   for (key in this.variables) {
 +
     // only own properties, not inherited ones:
 +
    if (this.variables.hasOwnProperty(key)) {
 +
      message += " " + key + ": " + this.variables[key] + "\n";
 +
    }
 
   }
 
   }
 
   return message;
 
   return message;
 
};
 
};
 
// also override toString of default Error constructor
 
// also override toString of default Error constructor
Error.prototype.toString = function() {  
+
Error.prototype.toString = function () {
   return "Javascript exception: " + this.name + "\n\nMessage: " + this.message + "\nFileName: " + this.fileName + "\nLineNumber: " + this.lineNumber;
+
   return "Javascript exception: " + this.name  
 +
    + "\n\nMessage: " + this.message  
 +
    + "\nFileName: " + this.fileName  
 +
    + "\nLineNumber: " + this.lineNumber;
 
};
 
};
 
 
/*
 
/*
  * Description: report exception to console or alert box.
+
  * @description report exception to console or alert box.
  * exception: JKeyException or JS internal exception
+
  * exception: jException or JS internal exception
 
  */
 
  */
function jkeyExceptionAlert(exception) {
+
function jExceptionAlert(exception) {
   if(window.console) { // IE dev. tools (F12), or FF with firebug console ENABLED
+
   if (window.console) { // IE dev.tools (=F12), or FF firebug console ENABLED
 
     console.log(exception); // TODO: in ie watch-console there is no debug fct?!
 
     console.log(exception); // TODO: in ie watch-console there is no debug fct?!
   } else { // perhaps if FF write to browser-javascript-console: throw new Error("text");
+
   } else { // perhaps in FF write to browser-javascript-console: throw new Error("text");
 
     alert(exception);
 
     alert(exception);
 
   }
 
   }
}
 
 
////////////////////////////////////
 
// Modal Layer Base functionality //
 
////////////////////////////////////
 
 
/*
 
* Description: Hide the modal layer (cyclical dependency with next method)
 
*/
 
function jkeyModalLayer_Hide() {
 
  $j(document).unbind("keydown", jkeyModalLayer_KeyDown);
 
  $j("#jkeymodal-layer").fadeOut(function() {
 
    $j("#jkeymodal-overlay").hide();
 
    $j(this).empty().hide();
 
  });
 
}
 
 
/*
 
* Description: Close (hide) modal layer on escape, backspace and arrow left key
 
* e: the keyboard event object
 
*/
 
function jkeyModalLayer_KeyDown(e) {
 
  if ((e.keyCode == 8) || (e.keyCode == 27) || (e.keyCode == 37)) { jkeyModalLayer_Hide(); }
 
}
 
 
/*
 
* Description: Create modal layer and execute myFunction
 
* myFunction: logic executed after modal layer was created
 
* myFunctionParams: generic parameters which are used by "myFunction"
 
*/
 
function jkeyModalLayer_Create(myFunction, myFunctionParams) {
 
  // find existing or create overlay & layer
 
  var jkeyModalOverlay = $j("#jkeymodal-overlay").length ? $j("#jkeymodal-overlay") : $j("<div id='jkeymodal-overlay'/>"),
 
    jkeyModalLayer = $j("#jkeymodal-layer").length ? $j("#jkeymodal-layer") : $j("<div id='jkeymodal-layer'/>");
 
  if (typeof(document.body.style.maxHeight) === "undefined") { // if IE 6
 
    $j("body","html").css({height: "100%", width: "100%"});
 
  }
 
  // append the overlay to document body
 
  $j("body").append(jkeyModalOverlay);
 
  // CSS: use full height, background opaque; then fade in
 
  jkeyModalOverlay.css("height", $j(document).height()).show();
 
  // activate keydown listener
 
  $j(document).keydown(jkeyModalLayer_KeyDown);
 
  // generic functionality
 
  jkeyModalLayer.append($j('<div style="float:right"/>')
 
    .append($j("<a/>")
 
      .append($j("<img/>").attr("src", $j.resource("iconCloseWindow")) )
 
      .hover(
 
        function() { $j(this).find("img:first").attr("src", $j.resource("iconCloseWindowHover")); },
 
        function() { $j(this).find("img:first").attr("src", $j.resource("iconCloseWindow")); })
 
      .click(function() { jkeyModalLayer_Hide(); })
 
    )
 
  );
 
  // Execute custom logic
 
  myFunction(myFunctionParams, jkeyModalLayer);
 
}
 
 
////////////////
 
// Image zoom //
 
////////////////
 
 
/*
 
* Description: load image over jkeyModalLayer
 
* jkeyModalLayer: access to DOM element
 
* newImg: image, enlarged if possible, which will be shown
 
* oriImg: original image currently being zoomed
 
*/
 
function jkeyModalLayer_LoadImage(jkeyModalLayer, newImg, oriImg) {
 
  var title = oriImg.title,
 
    imgWidth = newImg.width,
 
    imgHeight = newImg.height;
 
  if (imgWidth===0) { // in IE cloned image has no width; occurs only if newImg = oriImg.clone
 
    imgWidth = oriImg.width;
 
    imgHeight = oriImg.height;
 
  }
 
  // extend height of modal layer for no-zoom msg
 
  var zoomIsPossible = (imgWidth != oriImg.width),
 
    layerHeight = imgHeight + 105 + ((!zoomIsPossible) ? 60 : 0),
 
    layerWidth = Math.max(300, imgWidth + 70); // max to reserve min text width
 
  // delete alt text & add click function to hide modal
 
  $j(newImg).removeAttr("alt")
 
    .attr("title", title.replace($j.resource("toolTipImageZooming"),$j.resource("toolTipClose")))
 
    .click( function() {jkeyModalLayer_Hide();});
 
  // append content container
 
  jkeyModalLayer.css({width: layerWidth + "px", height: layerHeight + "px", "margin-left": -(layerWidth/2)})
 
  .append($j("<div/>")
 
    .css({"margin-left": (layerWidth-imgWidth) / 2 + "px", "margin-top": "35px"})
 
    .append(newImg)
 
  )
 
  .append($j("<div style=\"text-align:center; margin:2px; margin-top:8px;font-size:1.15em; font-weight:bold;\"/>")
 
  .append(title.replace("("+$j.resource("toolTipImageZooming")+")","")+"<br />")
 
  // URL to metadata page from "a[href]" around img
 
  .append($j($j.linkBuilder("imageMetadataLink", "", $j(oriImg).closest("a").attr("href"), "target='_blank'")))
 
  .append((!zoomIsPossible) ? ("<br/><br/><span style='color:red;'>" + $j.resource("zoomNotPossible") + "</span>") : "")
 
  );
 
  if ( !($j.browser.msie && $j.browser.version < 7)) { // take away IE6 modifications
 
    jkeyModalLayer.css({"margin-top": -((layerHeight + 8) / 2)}); // 8 from other margin-top
 
  }
 
  $j("body").append(jkeyModalLayer); // add to DOM & start
 
  jkeyModalLayer.fadeIn(50);
 
}
 
 
/*
 
* Description: add div container with image to modal layer
 
* paramsObj: needed parms for the current function
 
* parameter 1 = "link": reference to a image link including an img (dependent properties evaluated here)
 
* jkeyModalLayer: access to DOM element
 
*/
 
function jkeyModalLayer_ZoomImage(paramsObj, jkeyModalLayer) {
 
  var img = $j(paramsObj.link).find("img").get(0); // img inside a[href]
 
  var urlParts = img.src.split("/");
 
  if ((img.src.search(/\/thumb\//) === -1) || (urlParts[urlParts.length - 1].search(/px-/) === -1)) {
 
    // no larger picture possible, use existing
 
    jkeyModalLayer_LoadImage(jkeyModalLayer, $j(img).clone().get(0), img);
 
  } else { // images including "/thumb/" in path can be enlarged using URL-based wiki resize
 
    var stdThumbWidths = [1600,1280,1024,800,640,480,400,320,300,250,200,180,150,120,100,80],
 
      maxHeight = $j(window).height() - 105, // 70 are additional text; 35 are for space at top and bottom
 
      maxWidth = $j(window).width() - 50; // 50 are for space at left & right
 
    // calculate min of upscaling factors for height+width, multiply to get max possible img width
 
    var maxScaledWidth = img.width * Math.min(maxHeight/img.height, maxWidth/img.width);
 
    // reduce to next smaller standard thumb width (mediawiki preview settings plus additions)
 
    for (var i = 0; i < 16; i++) {
 
      if (stdThumbWidths[i] < maxScaledWidth) {
 
        maxScaledWidth = stdThumbWidths[i];
 
        break;
 
      }
 
    }
 
    urlParts[urlParts.length - 1] = maxScaledWidth + "px-" + urlParts[urlParts.length - 2];
 
    var newImg = new Image();
 
    // Load image. Load/error occur asynchronously, need independent calls
 
    // "random()" is necessary for IE6-8, else "zoom image, close, zoom again" fails.
 
    $j(newImg).attr("src", urlParts.join("/")+"&rnd="+$j.random(0,10000))
 
      .load(function() { // load succeeded: create modal layer after loading image, else values (width, etc.) are 0
 
        jkeyModalLayer_LoadImage(jkeyModalLayer, newImg, img);
 
      })
 
      .error(function() { // failed: load original picture
 
      // Main reason: thumbs can never be larger than ori size. Currently assuming this reason
 
      // Could test using API: https://commons.wikimedia.org/w/api.php?action=query&titles=Image:Lamium_purpureum_scan.jpg&prop=imageinfo&iiprop=size
 
      // Currently assuming and using original, unthumbed image from repository
 
      urlParts.pop();
 
      newImg = new Image();
 
      $j(newImg).attr("src", urlParts.join("/").replace("/thumb", "")+"&rnd="+$j.random(0,10000)) // remove "/thumb/" from url
 
        .load(function() {jkeyModalLayer_LoadImage(jkeyModalLayer, newImg, img);}) // load succeeded
 
        // 2nd level fail -> load from wikimedia.org
 
        .error(function() {
 
          newImg = new Image();
 
          $j(newImg).attr("src", "https://commons.wikimedia.org/w/thumb.php?f="+urlParts[urlParts.length-1]+"&width="+maxScaledWidth+"px"+"&rnd="+$j.random(0,10000))
 
            .load(function() {jkeyModalLayer_LoadImage(jkeyModalLayer, newImg, img);}) // load succeeded
 
            // 3rd level fail -> load unchanged wiki page thumb
 
            .error(function() {jkeyModalLayer_LoadImage(jkeyModalLayer, $j(img).clone().get(0), img);});       
 
        });
 
      }); // end first level error
 
  }
 
}
 
 
/*
 
* Description: Show image in modal layer
 
* caller: reference to a link around image
 
*/
 
function jkeyZoomImage(caller) {
 
  jkeyModalLayer_Create(jkeyModalLayer_ZoomImage, {link: caller});
 
  return false; // cancel default event
 
}
 
 
/*
 
* Description: Initialize all images in div.decisiontree with a modal zoom/preview functionality
 
*/
 
function jkeyInitImageZooming() {
 
  $j("div.decisiontree").find("div.floatnone img, div.floatright img, a.image img").each(function() {
 
    var jParent = $j(this).parent();
 
    if (jParent.is("a[href]")) { // = parent is link
 
      var metaURL = jParent.attr("href"),
 
        urlParts = this.src.split("/"),
 
        imgFileName = (this.src.search(/\/thumb\//) != -1) ? urlParts[urlParts.length - 2] : urlParts[urlParts.length - 1];
 
      // ignore links to targets other than media metadata page
 
      if (metaURL.search(imgFileName) == -1) { return; }
 
      // Else add new link before, move img there, remove old link
 
      // TODO replace with jParent.click( function()...); should work now that we use .clone(true)!
 
      jParent.replaceWith( $j("<a class='image' href=\""+metaURL+"\" onclick='return jkeyZoomImage(this);'><span/></a>").append(this) );
 
    } else { // new link around image
 
      jParent.prepend($j("<a class='image' href='#' onclick='return jkeyZoomImage(this);'/>").append(this));
 
    }
 
    // set or change title, set alt to title
 
    var newTitle = this.alt + ((this.alt.length === 0) ? "" : " ") + "(" + $j.resource("toolTipImageZooming") + ")";
 
    $j(this).attr({title:newTitle, alt:newTitle});
 
  } );
 
 
}
 
}
  
Line 460: Line 72:
 
/////////////////////////
 
/////////////////////////
  
/*
+
/**
  * Description: history header CONSTRUCTOR
+
  * @description: history header CONSTRUCTOR
 +
*
 +
* @returns {HistoryHeader}
 
  */
 
  */
 
function HistoryHeader() {
 
function HistoryHeader() {
  /*
+
/**
  * Description: creates new header
+
  * @description: creates new header for history of previous decisions
  * newIsActiveHistory: boolean is history set as active
+
  *  
  * newIsFirstHistory: if set hides active-history-flag
+
  * @requires $.resource()
  * newContent: the new description
+
  * @param {boolean} isActiveHistory boolean is history set as active
  */
+
  * @param {boolean} isFirstHistory hides active-history-flag if true
   this.create = function(newIsActiveHistory, newIsFirstHistory, newContent) {
+
  * @param {string} newContent the new description
 +
  * @returns {unresolved}
 +
  */
 +
   this.create = function (isActiveHistory, isFirstHistory, newContent) {
 
     // create base header
 
     // create base header
     this.item = $j('<tr class="histHeader"/>')
+
     this.item = $('<tr class="histHeader"/>')
    .append($j('<td class="histHeaderContent" colspan="3"/>').html(newContent + " &nbsp; ")
+
      .append($('<td class="histHeaderContent" colspan="3"/>').html((newContent ? newContent : $.resource("jKey_historyHeading")) + " &nbsp; ")
      .append($j('<span class="histHeaderActive"/>')
+
        .append($('<span class="histHeaderActive"/>')
        .append($j.imglinkBuilder("historyActiveOn", "", "class='histActiveOn' onclick='return jkeySwitchHistory(this);'"))
+
          .append($.imglinkBuilder("jKey_historyActive", "", "class='histActiveOn' onclick='return jkeySwitchHistory(this);'"))
        .toggle(!newIsFirstHistory)
+
          .toggle(!isFirstHistory)
        ));
+
          ));
     this.setCurrBlockActive(newIsActiveHistory);
+
     this.setCurrBlockActive(isActiveHistory);
 
     return this.item;
 
     return this.item;
 
   };
 
   };
 +
 
   // item specific fields
 
   // item specific fields
 
   /*
 
   /*
  * Description: get "active" state; return boolean
+
  * Description: get "active" state; return boolean
  */
+
  */
   this.isActive = function() {
+
   this.isActive = function () {
 
     return this.item.find("span.histHeaderActive a").hasClass("histActiveOn");
 
     return this.item.find("span.histHeaderActive a").hasClass("histActiveOn");
 
   };
 
   };
 
   /*
 
   /*
  * Description: set the active history flag
+
  * Description: set the active history flag
  * newIsActiveHistory: boolean
+
  * isActiveHistory: boolean
  */
+
  */
   this.setCurrBlockActive = function(newIsActiveHistory) { // do not use imglinkBuilder replaceWith kills layout
+
   this.setCurrBlockActive = function (isActiveHistory) { // do not use imglinkBuilder, replaceWith kills layout
 
     var histHeaderActiveCell = this.item.find("span.histHeaderActive a");
 
     var histHeaderActiveCell = this.item.find("span.histHeaderActive a");
 
     // change class & image
 
     // change class & image
     histHeaderActiveCell.attr({"class": (newIsActiveHistory ? "histActiveOn" : "histActiveOff")});
+
     histHeaderActiveCell.attr({"class": (isActiveHistory ? "histActiveOn" : "histActiveOff")});
     histHeaderActiveCell.find("img").attr("src", $j.resource(newIsActiveHistory ? "historyActiveOn" : "historyActiveOff"));
+
     histHeaderActiveCell.find("img")
 +
    .attr("src", $.resource(isActiveHistory ? "jKey_historyActive" : "jKey_historyInactive"))
 +
    .attr("title", $.resource(isActiveHistory ? "jKey_historyActiveTooltip" : "jKey_historyInactiveTooltip"));
 
   };
 
   };
}
+
}// HistoryHeader()
 
/*
 
/*
 
  * Description: history confirm subheading CONSTRUCTOR
 
  * Description: history confirm subheading CONSTRUCTOR
Line 504: Line 124:
 
function HistoryConfirmSubheading() {
 
function HistoryConfirmSubheading() {
 
   /* Description: create new subheading with newContent string */
 
   /* Description: create new subheading with newContent string */
   this.create = function(newContent) {
+
   this.create = function (newContent) {
     this.item = $j("<tr class='histConfirmSubhdg'/>")
+
     this.item = $("<tr class='histConfirmSubhdg'/>")
       .append($j("<td colspan='3' class='histConfirmSubhdgContent'/>").html(newContent));
+
       .append($("<td colspan='3' class='histConfirmSubhdgContent'/>").html(newContent));
    return this.item;
+
  };
+
}
+
/*
+
* Description: history subheading CONSTRUCTOR
+
*/
+
function HistorySubkeyHeading() {
+
  /* Description: create new subheading with newContent string */
+
  this.create = function(newContent) {
+
    this.item = $j("<tr class='histSubkeyHdg'/>")
+
    .append($j("<td colspan='3' class='histSubkeyHdgContent'/>").html(newContent));
+
 
     return this.item;
 
     return this.item;
 
   };
 
   };
Line 526: Line 135:
 
function HistoryResult() {
 
function HistoryResult() {
 
   /* Description: create new result item with newContent string */
 
   /* Description: create new result item with newContent string */
   this.create = function(newContent) {
+
   this.create = function (newContent) {
     this.item = $j("<tr class='histResult'/>")
+
     this.item = $("<tr class='histResult'/>")
    .append($j("<td class='histResultSymbol'/>").text("►"))
+
      .append($("<td class='histResultSymbol'/>").text("►"))
    .append($j("<td colspan='2' class='histResultContent'/>").html(newContent));
+
      .append($("<td colspan='2' class='histResultContent'/>").html(newContent));
 
     return this.item;
 
     return this.item;
 
   };
 
   };
Line 539: Line 148:
 
function HistoryNested() {
 
function HistoryNested() {
 
   /*
 
   /*
  * newContent: blocks with alternative paths  
+
  * newContent: blocks with alternative paths
  */
+
  */
   this.create = function(newContent) {
+
   this.create = function (newContent) {
 
     // create base step
 
     // create base step
     this.item = $j('<tr class="histNested"/>')
+
     this.item = $('<tr class="histNested"/>')
    .append($j('<td class="histNestedEmpty"/>'))
+
      .append($('<td class="histNestedEmpty"/>'))
    .append($j('<td class="histNestedContent" colspan="2"/>').html(newContent));
+
      .append($('<td class="histNestedContent" colspan="2"/>').html(newContent));
 
     return this.item;
 
     return this.item;
 
   };
 
   };
Line 554: Line 163:
 
function HistoryStep() {
 
function HistoryStep() {
 
   /*
 
   /*
  * Description: creates new step
+
  * Description: creates new step
  * newStepNumber: step number
+
  * newStepNumber: step number
  * newContent: description
+
  * newContent: description of previous decision
  * isUncertain: flag whether decision was uncertain
+
  * isUncertain: flag whether decision was uncertain
  * confCoupletID: id of confirm-couplet
+
  * confCoupletID: id of confirm-couplet
  * revCoupletID: id of revise-couplet
+
  * revCoupletID: id of revise-couplet
  */
+
  */
   this.create = function(newStepNumber, newContent, isUncertain, confCoupletID, revCoupletID) {
+
   this.create = function (newStepNumber, newContent, isUncertain, confCoupletID, revCoupletID) {
 
     // create base step
 
     // create base step
     this.item = $j('<tr class ="histStep"/>')
+
     this.item = $('<tr class ="histStep"/>')
    .append($j('<td class="histStepNumber"/>').text(newStepNumber + "."))
+
      .append($('<td class="histStepNumber"/>').text(newStepNumber + "."))
    .append($j('<td class="histStepContent"/>'))
+
      .append('<td class="histStepContent"/>')
     // Create revise-history action (changable to confirm).  
+
     // Create revise-history action (changable to confirm).
    .append($j('<td class="histStepActions"/>')
+
      .append(
      // (side-requirement: href must be confCoupletID:)
+
        $('<td class="histStepActions"/>')
      .append($j.linkBuilder("historyRevise", "", "#" + confCoupletID, " class='histStepActionRevise' onclick='return jkeyHistoryAction(this, \"" + revCoupletID + "\", \"" + confCoupletID + "\");'"))
+
        // (side-requirement: href MUST be confCoupletID:)
    );
+
          .append(
 +
            $.linkBuilder(
 +
              "jKey_historyRevise",
 +
              "",
 +
              "#" + confCoupletID,
 +
              " class='histStepActionRevise small-linkbtn' onclick='return jkeyHistoryAction(this, \"" + revCoupletID + "\", \"" + confCoupletID + "\");'"
 +
            )
 +
          )
 +
      );
 
     this.setContent(newContent);
 
     this.setContent(newContent);
 
     this.setUncertainty(isUncertain);
 
     this.setUncertainty(isUncertain);
Line 576: Line 193:
 
   };
 
   };
 
   /*
 
   /*
  * Description: set decision certainty of history step
+
  * Description: set decision certainty of history step
  * isUncertain: true -> display marker text
+
  * isUncertain: true -> display marker text
  */
+
  */
   this.setUncertainty = function(isUncertain) {
+
   this.setUncertainty = function (isUncertain) {
 
     this.item.find("td.histStepContent span.histStepCertainty")
 
     this.item.find("td.histStepContent span.histStepCertainty")
    .html(isUncertain ? ("&nbsp;" + $j.resource("historyUncertainFlag") + "&nbsp;") : "");
+
      .html(isUncertain ? ("&nbsp;" + $.resource("jKey_historyUncertainFlag") + "&nbsp;") : "");
 
   };
 
   };
 
   /*
 
   /*
  * Description: return previously recorded uncertainty for a history step (boolean)
+
  * Description: return previously recorded uncertainty for a history step (boolean)
  */
+
  */
   this.getUncertainty = function() {
+
   this.getUncertainty = function () {
 
     return this.item.find("td.histStepContent span.histStepCertainty").text().length > 0;
 
     return this.item.find("td.histStepContent span.histStepCertainty").text().length > 0;
 
   };
 
   };
 
   /*
 
   /*
  * Description: set the step content (override inheritance)
+
  * set the step content (override inheritance)
  * newContent: the new html value
+
  * @param newContent: the new html value
  */
+
  */
   this.setContent = function(newContent) {
+
   this.setContent = function (newContent) {
 
     this.item.find("td.histStepContent")
 
     this.item.find("td.histStepContent")
    .empty()
+
      .empty()
    .append(newContent)
+
      .append(newContent)
    .append($j('<span class="histStepCertainty"/>')   
+
      .append('<span class="histStepCertainty"/>');
    );
+
 
   };
 
   };
 
   /*
 
   /*
  * Description: get number in front of step
+
  * Description: get number in front of step
  */
+
  */
   this.getStepNumber = function() {
+
   this.getStepNumber = function () {
     return parseInt(this.item.find("td.histStepNumber").text(),10); // parseInt example: " 8." will return 8
+
     return parseInt(this.item.find("td.histStepNumber").text(), 10); // parseInt example: " 8." will return 8
 
   };
 
   };
 
   /*
 
   /*
  * Description: set confirm/revise action
+
  * Description: set confirm/revise action
  * setToConfirm: the new action state (true = confirm active, false = revise active)
+
  * setToConfirm: the new action state (true = confirm active, false = revise active)
  */
+
  */
   this.setConfirmable = function(setToConfirm) {
+
   this.setConfirmable = function (setToConfirm) {
 
     this.item.find("td.histStepActions a")
 
     this.item.find("td.histStepActions a")
    .attr({"class": (setToConfirm ? "histStepActionConfirm" : "histStepActionRevise")})
+
      .attr({"class": (setToConfirm ? "histStepActionConfirm small-linkbtn" : "histStepActionRevise small-linkbtn")})
    .text($j.resource((setToConfirm ? "historyConfirm" : "historyRevise")));
+
      .text($.resource((setToConfirm ? "jKey_historyConfirm" : "jKey_historyRevise")));
 
   };
 
   };
 
   /*
 
   /*
  * Description: check whether action is confirm or revise
+
  * Description: check whether action is confirm or revise
  */
+
  */
   this.isConfirmable = function() {
+
   this.isConfirmable = function () {
 
     return this.item.find("td.histStepActions a").is(".histStepActionConfirm");
 
     return this.item.find("td.histStepActions a").is(".histStepActionConfirm");
 
   };
 
   };
 
}
 
}
 
  
 
// global player history object, one key/history pair for each identification key on the html page
 
// global player history object, one key/history pair for each identification key on the html page
Line 631: Line 246:
 
function PlayerHistory(jPlayerDiv) {
 
function PlayerHistory(jPlayerDiv) {
 
   // local variables: item types = histHeader, histStep, histSubkeyHdg, histResult, histBlock;
 
   // local variables: item types = histHeader, histStep, histSubkeyHdg, histResult, histBlock;
   // no more lazy loading for header (used immediately!), and historySubkeyHeading/historyResult (small)
+
   // no more lazy loading for header (used immediately!), and historyResult (small)
 
   var jCurrHistBlock,
 
   var jCurrHistBlock,
 
     historyHeader = new HistoryHeader(),
 
     historyHeader = new HistoryHeader(),
 
     historyConfirmSubheading = new HistoryConfirmSubheading(),
 
     historyConfirmSubheading = new HistoryConfirmSubheading(),
    historySubkeyHeading = new HistorySubkeyHeading(),
 
 
     historyResult = new HistoryResult(),
 
     historyResult = new HistoryResult(),
 
     historyNested = new HistoryNested(),
 
     historyNested = new HistoryNested(),
     historyStep; // LAZY LOADING
+
     historyStep, // LAZY LOADING
 
   /*
 
   /*
  * Description: ensure that static class historyStep is loaded (lazy loading, deferring until used)
+
  * Description: ensure that static class historyStep is loaded (lazy loading, deferring until used)
  */
+
  */
  var histStep_lazyLoad = function() {
+
    histStep_lazyLoad = function () {
    if (historyStep === undefined) { historyStep = new HistoryStep(); }
+
      if (historyStep === undefined) {historyStep = new HistoryStep(); }
  };
+
    };
 
   /*
 
   /*
  * Description: create new history section in DOM tree, together with header row.
+
  * Description: create new history section in DOM tree, together with header row.
  * active: true = set as active history
+
  * active: true = set as active history
  * isFirstHistory: if set hides active-history-flag
+
  * isFirstHistory: if set hides active-history-flag
  */
+
  */
   this.createBlock = function(active, isFirstHistory, newContent) {
+
   this.createBlock = function (active, isFirstHistory, newContent) {
     return $j('<table cellpadding="0" cellspacing="0" class="'+(isFirstHistory ? "histTable" : "histBlock")+'"/>')
+
     return $('<table cellpadding="0" cellspacing="0" class="' + (isFirstHistory ? "histTable" : "histBlock") + '"/>')
       .append($j('<tr class="histLayout"/>').append($j("<td/><td/><td/>")))
+
       .append('<tr class="histLayout"/><td/><td/><td/></tr>')
       .append(historyHeader.create(active, isFirstHistory, "<b>" + ((newContent) ? newContent : $j.resource("historyHeading")) + "</b>"));
+
       .append(
 +
        historyHeader.create(
 +
          active,
 +
          isFirstHistory,
 +
          "<b>" + (newContent) ? newContent : $.resource("jKey_historyHeading") + "</b>"
 +
        )
 +
      );
 
   };
 
   };
 
   /*
 
   /*
  * Description: 3 methods to validate item: general, addable, specific type.
+
   * Description: add an item to historyTable
  * item: history table row
+
  */
  */
+
   this.addItem = function (item) {
   var isValidItem = function(item) { // maybe rename to isValidAddableItem ?!
+
    return (typeof(item) == 'object') && (item.tagName) && (item.tagName.toLowerCase() == "tr");
+
  };
+
  var isAddableItem = function(item) {
+
    return (isValidItem(item) && !$j(item).hasClass("histHeader"));
+
  };
+
  var isValidItemForType = function(item, type) {
+
    return (isValidItem(item) && $j(item).hasClass(type));
+
  };
+
  /*
+
  * Description: add an item to historyTable
+
  */
+
   this.addItem = function(item) {
+
    if (!isAddableItem(item)) {
+
      throw new JKeyException("NotValid", {info:"item type=" + typeof(item)});
+
    }
+
 
     jCurrHistBlock.append(item);
 
     jCurrHistBlock.append(item);
 
   };
 
   };
 
   /*
 
   /*
  * Description: change active history
+
  * Description: change active history
  */
+
  */
   this.changeActiveBlock = function(newActiveBlock) {
+
   this.changeActiveBlock = function (newActiveBlock) {
 
     this.setCurrBlockActive(false); // deactivate current block
 
     this.setCurrBlockActive(false); // deactivate current block
 
     this.setCurrBlock(newActiveBlock); // set new
 
     this.setCurrBlock(newActiveBlock); // set new
Line 686: Line 290:
 
   };
 
   };
 
   /*
 
   /*
  * Description: get history block
+
  * Description: get history block
  */
+
  */
   this.getCurrBlock = function() {
+
   this.getCurrBlock = function () {
 
     return jCurrHistBlock;
 
     return jCurrHistBlock;
 
   };
 
   };
 
   /*
 
   /*
  * Description: set history block
+
  * Description: set history block
  */
+
  */
   this.setCurrBlock = function(newBlock) {
+
   this.setCurrBlock = function (newBlock) {
 
     jCurrHistBlock = newBlock;
 
     jCurrHistBlock = newBlock;
 
   };
 
   };
 
   /*
 
   /*
  * Description: first item step visible confirm link
+
  * Description: first item step visible confirm link
  * historyTable: history ref
+
  * historyTable: history ref
  * return: item object or null if not found
+
  * return: item object or null if not found
  */
+
  */
   this.getfirstConfirmableStep = function() {
+
   this.getfirstConfirmableStep = function () {
 
     var retValue = null,
 
     var retValue = null,
    jAllSteps = jCurrHistBlock.find("tr:first").nextAll("tr.histStep");
+
      jAllSteps = jCurrHistBlock.find("tr:first").nextAll("tr.histStep"),
 +
      parentThis;
 
     if (jAllSteps.length) { // steps exist
 
     if (jAllSteps.length) { // steps exist
       var parentThis = this;
+
       parentThis = this;
       jAllSteps.each(function() {
+
       jAllSteps.each(function () {
 
         if ((retValue === null) && parentThis.withHistoryStep(this).isConfirmable()) {
 
         if ((retValue === null) && parentThis.withHistoryStep(this).isConfirmable()) {
 
           retValue = this;
 
           retValue = this;
Line 716: Line 321:
 
   };
 
   };
 
   /*
 
   /*
  * Description: change all history action links in history table
+
  * Description: change all history action links in history table
  * (revisable before, confirmable starting at firstConfirmableItem)
+
  * (revisable before, confirmable starting at firstConfirmableItem)
  * historyTable: history ref
+
  * historyTable: history ref
  * firstConfirmableItem: first confirmable item after updating
+
  * firstConfirmableItem: first confirmable item after updating
  */
+
  */
   this.updateConfirmability = function(firstConfirmableItem) {
+
   this.updateConfirmability = function (firstConfirmableItem) {
 
     var itemFound = false,
 
     var itemFound = false,
 
       // get steps within block, not including nested steps
 
       // get steps within block, not including nested steps
       jAllSteps = jCurrHistBlock.find("tr:first").nextAll("tr.histStep");
+
       jAllSteps = jCurrHistBlock.find("tr:first").nextAll("tr.histStep"),
 +
      parentThis;
 
     if (jAllSteps.length) { // entries exists
 
     if (jAllSteps.length) { // entries exists
       var parentThis = this;
+
       parentThis = this;
       jAllSteps.each(function() {
+
       jAllSteps.each(function () {
 
         if (this === firstConfirmableItem) {
 
         if (this === firstConfirmableItem) {
 
           itemFound = true; // true once item was passed in loop
 
           itemFound = true; // true once item was passed in loop
 
           // add confirm subheading in front of confirmable steps
 
           // add confirm subheading in front of confirmable steps
           $j(this).before(parentThis.createHistoryConfirmSubheading("<i>" + $j.resource("historyConfirmable") + "</i>"));
+
           $(this).before(parentThis.createHistoryConfirmSubheading("<i>" + $.resource("jKey_historyConfirmable") + "</i>"));
 
         }
 
         }
 
         // update history actions
 
         // update history actions
Line 739: Line 345:
 
   };
 
   };
 
   /*
 
   /*
  * Description: handle history path changes (cleanup of obsolete steps)
+
  * Description: handle history path changes (cleanup of obsolete steps)
  */
+
  */
   this.cleanupAfter = function(jStep) {
+
   this.cleanupAfter = function (jStep) {
 
     // rework history if decision path has changed. Remove confirm-sub-heading, item itself & following siblings
 
     // rework history if decision path has changed. Remove confirm-sub-heading, item itself & following siblings
 
     jStep.prevAll("tr.histConfirmSubhdg").remove();
 
     jStep.prevAll("tr.histConfirmSubhdg").remove();
 
     jStep.nextAll("tr").andSelf().remove();
 
     jStep.nextAll("tr").andSelf().remove();
 
   };
 
   };
   this.cleanupNestedBlocks = function() {
+
   this.cleanupNestedBlocks = function () {
 
     // find last normal history step (or history header; fallback if no steps exists, e.g. when using try-all as first decision)
 
     // find last normal history step (or history header; fallback if no steps exists, e.g. when using try-all as first decision)
 
     var jStep = jCurrHistBlock.find("tr:first").nextAll("tr.histHeader, tr.histStep").filter(":last");
 
     var jStep = jCurrHistBlock.find("tr:first").nextAll("tr.histHeader, tr.histStep").filter(":last");
Line 753: Line 359:
 
     }
 
     }
 
   };
 
   };
   this.cleanupConfirmableSteps = function() {
+
   this.cleanupConfirmableSteps = function () {
 
     var firstConfirmableStep = this.getfirstConfirmableStep();
 
     var firstConfirmableStep = this.getfirstConfirmableStep();
 
     if (firstConfirmableStep) {
 
     if (firstConfirmableStep) {
       this.cleanupAfter($j(firstConfirmableStep));
+
       this.cleanupAfter($(firstConfirmableStep));
 
     }
 
     }
 
   };
 
   };
  
 
   /*
 
   /*
  * Description: create a history sub-heading, nested block, or result history table row
+
  * Description: create a history sub-heading, nested block, or result history table row
  * newContent: content for item
+
  * newContent: content for item
  */  
+
  */
   this.createHistoryConfirmSubheading = function(newContent) {
+
   this.createHistoryConfirmSubheading = function (newContent) {
 
     return historyConfirmSubheading.create(newContent); // this.addItem( because will be added between block items
 
     return historyConfirmSubheading.create(newContent); // this.addItem( because will be added between block items
 
   };
 
   };
  this.createHistorySubkeyHeading = function(newContent) {
+
   this.createHistoryNested = function (newContent) {
    return historySubkeyHeading.create(newContent);
+
  };
+
   this.createHistoryNested = function(newContent) {
+
 
     this.addItem(historyNested.create(newContent).get(0));
 
     this.addItem(historyNested.create(newContent).get(0));
 
   };
 
   };
   this.createHistoryResult = function(newContent) {
+
   this.createHistoryResult = function (newContent) {
 
     this.addItem(historyResult.create(newContent).get(0));
 
     this.addItem(historyResult.create(newContent).get(0));
 
   };
 
   };
 
   /*
 
   /*
  * Description: creates a history step item
+
  * Description: creates a history step item
  */
+
  */
   this.createHistoryStep = function(newContent, isUncertain, confCoupletID, revCoupletID) {
+
   this.createHistoryStep = function (newContent, isUncertain, confCoupletID, revCoupletID) {
 
     histStep_lazyLoad();
 
     histStep_lazyLoad();
 
     this.addItem(historyStep.create(this.getNewStepNumber(), newContent, isUncertain, confCoupletID, revCoupletID).get(0));
 
     this.addItem(historyStep.create(this.getNewStepNumber(), newContent, isUncertain, confCoupletID, revCoupletID).get(0));
 
   };
 
   };
 
   /*
 
   /*
  * Description: return history step class initialized with item
+
  * Description: return history step class initialized with item
  */
+
  */
   this.withHistoryStep = function(item) {
+
   this.withHistoryStep = function (item) {
 
     histStep_lazyLoad();
 
     histStep_lazyLoad();
    if (!isValidItemForType(item, "histStep")) {
+
     historyStep.item = $(item);
      throw new JKeyException("NotValid", {info:"item type=" + typeof(item)});
+
    }
+
     historyStep.item = $j(item);
+
 
     return historyStep;
 
     return historyStep;
 
   };
 
   };
 
   // item specific fields
 
   // item specific fields
 
   /*
 
   /*
  * Description: get "active" state; return boolean
+
  * Description: get "active" state; return boolean
  */
+
  */
   this.isActive = function() {
+
   this.isActive = function () {
 
     historyHeader.item = jCurrHistBlock.find("tr.histHeader:first");
 
     historyHeader.item = jCurrHistBlock.find("tr.histHeader:first");
 
     return historyHeader.isActive();
 
     return historyHeader.isActive();
 
   };
 
   };
 
   /*
 
   /*
  * Description: set the active history flag
+
  * Description: set the active history flag
  * newIsActiveHistory: boolean
+
  * isActiveHistory: boolean
  */
+
  */
   this.setCurrBlockActive = function(newIsActive) {
+
   this.setCurrBlockActive = function (newIsActive) {
 
     historyHeader.item = jCurrHistBlock.find("tr.histHeader:first");
 
     historyHeader.item = jCurrHistBlock.find("tr.histHeader:first");
 
     historyHeader.setCurrBlockActive(newIsActive);
 
     historyHeader.setCurrBlockActive(newIsActive);
 
   };
 
   };
 
   /*
 
   /*
  * Description: get next available number for a history step in a history block (last + 1).
+
  * Description: get next available number for a history step in a history block (last + 1).
  * If the block is nested and has no steps yet, outer blocks are taken into account.
+
  * If the block is nested and has no steps yet, outer blocks are taken into account.
  * historyBlock : a jquery-history-block, defaults to jCurrHistBlock if omitted
+
  * historyBlock : a jquery-history-block, defaults to jCurrHistBlock if omitted
  */
+
  */
   this.getNewStepNumber = function(historyBlock) {
+
   this.getNewStepNumber = function (historyBlock) {
 
     if (!historyBlock) {
 
     if (!historyBlock) {
 
       historyBlock = jCurrHistBlock;
 
       historyBlock = jCurrHistBlock;
Line 824: Line 424:
 
       stepNumber = this.withHistoryStep(jLastStep.get(0)).getStepNumber();
 
       stepNumber = this.withHistoryStep(jLastStep.get(0)).getStepNumber();
 
     } else if (!historyBlock.is(".histTable")) { // unless outermost and not step (return 0): recurse to parent
 
     } else if (!historyBlock.is(".histTable")) { // unless outermost and not step (return 0): recurse to parent
       historyBlock = this.getParentBlock();    
+
       historyBlock = this.getParentBlock();
 
       return this.getNewStepNumber(historyBlock);
 
       return this.getNewStepNumber(historyBlock);
 
     }
 
     }
Line 830: Line 430:
 
   };
 
   };
 
   /*
 
   /*
  * Description: checks whether first step has to be confirmed
+
  * Description: checks whether first step has to be confirmed
  */
+
  */
   this.isFirstStepInBlockConfirmableStep = function() {
+
   this.isFirstStepInBlockConfirmableStep = function () {
 
     var firstStep = jCurrHistBlock.find("tr:first").nextAll("tr.histStep:first");
 
     var firstStep = jCurrHistBlock.find("tr:first").nextAll("tr.histStep:first");
 
     return (firstStep.prev("tr").hasClass("histConfirmSubhdg"));
 
     return (firstStep.prev("tr").hasClass("histConfirmSubhdg"));
 
   };
 
   };
 
   /*
 
   /*
  * Description: retrieve first nested block within current history block
+
  * Description: retrieve first nested block within current history block
  */
+
  */
   this.firstNestedStep = function() {
+
   this.firstNestedStep = function () {
 
     return jCurrHistBlock.find("tr:first").nextAll("tr.histNested:first");
 
     return jCurrHistBlock.find("tr:first").nextAll("tr.histNested:first");
 
   };
 
   };
 
   /*
 
   /*
  * Description: get parent block of current block (as $j object; if already outermost -> getParentBlock.length=0)
+
  * Description: get parent block of current block (as $ object; if already outermost -> getParentBlock.length=0)
  */
+
  */
   this.getParentBlock = function() {
+
   this.getParentBlock = function () {
 
     // first closest("table.histBlock, table.histTable") will find own block, then up and find parent:
 
     // first closest("table.histBlock, table.histTable") will find own block, then up and find parent:
 
     var jBlock = jCurrHistBlock.closest("table.histBlock, table.histTable");
 
     var jBlock = jCurrHistBlock.closest("table.histBlock, table.histTable");
Line 851: Line 451:
 
       jBlock = jBlock.parent().closest("table.histBlock, table.histTable");
 
       jBlock = jBlock.parent().closest("table.histBlock, table.histTable");
 
     }
 
     }
     return jBlock;  
+
     return jBlock;
 
   };
 
   };
 
+
 
 
   // Constructor logic - has to be at end of obj/class definition
 
   // Constructor logic - has to be at end of obj/class definition
 
   jCurrHistBlock = this.createBlock(true, true); // first and (here in constructor so far) only one
 
   jCurrHistBlock = this.createBlock(true, true); // first and (here in constructor so far) only one
   jPlayerDiv.append($j('<div class="jkeyHistory dt-box"/>').hide()
+
   jPlayerDiv.append(
    .append(jCurrHistBlock)
+
    $('<div class="jkeyHistory dt-box"/>').hide()
 +
      .append(jCurrHistBlock)
 
   );
 
   );
 
}
 
}
Line 870: Line 471:
 
  * return: boolean
 
  * return: boolean
 
  */
 
  */
function jkeyIsValidKeyRef(jRefTarget) {
+
function jkeyisKeyRef(jRefTarget) {
   return ( jRefTarget.is("td.dt-nodeid") || jRefTarget.is("tr.dt-row") || jRefTarget.is("div.decisiontree") );
+
   return (jRefTarget.is("td.dt-nodeid") || jRefTarget.is("tr.dt-row") || jRefTarget.is("div.decisiontree"));
 
}
 
}
  
 
/*
 
/*
 
  * Description: transform all rows of couplet
 
  * Description: transform all rows of couplet
  * jPlayerDiv: main div around player
+
  * jPlayerDiv: main div around player (unused?)
 
  * jDecisionRow: first row of a couplet - must be tested prior to calling this!
 
  * jDecisionRow: first row of a couplet - must be tested prior to calling this!
 
  * jPlayerCouplet: position where couplet will appended
 
  * jPlayerCouplet: position where couplet will appended
 +
* @todo parentlead does lead to jDecisionRow.length === 0
 
  */
 
  */
 
function jkeyTransformCouplet(jPlayerDiv, jDecisionRow, jPlayerCouplet) {
 
function jkeyTransformCouplet(jPlayerDiv, jDecisionRow, jPlayerCouplet) {
 
   if (jDecisionRow.length === 0) {
 
   if (jDecisionRow.length === 0) {
     throw new JKeyException("NotFound", {info:"Transform w/o valid jDecisionRow"});
+
     throw new jException("NotFound", {
 +
      info: "Transform w/o valid jDecisionRow"
 +
    });
 
   }
 
   }
 
   // row id is like id="Lz_1_row"; we need the id of first td inside: id="Lz_1"
 
   // row id is like id="Lz_1_row"; we need the id of first td inside: id="Lz_1"
   var currCoupletID = jDecisionRow.children("td[id]:first").attr("id");
+
   var currCoupletID = jDecisionRow.children("td[id]:first").attr("id"),
  var eachLeadout = function() { // this = a leadout link
+
    jNextCouplet,
    var nextCoupletID = this.hash,
+
    eachLeadout = function () {// this = a leadout link
      isInternalLink = ((nextCoupletID.length > 0) && (this.href.search("/"+wgPageName+"#") != -1));
+
      var nextCoupletID = this.hash,
    // Example for test above: wgPageName="ThisPage", this.href "http://.../AlsoThisPageTwo#xxx" -> include must / and #
+
        isInternalLink = ((nextCoupletID.length > 0) && (this.href.search("/" + wgPageName + "#") !== -1)),
    if (isInternalLink) {
+
        jKeyTable,
      var jNextCouplet = $j(nextCoupletID);
+
        jNodeID;
      // Check if valid element within a player was found
+
      // Example for test above: wgPageName="ThisPage", this.href "http://.../AlsoThisPageTwo#xxx" -> must include / and #
      isInternalLink = jkeyIsValidKeyRef(jNextCouplet);
+
 
       if (isInternalLink) {
 
       if (isInternalLink) {
         if (jNextCouplet.is("td.dt-nodeid")) { // already is the target node-id
+
         jNextCouplet = $(nextCoupletID);
          nextCoupletID = jNextCouplet.attr("id");
+
        // Check if valid element within a player was found
        } else { // might be row or div; get key div (closest finds itself), then table, then dt-nodeid
+
        isInternalLink = jkeyisKeyRef(jNextCouplet);
          // jKeyTable is not loop-invariable; a wiki page may have multiple keys!
+
        if (isInternalLink) {
          var jKeyTable = jNextCouplet.closest("div.decisiontree").find("table.dt-body:last"),
+
          if (jNextCouplet.is("td.dt-nodeid")) { // already is the target node-id
 +
            nextCoupletID = jNextCouplet.attr("id");
 +
          } else { // might be row or div; get key div (closest finds itself), then table, then dt-nodeid
 +
            // jKeyTable is not loop-invariable; a wiki page may have multiple keys!
 +
            jKeyTable = jNextCouplet.closest("div.decisiontree").find("table.dt-body:last");
 
             jNodeID = jKeyTable.find("td.dt-nodeid:first");
 
             jNodeID = jKeyTable.find("td.dt-nodeid:first");
          isInternalLink = (jNodeID.length>0);
+
            isInternalLink = (jNodeID.length > 0);
          if (isInternalLink) {
+
            if (isInternalLink) {
            nextCoupletID = jNodeID.attr("id");
+
              nextCoupletID = jNodeID.attr("id");
          } // else no leads found in div
+
            } // else no leads found in div
        }
+
          }
      } // else: local id NOT found or NOT valid for player
+
        } // else: local id NOT found or NOT valid for player
    } // END if isInternalLink
+
      } // END if isInternalLink
    // Following is NOT an else, isInternalLink may have been changed.
+
      // Following is NOT an else, isInternalLink may have been changed.
    nextCoupletID = (!isInternalLink) ? "" : nextCoupletID;
+
      nextCoupletID = (!isInternalLink) ? "" : nextCoupletID;
    // prepare resultlink (page or internal subkey) for player
+
      // prepare resultlink (page or internal subkey) for player
    $j(this)
+
      $(this)
      .attr("target", (isInternalLink ? "_self" : "_blank"))
+
        .attr("target", (isInternalLink ? "_self" : "_blank"))
      .addClass("linkbtn").removeAttr("style")
+
        .addClass("linkbtn").removeAttr("style")
      .click(function() {return jkeyDecision(this, false);});
+
        .click(function () {return jkeyDecision(this, false); });
    // save in data
+
      // pass autostart marker on (note: this.hash = this.hash + "jkey-autostart" is ok in FF, but IE8 behaves strangely. Using full href in IE8 seems ok!)
    $j.data(this, "coupletID", { curr:currCoupletID, next:nextCoupletID });
+
      this.href = this.href + (this.hash.length ? "" : "#") + "jkey-autostart";
  };
+
      // save in data
  var eachLeadon = function() { // this = a leadon link
+
      $.data(this, "coupletID", {curr: currCoupletID, next: nextCoupletID});
    var jThis = $j(this);
+
    },
    if (jThis.parent().hasClass("leadon")) { // for leadon (but not leadontext) overwrite display text
+
    eachLeadon = function () { // this = a leadon link
      jThis.html($j.resource("coupletContinue"));
+
      var jThis = $(this);
    }
+
      if (jThis.parent().hasClass("leadon")) { // for leadon (but not leadontext) overwrite display text
    jThis.addClass("linkbtn").removeAttr("style");
+
        jThis.html($.resource("jKey_coupletContinue"));
    // General problem: changing link onclick attribute works in FF, but is ignored by IE (known bug)
+
      }
    // One general solution is to build complete new link and delete previous one.
+
      jThis.addClass("linkbtn").removeAttr("style");
    // Also working is use of jquery.click(), but watch referencing "this". Here "this" is correct because each().
+
      // General problem: changing link onclick attribute works in FF, but is ignored by IE (known bug)
    jThis.click(function() {return jkeyDecision(this, false);});
+
      // One general solution is to build complete new link and delete previous one.
    // save in data
+
      // Also working is use of jquery.click(), but watch referencing "this". Here "this" is correct because of each().
    $j.data(this, "coupletID", { curr:currCoupletID, next:this.hash.substring(1, this.hash.length) });
+
      jThis.click(function () {return jkeyDecision(this, false); });
  };
+
      // save in data
  var prefixID = function() {return "jK" + this.id;};
+
      $.data(this, "coupletID", {curr: currCoupletID, next: this.hash.substring(1, this.hash.length)});
 +
    },
 +
    //prefixID = function () {return "jK" + this.id; },
 +
    suffixID = function () {
 +
      // leave the mw-customcollapsible untouched
 +
      // if (this.id.indexOf("mw-customcollapsible") === 0) {
 +
      //  return this.id;
 +
      // } else {
 +
      //  return this.id + "jK";
 +
      // }
 +
      return this.id + "jK";
 +
    };
 
   // Process all leads in couplet
 
   // Process all leads in couplet
 
   // Leads may be non-consecutive (general nested order = "1 2 2* 1* 3 3*" or nested subkeys = "1 alpha beta 1*->2 2->3 2* gamma epsilon 3 3*").
 
   // Leads may be non-consecutive (general nested order = "1 2 2* 1* 3 3*" or nested subkeys = "1 alpha beta 1*->2 2->3 2* gamma epsilon 3 3*").
   // jquery [attribute^=value] Matches elements that have specified attribute and it starting with value.
+
   // jquery [attribute^=value] Matches elements that have specified attribute, starting with value.
 
   // Trailing "_" after currCoupletID because non-consecutive IDs possible ("Lz_1_row", "Lz_1000_row"); Example: 3 leads within couplet = "Lz_1_row"/"Lz_1_2_row"/"Lz_1_3_row"
 
   // Trailing "_" after currCoupletID because non-consecutive IDs possible ("Lz_1_row", "Lz_1000_row"); Example: 3 leads within couplet = "Lz_1_row"/"Lz_1_2_row"/"Lz_1_3_row"
 
   // Notes: * Using nextAll().andSelf() would add first lead (jDecisionRow) at the end
 
   // Notes: * Using nextAll().andSelf() would add first lead (jDecisionRow) at the end
 
   // * Using .prev().nextAll() fails for first couplet of horizontal style, which has no row before!
 
   // * Using .prev().nextAll() fails for first couplet of horizontal style, which has no row before!
   jDecisionRow.parent().children("tr.dt-row[id^="+currCoupletID+"_]").each(function() {
+
   jDecisionRow.parent().children("tr.dt-row[id^=" + currCoupletID.replace(/(:|\.)/g,'\\$1') + "_]").each(function () {
     var jClonedRow = $j(this).clone(true);
+
     var jClonedRow = $(this).clone(true, true),
 +
      jNodeCell;
 
     // prefix id of decision row itself and all descendants
 
     // prefix id of decision row itself and all descendants
     jClonedRow.find("[id]").andSelf().attr("id", prefixID);
+
     //jClonedRow.find("[id]").andSelf().attr("id", prefixID);
 +
    //check if andSelf neccessary
 +
    //jClonedRow.find("[id]").andSelf().attr("id", suffixID);
 +
    jClonedRow.find("[id]").attr("id", suffixID);
 +
    /**
 +
    * test try default jQuery.makeCollapsible Tue Aug 08 2017 14:25:15 GMT+0200 (CEST)
 +
    * jClonedRow.find('.pseudolink')
 +
    * // remove MediaWiki's collapsible click event
 +
    * .unbind('click.mw-collapse')
 +
    * .attr('class', function (index, attr) {
 +
    * // check for MediaWiki class mw-customtoggle-myKey and add jK suffix
 +
    *  return attr.replace(/(mw-customtoggle-[^ ]+)/, "$1jK");
 +
    * });
 +
    * // remove all MediaWiki's collapsible class added previously by $.fn.makeCollapsible()
 +
    * // leave mw-collapsed untouched!
 +
    * jClonedRow
 +
    * .find('.mw-made-collapsible')
 +
    * .removeClass('mw-made-collapsible');
 +
    */
 
     // replace lead identifier with arrow (normal) or blank (horizontal side-by-side leads)
 
     // replace lead identifier with arrow (normal) or blank (horizontal side-by-side leads)
     var jNodeCell = jClonedRow.find("td.dt-nodeid:first");
+
     jNodeCell = jClonedRow.find("td.dt-nodeid:first");
 
     // CHECK why coupletID must be added here in addition to data stored generally for each link
 
     // CHECK why coupletID must be added here in addition to data stored generally for each link
 
     // couplet ID has to be extracted and stored as data
 
     // couplet ID has to be extracted and stored as data
     $j.data(jNodeCell.get(0), "couplet", { id : $j.trim(jNodeCell.text()) }); // needed for editor & multipleStep startup
+
     $.data(jNodeCell.get(0), "couplet", {id : $.trim(jNodeCell.text())}); // needed for editor & multipleStep startup
     jNodeCell.html((jClonedRow.get(0).className.search(/dt-row-hor\w+/) == -1) ? "►" : " ");
+
     jNodeCell.html((jClonedRow.get(0).className.search(/dt-row-hor\w+/) === -1) ? "►" : " "); //\u25ba is ►
 
     jClonedRow.find("td.leadalt").empty();
 
     jClonedRow.find("td.leadalt").empty();
 
     // Transform all relevant leadout (= result pages) and leadon/leadontext (next couplet) links
 
     // Transform all relevant leadout (= result pages) and leadon/leadontext (next couplet) links
Line 953: Line 590:
 
     jClonedRow.find("span.leadout a").each(eachLeadout);
 
     jClonedRow.find("span.leadout a").each(eachLeadout);
 
     jClonedRow.find("span.leadon a, div.leadontext a, span.leadontext a").each(eachLeadon);
 
     jClonedRow.find("span.leadon a, div.leadontext a, span.leadontext a").each(eachLeadon);
 +
    // initialize $.fn.makeCollapsible() again
 
     jPlayerCouplet.append(jClonedRow.show());
 
     jPlayerCouplet.append(jClonedRow.show());
 +
    // jPlayerCouplet.find('.mw-collapsible').makeCollapsible();
 +
    var cutsomToggleOptions = {
 +
      toggleClasses: true,
 +
      toggleText: {
 +
        collapseText: $.resource('CollapseBox_captionCollapse'),
 +
        expandText: $.resource('CollapseBox_captionExpand')
 +
      },
 +
      $customTogglers : jPlayerCouplet.find('.mw-customtoggle')
 +
    };
 +
    jPlayerCouplet.find('[id^="mw-customcollapsible-"]').makeCollapsible(cutsomToggleOptions);
 
   }); // END each lead row of couplet
 
   }); // END each lead row of couplet
 
}
 
}
  
 
/*
 
/*
  * Description: Load couplet in player
+
  * Description: Load couplet in player
 
  * jPlayerDiv: main div around player
 
  * jPlayerDiv: main div around player
 
  * coupletID: id of the couplet to be loaded
 
  * coupletID: id of the couplet to be loaded
Line 967: Line 615:
 
   jPlayerCouplet.empty(); // flush
 
   jPlayerCouplet.empty(); // flush
 
   // coupletID could be 'a.34', jquery needs 'a\.34'
 
   // coupletID could be 'a.34', jquery needs 'a\.34'
   // $j("#"... -> document scope, coupletID may be in other key on same page
+
   // $("#"... -> document scope, coupletID may be in other key on same page
   jkeyTransformCouplet(jPlayerDiv, $j("#" + coupletID.replace(/\./g,"\\.")).closest("tr"), jPlayerCouplet);
+
   jkeyTransformCouplet(jPlayerDiv, $("#" + coupletID.replace(/\./g, "\\.")).closest("tr"), jPlayerCouplet);
 
   // after history actions: results div may have to be hidden, and certainty div re-displayed
 
   // after history actions: results div may have to be hidden, and certainty div re-displayed
 
   jPlayerDiv.find("div.jkeyResultMsg").hide();
 
   jPlayerDiv.find("div.jkeyResultMsg").hide();
 
   jPlayerDiv.find("div.certaintyDiv").show()
 
   jPlayerDiv.find("div.certaintyDiv").show()
 
     .find("input#decisionUncertain").get(0).checked = isUncertain; // setting checkbox
 
     .find("input#decisionUncertain").get(0).checked = isUncertain; // setting checkbox
}
 
/*
 
* Description: Load result in  player
 
* jPlayerDiv: main div around player
 
* jResult = $j object containing the combined result html (commonnames, resultlink, qualifier); will be cloned
 
*/
 
function jkeyLoadResult(jPlayerDiv, jResult) {
 
  jPlayerDiv.find("div.jkeyResultMsg").empty().html($j.resource("mainResultMsg")).append(jResult.clone());
 
  jkeySetMode("finished", jPlayerDiv);
 
 
}
 
}
  
Line 996: Line 635:
 
     // Cancel if called with undefined element; may occur for "newstart"
 
     // Cancel if called with undefined element; may occur for "newstart"
 
     // when called based on undefined id from location.hash
 
     // when called based on undefined id from location.hash
     if (elementInKeyDiv === undefined) { return; }
+
     if (elementInKeyDiv === undefined) {return; }
     var jKeyDiv = $j(elementInKeyDiv).closest("div.decisiontree"),
+
     var jKeyDiv = $(elementInKeyDiv).closest("div.decisiontree"),
 
       jKeyTable = jKeyDiv.find("table.dt-body:last"),
 
       jKeyTable = jKeyDiv.find("table.dt-body:last"),
 
       jPlayerDiv = jKeyDiv.find("div.jkeyPlayer"),
 
       jPlayerDiv = jKeyDiv.find("div.jkeyPlayer"),
 
       jPlayerCouplet = jPlayerDiv.find("table.dt-body"),
 
       jPlayerCouplet = jPlayerDiv.find("table.dt-body"),
 
       jResultDiv = jPlayerDiv.find("div.jkeyResultMsg"),
 
       jResultDiv = jPlayerDiv.find("div.jkeyResultMsg"),
       jControls = jKeyDiv.find("td.jkeyControls");
+
       jControls = jKeyDiv.find(".jkeyControls"),
     if (mode == "firststart") { // change start permanently to restart icon/text, add overview/resume controls
+
      finished,
       jControls.find("span.jkeyPlayerStart1st").replaceWith("<span class='jkeyPlayerOverview nowrap'> &nbsp; &nbsp;" +
+
      overview,
        $j.imglinkBuilder("iconOverview", "playerOverview", "onclick='return jkeySetMode(\"overview\",this);'") +
+
      uniquePlayerID;
        "</span> <span class='jkeyPlayerResume nowrap'> &nbsp; &nbsp;" +
+
     if (mode === "firststart") { // change permanently: first-start to restart icon/text, add overview/resume controls, removes  toggleAllExtras!
        $j.imglinkBuilder("iconResume", "playerResume", "onclick='return jkeySetMode(\"resumed\",this);'")  +
+
       jControls.find(".jkeyPlayerStartNew").show();
        "</span> <span class='jkeyPlayerStartNew nowrap'> &nbsp; &nbsp;" +
+
      jControls.find(".jkeyPlayerStart1st, .jkeyToggleAllExtras").hide();
        $j.imglinkBuilder("iconStartNew", "playerStartNew", "onclick='return jkeySetMode(\"newstart\",this);'") +
+
      window.scrollTo(0, jKeyDiv[0].offsetTop); // scroll to current key
        "</span>");
+
 
       mode = "newstart";
 
       mode = "newstart";
 
     }
 
     }
     var finished = (mode=="finished"),
+
     finished = (mode === "finished");
      overview = !(mode=="resumed" || mode=="newstart" || finished);  
+
    overview = !(mode === "resumed" || mode === "newstart" || finished);
     jControls.find("span.jkeyPlayerOverview").toggle(!overview);
+
     jControls.find(".jkeyPlayerOverview").toggle(!overview);
     jControls.find("span.jkeyPlayerResume").toggle(overview);
+
     jControls.find(".jkeyPlayerResume").toggle(overview);
     jKeyDiv.find("div.dt-header").toggle(overview); // metadata box (description, audience, etc.)
+
     jKeyDiv.find(".dt-header,.dt-header-toggle").toggle(overview); // metadata box (description, audience, etc.) and right-floating "more" to show metadata box
 
     // Show finished message only in finished mode (+ set below for resumed)
 
     // Show finished message only in finished mode (+ set below for resumed)
 
     jResultDiv.toggle(finished);
 
     jResultDiv.toggle(finished);
     switch(mode) {
+
    // set entire wiki content area to text color css3 rgba with transparency 0.2 (jkeyCanvas is already 1.0).
     case("overview"): // STOP player, hide player, show original overview player
+
    // all separately colored text (links etc.) must be handled separately (add all, then remove inside player)
      jKeyDiv.find("div.jkeyLCtrls input.editbtn").show();
+
    // NOTE: css opacity not an option, opacity of children can not increase (= made more visible) !
 +
    $("#bodyContent")
 +
    .addClass("jkdimmed")
 +
    .find("a,a.new,a:visited,a:hover,.pseudolink,.linbtn,h2,h3,h4,h5,h6,.commonnames").addClass("jkdimmed");
 +
    jKeyDiv.find(".jkdimmed").removeClass("jkdimmed");
 +
     switch (mode) {
 +
     case ("overview"): // STOP player, hide player, show original overview player, undo low opacity content area
 
       jKeyDiv.removeClass("jkeyCanvas");
 
       jKeyDiv.removeClass("jkeyCanvas");
 
       jKeyTable.show();
 
       jKeyTable.show();
 
       jPlayerDiv.hide();
 
       jPlayerDiv.hide();
 +
      $("#bodyContent")
 +
      .removeClass("jkdimmed")
 +
      .find(".jkdimmed").removeClass("jkdimmed");
 
       break;
 
       break;
     case("newstart"):
+
     case ("newstart"):
 
       // Delete player-div in case of restart (also invalidates jPlayerCouplet)
 
       // Delete player-div in case of restart (also invalidates jPlayerCouplet)
 
       jPlayerDiv.remove();
 
       jPlayerDiv.remove();
 
       jKeyTable.hide();
 
       jKeyTable.hide();
 
       // create new player & table, transform couplet
 
       // create new player & table, transform couplet
       jPlayerCouplet = $j('<table class="dt-body" cellspacing="0" cellpadding="0"/>');
+
       jPlayerCouplet = $('<table class="dt-body" cellspacing="0" cellpadding="0"/>');
       jPlayerDiv = $j('<div class="jkeyPlayer"/>')
+
       jPlayerDiv = $('<div class="jkeyPlayer"/>')
 
         .append(jPlayerCouplet)
 
         .append(jPlayerCouplet)
 
         // Append a hidden result section (for finished mode)
 
         // Append a hidden result section (for finished mode)
         .append($j('<div class="jkeyResultMsg"/>').hide());
+
         .append($('<div class="jkeyResultMsg"/>').hide());
 
       if (!jKeyDiv.hasClass("jkey-simplified")) {
 
       if (!jKeyDiv.hasClass("jkey-simplified")) {
         jPlayerDiv.append($j('<div class="certaintyDiv noprint"/>')
+
         jPlayerDiv.append(
          .html('<input type="checkbox" id="decisionUncertain" value="1" /> <label for="decisionUncertain" style="font-weight:bold">' + $j.resource("certaintyLabel") + "</label>&nbsp;"  + $j.resource("certaintyHint") + '&nbsp; &nbsp;<span class="tryAllDiv noprint"/><input type="button" onclick="return jkeyTryAll(this);" value="'+$j.resource("tryAllAlternatives")+'" /></span>'));
+
          '<div class="certaintyDiv noprint">'
 +
            + '<input type="checkbox" id="decisionUncertain" value="1" /> '
 +
            + '<label for="decisionUncertain" style="font-weight:bold" title="'+$.resource("jKey_certaintyTooltip")+'">'
 +
            + $.resource("jKey_certaintyLabel") + "</label>&nbsp;"
 +
            + '&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;|&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; '
 +
            + $.linkBuilder("jKey_tryAllAlternatives","","#"," class='small-linkbtn' onclick='return jkeyTryAll(this);' title='"+$.resource("jKey_tryAllAlternativesTooltip")+"'")
 +
            + '</div>'
 +
        );
 
       }
 
       }
       // generate unique player id     
+
       // generate unique playerID (multiple keys may exist on 1 page!):
      var uniquePlayerID;
+
 
       do {
 
       do {
         uniquePlayerID = "jkp_"+$j.random(1,9999999);
+
         uniquePlayerID = "jkp_" + $.random(1, 9999999);
       } while (jkeyHistory.uniquePlayerID); // until not yet used
+
       } while (jkeyHistory.uniquePlayerID); // until ID not yet used
 
       // add id to connect with history
 
       // add id to connect with history
 
       jPlayerDiv.attr("id", uniquePlayerID);
 
       jPlayerDiv.attr("id", uniquePlayerID);
Line 1,053: Line 706:
 
       jKeyTable.before(jPlayerDiv); // add to DOM
 
       jKeyTable.before(jPlayerDiv); // add to DOM
 
       jkeyTransformCouplet(jKeyDiv, jKeyTable.find("tr.dt-row:first"), jPlayerCouplet);
 
       jkeyTransformCouplet(jKeyDiv, jKeyTable.find("tr.dt-row:first"), jPlayerCouplet);
       // NOTE: NO BREAK here, fallthrough to "resumed"
+
       // NOTE: NO BREAK here, fallthrough to "resumed" intended!
     case("resumed"): // = player resumed. This may have to show couplet or finished mode
+
     case ("resumed"): // = player resumed. This may have to show couplet or finished mode
 
       // style change for key div + hide original table & show table in player
 
       // style change for key div + hide original table & show table in player
      jKeyDiv.find("div.jkeyLCtrls input.editbtn").hide();
 
 
       jKeyDiv.addClass("jkeyCanvas");
 
       jKeyDiv.addClass("jkeyCanvas");
 
       jKeyTable.hide();
 
       jKeyTable.hide();
Line 1,062: Line 714:
 
       jPlayerDiv.show();
 
       jPlayerDiv.show();
 
       // redisplay result div if still filled
 
       // redisplay result div if still filled
       jResultDiv.toggle(jResultDiv.text().length>0);
+
       jResultDiv.toggle(jResultDiv.text().length > 0);
 
       break;
 
       break;
     case("finished"): // empty current couplet, hide certainty checkbox
+
     case ("finished"): // empty current couplet, hide certainty checkbox
 
       jPlayerCouplet.empty();
 
       jPlayerCouplet.empty();
 
       jPlayerDiv.find("div.certaintyDiv").hide();
 
       jPlayerDiv.find("div.certaintyDiv").hide();
Line 1,071: Line 723:
 
     return false; // cancel default event
 
     return false; // cancel default event
 
   } catch (err) {
 
   } catch (err) {
     jkeyExceptionAlert(err);
+
     jExceptionAlert(err);
 
   }
 
   }
 +
}
 +
 +
/*
 +
* Description: Load result in  player
 +
* jPlayerDiv: main div around player
 +
* jResult = $ object containing the combined result html (commonnames, resultlink, qualifier); will be cloned
 +
*/
 +
function jkeyLoadResult(jPlayerDiv, jResult) {
 +
  jPlayerDiv.find("div.jkeyResultMsg").empty().html($.resource("jKey_mainResultMsg")).append(jResult.clone());
 +
  jkeySetMode("finished", jPlayerDiv);
 
}
 
}
  
Line 1,079: Line 741:
 
  * caller: DOM link; confirm if class=histStepActionConfirm, revise if histStepActionRevise
 
  * caller: DOM link; confirm if class=histStepActionConfirm, revise if histStepActionRevise
 
  * revCoupletID, confCoupletID: id of couplet to be revised or confirmed
 
  * revCoupletID, confCoupletID: id of couplet to be revised or confirmed
  */  
+
  */
 
function jkeyHistoryAction(caller, revCoupletID, confCoupletID) {
 
function jkeyHistoryAction(caller, revCoupletID, confCoupletID) {
 
   try {
 
   try {
 
     // get currently active history (find first & look in hierarchy)
 
     // get currently active history (find first & look in hierarchy)
     var jCaller = $j(caller),
+
     var jCaller = $(caller),
 
       jStepItem = jCaller.closest("tr"), // history row around action link
 
       jStepItem = jCaller.closest("tr"), // history row around action link
 
       jPlayerDiv = jStepItem.closest("div.jkeyPlayer"),
 
       jPlayerDiv = jStepItem.closest("div.jkeyPlayer"),
       jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")];
+
       jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")],
 +
      firstConfirmableStep,
 +
      nextDecisionIsUncertain;
 
     // set history block to the one containing the caller (or outermost histTable itself)
 
     // set history block to the one containing the caller (or outermost histTable itself)
 
     jCurrHistory.changeActiveBlock(jCaller.closest("table.histBlock, table.histTable"));
 
     jCurrHistory.changeActiveBlock(jCaller.closest("table.histBlock, table.histTable"));
Line 1,095: Line 759:
 
       jStepItem = jStepItem.nextAll("tr.histStep:first");
 
       jStepItem = jStepItem.nextAll("tr.histStep:first");
 
       // Update history step certainty from player checkbox
 
       // Update history step certainty from player checkbox
       var firstConfirmableStep = jCurrHistory.getfirstConfirmableStep();
+
       firstConfirmableStep = jCurrHistory.getfirstConfirmableStep();
 
       if (firstConfirmableStep) { // null if none found
 
       if (firstConfirmableStep) { // null if none found
 
         // firstConfirmableStep is the couplet shown in player!
 
         // firstConfirmableStep is the couplet shown in player!
Line 1,105: Line 769:
 
     jCurrHistory.getCurrBlock().find("tr.histConfirmSubhdg").remove();
 
     jCurrHistory.getCurrBlock().find("tr.histConfirmSubhdg").remove();
 
     if (revCoupletID === "") { // RESULT
 
     if (revCoupletID === "") { // RESULT
       // try-all may result in multiple alternative results. Thus confirming a result needs to refresh rather than re-display    
+
       // try-all may result in multiple alternative results. Thus confirming a result needs to refresh rather than re-display
 
       jkeyLoadResult(jPlayerDiv, jCaller.closest("tr.histStep").next("tr.histResult").find("span.leadout"));
 
       jkeyLoadResult(jPlayerDiv, jCaller.closest("tr.histStep").next("tr.histResult").find("span.leadout"));
 
     } else { // Refresh player with couplet corresponding to jStep
 
     } else { // Refresh player with couplet corresponding to jStep
       var nextDecisionIsUncertain = false; // default
+
       nextDecisionIsUncertain = false; // default
 
       if (jStepItem.length) { // extract history step certainty
 
       if (jStepItem.length) { // extract history step certainty
 
         nextDecisionIsUncertain = jCurrHistory.withHistoryStep(jStepItem.get(0)).getUncertainty();
 
         nextDecisionIsUncertain = jCurrHistory.withHistoryStep(jStepItem.get(0)).getUncertainty();
Line 1,116: Line 780:
 
     jCurrHistory.updateConfirmability(jStepItem.get(0));
 
     jCurrHistory.updateConfirmability(jStepItem.get(0));
 
   } catch (err) {
 
   } catch (err) {
     jkeyExceptionAlert(err);
+
     jExceptionAlert(err);
 
   }
 
   }
 
   return false; // cancel default event
 
   return false; // cancel default event
Line 1,130: Line 794:
 
  */
 
  */
 
function jkeySimplifiedLeadtxt(leadLinkCaller) {
 
function jkeySimplifiedLeadtxt(leadLinkCaller) {
   var jContainer = $j(leadLinkCaller).closest("td.leadresult, th.leadtxt"),
+
   var jContainer = $(leadLinkCaller).closest("td.leadresult, th.leadtxt, td.leadtxt"), // TODO remove th.leadtxt later (Freitag, 23. März 2012 13:42)
     jSpan;
+
     jSpan,
 +
    jSimplifiedLeadTxt;
 
   if (jContainer.hasClass("leadtxt")) {
 
   if (jContainer.hasClass("leadtxt")) {
 
     // occurs if leadspan itself is formatted as link (having both leadspan and leadontext class)
 
     // occurs if leadspan itself is formatted as link (having both leadspan and leadontext class)
Line 1,137: Line 802:
 
   } else if (jContainer.hasClass("leadresult")) {
 
   } else if (jContainer.hasClass("leadresult")) {
 
     // pos of span.leadspan depends on arrangement (horizontal or not)
 
     // pos of span.leadspan depends on arrangement (horizontal or not)
     jSpan = (jContainer.get(0).className.search(/leadresult-hor\w+/) != -1) ? jContainer.closest("table").find("td.leadtxt:nth-child(" + (jContainer.get(0).cellIndex + 1) + ")").find("span.leadspan") : jContainer.prev("th.leadtxt").find("span.leadspan");
+
     if (jContainer.get(0).className.search(/leadresult-hor/) !== -1) {
 +
      // result is separated by a table line.
 +
      jSpan = jContainer.closest("table").find("td.leadtxt:nth-child(" + (jContainer.get(0).cellIndex + 1) + ")").find("span.leadspan");
 +
    } else {
 +
      jSpan = jContainer.closest("table").find("td.leadtxt").find("span.leadspan");
 +
    }
 
   }
 
   }
 
   if (jSpan.length === 0) {
 
   if (jSpan.length === 0) {
     throw new JKeyException("NotFound", {info:"No lead statement!"});
+
     throw new jException("NotFound", {
 +
      info: "No lead statement!",
 +
      jContainer: jContainer,
 +
      jContainer_closest_table: jContainer.closest("table"),
 +
      leadLinkCaller: leadLinkCaller
 +
    });
 
   }
 
   }
 
   // Clone span; remove links, breaks, imgs
 
   // Clone span; remove links, breaks, imgs
   var jSimplifiedLeadTxt = jSpan.clone(true);
+
   jSimplifiedLeadTxt = jSpan.clone(true);
   jSimplifiedLeadTxt.find("a").each( function() { $j(this).replaceWith($j(this).html()); } );
+
   jSimplifiedLeadTxt.find("a").each(function () {$(this).replaceWith($(this).html()); });
   jSimplifiedLeadTxt.find("br, img").remove();
+
   jSimplifiedLeadTxt.find("br, img").remove();
 
   return jSimplifiedLeadTxt;
 
   return jSimplifiedLeadTxt;
 
}
 
}
  
/*
+
/**
  * Description: Used in onclick events for both next couplet and final result, i.e. user made a decision.
+
  * @description: Used in onclick events for both next couplet and final result, i.e. user made a decision.
 
  *  Add current lead to history, show next couplet in player or finish (result)
 
  *  Add current lead to history, show next couplet in player or finish (result)
  * caller: DOM reference of the calling element (has $j.data with members: next & curr (couplet ID))
+
  * @param {selector} caller DOM reference of the calling element (has $.data with members: next & curr (couplet ID))
  * quiet: do not output couplets or results, only add to history
+
  * @param {boolean} quiet do not output couplets or results, only add to history
 
  */
 
  */
function jkeyDecision(caller, quiet) {  
+
function jkeyDecision(caller, quiet) {
 
   try {
 
   try {
     var jCaller = $j(caller),
+
     var jCaller = $(caller),
       jContainer = jCaller.closest("td.leadresult, th.leadtxt"),
+
       jContainer = jCaller.closest("td.leadresult, th.leadtxt, td.leadtxt"), // TODO remove th.leadtxt later (Freitag, 23. März 2012 13:42)
 
       jPlayerDiv = jContainer.closest("div.jkeyPlayer"),
 
       jPlayerDiv = jContainer.closest("div.jkeyPlayer"),
 
       jSimplifiedLeadTxt = jkeySimplifiedLeadtxt(caller), // single time this is called
 
       jSimplifiedLeadTxt = jkeySimplifiedLeadtxt(caller), // single time this is called
 
       jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")],
 
       jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")],
       nextCoupletID = $j.data(caller, "coupletID").next; // ref to id attribute of next couplet when used on next couplet link
+
       nextCoupletID = $.data(caller, "coupletID").next, // ref to id attribute of next couplet when used on next couplet link
 +
      firstConfirmableStep,
 +
      jParentBlock,
 +
      jStep,
 +
      firstConfItemActionLink,
 +
      jResult;
 
     // Decision in player may be identical to current confirmable history step
 
     // Decision in player may be identical to current confirmable history step
     var firstConfirmableStep = jCurrHistory.getfirstConfirmableStep();
+
     firstConfirmableStep = jCurrHistory.getfirstConfirmableStep();
 
     if (firstConfirmableStep) { // = null if not found
 
     if (firstConfirmableStep) { // = null if not found
       // is it a nested block and is it first step, which has to be confirmed
+
       // is it a nested block and is it first step, which has to be confirmed?
       // means user decided to change path so that the nested block will be removed
+
       // -> user decided to change path so that the nested block will be removed
       var jParentBlock = jCurrHistory.getParentBlock(); // null if not nested
+
       jParentBlock = jCurrHistory.getParentBlock(); // null if not nested
 
       // in inner, nested blocks, confirming first step implies removing the try-all structure
 
       // in inner, nested blocks, confirming first step implies removing the try-all structure
       if (!jParentBlock.is(".histTable") && jCurrHistory.isFirstStepInBlockConfirmableStep()) {  
+
       if (!jParentBlock.is(".histTable") && jCurrHistory.isFirstStepInBlockConfirmableStep()) {
 
         jCurrHistory.setCurrBlock(jParentBlock);
 
         jCurrHistory.setCurrBlock(jParentBlock);
 
         jCurrHistory.setCurrBlockActive(true);
 
         jCurrHistory.setCurrBlockActive(true);
 
         jCurrHistory.cleanupNestedBlocks();
 
         jCurrHistory.cleanupNestedBlocks();
       } else { //  
+
       } else { //
         var jStep = $j(firstConfirmableStep);
+
         jStep = $(firstConfirmableStep);
 
         // Is nextCoupletID identical to hash of firstConfirmableStep action link?
 
         // Is nextCoupletID identical to hash of firstConfirmableStep action link?
         var firstConfItemActionLink = jStep.find("td.histStepActions a").get(0);
+
         firstConfItemActionLink = jStep.find("td.histStepActions a").get(0);
         if (firstConfItemActionLink.hash == "#"+nextCoupletID) {  
+
         if (firstConfItemActionLink.hash === "#" + nextCoupletID) {
 
           // Selection in player is identical to current confirmable history action, i.e. does NOT change path.
 
           // Selection in player is identical to current confirmable history action, i.e. does NOT change path.
 
           // Execute confirm and exit jkeyDecision immediately after
 
           // Execute confirm and exit jkeyDecision immediately after
Line 1,190: Line 870:
 
       }
 
       }
 
     } else { // enable history div (disabled at start)
 
     } else { // enable history div (disabled at start)
       jPlayerDiv.find("div.jkeyHistory").show();    
+
       jPlayerDiv.find("div.jkeyHistory").show();
 
       // remove all following if path has changed
 
       // remove all following if path has changed
 
       jCurrHistory.cleanupNestedBlocks();
 
       jCurrHistory.cleanupNestedBlocks();
 
     }
 
     }
 
     // Add new step with simplified lead text to history
 
     // Add new step with simplified lead text to history
     jCurrHistory.createHistoryStep(jSimplifiedLeadTxt.html(),
+
     jCurrHistory.createHistoryStep(
 +
      jSimplifiedLeadTxt.html(),
 
       jPlayerDiv.find("div.certaintyDiv input#decisionUncertain").is(":checked"),
 
       jPlayerDiv.find("div.certaintyDiv input#decisionUncertain").is(":checked"),
       nextCoupletID, $j.data(caller, "coupletID").curr
+
       nextCoupletID,
 +
      $.data(caller, "coupletID").curr
 
     );
 
     );
 
     if (nextCoupletID.length) { // -> NEXT couplet
 
     if (nextCoupletID.length) { // -> NEXT couplet
 
       if (!quiet) {
 
       if (!quiet) {
         // Last parameter false: default (i.e. user can change this later) for NEXT decision is certain  
+
         // Last parameter false: default (i.e. user can change this later) for NEXT decision is certain
 
         jkeyLoadCouplet(jPlayerDiv, nextCoupletID, false);
 
         jkeyLoadCouplet(jPlayerDiv, nextCoupletID, false);
 
       }
 
       }
 
     } else { // -> RESULT
 
     } else { // -> RESULT
 
       // Prepare main result link, common names and resultqualifier (latter 2 may be missing!).
 
       // Prepare main result link, common names and resultqualifier (latter 2 may be missing!).
       var jResult = $j('<span class="leadout"/>')
+
       jResult = $('<span class="leadout"/>')
 
         .append(jContainer.find("span.commonnames").clone().append(" "))
 
         .append(jContainer.find("span.commonnames").clone().append(" "))
 
         .append(jCaller.clone().removeClass("linkbtn").unbind('click'))
 
         .append(jCaller.clone().removeClass("linkbtn").unbind('click'))
Line 1,213: Line 895:
 
       // Add History result row (use .clone(), result already used above)
 
       // Add History result row (use .clone(), result already used above)
 
       // ## TODO: is it possible to avoid redundant span element? Text node?
 
       // ## TODO: is it possible to avoid redundant span element? Text node?
       jCurrHistory.createHistoryResult($j('<span/>').append($j.resource("historyResult")).append(jResult));
+
       jCurrHistory.createHistoryResult($('<span/>').append($.resource("jKey_historyResult")).append(jResult));
 
       if (!quiet) { // load into main result area
 
       if (!quiet) { // load into main result area
 
         jkeyLoadResult(jPlayerDiv, jResult);
 
         jkeyLoadResult(jPlayerDiv, jResult);
Line 1,219: Line 901:
 
     }
 
     }
 
   } catch (err) {
 
   } catch (err) {
     jkeyExceptionAlert(err);
+
     jExceptionAlert(err);
 
   }
 
   }
 
   return false; // cancel default event
 
   return false; // cancel default event
 
}
 
}
  
/*
+
/**
  * Description: Used in onclick event of button "try all decisions"
+
  * @description: Used in onclick event of button "try all decisions"
  * caller: DOM reference to a link
+
  *  
 +
* @requires $.resource()
 +
* @param {selector} caller DOM reference to a link
 +
* @returns {Boolean} False
 
  */
 
  */
 
function jkeyTryAll(caller) {
 
function jkeyTryAll(caller) {
 
   var eachDecisionLink = function (jCurrHistory, jParentBlock, caller, idx) {
 
   var eachDecisionLink = function (jCurrHistory, jParentBlock, caller, idx) {
 
       // create new block for current link & add it to parent block. +idx = unary operator to cast to numeric
 
       // create new block for current link & add it to parent block. +idx = unary operator to cast to numeric
       var jBlock = jCurrHistory.createBlock(false, false, $j.resource("historyNested") + " " + (+idx + 1) + ":");
+
       var jBlock = jCurrHistory.createBlock(false, false, $.resource("jKey_historyNested") + " " + (+idx + 1) + ":");
 
       jCurrHistory.createHistoryNested(jBlock);
 
       jCurrHistory.createHistoryNested(jBlock);
 
       jCurrHistory.setCurrBlock(jBlock);
 
       jCurrHistory.setCurrBlock(jBlock);
       jkeyDecision(caller, true); // true = no couplet/result loading into main window, history only    
+
       jkeyDecision(caller, true); // true = no couplet/result loading into main window, history only
 
       jCurrHistory.setCurrBlock(jParentBlock); // revert current block to parent
 
       jCurrHistory.setCurrBlock(jParentBlock); // revert current block to parent
     };
+
     },
  var jPlayerDiv = $j(caller).closest("div.jkeyPlayer"),
+
    jPlayerDiv = $(caller).closest("div.jkeyPlayer"),
     jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")];
+
     jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")],
 +
    jNestingParent;
 
   // Only if nested steps not already present:
 
   // Only if nested steps not already present:
   if (jCurrHistory.firstNestedStep().length===0) {
+
   if (jCurrHistory.firstNestedStep().length === 0) {
 
     // jkeyTryAll always implies that all later steps must be removed
 
     // jkeyTryAll always implies that all later steps must be removed
 
     jCurrHistory.cleanupConfirmableSteps();
 
     jCurrHistory.cleanupConfirmableSteps();
     var jNestingParent = jCurrHistory.getCurrBlock(); // preserve current for loop
+
     jNestingParent = jCurrHistory.getCurrBlock(); // preserve current for loop
 
     // add all lead-on and lead-out links to history
 
     // add all lead-on and lead-out links to history
 
     jPlayerDiv.find("table.dt-body:first").find("span.leadon a, span.leadout a").each(function (idx) {
 
     jPlayerDiv.find("table.dt-body:first").find("span.leadon a, span.leadout a").each(function (idx) {
Line 1,250: Line 936:
 
   }
 
   }
 
   // activate first Nested history block; mark as active in history, then load couplet or result
 
   // activate first Nested history block; mark as active in history, then load couplet or result
   jCurrHistory.firstNestedStep().find("span.histHeaderActive a").click();  
+
   jCurrHistory.firstNestedStep().find("span.histHeaderActive a").click();
 
   return false; // cancel default event
 
   return false; // cancel default event
 
}
 
}
/*
+
/**
  * Description: switch between history blocks (multiple alternatives if couplet could not be decided)
+
  * @description: switch between history blocks (multiple alternatives if couplet could not be decided)
  * caller: DOM reference to a link
+
  *  
 +
* @param {type} caller DOM reference to a link
 +
* @returns {Boolean}
 
  */
 
  */
 
function jkeySwitchHistory(caller) {
 
function jkeySwitchHistory(caller) {
   var jClosestHistory = $j(caller).closest("table.histBlock, table.histTable");
+
   var jClosestHistory = $(caller).closest("table.histBlock, table.histTable");
 
   // set new & show last entry
 
   // set new & show last entry
 
   jkeyHistory[jClosestHistory.closest("div.jkeyPlayer").attr("id")].changeActiveBlock(jClosestHistory);
 
   jkeyHistory[jClosestHistory.closest("div.jkeyPlayer").attr("id")].changeActiveBlock(jClosestHistory);
Line 1,266: Line 954:
 
}
 
}
  
/*
+
/**
  * Description: Initialize interactive mode (step-by-step) for key if keys exist, init key editor delayed;
+
  * @description Initialize interactive mode (step-by-step) for key if keys exist, init key editor delayed;
  * Also: start player automatically if URL has hash pointing into a valid key
+
  * Also: start player automatically if URL has hash pointing into a valid key.
 +
* Adds event jkey:initialized
 +
*
 +
* @requires $.imglinkBuilder()
 +
* @requires $.resource()
 +
* @returns {undefined}
 
  */
 
  */
 
function jkeyInit() {
 
function jkeyInit() {
   var jKeys = $j("div.decisiontree");
+
   var jKeys = $("div.decisiontree"),
 +
    jKeysWithCtrls,
 +
    fragmentID,
 +
    jKeyAutostartPos,
 +
    isKeyRef;
 +
  // append jKey-specific resources to global jI18n, "true" = deep extension
 +
  $.extend(true, $.jI18n, {
 +
    en: {
 +
      jKey_historyActive : "http://upload.wikimedia.org/wikipedia/commons/thumb/9/94/Symbol_support_vote.svg/20px-Symbol_support_vote.svg.png",
 +
      jKey_historyInactive  : "http://upload.wikimedia.org/wikipedia/commons/thumb/3/37/Symbol_partial_support_vote.svg/20px-Symbol_partial_support_vote.svg.png",
 +
      jKey_historyActiveTooltip : "This is the currently active history; decisions made above will append to this",
 +
      jKey_historyInactiveTooltip  : "Click here to make this alternative the active history",
 +
      jKey_playerStart1st  : "Step-by-step identification",
 +
      jKey_playerStartNew  : "Start new identification",
 +
      jKey_playerOverview  : "Key overview (printable)",
 +
      jKey_playerResume  : "Resume",
 +
      jKey_coupletContinue : "&nbsp;Continue&nbsp;",
 +
      jKey_editorEdit  : "Edit Key",// unused
 +
      jKey_editorSave  : "Save",// unused
 +
      jKey_certaintyLabel  : "Flag this decision as uncertain",
 +
      jKey_certaintyTooltip : "If checkbox is activated, clicking on Next or a result (above) will flag the decision in the list of previous decisions with the word 'uncertain'.",
 +
      jKey_tryAllAlternatives  : "Undecided: Try all alternatives",
 +
      jKey_tryAllAlternativesTooltip : "If not even an uncertain decision is possible, all alternatives may be followed in parallel. After finishing the first alternative, activate the other alternatives in the history of previous decisions (below).",
 +
      jKey_mainResultMsg : "You identified: ",
 +
      jKey_historyHeading  : "Previous decisions",
 +
      jKey_historyConfirmable  : "Confirmable decisions:",
 +
      jKey_historyNested : "Alternative",
 +
      jKey_historyResult : "Result: ",
 +
      jKey_historyConfirm  : "confirm",
 +
      jKey_historyRevise : "revise",
 +
      jKey_historyUncertainFlag  : "(uncertain)",
 +
      jKey_toolTipIsActivePath : "Currently active identification path (multiple alternatives are being followed)" // unused
 +
    },
 +
    fr: {
 +
      jKey_playerStart1st  : "détermination interactive",
 +
      jKey_playerStartNew  : "détermination nouvelle",
 +
      jKey_playerOverview  : "Vue d’ensemble (imprimable)",
 +
      jKey_playerResume  : "continuer la détermination",
 +
      jKey_coupletContinue : "&nbsp;avancer&nbsp;",
 +
      jKey_editorEdit  : "éditer",
 +
      jKey_editorSave  : "enregistrer",
 +
      jKey_certaintyLabel  : "Cette décision a marqué comme incertain",
 +
      jKey_certaintyTooltip : "Si la case à cocher est activée, cliquer sur « avancer » ou un résultat (ci-dessus) marquera la décision dans la liste des décisions précédentes avec le mot « incertain ».",
 +
      jKey_tryAllAlternatives  : "Indécidable: Suivez toutes les alternatives",
 +
      jKey_tryAllAlternativesTooltip : "Si même une décision incertaine est possible, toutes les alternatives peuvent être suivies en parallèle. Après avoir terminé la première alternative, activez les autres alternatives dans l’historique des décisions précédentes (ci-dessous).",
 +
      jKey_mainResultMsg : "résultat: ",
 +
      jKey_historyActiveTooltip : "Alternatif actif. Les décisions prises ci-dessus sont enregistrés ici",
 +
      jKey_historyInactiveTooltip  : "Cliquez ici pour poursuivre cette alternative",
 +
      jKey_historyHeading  : "Décisions antérieures",
 +
      jKey_historyConfirmable  : "Les décisions en cours d’examen:",
 +
      jKey_historyResult : "résultat: ",
 +
      jKey_historyConfirm  : "confirmer",
 +
      jKey_historyRevise : "vérifier",
 +
      jKey_historyUncertainFlag  : "(incertain)",
 +
      jKey_toolTipIsActivePath : "Actuellement la route de détermination actif (plusieurs alternatives sont poursuivies)"
 +
    },
 +
    de: {
 +
      jKey_playerStart1st  : "Interaktive Bestimmung",
 +
      jKey_playerStartNew  : "Neue Bestimmung",
 +
      jKey_playerOverview  : "Übersicht (druckbar)",
 +
      jKey_playerResume  : "Bestimmung fortsetzen",
 +
      jKey_coupletContinue : "&nbsp;Weiter&nbsp;",
 +
      jKey_editorEdit  : "Bearbeiten",
 +
      jKey_editorSave  : "Speichern",
 +
      jKey_certaintyLabel  : "Diese Entscheidung als unsicher kennzeichnen",
 +
      jKey_certaintyTooltip : "Wenn die Checkbox aktiviert ist, wird beim Klick auf 'Weiter' oder ein Ergebnis diese Entscheidung in der Liste bisheriger Entscheidungen mit dem Wort 'unsicher' gekennzeichnet.",
 +
      jKey_tryAllAlternatives  : "Nicht entscheidbar: Verfolge alle Alternativen",
 +
      jKey_tryAllAlternativesTooltip : "Wenn überhaupt keine Entscheidung möglich ist, können die Alternativen parallel verfolgt werden. Nachdem die erste Alternative zu Ende geführt wurde, können die Übrigen in der Liste der bisherigen Entscheidungen (unten) aktiviert werden.",
 +
      jKey_mainResultMsg : "Ergebnis: ",
 +
      jKey_historyActiveTooltip : "Aktive Alternative. Die oben getroffenen Entscheidungen werden hier aufgezeichnet",
 +
      jKey_historyInactiveTooltip  : "Klicken Sie hier, um diese Alternative weiter zu verfolgen",
 +
      jKey_historyHeading  : "Bisherige Entscheidungen",
 +
      jKey_historyConfirmable  : "Entscheidungen in Überprüfung:",
 +
      jKey_historyResult : "Ergebnis: ",
 +
      jKey_historyConfirm  : "bestätigen",
 +
      jKey_historyRevise : "überprüfen",
 +
      jKey_historyUncertainFlag  : "(unsicher)",
 +
      jKey_toolTipIsActivePath : "Derzeit aktiver Bestimmungsweg (mehrere Alternativen werden verfolgt)"
 +
    },
 +
    it: {
 +
      jKey_playerStart1st  : "Esegui passo-dopo-passo",
 +
      jKey_playerStartNew  : "Nuova identificazione",
 +
      jKey_playerOverview  : "Sintesi completa (stampabile)",
 +
      jKey_playerResume  : "Ricomincia l’identificazione",
 +
      jKey_coupletContinue : "&nbsp;Continua&nbsp;",
 +
      jKey_editorEdit  : "Modifica",
 +
      jKey_editorSave  : "Salva",
 +
      jKey_certaintyLabel  : "Segna scelta come insicura", //REVISE
 +
      jKey_certaintyTooltip : "(click, then make your next decision)", //TRANSLATE
 +
      jKey_mainResultMsg : "Il risultato dell'identificazione è: ",
 +
      jKey_historyHeading  : "Scelte precedenti",
 +
      jKey_historyConfirmable  : "Scelta confermabile:",
 +
      jKey_historyResult : "Risultato: ",
 +
      jKey_historyConfirm  : "conferma",
 +
      jKey_historyRevise : "correggi",
 +
      jKey_historyUncertainFlag  : "(incerta)",
 +
      jKey_toolTipIsActivePath : "Percorso di identificazione attualmente attivo (vengono seguite alternative multiple)"
 +
    }
 +
  });
 
   if (jKeys.length) { // only if at least one key exists
 
   if (jKeys.length) { // only if at least one key exists
     $j("head").append("<style type=\"text/css\">" +
+
     $("head").append(
    // modal layer styles
+
      "<style type=\"text/css\">"
    "#jkeymodal-overlay {position:fixed; z-index:100; top:0px; left:0px; height:100%; width:100%; background:#000; opacity:0.8; display:none;}\n" +
+
        // Canvas: top padding needed for IE, FF/Chrome could do without
    "#jkeymodal-layer {position:fixed; z-index:101; top:50%; left:50%; padding:3px; border:3px solid; background-color:#FFFFFF; display:none;}\n" +
+
        + "div.jkeyCanvas {background-color:#FFFFFF; border:5px solid #FFC51A; padding:0.7em 1em 1em 1em; color:rgba(0,0,0,1.0);}\n"
    "#jkeymodal-layer img {display:block;}\n" +
+
        // do dim background around the jkeyCanvas, applied to body, links, etc.
    // /* IE6 hack: IE-CSS-expression is very slow; inserted only for IE < 7.
+
        + ".jkdimmed {color:rgba(0,0,0,0.2) !important;}\n"
    ($j.browser.msie && $j.browser.version < 7 ? "* html #jkeymodal-overlay {position: absolute; height:expression(document.body.scrollHeight > document.body.offsetHeight ? document.body.scrollHeight : document.body.offsetHeight + 'px');}\n* html #jkeymodal-layer {position: absolute; margin-top: expression(0 - parseInt(this.offsetHeight / 2) + (TBWindowMargin = document.documentElement && document.documentElement.scrollTop || document.body.scrollTop) + 'px');}\n" : "") +
+
        // Safari 4 has problems with 100% table width:
    // player styles. Note on Canvas: FF does not need top 1em, but IE does!
+
        + ( ( navigator.userAgent.toUpperCase().indexOf('SAFARI') > 0 && parseFloat(navigator.appVersion) < 5 ) ? "table.dt-caption {width:98%}\n" : "")
    "div.jkeyCanvas {background-color:#FFFFFF; border:5px solid #FFC51A; padding:0.7em 1em 1em 1em}\n" +
+
        + "div.jkeyControls a {vertical-align: middle;}\n"
    // Safari 4 has problems with 100% table width:
+
        + "div.jkeyPlayer table.dt-body {margin:0.5em}\n"
    ($j.browser.safari && $j.browser.version < 5 ? "table.dt-caption {width:98%}\n" : "") +  
+
        + "div.jkeyResultMsg {clear:right; margin:0 0 1em; padding:1em; font-weight:bold; background-color:#EEF0F0; border:1px solid #444444; }\n"
    "div.jkeyLCtrls {margin:0.3em 0 0;padding:0;}\n" +
+
        + "div.jkeyResultMsg span.leadout, div.jkeyResultMsg span.commonnames {background-color:transparent}\n"
    "td.jkeyControls {text-align:right; background-color:transparent; font-weight:bold;}\n" +
+
        + "div.certaintyDiv {margin:1.8em 0 1em 0; padding:0.8em 0.2em; font-size:83%; color:#555;}\n"
    "td.jkeyControls a {vertical-align: middle;}" +
+
        + "input[type=checkbox] {vertical-align:middle;}\n" /* MAKE GENERIC?? */
    "div.jkeyPlayer table.dt-body {margin:0.5em}\n" +
+
        + "div.jkeyHistory {margin-top:1em;}\n"
    "div.jkeyResultMsg {margin:0 0 1em; padding:1em; font-weight:bold; background-color:#EEF0F0; border:1px solid #444444; }\n" +
+
        + "table.histTable, table.histBlock {background-color:transparent;}\n"
    "div.jkeyResultMsg span.leadout, div.jkeyResultMsg span.commonnames {background-color:transparent}\n" +
+
        + "table.histBlock {border-left:1px solid; width:100%;}\n"
    "div.certaintyDiv {margin:2em 0 1em 0; padding:0.6em 0.2em; border-top:1px solid #D0D0D0;}\n" +
+
        + "td.histStepNumber, td.histNestedEmpty, td.histResultSymbol {width:1em; padding:0 0.6em;}\n"
    "input[type=checkbox] {vertical-align:middle;}\n" + // MAKE GENERIC??
+
        + "td.histHeaderContent, td.histConfirmSubhdgContent {padding-left:0.6em; font-weight:bold;}\n"
    "div.jkeyHistory {margin-top:1em;}\n" +
+
        + "td.histStepActions {text-align:center; width:50px; padding-left:1em; }\n"
    "table.histTable, table.histBlock {background-color:transparent;}\n" +
+
        + "td.histStepNumber, td.histStepActions, td.histResultSymbol {vertical-align:top;}\n"
    "table.histBlock {border-left:1px solid; width:100%;}\n" +
+
        + "span.histStepCertainty {background-color:#FFA07A;}\n"
    "td.histStepNumber, td.histNestedEmpty, td.histResultSymbol {width:1em; padding:0 0.6em;}\n" +
+
        + "div.jkeyPlayer table.dt-body td.dt-nodeid, div.jkeyPlayer table.dt-body th.leadtxt, div.jkeyPlayer table.dt-body td.leadtxt, div.jkeyPlayer table.dt-body td.leadresult {padding-top:1em;line-height:1.7em}\n"
    "td.histHeaderContent, td.histConfirmSubhdgContent {padding-left:0.6em;}\n" +
+
        + "</style>"
    "td.histStepActions {text-align:center; width:50px; padding-left:1em; }\n" +
+
     );
    "td.histStepNumber, td.histStepActions, td.histResultSymbol {vertical-align:top;}\n" +
+
     // flag: hide top metadata display (geoscope, creator, etc.)
    "span.histStepCertainty {background-color:#FFA07A;}\n" +
+
     jKeys.filter(".jkey-hidekeymetadata").find("span.collapseButton:first a").click();
    "div.jkeyPlayer table.dt-body td.dt-nodeid, div.jkeyPlayer table.dt-body th.leadtxt, div.jkeyPlayer table.dt-body td.leadresult {padding-top:1em;line-height:1.7em}\n" +
+
    // where span.leadout followed by span.leadon, the two a.linkbtn easily overlap. OK with padding 1.5px!
+
     "a.linkbtn {background-color:#EEF0F0; border:1px solid #444444; padding:1.5px 6px; text-align:center; font-weight:bold; white-space:nowrap;}\n" +
+
     "a.linkbtn:visited, a.linkbtn:hover {color:#444444; text-decoration:none;}\n" +
+
     "a.linkbtn:hover {background-color:#DFDFDF;}\n" +
+
    "@media screen, handheld, projection {*.printonly {display:none;}} @media print {*.noprint {display:none;}}\n" +
+
    "</style>");
+
 
     // flag: show interactive key controls only for keys without class "jkey-nocontrols"
 
     // flag: show interactive key controls only for keys without class "jkey-nocontrols"
     var filtered = jKeys.not(".jkey-nocontrols");
+
     jKeysWithCtrls = jKeys.not(".jkey-nocontrols");
     filtered.find("table.dt-caption tr:first").append($j('<td class="jkeyControls noprint"/>').html("<span class='jkeyPlayerStart1st nowrap'> &nbsp; &nbsp;" + $j.imglinkBuilder("iconStart1st", "playerStart1st", "onclick='return jkeySetMode(\"firststart\",this);'") + "</span>"));
+
     // add player control + always add "show-all-extras" checkbox (toggleAllExtras) in Overview mode (later removed)
    // always add "show-all-extras" checkbox (edit button will also be added later here)
+
    jKeysWithCtrls
    filtered.find("div.dt-header").append("<div class='jkeyLCtrls nowrap' style='font-size:0.8em'> &nbsp; &nbsp;<input type='checkbox' id='toggleAllExtras' value='1' onclick='toggleAllCollapsible(this.checked)' /> <label for='toggleAllExtras'>" + $j.resource("expandAll") + "</label></span>");
+
      .find(".dt-caption")
     // if current URL contains hash (>1 to ignore "#" itself): attempt to start that player (will check for undefined ids)
+
      .prepend('<div class="jkeyControls" style="float:right;text-align:right; background-color:transparent; font-weight:bold; margin-bottom:6px"><span class="jkeyPlayerStart1st nowrap linkbtn" style="padding:4px 6px;">'
    if (document.location.hash.length > 1) {
+
        + $.imglinkBuilder("jKey_iconStart1st", "jKey_playerStart1st", "onclick='return jkeySetMode(\"firststart\",this);'")
      if (jkeyIsValidKeyRef($j(document.location.hash))) {
+
        + '</span><span class="jkeyToggleAllExtras nowrap"><br/><input type="checkbox" id="toggleAllExtras" class="toggleAllExtras" value="1" onclick="$.toggleAllCollapsible(this.checked, this)" onkeyup="$.toggleAllCollapsible(this.checked, this)" />'
        jkeySetMode("firststart", $j(document.location.hash));
+
        + ' <label for="toggleAllExtras" style="font-weight:normal;font-size:83%">' + $.resource("jKey_expandAll")
 +
        + '</label><br/></span><span class="jkeyPlayerOverview nowrap">' +
 +
        $.imglinkBuilder("jKey_iconOverview", "jKey_playerOverview", "onclick='return jkeySetMode(\"overview\",this);'")
 +
        + '<br/></span><span class="jkeyPlayerResume nowrap">'
 +
        + $.imglinkBuilder("jKey_iconResume", "jKey_playerResume", "onclick='return jkeySetMode(\"resumed\",this);'")
 +
        + '<br/></span><span class="jkeyPlayerStartNew nowrap">'
 +
        + $.imglinkBuilder("jKey_iconStartNew", "jKey_playerStartNew", "onclick='return jkeySetMode(\"newstart\",this);'") +
 +
        '</span></div>')
 +
      .find(".jkeyPlayerOverview,.jkeyPlayerResume,.jkeyPlayerStartNew").hide(); // RETEST WHETHER DIRECT HIDE/DISPLAY:NONE NOW POSSIBLE! putting display:none causes layout problems when showing them later!
 +
     // Evaluate URL-hash-based and div-class-based autostart options:
 +
    fragmentID = document.location.hash;
 +
    if (fragmentID.length > 1) { // is non-empty hash (>1 to ignore "#" itself):
 +
      jKeyAutostartPos = fragmentID.indexOf("jkey-autostart");
 +
      if (jKeyAutostartPos > -1) { // test for presence of "jkey-autostart" inside hash
 +
        fragmentID = fragmentID.substring(0, jKeyAutostartPos); // remove jkey-autostart
 +
        if (fragmentID.length === 1) { // just the "#"
 +
          isKeyRef = false;
 +
        } else {
 +
          isKeyRef = jkeyisKeyRef($(fragmentID));
 +
        }
 +
        if (isKeyRef) { // existing ID, start this key (multiple keys may exist)
 +
          jkeySetMode("firststart", fragmentID);
 +
        } else { // else start first key
 +
          jkeySetMode("firststart", jKeys.get(0));
 +
        }
 
       }
 
       }
     } else { // check for flag: jkey-autostart -> automatically change to interactive mode  
+
     } else { // no URL-based autostart -> check for flag: jkey-autostart -> automatically change to interactive mode
       jKeys.filter(".jkey-autostart").each(function() {
+
       jKeys.filter(".jkey-autostart").each(function () {
         jkeySetMode("firststart", "#"+this.id);
+
         jkeySetMode("firststart", "#" + this.id);
 
       });
 
       });
 
     }
 
     }
     if (wgUserName) { // creation of edit button if signed-in
+
     jKeys.trigger('jkey:initialized');
    // load editor (large code!) only on demand
+
    // TODO: editor should be activated only for ff/opera/newest IE
+
      $j("div.decisiontree").find("div.jkeyLCtrls:first").append(" &nbsp; <input class='editbtn' type='button' style='font-size:0.76em;vertical-align:middle;' value='" + $j.resource("editorEdit") + "' onclick='return jkeyInitEditor(this)' />");
+
    }
+
 
+
 
   }
 
   }
}
+
}// jkeyInit()
  
function jkeyInitEditor(caller) {
+
$(document).ready(function () { // Called after html + all js loaded
  $j.getScript(wgServer + wgScript + "?title=MediaWiki:JKeyEditor.js&action=raw&ctype=text/javascript",
+
   jkeyInit();
    function(){jedtInitKey(caller)});
+
  return false;
+
}
+
 
+
/*
+
* Description: Called after html document and all js-sources are loaded
+
*/
+
$j(document).ready(function() {
+
  initCollapseButtons(); // strongly changes page layout: execute first
+
  jkeyInitImageZooming();
+
  initTargetHighlighting(); // page-internal jumps
+
   jkeyInit(); // init player & editor
+
 
});
 
});

Revision as of 23:56, 26 October 2017

/*** TEMP NOTE: current class flags: jkey-hidekeymetadata (TEST), jkey-nocontrols (OK) jkey-autostart (FAIL nested keys) jkey-simplified (NEEDS TESTING) ***/
/**
 * @description jKey.js - Copyright (c) 2009-2012 Stephan Opitz, Andreas Plank & G. Hagedorn, JKI Berlin Dahlem
 * This program is free software; you can redistribute it and/or modify it under the terms of the EUPL v.1.1
 * or (at your option) the GNU General Public License as published by the Free Software Foundation; either
 * GPL v.3 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 General Public License (http://www.gnu.org/licenses/) for more details.
 * @requires: $.imglinkBuilder
 * @requires: $.linkBuilder
 * @requires: $.random
 * @requires: $.toggleAllCollapsible
 * TODO add documentation
 */
/* Settings for JSLint: */
/*jslint sloppy:false, maxerr:100, indent:2 */
/*global $, jQuery, document, console, window, alert, wgPageName, wgServer, wgScript, wgUserName, jedtInit */

// set ECMAScript 5 Strict Mode. Done globally, not within each function. Later the whole jKey should be wrapped as a gadget providing scope for this
"use strict";

////////////////////////
// Exception handling //
////////////////////////

/**
 * @description: Exception constructor, providing toString method
 * @param {string} errorCode  identifying errors
 * @param {object} variables optional for detailed information
 * @returns {jException}
 */
function jException(errorCode, variables) {
  this.errorCode = errorCode;
  this.variables = variables;
}
/**
 * @augments jException()
 * @returns {String}
 */
jException.prototype.toString = function () {
  var message = "JKey Exception " + this.errorCode + "\n\n",
   key;
  for (key in this.variables) {
    // only own properties, not inherited ones:
    if (this.variables.hasOwnProperty(key)) {
      message += " " + key + ": " + this.variables[key] + "\n";
    }
  }
  return message;
};
// also override toString of default Error constructor
Error.prototype.toString = function () {
  return "Javascript exception: " + this.name 
    + "\n\nMessage: " + this.message 
    + "\nFileName: " + this.fileName 
    + "\nLineNumber: " + this.lineNumber;
};
/*
 * @description report exception to console or alert box.
 * exception: jException or JS internal exception
 */
function jExceptionAlert(exception) {
  if (window.console) { // IE dev.tools (=F12), or FF firebug console ENABLED
    console.log(exception); // TODO: in ie watch-console there is no debug fct?!
  } else { // perhaps in FF write to browser-javascript-console: throw new Error("text");
    alert(exception);
  }
}

/////////////////////////
// JKey player history //
/////////////////////////

/**
 * @description: history header CONSTRUCTOR
 * 
 * @returns {HistoryHeader}
 */
function HistoryHeader() {
 /**
  * @description: creates new header for history of previous decisions
  * 
  * @requires $.resource()
  * @param {boolean} isActiveHistory boolean is history set as active
  * @param {boolean} isFirstHistory hides active-history-flag if true
  * @param {string} newContent the new description
  * @returns {unresolved}
  */
  this.create = function (isActiveHistory, isFirstHistory, newContent) {
    // create base header
    this.item = $('<tr class="histHeader"/>')
      .append($('<td class="histHeaderContent" colspan="3"/>').html((newContent ? newContent : $.resource("jKey_historyHeading")) + " &nbsp; ")
        .append($('<span class="histHeaderActive"/>')
          .append($.imglinkBuilder("jKey_historyActive", "", "class='histActiveOn' onclick='return jkeySwitchHistory(this);'"))
          .toggle(!isFirstHistory)
          ));
    this.setCurrBlockActive(isActiveHistory);
    return this.item;
  };

  // item specific fields
  /*
  * Description: get "active" state; return boolean
  */
  this.isActive = function () {
    return this.item.find("span.histHeaderActive a").hasClass("histActiveOn");
  };
  /*
  * Description: set the active history flag
  * isActiveHistory: boolean
  */
  this.setCurrBlockActive = function (isActiveHistory) { // do not use imglinkBuilder, replaceWith kills layout
    var histHeaderActiveCell = this.item.find("span.histHeaderActive a");
    // change class & image
    histHeaderActiveCell.attr({"class": (isActiveHistory ? "histActiveOn" : "histActiveOff")});
    histHeaderActiveCell.find("img")
     .attr("src", $.resource(isActiveHistory ? "jKey_historyActive" : "jKey_historyInactive"))
     .attr("title", $.resource(isActiveHistory ? "jKey_historyActiveTooltip" : "jKey_historyInactiveTooltip"));
  };
}// HistoryHeader()
/*
 * Description: history confirm subheading CONSTRUCTOR
 */
function HistoryConfirmSubheading() {
  /* Description: create new subheading with newContent string */
  this.create = function (newContent) {
    this.item = $("<tr class='histConfirmSubhdg'/>")
      .append($("<td colspan='3' class='histConfirmSubhdgContent'/>").html(newContent));
    return this.item;
  };
}
/*
 * Description: history result CONSTRUCTOR
 */
function HistoryResult() {
  /* Description: create new result item with newContent string */
  this.create = function (newContent) {
    this.item = $("<tr class='histResult'/>")
      .append($("<td class='histResultSymbol'/>").text("►"))
      .append($("<td colspan='2' class='histResultContent'/>").html(newContent));
    return this.item;
  };
}

/*
 * Description: history nested CONSTRUCTOR
 */
function HistoryNested() {
  /*
  * newContent: blocks with alternative paths
  */
  this.create = function (newContent) {
    // create base step
    this.item = $('<tr class="histNested"/>')
      .append($('<td class="histNestedEmpty"/>'))
      .append($('<td class="histNestedContent" colspan="2"/>').html(newContent));
    return this.item;
  };
}
/*
 * Description: history step CONSTRUCTOR
 */
function HistoryStep() {
  /*
  * Description: creates new step
  * newStepNumber: step number
  * newContent: description of previous decision
  * isUncertain: flag whether decision was uncertain
  * confCoupletID: id of confirm-couplet
  * revCoupletID: id of revise-couplet
  */
  this.create = function (newStepNumber, newContent, isUncertain, confCoupletID, revCoupletID) {
    // create base step
    this.item = $('<tr class ="histStep"/>')
      .append($('<td class="histStepNumber"/>').text(newStepNumber + "."))
      .append('<td class="histStepContent"/>')
    // Create revise-history action (changable to confirm).
      .append(
        $('<td class="histStepActions"/>')
        // (side-requirement: href MUST be confCoupletID:)
          .append(
            $.linkBuilder(
              "jKey_historyRevise",
              "",
              "#" + confCoupletID,
              " class='histStepActionRevise small-linkbtn' onclick='return jkeyHistoryAction(this, \"" + revCoupletID + "\", \"" + confCoupletID + "\");'"
            )
          )
      );
    this.setContent(newContent);
    this.setUncertainty(isUncertain);
    return this.item;
  };
  /*
  * Description: set decision certainty of history step
  * isUncertain: true -> display marker text
  */
  this.setUncertainty = function (isUncertain) {
    this.item.find("td.histStepContent span.histStepCertainty")
      .html(isUncertain ? ("&nbsp;" + $.resource("jKey_historyUncertainFlag") + "&nbsp;") : "");
  };
  /*
  * Description: return previously recorded uncertainty for a history step (boolean)
  */
  this.getUncertainty = function () {
    return this.item.find("td.histStepContent span.histStepCertainty").text().length > 0;
  };
  /*
  * set the step content (override inheritance)
  * @param newContent: the new html value
  */
  this.setContent = function (newContent) {
    this.item.find("td.histStepContent")
      .empty()
      .append(newContent)
      .append('<span class="histStepCertainty"/>');
  };
  /*
  * Description: get number in front of step
  */
  this.getStepNumber = function () {
    return parseInt(this.item.find("td.histStepNumber").text(), 10); // parseInt example: " 8." will return 8
  };
  /*
  * Description: set confirm/revise action
  * setToConfirm: the new action state (true = confirm active, false = revise active)
  */
  this.setConfirmable = function (setToConfirm) {
    this.item.find("td.histStepActions a")
      .attr({"class": (setToConfirm ? "histStepActionConfirm small-linkbtn" : "histStepActionRevise small-linkbtn")})
      .text($.resource((setToConfirm ? "jKey_historyConfirm" : "jKey_historyRevise")));
  };
  /*
  * Description: check whether action is confirm or revise
  */
  this.isConfirmable = function () {
    return this.item.find("td.histStepActions a").is(".histStepActionConfirm");
  };
}

// global player history object, one key/history pair for each identification key on the html page
var jkeyHistory = {};
/*
 * Description: player history CONSTRUCTOR
 */
function PlayerHistory(jPlayerDiv) {
  // local variables: item types = histHeader, histStep, histSubkeyHdg, histResult, histBlock;
  // no more lazy loading for header (used immediately!), and historyResult (small)
  var jCurrHistBlock,
    historyHeader = new HistoryHeader(),
    historyConfirmSubheading = new HistoryConfirmSubheading(),
    historyResult = new HistoryResult(),
    historyNested = new HistoryNested(),
    historyStep, // LAZY LOADING
  /*
  * Description: ensure that static class historyStep is loaded (lazy loading, deferring until used)
  */
    histStep_lazyLoad = function () {
      if (historyStep === undefined) {historyStep = new HistoryStep(); }
    };
  /*
  * Description: create new history section in DOM tree, together with header row.
  * active: true = set as active history
  * isFirstHistory: if set hides active-history-flag
  */
  this.createBlock = function (active, isFirstHistory, newContent) {
    return $('<table cellpadding="0" cellspacing="0" class="' + (isFirstHistory ? "histTable" : "histBlock") + '"/>')
      .append('<tr class="histLayout"/><td/><td/><td/></tr>')
      .append(
        historyHeader.create(
          active,
          isFirstHistory,
          "<b>" + (newContent) ? newContent : $.resource("jKey_historyHeading") + "</b>"
        )
      );
  };
  /*
  * Description: add an item to historyTable
  */
  this.addItem = function (item) {
    jCurrHistBlock.append(item);
  };
  /*
  * Description: change active history
  */
  this.changeActiveBlock = function (newActiveBlock) {
    this.setCurrBlockActive(false); // deactivate current block
    this.setCurrBlock(newActiveBlock); // set new
    this.setCurrBlockActive(true); // activate new
  };
  /*
  * Description: get history block
  */
  this.getCurrBlock = function () {
    return jCurrHistBlock;
  };
  /*
  * Description: set history block
  */
  this.setCurrBlock = function (newBlock) {
    jCurrHistBlock = newBlock;
  };
  /*
  * Description: first item step visible confirm link
  * historyTable: history ref
  * return: item object or null if not found
  */
  this.getfirstConfirmableStep = function () {
    var retValue = null,
      jAllSteps = jCurrHistBlock.find("tr:first").nextAll("tr.histStep"),
      parentThis;
    if (jAllSteps.length) { // steps exist
      parentThis = this;
      jAllSteps.each(function () {
        if ((retValue === null) && parentThis.withHistoryStep(this).isConfirmable()) {
          retValue = this;
        }
      });
    }
    return retValue;
  };
  /*
  * Description: change all history action links in history table
  * (revisable before, confirmable starting at firstConfirmableItem)
  * historyTable: history ref
  * firstConfirmableItem: first confirmable item after updating
  */
  this.updateConfirmability = function (firstConfirmableItem) {
    var itemFound = false,
      // get steps within block, not including nested steps
      jAllSteps = jCurrHistBlock.find("tr:first").nextAll("tr.histStep"),
      parentThis;
    if (jAllSteps.length) { // entries exists
      parentThis = this;
      jAllSteps.each(function () {
        if (this === firstConfirmableItem) {
          itemFound = true; // true once item was passed in loop
          // add confirm subheading in front of confirmable steps
          $(this).before(parentThis.createHistoryConfirmSubheading("<i>" + $.resource("jKey_historyConfirmable") + "</i>"));
        }
        // update history actions
        parentThis.withHistoryStep(this).setConfirmable(itemFound);
      }); // TODO: parentThis and itemFound create CLOSUREs - change code?
    }
  };
  /*
  * Description: handle history path changes (cleanup of obsolete steps)
  */
  this.cleanupAfter = function (jStep) {
    // rework history if decision path has changed. Remove confirm-sub-heading, item itself & following siblings
    jStep.prevAll("tr.histConfirmSubhdg").remove();
    jStep.nextAll("tr").andSelf().remove();
  };
  this.cleanupNestedBlocks = function () {
    // find last normal history step (or history header; fallback if no steps exists, e.g. when using try-all as first decision)
    var jStep = jCurrHistBlock.find("tr:first").nextAll("tr.histHeader, tr.histStep").filter(":last");
    if (jStep.length) {
      jStep.nextAll("tr").remove();
    }
  };
  this.cleanupConfirmableSteps = function () {
    var firstConfirmableStep = this.getfirstConfirmableStep();
    if (firstConfirmableStep) {
      this.cleanupAfter($(firstConfirmableStep));
    }
  };

  /*
  * Description: create a history sub-heading, nested block, or result history table row
  * newContent: content for item
  */
  this.createHistoryConfirmSubheading = function (newContent) {
    return historyConfirmSubheading.create(newContent); // this.addItem( because will be added between block items
  };
  this.createHistoryNested = function (newContent) {
    this.addItem(historyNested.create(newContent).get(0));
  };
  this.createHistoryResult = function (newContent) {
    this.addItem(historyResult.create(newContent).get(0));
  };
  /*
  * Description: creates a history step item
  */
  this.createHistoryStep = function (newContent, isUncertain, confCoupletID, revCoupletID) {
    histStep_lazyLoad();
    this.addItem(historyStep.create(this.getNewStepNumber(), newContent, isUncertain, confCoupletID, revCoupletID).get(0));
  };
  /*
  * Description: return history step class initialized with item
  */
  this.withHistoryStep = function (item) {
    histStep_lazyLoad();
    historyStep.item = $(item);
    return historyStep;
  };
  // item specific fields
  /*
  * Description: get "active" state; return boolean
  */
  this.isActive = function () {
    historyHeader.item = jCurrHistBlock.find("tr.histHeader:first");
    return historyHeader.isActive();
  };
  /*
  * Description: set the active history flag
  * isActiveHistory: boolean
  */
  this.setCurrBlockActive = function (newIsActive) {
    historyHeader.item = jCurrHistBlock.find("tr.histHeader:first");
    historyHeader.setCurrBlockActive(newIsActive);
  };
  /*
  * Description: get next available number for a history step in a history block (last + 1).
  * If the block is nested and has no steps yet, outer blocks are taken into account.
  * historyBlock : a jquery-history-block, defaults to jCurrHistBlock if omitted
  */
  this.getNewStepNumber = function (historyBlock) {
    if (!historyBlock) {
      historyBlock = jCurrHistBlock;
    }
    var stepNumber = 0,
      jLastStep = historyBlock.find("tr:first").nextAll("tr.histStep:last");
    if (jLastStep.length) {
      stepNumber = this.withHistoryStep(jLastStep.get(0)).getStepNumber();
    } else if (!historyBlock.is(".histTable")) { // unless outermost and not step (return 0): recurse to parent
      historyBlock = this.getParentBlock();
      return this.getNewStepNumber(historyBlock);
    }
    return stepNumber + 1;
  };
  /*
  * Description: checks whether first step has to be confirmed
  */
  this.isFirstStepInBlockConfirmableStep = function () {
    var firstStep = jCurrHistBlock.find("tr:first").nextAll("tr.histStep:first");
    return (firstStep.prev("tr").hasClass("histConfirmSubhdg"));
  };
  /*
  * Description: retrieve first nested block within current history block
  */
  this.firstNestedStep = function () {
    return jCurrHistBlock.find("tr:first").nextAll("tr.histNested:first");
  };
  /*
  * Description: get parent block of current block (as $ object; if already outermost -> getParentBlock.length=0)
  */
  this.getParentBlock = function () {
    // first closest("table.histBlock, table.histTable") will find own block, then up and find parent:
    var jBlock = jCurrHistBlock.closest("table.histBlock, table.histTable");
    if (!jBlock.is(".histTable")) { // unless already outermost
      jBlock = jBlock.parent().closest("table.histBlock, table.histTable");
    }
    return jBlock;
  };

  // Constructor logic - has to be at end of obj/class definition
  jCurrHistBlock = this.createBlock(true, true); // first and (here in constructor so far) only one
  jPlayerDiv.append(
    $('<div class="jkeyHistory dt-box"/>').hide()
      .append(jCurrHistBlock)
  );
}

/////////////////
// JKey player //
/////////////////

/*
 * Description: is jRefTarget a valid location inside a key?
 * jRefTarget: an idref as jquery object
 * return: boolean
 */
function jkeyisKeyRef(jRefTarget) {
  return (jRefTarget.is("td.dt-nodeid") || jRefTarget.is("tr.dt-row") || jRefTarget.is("div.decisiontree"));
}

/*
 * Description: transform all rows of couplet
 * jPlayerDiv: main div around player (unused?)
 * jDecisionRow: first row of a couplet - must be tested prior to calling this!
 * jPlayerCouplet: position where couplet will appended
 * @todo parentlead does lead to jDecisionRow.length === 0
 */
function jkeyTransformCouplet(jPlayerDiv, jDecisionRow, jPlayerCouplet) {
  if (jDecisionRow.length === 0) {
    throw new jException("NotFound", {
      info: "Transform w/o valid jDecisionRow"
    });
  }
  // row id is like id="Lz_1_row"; we need the id of first td inside: id="Lz_1"
  var currCoupletID = jDecisionRow.children("td[id]:first").attr("id"),
    jNextCouplet,
    eachLeadout = function () {// this = a leadout link
      var nextCoupletID = this.hash,
        isInternalLink = ((nextCoupletID.length > 0) && (this.href.search("/" + wgPageName + "#") !== -1)),
        jKeyTable,
        jNodeID;
      // Example for test above: wgPageName="ThisPage", this.href "http://.../AlsoThisPageTwo#xxx" -> must include / and #
      if (isInternalLink) {
        jNextCouplet = $(nextCoupletID);
        // Check if valid element within a player was found
        isInternalLink = jkeyisKeyRef(jNextCouplet);
        if (isInternalLink) {
          if (jNextCouplet.is("td.dt-nodeid")) { // already is the target node-id
            nextCoupletID = jNextCouplet.attr("id");
          } else { // might be row or div; get key div (closest finds itself), then table, then dt-nodeid
            // jKeyTable is not loop-invariable; a wiki page may have multiple keys!
            jKeyTable = jNextCouplet.closest("div.decisiontree").find("table.dt-body:last");
            jNodeID = jKeyTable.find("td.dt-nodeid:first");
            isInternalLink = (jNodeID.length > 0);
            if (isInternalLink) {
              nextCoupletID = jNodeID.attr("id");
            } // else no leads found in div
          }
        } // else: local id NOT found or NOT valid for player
      } // END if isInternalLink
      // Following is NOT an else, isInternalLink may have been changed.
      nextCoupletID = (!isInternalLink) ? "" : nextCoupletID;
      // prepare resultlink (page or internal subkey) for player
      $(this)
        .attr("target", (isInternalLink ? "_self" : "_blank"))
        .addClass("linkbtn").removeAttr("style")
        .click(function () {return jkeyDecision(this, false); });
      // pass autostart marker on (note: this.hash = this.hash + "jkey-autostart" is ok in FF, but IE8 behaves strangely. Using full href in IE8 seems ok!)
      this.href = this.href + (this.hash.length ? "" : "#") + "jkey-autostart";
      // save in data
      $.data(this, "coupletID", {curr: currCoupletID, next: nextCoupletID});
    },
    eachLeadon = function () { // this = a leadon link
      var jThis = $(this);
      if (jThis.parent().hasClass("leadon")) { // for leadon (but not leadontext) overwrite display text
        jThis.html($.resource("jKey_coupletContinue"));
      }
      jThis.addClass("linkbtn").removeAttr("style");
      // General problem: changing link onclick attribute works in FF, but is ignored by IE (known bug)
      // One general solution is to build complete new link and delete previous one.
      // Also working is use of jquery.click(), but watch referencing "this". Here "this" is correct because of each().
      jThis.click(function () {return jkeyDecision(this, false); });
      // save in data
      $.data(this, "coupletID", {curr: currCoupletID, next: this.hash.substring(1, this.hash.length)});
    },
    //prefixID = function () {return "jK" + this.id; },
    suffixID = function () {
      // leave the mw-customcollapsible untouched
      // if (this.id.indexOf("mw-customcollapsible") === 0) {
      //   return this.id;
      // } else {
      //   return this.id + "jK";
      // }
      return this.id + "jK";
    };
  // Process all leads in couplet
  // Leads may be non-consecutive (general nested order = "1 2 2* 1* 3 3*" or nested subkeys = "1 alpha beta 1*->2 2->3 2* gamma epsilon 3 3*").
  // jquery [attribute^=value] Matches elements that have specified attribute, starting with value.
  // Trailing "_" after currCoupletID because non-consecutive IDs possible ("Lz_1_row", "Lz_1000_row"); Example: 3 leads within couplet = "Lz_1_row"/"Lz_1_2_row"/"Lz_1_3_row"
  // Notes: * Using nextAll().andSelf() would add first lead (jDecisionRow) at the end
  // * Using .prev().nextAll() fails for first couplet of horizontal style, which has no row before!
  jDecisionRow.parent().children("tr.dt-row[id^=" + currCoupletID.replace(/(:|\.)/g,'\\$1') + "_]").each(function () {
    var jClonedRow = $(this).clone(true, true),
      jNodeCell;
    // prefix id of decision row itself and all descendants
    //jClonedRow.find("[id]").andSelf().attr("id", prefixID);
    //check if andSelf neccessary
    //jClonedRow.find("[id]").andSelf().attr("id", suffixID);
    jClonedRow.find("[id]").attr("id", suffixID);
    /**
     * test try default jQuery.makeCollapsible Tue Aug 08 2017 14:25:15 GMT+0200 (CEST)
     * jClonedRow.find('.pseudolink')
     * // remove MediaWiki's collapsible click event
     * .unbind('click.mw-collapse')
     * .attr('class', function (index, attr) {
     * // check for MediaWiki class mw-customtoggle-myKey and add jK suffix
     *   return attr.replace(/(mw-customtoggle-[^ ]+)/, "$1jK");
     * });
     * // remove all MediaWiki's collapsible class added previously by $.fn.makeCollapsible()
     * // leave mw-collapsed untouched!
     * jClonedRow
     * .find('.mw-made-collapsible')
     * .removeClass('mw-made-collapsible');
     */
    // replace lead identifier with arrow (normal) or blank (horizontal side-by-side leads)
    jNodeCell = jClonedRow.find("td.dt-nodeid:first");
    // CHECK why coupletID must be added here in addition to data stored generally for each link
    // couplet ID has to be extracted and stored as data
    $.data(jNodeCell.get(0), "couplet", {id : $.trim(jNodeCell.text())}); // needed for editor & multipleStep startup
    jNodeCell.html((jClonedRow.get(0).className.search(/dt-row-hor\w+/) === -1) ? "►" : " "); //\u25ba is ►
    jClonedRow.find("td.leadalt").empty();
    // Transform all relevant leadout (= result pages) and leadon/leadontext (next couplet) links
    // (depending on row mode, multiple links may exist)
    jClonedRow.find("span.leadout a").each(eachLeadout);
    jClonedRow.find("span.leadon a, div.leadontext a, span.leadontext a").each(eachLeadon);
    // initialize $.fn.makeCollapsible() again
    jPlayerCouplet.append(jClonedRow.show());
    // jPlayerCouplet.find('.mw-collapsible').makeCollapsible();
    var cutsomToggleOptions = {
      toggleClasses: true,
      toggleText: {
        collapseText: $.resource('CollapseBox_captionCollapse'),
        expandText: $.resource('CollapseBox_captionExpand')
      },
      $customTogglers : jPlayerCouplet.find('.mw-customtoggle')
    };
    jPlayerCouplet.find('[id^="mw-customcollapsible-"]').makeCollapsible(cutsomToggleOptions);
  }); // END each lead row of couplet
}

/*
 * Description: Load couplet in player
 * jPlayerDiv: main div around player
 * coupletID: id of the couplet to be loaded
 * isCertain: preset for certainty checkbox (normally true, but may be false when restoring from history)
 */
function jkeyLoadCouplet(jPlayerDiv, coupletID, isUncertain) {
  var jPlayerCouplet = jPlayerDiv.find("table.dt-body");
  jPlayerCouplet.empty(); // flush
  // coupletID could be 'a.34', jquery needs 'a\.34'
  // $("#"... -> document scope, coupletID may be in other key on same page
  jkeyTransformCouplet(jPlayerDiv, $("#" + coupletID.replace(/\./g, "\\.")).closest("tr"), jPlayerCouplet);
  // after history actions: results div may have to be hidden, and certainty div re-displayed
  jPlayerDiv.find("div.jkeyResultMsg").hide();
  jPlayerDiv.find("div.certaintyDiv").show()
    .find("input#decisionUncertain").get(0).checked = isUncertain; // setting checkbox
}

/*
 * Description: toggle visibility of player controls (top right player area)
 * mode: string for current control mode;
 *  values are "resumed", "overview", "newstart", "finished"
 * elementInKeyDiv: any element inside div.decisiontree or div.decisiontree itself, Either as DOM ref OR as "#id" string
 * (.closest() works inclusive!), usually a caller of a control (link, button).
 * returns: jKeyDiv to be further used elsewhere
 */
function jkeySetMode(mode, elementInKeyDiv) {
  try {
    // Cancel if called with undefined element; may occur for "newstart"
    // when called based on undefined id from location.hash
    if (elementInKeyDiv === undefined) {return; }
    var jKeyDiv = $(elementInKeyDiv).closest("div.decisiontree"),
      jKeyTable = jKeyDiv.find("table.dt-body:last"),
      jPlayerDiv = jKeyDiv.find("div.jkeyPlayer"),
      jPlayerCouplet = jPlayerDiv.find("table.dt-body"),
      jResultDiv = jPlayerDiv.find("div.jkeyResultMsg"),
      jControls = jKeyDiv.find(".jkeyControls"),
      finished,
      overview,
      uniquePlayerID;
    if (mode === "firststart") { // change permanently: first-start to restart icon/text, add overview/resume controls, removes  toggleAllExtras!
      jControls.find(".jkeyPlayerStartNew").show();
      jControls.find(".jkeyPlayerStart1st, .jkeyToggleAllExtras").hide();
      window.scrollTo(0, jKeyDiv[0].offsetTop); // scroll to current key
      mode = "newstart";
    }
    finished = (mode === "finished");
    overview = !(mode === "resumed" || mode === "newstart" || finished);
    jControls.find(".jkeyPlayerOverview").toggle(!overview);
    jControls.find(".jkeyPlayerResume").toggle(overview);
    jKeyDiv.find(".dt-header,.dt-header-toggle").toggle(overview); // metadata box (description, audience, etc.) and right-floating "more" to show metadata box
    // Show finished message only in finished mode (+ set below for resumed)
    jResultDiv.toggle(finished);
    // set entire wiki content area to text color css3 rgba with transparency 0.2 (jkeyCanvas is already 1.0).
    // all separately colored text (links etc.) must be handled separately (add all, then remove inside player)
    // NOTE: css opacity not an option, opacity of children can not increase (= made more visible) !
    $("#bodyContent")
     .addClass("jkdimmed")
     .find("a,a.new,a:visited,a:hover,.pseudolink,.linbtn,h2,h3,h4,h5,h6,.commonnames").addClass("jkdimmed");
    jKeyDiv.find(".jkdimmed").removeClass("jkdimmed");
    switch (mode) {
    case ("overview"): // STOP player, hide player, show original overview player, undo low opacity content area
      jKeyDiv.removeClass("jkeyCanvas");
      jKeyTable.show();
      jPlayerDiv.hide();
      $("#bodyContent")
       .removeClass("jkdimmed")
       .find(".jkdimmed").removeClass("jkdimmed");
      break;
    case ("newstart"):
      // Delete player-div in case of restart (also invalidates jPlayerCouplet)
      jPlayerDiv.remove();
      jKeyTable.hide();
      // create new player & table, transform couplet
      jPlayerCouplet = $('<table class="dt-body" cellspacing="0" cellpadding="0"/>');
      jPlayerDiv = $('<div class="jkeyPlayer"/>')
        .append(jPlayerCouplet)
        // Append a hidden result section (for finished mode)
        .append($('<div class="jkeyResultMsg"/>').hide());
      if (!jKeyDiv.hasClass("jkey-simplified")) {
        jPlayerDiv.append(
          '<div class="certaintyDiv noprint">'
            + '<input type="checkbox" id="decisionUncertain" value="1" /> '
            + '<label for="decisionUncertain" style="font-weight:bold" title="'+$.resource("jKey_certaintyTooltip")+'">'
            + $.resource("jKey_certaintyLabel") + "</label>&nbsp;"
            + '&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;|&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; '
            + $.linkBuilder("jKey_tryAllAlternatives","","#"," class='small-linkbtn' onclick='return jkeyTryAll(this);' title='"+$.resource("jKey_tryAllAlternativesTooltip")+"'")
            + '</div>'
        );
      }
      // generate unique playerID (multiple keys may exist on 1 page!):
      do {
        uniquePlayerID = "jkp_" + $.random(1, 9999999);
      } while (jkeyHistory.uniquePlayerID); // until ID not yet used
      // add id to connect with history
      jPlayerDiv.attr("id", uniquePlayerID);
      // initalize player history class (lazy loading)
      jkeyHistory[uniquePlayerID] = new PlayerHistory(jPlayerDiv);
      // Initialize player with first decision row (table may start with spacer row)
      jKeyTable.before(jPlayerDiv); // add to DOM
      jkeyTransformCouplet(jKeyDiv, jKeyTable.find("tr.dt-row:first"), jPlayerCouplet);
      // NOTE: NO BREAK here, fallthrough to "resumed" intended!
    case ("resumed"): // = player resumed. This may have to show couplet or finished mode
      // style change for key div + hide original table & show table in player
      jKeyDiv.addClass("jkeyCanvas");
      jKeyTable.hide();
      jPlayerCouplet.show();
      jPlayerDiv.show();
      // redisplay result div if still filled
      jResultDiv.toggle(jResultDiv.text().length > 0);
      break;
    case ("finished"): // empty current couplet, hide certainty checkbox
      jPlayerCouplet.empty();
      jPlayerDiv.find("div.certaintyDiv").hide();
      break;
    } // end case
    return false; // cancel default event
  } catch (err) {
    jExceptionAlert(err);
  }
}

/*
 * Description: Load result in  player
 * jPlayerDiv: main div around player
 * jResult = $ object containing the combined result html (commonnames, resultlink, qualifier); will be cloned
 */
function jkeyLoadResult(jPlayerDiv, jResult) {
  jPlayerDiv.find("div.jkeyResultMsg").empty().html($.resource("jKey_mainResultMsg")).append(jResult.clone());
  jkeySetMode("finished", jPlayerDiv);
}

/*
 * Description: perform "confirm" or "revise" action from history
 * caller: DOM link; confirm if class=histStepActionConfirm, revise if histStepActionRevise
 * revCoupletID, confCoupletID: id of couplet to be revised or confirmed
 */
function jkeyHistoryAction(caller, revCoupletID, confCoupletID) {
  try {
    // get currently active history (find first & look in hierarchy)
    var jCaller = $(caller),
      jStepItem = jCaller.closest("tr"), // history row around action link
      jPlayerDiv = jStepItem.closest("div.jkeyPlayer"),
      jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")],
      firstConfirmableStep,
      nextDecisionIsUncertain;
    // set history block to the one containing the caller (or outermost histTable itself)
    jCurrHistory.changeActiveBlock(jCaller.closest("table.histBlock, table.histTable"));
    // Handle possible history change
    if (jCaller.hasClass("histStepActionConfirm")) { // confirm was clicked
      revCoupletID = confCoupletID; // revCoupletID = next couplet to load
      // Advance to next history step
      jStepItem = jStepItem.nextAll("tr.histStep:first");
      // Update history step certainty from player checkbox
      firstConfirmableStep = jCurrHistory.getfirstConfirmableStep();
      if (firstConfirmableStep) { // null if none found
        // firstConfirmableStep is the couplet shown in player!
        jCurrHistory.withHistoryStep(firstConfirmableStep)
          .setUncertainty(jPlayerDiv.find("div.certaintyDiv input#decisionUncertain").is(":checked"));
      }
    }
    // Remove confirm "confirmable-decisions" sub-heading; will be recreated in updateConfirmability if necessary
    jCurrHistory.getCurrBlock().find("tr.histConfirmSubhdg").remove();
    if (revCoupletID === "") { // RESULT
      // try-all may result in multiple alternative results. Thus confirming a result needs to refresh rather than re-display
      jkeyLoadResult(jPlayerDiv, jCaller.closest("tr.histStep").next("tr.histResult").find("span.leadout"));
    } else { // Refresh player with couplet corresponding to jStep
      nextDecisionIsUncertain = false; // default
      if (jStepItem.length) { // extract history step certainty
        nextDecisionIsUncertain = jCurrHistory.withHistoryStep(jStepItem.get(0)).getUncertainty();
      }
      jkeyLoadCouplet(jPlayerDiv, revCoupletID, nextDecisionIsUncertain);
    }
    jCurrHistory.updateConfirmability(jStepItem.get(0));
  } catch (err) {
    jExceptionAlert(err);
  }
  return false; // cancel default event
}

/*
 * Description: Get and simplify corresponding leadtxt (without links etc., used for history items)
 * Notes: Templates Lead and Decision Horizontal use class:leadon and caller-link (class leadon) is in td.leadresult
 * > (Decision Horizontal uses both leadresult and leadresult-hor1 as class names)
 * > Template Lead Link (cross-refs) uses class:leadontext,
 * > Decision S2 uses class:leadspan; here caller is in th.leadtxt!
 * leadLinkCaller: DOM reference of the calling element, which has to be a link (e.g: under leadon span)
 */
function jkeySimplifiedLeadtxt(leadLinkCaller) {
  var jContainer = $(leadLinkCaller).closest("td.leadresult, th.leadtxt, td.leadtxt"), // TODO remove th.leadtxt later (Freitag, 23. März 2012 13:42)
    jSpan,
    jSimplifiedLeadTxt;
  if (jContainer.hasClass("leadtxt")) {
    // occurs if leadspan itself is formatted as link (having both leadspan and leadontext class)
    jSpan = jContainer.find("span.leadspan");
  } else if (jContainer.hasClass("leadresult")) {
    // pos of span.leadspan depends on arrangement (horizontal or not)
    if (jContainer.get(0).className.search(/leadresult-hor/) !== -1) {
      // result is separated by a table line.
      jSpan = jContainer.closest("table").find("td.leadtxt:nth-child(" + (jContainer.get(0).cellIndex + 1) + ")").find("span.leadspan");
    } else {
      jSpan = jContainer.closest("table").find("td.leadtxt").find("span.leadspan");
    }
  }
  if (jSpan.length === 0) {
    throw new jException("NotFound", {
      info: "No lead statement!",
      jContainer: jContainer,
      jContainer_closest_table: jContainer.closest("table"),
      leadLinkCaller: leadLinkCaller
    });
  }
  // Clone span; remove links, breaks, imgs
  jSimplifiedLeadTxt = jSpan.clone(true);
  jSimplifiedLeadTxt.find("a").each(function () {$(this).replaceWith($(this).html()); });
  jSimplifiedLeadTxt.find("br, img").remove();
  return jSimplifiedLeadTxt;
}

/**
 * @description: Used in onclick events for both next couplet and final result, i.e. user made a decision.
 *  Add current lead to history, show next couplet in player or finish (result)
 * @param {selector} caller DOM reference of the calling element (has $.data with members: next & curr (couplet ID))
 * @param {boolean} quiet do not output couplets or results, only add to history
 */
function jkeyDecision(caller, quiet) {
  try {
    var jCaller = $(caller),
      jContainer = jCaller.closest("td.leadresult, th.leadtxt, td.leadtxt"), // TODO remove th.leadtxt later (Freitag, 23. März 2012 13:42)
      jPlayerDiv = jContainer.closest("div.jkeyPlayer"),
      jSimplifiedLeadTxt = jkeySimplifiedLeadtxt(caller), // single time this is called
      jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")],
      nextCoupletID = $.data(caller, "coupletID").next, // ref to id attribute of next couplet when used on next couplet link
      firstConfirmableStep,
      jParentBlock,
      jStep,
      firstConfItemActionLink,
      jResult;
    // Decision in player may be identical to current confirmable history step
    firstConfirmableStep = jCurrHistory.getfirstConfirmableStep();
    if (firstConfirmableStep) { // = null if not found
      // is it a nested block and is it first step, which has to be confirmed?
      // -> user decided to change path so that the nested block will be removed
      jParentBlock = jCurrHistory.getParentBlock(); // null if not nested
      // in inner, nested blocks, confirming first step implies removing the try-all structure
      if (!jParentBlock.is(".histTable") && jCurrHistory.isFirstStepInBlockConfirmableStep()) {
        jCurrHistory.setCurrBlock(jParentBlock);
        jCurrHistory.setCurrBlockActive(true);
        jCurrHistory.cleanupNestedBlocks();
      } else { //
        jStep = $(firstConfirmableStep);
        // Is nextCoupletID identical to hash of firstConfirmableStep action link?
        firstConfItemActionLink = jStep.find("td.histStepActions a").get(0);
        if (firstConfItemActionLink.hash === "#" + nextCoupletID) {
          // Selection in player is identical to current confirmable history action, i.e. does NOT change path.
          // Execute confirm and exit jkeyDecision immediately after
          jkeyHistoryAction(firstConfItemActionLink, nextCoupletID, nextCoupletID);
          return false;
        }
        // from here, path has changed; rework history. 1. Remove confirm sub-heading, item itself & following siblings
        jCurrHistory.cleanupAfter(jStep);
        // empty result-div of player (no longer valid)
        jPlayerDiv.children("div.jkeyResultMsg").empty().hide();
      }
    } else { // enable history div (disabled at start)
      jPlayerDiv.find("div.jkeyHistory").show();
      // remove all following if path has changed
      jCurrHistory.cleanupNestedBlocks();
    }
    // Add new step with simplified lead text to history
    jCurrHistory.createHistoryStep(
      jSimplifiedLeadTxt.html(),
      jPlayerDiv.find("div.certaintyDiv input#decisionUncertain").is(":checked"),
      nextCoupletID,
      $.data(caller, "coupletID").curr
    );
    if (nextCoupletID.length) { // -> NEXT couplet
      if (!quiet) {
        // Last parameter false: default (i.e. user can change this later) for NEXT decision is certain
        jkeyLoadCouplet(jPlayerDiv, nextCoupletID, false);
      }
    } else { // -> RESULT
      // Prepare main result link, common names and resultqualifier (latter 2 may be missing!).
      jResult = $('<span class="leadout"/>')
        .append(jContainer.find("span.commonnames").clone().append(" "))
        .append(jCaller.clone().removeClass("linkbtn").unbind('click'))
        .append(jContainer.find("span.resultqualifier").clone().prepend(" "));
      jResult.find("br, img").remove(); // don't combine with above!
      // Add History result row (use .clone(), result already used above)
      // ## TODO: is it possible to avoid redundant span element? Text node?
      jCurrHistory.createHistoryResult($('<span/>').append($.resource("jKey_historyResult")).append(jResult));
      if (!quiet) { // load into main result area
        jkeyLoadResult(jPlayerDiv, jResult);
      }
    }
  } catch (err) {
    jExceptionAlert(err);
  }
  return false; // cancel default event
}

/**
 * @description: Used in onclick event of button "try all decisions"
 * 
 * @requires $.resource()
 * @param {selector} caller DOM reference to a link
 * @returns {Boolean} False
 */
function jkeyTryAll(caller) {
  var eachDecisionLink = function (jCurrHistory, jParentBlock, caller, idx) {
      // create new block for current link & add it to parent block. +idx = unary operator to cast to numeric
      var jBlock = jCurrHistory.createBlock(false, false, $.resource("jKey_historyNested") + " " + (+idx + 1) + ":");
      jCurrHistory.createHistoryNested(jBlock);
      jCurrHistory.setCurrBlock(jBlock);
      jkeyDecision(caller, true); // true = no couplet/result loading into main window, history only
      jCurrHistory.setCurrBlock(jParentBlock); // revert current block to parent
    },
    jPlayerDiv = $(caller).closest("div.jkeyPlayer"),
    jCurrHistory = jkeyHistory[jPlayerDiv.attr("id")],
    jNestingParent;
  // Only if nested steps not already present:
  if (jCurrHistory.firstNestedStep().length === 0) {
    // jkeyTryAll always implies that all later steps must be removed
    jCurrHistory.cleanupConfirmableSteps();
    jNestingParent = jCurrHistory.getCurrBlock(); // preserve current for loop
    // add all lead-on and lead-out links to history
    jPlayerDiv.find("table.dt-body:first").find("span.leadon a, span.leadout a").each(function (idx) {
      eachDecisionLink(jCurrHistory, jNestingParent, this, idx);
    });
  }
  // activate first Nested history block; mark as active in history, then load couplet or result
  jCurrHistory.firstNestedStep().find("span.histHeaderActive a").click();
  return false; // cancel default event
}
/**
 * @description: switch between history blocks (multiple alternatives if couplet could not be decided)
 * 
 * @param {type} caller DOM reference to a link
 * @returns {Boolean}
 */
function jkeySwitchHistory(caller) {
  var jClosestHistory = $(caller).closest("table.histBlock, table.histTable");
  // set new & show last entry
  jkeyHistory[jClosestHistory.closest("div.jkeyPlayer").attr("id")].changeActiveBlock(jClosestHistory);
  // find directly last step (excluding nested), 2x click is: revise, then confirm
  jClosestHistory.find("tr:first").nextAll("tr.histStep:last").find("td.histStepActions a").click().click();
  return false; // cancel default event
}

/**
 * @description Initialize interactive mode (step-by-step) for key if keys exist, init key editor delayed;
 * Also: start player automatically if URL has hash pointing into a valid key.
 * Adds event jkey:initialized
 * 
 * @requires $.imglinkBuilder()
 * @requires $.resource()
 * @returns {undefined}
 */
function jkeyInit() {
  var jKeys = $("div.decisiontree"),
    jKeysWithCtrls,
    fragmentID,
    jKeyAutostartPos,
    isKeyRef;
  // append jKey-specific resources to global jI18n, "true" = deep extension
  $.extend(true, $.jI18n, {
    en: {
      jKey_historyActive : "http://upload.wikimedia.org/wikipedia/commons/thumb/9/94/Symbol_support_vote.svg/20px-Symbol_support_vote.svg.png",
      jKey_historyInactive  : "http://upload.wikimedia.org/wikipedia/commons/thumb/3/37/Symbol_partial_support_vote.svg/20px-Symbol_partial_support_vote.svg.png",
      jKey_historyActiveTooltip : "This is the currently active history; decisions made above will append to this",
      jKey_historyInactiveTooltip  : "Click here to make this alternative the active history",
      jKey_playerStart1st  : "Step-by-step identification",
      jKey_playerStartNew  : "Start new identification",
      jKey_playerOverview  : "Key overview (printable)",
      jKey_playerResume  : "Resume",
      jKey_coupletContinue : "&nbsp;Continue&nbsp;",
      jKey_editorEdit  : "Edit Key",// unused
      jKey_editorSave  : "Save",// unused
      jKey_certaintyLabel  : "Flag this decision as uncertain",
      jKey_certaintyTooltip : "If checkbox is activated, clicking on Next or a result (above) will flag the decision in the list of previous decisions with the word 'uncertain'.",
      jKey_tryAllAlternatives  : "Undecided: Try all alternatives",
      jKey_tryAllAlternativesTooltip : "If not even an uncertain decision is possible, all alternatives may be followed in parallel. After finishing the first alternative, activate the other alternatives in the history of previous decisions (below).",
      jKey_mainResultMsg : "You identified: ",
      jKey_historyHeading  : "Previous decisions",
      jKey_historyConfirmable  : "Confirmable decisions:",
      jKey_historyNested : "Alternative",
      jKey_historyResult : "Result: ",
      jKey_historyConfirm  : "confirm",
      jKey_historyRevise : "revise",
      jKey_historyUncertainFlag  : "(uncertain)",
      jKey_toolTipIsActivePath : "Currently active identification path (multiple alternatives are being followed)" // unused
    },
    fr: {
      jKey_playerStart1st  : "détermination interactive",
      jKey_playerStartNew  : "détermination nouvelle",
      jKey_playerOverview  : "Vue d’ensemble (imprimable)",
      jKey_playerResume  : "continuer la détermination",
      jKey_coupletContinue : "&nbsp;avancer&nbsp;",
      jKey_editorEdit  : "éditer",
      jKey_editorSave  : "enregistrer",
      jKey_certaintyLabel  : "Cette décision a marqué comme incertain",
      jKey_certaintyTooltip : "Si la case à cocher est activée, cliquer sur « avancer » ou un résultat (ci-dessus) marquera la décision dans la liste des décisions précédentes avec le mot « incertain ».",
      jKey_tryAllAlternatives  : "Indécidable: Suivez toutes les alternatives",
      jKey_tryAllAlternativesTooltip : "Si même une décision incertaine est possible, toutes les alternatives peuvent être suivies en parallèle. Après avoir terminé la première alternative, activez les autres alternatives dans l’historique des décisions précédentes (ci-dessous).",
      jKey_mainResultMsg : "résultat: ",
      jKey_historyActiveTooltip : "Alternatif actif. Les décisions prises ci-dessus sont enregistrés ici",
      jKey_historyInactiveTooltip  : "Cliquez ici pour poursuivre cette alternative",
      jKey_historyHeading  : "Décisions antérieures",
      jKey_historyConfirmable  : "Les décisions en cours d’examen:",
      jKey_historyResult : "résultat: ",
      jKey_historyConfirm  : "confirmer",
      jKey_historyRevise : "vérifier",
      jKey_historyUncertainFlag  : "(incertain)",
      jKey_toolTipIsActivePath : "Actuellement la route de détermination actif (plusieurs alternatives sont poursuivies)"
    },
    de: {
      jKey_playerStart1st  : "Interaktive Bestimmung",
      jKey_playerStartNew  : "Neue Bestimmung",
      jKey_playerOverview  : "Übersicht (druckbar)",
      jKey_playerResume  : "Bestimmung fortsetzen",
      jKey_coupletContinue : "&nbsp;Weiter&nbsp;",
      jKey_editorEdit  : "Bearbeiten",
      jKey_editorSave  : "Speichern",
      jKey_certaintyLabel  : "Diese Entscheidung als unsicher kennzeichnen",
      jKey_certaintyTooltip : "Wenn die Checkbox aktiviert ist, wird beim Klick auf 'Weiter' oder ein Ergebnis diese Entscheidung in der Liste bisheriger Entscheidungen mit dem Wort 'unsicher' gekennzeichnet.",
      jKey_tryAllAlternatives  : "Nicht entscheidbar: Verfolge alle Alternativen",
      jKey_tryAllAlternativesTooltip : "Wenn überhaupt keine Entscheidung möglich ist, können die Alternativen parallel verfolgt werden. Nachdem die erste Alternative zu Ende geführt wurde, können die Übrigen in der Liste der bisherigen Entscheidungen (unten) aktiviert werden.",
      jKey_mainResultMsg : "Ergebnis: ",
      jKey_historyActiveTooltip : "Aktive Alternative. Die oben getroffenen Entscheidungen werden hier aufgezeichnet",
      jKey_historyInactiveTooltip  : "Klicken Sie hier, um diese Alternative weiter zu verfolgen",
      jKey_historyHeading  : "Bisherige Entscheidungen",
      jKey_historyConfirmable  : "Entscheidungen in Überprüfung:",
      jKey_historyResult : "Ergebnis: ",
      jKey_historyConfirm  : "bestätigen",
      jKey_historyRevise : "überprüfen",
      jKey_historyUncertainFlag  : "(unsicher)",
      jKey_toolTipIsActivePath : "Derzeit aktiver Bestimmungsweg (mehrere Alternativen werden verfolgt)"
    },
    it: {
      jKey_playerStart1st  : "Esegui passo-dopo-passo",
      jKey_playerStartNew  : "Nuova identificazione",
      jKey_playerOverview  : "Sintesi completa (stampabile)",
      jKey_playerResume  : "Ricomincia l’identificazione",
      jKey_coupletContinue : "&nbsp;Continua&nbsp;",
      jKey_editorEdit  : "Modifica",
      jKey_editorSave  : "Salva",
      jKey_certaintyLabel  : "Segna scelta come insicura", //REVISE
      jKey_certaintyTooltip : "(click, then make your next decision)", //TRANSLATE
      jKey_mainResultMsg : "Il risultato dell'identificazione è: ",
      jKey_historyHeading  : "Scelte precedenti",
      jKey_historyConfirmable  : "Scelta confermabile:",
      jKey_historyResult : "Risultato: ",
      jKey_historyConfirm  : "conferma",
      jKey_historyRevise : "correggi",
      jKey_historyUncertainFlag  : "(incerta)",
      jKey_toolTipIsActivePath : "Percorso di identificazione attualmente attivo (vengono seguite alternative multiple)"
    }
  });
  if (jKeys.length) { // only if at least one key exists
    $("head").append(
      "<style type=\"text/css\">"
        // Canvas: top padding needed for IE, FF/Chrome could do without
        + "div.jkeyCanvas {background-color:#FFFFFF; border:5px solid #FFC51A; padding:0.7em 1em 1em 1em; color:rgba(0,0,0,1.0);}\n"
        // do dim background around the jkeyCanvas, applied to body, links, etc.
        + ".jkdimmed {color:rgba(0,0,0,0.2) !important;}\n"
        // Safari 4 has problems with 100% table width:
        + ( ( navigator.userAgent.toUpperCase().indexOf('SAFARI') > 0 && parseFloat(navigator.appVersion) < 5 ) ? "table.dt-caption {width:98%}\n" : "")
        + "div.jkeyControls a {vertical-align: middle;}\n"
        + "div.jkeyPlayer table.dt-body {margin:0.5em}\n"
        + "div.jkeyResultMsg {clear:right; margin:0 0 1em; padding:1em; font-weight:bold; background-color:#EEF0F0; border:1px solid #444444; }\n"
        + "div.jkeyResultMsg span.leadout, div.jkeyResultMsg span.commonnames {background-color:transparent}\n"
        + "div.certaintyDiv {margin:1.8em 0 1em 0; padding:0.8em 0.2em; font-size:83%; color:#555;}\n"
        + "input[type=checkbox] {vertical-align:middle;}\n" /* MAKE GENERIC?? */
        + "div.jkeyHistory {margin-top:1em;}\n"
        + "table.histTable, table.histBlock {background-color:transparent;}\n"
        + "table.histBlock {border-left:1px solid; width:100%;}\n"
        + "td.histStepNumber, td.histNestedEmpty, td.histResultSymbol {width:1em; padding:0 0.6em;}\n"
        + "td.histHeaderContent, td.histConfirmSubhdgContent {padding-left:0.6em; font-weight:bold;}\n"
        + "td.histStepActions {text-align:center; width:50px; padding-left:1em; }\n"
        + "td.histStepNumber, td.histStepActions, td.histResultSymbol {vertical-align:top;}\n"
        + "span.histStepCertainty {background-color:#FFA07A;}\n"
        + "div.jkeyPlayer table.dt-body td.dt-nodeid, div.jkeyPlayer table.dt-body th.leadtxt, div.jkeyPlayer table.dt-body td.leadtxt, div.jkeyPlayer table.dt-body td.leadresult {padding-top:1em;line-height:1.7em}\n"
        + "</style>"
    );
    // flag: hide top metadata display (geoscope, creator, etc.)
    jKeys.filter(".jkey-hidekeymetadata").find("span.collapseButton:first a").click();
    // flag: show interactive key controls only for keys without class "jkey-nocontrols"
    jKeysWithCtrls = jKeys.not(".jkey-nocontrols");
    // add player control + always add "show-all-extras" checkbox (toggleAllExtras) in Overview mode (later removed)
    jKeysWithCtrls
      .find(".dt-caption")
      .prepend('<div class="jkeyControls" style="float:right;text-align:right; background-color:transparent; font-weight:bold; margin-bottom:6px"><span class="jkeyPlayerStart1st nowrap linkbtn" style="padding:4px 6px;">'
        + $.imglinkBuilder("jKey_iconStart1st", "jKey_playerStart1st", "onclick='return jkeySetMode(\"firststart\",this);'")
        + '</span><span class="jkeyToggleAllExtras nowrap"><br/><input type="checkbox" id="toggleAllExtras" class="toggleAllExtras" value="1" onclick="$.toggleAllCollapsible(this.checked, this)" onkeyup="$.toggleAllCollapsible(this.checked, this)" />'
        + ' <label for="toggleAllExtras" style="font-weight:normal;font-size:83%">' + $.resource("jKey_expandAll")
        + '</label><br/></span><span class="jkeyPlayerOverview nowrap">' +
        $.imglinkBuilder("jKey_iconOverview", "jKey_playerOverview", "onclick='return jkeySetMode(\"overview\",this);'")
        + '<br/></span><span class="jkeyPlayerResume nowrap">'
        + $.imglinkBuilder("jKey_iconResume", "jKey_playerResume", "onclick='return jkeySetMode(\"resumed\",this);'")
        + '<br/></span><span class="jkeyPlayerStartNew nowrap">'
        + $.imglinkBuilder("jKey_iconStartNew", "jKey_playerStartNew", "onclick='return jkeySetMode(\"newstart\",this);'") +
        '</span></div>')
      .find(".jkeyPlayerOverview,.jkeyPlayerResume,.jkeyPlayerStartNew").hide(); // RETEST WHETHER DIRECT HIDE/DISPLAY:NONE NOW POSSIBLE! putting display:none causes layout problems when showing them later!
    // Evaluate URL-hash-based and div-class-based autostart options:
    fragmentID = document.location.hash;
    if (fragmentID.length > 1) { // is non-empty hash (>1 to ignore "#" itself):
      jKeyAutostartPos = fragmentID.indexOf("jkey-autostart");
      if (jKeyAutostartPos > -1) { // test for presence of "jkey-autostart" inside hash
        fragmentID = fragmentID.substring(0, jKeyAutostartPos); // remove jkey-autostart
        if (fragmentID.length === 1) { // just the "#"
          isKeyRef = false;
        } else {
          isKeyRef = jkeyisKeyRef($(fragmentID));
        }
        if (isKeyRef) { // existing ID, start this key (multiple keys may exist)
          jkeySetMode("firststart", fragmentID);
        } else { // else start first key
          jkeySetMode("firststart", jKeys.get(0));
        }
      }
    } else { // no URL-based autostart -> check for flag: jkey-autostart -> automatically change to interactive mode
      jKeys.filter(".jkey-autostart").each(function () {
        jkeySetMode("firststart", "#" + this.id);
      });
    }
    jKeys.trigger('jkey:initialized');
  }
}// jkeyInit()

$(document).ready(function () { // Called after html + all js loaded
  jkeyInit();
});