Difference between revisions of "MediaWiki:Gadget-popups.js"

Jump to navigation Jump to search
(Created page with "// STARTFILE: main.js // ********************************************************************** // ** ** // **...")
 
Line 14: Line 14:
 
// **                                                                  **
 
// **                                                                  **
 
// **********************************************************************
 
// **********************************************************************
/* eslint-env browser  */
 
/* global $, jQuery, mw, window */
 
  
// Fix later
 
/* global log, errlog, popupStrings, wikEdUseWikEd, WikEdUpdateFrame */
 
/* eslint no-mixed-spaces-and-tabs: 0, no-empty: 0 */
 
 
$(function () {
 
 
//////////////////////////////////////////////////
 
//////////////////////////////////////////////////
 
// Globals
 
// Globals
Line 27: Line 20:
  
 
// Trying to shove as many of these as possible into the pg (popup globals) object
 
// Trying to shove as many of these as possible into the pg (popup globals) object
var pg = {
+
function pg(){} // dummy to stop errors
 +
window.pg = {
 
re: {},              // regexps
 
re: {},              // regexps
 
ns: {},              // namespaces
 
ns: {},              // namespaces
 
string: {},          // translatable strings
 
string: {},          // translatable strings
 
wiki: {},            // local site info
 
wiki: {},            // local site info
user: {},            // current user info
 
 
misc: {},            // YUCK PHOOEY
 
misc: {},            // YUCK PHOOEY
 
option: {},          // options, see newOption etc
 
option: {},          // options, see newOption etc
Line 42: Line 35:
 
counter: {},          // .. and all sorts of counters
 
counter: {},          // .. and all sorts of counters
 
current: {},          // state info
 
current: {},          // state info
fn: {},              // functions
 
 
endoflist: null
 
endoflist: null
 
};
 
};
/* Bail if the gadget/script is being loaded twice */
+
window.pop = {               // wrap various functions in here
if( window.pg ) {
+
init: {},
return;
+
util: {},
 +
endoflist: null
 +
};
 +
function popupsReady() {
 +
if (!window.pg) { return false; }
 +
if (!pg.flag) { return false; }
 +
if (!pg.flag.finishedLoading) { return false; }
 +
return true;
 
}
 
}
/* Export to global context */
 
window.pg = pg;
 
  
 
/// Local Variables: ///
 
/// Local Variables: ///
Line 62: Line 59:
 
//<NOLITE>
 
//<NOLITE>
 
// the main initial call
 
// the main initial call
if (getValueOf('popupOnEditSelection') && document && document.editform && document.editform.wpTextbox1) {
+
if (getValueOf('popupOnEditSelection') && window.doSelectionPopup && document && document.editform && document.editform.wpTextbox1) {
 
document.editform.wpTextbox1.onmouseup=doSelectionPopup;
 
document.editform.wpTextbox1.onmouseup=doSelectionPopup;
 
}
 
}
Line 90: Line 87:
 
log(simplePrintf('setupTooltipsLoop(%s,%s,%s,%s,%s)', arguments));
 
log(simplePrintf('setupTooltipsLoop(%s,%s,%s,%s,%s)', arguments));
 
var finish=begin+howmany;
 
var finish=begin+howmany;
var loopend = Math.min(finish, anchors.length);
+
var loopend = min(finish, anchors.length);
 
var j=loopend - begin;
 
var j=loopend - begin;
 
log ('setupTooltips: anchors.length=' + anchors.length + ', begin=' + begin +
 
log ('setupTooltips: anchors.length=' + anchors.length + ', begin=' + begin +
Line 123: Line 120:
 
var tocLinks=toc.getElementsByTagName('A');
 
var tocLinks=toc.getElementsByTagName('A');
 
var tocLen = tocLinks.length;
 
var tocLen = tocLinks.length;
for (var j=0; j<tocLen; ++j) {
+
for (j=0; j<tocLen; ++j) {
 
removeTooltip(tocLinks[j], true);
 
removeTooltip(tocLinks[j], true);
 
}
 
}
Line 165: Line 162:
 
np.mainDiv.style.maxWidth = popupMaxWidth + 'px';
 
np.mainDiv.style.maxWidth = popupMaxWidth + 'px';
 
np.maxWidth = popupMaxWidth;
 
np.maxWidth = popupMaxWidth;
 +
 +
try {
 +
// hack for IE
 +
// see http://www.svendtofte.com/code/max_width_in_ie/
 +
// use setExpression as documented here on msdn: http://tinyurl dot com/dqljn
 +
 +
if (np.mainDiv.style.setExpression) {
 +
np.mainDiv.style.setExpression(
 +
'width', 'document.body.clientWidth > ' +
 +
popupMaxWidth + ' ? "' +popupMaxWidth + 'px": "auto"');
 +
}
 +
}
 +
catch (errors) {
 +
errlog( "Running on IE8 are we not?: " + errors );
 +
}
 
};
 
};
 
np.addHook(setMaxWidth, 'unhide', 'before');
 
np.addHook(setMaxWidth, 'unhide', 'before');
 
}
 
}
 
//<NOLITE>
 
//<NOLITE>
np.addHook(addPopupShortcuts, 'unhide', 'after');
+
if (window.addPopupShortcuts && window.rmPopupShortcuts) {
np.addHook(rmPopupShortcuts, 'hide', 'before');
+
np.addHook(addPopupShortcuts, 'unhide', 'after');
 +
np.addHook(rmPopupShortcuts, 'hide', 'before');
 +
}
 
//</NOLITE>
 
//</NOLITE>
 
}
 
}
  
function removeModifierKeyHandler(a) {
 
//remove listeners for modifier key if any that were added in mouseOverWikiLink
 
document.removeEventListener('keydown', a.modifierKeyHandler, false);
 
document.removeEventListener('keyup', a.modifierKeyHandler, false);
 
}
 
  
 
function mouseOverWikiLink(evt) {
 
function mouseOverWikiLink(evt) {
 +
if (!window.popupsReady || !window.popupsReady()) { return; }
 
if (!evt && window.event) {evt=window.event;}
 
if (!evt && window.event) {evt=window.event;}
 
// if the modifier is needed, listen for it,
 
// we will remove the listener when we mouseout of this link or kill popup.
 
if (getValueOf('popupModifier')) {
 
// if popupModifierAction = enable, we should popup when the modifier is pressed
 
// if popupModifierAction = disable, we should popup unless the modifier is pressed
 
    var action = getValueOf('popupModifierAction');
 
    var key = action=='disable' ? 'keyup' : 'keydown';
 
    var a = this;
 
    a.modifierKeyHandler = function(evt) {
 
mouseOverWikiLink2(a, evt);
 
};
 
    document.addEventListener(key, a.modifierKeyHandler, false);
 
}
 
 
 
return mouseOverWikiLink2(this, evt);
 
return mouseOverWikiLink2(this, evt);
 
}
 
}
  
/**
 
* Gets the references list item that the provided footnote link targets. This
 
* is typically a li element within the ol.references element inside the reflist.
 
* @param {Element} a - A footnote link.
 
* @returns {Element|boolean} The targeted element, or false if one can't be found.
 
*/
 
 
function footnoteTarget(a) {
 
function footnoteTarget(a) {
 
var aTitle=Title.fromAnchor(a);
 
var aTitle=Title.fromAnchor(a);
Line 227: Line 216:
  
 
function footnotePreview(x, navpop) {
 
function footnotePreview(x, navpop) {
setPopupHTML('<hr />' + x.innerHTML, 'popupPreview', navpop.idNumber);
+
setPopupHTML('<hr />' + x.innerHTML, 'popupPreview', navpop.idNumber, getValueOf('popupSubpopups')
 +
? function() { setupTooltips(document.getElementById('popupPreview' + navpop.idNumber)); }
 +
: null);
 +
}
 +
 
 +
// var modid=0;
 +
// if(!window.opera) { window.opera={postError: console.log}; }
 +
 
 +
function modifierKeyHandler(a) {
 +
return function(evt) {
 +
// opera.postError('modifierKeyHandler called' + (++modid));
 +
// opera.postError(''+evt + modid);
 +
// for (var i in evt) {
 +
// opera.postError('' + modid + ' ' + i + ' ' + evt[i]);
 +
// }
 +
// opera.postError(''+evt.ctrlKey + modid);
 +
var mod=getValueOf('popupModifier');
 +
if (!mod) { return true; }
 +
 
 +
if (!evt && window.event) {evt=window.event;}
 +
// opera.postError('And now....'+modid);
 +
// opera.postError(''+evt+modid);
 +
// opera.postError(''+evt.ctrlKey+modid);
 +
 
 +
var modPressed = modifierPressed(evt);
 +
var action = getValueOf('popupModifierAction');
 +
 
 +
// FIXME: probable bug - modifierPressed should be modPressed below?
 +
if ( action === 'disable' && modifierPressed ) { return true; }
 +
if ( action === 'enable' && !modifierPressed ) { return true; }
 +
 
 +
mouseOverWikiLink2(a, evt);
 +
};
 
}
 
}
  
Line 235: Line 256:
  
 
if (!evt && window.event) {evt=window.event;}
 
if (!evt && window.event) {evt=window.event;}
 +
// opera.postError('And now....'+modid);
 +
// opera.postError(''+evt+modid);
 +
// opera.postError(''+evt.ctrlKey+modid);
  
 
return ( evt && mod && evt[mod.toLowerCase() + 'Key'] );
 
return ( evt && mod && evt[mod.toLowerCase() + 'Key'] );
 +
 
}
 
}
  
// Checks if the correct modifier pressed/unpressed if needed
+
function dealWithModifier(a,evt) {
function isCorrectModifier(a,evt) {
+
if (!getValueOf('popupModifier')) { return false; }
if (!getValueOf('popupModifier')) { return true; }
 
// if popupModifierAction = enable, we should popup when the modifier is pressed
 
// if popupModifierAction = disable, we should popup unless the modifier is pressed
 
 
var action = getValueOf('popupModifierAction');
 
var action = getValueOf('popupModifierAction');
return ( action ==  'enable' && modifierPressed(evt) ||
+
if ( action ==  'enable' && !modifierPressed(evt) ||
        action == 'disable' && !modifierPressed(evt) );
+
    action == 'disable' && modifierPressed(evt) ) {
 +
// if the modifier is needed and not pressed, listen for it until
 +
// we mouseout of this link.
 +
restoreTitle(a);
 +
var addHandler='addEventListener';
 +
var rmHandler='removeEventListener';
 +
var on='';
 +
if (!document.addEventListener) {
 +
addHandler='attachEvent';
 +
rmHandler='detachEvent';
 +
on='on';
 +
}
 +
if (!document[addHandler]) { // forget it
 +
return;
 +
}
 +
 
 +
a.modifierKeyHandler=modifierKeyHandler(a);
 +
 
 +
switch (action) {
 +
case 'enable':
 +
document[addHandler](on+'keydown', a.modifierKeyHandler, false);
 +
a[addHandler](on+'mouseout', function() {
 +
document[rmHandler](on+'keydown',
 +
a.modifierKeyHandler, false);
 +
}, true);
 +
break;
 +
case 'disable':
 +
document[addHandler](on+'keyup', a.modifierKeyHandler, false);
 +
}
 +
 
 +
return true;
 +
}
 +
return false;
 
}
 
}
  
 
function mouseOverWikiLink2(a, evt) {
 
function mouseOverWikiLink2(a, evt) {
if (!isCorrectModifier(a,evt)) { return; }
+
if (dealWithModifier(a,evt)) { return; }
 
if ( getValueOf('removeTitles') ) { removeTitle(a); }
 
if ( getValueOf('removeTitles') ) { removeTitle(a); }
 
if ( a==pg.current.link && a.navpopup && a.navpopup.isVisible() ) { return; }
 
if ( a==pg.current.link && a.navpopup && a.navpopup.isVisible() ) { return; }
 
pg.current.link=a;
 
pg.current.link=a;
  
if (getValueOf('simplePopups') && !pg.option.popupStructure) {
+
if (getValueOf('simplePopups') && pg.option.popupStructure===null) {
 
// reset *default value* of popupStructure
 
// reset *default value* of popupStructure
 
setDefault('popupStructure', 'original');
 
setDefault('popupStructure', 'original');
Line 265: Line 319:
  
 
if (!a.navpopup) {
 
if (!a.navpopup) {
a.navpopup=newNavpopup(a, article);
+
// FIXME: this doesn't behave well if you mouse out of a popup
pg.current.linksHash[a.href] = a.navpopup;
+
// directly into a link with the same href
pg.current.links.push(a);
+
if (pg.current.linksHash[a.href] && false) {
 +
a.navpopup = pg.current.linksHash[a.href];
 +
}
 +
else {
 +
a.navpopup=newNavpopup(a, article);
 +
pg.current.linksHash[a.href] = a.navpopup;
 +
pg.current.links.push(a);
 +
}
 
}
 
}
if (a.navpopup.pending === null || a.navpopup.pending !== 0) {
+
if (a.navpopup.pending===null || a.navpopup.pending!==0) {
 
// either fresh popups or those with unfinshed business are redone from scratch
 
// either fresh popups or those with unfinshed business are redone from scratch
 
simplePopupContent(a, article);
 
simplePopupContent(a, article);
 
}
 
}
 
a.navpopup.showSoonIfStable(a.navpopup.delay);
 
a.navpopup.showSoonIfStable(a.navpopup.delay);
 +
 +
getValueOf('popupInitialWidth');
  
 
clearInterval(pg.timer.checkPopupPosition);
 
clearInterval(pg.timer.checkPopupPosition);
Line 287: Line 350:
 
s['on' + getValueOf('popupPreviewButtonEvent')] = function() {
 
s['on' + getValueOf('popupPreviewButtonEvent')] = function() {
 
a.simpleNoMore=true;
 
a.simpleNoMore=true;
d.style.display = "none";
 
 
nonsimplePopupContent(a,article);
 
nonsimplePopupContent(a,article);
 
};
 
};
Line 293: Line 355:
 
setPopupHTML(d, 'popupPreview', a.navpopup.idNumber);
 
setPopupHTML(d, 'popupPreview', a.navpopup.idNumber);
 
}
 
}
 +
return;
 
}
 
}
  
if (a.navpopup.pending !== 0 ) {
+
if (a.navpopup.pending!==0 ) {
 
nonsimplePopupContent(a, article);
 
nonsimplePopupContent(a, article);
 
}
 
}
 
}
 
}
  
// simplePopupContent: the content that do not require additional download
+
// simplePopupContent: the content that is shown even when simplePopups is true
// (it is shown even when simplePopups is true)
 
 
function simplePopupContent(a, article) {
 
function simplePopupContent(a, article) {
 
/* FIXME hack */ a.navpopup.hasPopupMenu=false;
 
/* FIXME hack */ a.navpopup.hasPopupMenu=false;
Line 343: Line 405:
 
}
 
}
  
// Should we show nonsimple context?
 
// If simplePopups is set to true, then we do not show nonsimple context,
 
// but if a bottom "show preview" was clicked we do show nonsimple context
 
function shouldShowNonSimple(a) {
 
  return !getValueOf('simplePopups') || a.simpleNoMore;
 
}
 
 
// Should we show nonsimple context govern by the option (e.g. popupUserInfo)?
 
// If the user explicitly asked for nonsimple context by setting the option to true,
 
// then we show it even in nonsimple mode.
 
function shouldShow(a,option) {
 
if (shouldShowNonSimple(a)) {
 
return getValueOf(option);
 
} else {
 
return (typeof window[option] != 'undefined' )  && window[option];
 
}
 
}
 
  
 
function nonsimplePopupContent(a, article) {
 
function nonsimplePopupContent(a, article) {
Line 366: Line 411:
 
var oldid=(typeof params.oldid=='undefined' ? null : params.oldid);
 
var oldid=(typeof params.oldid=='undefined' ? null : params.oldid);
 
//<NOLITE>
 
//<NOLITE>
if(shouldShow(a,'popupPreviewDiffs')) {
+
if(getValueOf('popupPreviewDiffs') && window.loadDiff) {
 
diff=params.diff;
 
diff=params.diff;
 
}
 
}
if(shouldShow(a,'popupPreviewHistory')) {
+
if(getValueOf('popupPreviewHistory')) {
 
history=(params.action=='history');
 
history=(params.action=='history');
 
}
 
}
 
//</NOLITE>
 
//</NOLITE>
 
a.navpopup.pending=0;
 
a.navpopup.pending=0;
var referenceElement = footnoteTarget(a);
+
var x;
if (referenceElement) {
+
footnotePreview(referenceElement, a.navpopup);
+
if (x=footnoteTarget(a)) {
 +
footnotePreview(x, a.navpopup);
 
//<NOLITE>
 
//<NOLITE>
 
} else if ( diff || diff === 0 ) {
 
} else if ( diff || diff === 0 ) {
Line 382: Line 428:
 
} else if ( history ) {
 
} else if ( history ) {
 
loadAPIPreview('history', article, a.navpopup);
 
loadAPIPreview('history', article, a.navpopup);
} else if ( shouldShowNonSimple(a) && pg.re.contribs.test(a.href) ) {
+
} else if ( pg.re.contribs.test(a.href) ) {
 
loadAPIPreview('contribs', article, a.navpopup);
 
loadAPIPreview('contribs', article, a.navpopup);
} else if ( shouldShowNonSimple(a) && pg.re.backlinks.test(a.href) ) {
+
} else if ( pg.re.backlinks.test(a.href) ) {
 
loadAPIPreview('backlinks', article, a.navpopup);
 
loadAPIPreview('backlinks', article, a.navpopup);
 
} else if ( // FIXME should be able to get all preview combinations with options
 
} else if ( // FIXME should be able to get all preview combinations with options
 
article.namespaceId()==pg.nsImageId &&
 
article.namespaceId()==pg.nsImageId &&
( shouldShow(a,'imagePopupsForImages') || ! anchorContainsImage(a) )
+
( getValueOf('imagePopupsForImages') || ! anchorContainsImage(a) )
 
) {
 
) {
 
loadAPIPreview('imagepagepreview', article, a.navpopup);
 
loadAPIPreview('imagepagepreview', article, a.navpopup);
Line 395: Line 441:
 
} else {
 
} else {
 
if (article.namespaceId() == pg.nsCategoryId &&
 
if (article.namespaceId() == pg.nsCategoryId &&
shouldShow(a,'popupCategoryMembers')) {
+
getValueOf('popupCategoryMembers')) {
 
loadAPIPreview('category', article, a.navpopup);
 
loadAPIPreview('category', article, a.navpopup);
 
} else if ((article.namespaceId() == pg.nsUserId || article.namespaceId() == pg.nsUsertalkId) &&
 
} else if ((article.namespaceId() == pg.nsUserId || article.namespaceId() == pg.nsUsertalkId) &&
shouldShow(a,'popupUserInfo')) {
+
getValueOf('popupUserInfo')) {
 
loadAPIPreview('userinfo', article, a.navpopup);
 
loadAPIPreview('userinfo', article, a.navpopup);
 
}
 
}
if (shouldShowNonSimple(a)) startArticlePreview(article, oldid, a.navpopup);
+
startArticlePreview(article, oldid, a.navpopup);
 
}
 
}
 
}
 
}
  
 
function pendingNavpopTask(navpop) {
 
function pendingNavpopTask(navpop) {
if (navpop && navpop.pending === null) { navpop.pending=0; }
+
if (navpop && navpop.pending===null) { navpop.pending=0; }
 
++navpop.pending;
 
++navpop.pending;
 
debugData(navpop);
 
debugData(navpop);
Line 422: Line 468:
  
 
function loadPreview(article, oldid, navpop) {
 
function loadPreview(article, oldid, navpop) {
 +
pendingNavpopTask(navpop);
 
if (!navpop.redir) { navpop.originalArticle=article; }
 
if (!navpop.redir) { navpop.originalArticle=article; }
article.oldid = oldid;
+
if (!navpop.visible && getValueOf('popupLazyDownloads')) {
loadAPIPreview('revision', article, navpop);
+
var id=(navpop.redir) ? 'DOWNLOAD_PREVIEW_REDIR_HOOK' : 'DOWNLOAD_PREVIEW_HOOK';
 +
navpop.addHook(function() {
 +
getWiki(article, insertPreview, oldid, navpop);
 +
return true; }, 'unhide', 'before', id);
 +
} else {
 +
getWiki(article, insertPreview, oldid, navpop);
 +
}
 
}
 
}
  
Line 433: Line 486:
 
// mediawiki does overwrite; eg [[User:Lupin/foo3#Done]]
 
// mediawiki does overwrite; eg [[User:Lupin/foo3#Done]]
 
if ( navpop.article.anchor ) { target.anchor = navpop.article.anchor; }
 
if ( navpop.article.anchor ) { target.anchor = navpop.article.anchor; }
 +
var trailingRubbish=redirMatch[4];
 
navpop.redir++;
 
navpop.redir++;
 
navpop.redirTarget=target;
 
navpop.redirTarget=target;
 
//<NOLITE>
 
//<NOLITE>
var warnRedir = redirLink(target, navpop.article);
+
if (window.redirLink) {
setPopupHTML(warnRedir, 'popupWarnRedir', navpop.idNumber);
+
var warnRedir = redirLink(target, navpop.article);
 +
setPopupHTML(warnRedir, 'popupWarnRedir', navpop.idNumber);
 +
}
 
//</NOLITE>
 
//</NOLITE>
 
navpop.article=target;
 
navpop.article=target;
Line 448: Line 504:
  
 
var redirMatch = pg.re.redirect.exec(download.data);
 
var redirMatch = pg.re.redirect.exec(download.data);
if (download.owner.redir === 0 && redirMatch) {
+
if (download.owner.redir===0 && redirMatch) {
 +
completedNavpopTask(download.owner);
 
loadPreviewFromRedir(redirMatch, download.owner);
 
loadPreviewFromRedir(redirMatch, download.owner);
 
return;
 
return;
Line 466: Line 523:
 
var wikiText=download.data;
 
var wikiText=download.data;
 
var navpop=download.owner;
 
var navpop=download.owner;
 +
completedNavpopTask(navpop);
 
var art=navpop.redirTarget || navpop.originalArticle;
 
var art=navpop.redirTarget || navpop.originalArticle;
  
 
//<NOLITE>
 
//<NOLITE>
 
makeFixDabs(wikiText, navpop);
 
makeFixDabs(wikiText, navpop);
if (getValueOf('popupSummaryData')) {
+
if (getValueOf('popupSummaryData') && window.getPageInfo) {
getPageInfo(wikiText, download);
+
var info=getPageInfo(wikiText, download);
 
setPopupTrailer(getPageInfo(wikiText, download), navpop.idNumber);
 
setPopupTrailer(getPageInfo(wikiText, download), navpop.idNumber);
 
}
 
}
Line 511: Line 569:
 
function anchorize(d, anch) {
 
function anchorize(d, anch) {
 
if (!anch) { return d; }
 
if (!anch) { return d; }
var anchRe=RegExp('(?:=+\\s*' + literalizeRegex(anch).replace(/[_ ]/g, '[_ ]') + '\\s*=+|\\{\\{\\s*'+getValueOf('popupAnchorRegexp')+'\\s*(?:\\|[^|}]*)*?\\s*'+literalizeRegex(anch)+'\\s*(?:\\|[^}]*)?}})');
+
var anchRe=RegExp('(?:=+\\s*' + literalizeRegex(anch).replace(/[_ ]/g, '[_ ]') + '\\s*=+|\\{\\{\\s*'+getValueOf('popupAnchorRegexp')+'\\s*(?:\\|[^|}]*)*?\\s*'+literalizeRegex(anch)+'\\s*(?:\\|[^}]*)?\}\})');
 
var match=d.match(anchRe);
 
var match=d.match(anchRe);
 
if(match && match.length > 0 && match[0]) { return d.substring(d.indexOf(match[0])); }
 
if(match && match.length > 0 && match[0]) { return d.substring(d.indexOf(match[0])); }
Line 528: Line 586:
  
 
function killPopup() {
 
function killPopup() {
removeModifierKeyHandler(this);
+
if (getValueOf('popupShortcutKeys') && window.rmPopupShortcuts) { rmPopupShortcuts(); }
if (getValueOf('popupShortcutKeys')) { rmPopupShortcuts(); }
 
 
if (!pg) { return; }
 
if (!pg) { return; }
if (pg.current.link && pg.current.link.navpopup) { pg.current.link.navpopup.banish(); }
+
pg.current.link && pg.current.link.navpopup && pg.current.link.navpopup.banish();
 
pg.current.link=null;
 
pg.current.link=null;
 
abortAllDownloads();
 
abortAllDownloads();
if (pg.timer.checkPopupPosition) {
+
if (pg.timer.checkPopupPosition !== null) {
 
clearInterval(pg.timer.checkPopupPosition);
 
clearInterval(pg.timer.checkPopupPosition);
 
pg.timer.checkPopupPosition=null;
 
pg.timer.checkPopupPosition=null;
Line 606: Line 663:
 
o.vmode   = true;
 
o.vmode   = true;
  
o.root = oRoot ? oRoot : o ;
+
o.root = oRoot && oRoot !== null ? oRoot : o ;
  
 
if (isNaN(parseInt(o.root.style.left, 10))) { o.root.style.left  = "0px"; }
 
if (isNaN(parseInt(o.root.style.left, 10))) { o.root.style.left  = "0px"; }
Line 685: Line 742:
 
pg.structures.original.popupLayout=function () {
 
pg.structures.original.popupLayout=function () {
 
return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle',
 
return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle',
'popupUserData', 'popupData', 'popupOtherLinks',
+
'popupData', 'popupOtherLinks',
 
'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks',
 
'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks',
 
  'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
 
  'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
Line 738: Line 795:
 
// other page -> edit - history - un|watch - talk|edit|new
 
// other page -> edit - history - un|watch - talk|edit|new
 
var editstr='<<edit|shortcut=e>>';
 
var editstr='<<edit|shortcut=e>>';
var editOldidStr='if(oldid){<<editOld|shortcut=e>>|<<revert|shortcut=v|rv>>|<<edit|cur>>}else{' +
+
var editOldidStr='if(oldid){<<editOld|shortcut=e>>|<<revert|shortcut=v|rv>>|<<edit|cur>>}else{'
editstr + '}';
+
+ editstr + '}';
 
var historystr='<<history|shortcut=h>>';
 
var historystr='<<history|shortcut=h>>';
 
var watchstr='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';
 
var watchstr='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';
  
str += '<br>if(talk){' +
+
str+='<br>if(talk){' +
 
editOldidStr+'|<<new|shortcut=+>>' + '*' + historystr+'*'+watchstr + '*' +
 
editOldidStr+'|<<new|shortcut=+>>' + '*' + historystr+'*'+watchstr + '*' +
 
'<b><<article|shortcut=a>></b>|<<editArticle|edit>>' +
 
'<b><<article|shortcut=a>></b>|<<editArticle|edit>>' +
 
'}else{' + // not a talk page
 
'}else{' + // not a talk page
 
editOldidStr + '*' + historystr + '*' + watchstr + '*' +
 
editOldidStr + '*' + historystr + '*' + watchstr + '*' +
'<b><<talk|shortcut=t>></b>|<<editTalk|edit>>|<<newTalk|shortcut=+|new>>}';
+
'<b><<talk|shortcut=t>></b>|<<editTalk|edit>>|<<newTalk|shortcut=+|new>>'
 +
+ '}';
  
 
// misc links
 
// misc links
Line 767: Line 825:
 
};
 
};
 
pg.structures.fancy.popupTopLinks=function(x) {
 
pg.structures.fancy.popupTopLinks=function(x) {
var hist='<<history|shortcut=h|hist>>|<<lastEdit|shortcut=/|last>>|<<editors|shortcut=E|eds>>';
+
var hist='<<history|shortcut=h|hist>>|<<lastEdit|shortcut=/|last>>if(mainspace_en){|<<editors|shortcut=E|eds>>}';
 
var watch='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';
 
var watch='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';
 
var move='<<move|shortcut=m|move>>';
 
var move='<<move|shortcut=m|move>>';
Line 799: Line 857:
 
};
 
};
 
pg.structures.fancy2.popupLayout=function () { // move toplinks to after the title
 
pg.structures.fancy2.popupLayout=function () { // move toplinks to after the title
return ['popupError', 'popupImage', 'popupTitle', 'popupUserData', 'popupData', 'popupTopLinks', 'popupOtherLinks',
+
return ['popupError', 'popupImage', 'popupTitle', 'popupData', 'popupTopLinks', 'popupOtherLinks',
 
'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
 
'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
 
'popupMiscTools', ['popupRedlink'],
 
'popupMiscTools', ['popupRedlink'],
Line 810: Line 868:
 
return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle', 'popupOtherLinks',
 
return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle', 'popupOtherLinks',
 
'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
 
'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
'popupUserData', 'popupData', 'popupMiscTools', ['popupRedlink'],
+
'popupData', 'popupMiscTools', ['popupRedlink'],
 
'popupPrePreviewSep', 'popupPreview', 'popupSecondPreview', 'popupPreviewMore', 'popupPostPreview', 'popupFixDab'];
 
'popupPrePreviewSep', 'popupPreview', 'popupSecondPreview', 'popupPreviewMore', 'popupPostPreview', 'popupFixDab'];
 
};
 
};
 
+
function toggleSticky(uid) {
 +
var popDiv=document.getElementById('navpopup_maindiv'+uid);
 +
if (!popDiv) { return; }
 +
if (!popDiv.navpopup.sticky) { popDiv.navpopup.stick(); }
 +
else {
 +
popDiv.navpopup.unstick();
 +
popDiv.navpopup.hide();
 +
}
 +
}
 
pg.structures.menus.popupTopLinks = function (x, shorter) {
 
pg.structures.menus.popupTopLinks = function (x, shorter) {
 
// FIXME maybe this stuff should be cached
 
// FIXME maybe this stuff should be cached
Line 819: Line 885:
 
var dropdiv='<div class="popup_drop">';
 
var dropdiv='<div class="popup_drop">';
 
var enddiv='</div>';
 
var enddiv='</div>';
 +
var endspan='</span>';
 
var hist='<<history|shortcut=h>>';
 
var hist='<<history|shortcut=h>>';
 
if (!shorter) { hist = '<menurow>' + hist +
 
if (!shorter) { hist = '<menurow>' + hist +
'|<<historyfeed|rss>>|<<editors|shortcut=E>></menurow>'; }
+
'|<<historyfeed|rss>>if(mainspace_en){|<<editors|shortcut=E>>}</menurow>'; }
 
var lastedit='<<lastEdit|shortcut=/|show last edit>>';
 
var lastedit='<<lastEdit|shortcut=/|show last edit>>';
var thank='if(diff){<<thank|send thanks>>}';
 
 
var jsHistory='<<lastContrib|last set of edits>><<sinceMe|changes since mine>>';
 
var jsHistory='<<lastContrib|last set of edits>><<sinceMe|changes since mine>>';
 
var linkshere='<<whatLinksHere|shortcut=l|what links here>>';
 
var linkshere='<<whatLinksHere|shortcut=l|what links here>>';
Line 850: Line 916:
 
}
 
}
 
s.push( '<menu>');
 
s.push( '<menu>');
s.push( editRow + markPatrolled + newTopic + hist + lastedit + thank );
+
s.push( editRow + markPatrolled + newTopic + hist + lastedit );
 
if (!shorter) { s.push(jsHistory); }
 
if (!shorter) { s.push(jsHistory); }
 
s.push( move + linkshere + related);
 
s.push( move + linkshere + related);
Line 868: Line 934:
  
 
s.push('if(user){*' + dropdiv + menuTitle('user'));
 
s.push('if(user){*' + dropdiv + menuTitle('user'));
s.push('<menu>');
+
s.push('<menu>'); +
 
s.push('<menurow><<userPage|shortcut=u|user&nbsp;page>>|<<userSpace|space>></menurow>');
 
s.push('<menurow><<userPage|shortcut=u|user&nbsp;page>>|<<userSpace|space>></menurow>');
 
s.push('<<userTalk|shortcut=t|user talk>><<editUserTalk|edit user talk>>' +
 
s.push('<<userTalk|shortcut=t|user talk>><<editUserTalk|edit user talk>>' +
Line 877: Line 943:
 
s.push('if(wikimedia){<<count|shortcut=#|edit counter>>}');
 
s.push('if(wikimedia){<<count|shortcut=#|edit counter>>}');
 
s.push('if(admin){<menurow><<unblock|unblockShort>>|<<block|shortcut=b|block user>></menurow>}');
 
s.push('if(admin){<menurow><<unblock|unblockShort>>|<<block|shortcut=b|block user>></menurow>}');
s.push('<<blocklog|shortcut=B|block log>>');
+
s.push('<<blocklog|shortcut=B|block log>>' + getValueOf('popupExtraUserMenu'));
 
s.push('</menu>'  + enddiv + '}');
 
s.push('</menu>'  + enddiv + '}');
  
Line 904: Line 970:
 
};
 
};
 
pg.structures.shortmenus.popupRedirTopLinks=pg.structures.shortmenus.popupTopLinks;
 
pg.structures.shortmenus.popupRedirTopLinks=pg.structures.shortmenus.popupTopLinks;
 +
 +
copyStructure('shortmenus', 'dabshortmenus');
 +
pg.structures.dabshortmenus.popupLayout=function () {
 +
return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle', 'popupOtherLinks',
 +
'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
 +
'popupData', 'popupMiscTools', ['popupRedlink'], 'popupFixDab',
 +
'popupPrePreviewSep', 'popupPreview', 'popupSecondPreview', 'popupPreviewMore', 'popupPostPreview'];
 +
};
 +
 +
copyStructure('menus', 'dabmenus');
 +
pg.structures.dabmenus.popupLayout=pg.structures.dabshortmenus.popupLayout;
 +
  
 
//</NOLITE>
 
//</NOLITE>
Line 985: Line 1,063:
 
}
 
}
  
/*eslint-disable*/
 
 
function skipToEnd(str,sep) {
 
function skipToEnd(str,sep) {
 
return {segment: str, remainder: ''};
 
return {segment: str, remainder: ''};
 
}
 
}
/*eslint-enable */
 
  
 
function findNext(str, ch) {
 
function findNext(str, ch) {
Line 1,001: Line 1,077:
 
function setCheckbox(param, box) {
 
function setCheckbox(param, box) {
 
var val=mw.util.getParamValue(param);
 
var val=mw.util.getParamValue(param);
if (val) {
+
if (val!==null) {
 
switch (val) {
 
switch (val) {
 
case '1': case 'yes': case 'true':
 
case '1': case 'yes': case 'true':
Line 1,013: Line 1,089:
  
 
function autoEdit() {
 
function autoEdit() {
setupPopups( function () {
+
if (!setupPopups.completed) { setupPopups(); }
if (mw.util.getParamValue('autoimpl') !== popupString('autoedit_version') ) { return false; }
+
if (!mw.config.get('wgEnableAPI') || mw.util.getParamValue('autoimpl') !== popupString('autoedit_version') ) { return false; }
if (mw.util.getParamValue('autowatchlist') && mw.util.getParamValue('actoken')===autoClickToken()) {
+
if (mw.util.getParamValue('autowatchlist') && mw.util.getParamValue('actoken')===autoClickToken()) {
pg.fn.modifyWatchlist(mw.util.getParamValue('title'), mw.util.getParamValue('action'));
+
modifyWatchlist(mw.util.getParamValue('title'), mw.util.getParamValue('action'));
}
+
}
if (!document.editform) { return false; }
+
if (!document.editform) { return false; }
if (autoEdit.alreadyRan) { return false; }
+
if (window.autoEdit.alreadyRan) { return false; }
autoEdit.alreadyRan=true;
+
window.autoEdit.alreadyRan=true;
var cmdString=mw.util.getParamValue('autoedit');
+
var cmdString=mw.util.getParamValue('autoedit');
if (cmdString) {
+
if (cmdString) {
try {
+
try {
var editbox=document.editform.wpTextbox1;
+
var editbox=document.editform.wpTextbox1;
var cmdList=parseCmd(cmdString);
+
} catch (dang) { return; }
var input=editbox.value;
+
var cmdList=parseCmd(cmdString);
var output=execCmds(input, cmdList);
+
var input=editbox.value;
editbox.value=output;
+
var output=execCmds(input, cmdList);
} catch (dang) { return; }
+
editbox.value=output;
// wikEd user script compatibility
+
// wikEd user script compatibility
if (typeof(wikEdUseWikEd) != 'undefined') {
+
if (typeof(wikEdUseWikEd) != 'undefined') {
if (wikEdUseWikEd === true) {
+
if (wikEdUseWikEd === true) {
WikEdUpdateFrame();
+
WikEdUpdateFrame();
}
 
 
}
 
}
 
}
 
}
setCheckbox('autominor', document.editform.wpMinoredit);
+
}
setCheckbox('autowatch', document.editform.wpWatchthis);
+
setCheckbox('autominor', document.editform.wpMinoredit);
+
setCheckbox('autowatch', document.editform.wpWatchthis);
var rvid = mw.util.getParamValue('autorv');
+
 
if (rvid) {
+
var rvid = mw.util.getParamValue('autorv');
var url=pg.wiki.apiwikibase + '?action=query&format=json&formatversion=2&prop=revisions&revids='+rvid;
+
if (rvid) {
startDownload(url, null, autoEdit2);
+
var url=pg.wiki.apiwikibase + '?action=query&format=json&prop=revisions&revids='+rvid;
} else { autoEdit2(); }
+
startDownload(url, null, autoEdit2);
} );
+
} else { autoEdit2(); }
 
}
 
}
  
Line 1,054: Line 1,129:
 
if (d && d.data && mw.util.getParamValue('autorv')) {
 
if (d && d.data && mw.util.getParamValue('autorv')) {
 
var s = getRvSummary(summary, d.data);
 
var s = getRvSummary(summary, d.data);
if (s === false) {
+
if (s===false) {
 
summaryprompt=true;
 
summaryprompt=true;
 
summarynotice=popupString('Failed to get revision information, please edit manually.\n\n');
 
summarynotice=popupString('Failed to get revision information, please edit manually.\n\n');
Line 1,109: Line 1,184:
 
var o=getJsObj(json);
 
var o=getJsObj(json);
 
var edit = anyChild(o.query.pages).revisions[0];
 
var edit = anyChild(o.query.pages).revisions[0];
var timestamp = edit.timestamp.split(/[A-Z]/g).join(' ').replace(/^ *| *$/g, '');
+
} catch (badness) {return false;}
return simplePrintf(template, [edit.revid, timestamp, edit.userhidden ? '(hidden)' : edit.user ]);
+
var timestamp = edit.timestamp.split(/[A-Z]/g).join(' ').replace(/^ *| *$/g, '');
} catch (badness) {
+
return simplePrintf(template, [edit.revid, timestamp, edit.user]);
return false;
 
}
 
 
}
 
}
  
Line 1,131: Line 1,204:
 
*/
 
*/
 
function Downloader(url) {
 
function Downloader(url) {
if (typeof XMLHttpRequest!='undefined') { this.http = new XMLHttpRequest(); }
+
// Source: http://jibbering.com/2002/4/httprequest.html
 +
/** xmlhttprequest object which we're wrapping */
 +
this.http = false;
 +
 
 +
/*@cc_on @*/
 +
/*@if (@_jscript_version >= 5)
 +
// JScript gives us Conditional compilation,
 +
// we can cope with old IE versions.
 +
// and security blocked creation of the objects.
 +
try {
 +
this.http = new ActiveXObject("Msxml2.XMLHTTP");
 +
} catch (e) {
 +
try {
 +
this.http = new ActiveXObject("Microsoft.XMLHTTP");
 +
} catch (E) {
 +
// this.http = false;
 +
}
 +
}
 +
@end @*/
 +
 
 +
if (! this.http && typeof XMLHttpRequest!='undefined') { this.http = new XMLHttpRequest(); }
 
/**
 
/**
 
The url to download
 
The url to download
Line 1,166: Line 1,259:
 
this.aborted = false;
 
this.aborted = false;
 
/**
 
/**
  HTTP method. See https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html for possibilities.
+
  HTTP method. See http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html for possibilities.
 
  @type String
 
  @type String
 
*/
 
*/
Line 1,196: Line 1,289:
 
if (!this.http) { return null; }
 
if (!this.http) { return null; }
 
this.http.open(this.method, this.url, this.async);
 
this.http.open(this.method, this.url, this.async);
this.http.setRequestHeader( 'Api-User-Agent', pg.misc.userAgent );
 
 
};
 
};
 
/** Gets the state of the download. */
 
/** Gets the state of the download. */
Line 1,316: Line 1,408:
 
pg.misc.downloadsInProgress[x].abort();
 
pg.misc.downloadsInProgress[x].abort();
 
delete pg.misc.downloadsInProgress[x];
 
delete pg.misc.downloadsInProgress[x];
} catch (e) {}
+
} catch (e) { }
 
}
 
}
 
}
 
}
Line 1,322: Line 1,414:
 
// STARTFILE: livepreview.js
 
// STARTFILE: livepreview.js
 
// TODO: location is often not correct (eg relative links in previews)
 
// TODO: location is often not correct (eg relative links in previews)
// NOTE: removed md5 and image and math parsing. was broken, lots of bytes.
+
 
 
/**
 
/**
 
  * InstaView - a Mediawiki to HTML converter in JavaScript
 
  * InstaView - a Mediawiki to HTML converter in JavaScript
 
  * Version 0.6.1
 
  * Version 0.6.1
 
  * Copyright (C) Pedro Fayolle 2005-2006
 
  * Copyright (C) Pedro Fayolle 2005-2006
  * https://en.wikipedia.org/wiki/User:Pilaf
+
  * http://en.wikipedia.org/wiki/User:Pilaf
 
  * Distributed under the BSD license
 
  * Distributed under the BSD license
 
  *
 
  *
Line 1,377: Line 1,469:
 
// Only used for Insta previews with images. (not in popups)
 
// Only used for Insta previews with images. (not in popups)
 
math: '/math/',
 
math: '/math/',
images: '//upload.wikimedia.org/wikipedia/en/', // FIXME getImageUrlStart(pg.wiki.hostname),
+
images: '//upload.wikimedia.org/wikipedia/en/', // FIXME ( window.getImageUrlStart ? getImageUrlStart(pg.wiki.hostname) : ''),
 
images_fallback: '//upload.wikimedia.org/wikipedia/commons/',
 
images_fallback: '//upload.wikimedia.org/wikipedia/commons/',
 
},
 
},
Line 1,391: Line 1,483:
  
 
// options with default values or backreferences
 
// options with default values or backreferences
Insta.conf.user.name = Insta.conf.user.name || 'Wikipedian';
+
with (Insta.conf) {
Insta.conf.user.signature = '[['+Insta.conf.locale.user+':'+Insta.conf.user.name+'|'+Insta.conf.user.name+']]';
+
user.name = user.name || 'Wikipedian';
//Insta.conf.paths.images = '//upload.wikimedia.org/wikipedia/' + Insta.conf.wiki.lang + '/';
+
user.signature = '[['+locale.user+':'+user.name+'|'+user.name+']]';
 +
//paths.images = '//upload.wikimedia.org/wikipedia/' + wiki.lang + '/';
 +
}
  
 
// define constants
 
// define constants
Line 1,452: Line 1,546:
 
return htmlescape_text(s).replace(/'/g,"&#39;").replace(/"/g,"&quot;");
 
return htmlescape_text(s).replace(/'/g,"&#39;").replace(/"/g,"&quot;");
 
}
 
}
 +
 +
function max(a,b) { return (a>b)?a:b; }
 +
function min(a,b) { return (a<b)?a:b; }
  
 
// return the first non matching character position between two strings
 
// return the first non matching character position between two strings
 
function str_imatch(a, b)
 
function str_imatch(a, b)
 
{
 
{
for (var i=0, l=Math.min(a.length, b.length); i<l; i++) {
+
for (var i=0, l=min(a.length, b.length); i<l; i++) {
 
if (a.charAt(i)!=b.charAt(i)) { break; }
 
if (a.charAt(i)!=b.charAt(i)) { break; }
 
}
 
}
Line 1,485: Line 1,582:
  
 
// close uncontinued lists
 
// close uncontinued lists
for (var prevPos=prev.length-1; prevPos >= ipos; prevPos--) {
+
for (var i=prev.length-1; i >= ipos; i--) {
  
var pi = prev.charAt(prevPos);
+
var pi = prev.charAt(i);
  
 
if (pi=='*') { ps('</ul>'); }
 
if (pi=='*') { ps('</ul>'); }
 
else if (pi=='#') { ps('</ol>'); }
 
else if (pi=='#') { ps('</ol>'); }
 
// close a dl only if the new item is not a dl item (:, ; or empty)
 
// close a dl only if the new item is not a dl item (:, ; or empty)
else if($.inArray(l_match[1].charAt(prevPos), ['','*','#'])) { ps('</dl>'); }
+
else switch (l_match[1].charAt(i)) { case'':case'*':case'#': ps('</dl>') }
 
}
 
}
  
 
// open new lists
 
// open new lists
for (var matchPos=ipos; matchPos<l_match[1].length; matchPos++) {
+
for (var i=ipos; i<l_match[1].length; i++) {
  
var li = l_match[1].charAt(matchPos);
+
var li = l_match[1].charAt(i);
  
 
if (li=='*') { ps('<ul>'); }
 
if (li=='*') { ps('<ul>'); }
 
else if (li=='#') { ps('<ol>'); }
 
else if (li=='#') { ps('<ol>'); }
 
// open a new dl only if the prev item is not a dl item (:, ; or empty)
 
// open a new dl only if the prev item is not a dl item (:, ; or empty)
else if ($.inArray(prev.charAt(matchPos), ['','*','#'])) { ps('<dl>'); }
+
else { switch(prev.charAt(i)) { case'':case'*':case'#': ps('<dl>') } }
 
}
 
}
  
Line 1,509: Line 1,606:
  
 
case '*': case '#':
 
case '*': case '#':
ps('<li>' + parse_inline_nowiki(l_match[2]));
+
ps('<li>' + parse_inline_nowiki(l_match[2])); break
break;
 
  
 
case ';':
 
case ';':
 
ps('<dt>');
 
ps('<dt>');
  
var dt_match = l_match[2].match(/(.*?)(:.*?)$/);
+
var dt_match;
  
 
// handle ;dt :dd format
 
// handle ;dt :dd format
if (dt_match) {
+
if (dt_match = l_match[2].match(/(.*?)(:.*?)$/)) {
ps(parse_inline_nowiki(dt_match[1]));
+
 
ll.unshift(dt_match[2]);
+
ps(parse_inline_nowiki(dt_match[1]))
 +
ll.unshift(dt_match[2])
 +
 
 +
} else ps(parse_inline_nowiki(l_match[2]))
  
} else ps(parse_inline_nowiki(l_match[2]));
 
 
break;
 
break;
  
 
case ':':
 
case ':':
ps('<dd>' + parse_inline_nowiki(l_match[2]));
+
ps('<dd>' + parse_inline_nowiki(l_match[2]))
 
}
 
}
  
prev=l_match[1];
+
prev=l_match[1]
 
}
 
}
  
 
// close remaining lists
 
// close remaining lists
for (var i=prev.length-1; i>=0; i--) {
+
for (var i=prev.length-1; i>=0; i--)
ps(f('</?>', (prev.charAt(i)=='*')? 'ul': ((prev.charAt(i)=='#')? 'ol': 'dl')));
+
ps(f('</?>', (prev.charAt(i)=='*')? 'ul': ((prev.charAt(i)=='#')? 'ol': 'dl')))
}
 
 
}
 
}
  
 
function parse_table()
 
function parse_table()
 
{
 
{
endl(f('<table>', $(/^\{\|( .*)$/)? $r[1]: ''));
+
endl(f('<table>', $(/^\{\|( .*)$/)? $r[1]: ''))
  
 
for (;remain();) if ($('|')) switch (_(1)) {
 
for (;remain();) if ($('|')) switch (_(1)) {
case '}':
+
case '}': endl('</table>'); return
endl('</table>');
+
case '-': endl(f('<tr>', $(/\|-*(.*)/)[1])); break
return;
+
default: parse_table_data()
case '-':
 
endl(f('<tr>', $(/\|-*(.*)/)[1]));
 
break;
 
default:
 
parse_table_data();
 
 
}
 
}
else if ($('!')) { parse_table_data(); }
+
else if ($('!')) parse_table_data()
else { sh(); }
+
else sh()
 
}
 
}
  
 
function parse_table_data()
 
function parse_table_data()
 
{
 
{
var td_line, match_i;
+
var td_line, match_i
  
 
// 1: "|+", '|' or '+'
 
// 1: "|+", '|' or '+'
Line 1,564: Line 1,656:
 
// 3: attributes ??
 
// 3: attributes ??
 
// TODO: finish commenting this regexp
 
// TODO: finish commenting this regexp
var td_match = sh().match(/^(\|\+|\||!)((?:([^[|]*?)\|(?!\|))?(.*))$/);
+
var td_match = sh().match(/^(\|\+|\||!)((?:([^[|]*?)\|(?!\|))?(.*))$/)
  
 
if (td_match[1] == '|+') ps('<caption');
 
if (td_match[1] == '|+') ps('<caption');
else ps('<t' + ((td_match[1]=='|')?'d':'h'));
+
else ps('<t' + ((td_match[1]=='|')?'d':'h'))
  
 
if (typeof td_match[3] != 'undefined') {
 
if (typeof td_match[3] != 'undefined') {
  
 
//ps(' ' + td_match[3])
 
//ps(' ' + td_match[3])
match_i = 4;
+
match_i = 4
  
} else match_i = 2;
+
} else match_i = 2
  
ps('>');
+
ps('>')
  
 
if (td_match[1] != '|+') {
 
if (td_match[1] != '|+') {
Line 1,582: Line 1,674:
 
// use || or !! as a cell separator depending on context
 
// use || or !! as a cell separator depending on context
 
// NOTE: when split() is passed a regexp make sure to use non-capturing brackets
 
// NOTE: when split() is passed a regexp make sure to use non-capturing brackets
td_line = td_match[match_i].split((td_match[1] == '|')? '||': /(?:\|\||!!)/);
+
td_line = td_match[match_i].split((td_match[1] == '|')? '||': /(?:\|\||!!)/)
  
ps(parse_inline_nowiki(td_line.shift()));
+
ps(parse_inline_nowiki(td_line.shift()))
  
while (td_line.length) ll.unshift(td_match[1] + td_line.pop());
+
while (td_line.length) ll.unshift(td_match[1] + td_line.pop())
  
} else ps(td_match[match_i]);
+
} else ps(td_match[match_i])
  
var tc = 0, td = [];
+
var tc = 0, td = []
  
while (remain()) {
+
for (;remain(); td.push(sh()))
td.push(sh());
+
if ($('|')) {
if ($('|')) {
+
if (!tc) break // we're at the outer-most level (no nested tables), skip to td parse
if (!tc) break; // we're at the outer-most level (no nested tables), skip to td parse
+
else if (_(1)=='}') tc--
else if (_(1)=='}') tc--;
 
}
 
else if (!tc && $('!')) break;
 
else if ($('{|')) tc++;
 
 
}
 
}
 +
else if (!tc && $('!')) break
 +
else if ($('{|')) tc++
  
if (td.length) ps(Insta.convert(td));
+
if (td.length) ps(Insta.convert(td))
 
}
 
}
  
 
function parse_pre()
 
function parse_pre()
 
{
 
{
ps('<pre>');
+
ps('<pre>')
do {
+
do endl(parse_inline_nowiki(ll[0].substring(1)) + "\n"); while (remain() && $(' '))
endl(parse_inline_nowiki(ll[0].substring(1)) + "\n");
+
ps('</pre>')
} while (remain() && $(' '));
 
ps('</pre>');
 
 
}
 
}
  
 
function parse_block_image()
 
function parse_block_image()
 
{
 
{
ps(parse_image(sh()));
+
ps(parse_image(sh()))
 
}
 
}
  
Line 1,624: Line 1,712:
 
// get what's in between "[[Image:" and "]]"
 
// get what's in between "[[Image:" and "]]"
 
var tag = str.substring(str.indexOf(':') + 1, str.length - 2);
 
var tag = str.substring(str.indexOf(':') + 1, str.length - 2);
/* eslint-disable no-unused-vars */
+
 
 
var width;
 
var width;
 
var attr = [], filename, caption = '';
 
var attr = [], filename, caption = '';
 
var thumb=0, frame=0, center=0;
 
var thumb=0, frame=0, center=0;
 
var align='';
 
var align='';
/* eslint-enable no-unused-vars */
 
  
 
if (tag.match(/\|/)) {
 
if (tag.match(/\|/)) {
Line 1,657: Line 1,744:
 
var w_match;
 
var w_match;
  
for (;attr.length; attr.shift()) {
+
for (;attr.length; attr.shift())
w_match = attr[0].match(/^(\d*)(?:[px]*\d*)?px$/);
+
if (w_match = attr[0].match(/^(\d*)(?:[px]*\d*)?px$/)) width = w_match[1]
if (w_match) width = w_match[1];
+
else switch(attr[0]) {
else switch(attr[0]) {
+
case 'thumb':
case 'thumb':
+
case 'thumbnail':
case 'thumbnail':
+
thumb=true;
thumb=true;
+
case 'frame':
frame=true;
+
frame=true;
break;
+
break;
case 'frame':
+
case 'none':
frame=true;
+
case 'right':
break;
+
case 'left':
case 'none':
+
center=false;
case 'right':
+
align=attr[0];
case 'left':
+
break;
center=false;
+
case 'center':
align=attr[0];
+
center=true;
break;
+
align='none';
case 'center':
+
break;
center=true;
+
default:
align='none';
+
if (attr.length == 1) caption = attr[0];
break;
 
default:
 
if (attr.length == 1) caption = attr[0];
 
 
}
 
}
}
 
  
 
} else filename = tag;
 
} else filename = tag;
  
return '';
+
 
 +
var o='';
 +
 
 +
if (frame) {
 +
 
 +
if (align=='') align = 'right';
 +
 
 +
o += f("<div class='thumb t?'>", align);
 +
 
 +
if (thumb) {
 +
if (!width) width = Insta.conf.wiki.default_thumb_width;
 +
 
 +
o += f("<div style='width:?px;'>?", 2+width*1, make_image(filename, caption, width)) +
 +
f("<div class='thumbcaption'><div class='magnify' style='float:right'><a href='?' title='Enlarge'></a></div>?</div>",
 +
htmlescape_attr(Insta.conf.paths.articles + Insta.conf.locale.image + ':' + filename),
 +
parse_inline_nowiki(caption)
 +
)
 +
} else {
 +
o += '<div>' + make_image(filename, caption) + f("<div class='thumbcaption'>?</div>", parse_inline_nowiki(caption))
 +
}
 +
 
 +
o += '</div></div>';
 +
 
 +
} else if (align != '') {
 +
o += f("<div class='float?'><span>?</span></div>", align, make_image(filename, caption, width));
 +
} else {
 +
return make_image(filename, caption, width);
 +
}
 +
 
 +
return center? f("<div class='center'>?</div>", o): o;
 
//</NOLITE>
 
//</NOLITE>
 
}
 
}
Line 1,692: Line 1,804:
 
function parse_inline_nowiki(str)
 
function parse_inline_nowiki(str)
 
{
 
{
var start, lastend=0;
+
var start, lastend=0
 
var substart=0, nestlev=0, open, close, subloop;
 
var substart=0, nestlev=0, open, close, subloop;
 
var html='';
 
var html='';
Line 1,720: Line 1,832:
 
nestlev++;
 
nestlev++;
 
}
 
}
} while (subloop);
+
} while (subloop)
 
}
 
}
  
 
return html + parse_inline_wiki(str.substr(lastend));
 
return html + parse_inline_wiki(str.substr(lastend));
 +
}
 +
 +
function make_image(filename, caption, width)
 +
{
 +
//<NOLITE>
 +
// uppercase first letter in file name
 +
filename = filename.charAt(0).toUpperCase() + filename.substr(1);
 +
// replace spaces with underscores
 +
filename = filename.replace(/ /g, '_');
 +
 +
caption = strip_inline_wiki(caption);
 +
 +
var md5 = hex_md5(filename);
 +
 +
var source = md5.charAt(0) + '/' + md5.substr(0,2) + '/' + filename;
 +
 +
if (width) width = "width='" + width + "px'";
 +
 +
var img = "<img onerror=\""+pg.escapeQuotesHTML("this.onerror=null;this.src='"+pg.jsescape(Insta.conf.paths.images_fallback + source)+"'")+"\" src=\""+pg.escapeQuotesHTML(Insta.conf.paths.images + source)+"\" "+(caption!='' ? "alt=\"" + pg.escapeQuotesHTML(caption) + "\"" : '')+" "+width+">";
 +
 +
return f("<a class='image' ? href=\"?\">?</a>", (caption!='')? "title=\"" + pg.escapeQuotesHTML(caption) + "\"" : '', pg.escapeQuotesHTML(Insta.conf.paths.articles + Insta.conf.locale.image + ':' + filename), img);
 +
//</NOLITE>
 
}
 
}
  
Line 1,756: Line 1,890:
 
nestlev++;
 
nestlev++;
 
}
 
}
} while (loop);
+
} while (loop)
  
 
} else break;
 
} else break;
Line 1,787: Line 1,921:
 
function parse_inline_wiki(str)
 
function parse_inline_wiki(str)
 
{
 
{
 +
var aux_match;
 +
 
str = parse_inline_images(str);
 
str = parse_inline_images(str);
 
str = parse_inline_formatting(str);
 
str = parse_inline_formatting(str);
  
 
// math
 
// math
str = str.replace(/<(?:)math>(.*?)<\/math>/ig, '');
+
while (aux_match = str.match(/<(?:)math>(.*?)<\/math>/i)) {
 +
var math_md5 = hex_md5(aux_match[1]);
 +
str = str.replace(aux_match[0], f("<img src='?.png'>", Insta.conf.paths.math+math_md5));
 +
}
  
 
// Build a Mediawiki-formatted date string
 
// Build a Mediawiki-formatted date string
var date = new Date();
+
var date = new Date;
 
var minutes = date.getUTCMinutes();
 
var minutes = date.getUTCMinutes();
 
if (minutes < 10) minutes = '0' + minutes;
 
if (minutes < 10) minutes = '0' + minutes;
date = f("?:?, ? ? ? (UTC)", date.getUTCHours(), minutes, date.getUTCDate(), Insta.conf.locale.months[date.getUTCMonth()], date.getUTCFullYear());
+
var date = f("?:?, ? ? ? (UTC)", date.getUTCHours(), minutes, date.getUTCDate(), Insta.conf.locale.months[date.getUTCMonth()], date.getUTCFullYear());
  
 
// text formatting
 
// text formatting
Line 1,807: Line 1,946:
  
 
// [[:Category:...]], [[:Image:...]], etc...
 
// [[:Category:...]], [[:Image:...]], etc...
replace(RegExp('\\[\\[:((?:'+Insta.conf.locale.category+'|Image|File|'+Insta.conf.locale.image+'|'+Insta.conf.wiki.interwiki+'):[^|]*?)\\]\\](\\w*)','gi'), function($0,$1,$2){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($1) + htmlescape_text($2));}).
+
replace(RegExp('\\[\\[:((?:'+Insta.conf.locale.category+'|Image|File|'+Insta.conf.locale.image+'|'+Insta.conf.wiki.interwiki+'):[^|]*?)\\]\\](\w*)','gi'), function($0,$1,$2){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($1) + htmlescape_text($2));}).
 
// remove straight category and interwiki tags
 
// remove straight category and interwiki tags
 
replace(RegExp('\\[\\[(?:'+Insta.conf.locale.category+'|'+Insta.conf.wiki.interwiki+'):.*?\\]\\]','gi'),'').
 
replace(RegExp('\\[\\[(?:'+Insta.conf.locale.category+'|'+Insta.conf.wiki.interwiki+'):.*?\\]\\]','gi'),'').
Line 1,821: Line 1,960:
  
 
// [[Common links]]
 
// [[Common links]]
replace(/\[\[([^[|]*?)\]\](\w*)/g, function($0,$1,$2){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($1) + htmlescape_text($2)); }).
+
replace(/\[\[([^|]*?)\]\](\w*)/g, function($0,$1,$2){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($1) + htmlescape_text($2)); }).
  
 
// [[Replaced|Links]]
 
// [[Replaced|Links]]
replace(/\[\[([^[]*?)\|([^\]]+?)\]\](\w*)/g, function($0,$1,$2,$3){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($2) + htmlescape_text($3)); }).
+
replace(/\[\[(.*?)\|([^\]]+?)\]\](\w*)/g, function($0,$1,$2,$3){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($2) + htmlescape_text($3)); }).
  
 
// [[Stripped:Namespace|Namespace]]
 
// [[Stripped:Namespace|Namespace]]
Line 1,832: Line 1,971:
 
replace(/\[(https?|news|ftp|mailto|gopher|irc):(\/*)([^\]]*?) (.*?)\]/g, function($0,$1,$2,$3,$4){return f("<a class='external' href='?:?'>?</a>", htmlescape_attr($1), htmlescape_attr($2) + htmlescape_attr($3), htmlescape_text($4)); }).
 
replace(/\[(https?|news|ftp|mailto|gopher|irc):(\/*)([^\]]*?) (.*?)\]/g, function($0,$1,$2,$3,$4){return f("<a class='external' href='?:?'>?</a>", htmlescape_attr($1), htmlescape_attr($2) + htmlescape_attr($3), htmlescape_text($4)); }).
 
replace(/\[http:\/\/(.*?)\]/g, function($0,$1){return f("<a class='external' href='http://?'>[#]</a>", htmlescape_attr($1)); }).
 
replace(/\[http:\/\/(.*?)\]/g, function($0,$1){return f("<a class='external' href='http://?'>[#]</a>", htmlescape_attr($1)); }).
replace(/\[(news|ftp|mailto|gopher|irc):(\/*)(.*?)\]/g, function($0,$1,$2,$3){return f("<a class='external' href='?:?'>?:?</a>", htmlescape_attr($1), htmlescape_attr($2) + htmlescape_attr($3), htmlescape_text($1), htmlescape_text($2) + htmlescape_text($3)); }).
+
replace(/\[(news|ftp|mailto|gopher|irc):(\/*)(.*?)\]/g, function($0,$1,$2,$3,$4){return f("<a class='external' href='?:?'>?:?</a>", htmlescape_attr($1), htmlescape_attr($2) + htmlescape_attr($3), htmlescape_text($1), htmlescape_text($2) + htmlescape_text($3)); }).
 
replace(/(^| )(https?|news|ftp|mailto|gopher|irc):(\/*)([^ $]*[^.,!?;: $])/g, function($0,$1,$2,$3,$4){return f("?<a class='external' href='?:?'>?:?</a>", htmlescape_text($1), htmlescape_attr($2), htmlescape_attr($3) + htmlescape_attr($4), htmlescape_text($2), htmlescape_text($3) + htmlescape_text($4)); }).
 
replace(/(^| )(https?|news|ftp|mailto|gopher|irc):(\/*)([^ $]*[^.,!?;: $])/g, function($0,$1,$2,$3,$4){return f("?<a class='external' href='?:?'>?:?</a>", htmlescape_text($1), htmlescape_attr($2), htmlescape_attr($3) + htmlescape_attr($4), htmlescape_text($2), htmlescape_text($3) + htmlescape_text($4)); }).
  
 
replace('__NOTOC__','').
 
replace('__NOTOC__','').
 
replace('__NOEDITSECTION__','');
 
replace('__NOEDITSECTION__','');
 +
}
 +
/*
 +
*/
 +
function strip_inline_wiki(str)
 +
{
 +
return str
 +
.replace(/\[\[[^\]]*\|(.*?)\]\]/g,'$1')
 +
.replace(/\[\[(.*?)\]\]/g,'$1')
 +
.replace(/''(.*?)''/g,'$1');
 
}
 
}
  
 
// begin parsing
 
// begin parsing
 
for (;remain();) if ($(/^(={1,6})(.*)\1(.*)$/)) {
 
for (;remain();) if ($(/^(={1,6})(.*)\1(.*)$/)) {
p=0;
+
p=0
endl(f('<h?>?</h?>?', $r[1].length, parse_inline_nowiki($r[2]), $r[1].length, $r[3]));
+
endl(f('<h?>?</h?>?', $r[1].length, parse_inline_nowiki($r[2]), $r[1].length, $r[3]))
  
 
} else if ($(/^[*#:;]/)) {
 
} else if ($(/^[*#:;]/)) {
p=0;
+
p=0
parse_list();
+
parse_list()
  
 
} else if ($(' ')) {
 
} else if ($(' ')) {
p=0;
+
p=0
parse_pre();
+
parse_pre()
  
 
} else if ($('{|')) {
 
} else if ($('{|')) {
p=0;
+
p=0
parse_table();
+
parse_table()
  
 
} else if ($(/^----+$/)) {
 
} else if ($(/^----+$/)) {
p=0;
+
p=0
endl('<hr />');
+
endl('<hr />')
  
 
} else if ($(Insta.BLOCK_IMAGE)) {
 
} else if ($(Insta.BLOCK_IMAGE)) {
p=0;
+
p=0
parse_block_image();
+
parse_block_image()
  
 
} else {
 
} else {
Line 1,868: Line 2,016:
 
// handle paragraphs
 
// handle paragraphs
 
if ($$('')) {
 
if ($$('')) {
p = (remain()>1 && ll[1]===(''));
+
if (p = (remain()>1 && ll[1]==(''))) endl('<p><br>')
if (p) endl('<p><br>');
 
 
} else {
 
} else {
 
if(!p) {
 
if(!p) {
ps('<p>');
+
ps('<p>')
p=1;
+
p=1
 
}
 
}
ps(parse_inline_nowiki(ll[0]) + ' ');
+
ps(parse_inline_nowiki(ll[0]) + ' ')
 
}
 
}
  
Line 1,881: Line 2,028:
 
}
 
}
  
return o;
+
return o
 
};
 
};
  
function wiki2html(txt,baseurl) {
+
window.wiki2html=function(txt,baseurl) {
 
Insta.conf.baseUrl=baseurl;
 
Insta.conf.baseUrl=baseurl;
 
return Insta.convert(txt);
 
return Insta.convert(txt);
}
+
};
 
// ENDFILE: livepreview.js
 
// ENDFILE: livepreview.js
 
// STARTFILE: pageinfo.js
 
// STARTFILE: pageinfo.js
Line 1,997: Line 2,144:
  
 
function popupFilterDisambigDetect(data, download, article) {
 
function popupFilterDisambigDetect(data, download, article) {
if (!getValueOf('popupAllDabsStubs') && article.namespace()) { return ''; }
+
if (getValueOf('popupOnlyArticleDabStub') && article.namespace()) { return ''; }
 
return (isDisambig(data, article)) ? popupString('disambig') : '';
 
return (isDisambig(data, article)) ? popupString('disambig') : '';
 
}
 
}
Line 2,152: Line 2,299:
 
var splitted=h.split('?');
 
var splitted=h.split('?');
 
splitted[0]=splitted[0].split('&').join('%26');
 
splitted[0]=splitted[0].split('&').join('%26');
 +
 +
if (pg.flag.linksLikeIE6) {
 +
splitted[0]=encodeURI(decode_utf8(splitted[0]));
 +
}
  
 
h=splitted.join('?');
 
h=splitted.join('?');
  
 
var contribs=pg.re.contribs.exec(h);
 
var contribs=pg.re.contribs.exec(h);
if (contribs) {
+
if (contribs !== null) {
 
if (contribs[1]=='title=') { contribs[3]=contribs[3].split('+').join(' '); }
 
if (contribs[1]=='title=') { contribs[3]=contribs[3].split('+').join(' '); }
 
var u=new Title(contribs[3]);
 
var u=new Title(contribs[3]);
Line 2,164: Line 2,315:
  
 
var email=pg.re.email.exec(h);
 
var email=pg.re.email.exec(h);
if (email) {
+
if (email !== null) {
 
this.setUtf(this.decodeNasties(mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' + new Title(email[3]).stripNamespace()));
 
this.setUtf(this.decodeNasties(mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' + new Title(email[3]).stripNamespace()));
 
return this;
 
return this;
Line 2,185: Line 2,336:
 
// hopefully it's not a disguised user-related or specially treated special page
 
// hopefully it's not a disguised user-related or specially treated special page
 
var m=pg.re.main.exec(h);
 
var m=pg.re.main.exec(h);
if(m === null) { this.value=null; }
+
if(m===null) { this.value=null; }
 
else {
 
else {
 
var fromBotInterface = /[?](.+[&])?title=/.test(h);
 
var fromBotInterface = /[?](.+[&])?title=/.test(h);
Line 2,228: Line 2,379:
 
Title.prototype.fromWikiText=function(txt) {
 
Title.prototype.fromWikiText=function(txt) {
 
// FIXME - testing needed
 
// FIXME - testing needed
txt=myDecodeURI(txt);
+
if (!pg.flag.linksLikeIE6) { txt=myDecodeURI(txt); }
 
this.setUtf(txt);
 
this.setUtf(txt);
 
return this;
 
return this;
Line 2,255: Line 2,406:
 
// and return the corresponding talk page otherwise
 
// and return the corresponding talk page otherwise
 
//
 
//
// Per https://www.mediawiki.org/wiki/Manual:Namespace#Subject_and_talk_namespaces
+
// Per http://www.mediawiki.org/wiki/Manual:Namespace#Subject_and_talk_namespaces
 
// * All discussion namespaces have odd-integer indices
 
// * All discussion namespaces have odd-integer indices
 
// * The discussion namespace index for a specific namespace with index n is n + 1
 
// * The discussion namespace index for a specific namespace with index n is n + 1
if (this.value === null) { return null; }
+
if (this.value===null) { return null; }
 
 
 
var namespaceId = this.namespaceId();
 
var namespaceId = this.namespaceId();
if (namespaceId>=0 && namespaceId % 2 === 0) //non-special and subject namespace
+
if (namespaceId>=0 && namespaceId % 2 == 0) //non-special and subject namespace
 
{
 
{
 
var localizedNamespace = mw.config.get('wgFormattedNamespaces')[namespaceId+1];
 
var localizedNamespace = mw.config.get('wgFormattedNamespaces')[namespaceId+1];
 
if (typeof localizedNamespace!=='undefined')
 
if (typeof localizedNamespace!=='undefined')
 
{
 
{
if (localizedNamespace === '') {
+
if (localizedNamespace==='') return this.value = this.stripNamespace();
this.value = this.stripNamespace();
+
this.value = localizedNamespace.split(' ').join('_') + ':' + this.stripNamespace();
} else {
 
this.value = localizedNamespace.split(' ').join('_') + ':' + this.stripNamespace();
 
}
 
 
return this.value;
 
return this.value;
 
}
 
}
Line 2,303: Line 2,451:
 
Title.prototype.toArticleFromTalkPage=function() {
 
Title.prototype.toArticleFromTalkPage=function() {
 
//largely copy/paste from toTalkPage above.
 
//largely copy/paste from toTalkPage above.
if (this.value === null) { return null; }
+
if (this.value===null) { return null; }
 
 
 
var namespaceId = this.namespaceId();
 
var namespaceId = this.namespaceId();
if (namespaceId >= 0 && namespaceId % 2 == 1) //non-special and talk namespace
+
if (namespaceId>=0 && namespaceId % 2 == 1) //non-special and talk namespace
 
{
 
{
 
var localizedNamespace = mw.config.get('wgFormattedNamespaces')[namespaceId-1];
 
var localizedNamespace = mw.config.get('wgFormattedNamespaces')[namespaceId-1];
 
if (typeof localizedNamespace!=='undefined')
 
if (typeof localizedNamespace!=='undefined')
 
{
 
{
if (localizedNamespace === '') {
+
if (localizedNamespace==='') return this.value = this.stripNamespace();
this.value = this.stripNamespace();
+
this.value = localizedNamespace.split(' ').join('_') + ':' + this.stripNamespace();
} else {
 
this.value = localizedNamespace.split(' ').join('_') + ':' + this.stripNamespace();
 
}
 
 
return this.value;
 
return this.value;
 
}
 
}
Line 2,342: Line 2,487:
 
if (n<0) { return this.value; }
 
if (n<0) { return this.value; }
 
var namespaceId = this.namespaceId();
 
var namespaceId = this.namespaceId();
if (namespaceId === pg.nsMainspaceId) return this.value;
+
if (namespaceId===pg.nsMainspaceId) return this.value;
 
return this.value.substring(n+1);
 
return this.value.substring(n+1);
 
};
 
};
Line 2,356: Line 2,501:
 
var anch=urlfrag.indexOf('#');
 
var anch=urlfrag.indexOf('#');
 
this.value=safeDecodeURI(urlfrag.substring(0,anch));
 
this.value=safeDecodeURI(urlfrag.substring(0,anch));
this.anchor=this.value.substring(anch+1);
+
this.anchor=value.substring(anch+1);
 
};
 
};
 
Title.prototype.append=function(x){
 
Title.prototype.append=function(x){
Line 2,362: Line 2,507:
 
};
 
};
 
Title.prototype.urlString=function(x) {
 
Title.prototype.urlString=function(x) {
if(!x) { x={}; }
+
x || ( x={} );
 
var v=this.toString(true);
 
var v=this.toString(true);
 
if (!x.omitAnchor && this.anchor) { v+= '#' + this.urlAnchor(); }
 
if (!x.omitAnchor && this.anchor) { v+= '#' + this.urlAnchor(); }
Line 2,374: Line 2,519:
 
return pg.wiki.titlebase + this.urlString();
 
return pg.wiki.titlebase + this.urlString();
 
};
 
};
 +
 +
 +
function paramValue(param, url) {
 +
var s=url.parenSplit(RegExp('[?&]' + literalizeRegex(param) + '=([^?&]*)'));
 +
if (!url) { return null; }
 +
return s[1] || null;
 +
}
  
 
function parseParams(url) {
 
function parseParams(url) {
 
var specialDiff = pg.re.specialdiff.exec(url);
 
var specialDiff = pg.re.specialdiff.exec(url);
if (specialDiff)
+
if (specialDiff!==null)
 
{
 
{
 
var split= specialDiff[1].split('/');
 
var split= specialDiff[1].split('/');
Line 2,406: Line 2,558:
 
ret.oldid = helper;
 
ret.oldid = helper;
 
}
 
}
 +
return ret;
 +
}
 +
 +
// all sorts of stuff here
 +
// FIXME almost everything needs to be rewritten
 +
 +
function oldidFromAnchor(a) { return paramValue('oldid', a.href); }
 +
//function diffFromAnchor(a) { return paramValue('diff', a.href); }
 +
 +
 +
function wikiMarkupToAddressFragment (str) { // for images
 +
var ret = safeDecodeURI(str);
 +
ret = ret.split(' ').join('_');
 +
ret = encodeURI(ret);
 
return ret;
 
return ret;
 
}
 
}
Line 2,433: Line 2,599:
  
 
//<NOLITE>
 
//<NOLITE>
 +
function isIpUser(user) {return pg.re.ipUser.test(user);}
 +
 
function isDisambig(data, article) {
 
function isDisambig(data, article) {
 
if (!getValueOf('popupAllDabsStubs') && article.namespace()) { return false; }
 
if (!getValueOf('popupAllDabsStubs') && article.namespace()) { return false; }
Line 2,457: Line 2,625:
  
 
function isInStrippableNamespace(article) {
 
function isInStrippableNamespace(article) {
// Does the namespace allow subpages
+
//I believe that this method means to return whether the given article is in a namspace without subpages. Meaning, it's broken.
// Note, would be better if we had access to wgNamespacesWithSubpages
+
return ( article.namespace() !== '' );
return ( article.namespaceId() !== 0 );
 
 
}
 
}
  
function isInMainNamespace(article) { return article.namespaceId() === 0; }
+
function isInMainNamespace(article) { return !isInStrippableNamespace(article); }
  
 
function anchorContainsImage(a) {
 
function anchorContainsImage(a) {
 
// iterate over children of anchor a
 
// iterate over children of anchor a
 
// see if any are images
 
// see if any are images
if (a === null) { return false; }
+
if (a===null) { return false; }
var kids=a.childNodes;
+
kids=a.childNodes;
 
for (var i=0; i<kids.length; ++i) { if (kids[i].nodeName=='IMG') { return true; } }
 
for (var i=0; i<kids.length; ++i) { if (kids[i].nodeName=='IMG') { return true; } }
 
return false;
 
return false;
Line 2,483: Line 2,650:
 
if (a.onmousedown || a.getAttribute('nopopup')) { return false; }
 
if (a.onmousedown || a.getAttribute('nopopup')) { return false; }
 
var h=a.href;
 
var h=a.href;
if (h === document.location.href+'#') { return false; }
+
if (h===document.location.href+'#') { return false; }
 
if (!pg.re.basenames.test(h)) { return false; }
 
if (!pg.re.basenames.test(h)) { return false; }
 
if (!pg.re.urlNoPopup.test(h)) { return true; }
 
if (!pg.re.urlNoPopup.test(h)) { return true; }
Line 2,510: Line 2,677:
 
}
 
}
 
// ENDFILE: titles.js
 
// ENDFILE: titles.js
 +
// STARTFILE: cookies.js
 +
//<NOLITE>
 +
//////////////////////////////////////////////////
 +
// Cookie handling
 +
// from http://www.quirksmode.org/js/cookies.html
 +
 +
var Cookie= {
 +
create: function(name,value,days)
 +
{
 +
var expires;
 +
if (days)
 +
{
 +
var date = new Date();
 +
date.setTime(date.getTime()+(days*24*60*60*1000));
 +
expires = "; expires="+date.toGMTString();
 +
}
 +
else { expires = ""; }
 +
document.cookie = name+"="+value+expires+"; path=/";
 +
},
 +
 +
read: function(name)
 +
{
 +
var nameEQ = name + "=";
 +
var ca = document.cookie.split(';');
 +
for(var i=0;i < ca.length;i++)
 +
{
 +
var c = ca[i];
 +
while (c.charAt(0)==' ') { c = c.substring(1,c.length); }
 +
if (c.indexOf(nameEQ) === 0) { return c.substring(nameEQ.length,c.length); }
 +
}
 +
return null;
 +
},
 +
 +
erase: function(name)
 +
{
 +
Cookie.create(name,"",-1);
 +
}
 +
};
 +
//</NOLITE>
 +
// ENDFILE: cookies.js
 
// STARTFILE: getpage.js
 
// STARTFILE: getpage.js
 
//////////////////////////////////////////////////
 
//////////////////////////////////////////////////
Line 2,517: Line 2,724:
 
// Schematic for a getWiki call
 
// Schematic for a getWiki call
 
//
 
//
//             getPageWithCaching
+
//   getWiki->-getPageWithCaching
 
// |
 
// |
 
//   false |   true
 
//   false |   true
Line 2,523: Line 2,730:
 
//  \.
 
//  \.
 
// (async)->addPageToCache(download)->-onComplete(download)
 
// (async)->addPageToCache(download)->-onComplete(download)
 +
 +
 +
/** @todo {document}
 +
@param {Title} article
 +
@param {Function} onComplete
 +
@param {integer} oldid
 +
@param {Navapopup} owner
 +
*/
 +
function getWiki(article, onComplete, oldid, owner) {
 +
// set ctype=text/css to get around opera gzip bug
 +
var url = pg.wiki.titlebase;
 +
if (article.namespaceId() >= 0)
 +
url += article.removeAnchor().urlString();
 +
if (oldid || oldid===0 || oldid==='0')
 +
url += '&oldid='+oldid;
 +
url += '&action=raw&ctype=text/css&maxage=0&smaxage=0';
 +
 +
getPageWithCaching(url, onComplete, owner);
 +
}
  
 
// check cache to see if page exists
 
// check cache to see if page exists
Line 2,529: Line 2,755:
 
log('getPageWithCaching, url='+url);
 
log('getPageWithCaching, url='+url);
 
var i=findInPageCache(url);
 
var i=findInPageCache(url);
var d;
 
 
if (i > -1) {
 
if (i > -1) {
d=fakeDownload(url, owner.idNumber, onComplete,
+
var d=fakeDownload(url, owner.idNumber, onComplete,
pg.cache.pages[i].data, pg.cache.pages[i].lastModified,
+
  pg.cache.pages[i].data, pg.cache.pages[i].lastModified,
owner);
+
  owner);
 
} else {
 
} else {
d=getPage(url, onComplete, owner);
+
var d=getPage(url, onComplete, owner);
 
if (d && owner && owner.addDownload) {
 
if (d && owner && owner.addDownload) {
 
owner.addDownload(d);
 
owner.addDownload(d);
Line 2,562: Line 2,787:
 
}
 
}
 
// ENDFILE: getpage.js
 
// ENDFILE: getpage.js
// STARTFILE: parensplit.js
+
// STARTFILE: md5-2.2alpha.js
//////////////////////////////////////////////////
+
//<NOLITE>
// parenSplit
+
/*
 +
* A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
 +
* Digest Algorithm, as defined in RFC 1321.
 +
* Version 2.2-alpha Copyright (C) Paul Johnston 1999 - 2005
 +
* Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
 +
* Distributed under the BSD License
 +
* See http://pajhome.org.uk/crypt/md5 for more info.
 +
*/
 +
 
 +
/*
 +
* Configurable variables. You may need to tweak these to be compatible with
 +
* the server-side, but the defaults work in most cases.
 +
*/
 +
var hexcase = 0;  /* hex output format. 0 - lowercase; 1 - uppercase */
 +
var b64pad  = ""; /* base-64 pad character. "=" for strict RFC compliance  */
 +
 
 +
/*
 +
* These are the functions you'll usually want to call
 +
* They take string arguments and return either hex or base-64 encoded strings
 +
*/
 +
function hex_md5(s) { return rstr2hex(rstr_md5(str2rstr_utf8(s))); }
 +
function b64_md5(s) { return rstr2b64(rstr_md5(str2rstr_utf8(s))); }
 +
function any_md5(s, e) { return rstr2any(rstr_md5(str2rstr_utf8(s)), e); }
 +
function hex_hmac_md5(k, d)
 +
  { return rstr2hex(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
 +
function b64_hmac_md5(k, d)
 +
  { return rstr2b64(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
 +
function any_hmac_md5(k, d, e)
 +
  { return rstr2any(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d)), e); }
  
// String.prototype.parenSplit should do what ECMAscript says String.prototype.split does,
+
/*
// interspersing paren matches (regex capturing groups) between the split elements.
+
* Perform a simple self-test to see if the VM is working
// i.e. 'abc'.split(/(b)/)) should return ['a','b','c'], not ['a','c']
+
*/
 +
function md5_vm_test()
 +
{
 +
  return hex_md5("abc") == "900150983cd24fb0d6963f7d28e17f72";
 +
}
  
if (String('abc'.split(/(b)/))!='a,b,c') {
+
/*
// broken String.split, e.g. konq, IE < 10
+
* Calculate the MD5 of a raw string
String.prototype.parenSplit=function (re) {
+
*/
re=nonGlobalRegex(re);
+
function rstr_md5(s)
var s=this;
+
{
var m=re.exec(s);
+
  return binl2rstr(binl_md5(rstr2binl(s), s.length * 8));
var ret=[];
+
}
while (m && s) {
+
 
// without the following loop, we have
+
/*
// 'ab'.parenSplit(/a|(b)/) != 'ab'.split(/a|(b)/)
+
* Calculate the HMAC-MD5, of a key and some data (raw strings)
for(var i=0; i<m.length; ++i) {
+
*/
if (typeof m[i]=='undefined') m[i]='';
+
function rstr_hmac_md5(key, data)
}
+
{
ret.push(s.substring(0,m.index));
+
  var bkey = rstr2binl(key);
ret = ret.concat(m.slice(1));
+
  if(bkey.length > 16) bkey = binl_md5(bkey, key.length * 8);
s=s.substring(m.index + m[0].length);
+
 
m=re.exec(s);
+
  var ipad = Array(16), opad = Array(16);
}
+
  for(var i = 0; i < 16; i++)
ret.push(s);
+
  {
return ret;
+
ipad[i] = bkey[i] ^ 0x36363636;
};
+
opad[i] = bkey[i] ^ 0x5C5C5C5C;
} else {
+
  }
String.prototype.parenSplit=function (re) { return this.split(re); };
+
 
String.prototype.parenSplit.isNative=true;
+
  var hash = binl_md5(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
 +
  return binl2rstr(binl_md5(opad.concat(hash), 512 + 128));
 +
}
 +
 
 +
/*
 +
* Convert a raw string to a hex string
 +
*/
 +
function rstr2hex(input)
 +
{
 +
  var hex_tab = hexcase ? "0123456789ABCDEF" : "0123456789abcdef";
 +
  var output = "";
 +
  var x;
 +
  for(var i = 0; i < input.length; i++)
 +
  {
 +
x = input.charCodeAt(i);
 +
output += hex_tab.charAt((x >>> 4) & 0x0F)
 +
  +  hex_tab.charAt( x   & 0x0F);
 +
  }
 +
  return output;
 
}
 
}
  
function nonGlobalRegex(re) {
+
/*
var s=re.toString();
+
* Convert a raw string to a base-64 string
var flags='';
+
*/
for (var j=s.length; s.charAt(j) != '/'; --j) {
+
function rstr2b64(input)
if (s.charAt(j) != 'g') { flags += s.charAt(j); }
+
{
 +
  var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
 +
  var output = "";
 +
  var len = input.length;
 +
  for(var i = 0; i < len; i += 3)
 +
  {
 +
var triplet = (input.charCodeAt(i) << 16)
 +
| (i + 1 < len ? input.charCodeAt(i+1) << 8 : 0)
 +
| (i + 2 < len ? input.charCodeAt(i+2) : 0);
 +
for(var j = 0; j < 4; j++)
 +
{
 +
  if(i * 8 + j * 6 > input.length * 8) output += b64pad;
 +
  else output += tab.charAt((triplet >>> 6*(3-j)) & 0x3F);
 
}
 
}
var t=s.substring(1,j);
+
  }
return RegExp(t,flags);
+
  return output;
 
}
 
}
// ENDFILE: parensplit.js
 
// STARTFILE: tools.js
 
// IE madness with encoding
 
// ========================
 
//
 
// suppose throughout that the page is in utf8, like wikipedia
 
//
 
// if a is an anchor DOM element and a.href should consist of
 
//
 
// http://host.name.here/wiki/foo?bar=baz
 
//
 
// then IE gives foo as "latin1-encoded" utf8; we have foo = decode_utf8(decodeURI(foo_ie))
 
// but IE gives bar=baz correctly as plain utf8
 
//
 
// ---------------------------------
 
//
 
// IE's xmlhttp doesn't understand utf8 urls. Have to use encodeURI here.
 
//
 
// ---------------------------------
 
//
 
// summat else
 
  
// Source: http://aktuell.de.selfhtml.org/artikel/javascript/utf8b64/utf8.htm
+
/*
 +
* Convert a raw string to an arbitrary string encoding
 +
*/
 +
function rstr2any(input, encoding)
 +
{
 +
  var divisor = encoding.length;
 +
  var remainders = Array();
 +
  var i, q, x, quotient;
 +
 
 +
  /* Convert to an array of 16-bit big-endian values, forming the dividend */
 +
  var dividend = Array(input.length / 2);
 +
  for(i = 0; i < dividend.length; i++)
 +
  {
 +
dividend[i] = (input.charCodeAt(i * 2) << 8) | input.charCodeAt(i * 2 + 1);
 +
  }
  
//<NOLITE>
+
  /*
 +
  * Repeatedly perform a long division. The binary array forms the dividend,
 +
  * the length of the encoding is the divisor. Once computed, the quotient
 +
  * forms the dividend for the next step. We stop when the dividend is zero.
 +
  * All remainders are stored for later use.
 +
  */
 +
  while(dividend.length > 0)
 +
  {
 +
quotient = Array();
 +
x = 0;
 +
for(i = 0; i < dividend.length; i++)
 +
{
 +
  x = (x << 16) + dividend[i];
 +
  q = Math.floor(x / divisor);
 +
  x -= q * divisor;
 +
  if(quotient.length > 0 || q > 0)
 +
quotient[quotient.length] = q;
 +
}
 +
remainders[remainders.length] = x;
 +
dividend = quotient;
 +
  }
  
 +
  /* Convert the remainders to the output string */
 +
  var output = "";
 +
  for(i = remainders.length - 1; i >= 0; i--)
 +
output += encoding.charAt(remainders[i]);
  
function getJsObj(json) {
+
  return output;
try {
 
var json_ret = JSON.parse(json);
 
if( json_ret.warnings ) {
 
for( var w=0; w < json_ret.warnings.length; w++ ) {
 
if( json_ret.warnings[w]['*'] ) {
 
log( json_ret.warnings[w]['*'] );
 
} else {
 
log( json_ret.warnings[w]['warnings'] );
 
}
 
}
 
} else if ( json_ret.error ) {
 
errlog( json_ret.error.code + ': ' + json_ret.error.info );
 
}
 
return json_ret;
 
} catch (someError) {
 
errlog('Something went wrong with getJsObj, json='+json);
 
return 1;
 
}
 
 
}
 
}
  
function anyChild(obj) {
+
/*
for (var p in obj) {
+
* Encode a string as utf-8.
return obj[p];
+
* For efficiency, this assumes the input is valid utf-16.
 +
*/
 +
function str2rstr_utf8(input)
 +
{
 +
  var output = "";
 +
  var i = -1;
 +
  var x, y;
 +
 
 +
  while(++i < input.length)
 +
  {
 +
/* Decode utf-16 surrogate pairs */
 +
x = input.charCodeAt(i);
 +
y = i + 1 < input.length ? input.charCodeAt(i + 1) : 0;
 +
if(0xD800 <= x && x <= 0xDBFF && 0xDC00 <= y && y <= 0xDFFF)
 +
{
 +
  x = 0x10000 + ((x & 0x03FF) << 10) + (y & 0x03FF);
 +
  i++;
 
}
 
}
return null;
 
}
 
  
//</NOLITE>
+
/* Encode output as utf-8 */
 
+
if(x <= 0x7F)
function upcaseFirst(str) {
+
  output += String.fromCharCode(x);
if (typeof str != typeof '' || str === '') return '';
+
else if(x <= 0x7FF)
return str.charAt(0).toUpperCase() + str.substring(1);
+
  output += String.fromCharCode(0xC0 | ((x >>> 6 ) & 0x1F),
 +
0x80 | ( x   & 0x3F));
 +
else if(x <= 0xFFFF)
 +
  output += String.fromCharCode(0xE0 | ((x >>> 12) & 0x0F),
 +
0x80 | ((x >>> 6 ) & 0x3F),
 +
0x80 | ( x   & 0x3F));
 +
else if(x <= 0x1FFFFF)
 +
  output += String.fromCharCode(0xF0 | ((x >>> 18) & 0x07),
 +
0x80 | ((x >>> 12) & 0x3F),
 +
0x80 | ((x >>> 6 ) & 0x3F),
 +
0x80 | ( x   & 0x3F));
 +
  }
 +
  return output;
 
}
 
}
  
 +
/*
 +
* Encode a string as utf-16
 +
*/
 +
function str2rstr_utf16le(input)
 +
{
 +
  var output = "";
 +
  for(var i = 0; i < input.length; i++)
 +
output += String.fromCharCode( input.charCodeAt(i)   & 0xFF,
 +
  (input.charCodeAt(i) >>> 8) & 0xFF);
 +
  return output;
 +
}
  
function findInArray(arr, foo) {
+
function str2rstr_utf16be(input)
if (!arr || !arr.length) { return -1; }
+
{
var len=arr.length;
+
  var output = "";
for (var i=0; i<len; ++i) { if (arr[i]==foo) { return i; } }
+
  for(var i = 0; i < input.length; i++)
return -1;
+
output += String.fromCharCode((input.charCodeAt(i) >>> 8) & 0xFF,
 +
  input.charCodeAt(i)   & 0xFF);
 +
  return output;
 
}
 
}
  
/* eslint-disable no-unused-vars */
+
/*
function nextOne (array, value) {
+
* Convert a raw string to an array of little-endian words
// NB if the array has two consecutive entries equal
+
* Characters >255 have their high-byte silently ignored.
// then this will loop on successive calls
+
*/
var i=findInArray(array, value);
+
function rstr2binl(input)
if (i<0) { return null; }
+
{
return array[i+1];
+
  var output = Array(input.length >> 2);
 +
  for(var i = 0; i < output.length; i++)
 +
output[i] = 0;
 +
  for(var i = 0; i < input.length * 8; i += 8)
 +
output[i>>5] |= (input.charCodeAt(i / 8) & 0xFF) << (i%32);
 +
  return output;
 
}
 
}
/* eslint-enable no-unused-vars */
 
  
function literalizeRegex(str){
+
/*
return mw.util.escapeRegExp(str);
+
* Convert an array of little-endian words to a string
 +
*/
 +
function binl2rstr(input)
 +
{
 +
  var output = "";
 +
  for(var i = 0; i < input.length * 32; i += 8)
 +
output += String.fromCharCode((input[i>>5] >>> (i % 32)) & 0xFF);
 +
  return output;
 
}
 
}
  
String.prototype.entify=function() {
+
/*
//var shy='&shy;';
+
* Calculate the MD5 of an array of little-endian words, and a bit length.
return this.split('&').join('&amp;').split('<').join('&lt;').split('>').join('&gt;'/*+shy*/).split('"').join('&quot;');
+
*/
};
+
function binl_md5(x, len)
 +
{
 +
  /* append padding */
 +
  x[len >> 5] |= 0x80 << ((len) % 32);
 +
  x[(((len + 64) >>> 9) << 4) + 14] = len;
 +
 
 +
  var a =  1732584193;
 +
  var b = -271733879;
 +
  var c = -1732584194;
 +
  var d =  271733878;
 +
 
 +
  for(var i = 0; i < x.length; i += 16)
 +
  {
 +
var olda = a;
 +
var oldb = b;
 +
var oldc = c;
 +
var oldd = d;
 +
 
 +
a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936);
 +
d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586);
 +
c = md5_ff(c, d, a, b, x[i+ 2], 17, 606105819);
 +
b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330);
 +
a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897);
 +
d = md5_ff(d, a, b, c, x[i+ 5], 12, 1200080426);
 +
c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341);
 +
b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983);
 +
a = md5_ff(a, b, c, d, x[i+ 8], 7 , 1770035416);
 +
d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417);
 +
c = md5_ff(c, d, a, b, x[i+10], 17, -42063);
 +
b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162);
 +
a = md5_ff(a, b, c, d, x[i+12], 7 , 1804603682);
 +
d = md5_ff(d, a, b, c, x[i+13], 12, -40341101);
 +
c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290);
 +
b = md5_ff(b, c, d, a, x[i+15], 22, 1236535329);
 +
 
 +
a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510);
 +
d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632);
 +
c = md5_gg(c, d, a, b, x[i+11], 14, 643717713);
 +
b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302);
 +
a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691);
 +
d = md5_gg(d, a, b, c, x[i+10], 9 , 38016083);
 +
c = md5_gg(c, d, a, b, x[i+15], 14, -660478335);
 +
b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848);
 +
a = md5_gg(a, b, c, d, x[i+ 9], 5 , 568446438);
 +
d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690);
 +
c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961);
 +
b = md5_gg(b, c, d, a, x[i+ 8], 20, 1163531501);
 +
a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467);
 +
d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784);
 +
c = md5_gg(c, d, a, b, x[i+ 7], 14, 1735328473);
 +
b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734);
 +
 
 +
a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558);
 +
d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463);
 +
c = md5_hh(c, d, a, b, x[i+11], 16, 1839030562);
 +
b = md5_hh(b, c, d, a, x[i+14], 23, -35309556);
 +
a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060);
 +
d = md5_hh(d, a, b, c, x[i+ 4], 11, 1272893353);
 +
c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632);
 +
b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640);
 +
a = md5_hh(a, b, c, d, x[i+13], 4 , 681279174);
 +
d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222);
 +
c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979);
 +
b = md5_hh(b, c, d, a, x[i+ 6], 23, 76029189);
 +
a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487);
 +
d = md5_hh(d, a, b, c, x[i+12], 11, -421815835);
 +
c = md5_hh(c, d, a, b, x[i+15], 16, 530742520);
 +
b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651);
  
// Array filter function
+
a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844);
function removeNulls(val) { return val !== null; }
+
d = md5_ii(d, a, b, c, x[i+ 7], 10, 1126891415);
 +
c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905);
 +
b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055);
 +
a = md5_ii(a, b, c, d, x[i+12], 6 , 1700485571);
 +
d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606);
 +
c = md5_ii(c, d, a, b, x[i+10], 15, -1051523);
 +
b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799);
 +
a = md5_ii(a, b, c, d, x[i+ 8], 6 , 1873313359);
 +
d = md5_ii(d, a, b, c, x[i+15], 10, -30611744);
 +
c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380);
 +
b = md5_ii(b, c, d, a, x[i+13], 21, 1309151649);
 +
a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070);
 +
d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379);
 +
c = md5_ii(c, d, a, b, x[i+ 2], 15, 718787259);
 +
b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551);
  
function joinPath(list) {
+
a = safe_add(a, olda);
return list.filter(removeNulls).join('/');
+
b = safe_add(b, oldb);
 +
c = safe_add(c, oldc);
 +
d = safe_add(d, oldd);
 +
  }
 +
  return Array(a, b, c, d);
 
}
 
}
  
 +
/*
 +
* These functions implement the four basic operations the algorithm uses.
 +
*/
 +
function md5_cmn(q, a, b, x, s, t)
 +
{
 +
  return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s),b);
 +
}
 +
function md5_ff(a, b, c, d, x, s, t)
 +
{
 +
  return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t);
 +
}
 +
function md5_gg(a, b, c, d, x, s, t)
 +
{
 +
  return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t);
 +
}
 +
function md5_hh(a, b, c, d, x, s, t)
 +
{
 +
  return md5_cmn(b ^ c ^ d, a, b, x, s, t);
 +
}
 +
function md5_ii(a, b, c, d, x, s, t)
 +
{
 +
  return md5_cmn(c ^ (b | (~d)), a, b, x, s, t);
 +
}
  
function simplePrintf(str, subs) {
+
/*
if (!str || !subs) { return str; }
+
* Add integers, wrapping at 2^32. This uses 16-bit operations internally
var ret=[];
+
* to work around bugs in some JS interpreters.
var s=str.parenSplit(/(%s|\$[0-9]+)/);
+
*/
var i=0;
+
function safe_add(x, y)
do {
+
{
ret.push(s.shift());
+
  var lsw = (x & 0xFFFF) + (y & 0xFFFF);
if ( !s.length ) { break; }
+
  var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
var cmd=s.shift();
+
  return (msw << 16) | (lsw & 0xFFFF);
if (cmd == '%s') {
 
if ( i < subs.length ) { ret.push(subs[i]); } else { ret.push(cmd); }
 
++i;
 
} else {
 
var j=parseInt( cmd.replace('$', ''), 10 ) - 1;
 
if ( j > -1 && j < subs.length ) { ret.push(subs[j]); } else { ret.push(cmd); }
 
}
 
} while (s.length > 0);
 
return ret.join('');
 
 
}
 
}
/* eslint-disable no-unused-vars */
 
function isString(x) { return (typeof x === 'string' || x instanceof String); }
 
function isNumber(x) { return (typeof x === 'number' || x instanceof Number); }
 
function isRegExp(x) { return x instanceof RegExp; }
 
function isArray (x) { return x instanceof Array; }
 
function isObject(x) { return x instanceof Object; }
 
function isFunction(x) {
 
return !isRegExp(x) && ($.isFunction(x) || x instanceof Function);
 
}
 
/* eslint-enable no-unused-vars */
 
  
function repeatString(s,mult) {
+
/*
var ret='';
+
* Bitwise rotate a 32-bit number to the left.
for (var i=0; i<mult; ++i) { ret += s; }
+
*/
return ret;
+
function bit_rol(num, cnt)
 +
{
 +
  return (num << cnt) | (num >>> (32 - cnt));
 
}
 
}
 +
//</NOLITE>
 +
// ENDFILE: md5-2.2alpha.js
 +
// STARTFILE: parensplit.js
 +
//////////////////////////////////////////////////
 +
// parenSplit
  
function zeroFill(s, min) {
+
// String.prototype.parenSplit should do what ECMAscript says String.prototype.split does,
min = min || 2;
+
// interspersing paren matches (regex capturing groups) between the split elements.
var t=s.toString();
+
// i.e. 'abc'.split(/(b)/)) should return ['a','b','c'], not ['a','c']
return repeatString('0', min - t.length) + t;
 
}
 
  
function map(f, o) {
+
if (String('abc'.split(/(b)/))!='a,b,c') {
if (isArray(o)) { return map_array(f,o); }
+
// broken String.split, e.g. konq, IE
return map_object(f,o);
+
String.prototype.parenSplit=function (re) {
 +
re=nonGlobalRegex(re);
 +
var s=this;
 +
var m=re.exec(s);
 +
var ret=[];
 +
while (m && s) {
 +
// without the following loop, we have
 +
// 'ab'.parenSplit(/a|(b)/) != 'ab'.split(/a|(b)/)
 +
for(var i=0; i<m.length; ++i) {
 +
if (typeof m[i]=='undefined') m[i]='';
 +
}
 +
ret.push(s.substring(0,m.index));
 +
ret = ret.concat(m.slice(1));
 +
s=s.substring(m.index + m[0].length);
 +
m=re.exec(s);
 +
}
 +
ret.push(s);
 +
return ret;
 +
};
 +
} else {
 +
String.prototype.parenSplit=function (re) { return this.split(re); };
 +
String.prototype.parenSplit.isNative=true;
 
}
 
}
function map_array(f,o) {
+
 
var ret=[];
+
function nonGlobalRegex(re) {
for (var i=0; i<o.length; ++i) {
+
var s=re.toString();
ret.push(f(o[i]));
+
flags='';
 +
for (var j=s.length; s.charAt(j) != '/'; --j) {
 +
if (s.charAt(j) != 'g') { flags += s.charAt(j); }
 
}
 
}
return ret;
+
var t=s.substring(1,j);
 +
return RegExp(t,flags);
 
}
 
}
function map_object(f,o) {
+
// ENDFILE: parensplit.js
var ret={};
+
// STARTFILE: tools.js
for (var i in o) { ret[o]=f(o[i]); }
+
// IE madness with encoding
return ret;
+
// ========================
}
 
 
 
pg.escapeQuotesHTML = function ( text ) {
 
return text
 
.replace(/&/g, "&amp;")
 
.replace(/"/g, "&quot;")
 
.replace(/</g, "&lt;")
 
.replace(/>/g, "&gt;");
 
};
 
 
 
// ENDFILE: tools.js
 
// STARTFILE: dab.js
 
//<NOLITE>
 
//////////////////////////////////////////////////
 
// Dab-fixing code
 
 
//
 
//
 +
// suppose throughout that the page is in utf8, like wikipedia
 +
//
 +
// if a is an anchor DOM element and a.href should consist of
 +
//
 +
// http://host.name.here/wiki/foo?bar=baz
 +
//
 +
// then IE gives foo as "latin1-encoded" utf8; we have foo = decode_utf8(decodeURI(foo_ie))
 +
// but IE gives bar=baz correctly as plain utf8
 +
//
 +
// ---------------------------------
 +
//
 +
// IE's xmlhttp doesn't understand utf8 urls. Have to use encodeURI here.
 +
//
 +
// ---------------------------------
 +
//
 +
// summat else
  
 +
// Source: http://aktuell.de.selfhtml.org/artikel/javascript/utf8b64/utf8.htm
  
function retargetDab(newTarget, oldTarget, friendlyCurrentArticleName, titleToEdit) {
+
//<NOLITE>
log('retargetDab: newTarget='+newTarget + ' oldTarget=' + oldTarget);
+
function encode_utf8(rohtext) {
return changeLinkTargetLink(
+
// dient der Normalisierung des Zeilenumbruchs
{newTarget: newTarget,
+
rohtext = rohtext.replace(/\r\n/g,"\n");
text: newTarget.split(' ').join('&nbsp;'),
+
var utftext = "";
hint: tprintf('disambigHint', [newTarget]),
+
for(var n=0; n<rohtext.length; n++)
summary: simplePrintf(
+
{
getValueOf('popupFixDabsSummary'), [friendlyCurrentArticleName, newTarget ]),
+
// ermitteln des Unicodes des  aktuellen Zeichens
clickButton: getValueOf('popupDabsAutoClick'), minor: true, oldTarget: oldTarget,
+
var c=rohtext.charCodeAt(n);
watch: getValueOf('popupWatchDisambiggedPages'),
+
// alle Zeichen von 0-127 => 1byte
title: titleToEdit});
+
if (c<128)
 +
utftext += String.fromCharCode(c);
 +
// alle Zeichen von 127 bis 2047 => 2byte
 +
else if((c>127) && (c<2048)) {
 +
utftext += String.fromCharCode((c>>6)|192);
 +
utftext += String.fromCharCode((c&63)|128);}
 +
// alle Zeichen von 2048 bis 66536 => 3byte
 +
else {
 +
utftext += String.fromCharCode((c>>12)|224);
 +
utftext += String.fromCharCode(((c>>6)&63)|128);
 +
utftext += String.fromCharCode((c&63)|128);}
 +
}
 +
return utftext;
 
}
 
}
  
function listLinks(wikitext, oldTarget, titleToEdit) {
+
function getJsObj(json) {
// mediawiki strips trailing spaces, so we do the same
+
try {
// testcase: https://en.wikipedia.org/w/index.php?title=Radial&oldid=97365633
+
var json_ret = JSON.parse(json);
var reg=RegExp('\\[\\[([^|]*?) *(\\||\\]\\])', 'gi');
+
} catch (someError) {
var ret=[];
+
errlog('Something went wrong with getJsobj, json='+json);
var splitted=wikitext.parenSplit(reg);
+
return 1;
// ^[a-z]+ should match interwiki links, hopefully (case-insensitive)
+
}
// and ^[a-z]* should match those and [[:Category...]] style links too
+
if( json_ret['warnings'] ) {
var omitRegex=RegExp('^[a-z]*:|^[Ss]pecial:|^[Ii]mage|^[Cc]ategory');
+
for( var w=0; w < json_ret['warnings'].length; w++ ) {
var friendlyCurrentArticleName= oldTarget.toString();
+
log( json_ret['warnings'][w]['*'] );
var wikPos = getValueOf('popupDabWiktionary');
+
}
 +
} else if ( json_ret['error'] ) {
 +
errlog( json_ret['error'].code + ': ' + json_ret['error'].info );
 +
}
 +
return json_ret;
 +
}
  
for (var i=1; i<splitted.length; i=i+3) {
+
function anyChild(obj) {
if (typeof splitted[i] == typeof 'string' && splitted[i].length>0 && !omitRegex.test(splitted[i])) {
+
for (var p in obj) {
ret.push( retargetDab(splitted[i], oldTarget, friendlyCurrentArticleName, titleToEdit) );
+
return obj[p];
} /* if */
+
}
} /* for loop */
+
return null;
 +
}
  
ret = rmDupesFromSortedList(ret.sort());
+
//</NOLITE>
  
if (wikPos) {
+
function decode_utf8(utftext) {
var wikTarget='wiktionary:' +
+
var plaintext = ""; var i=0, c=0, c1=0, c2=0;
friendlyCurrentArticleName.replace( RegExp('^(.+)\\s+[(][^)]+[)]\\s*$'), '$1' );
+
// while-Schleife, weil einige Zeichen uebersprungen werden
 
+
while(i<utftext.length)
var meth;
+
{
if (wikPos.toLowerCase() == 'first') { meth = 'unshift'; }
+
c = utftext.charCodeAt(i);
else { meth = 'push'; }
+
if (c<128) {
 
+
plaintext += String.fromCharCode(c);
ret[meth]( retargetDab(wikTarget, oldTarget, friendlyCurrentArticleName, titleToEdit) );
+
i++;}
 +
else if((c>191) && (c<224)) {
 +
c2 = utftext.charCodeAt(i+1);
 +
plaintext += String.fromCharCode(((c&31)<<6) | (c2&63));
 +
i+=2;}
 +
else {
 +
c2 = utftext.charCodeAt(i+1); c3 = utftext.charCodeAt(i+2);
 +
plaintext += String.fromCharCode(((c&15)<<12) | ((c2&63)<<6) | (c3&63));
 +
i+=3;}
 
}
 
}
 
+
return plaintext;
ret.push(changeLinkTargetLink(
 
{ newTarget: null,
 
text: popupString('remove this link').split(' ').join('&nbsp;'),
 
hint: popupString("remove all links to this disambig page from this article"),
 
clickButton: getValueOf('popupDabsAutoClick'), oldTarget: oldTarget,
 
summary: simplePrintf(getValueOf('popupRmDabLinkSummary'), [friendlyCurrentArticleName]),
 
watch: getValueOf('popupWatchDisambiggedPages'),
 
title: titleToEdit
 
}));
 
return ret;
 
 
}
 
}
  
function rmDupesFromSortedList(list) {
 
var ret=[];
 
for (var i=0; i<list.length; ++i) {
 
if (ret.length === 0 || list[i]!=ret[ret.length-1]) { ret.push(list[i]); }
 
}
 
return ret;
 
}
 
  
function makeFixDab(data, navpop) {
+
function upcaseFirst(str) {
// grab title from parent popup if there is one; default exists in changeLinkTargetLink
+
if (typeof str != typeof '' || str=='') return '';
var titleToEdit=(navpop.parentPopup && navpop.parentPopup.article.toString());
+
return str.charAt(0).toUpperCase() + str.substring(1);
var list=listLinks(data, navpop.originalArticle, titleToEdit);
 
if (list.length === 0) { log('listLinks returned empty list'); return null; }
 
var html='<hr />' + popupString('Click to disambiguate this link to:') + '<br>';
 
html+=list.join(', ');
 
return html;
 
 
}
 
}
  
  
function makeFixDabs(wikiText, navpop) {
+
function findInArray(arr, foo) {
if (getValueOf('popupFixDabs') && isDisambig(wikiText, navpop.article) &&
+
if (!arr || !arr.length) { return -1; }
Title.fromURL(location.href).namespaceId() != pg.nsSpecialId &&
+
var len=arr.length;
navpop.article.talkPage() ) {
+
for (var i=0; i<len; ++i) { if (arr[i]==foo) { return i; } }
setPopupHTML(makeFixDab(wikiText, navpop), 'popupFixDab', navpop.idNumber);
+
return -1;
}
+
}
 +
 
 +
function nextOne (array, value) {
 +
// NB if the array has two consecutive entries equal
 +
// then this will loop on successive calls
 +
var i=findInArray(array, value);
 +
if (i<0) { return null; }
 +
return array[i+1];
 
}
 
}
  
function popupRedlinkHTML(article) {
+
function literalizeRegex(str){
return changeLinkTargetLink(
+
return str.replace(RegExp('([-.|()\\\\+?*^${}\\[\\]])', 'g'), '\\$1');
{ newTarget: null, text: popupString('remove this link').split(' ').join('&nbsp;'),
 
hint: popupString("remove all links to this page from this article"),
 
clickButton: getValueOf('popupRedlinkAutoClick'),
 
oldTarget: article.toString(),
 
summary: simplePrintf(getValueOf('popupRedlinkSummary'), [article.toString()])});
 
 
}
 
}
//</NOLITE>
 
// ENDFILE: dab.js
 
// STARTFILE: htmloutput.js
 
  
// this has to use a timer loop as we don't know if the DOM element exists when we want to set the text
+
String.prototype.entify=function() {
function setPopupHTML (str, elementId, popupId, onSuccess, append) {
+
//var shy='&shy;';
if (typeof popupId === 'undefined') {
+
return this.split('&').join('&amp;').split('<').join('&lt;').split('>').join('&gt;'/*+shy*/).split('"').join('&quot;');
//console.error('popupId is not defined in setPopupHTML, html='+str.substring(0,100));
+
};
popupId = pg.idNumber;
+
 
 +
function findThis(array, value) {
 +
if (typeof array.length == 'undefined') { return null; }
 +
for (var i=0; i<array.length; ++i) {
 +
if (array[i]==value) { return i; }
 
}
 
}
 +
return null;
 +
}
  
var popupElement=document.getElementById(elementId+popupId);
+
function removeNulls(list) {
if (popupElement) {
+
var ret=[];
if (!append) { popupElement.innerHTML=''; }
+
for (var i=0; i<list.length; ++i) {
if (isString(str)) {
+
if (list[i]) {
popupElement.innerHTML+=str;
+
ret.push(list[i]);
} else {
 
popupElement.appendChild(str);
 
 
}
 
}
if (onSuccess) { onSuccess(); }
+
}
setTimeout(checkPopupPosition, 100);
+
return ret;
return true;
+
}
} else {
+
function joinPath(list) {
// call this function again in a little while...
+
return removeNulls(list).join('/');
setTimeout(function(){
 
setPopupHTML(str,elementId,popupId,onSuccess);
 
}, 600);
 
}
 
return null;
 
 
}
 
}
  
//<NOLITE>
 
function setPopupTrailer(str,id) {return setPopupHTML(str, 'popupData', id);}
 
//</NOLITE>
 
  
// args.navpopup is mandatory
+
function simplePrintf(str, subs) {
// optional: args.redir, args.redirTarget
+
if (!str || !subs) { return str; }
// FIXME: ye gods, this is ugly stuff
+
var ret=[];
function fillEmptySpans(args) {  
+
var s=str.parenSplit(/(%s|\$[0-9]+)/);
// if redir is present and true then redirTarget is mandatory
+
var i=0;
var redir=true;
+
do {
var rcid;
+
ret.push(s.shift());
if (typeof args != 'object' || typeof args.redir == 'undefined' || !args.redir) { redir=false; }
+
if ( !s.length ) { break; }
var a=args.navpopup.parentAnchor;
+
var cmd=s.shift();
 +
if (cmd == '%s') {
 +
if ( i < subs.length ) { ret.push(subs[i]); } else { ret.push(cmd); }
 +
++i;
 +
} else {
 +
var j=parseInt( cmd.replace('$', ''), 10 ) - 1;
 +
if ( j > -1 && j < subs.length ) { ret.push(subs[j]); } else { ret.push(cmd); }
 +
}
 +
} while (s.length > 0);
 +
return ret.join('');
 +
}
  
var article, hint=null, oldid=null, params={};
+
function max(a,b){return a<b ? b : a;}
if (redir && typeof args.redirTarget == typeof {}) {
+
function min(a,b){return a>b ? b : a;}
article=args.redirTarget;
 
//hint=article.hintValue();
 
} else {
 
article=(new Title()).fromAnchor(a);
 
hint=a.originalTitle || article.hintValue();
 
params=parseParams(a.href);
 
oldid=(getValueOf('popupHistoricalLinks')) ? params.oldid : null;
 
rcid=params.rcid;
 
}
 
var x={ article:article, hint: hint, oldid: oldid, rcid: rcid, navpop:args.navpopup, params:params };
 
  
var structure=pg.structures[getValueOf('popupStructure')];
+
function isString(x) { return (typeof x === 'string' || x instanceof String); }
if (typeof structure != 'object') {
+
//function isNumber(x) { return (typeof x === 'number' || x instanceof Number); }
setPopupHTML('popupError', 'Unknown structure (this should never happen): '+
+
function isRegExp(x) { return x instanceof RegExp; }
pg.option.popupStructure, args.navpopup.idNumber);
+
function isArray (x) { return x instanceof Array; }
return;
+
function isObject(x) { return x instanceof Object; }
}
+
function isFunction(x) {
var spans=flatten(pg.misc.layout);
+
return !isRegExp(x) && ($.isFunction(x) || x instanceof Function);
var numspans = spans.length;
 
var redirs=pg.misc.redirSpans;
 
 
 
for (var i=0; i<numspans; ++i) {
 
var found = redirs && (redirs.indexOf( spans[i] ) !== -1);
 
//log('redir='+redir+', found='+found+', spans[i]='+spans[i]);
 
if ( (found && !redir) || (!found && redir) ) {
 
//log('skipping this set of the loop');
 
continue;
 
}
 
var structurefn=structure[spans[i]];
 
var setfn = setPopupHTML;
 
if (getValueOf('popupActiveNavlinks') &&
 
(spans[i].indexOf('popupTopLinks')===0 || spans[i].indexOf('popupRedirTopLinks')===0)
 
) {
 
setfn = setPopupTipsAndHTML;
 
}
 
switch (typeof structurefn) {
 
case 'function':
 
log('running '+spans[i]+'({article:'+x.article+', hint:'+x.hint+', oldid: '+x.oldid+'})');
 
setfn(structurefn(x), spans[i], args.navpopup.idNumber);
 
break;
 
case 'string':
 
setfn(structurefn, spans[i], args.navpopup.idNumber);
 
break;
 
default:
 
errlog('unknown thing with label '+spans[i] + ' (span index was ' + i + ')');
 
break;
 
}
 
}
 
 
}
 
}
  
// flatten an array
+
function repeatString(s,mult) {
function flatten(list, start) {
+
var ret='';
var ret=[];
+
for (var i=0; i<mult; ++i) { ret += s; }
if (typeof start == 'undefined') { start=0; }
 
for (var i=start; i<list.length; ++i) {
 
if (typeof list[i] == typeof []) {
 
return ret.concat(flatten(list[i])).concat(flatten(list, i+1));
 
}
 
else { ret.push(list[i]); }
 
}
 
 
return ret;
 
return ret;
 
}
 
}
  
// Generate html for whole popup
+
function zeroFill(s, min) {
function popupHTML (a) {
+
min = min || 2;
getValueOf('popupStructure');
+
var t=s.toString();
var structure=pg.structures[pg.option.popupStructure];
+
return repeatString('0', min - t.length) + t;
if (typeof structure != 'object') {
 
//return 'Unknown structure: '+pg.option.popupStructure;
 
// override user choice
 
pg.option.popupStructure=pg.optionDefault.popupStructure;
 
return popupHTML(a);
 
}
 
if (typeof structure.popupLayout != 'function') { return 'Bad layout'; }
 
pg.misc.layout=structure.popupLayout();
 
if ($.isFunction(structure.popupRedirSpans)) { pg.misc.redirSpans=structure.popupRedirSpans(); }
 
else { pg.misc.redirSpans=[]; }
 
return makeEmptySpans(pg.misc.layout, a.navpopup);
 
 
}
 
}
  
function makeEmptySpans (list, navpop) {
+
function map(f, o) {
var ret='';
+
if (isArray(o)) { return map_array(f,o); }
for (var i=0; i<list.length; ++i) {
+
return map_object(f,o);
if (typeof list[i] == typeof '') {
+
}
ret += emptySpanHTML(list[i], navpop.idNumber, 'div');
+
function map_array(f,o) {
} else if (typeof list[i] == typeof [] && list[i].length > 0 ) {
+
var ret=[];
ret = ret.parenSplit(RegExp('(</[^>]*?>$)')).join(makeEmptySpans(list[i], navpop));
+
for (var i=0; i<o.length; ++i) {
} else if (typeof list[i] == typeof {} && list[i].nodeType ) {
+
ret.push(f(o[i]));
ret += emptySpanHTML(list[i].name, navpop.idNumber, list[i].nodeType);
 
}
 
 
}
 
}
 +
return ret;
 +
}
 +
function map_object(f,o) {
 +
var ret={};
 +
for (var i in o) { ret[o]=f(o[i]); }
 
return ret;
 
return ret;
 
}
 
}
  
 +
pg.escapeQuotesHTML = function ( text ) { var re = new RegExp( '&', "g" ); text = text.replace( re, "&amp;" ); re = new RegExp( '"', "g" ); text = text.replace( re, "&quot;" ); re = new RegExp( '<', "g" ); text = text.replace( re, "&lt;" ); re = new RegExp( '>', "g" ); text = text.replace( re, "&gt;" ); return text;}
  
function emptySpanHTML(name, id, tag, classname) {
+
pg.jsescape = function(s)
tag = tag || 'span';
+
{
if (!classname) { classname = emptySpanHTML.classAliases[name]; }
+
if (typeof s !== "string") throw "Invalid type in pg.jsescape";
classname = classname || name;
+
var res = "";
if (name == getValueOf('popupDragHandle')) { classname += ' popupDragHandle'; }
+
//this can be optimized by copying substrings instead of char by char!
return simplePrintf('<%s id="%s" class="%s"></%s>', [tag, name + id, classname, tag]);
+
for (var i=0; i<s.length; i++)
 +
{
 +
var c = s[i];
 +
switch (c)
 +
{
 +
  case '\b': res += '\\b'; continue;
 +
  case '\f': res += '\\f'; continue;
 +
  case '\n': res += '\\n'; continue;
 +
  case '\0': res += '\\0'; continue;
 +
  case '\r': res += '\\r'; continue;
 +
  case '\t': res += '\\t'; continue;
 +
  case '\v': res += '\\v'; continue;
 +
  case '\\': res += '\\\\'; continue;
 +
  case '\"':  res += '\\\"'; continue;
 +
  case '\'':  res += '\\\''; continue;
 +
  continue;
 +
  default:
 +
if (c < ' ' || c==='<' || c==='>' || c==="'")
 +
{
 +
var unicodeChar = c.charCodeAt(0).toString(16).toUpperCase();
 +
res += "\\u" + (unicodeChar.length>1?"00":"000") + unicodeChar;
 +
}
 +
else
 +
{
 +
res += c;
 +
}
 +
}
 +
  }
 +
  return res;
 
}
 
}
emptySpanHTML.classAliases={ 'popupSecondPreview': 'popupPreview' };
 
  
// generate html for popup image
+
// ENDFILE: tools.js
// <a id="popupImageLinkn"><img id="popupImagen">
+
// STARTFILE: dab.js
// where n=idNumber
+
//<NOLITE>
function imageHTML(article, idNumber) {
+
//////////////////////////////////////////////////
return simplePrintf('<a id="popupImageLink$1">' +
+
// Dab-fixing code
'<img align="right" valign="top" id="popupImg$1" style="display: none;"></img>' +
+
//
'</a>', [ idNumber ]);
 
}
 
  
function popTipsSoonFn(id, when, popData) {
 
if (!when) { when=250; }
 
var popTips=function(){ setupTooltips(document.getElementById(id), false, true, popData); };
 
return function() { setTimeout( popTips, when, popData ); };
 
}
 
  
function setPopupTipsAndHTML(html, divname, idnumber, popData) {
+
function retargetDab(newTarget, oldTarget, friendlyCurrentArticleName, titleToEdit) {
setPopupHTML(html, divname, idnumber,
+
log('retargetDab: newTarget='+newTarget + ' oldTarget=' + oldTarget);
getValueOf('popupSubpopups') ?
+
return changeLinkTargetLink(
popTipsSoonFn(divname + idnumber, null, popData) :  
+
{newTarget: newTarget,
null);
+
text: newTarget.split(' ').join('&nbsp;'),
 +
hint: tprintf('disambigHint', [newTarget]),
 +
summary: simplePrintf(
 +
getValueOf('popupFixDabsSummary'), [friendlyCurrentArticleName, newTarget ]),
 +
clickButton: getValueOf('popupDabsAutoClick'), minor: true, oldTarget: oldTarget,
 +
watch: getValueOf('popupWatchDisambiggedPages'),
 +
title: titleToEdit});
 
}
 
}
// ENDFILE: htmloutput.js
 
// STARTFILE: mouseout.js
 
//////////////////////////////////////////////////
 
// fuzzy checks
 
  
function fuzzyCursorOffMenus(x,y, fuzz, parent) {
+
function listLinks(wikitext, oldTarget, titleToEdit) {
if (!parent) { return null; }
+
// mediawiki strips trailing spaces, so we do the same
var uls=parent.getElementsByTagName('ul');
+
// testcase: http://en.wikipedia.org/w/index.php?title=Radial&oldid=97365633
for (var i=0; i<uls.length; ++i) {
+
var reg=RegExp('\\[\\[([^|]*?) *(\\||\\]\\])', 'gi');
if (uls[i].className=='popup_menu') {
+
var ret=[];
if (uls[i].offsetWidth > 0) return false;
+
var splitted=wikitext.parenSplit(reg);
} // else {document.title+='.';}
+
// ^[a-z]+ should match interwiki links, hopefully (case-insensitive)
 +
// and ^[a-z]* should match those and [[:Category...]] style links too
 +
var omitRegex=RegExp('^[a-z]*:|^[Ss]pecial:|^[Ii]mage|^[Cc]ategory');
 +
var friendlyCurrentArticleName= oldTarget.toString();
 +
var wikPos = getValueOf('popupDabWiktionary');
 +
 
 +
for (var i=1; i<splitted.length; i=i+3) {
 +
if (typeof splitted[i] == typeof 'string' && splitted[i].length>0 && !omitRegex.test(splitted[i])) {
 +
ret.push( retargetDab(splitted[i], oldTarget, friendlyCurrentArticleName, titleToEdit) );
 +
} /* if */
 +
} /* for loop */
 +
 
 +
ret = rmDupesFromSortedList(ret.sort());
 +
 
 +
if (wikPos) {
 +
var wikTarget='wiktionary:' +
 +
friendlyCurrentArticleName.replace( RegExp('^(.+)\\s+[(][^)]+[)]\\s*$'), '$1' );
 +
 
 +
var meth;
 +
if (wikPos.toLowerCase() == 'first') { meth = 'unshift'; }
 +
else { meth = 'push'; }
 +
 
 +
ret[meth]( retargetDab(wikTarget, oldTarget, friendlyCurrentArticleName, titleToEdit) );
 
}
 
}
return true;
+
 
 +
ret.push(changeLinkTargetLink(
 +
{ newTarget: null,
 +
text: popupString('remove this link').split(' ').join('&nbsp;'),
 +
hint: popupString("remove all links to this disambig page from this article"),
 +
clickButton: "wpDiff", oldTarget: oldTarget,
 +
summary: simplePrintf(getValueOf('popupRmDabLinkSummary'), [friendlyCurrentArticleName]),
 +
watch: getValueOf('popupWatchDisambiggedPages'),
 +
title: titleToEdit
 +
}));
 +
return ret;
 
}
 
}
  
function checkPopupPosition () { // stop the popup running off the right of the screen
+
function rmDupesFromSortedList(list) {
// FIXME avoid pg.current.link
+
var ret=[];
if (pg.current.link && pg.current.link.navpopup)
+
for (var i=0; i<list.length; ++i) {
pg.current.link.navpopup.limitHorizontalPosition();
+
if (ret.length===0 || list[i]!=ret[ret.length-1]) { ret.push(list[i]); }
 +
}
 +
return ret;
 
}
 
}
  
function mouseOutWikiLink () {
+
function makeFixDab(data, navpop) {
//console ('mouseOutWikiLink');
+
// grab title from parent popup if there is one; default exists in changeLinkTargetLink
var a=this;
+
var titleToEdit=(navpop.parentPopup && navpop.parentPopup.article.toString());
+
var list=listLinks(data, navpop.originalArticle, titleToEdit);
removeModifierKeyHandler(a);
+
if (list.length===0) { log('listLinks returned empty list'); return null; }
+
var html='<hr />' + popupString('Click to disambiguate this link to:') + '<br>';
if (a.navpopup === null || typeof a.navpopup === 'undefined') return;
+
html+=list.join(', ');
if ( ! a.navpopup.isVisible() ) {
+
return html;
a.navpopup.banish();
+
}
return;
+
 
 +
 
 +
function makeFixDabs(wikiText, navpop) {
 +
if (getValueOf('popupFixDabs') && isDisambig(wikiText, navpop.article) &&
 +
Title.fromURL(location.href).namespaceId() != pg.nsSpecialId &&
 +
navpop.article.talkPage() ) {
 +
setPopupHTML(makeFixDab(wikiText, navpop), 'popupFixDab', navpop.idNumber);
 
}
 
}
restoreTitle(a);
 
Navpopup.tracker.addHook(posCheckerHook(a.navpopup));
 
 
}
 
}
  
function posCheckerHook(navpop) {
+
function popupRedlinkHTML(article) {
return function() {
+
return changeLinkTargetLink(
if (!navpop.isVisible()) { return true; /* remove this hook */ }
+
{ newTarget: null, text: popupString('remove this link').split(' ').join('&nbsp;'),
if (Navpopup.tracker.dirty) {
+
hint: popupString("remove all links to this page from this article"),
return false;
+
clickButton: "wpDiff",
}
+
oldTarget: article.toString(),
var x=Navpopup.tracker.x, y=Navpopup.tracker.y;
+
summary: simplePrintf(getValueOf('popupRedlinkSummary'), [article.toString()])});
var mouseOverNavpop = navpop.isWithin(x,y,navpop.fuzz, navpop.mainDiv) ||
+
}
!fuzzyCursorOffMenus(x,y,navpop.fuzz, navpop.mainDiv);
+
//</NOLITE>
 +
// ENDFILE: dab.js
 +
// STARTFILE: htmloutput.js
 +
 
 +
function appendPopupContent(obj, elementId, popupId, onSuccess) {
 +
return setPopupHTML(obj, elementId, popupId, onSuccess, true);
 +
}
  
// FIXME it'd be prettier to do this internal to the Navpopup objects
+
// this has to use a timer loop as we don't know if the DOM element exists when we want to set the text
var t=getValueOf('popupHideDelay');
+
function setPopupHTML (str, elementId, popupId, onSuccess, append) {
if (t) { t = t * 1000; }
+
if (elementId=='popupPreview') {
if (!t) {
+
}
if(!mouseOverNavpop) {
+
if (typeof popupId === 'undefined') {
if(navpop.parentAnchor) {
+
//console.error('popupId is not defined in setPopupHTML, html='+str.substring(0,100));
restoreTitle( navpop.parentAnchor );
+
popupId = pg.idNumber;
}
+
}
navpop.banish();
+
 
return true; /* remove this hook */
+
var popupElement=document.getElementById(elementId+popupId);
}
+
if (popupElement) {
return false;
+
if (!append) { popupElement.innerHTML=''; }
 +
if (isString(str)) {
 +
popupElement.innerHTML+=str;
 +
} else {
 +
popupElement.appendChild(str);
 
}
 
}
// we have a hide delay set
+
if (onSuccess) { onSuccess(); }
var d=+(new Date());
+
setTimeout(checkPopupPosition, 100);
if ( !navpop.mouseLeavingTime ) {
+
return true;
navpop.mouseLeavingTime = d;
+
} else {
return false;
+
// call this function again in a little while...
}
+
setTimeout(function(){
if ( mouseOverNavpop ) {
+
setPopupHTML(str,elementId,popupId,onSuccess);
navpop.mouseLeavingTime=null;
+
}, 600);
return false;
+
}
}
+
return null;
if (d - navpop.mouseLeavingTime > t) {
 
navpop.mouseLeavingTime=null;
 
navpop.banish(); return true; /* remove this hook */
 
}
 
return false;
 
};
 
 
}
 
}
  
function runStopPopupTimer(navpop) {
+
//<NOLITE>
// at this point, we should have left the link but remain within the popup
+
function setPopupTrailer(str,id) {return setPopupHTML(str, 'popupData', id);}
// so we call this function again until we leave the popup.
+
//</NOLITE>
if (!navpop.stopPopupTimer) {
+
 
navpop.stopPopupTimer=setInterval(posCheckerHook(navpop), 500);
+
 
navpop.addHook(function(){clearInterval(navpop.stopPopupTimer);},
+
function fillEmptySpans(args) { return fillEmptySpans2(args); }
  'hide', 'before');
 
}
 
}
 
// ENDFILE: mouseout.js
 
// STARTFILE: previewmaker.js
 
/**
 
  @fileoverview
 
  Defines the {@link Previewmaker} object, which generates short previews from wiki markup.
 
*/
 
  
/**
+
// args.navpopup is mandatory
  Creates a new Previewmaker
+
// optional: args.redir, args.redirTarget
  @constructor
+
// FIXME: ye gods, this is ugly stuff
  @class The Previewmaker class. Use an instance of this to generate short previews from Wikitext.
+
function fillEmptySpans2(args) { // if redir is present and true then redirTarget is mandatory
  @param {String} wikiText The Wikitext source of the page we wish to preview.
+
var redir=true;
  @param {String} baseUrl The url we should prepend when creating relative urls.
+
if (typeof args != 'object' || typeof args.redir == 'undefined' || !args.redir) { redir=false; }
  @param {Navpopup} owner The navpop associated to this preview generator
+
var a=args.navpopup.parentAnchor;
*/
 
function Previewmaker(wikiText, baseUrl, owner) {
 
/** The wikitext which is manipulated to generate the preview. */
 
this.originalData=wikiText;
 
this.baseUrl=baseUrl;
 
this.owner=owner;
 
  
this.maxCharacters=getValueOf('popupMaxPreviewCharacters');
+
var article, hint=null, oldid=null, params={};
this.maxSentences=getValueOf('popupMaxPreviewSentences');
+
if (redir && typeof args.redirTarget == typeof {}) {
 +
article=args.redirTarget;
 +
//hint=article.hintValue();
 +
} else {
 +
article=(new Title()).fromAnchor(a);
 +
hint=a.originalTitle || article.hintValue();
 +
params=parseParams(a.href);
 +
oldid=(getValueOf('popupHistoricalLinks')) ? params.oldid : null;
 +
rcid=params.rcid;
 +
}
 +
var x={ article:article, hint: hint, oldid: oldid, rcid: rcid, navpop:args.navpopup, params:params };
  
this.setData();
+
var structure=pg.structures[getValueOf('popupStructure')];
}
+
if (typeof structure != 'object') {
Previewmaker.prototype.setData=function() {
+
setPopupHTML('popupError', 'Unknown structure (this should never happen): '+
var maxSize=Math.max(10000, 2*this.maxCharacters);
+
pg.option.popupStructure, args.navpopup.idNumber);
this.data=this.originalData.substring(0,maxSize);
+
return;
};
 
/** Remove HTML comments
 
@private
 
*/
 
Previewmaker.prototype.killComments = function () {
 
// this also kills one trailing newline, eg [[diamyo]]
 
this.data=this.data.replace(RegExp('^<!--[^$]*?-->\\n|\\n<!--[^$]*?-->(?=\\n)|<!--[^$]*?-->', 'g'), '');
 
};
 
/**
 
  @private
 
*/
 
Previewmaker.prototype.killDivs = function () {
 
// say goodbye, divs (can be nested, so use * not *?)
 
this.data=this.data.replace(RegExp('< *div[^>]* *>[\\s\\S]*?< */ *div *>',
 
  'gi'), '');
 
};
 
/**
 
  @private
 
*/
 
Previewmaker.prototype.killGalleries = function () {
 
this.data=this.data.replace(RegExp('< *gallery[^>]* *>[\\s\\S]*?< */ *gallery *>',
 
  'gi'), '');
 
};
 
/**
 
  @private
 
*/
 
Previewmaker.prototype.kill = function(opening, closing, subopening, subclosing, repl) {
 
var oldk=this.data;
 
var k=this.killStuff(this.data, opening, closing, subopening, subclosing, repl);
 
while (k.length < oldk.length) {
 
oldk=k;
 
k=this.killStuff(k, opening, closing, subopening, subclosing, repl);
 
 
}
 
}
this.data=k;
+
var spans=flatten(pg.misc.layout);
};
+
var numspans = spans.length;
/**
+
var redirs=pg.misc.redirSpans;
  @private
+
 
*/
+
for (var i=0; i<numspans; ++i) {
Previewmaker.prototype.killStuff = function (txt, opening, closing, subopening, subclosing, repl) {
+
var f=findThis(redirs, spans[i]);
var op=this.makeRegexp(opening);
+
//log('redir='+redir+', f='+f+', spans[i]='+spans[i]);
var cl=this.makeRegexp(closing, '^');
+
if ( (f!==null && !redir) || (f===null && redir) ) {
var sb=subopening ? this.makeRegexp(subopening, '^') : null;
+
//log('skipping this set of the loop');
var sc=subclosing ? this.makeRegexp(subclosing, '^') : cl;
+
continue;
if (!op || !cl) {
+
}
alert('Navigation Popups error: op or cl is null! something is wrong.');
+
var structurefn=structure[spans[i]];
return;
+
var setfn = setPopupHTML;
 +
if (getValueOf('popupActiveNavlinks') &&
 +
(spans[i].indexOf('popupTopLinks')==0 || spans[i].indexOf('popupRedirTopLinks')==0)
 +
) {
 +
setfn = setPopupTipsAndHTML;
 +
}
 +
switch (typeof structurefn) {
 +
case 'function':
 +
//log('running '+spans[i]+'({article:'+x.article+', hint:'+x.hint+', oldid: '+x.oldid+'})');
 +
setfn(structurefn(x), spans[i], args.navpopup.idNumber);
 +
break;
 +
case 'string':
 +
setfn(structurefn, spans[i], args.navpopup.idNumber);
 +
break;
 +
default:
 +
errlog('unknown thing with label '+spans[i]);
 +
break;
 +
}
 
}
 
}
if (!op.test(txt)) { return txt; }
+
}
var ret='';
+
 
var opResult = op.exec(txt);
+
// flatten an array
ret = txt.substring(0,opResult.index);
+
function flatten(list, start) {
txt=txt.substring(opResult.index+opResult[0].length);
+
var ret=[];
var depth = 1;
+
if (typeof start == 'undefined') { start=0; }
while (txt.length > 0) {
+
for (var i=start; i<list.length; ++i) {
var removal=0;
+
if (typeof list[i] == typeof []) {
if (depth==1 && cl.test(txt)) {
+
return ret.concat(flatten(list[i])).concat(flatten(list, i+1));
depth--;
 
removal=cl.exec(txt)[0].length;
 
} else if (depth > 1 && sc.test(txt)) {
 
depth--;
 
removal=sc.exec(txt)[0].length;
 
}else if (sb && sb.test(txt)) {
 
depth++;
 
removal=sb.exec(txt)[0].length;
 
 
}
 
}
if ( !removal ) { removal = 1; }
+
else { ret.push(list[i]); }
txt=txt.substring(removal);
 
if (depth === 0) { break; }
 
 
}
 
}
return ret + (repl || '') + txt;
+
return ret;
};
+
}
/**
+
 
  @private
+
// Generate html for whole popup
*/
+
function popupHTML (a) {
Previewmaker.prototype.makeRegexp = function (x, prefix, suffix) {
+
getValueOf('popupStructure');
prefix = prefix || '';
+
var structure=pg.structures[pg.option.popupStructure];
suffix = suffix || '';
+
if (typeof structure != 'object') {
var reStr='';
+
//return 'Unknown structure: '+pg.option.popupStructure;
var flags='';
+
// override user choice
if (isString(x)) {
+
pg.option.popupStructure=pg.optionDefault.popupStructure;
reStr=prefix + literalizeRegex(x) + suffix;
+
return popupHTML(a);
} else if (isRegExp(x)) {
 
var s=x.toString().substring(1);
 
var sp=s.split('/');
 
flags=sp[sp.length-1];
 
sp[sp.length-1]='';
 
s=sp.join('/');
 
s=s.substring(0,s.length-1);
 
reStr= prefix + s + suffix;
 
} else {
 
log ('makeRegexp failed');
 
 
}
 
}
 +
if (typeof structure.popupLayout != 'function') { return 'Bad layout'; }
 +
pg.misc.layout=structure.popupLayout();
 +
if ($.isFunction(structure.popupRedirSpans)) { pg.misc.redirSpans=structure.popupRedirSpans(); }
 +
else { pg.misc.redirSpans=[]; }
 +
return makeEmptySpans(pg.misc.layout, a.navpopup);
 +
}
  
log ('makeRegexp: got reStr=' + reStr + ', flags=' + flags);
+
function makeEmptySpans (list, navpop) {
return RegExp(reStr, flags);
+
var ret='';
};
+
for (var i=0; i<list.length; ++i) {
/**
+
if (typeof list[i] == typeof '') {
  @private
+
ret += emptySpanHTML(list[i], navpop.idNumber, 'div');
*/
+
} else if (typeof list[i] == typeof [] && list[i].length > 0 ) {
Previewmaker.prototype.killBoxTemplates = function () {
+
ret = ret.parenSplit(RegExp('(</[^>]*?>$)')).join(makeEmptySpans(list[i], navpop));
 +
} else if (typeof list[i] == typeof {} && list[i].nodeType ) {
 +
ret += emptySpanHTML(list[i].name, navpop.idNumber, list[i].nodeType);
 +
}
 +
}
 +
return ret;
 +
}
 +
 
 +
 
 +
function emptySpanHTML(name, id, tag, classname) {
 +
tag = tag || 'span';
 +
if (!classname) { classname = emptySpanHTML.classAliases[name]; }
 +
classname = classname || name;
 +
if (name == getValueOf('popupDragHandle')) { classname += ' popupDragHandle'; }
 +
return simplePrintf('<%s id="%s" class="%s"></%s>', [tag, name + id, classname, tag]);
 +
}
 +
emptySpanHTML.classAliases={ 'popupSecondPreview': 'popupPreview' };
  
// taxobox removal... in fact, there's a saudiprincebox_begin, so let's be more general
+
// generate html for popup image
// also, have float_begin, ... float_end
+
// <a id="popupImageLinkn"><img id="popupImagen">
this.kill(RegExp('[{][{][^{}\\s|]*?(float|box)[_ ](begin|start)', 'i'), /[}][}]\s*/, '{{');
+
// where n=idNumber
 +
function imageHTML(article, idNumber) {
 +
return simplePrintf('<a id="popupImageLink$1">' +
 +
'<img align="right" valign="top" id="popupImg$1" style="display: none;"></img>' +
 +
'</a>', [ idNumber ]);
 +
}
  
// infoboxes etc
+
function popTipsSoonFn(id, when, popData) {
// from [[User:Zyxw/popups.js]]: kill frames too
+
when || ( when=250 );
this.kill(RegExp('[{][{][^{}\\s|]*?(infobox|elementbox|frame)[_ ]', 'i'), /[}][}]\s*/, '{{');
+
var popTips=function(){ setupTooltips(document.getElementById(id), false, true, popData); };
 +
return function() { setTimeout( popTips, when, popData ); };
 +
}
  
};
+
function setPopupTipsAndHTML(html, divname, idnumber, popData) {
/**
+
setPopupHTML(html, divname, idnumber,
  @private
+
getValueOf('popupSubpopups') ?
*/
+
popTipsSoonFn(divname + idnumber, null, popData) :
Previewmaker.prototype.killTemplates = function () {
+
null);
this.kill('{{', '}}', '{', '}', ' ');
+
}
};
+
// ENDFILE: htmloutput.js
/**
+
// STARTFILE: mouseout.js
  @private
+
//////////////////////////////////////////////////
*/
+
// fuzzy checks
Previewmaker.prototype.killTables = function () {
 
// tables are bad, too
 
// this can be slow, but it's an inprovement over a browser hang
 
// torture test: [[Comparison_of_Intel_Central_Processing_Units]]
 
this.kill('{|', /[|]}\s*/, '{|');
 
this.kill(/<table.*?>/i, /<\/table.*?>/i, /<table.*?>/i);
 
// remove lines starting with a pipe for the hell of it (?)
 
this.data=this.data.replace(RegExp('^[|].*$', 'mg'), '');
 
};
 
/**
 
  @private
 
*/
 
Previewmaker.prototype.killImages = function () {
 
var forbiddenNamespaceAliases = [];
 
jQuery.each(mw.config.get('wgNamespaceIds'), function(_localizedNamespaceLc, _namespaceId) {
 
if (_namespaceId!=pg.nsImageId && _namespaceId!=pg.nsCategoryId) return;
 
forbiddenNamespaceAliases.push(_localizedNamespaceLc.split(' ').join('[ _]')); //todo: escape regexp fragments!
 
});
 
 
// images and categories are a nono
 
this.kill(RegExp('[[][[]\\s*(' + forbiddenNamespaceAliases.join('|') + ')\\s*:', 'i'),
 
  /\]\]\s*/, '[', ']');
 
};
 
/**
 
  @private
 
*/
 
Previewmaker.prototype.killHTML = function () {
 
// kill <ref ...>...</ref>
 
this.kill(/<ref\b[^/>]*?>/i, /<\/ref>/i);
 
  
// let's also delete entire lines starting with <. it's worth a try.
+
function fuzzyCursorOffMenus(x,y, fuzz, parent) {
this.data=this.data.replace(RegExp('(^|\\n) *<.*', 'g'), '\n');
+
if (!parent) { return null; }
 +
var uls=parent.getElementsByTagName('ul');
 +
for (var i=0; i<uls.length; ++i) {
 +
if (uls[i].className=='popup_menu') {
 +
if (uls[i].offsetWidth > 0) return false;
 +
} // else {document.title+='.';}
 +
}
 +
return true;
 +
}
  
// and those pesky html tags, but not <nowiki> or <blockquote>
+
function checkPopupPosition () { // stop the popup running off the right of the screen
var splitted=this.data.parenSplit(/(<[\w\W]*?(?:>|$|(?=<)))/);
+
// FIXME avoid pg.current.link
var len=splitted.length;
+
pg.current.link && pg.current.link.navpopup &&
for (var i=1; i<len; i=i+2) {
+
pg.current.link.navpopup.limitHorizontalPosition();
switch (splitted[i]) {
+
}
case '<nowiki>':
+
 
case '</nowiki>':
+
function mouseOutWikiLink () {
case '<blockquote>':
+
if (!window.popupsReady || !window.popupsReady()) { return; }
case '</blockquote>':
+
//console ('mouseOutWikiLink');
break;
+
var a=this;
default:
+
if (a.navpopup==null) return;
splitted[i]='';
+
if ( ! a.navpopup.isVisible() ) {
}
+
a.navpopup.banish();
 +
return;
 
}
 
}
this.data=splitted.join('');
+
restoreTitle(a);
};
+
Navpopup.tracker.addHook(posCheckerHook(a.navpopup));
/**
+
}
  @private
 
*/
 
Previewmaker.prototype.killChunks = function() { // heuristics alert
 
// chunks of italic text? you crazy, man?
 
var italicChunkRegex=new RegExp
 
("((^|\\n)\\s*:*\\s*''[^']([^']|'''|'[^']){20}(.|\\n[^\\n])*''[.!?\\s]*\\n)+", 'g');
 
// keep stuff separated, though, so stick in \n (fixes [[Union Jack]]?
 
this.data=this.data.replace(italicChunkRegex, '\n');
 
};
 
/**
 
  @private
 
*/
 
Previewmaker.prototype.mopup = function () {
 
// we simply *can't* be doing with horizontal rules right now
 
this.data=this.data.replace(RegExp('^-{4,}','mg'),'');
 
  
// no indented lines
+
function posCheckerHook(navpop) {
this.data=this.data.replace(RegExp('(^|\\n) *:[^\\n]*','g'), '');
+
return function() {
 +
if (!navpop.isVisible()) { return true; /* remove this hook */ }
 +
if (Navpopup.tracker.dirty) {
 +
return false;
 +
}
 +
var x=Navpopup.tracker.x, y=Navpopup.tracker.y;
 +
var mouseOverNavpop = navpop.isWithin(x,y,navpop.fuzz, navpop.mainDiv) ||
 +
!fuzzyCursorOffMenus(x,y,navpop.fuzz, navpop.mainDiv);
  
// replace __TOC__, __NOTOC__ and whatever else there is
+
// FIXME it'd be prettier to do this internal to the Navpopup objects
// this'll probably do
+
var t=getValueOf('popupHideDelay');
this.data=this.data.replace(RegExp('^__[A-Z_]*__ *$', 'gmi'),'');
+
if (t) { t = t * 1000; }
};
+
if (!t) {
 +
if(!mouseOverNavpop) {
 +
if(navpop.parentAnchor) {
 +
restoreTitle( navpop.parentAnchor );
 +
}
 +
navpop.banish();
 +
return true; /* remove this hook */
 +
}
 +
return false;
 +
}
 +
// we have a hide delay set
 +
var d=+(new Date());
 +
if ( !navpop.mouseLeavingTime ) {
 +
navpop.mouseLeavingTime = d;
 +
return false;
 +
}
 +
if ( mouseOverNavpop ) {
 +
navpop.mouseLeavingTime=null;
 +
return false;
 +
}
 +
if (d - navpop.mouseLeavingTime > t) {
 +
navpop.mouseLeavingTime=null;
 +
navpop.banish(); return true; /* remove this hook */
 +
}
 +
return false;
 +
};
 +
}
 +
 
 +
function runStopPopupTimer(navpop) {
 +
// at this point, we should have left the link but remain within the popup
 +
// so we call this function again until we leave the popup.
 +
if (!navpop.stopPopupTimer) {
 +
navpop.stopPopupTimer=setInterval(posCheckerHook(navpop), 500);
 +
navpop.addHook(function(){clearInterval(navpop.stopPopupTimer);},
 +
  'hide', 'before');
 +
}
 +
}
 +
// ENDFILE: mouseout.js
 +
// STARTFILE: previewmaker.js
 
/**
 
/**
   @private
+
   @fileoverview
 +
  Defines the {@link Previewmaker} object, which generates short previews from wiki markup.
 
*/
 
*/
Previewmaker.prototype.firstBit = function () {
 
// dont't be givin' me no subsequent paragraphs, you hear me?
 
/// first we "normalize" section headings, removing whitespace after, adding before
 
var d=this.data;
 
  
if (getValueOf('popupPreviewCutHeadings')) {
+
/**
this.data=this.data.replace(RegExp('\\s*(==+[^=]*==+)\\s*', 'g'), '\n\n$1 ');
+
  Creates a new Previewmaker
/// then we want to get rid of paragraph breaks whose text ends badly
+
  @constructor
this.data=this.data.replace(RegExp('([:;]) *\\n{2,}', 'g'), '$1\n');
+
  @class The Previewmaker class. Use an instance of this to generate short previews from Wikitext.
 
+
  @param {String} wikiText The Wikitext source of the page we wish to preview.
this.data=this.data.replace(RegExp('^[\\s\\n]*'), '');
+
  @param {String} baseUrl The url we should prepend when creating relative urls.
var stuff=(RegExp('^([^\\n]|\\n[^\\n\\s])*')).exec(this.data);
+
   @param {Navpopup} owner The navpop associated to this preview generator
if (stuff) { d = stuff[0]; }
 
if (!getValueOf('popupPreviewFirstParOnly')) { d = this.data; }
 
 
 
/// now put \n\n after sections so that bullets and numbered lists work
 
d=d.replace(RegExp('(==+[^=]*==+)\\s*', 'g'), '$1\n\n');
 
}
 
 
 
 
 
// Split sentences. Superfluous sentences are RIGHT OUT.
 
// note: exactly 1 set of parens here needed to make the slice work
 
d = d.parenSplit(RegExp('([!?.]+["'+"'"+']*\\s)','g'));
 
// leading space is bad, mmkay?
 
d[0]=d[0].replace(RegExp('^\\s*'), '');
 
 
 
var notSentenceEnds=RegExp('([^.][a-z][.] *[a-z]|etc|sic|Dr|Mr|Mrs|Ms|St|no|op|cit|\\[[^\\]]*|\\s[A-Zvclm])$', 'i');
 
d = this.fixSentenceEnds(d, notSentenceEnds);
 
 
 
this.fullLength=d.join('').length;
 
var n=this.maxSentences;
 
var dd=this.firstSentences(d,n);
 
 
 
do {
 
dd=this.firstSentences(d,n); --n;
 
} while ( dd.length > this.maxCharacters && n !== 0 );
 
 
 
this.data = dd;
 
};
 
/**
 
   @private
 
 
*/
 
*/
Previewmaker.prototype.fixSentenceEnds = function(strs, reg) {
+
function Previewmaker(wikiText, baseUrl, owner) {
// take an array of strings, strs
+
/** The wikitext which is manipulated to generate the preview. */
// join strs[i] to strs[i+1] & strs[i+2] if strs[i] matches regex reg
+
this.originalData=wikiText;
 
+
this.setData();
for (var i=0; i<strs.length-2; ++i) {
+
this.baseUrl=baseUrl;
if (reg.test(strs[i])) {
+
this.owner=owner;
var a=[];
+
this.maxCharacters=getValueOf('popupMaxPreviewCharacters');
for (var j=0; j<strs.length; ++j) {
+
this.maxSentences=getValueOf('popupMaxPreviewSentences');
if (j<i)   a[j]=strs[j];
+
}
if (j==i) a[i]=strs[i]+strs[i+1]+strs[i+2];
+
Previewmaker.prototype.setData=function() {
if (j>i+2) a[j-2]=strs[j];
+
var maxSize=max(10000, 2*this.maxCharacters);
}
+
this.data=this.originalData.substring(0,maxSize);
return this.fixSentenceEnds(a,reg);
+
};
}
+
/** Remove HTML comments
}
+
@private
return strs;
+
*/
 +
Previewmaker.prototype.killComments = function () {
 +
// this also kills one trailing newline, eg [[diamyo]]
 +
this.data=this.data.replace(RegExp('<!--[\\s\\S]*?-->\\n?', 'g'), '');  
 
};
 
};
 
/**
 
/**
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.firstSentences = function(strs, howmany) {
+
Previewmaker.prototype.killDivs = function () {
var t=strs.slice(0, 2*howmany);
+
// say goodbye, divs (can be nested, so use * not *?)
return t.join('');
+
this.data=this.data.replace(RegExp('< *div[^>]* *>[\\s\\S]*?< */ *div *>',
 +
  'gi'), '');
 
};
 
};
 
/**
 
/**
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.killBadWhitespace = function() {
+
Previewmaker.prototype.killGalleries = function () {
// also cleans up isolated '''', eg [[Suntory Sungoliath]]
+
this.data=this.data.replace(RegExp('< *gallery[^>]* *>[\\s\\S]*?< */ *gallery *>',
this.data=this.data.replace(RegExp('^ *\'+ *$', 'gm'), '');
+
  'gi'), '');
 
};
 
};
 
/**
 
/**
  Runs the various methods to generate the preview.
 
  The preview is stored in the <code>html</html> field.
 
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.makePreview = function() {
+
Previewmaker.prototype.kill = function(opening, closing, subopening, subclosing, repl) {
if (this.owner.article.namespaceId()!=pg.nsTemplateId &&
+
var oldk=this.data;
this.owner.article.namespaceId()!=pg.nsImageId ) {
+
var k=this.killStuff(this.data, opening, closing, subopening, subclosing, repl);
this.killComments();
+
while (k.length < oldk.length) {
this.killDivs();
+
oldk=k;
this.killGalleries();
+
k=this.killStuff(k, opening, closing, subopening, subclosing, repl);
this.killBoxTemplates();
+
}
 
+
this.data=k;
if (getValueOf('popupPreviewKillTemplates')) {
 
this.killTemplates();
 
} else {
 
this.killMultilineTemplates();
 
}
 
this.killTables();
 
this.killImages();
 
this.killHTML();
 
this.killChunks();
 
this.mopup();
 
 
 
this.firstBit();
 
this.killBadWhitespace();
 
}
 
else
 
{
 
this.killHTML();
 
}
 
this.html=wiki2html(this.data, this.baseUrl); // needs livepreview
 
this.fixHTML();
 
this.stripLongTemplates();
 
 
};
 
};
 
/**
 
/**
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.esWiki2HtmlPart = function(data) {
+
Previewmaker.prototype.killStuff = function (txt, opening, closing, subopening, subclosing, repl) {
  var reLinks = /(?:\[\[([^|\]]*)(?:\|([^|\]]*))*]]([a-z]*))/gi; //match a wikilink
+
var op=this.makeRegexp(opening);
  reLinks.lastIndex = 0; //reset regex
+
var cl=this.makeRegexp(closing, '^');
 
+
var sb=subopening ? this.makeRegexp(subopening, '^') : null;
  var match;
+
var sc=subclosing ? this.makeRegexp(subclosing, '^') : cl;
  var result = "";
+
if (!op || !cl) {
  var postfixIndex = 0;
+
alert('Navigation Popups error: op or cl is null! something is wrong.');
  while ((match = reLinks.exec(data))) //match all wikilinks
+
return;
  {
+
}
//FIXME: the way that link is built here isn't perfect. It is clickable, but popups preview won't recognize it in some cases.
+
if (!op.test(txt)) { return txt; }
result += pg.escapeQuotesHTML(data.substring(postfixIndex, match.index)) +
+
var ret='';
  '<a href="'+Insta.conf.paths.articles+pg.escapeQuotesHTML(match[1])+'">'+pg.escapeQuotesHTML((match[2]?match[2]:match[1])+match[3])+"</a>";
+
var opResult = op.exec(txt);
postfixIndex = reLinks.lastIndex;
+
ret = txt.substring(0,opResult.index);
  }
+
txt=txt.substring(opResult.index+opResult[0].length);
  //append the rest
+
var depth = 1;
  result += pg.escapeQuotesHTML(data.substring(postfixIndex));
+
while (txt.length > 0) {
 
+
var removal=0;
  return result;
+
if (depth==1 && cl.test(txt)) {
};
+
depth--;
Previewmaker.prototype.editSummaryPreview=function() {
+
removal=cl.exec(txt)[0].length;
var reAes  = /\/\* *(.*?) *\*\//g; //match the first section marker
+
} else if (depth > 1 && sc.test(txt)) {
reAes.lastIndex = 0; //reset regex
+
depth--;
+
removal=sc.exec(txt)[0].length;
var match;
+
}else if (sb && sb.test(txt)) {
+
depth++;
match = reAes.exec(this.data);
+
removal=sb.exec(txt)[0].length;
if (match)
+
}
{
+
if ( !removal ) { removal = 1; }
//we have a section link. Split it, process it, combine it.
+
txt=txt.substring(removal);
var prefix = this.data.substring(0,match.index-1);
+
if (depth==0) { break; }
var section = match[1];
 
var postfix = this.data.substring(reAes.lastIndex);
 
 
var start = "<span class='autocomment'>";
 
var end = "</span>";
 
if (prefix.length>0) start = this.esWiki2HtmlPart(prefix) + " " + start + "- ";
 
if (postfix.length>0) end = ": " + end + this.esWiki2HtmlPart(postfix);
 
 
 
 
var t=new Title().fromURL(this.baseUrl);
 
t.anchorFromUtf(section);
 
var sectionLink = Insta.conf.paths.articles + pg.escapeQuotesHTML(t.toString(true)) + '#' + pg.escapeQuotesHTML(t.anchor);
 
return start + '<a href="'+sectionLink+'">&rarr;</a> '+pg.escapeQuotesHTML(section) + end;
 
 
}
 
}
+
return ret + (repl || '') + txt;
//else there's no section link, htmlify the whole thing.
 
return this.esWiki2HtmlPart(this.data);
 
 
};
 
};
 
+
/**
//<NOLITE>
+
  @private
/** Test function for debugging preview problems one step at a time.
+
*/
*/
+
Previewmaker.prototype.makeRegexp = function (x, prefix, suffix) {
/*eslint-disable */
+
prefix = prefix || '';
function previewSteps(txt) {
+
suffix = suffix || '';
try {
+
var reStr='';
txt=txt || document.editform.wpTextbox1.value;
+
var flags='';
} catch (err) {
+
if (isString(x)) {
if (pg.cache.pages.length > 0) {
+
reStr=prefix + literalizeRegex(x) + suffix;
txt=pg.cache.pages[pg.cache.pages.length-1].data;
+
} else if (isRegExp(x)) {
} else {
+
var s=x.toString().substring(1);
alert('provide text or use an edit page');
+
var sp=s.split('/');
}
+
flags=sp[sp.length-1];
 +
sp[sp.length-1]='';
 +
s=sp.join('/');
 +
s=s.substring(0,s.length-1);
 +
reStr= prefix + s + suffix;
 +
} else {
 +
log ('makeRegexp failed');
 
}
 
}
txt=txt.substring(0,10000);
 
var base=pg.wiki.articlebase + Title.fromURL(document.location.href).urlString();
 
var p=new Previewmaker(txt, base, pg.current.link.navpopup);
 
if (this.owner.article.namespaceId() != pg.nsTemplateId) {
 
p.killComments(); if (!confirm('done killComments(). Continue?\n---\n' + p.data)) { return; }
 
p.killDivs(); if (!confirm('done killDivs(). Continue?\n---\n' + p.data)) { return; }
 
p.killGalleries(); if (!confirm('done killGalleries(). Continue?\n---\n' + p.data)) { return; }
 
p.killBoxTemplates(); if (!confirm('done killBoxTemplates(). Continue?\n---\n' + p.data)) { return; }
 
  
if (getValueOf('popupPreviewKillTemplates')) {
+
log ('makeRegexp: got reStr=' + reStr + ', flags=' + flags);
p.killTemplates(); if (!confirm('done killTemplates(). Continue?\n---\n' + p.data)) { return; }
+
return RegExp(reStr, flags);
} else {
+
};
p.killMultilineTemplates(); if (!confirm('done killMultilineTemplates(). Continue?\n---\n' + p.data)) { return; }
+
/**
}
+
  @private
 +
*/
 +
Previewmaker.prototype.killBoxTemplates = function () {
  
p.killTables(); if (!confirm('done killTables(). Continue?\n---\n' + p.data)) { return; }
+
// taxobox removal... in fact, there's a saudiprincebox_begin, so let's be more general
p.killImages(); if (!confirm('done killImages(). Continue?\n---\n' + p.data)) { return; }
+
// also, have float_begin, ... float_end
p.killHTML(); if (!confirm('done killHTML(). Continue?\n---\n' + p.data)) { return; }
+
this.kill(RegExp('[{][{][^{}\\s|]*?(float|box)[_ ](begin|start)', 'i'), /[}][}]\s*/, '{{');
p.killChunks(); if (!confirm('done killChunks(). Continue?\n---\n' + p.data)) { return; }
 
p.mopup(); if (!confirm('done mopup(). Continue?\n---\n' + p.data)) { return; }
 
  
p.firstBit(); if (!confirm('done firstBit(). Continue?\n---\n' + p.data)) { return; }
+
// infoboxes etc
p.killBadWhitespace(); if (!confirm('done killBadWhitespace(). Continue?\n---\n' + p.data)) { return; }
+
// from [[User:Zyxw/popups.js]]: kill frames too
}
+
this.kill(RegExp('[{][{][^{}\\s|]*?(infobox|elementbox|frame)[_ ]', 'i'), /[}][}]\s*/, '{{');
 
 
p.html=wiki2html(p.data, base); // needs livepreview
 
p.fixHTML(); if (!confirm('done fixHTML(). Continue?\n---\n' + p.html)) { return; }
 
p.stripLongTemplates(); if (!confirm('done stripLongTemplates(). Continue?\n---\n' + p.html)) { return; }
 
alert('finished preview - end result follows.\n---\n' + p.html);
 
}
 
/*eslint-enable */
 
//</NOLITE>
 
  
 +
};
 
/**
 
/**
  Works around livepreview bugs.
 
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.fixHTML = function() {
+
Previewmaker.prototype.killTemplates = function () {
if(!this.html) return;
+
this.kill('{{', '}}', '{', '}', ' ');
 
 
  var ret = this.html;
 
 
 
// fix question marks in wiki links
 
// maybe this'll break some stuff :-(
 
ret=ret.replace(RegExp('(<a href="' + pg.wiki.articlePath + '/[^"]*)[?](.*?")', 'g'), '$1%3F$2');
 
ret=ret.replace(RegExp('(<a href=\'' + pg.wiki.articlePath + '/[^\']*)[?](.*?\')', 'g'), '$1%3F$2');
 
// FIXME fix up % too
 
 
 
this.html=ret;
 
};
 
/**
 
  Generates the preview and displays it in the current popup.
 
 
 
  Does nothing if the generated preview is invalid or consists of whitespace only.
 
  Also activates wikilinks in the preview for subpopups if the popupSubpopups option is true.
 
*/
 
Previewmaker.prototype.showPreview = function () {
 
this.makePreview();
 
if (typeof this.html != typeof '') return;
 
if (RegExp('^\\s*$').test(this.html)) return;
 
setPopupHTML('<hr />', 'popupPrePreviewSep', this.owner.idNumber);
 
setPopupTipsAndHTML(this.html, 'popupPreview', this.owner.idNumber, { owner: this.owner });
 
var more = (this.fullLength > this.data.length) ? this.moreLink() : '';
 
setPopupHTML(more, 'popupPreviewMore', this.owner.idNumber);
 
 
};
 
};
 
/**
 
/**
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.moreLink=function() {
+
Previewmaker.prototype.killTables = function () {
var a=document.createElement('a');
+
// tables are bad, too
a.className='popupMoreLink';
+
// this can be slow, but it's an inprovement over a browser hang
a.innerHTML=popupString('more...');
+
// torture test: [[Comparison_of_Intel_Central_Processing_Units]]
var savedThis=this;
+
this.kill('{|', /[|]}\s*/, '{|');
a.onclick=function() {
+
this.kill(/<table.*?>/i, /<\/table.*?>/i, /<table.*?>/i);
savedThis.maxCharacters+=2000;
+
// remove lines starting with a pipe for the hell of it (?)
savedThis.maxSentences+=20;
+
this.data=this.data.replace(RegExp('^[|].*$', 'mg'), '');
savedThis.setData();
 
savedThis.showPreview();
 
};
 
return a;
 
 
};
 
};
 
 
/**
 
/**
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.stripLongTemplates = function() {
+
Previewmaker.prototype.killImages = function () {
// operates on the HTML!
+
var forbiddenNamespaceAliases = [];
this.html=this.html.replace(RegExp('^.{0,1000}[{][{][^}]*?(<(p|br)( /)?>\\s*){2,}([^{}]*?[}][}])?', 'gi'), '');
+
jQuery.each(mw.config.get('wgNamespaceIds'), function(_localizedNamespaceLc, _namespaceId) {
this.html=this.html.split('\n').join(' '); // workaround for <pre> templates
+
if (_namespaceId!=pg.nsImageId && _namespaceId!=pg.nsCategoryId) return;
this.html=this.html.replace(RegExp('[{][{][^}]*<pre>[^}]*[}][}]','gi'), '');
+
forbiddenNamespaceAliases.push(_localizedNamespaceLc.split(' ').join('[ _]')); //todo: escape regexp fragments!
 +
});
 +
 +
// images and categories are a nono
 +
this.kill(RegExp('[[][[]\\s*(' + forbiddenNamespaceAliases.join('|') + ')\\s*:', 'i'),
 +
  /\]\]\s*/, '[', ']');
 
};
 
};
 
/**
 
/**
 
   @private
 
   @private
 
*/
 
*/
Previewmaker.prototype.killMultilineTemplates = function() {
+
Previewmaker.prototype.killHTML = function () {
this.kill('{{{', '}}}');
+
// kill <ref ...>...</ref>
this.kill(RegExp('\\s*[{][{][^{}]*\\n'), '}}', '{{');
+
this.kill(/<ref\b[^/>]*?>/i, /<\/ref>/i);
};
+
 
// ENDFILE: previewmaker.js
+
// let's also delete entire lines starting with <. it's worth a try.
// STARTFILE: querypreview.js
+
this.data=this.data.replace(RegExp('(^|\\n) *<.*', 'g'), '\n');
function loadAPIPreview(queryType, article, navpop) {
+
 
var art=new Title(article).urlString();
+
// and those pesky html tags, but not <nowiki> or <blockquote>
var url=pg.wiki.apiwikibase + '?format=json&formatversion=2&action=query&';
+
var splitted=this.data.parenSplit(/(<[\w\W]*?(?:>|$|(?=<)))/);
var htmlGenerator=function(/*a, d*/){alert('invalid html generator');};
+
var len=splitted.length;
var usernameart = '';
+
for (var i=1; i<len; i=i+2) {
switch (queryType) {
+
switch (splitted[i]) {
case 'history':
+
case '<nowiki>':
url += 'titles=' + art + '&prop=revisions&rvlimit=' +
+
case '</nowiki>':
getValueOf('popupHistoryPreviewLimit');
+
case '<blockquote>':
htmlGenerator=APIhistoryPreviewHTML;
+
case '</blockquote>':
break;
+
break;
case 'category':
+
default:
url += 'list=categorymembers&cmtitle=' + art;
+
splitted[i]='';
htmlGenerator=APIcategoryPreviewHTML;
 
break;
 
case 'userinfo':
 
var username = new Title( article ).userName();
 
usernameart = encodeURIComponent( username );
 
if (pg.re.ipUser.test(username)) {
 
url += 'list=blocks&bkprop=range&bkip=' + usernameart;
 
} else {
 
url += 'list=users|usercontribs&usprop=blockinfo|groups|editcount|registration|gender&ususers=' + usernameart + "&meta=globaluserinfo&guiprop=groups|unattached&guiuser="+ usernameart + "&uclimit=1&ucprop=timestamp&ucuser=" + usernameart;
 
 
}
 
}
htmlGenerator=APIuserInfoPreviewHTML;
 
break;
 
case 'contribs':
 
usernameart = encodeURIComponent( new Title( article ).userName() );
 
url += 'list=usercontribs&ucuser=' + usernameart +
 
'&uclimit=' + getValueOf('popupContribsPreviewLimit');
 
htmlGenerator=APIcontribsPreviewHTML;
 
break;
 
case 'imagepagepreview':
 
var trail='';
 
if (getValueOf('popupImageLinks')) { trail = '&list=imageusage&iutitle=' + art; }
 
url += 'titles=' + art + '&prop=revisions|imageinfo&rvprop=content' + trail;
 
htmlGenerator=APIimagepagePreviewHTML;
 
break;
 
case 'backlinks':
 
url += 'list=backlinks&bltitle=' + art;
 
htmlGenerator=APIbacklinksPreviewHTML;
 
break;
 
case 'revision':
 
if (article.oldid) {
 
url += 'revids=' + article.oldid;
 
} else {
 
url += 'titles=' + article.removeAnchor().urlString();
 
}
 
url += '&prop=revisions|pageprops|info|images|categories&rvprop=ids|timestamp|flags|comment|user|content&cllimit=max&imlimit=max';
 
htmlGenerator=APIrevisionPreviewHTML;
 
break;
 
 
}
 
}
pendingNavpopTask(navpop);
+
this.data=splitted.join('');
var callback=function(d){
+
};
log( "callback of API functions was hit" );
+
/**
showAPIPreview(queryType, htmlGenerator(article,d,navpop), navpop.idNumber, navpop, d);
+
  @private
};
+
*/
var go = function(){
+
Previewmaker.prototype.killChunks = function() { // heuristics alert
getPageWithCaching(url, callback, navpop);
+
// chunks of italic text? you crazy, man?
return true;
+
var italicChunkRegex=new RegExp
};
+
("((^|\\n)\\s*:*\\s*''[^']([^']|'''|'[^']){20}(.|\\n[^\\n])*''[.!?\\s]*\\n)+", 'g');
 +
// keep stuff separated, though, so stick in \n (fixes [[Union Jack]]?
 +
this.data=this.data.replace(italicChunkRegex, '\n');
 +
};
 +
/**
 +
  @private
 +
*/
 +
Previewmaker.prototype.mopup = function () {
 +
// we simply *can't* be doing with horizontal rules right now
 +
this.data=this.data.replace(RegExp('^-{4,}','mg'),'');
 +
 
 +
// no indented lines
 +
this.data=this.data.replace(RegExp('(^|\\n) *:[^\\n]*','g'), '');
 +
 
 +
// replace __TOC__, __NOTOC__ and whatever else there is
 +
// this'll probably do
 +
this.data=this.data.replace(RegExp('^__[A-Z_]*__ *$', 'gmi'),'');
 +
};
 +
/**
 +
  @private
 +
*/
 +
Previewmaker.prototype.firstBit = function () {
 +
// dont't be givin' me no subsequent paragraphs, you hear me?
 +
/// first we "normalize" section headings, removing whitespace after, adding before
 +
var d=this.data;
 +
 
 +
if (getValueOf('popupPreviewCutHeadings')) {
 +
this.data=this.data.replace(RegExp('\\s*(==+[^=]*==+)\\s*', 'g'), '\n\n$1 ');
 +
/// then we want to get rid of paragraph breaks whose text ends badly
 +
this.data=this.data.replace(RegExp('([:;]) *\\n{2,}', 'g'), '$1\n');
  
if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
+
this.data=this.data.replace(RegExp('^[\\s\\n]*'), '');
else { navpop.addHook(go, 'unhide', 'before', 'DOWNLOAD_'+queryType+'_QUERY_DATA'); }
+
stuff=(RegExp('^([^\\n]|\\n[^\\n\\s])*')).exec(this.data);
}
+
if (stuff) { d = stuff[0]; }
 +
if (!getValueOf('popupPreviewFirstParOnly')) { d = this.data; }
  
function linkList(list) {
+
/// now put \n\n after sections so that bullets and numbered lists work
list.sort(function(x,y) { return (x==y ? 0 : (x<y ? -1 : 1)); });
+
d=d.replace(RegExp('(==+[^=]*==+)\\s*', 'g'), '$1\n\n');
var buf=[];
 
for (var i=0; i<list.length; ++i) {
 
buf.push(wikiLink({article: new Title(list[i]),
 
  text: list[i].split(' ').join('&nbsp;'),
 
  action:  'view'}));
 
 
}
 
}
return buf.join(', ');
 
}
 
  
function getTimeOffset() {
 
var tz = mw.user.options.get('timecorrection');
 
  
if(tz) {
+
// Split sentences. Superfluous sentences are RIGHT OUT.
if( tz.indexOf('|') > -1 ) {
+
// note: exactly 1 set of parens here needed to make the slice work
// New format
+
d = d.parenSplit(RegExp('([!?.]+["'+"'"+']*\\s)','g'));
return parseInt(tz.split('|')[1],10);
+
// leading space is bad, mmkay?
} else if ( tz.indexOf(':') > -1 ) {
+
d[0]=d[0].replace(RegExp('^\\s*'), '');
// Old format
+
 
return( parseInt(tz,10)*60 + parseInt(tz.split(':')[1],10) );
+
var notSentenceEnds=RegExp('([^.][a-z][.] *[a-z]|etc|sic|Dr|Mr|Mrs|Ms|St|no|op|cit|\\[[^\\]]*|\\s[A-Zvclm])$', 'i');
}
+
d = this.fixSentenceEnds(d, notSentenceEnds);
}
 
return 0;
 
}
 
  
/*
+
this.fullLength=d.join('').length;
* Creates a HTML table that's shown in the history and user-contribs popups.
+
var maxChars=getValueOf('popupMaxPreviewCharacters') + this.extraCharacters;
* @param {Object[]} h - a list of revisions, returned from the API
+
var n=this.maxSentences;
* @param {boolean} reallyContribs - true only if we're displaying user contributions
+
var dd=this.firstSentences(d,n);  
*/
 
function editPreviewTable(article, h, reallyContribs, timeOffset) {
 
var html=['<table>'];
 
var day=null;
 
var curart=article;
 
var page=null;
 
  
var makeFirstColumnLinks;
+
do {
if(reallyContribs) {
+
dd=this.firstSentences(d,n); --n;
 +
} while ( dd.length > this.maxCharacters && n != 0 );
  
// We're showing user contributions, so make (diff | hist) links
+
this.data = dd;
makeFirstColumnLinks = function(currentRevision) {
+
};
var result = '(';
+
/**
result += '<a href="' + pg.wiki.titlebase +
+
  @private
new Title(currentRevision.title).urlString() + '&diff=prev' +
+
*/
'&oldid=' + currentRevision.revid + '">' + popupString('diff') + '</a>';
+
Previewmaker.prototype.fixSentenceEnds = function(strs, reg) {
result += '&nbsp;|&nbsp;';
+
// take an array of strings, strs
result += '<a href="' + pg.wiki.titlebase +
+
// join strs[i] to strs[i+1] & strs[i+2] if strs[i] matches regex reg
new Title(currentRevision.title).urlString() + '&action=history">' +
 
popupString('hist') + '</a>';
 
result += ')';
 
return result;
 
};
 
} else {
 
  
// It's a regular history page, so make (cur | last) links
+
var abbrevRe=/\b[a-z][^a-z]*$/i;
var firstRevid = h[0].revid;
 
makeFirstColumnLinks = function(currentRevision) {
 
var result = '(';
 
result += '<a href="' + pg.wiki.titlebase + new Title(curart).urlString() +
 
'&diff=' + firstRevid + '&oldid=' + currentRevision.revid + '">' + popupString('cur') + '</a>';
 
result += '&nbsp;|&nbsp;';
 
result += '<a href="' + pg.wiki.titlebase + new Title(curart).urlString() +
 
'&diff=prev&oldid=' + currentRevision.revid + '">' + popupString('last') + '</a>';
 
result += ')';
 
return result;
 
};
 
}
 
  
for (var i=0; i<h.length; ++i) {
+
for (var i=0; i<strs.length-2; ++i) {
if (reallyContribs) {  
+
if (reg.test(strs[i])) {
page = h[i].title;
+
a=[];
curart = new Title(page);
+
for (var j=0; j<strs.length; ++j) {
 +
if (j<i)  a[j]=strs[j];
 +
if (j==i)  a[i]=strs[i]+strs[i+1]+strs[i+2];
 +
if (j>i+2) a[j-2]=strs[j];
 +
}
 +
return this.fixSentenceEnds(a,reg);
 
}
 
}
var minor = h[i].minor ? '<b>m </b>' : '';
+
// BUGGY STUFF - trying to fix up [[S. C. Johnson & Son]] preview
var editDate = adjustDate(getDateFromTimestamp(h[i].timestamp), timeOffset);
+
if (false && abbrevRe.test(strs[i])) {
var thisDay = dayFormat(editDate);
+
var j=i, buf='';
var thisTime = timeFormat(editDate);
+
do {
if (thisDay == day) {
+
buf=buf+strs[i]+strs[i+1];
thisDay = '';
+
i=i+2;
 +
} while (i<strs.length-2 && abbrevRe.test(strs[i]));
 +
strs[i]=buf+strs[i];
 +
var a=(j?strs.slice(0,j-1):[]).concat(strs.slice(i));
 +
return this.fixSentenceEnds(a,reg);
 +
}
 +
}
 +
return strs;
 +
};
 +
/**
 +
  @private
 +
*/
 +
Previewmaker.prototype.firstSentences = function(strs, howmany) {
 +
var t=strs.slice(0, 2*howmany);
 +
return t.join('');
 +
};
 +
/**
 +
  @private
 +
*/
 +
Previewmaker.prototype.killBadWhitespace = function() {
 +
// also cleans up isolated '''', eg [[Suntory Sungoliath]]
 +
this.data=this.data.replace(RegExp('^ *\'+ *$', 'gm'), '');
 +
};
 +
/**
 +
  Runs the various methods to generate the preview.
 +
  The preview is stored in the <code>html</html> field.
 +
  @private
 +
*/
 +
Previewmaker.prototype.makePreview = function() {
 +
if (this.owner.article.namespaceId()!=pg.nsTemplateId &&
 +
this.owner.article.namespaceId()!=pg.nsImageId ) {
 +
this.killComments();
 +
this.killDivs();
 +
this.killGalleries();
 +
this.killBoxTemplates();
 +
 
 +
if (getValueOf('popupPreviewKillTemplates')) {
 +
this.killTemplates();
 
} else {
 
} else {
day = thisDay;
+
this.killMultilineTemplates();
 
}
 
}
if (thisDay) {
+
this.killTables();
html.push( '<tr><td colspan=3><span class="popup_history_date">' +
+
this.killImages();
  thisDay+'</span></td></tr>' );
+
this.killHTML();
}
+
this.killChunks();
html.push('<tr class="popup_history_row_' + ( (i%2) ? 'odd' : 'even') + '">');
+
this.mopup();
html.push('<td>' + makeFirstColumnLinks(h[i]) + '</td>');
+
 
html.push('<td>' +
+
this.firstBit();
'<a href="' + pg.wiki.titlebase + new Title(curart).urlString() +
+
this.killBadWhitespace();
'&oldid=' + h[i].revid + '">' + thisTime + '</a></td>');
 
var col3url='', col3txt='';
 
if (!reallyContribs) {
 
var user=h[i].user;
 
if( !h[i].userhidden ) {
 
if( pg.re.ipUser.test(user) ) {
 
col3url=pg.wiki.titlebase + mw.config.get('wgFormattedNamespaces')[pg.nsSpecialId] + ':Contributions&target=' + new Title(user).urlString();
 
} else {
 
col3url=pg.wiki.titlebase + mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' + new Title(user).urlString();
 
}
 
col3txt=pg.escapeQuotesHTML(user);
 
} else {
 
col3url=getValueOf('popupRevDelUrl');
 
col3txt=pg.escapeQuotesHTML( popupString('revdel'));
 
}
 
} else {
 
col3url=pg.wiki.titlebase + curart.urlString();
 
col3txt=pg.escapeQuotesHTML(page);
 
}
 
html.push('<td>' + (reallyContribs ? minor : '') +
 
'<a href="' + col3url + '">' + col3txt + '</a></td>');
 
var comment='';
 
var c=h[i].comment || h[i].content;
 
if (c) {
 
comment=new Previewmaker(c, new Title(curart).toUrl()).editSummaryPreview();
 
} else if ( h[i].commenthidden ) {
 
comment=popupString('revdel');
 
}
 
html.push('<td>' + (!reallyContribs ? minor : '') + comment + '</td>');
 
html.push('</tr>');
 
html=[html.join('')];
 
 
}
 
}
html.push('</table>');
+
else
return html.join('');
+
{
}
+
this.killHTML();
 
 
function getDateFromTimestamp(t) {
 
var s=t.split(/[^0-9]/);
 
switch(s.length) {
 
case 0: return null;
 
case 1: return new Date(s[0]);
 
case 2: return new Date(s[0], s[1]-1);
 
case 3: return new Date(s[0], s[1]-1, s[2]);
 
case 4: return new Date(s[0], s[1]-1, s[2], s[3]);
 
case 5: return new Date(s[0], s[1]-1, s[2], s[3], s[4]);
 
case 6: return new Date(s[0], s[1]-1, s[2], s[3], s[4], s[5]);
 
default: return new Date(s[0], s[1]-1, s[2], s[3], s[4], s[5], s[6]);
 
 
}
 
}
}
+
this.html=wiki2html(this.data, this.baseUrl); // needs livepreview
 +
this.fixHTML();
 +
this.stripLongTemplates();
 +
};
 +
/**
 +
  @private
 +
*/
 +
Previewmaker.prototype.esWiki2HtmlPart = function(data) {
 +
  var reLinks = /(?:\[\[([^|\]]*)(?:\|([^|\]]*))*]]([a-z]*))/gi; //match a wikilink
 +
  reLinks.lastIndex = 0; //reset regex
  
function adjustDate(d, offset) {
+
  var match;
// offset is in minutes
+
  var result = "";
var o=offset * 60 * 1000;
+
  var postfixIndex = 0;
return new Date( +d + o);
+
  while ((match = reLinks.exec(data)) != null) //match all wikilinks
}
+
  {
 +
//FIXME: the way that link is built here isn't perfect. It is clickable, but popups preview won't recognize it in some cases.
 +
result += pg.escapeQuotesHTML(data.substring(postfixIndex, match.index)) +
 +
  '<a href="'+Insta.conf.paths.articles+pg.escapeQuotesHTML(match[1])+'">'+pg.escapeQuotesHTML((match[2]?match[2]:match[1])+match[3])+"</a>";
 +
postfixIndex = reLinks.lastIndex;
 +
  }
 +
  //append the rest
 +
  result += pg.escapeQuotesHTML(data.substring(postfixIndex));
 +
 
 +
  return result;
 +
};
 +
Previewmaker.prototype.editSummaryPreview=function() {
 +
var reAes  = /\/\* *(.*?) *\*\//g; //match the first section marker
 +
reAes.lastIndex = 0; //reset regex
 +
 +
var match;
 +
 +
match = reAes.exec(this.data);
 +
if (match)
 +
{
 +
//we have a section link. Split it, process it, combine it.
 +
var prefix = this.data.substring(0,match.index-1);
 +
var section = match[1];
 +
var postfix = this.data.substring(reAes.lastIndex);
 +
 +
var start = "<span class='autocomment'>";
 +
var end = "</span>";
 +
if (prefix.length>0) start = this.esWiki2HtmlPart(prefix) + " " + start + "- ";
 +
if (postfix.length>0) end = ": " + end + this.esWiki2HtmlPart(postfix);
 +
  
function dayFormat(editDate, utc) {
+
var t=new Title().fromURL(this.baseUrl);
if (utc) { return map(zeroFill, [editDate.getUTCFullYear(), editDate.getUTCMonth()+1, editDate.getUTCDate()]).join('-'); }
+
t.anchorFromUtf(section);
return map(zeroFill, [editDate.getFullYear(), editDate.getMonth()+1, editDate.getDate()]).join('-');
+
var sectionLink = Insta.conf.paths.articles + pg.escapeQuotesHTML(t.toString(true)) + '#' + pg.escapeQuotesHTML(t.anchor);
}
+
return start + '<a href="'+sectionLink+'">&rarr;</a> '+pg.escapeQuotesHTML(section) + end;
 
 
function timeFormat(editDate, utc) {
 
if (utc) { return map(zeroFill, [editDate.getUTCHours(), editDate.getUTCMinutes(), editDate.getUTCSeconds()]).join(':'); }
 
return map(zeroFill, [editDate.getHours(), editDate.getMinutes(), editDate.getSeconds()]).join(':');
 
}
 
 
 
function showAPIPreview(queryType, html, id, navpop, download) {
 
// DJ: done
 
var target='popupPreview';
 
completedNavpopTask(navpop);
 
 
 
switch (queryType) {
 
case 'imagelinks':
 
case 'category':
 
target='popupPostPreview'; break;
 
case 'userinfo':
 
target='popupUserData'; break;
 
case 'revision':
 
insertPreview(download);
 
return;
 
 
}
 
}
setPopupTipsAndHTML(html, target, id);
+
}
+
//else there's no section link, htmlify the whole thing.
 +
return this.esWiki2HtmlPart(this.data);
 +
};
  
function APIrevisionPreviewHTML(article, download) {
+
//<NOLITE>
try{
+
/** Test function for debugging preview problems one step at a time.
var jsObj=getJsObj(download.data);
+
*/
var page=anyChild(jsObj.query.pages);
+
function previewSteps(txt) {
if( page.missing ) {
 
// TODO we need to fix this proper later on
 
download.owner = null;
 
return;
 
}
 
var content = (
 
page &&
 
page.revisions &&
 
page.revisions[0].contentmodel === 'wikitext'
 
) ? page.revisions[0].content : null;
 
if( typeof content === 'string' )
 
{
 
download.data = content;
 
download.lastModified = new Date(page.revisions[0].timestamp);
 
}
 
} catch(someError) {
 
return 'Revision preview failed :(';
 
}
 
}
 
 
 
function APIbacklinksPreviewHTML(article, download/*, navpop*/ ) {
 
 
try {
 
try {
var jsObj=getJsObj(download.data);
+
txt=txt || document.editform.wpTextbox1.value;
var list=jsObj.query.backlinks;
+
} catch (err) {
 
+
if (pg.cache.pages.length > 0) {
var html=[];
+
txt=pg.cache.pages[pg.cache.pages.length-1].data;
if (!list) { return popupString('No backlinks found'); }
+
} else {
for ( var i=0; i < list.length; i++ ) {
+
alert('provide text or use an edit page');
var t=new Title(list[i].title);
 
html.push('<a href="' + pg.wiki.titlebase + t.urlString() + '">' + t + '</a>');
 
}
 
html=html.join(', ');
 
if (jsObj['continue'] && jsObj['continue'].blcontinue) {
 
html += popupString(' and more');
 
 
}
 
}
return html;
 
} catch (someError) {
 
return 'backlinksPreviewHTML went wonky';
 
 
}
 
}
}
+
txt=txt.substring(0,10000);
 +
var base=pg.wiki.articlebase + Title.fromURL(document.location.href).urlString();
 +
var p=new Previewmaker(txt, base, pg.current.link.navpopup);
 +
if (this.owner.article.namespaceId() != pg.nsTemplateId) {
 +
p.killComments(); if (!confirm('done killComments(). Continue?\n---\n' + p.data)) { return; }
 +
p.killDivs(); if (!confirm('done killDivs(). Continue?\n---\n' + p.data)) { return; }
 +
p.killGalleries(); if (!confirm('done killGalleries(). Continue?\n---\n' + p.data)) { return; }
 +
p.killBoxTemplates(); if (!confirm('done killBoxTemplates(). Continue?\n---\n' + p.data)) { return; }
  
pg.fn.APIsharedImagePagePreviewHTML = function APIsharedImagePagePreviewHTML(obj) {
+
if (getValueOf('popupPreviewKillTemplates')) {
log( "APIsharedImagePagePreviewHTML" );
+
p.killTemplates(); if (!confirm('done killTemplates(). Continue?\n---\n' + p.data)) { return; }
var popupid = obj.requestid;
+
} else {
if( obj.query && obj.query.pages )
+
p.killMultilineTemplates(); if (!confirm('done killMultilineTemplates(). Continue?\n---\n' + p.data)) { return; }
{
 
var page=anyChild(obj.query.pages );
 
var content = (
 
page &&
 
page.revisions &&
 
page.revisions[0].contentmodel === 'wikitext'
 
) ? page.revisions[0].content : null;
 
if( typeof content === 'string' )  
 
{
 
/* Not entirely safe, but the best we can do */
 
var p=new Previewmaker(content, pg.current.link.navpopup.article, pg.current.link.navpopup);
 
p.makePreview();
 
setPopupHTML( p.html, "popupSecondPreview", popupid );
 
 
}
 
}
}
 
};
 
  
function APIimagepagePreviewHTML(article, download, navpop) {
+
p.killTables(); if (!confirm('done killTables(). Continue?\n---\n' + p.data)) { return; }
try {
+
p.killImages(); if (!confirm('done killImages(). Continue?\n---\n' + p.data)) { return; }
var jsObj=getJsObj(download.data);
+
p.killHTML(); if (!confirm('done killHTML(). Continue?\n---\n' + p.data)) { return; }
var page=anyChild(jsObj.query.pages);
+
p.killChunks(); if (!confirm('done killChunks(). Continue?\n---\n' + p.data)) { return; }
var content= (
+
p.mopup(); if (!confirm('done mopup(). Continue?\n---\n' + p.data)) { return; }
page &&
 
page.revisions &&
 
page.revisions[0].contentmodel === 'wikitext'
 
) ? page.revisions[0].content : null;
 
var ret='';
 
var alt='';
 
try{alt=navpop.parentAnchor.childNodes[0].alt;} catch(e){}
 
if (alt) {
 
ret = ret + '<hr /><b>' + popupString('Alt text:') + '</b> ' + pg.escapeQuotesHTML(alt);
 
}
 
if (typeof content === 'string') {
 
var p=prepPreviewmaker(content, article, navpop);
 
p.makePreview();
 
if (p.html) { ret += '<hr />' + p.html; }
 
if (getValueOf('popupSummaryData')) {
 
var info=getPageInfo(content, download);
 
log(info);
 
setPopupTrailer(info, navpop.idNumber);
 
}
 
}
 
if (page && page.imagerepository == "shared" ) {
 
var art=new Title(article);
 
var encart = encodeURIComponent( "File:" + art.stripNamespace() );
 
var shared_url =  pg.wiki.apicommonsbase + '?format=json&formatversion=2' +
 
'&callback=pg.fn.APIsharedImagePagePreviewHTML' +
 
'&requestid=' + navpop.idNumber +
 
'&action=query&prop=revisions&rvprop=content&titles=' + encart;
 
  
ret = ret +'<hr />' + popupString( 'Image from Commons') +
+
p.firstBit(); if (!confirm('done firstBit(). Continue?\n---\n' + p.data)) { return; }
': <a href="' + pg.wiki.commonsbase + '?title=' + encart + '">' +
+
p.killBadWhitespace(); if (!confirm('done killBadWhitespace(). Continue?\n---\n' + p.data)) { return; }
popupString( 'Description page') + '</a>';
 
mw.loader.load( shared_url );
 
}
 
showAPIPreview('imagelinks', APIimagelinksPreviewHTML(article,download), navpop.idNumber, download);
 
return ret;
 
} catch (someError) {
 
return 'API imagepage preview failed :(';
 
 
}
 
}
}
 
  
function APIimagelinksPreviewHTML(article, download) {
+
p.html=wiki2html(p.data, base); // needs livepreview
try {
+
p.fixHTML(); if (!confirm('done fixHTML(). Continue?\n---\n' + p.html)) { return; }
var jsobj=getJsObj(download.data);
+
p.stripLongTemplates(); if (!confirm('done stripLongTemplates(). Continue?\n---\n' + p.html)) { return; }
var list=jsobj.query.imageusage;
+
alert('finished preview - end result follows.\n---\n' + p.html);
if (list) {
 
var ret=[];
 
for (var i=0; i < list.length; i++) {
 
ret.push(list[i].title);
 
}
 
if (ret.length === 0) { return popupString('No image links found'); }
 
return '<h2>' + popupString('File links') + '</h2>' + linkList(ret);
 
} else {
 
return popupString('No image links found');
 
}
 
} catch(someError) {
 
return 'Image links preview generation failed :(';
 
}
 
 
}
 
}
 +
//</NOLITE>
  
function APIcategoryPreviewHTML(article, download) {
+
/**
try{
+
  Works around livepreview bugs.
var jsobj=getJsObj(download.data);
+
  @private
var list=jsobj.query.categorymembers;
+
*/
var ret=[];
+
Previewmaker.prototype.fixHTML = function() {
for (var p=0; p < list.length; p++) {
+
if(!this.html) return;
  ret.push(list[p].title);
 
}
 
if (ret.length === 0) { return popupString('Empty category'); }
 
ret = '<h2>' + tprintf('Category members (%s shown)', [ret.length]) + '</h2>' +linkList(ret);
 
if (jsobj['continue'] && jsobj['continue'].cmcontinue) {
 
ret += popupString(' and more');
 
}
 
return ret;
 
} catch(someError) {
 
return 'Category preview failed :(';
 
}
 
}
 
  
function APIuserInfoPreviewHTML(article, download) {
+
  var ret = this.html;
var ret=[];
 
var queryobj = {};
 
try{
 
queryobj=getJsObj(download.data).query;
 
} catch(someError) { return 'Userinfo preview failed :('; }
 
  
var user=anyChild(queryobj.users);
+
// fix question marks in wiki links
if (user) {
+
// maybe this'll break some stuff :-(
var globaluserinfo=queryobj.globaluserinfo;
+
ret=ret.replace(RegExp('\(<a href="' + pg.wiki.articlePath + '/[^"]*\)[?]\(.*?"\)', 'g'), '$1%3F$2');
if (user.invalid === '') {
+
ret=ret.replace(RegExp('\(<a href=\'' + pg.wiki.articlePath + '/[^\']*\)[?]\(.*?\'\)', 'g'), '$1%3F$2');
ret.push( popupString( 'Invalid user') );
+
// FIXME fix up % too
} else if (user.missing === '') {
 
ret.push( popupString( 'Not a registered username') );
 
}
 
if( user.blockedby )
 
ret.push('<b>' + popupString('BLOCKED') + '</b>');
 
if( globaluserinfo && ( 'locked' in globaluserinfo || 'hidden' in globaluserinfo ) ) {
 
var lockedSulAccountIsAttachedToThis = true;
 
for( var i=0; globaluserinfo.unattached && i < globaluserinfo.unattached.length; i++) {
 
if ( globaluserinfo.unattached[i].wiki === mw.config.get('wgDBname') ) {
 
lockedSulAccountIsAttachedToThis=false;
 
break;
 
}
 
}
 
if (lockedSulAccountIsAttachedToThis) {
 
if ( 'locked' in globaluserinfo ) ret.push('<b><i>' + popupString('LOCKED') + '</i></b>');
 
if ( 'hidden' in globaluserinfo ) ret.push('<b><i>' + popupString('HIDDEN') + '</i></b>');
 
}
 
}
 
if( getValueOf('popupShowGender') && user.gender ) {
 
switch( user.gender ) {
 
case "male": ret.push( popupString( "\u2642" ) ); break;
 
case "female": ret.push( popupString( "\u2640" ) ); break;
 
}
 
}
 
if( user.groups ) {
 
for( var j=0; j < user.groups.length; j++) {
 
var currentGroup = user.groups[j];
 
if( ["*", "user", "autoconfirmed", "extendedconfirmed"].indexOf( currentGroup ) === -1 ) {
 
ret.push( pg.escapeQuotesHTML(user.groups[j]) );
 
}
 
}
 
}
 
if( globaluserinfo && globaluserinfo.groups ) {
 
for( var k=0; k < globaluserinfo.groups.length; k++) {
 
ret.push( '<i>'+pg.escapeQuotesHTML(globaluserinfo.groups[k])+'</i>' );
 
}
 
}
 
if( user.registration )
 
ret.push( pg.escapeQuotesHTML((user.editcount?user.editcount:'0') + popupString(' edits since: ') + (user.registration?dayFormat(getDateFromTimestamp(user.registration)):'')) );
 
}
 
 
 
if (queryobj.usercontribs && queryobj.usercontribs.length) {
 
ret.push( popupString('last edit on ') + dayFormat(getDateFromTimestamp(queryobj.usercontribs[0].timestamp)) );
 
}
 
 
 
if (queryobj.blocks) {
 
ret.push( popupString( 'IP user') ); //we only request list=blocks for IPs
 
for (var l=0; l<queryobj.blocks.length; l++) {
 
ret.push('<b>' + popupString(queryobj.blocks[l].rangestart === queryobj.blocks[l].rangeend ? 'BLOCKED' : 'RANGEBLOCKED') + '</b>' );
 
}
 
}
 
 
 
ret = '<hr />' + ret.join( ', ' );
+
this.html=ret;
return ret;
+
};
}
+
/**
 +
  Generates the preview and displays it in the current popup.
  
function APIcontribsPreviewHTML(article, download, navpop) {
+
  Does nothing if the generated preview is invalid or consists of whitespace only.
return APIhistoryPreviewHTML(article, download, navpop, true);
+
  Also activates wikilinks in the preview for subpopups if the popupSubpopups option is true.
}
+
*/
 
+
Previewmaker.prototype.showPreview = function () {
function APIhistoryPreviewHTML(article, download, navpop, reallyContribs) {
+
this.makePreview();
try {
+
if (typeof this.html != typeof '') return;
var jsobj=getJsObj(download.data);
+
if (RegExp('^\\s*$').test(this.html)) return;
var edits = [];
+
setPopupHTML('<hr />', 'popupPrePreviewSep', this.owner.idNumber);
if( reallyContribs ) {
+
setPopupTipsAndHTML(this.html, 'popupPreview', this.owner.idNumber, { owner: this.owner });
edits=jsobj.query.usercontribs;
+
var more = (this.fullLength > this.data.length) ? this.moreLink() : '';
} else {
+
setPopupHTML(more, 'popupPreviewMore', this.owner.idNumber);
edits=anyChild(jsobj.query.pages).revisions;
+
};
}
+
/**
 
+
  @private
var ret=editPreviewTable(article, edits, reallyContribs, getTimeOffset());
+
*/
return ret;
+
Previewmaker.prototype.moreLink=function() {
} catch (someError) {
+
var a=document.createElement('a');
return 'History preview failed :-(';
+
a.className='popupMoreLink';
 +
a.innerHTML=popupString('more...');
 +
var savedThis=this;
 +
a.onclick=function() {
 +
savedThis.maxCharacters+=2000;
 +
savedThis.maxSentences+=20;
 +
savedThis.setData();
 +
savedThis.showPreview();
 
}
 
}
 +
return a;
 
}
 
}
  
 
+
/**
//</NOLITE>
+
  @private
// ENDFILE: querypreview.js
+
*/
// STARTFILE: debug.js
+
Previewmaker.prototype.stripLongTemplates = function() {
////////////////////////////////////////////////////////////////////
+
// operates on the HTML!
// Debugging functions
+
this.html=this.html.replace(RegExp('^.{0,1000}[{][{][^}]*?(<(p|br)( /)?>\\s*){2,}([^{}]*?[}][}])?', 'gi'), '');
////////////////////////////////////////////////////////////////////
+
this.html=this.html.split('\n').join(' '); // workaround for <pre> templates
 
+
this.html=this.html.replace(RegExp('[{][{][^}]*<pre>[^}]*[}][}]','gi'), '');
function setupDebugging() {
+
};
//<NOLITE>
+
/**
if (window.popupDebug) { // popupDebug is set from .version
+
  @private
window.log=function(x) { //if(gMsg!='')gMsg += '\n'; gMsg+=time() + ' ' + x; };
+
*/
window.console.log(x);
+
Previewmaker.prototype.killMultilineTemplates = function() {
};
+
this.kill('{{{', '}}}');
window.errlog=function(x) {
+
this.kill(RegExp('\\s*[{][{][^{}]*\\n'), '}}', '{{');
window.console.error(x);
+
};
};
+
// ENDFILE: previewmaker.js
log('Initializing logger');
+
// STARTFILE: querypreview.js
} else {
+
function loadAPIPreview(queryType, article, navpop) {
//</NOLITE>
+
var art=new Title(article).urlString();
window.log = function() {};
+
var url=pg.wiki.apiwikibase + '?format=json&action=query&';
window.errlog = function() {};
+
var htmlGenerator=function(a,d){alert('invalid html generator');};
//<NOLITE>
+
switch (queryType) {
}
+
case 'history':
//</NOLITE>
+
url += 'meta=userinfo&uiprop=options&titles=' + art + '&prop=revisions&rvlimit=' +
}
+
getValueOf('popupHistoryPreviewLimit');
// ENDFILE: debug.js
+
htmlGenerator=APIhistoryPreviewHTML;
// STARTFILE: images.js
+
break;
 
+
case 'category':
// load image of type Title.
+
url += 'list=categorymembers&rawcontinue=&cmtitle=' + art;
function loadImage(image, navpop) {
+
htmlGenerator=APIcategoryPreviewHTML;
if (typeof image.stripNamespace != 'function') { alert('loadImages bad'); }
+
break;
// API call to retrieve image info.
+
case 'userinfo':
 
+
  var username = new Title( article ).userName();
if ( !getValueOf('popupImages') ) return;
+
var usernameart = encodeURIComponent( username );
if ( !isValidImageName(image) ) return false;
+
if (pg.re.ipUser.test(username)) {
+
url += 'list=blocks&bkprop=range&bkip=' + usernameart;
var art=image.urlString();
+
} else {
 
+
url += 'list=users&usprop=blockinfo|groups|editcount|registration&ususers=' + usernameart + "&meta=globaluserinfo&guiprop=groups|unattached&guiuser="+ usernameart;
var url=pg.wiki.apiwikibase + '?format=json&formatversion=2&action=query';
+
}
url += '&prop=imageinfo&iiprop=url|mime&iiurlwidth=' + getValueOf('popupImageSizeLarge');
+
htmlGenerator=APIuserInfoPreviewHTML;
url += '&titles=' + art;
+
break;
 
+
case 'contribs':
 +
var usernameart = encodeURIComponent( new Title( article ).userName() );
 +
url += 'list=usercontribs&meta=userinfo&uiprop=options&ucuser=' + usernameart +
 +
'&uclimit=' + getValueOf('popupContribsPreviewLimit');
 +
htmlGenerator=APIcontribsPreviewHTML;
 +
break;
 +
case 'imagepagepreview':
 +
var trail='';
 +
if (getValueOf('popupImageLinks')) { trail = '&list=imageusage&iutitle=' + art; }
 +
url += 'titles=' + art + '&prop=revisions|imageinfo&rvprop=content' + trail;
 +
htmlGenerator=APIimagepagePreviewHTML;
 +
break;
 +
case 'backlinks':
 +
url += 'list=backlinks&rawcontinue=&bltitle=' + art;
 +
htmlGenerator=APIbacklinksPreviewHTML;
 +
break;
 +
}
 
pendingNavpopTask(navpop);
 
pendingNavpopTask(navpop);
 +
if( !mw.config.get('wgEnableAPI') ) {
 +
/* The API is not available */
 +
htmlGenerator=function(a,d){
 +
return 'This function of navigation popups now requires a MediaWiki ' +
 +
'installation with the <a href="http://www.mediawiki.org/wiki/API">API</a> enabled.'; };
 +
}
 
var callback=function(d){
 
var callback=function(d){
popupsInsertImage(navpop.idNumber, navpop, d);
+
log( "callback of API functions was hit" );
 +
showAPIPreview(queryType, htmlGenerator(article,d,navpop), navpop.idNumber, navpop, d);
 
};
 
};
 +
if (pg.flag.isIE) {
 +
url = url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
 +
}
 
var go = function(){
 
var go = function(){
 
getPageWithCaching(url, callback, navpop);
 
getPageWithCaching(url, callback, navpop);
 
return true;
 
return true;
};
+
}
 
if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
 
if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
else { navpop.addHook(go, 'unhide', 'after', 'DOWNLOAD_IMAGE_QUERY_DATA'); }
+
else { navpop.addHook(go, 'unhide', 'before', 'DOWNLOAD_'+queryType+'_QUERY_DATA'); }
 +
}
  
 +
function linkList(list) {
 +
list.sort(function(x,y) { return (x==y ? 0 : (x<y ? -1 : 1)); });
 +
var buf=[];
 +
for (var i=0; i<list.length; ++i) {
 +
buf.push(wikiLink({article: new Title(list[i]),
 +
  text: list[i].split(' ').join('&nbsp;'),
 +
  action:  'view'}));
 +
}
 +
return buf.join(', ');
 
}
 
}
  
function popupsInsertImage(id, navpop, download) {
+
function getTimeOffset(tz) {
log( "popupsInsertImage");
+
if( tz ) {
var imageinfo;
+
if( tz.indexOf('|') > -1 ) {
try {
+
// New format
var jsObj=getJsObj(download.data);
+
return parseInt(tz.split('|')[1],10);
var imagepage=anyChild(jsObj.query.pages);
+
} else if ( tz.indexOf(':') > -1 ) {
if (typeof imagepage.imageinfo === 'undefined') return;
+
// Old format
imageinfo = imagepage.imageinfo[0];
+
return( parseInt(tz,10)*60 + parseInt(tz.split(':')[1],10) );
} catch (someError) {
+
}
log( "popupsInsertImage failed :(" );
 
return;
 
 
}
 
}
 +
return 0;
 +
}
  
var popupImage = document.getElementById("popupImg"+id);
+
function editPreviewTable(article, h, reallyContribs, timeOffset) {
if (!popupImage) {
+
var html=['<table>'];
log( "could not find insertion point for image");
+
var day=null;
return;
+
var curart=article;
}
+
for (var i=0; i<h.length; ++i) {
 
+
if (reallyContribs) {  
popupImage.width=getValueOf('popupImageSize');
+
var page=h[i]['title']; curart = new Title(page);
popupImage.style.display='inline';
 
 
 
// Set the source for the image.
 
if( imageinfo.thumburl )
 
popupImage.src=imageinfo.thumburl;
 
else if( imageinfo.mime.indexOf("image") === 0 ){
 
popupImage.src=imageinfo.url;
 
log( "a thumb could not be found, using original image" );
 
} else log( "fullsize imagethumb, but not sure if it's an image");
 
 
 
 
 
var a=document.getElementById("popupImageLink"+id);
 
if (a === null) { return null; }
 
 
 
// Determine the action of the surrouding imagelink.
 
switch (getValueOf('popupThumbAction')) {
 
case 'imagepage':
 
if (pg.current.article.namespaceId()!=pg.nsImageId) {
 
a.href=imageinfo.descriptionurl;
 
// FIXME: unreliable pg.idNumber
 
popTipsSoonFn('popupImage' + id)();
 
break;
 
 
}
 
}
/* falls through */
+
var minor=typeof h[i]['minor']=='undefined' ? '' : '<b>m </b>';
case 'sizetoggle':
+
var editDate=adjustDate(getDateFromTimestamp(h[i].timestamp), timeOffset);
a.onclick=toggleSize;
+
var thisDay = dayFormat(editDate);
a.title=popupString('Toggle image size');
+
var thisTime = timeFormat(editDate);
return;
+
if (thisDay==day) { thisDay=''; }
case 'linkfull':
+
else { day=thisDay; }
a.href = imageinfo.url;
+
if (thisDay) {
a.title=popupString('Open full-size image');
+
html.push( '<tr><td colspan=3><span class="popup_history_date">' +
return;
+
  thisDay+'</span></td></tr>' );
 +
}
 +
html.push('<tr class="popup_history_row_' + ( (i%2) ? 'odd' : 'even') + '">');
 +
html.push('<td>(<a href="' + pg.wiki.titlebase + new Title(curart).urlString() +
 +
'&diff=prev&oldid=' + h[i]['revid'] + '">' + popupString('last') + '</a>)</td>');
 +
html.push('<td>' +
 +
'<a href="' + pg.wiki.titlebase + new Title(curart).urlString() +
 +
'&oldid=' + h[i]['revid'] + '">' + thisTime + '</a></td>');
 +
var col3url='', col3txt='';
 +
if (!reallyContribs) {
 +
var user=h[i]['user'];
 +
if( typeof h[i]['userhidden'] == "undefined" ) {
 +
if( pg.re.ipUser.test(user) ) {
 +
col3url=pg.wiki.titlebase + mw.config.get('wgFormattedNamespaces')[pg.nsSpecialId] + ':Contributions&target=' + new Title(user).urlString();
 +
} else {
 +
col3url=pg.wiki.titlebase + mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' + new Title(user).urlString();
 +
}
 +
col3txt=pg.escapeQuotesHTML(user);
 +
} else {
 +
col3url=getValueOf('popupRevDelUrl');
 +
col3txt=pg.escapeQuotesHTML( popupString('revdel'));
 +
}
 +
} else {
 +
col3url=pg.wiki.titlebase + curart.urlString();
 +
col3txt=pg.escapeQuotesHTML(page);
 +
}
 +
html.push('<td>' + (reallyContribs ? minor : '') +
 +
'<a href="' + col3url + '">' + col3txt + '</a></td>');
 +
var comment='';
 +
var c=h[i].comment || h[i]['*'];
 +
if (c) {
 +
comment=new Previewmaker(c, new Title(curart).toUrl()).editSummaryPreview();
 +
} else if (typeof h[i]['commenthidden'] != "undefined" ) {
 +
comment=popupString('revdel');
 +
}
 +
html.push('<td>' + (!reallyContribs ? minor : '') + comment + '</td>');
 +
html.push('</tr>');
 +
html=[html.join('')];
 
}
 
}
 
+
html.push('</table>');
 +
return html.join('');
 
}
 
}
  
// Toggles the image between inline small and navpop fullwidth.
+
function getDateFromTimestamp(t) {
// It's the same image, no actual sizechange occurs, only display width.
+
var s=t.split(/[^0-9]/);
function toggleSize() {
+
switch(s.length) {
var imgContainer=this;
+
case 0: return null;
if (!imgContainer) {
+
case 1: return new Date(s[0]);
alert('imgContainer is null :/');
+
case 2: return new Date(s[0], s[1]-1);
return;
+
case 3: return new Date(s[0], s[1]-1, s[2]);
}
+
case 4: return new Date(s[0], s[1]-1, s[2], s[3]);
var img=imgContainer.firstChild;
+
case 5: return new Date(s[0], s[1]-1, s[2], s[3], s[4]);
if (!img) {
+
case 6: return new Date(s[0], s[1]-1, s[2], s[3], s[4], s[5]);
alert('img is null :/');
+
default: return new Date(s[0], s[1]-1, s[2], s[3], s[4], s[5], s[6]);
return;
 
 
}
 
}
 +
}
  
if (!img.style.width || img.style.width==='') {
+
function adjustDate(d, offset) {
img.style.width='100%';
+
// offset is in minutes
} else {
+
var o=offset * 60 * 1000;
img.style.width='';
+
return new Date( +d + o);
}
 
 
}
 
}
  
// Returns one title of an image from wikiText.
+
function dayFormat(editDate, utc) {
function getValidImageFromWikiText(wikiText) {
+
if (utc) { return map(zeroFill, [editDate.getUTCFullYear(), editDate.getUTCMonth()+1, editDate.getUTCDate()]).join('-'); }
// nb in pg.re.image we're interested in the second bracketed expression
+
return map(zeroFill, [editDate.getFullYear(), editDate.getMonth()+1, editDate.getDate()]).join('-');
// this may change if the regex changes :-(
+
}
//var match=pg.re.image.exec(wikiText);
 
var matched=null;
 
var match;
 
// strip html comments, used by evil bots :-(
 
var t = removeMatchesUnless(wikiText, RegExp('(<!--[\\s\\S]*?-->)'), 1,
 
RegExp('^<!--[^[]*popup', 'i'));
 
  
while ( ( match = pg.re.image.exec(t) ) ) {
+
function timeFormat(editDate, utc) {
// now find a sane image name - exclude templates by seeking {
+
if (utc) { return map(zeroFill, [editDate.getUTCHours(), editDate.getUTCMinutes(), editDate.getUTCSeconds()]).join(':'); }
var m = match[2] || match[6];
+
return map(zeroFill, [editDate.getHours(), editDate.getMinutes(), editDate.getSeconds()]).join(':');
if ( isValidImageName(m) ) {
 
matched=m;
 
break;
 
}
 
}
 
pg.re.image.lastIndex=0;
 
if (!matched) { return null; }
 
return mw.config.get('wgFormattedNamespaces')[pg.nsImageId]+':'+upcaseFirst(matched);
 
 
}
 
}
  
function removeMatchesUnless(str, re1, parencount, re2) {
+
function showAPIPreview(queryType, html, id, navpop, download) {
var split=str.parenSplit(re1);
+
// DJ: done
var c=parencount + 1;
+
var target='popupPreview';
for (var i=0; i<split.length; ++i) {
+
switch (queryType) {
if ( i%c === 0 || re2.test(split[i]) ) { continue; }
+
case 'imagelinks':
split[i]='';
+
case 'category':
 +
case 'userinfo':
 +
target='popupPostPreview'; break;
 
}
 
}
return split.join('');
+
setPopupTipsAndHTML(html, target, id);
 +
completedNavpopTask(navpop);
 
}
 
}
  
//</NOLITE>
+
function APIbacklinksPreviewHTML(article, download, navpop) {
// ENDFILE: images.js
+
try {
// STARTFILE: namespaces.js
+
var jsObj=getJsObj(download.data);
// Set up namespaces and other non-strings.js localization
+
  var list=jsObj.query.backlinks;
// (currently that means redirs too)
+
} catch (someError) { return 'backlinksPreviewHTML went wonky'; }
 +
var html=[];
 +
if (!list) { return popupString('No backlinks found'); }
 +
for ( var i=0; i < list.length; i++ ) {
 +
var t=new Title(list[i]['title']);
 +
html.push('<a href="' + pg.wiki.titlebase + t.urlString() + '">' + t + '</a>');
 +
}
 +
html=html.join(', ');
 +
if (jsObj['query-continue'] && jsObj['query-continue'].backlinks && jsObj['query-continue'].backlinks.blcontinue) {
 +
html += popupString(' and more');
 +
}
 +
return html;
 +
}
  
function setNamespaces() {
+
function APIsharedImagePagePreviewHTML(obj) {
pg.nsSpecialId  = -1;
+
log( "APIsharedImagePagePreviewHTML" );
pg.nsMainspaceId = 0;
+
var popupid = obj['requestid'];
pg.nsImageId    = 6;
+
if( obj['query'] && obj['query']['pages'] )
pg.nsUserId      = 2;
+
{
pg.nsUsertalkId  = 3;
+
var page=anyChild(obj['query']['pages']);
pg.nsCategoryId  = 14;
+
var content=(page && page.revisions ) ? page.revisions[0]['*'] : null;
pg.nsTemplateId  = 10;
+
if( content )
 +
{
 +
/* Not entirely safe, but the best we can do */
 +
var p=new Previewmaker(content, pg.current.link.navpopup.article, pg.current.link.navpopup);
 +
p.makePreview();
 +
setPopupHTML( p.html, "popupSecondPreview", popupid );
 +
}
 +
}
 
}
 
}
  
 +
function APIimagepagePreviewHTML(article, download, navpop) {
 +
try {
 +
var jsObj=getJsObj(download.data);
 +
var page=anyChild(jsObj.query.pages);
 +
var content=(page && page.revisions ) ? page.revisions[0]['*'] : null;
 +
} catch (someError) {
 +
return 'API imagepage preview failed :(';
 +
}
 +
var ret='';
 +
var alt='';
 +
try{alt=navpop.parentAnchor.childNodes[0].alt;} catch(e){}
 +
if (alt) {
 +
ret = ret + '<hr /><b>' + popupString('Alt text:') + '</b> ' + pg.escapeQuotesHTML(alt);
 +
}
 +
if (content) {
 +
var p=prepPreviewmaker(content, article, navpop);
 +
p.makePreview();
 +
if (p.html) { ret += '<hr />' + p.html; }
 +
}
 +
if (content!==null && getValueOf('popupSummaryData')) {
 +
var info=getPageInfo(content, download);
 +
log(info);
 +
setPopupTrailer(info, navpop.idNumber);
 +
}
 +
if (page && page.imagerepository == "shared" ) {
 +
var art=new Title(article);
 +
var encart = encodeURIComponent( "File:" + art.stripNamespace() );
 +
var shared_url =  pg.wiki.apicommonsbase + '?format=json&callback=APIsharedImagePagePreviewHTML' +
 +
'&requestid=' + navpop.idNumber +
 +
'&action=query&prop=revisions&rvprop=content&titles=' + encart;
 +
if (pg.flag.isIE) {
 +
shared_url = shared_url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
 +
}
  
function setRedirs() {
+
ret = ret +'<hr />' + popupString( 'Image from Commons') +
var r='redirect';
+
': <a href="' + pg.wiki.commonsbase + '?title=' + encart + '">' +
var R='REDIRECT';
+
popupString( 'Description page') + '</a>';
var redirLists={
+
mw.loader.load( shared_url );
//<NOLITE>
+
}
'ar':  [ R, 'تحويل' ],
+
showAPIPreview('imagelinks', APIimagelinksPreviewHTML(article,download), navpop.idNumber, download);
'be': [ r, 'перанакіраваньне' ],
+
return ret;
'bg':  [ r, 'пренасочване', 'виж' ],
 
'bs':  [ r, 'Preusmjeri', 'preusmjeri', 'PREUSMJERI' ],
 
'cs':  [ R, 'PŘESMĚRUJ' ],
 
'cy':  [ r, 'ail-cyfeirio' ],
 
'de':  [ R, 'WEITERLEITUNG' ],
 
'el':  [ R, 'ΑΝΑΚΑΤΕΥΘΥΝΣΗ'],
 
'eo':  [ R, 'ALIDIREKTU', 'ALIDIREKTI' ],
 
'es':  [ R, 'REDIRECCIÓN' ],
 
'et':  [ r, 'suuna' ],
 
'ga':  [ r, 'athsheoladh' ],
 
'gl':  [ r, 'REDIRECCIÓN', 'REDIRECIONAMENTO'],
 
'he':  [ R, 'הפניה' ],
 
'hu':  [ R, 'ÁTIRÁNYÍTÁS' ],
 
'is':  [ r, 'tilvísun', 'TILVÍSUN' ],
 
'it':  [ R, 'RINVIA', 'Rinvia'],
 
'ja':  [ R, '転送' ],
 
'mk':  [ r, 'пренасочување', 'види' ],
 
'nds': [ r, 'wiederleiden' ],
 
'nl':  [ R, 'DOORVERWIJZING' ],
 
'nn':  [ r, 'omdiriger' ],
 
'pl':  [ R, 'PATRZ', 'PRZEKIERUJ', 'TAM' ],
 
'pt':  [ R, 'redir' ],
 
'ru':  [ R, 'ПЕРЕНАПРАВЛЕНИЕ', 'ПЕРЕНАПР' ],
 
'sk':  [ r, 'presmeruj' ],
 
'sr':  [ r, 'Преусмери', 'преусмери', 'ПРЕУСМЕРИ', 'Preusmeri', 'preusmeri', 'PREUSMERI' ],
 
'tt':  [ R, 'yünältü', 'перенаправление', 'перенапр' ],
 
'uk':  [ R, 'ПЕРЕНАПРАВЛЕННЯ', 'ПЕРЕНАПР' ],
 
'vi':  [ r, 'đổi' ],
 
'zh':  [ R, '重定向'] // no comma
 
//</NOLITE>
 
};
 
var redirList=redirLists[ pg.wiki.lang ] || [r, R];
 
// Mediawiki is very tolerant about what comes after the #redirect at the start
 
pg.re.redirect=RegExp('^\\s*[#](' + redirList.join('|') + ').*?\\[{2}([^\\|\\]]*)(|[^\\]]*)?\\]{2}\\s*(.*)', 'i');
 
 
}
 
}
  
function setInterwiki() {
+
function APIimagelinksPreviewHTML(article, download) {
if (pg.wiki.wikimedia) {
+
try {
// From https://meta.wikimedia.org/wiki/List_of_Wikipedias
+
var jsobj=getJsObj(download.data);
pg.wiki.interwiki='aa|ab|ace|af|ak|als|am|an|ang|ar|arc|arz|as|ast|av|ay|az|ba|bar|bat-smg|bcl|be|be-x-old|bg|bh|bi|bjn|bm|bn|bo|bpy|br|bs|bug|bxr|ca|cbk-zam|cdo|ce|ceb|ch|cho|chr|chy|ckb|co|cr|crh|cs|csb|cu|cv|cy|da|de|diq|dsb|dv|dz|ee|el|eml|en|eo|es|et|eu|ext|fa|ff|fi|fiu-vro|fj|fo|fr|frp|frr|fur|fy|ga|gag|gan|gd|gl|glk|gn|got|gu|gv|ha|hak|haw|he|hi|hif|ho|hr|hsb|ht|hu|hy|hz|ia|id|ie|ig|ii|ik|ilo|io|is|it|iu|ja|jbo|jv|ka|kaa|kab|kbd|kg|ki|kj|kk|kl|km|kn|ko|koi|kr|krc|ks|ksh|ku|kv|kw|ky|la|lad|lb|lbe|lg|li|lij|lmo|ln|lo|lt|ltg|lv|map-bms|mdf|mg|mh|mhr|mi|mk|ml|mn|mo|mr|mrj|ms|mt|mus|mwl|my|myv|mzn|na|nah|nap|nds|nds-nl|ne|new|ng|nl|nn|no|nov|nrm|nv|ny|oc|om|or|os|pa|pag|pam|pap|pcd|pdc|pfl|pi|pih|pl|pms|pnb|pnt|ps|pt|qu|rm|rmy|rn|ro|roa-rup|roa-tara|ru|rue|rw|sa|sah|sc|scn|sco|sd|se|sg|sh|si|simple|sk|sl|sm|sn|so|sq|sr|srn|ss|st|stq|su|sv|sw|szl|ta|te|tet|tg|th|ti|tk|tl|tn|to|tpi|tr|ts|tt|tum|tw|ty|udm|ug|uk|ur|uz|ve|vec|vi|vls|vo|wa|war|wo|wuu|xal|xh|yi|yo|za|zea|zh|zh-classical|zh-min-nan|zh-yue|zu';
+
var list=jsobj.query.imageusage;
pg.re.interwiki=RegExp('^'+pg.wiki.interwiki+':');
+
if (!list) { return popupString('No image links found'); }
} else {
+
} catch(someError) { return 'Image links preview generation failed :('; }
pg.wiki.interwiki=null;
+
var ret=[];
pg.re.interwiki=RegExp('^$');
+
for (var i=0; i < list.length; i++) {
}
+
ret.push(list[i]['title']);
 +
}
 +
if (ret.length === 0) { return popupString('No image links found'); }
 +
return '<h2>' + popupString('File links') + '</h2>' + linkList(ret);
 
}
 
}
  
// return a regexp pattern matching all variants to write the given namespace
+
function APIcategoryPreviewHTML(article, download) {
function nsRe(namespaceId) {
+
try{
var imageNamespaceVariants = [];
+
var jsobj=getJsObj(download.data);
jQuery.each(mw.config.get('wgNamespaceIds'), function(_localizedNamespaceLc, _namespaceId) {
+
var list=jsobj.query.categorymembers;
if (_namespaceId!=namespaceId) return;
+
} catch(someError) { return 'Category preview failed :('; }
_localizedNamespaceLc = upcaseFirst(_localizedNamespaceLc);
+
var ret=[];
imageNamespaceVariants.push(mw.util.escapeRegExp(_localizedNamespaceLc).split(' ').join('[ _]'));
+
for (var p=0; p < list.length; p++) {
imageNamespaceVariants.push(mw.util.escapeRegExp(encodeURI(_localizedNamespaceLc)));
+
  ret.push(list[p]['title']);
});
+
}
 
+
if (ret.length === 0) { return popupString('Empty category'); }
return '(?:' + imageNamespaceVariants.join('|') + ')';
+
ret = '<h2>' + tprintf('Category members (%s shown)', [ret.length]) + '</h2>' +linkList(ret);
 +
if (jsobj['query-continue'] && jsobj['query-continue'].categorymembers && jsobj['query-continue'].categorymembers.cmcontinue) {
 +
ret += popupString(' and more');
 +
}
 +
return ret;
 
}
 
}
  
function nsReImage() {
+
function APIuserInfoPreviewHTML(article, download) {
return nsRe(pg.nsImageId);
+
var ret=[];
}
+
try{
// ENDFILE: namespaces.js
+
var queryobj=getJsObj(download.data).query;
// STARTFILE: selpop.js
+
} catch(someError) { return 'Userinfo preview failed :('; }
//<NOLITE>
+
function getEditboxSelection() {
+
var user=anyChild(queryobj.users);
// see http://www.webgurusforum.com/8/12/0
+
if (user) {
var editbox;
+
var globaluserinfo=queryobj.globaluserinfo;
try {
+
if (user.invalid == '') {
editbox=document.editform.wpTextbox1;
+
ret.push( popupString( 'Invalid user') );
} catch (dang) { return; }
+
} else if (user.missing == '') {
// IE, Opera
+
ret.push( popupString( 'Not a registered username') );
if (document.selection) { return document.selection.createRange().text; }
+
}
// Mozilla
+
if( user.blockedby )
var selStart = editbox.selectionStart;
+
ret.push('<b>' + popupString('BLOCKED') + '</b>');
var selEnd = editbox.selectionEnd;
+
if( globaluserinfo && (globaluserinfo.locked != null || globaluserinfo.hidden != null) ) {
return (editbox.value).substring(selStart, selEnd);
+
var lockedSulAccountIsAttachedToThis = true;
 +
for( var i=0; globaluserinfo.unattached && i < globaluserinfo.unattached.length; i++) {
 +
if ( globaluserinfo.unattached[i].wiki===mw.config.get('wgDBname') ) { lockedSulAccountIsAttachedToThis=false; break; }
 +
}
 +
if (lockedSulAccountIsAttachedToThis) {
 +
if (globaluserinfo.locked != null) ret.push('<b><i>' + popupString('LOCKED') + '</i></b>');
 +
if (globaluserinfo.hidden != null) ret.push('<b><i>' + popupString('HIDDEN') + '</i></b>');
 +
}
 +
}
 +
for( var i=0; (user.groups && i < user.groups.length); i++) {
 +
switch (user.groups[i]) {
 +
case '*':
 +
case 'user':
 +
case 'autoconfirmed':
 +
break;
 +
default:
 +
ret.push( pg.escapeQuotesHTML(user.groups[i]) );
 +
}
 +
}
 +
for( var i=0; (globaluserinfo && globaluserinfo.groups && i < globaluserinfo.groups.length); i++) {
 +
ret.push( '<i>'+pg.escapeQuotesHTML(globaluserinfo.groups[i])+'</i>' );
 +
}
 +
if( user.editcount || user.registration )
 +
ret.push( pg.escapeQuotesHTML((user.editcount?user.editcount:'') + popupString(' edits since: ') + (user.registration?dayFormat(getDateFromTimestamp(user.registration)):'')) );
 +
}
 +
 +
if (queryobj.blocks) {
 +
ret.push( popupString( 'IP user') ); //we only request list=blocks for IPs
 +
for (var i=0; i<queryobj.blocks.length; i++) {
 +
ret.push('<b>' + popupString(queryobj.blocks[i].rangestart===queryobj.blocks[i].rangeend ? 'BLOCKED' : 'RANGEBLOCKED') + '</b>' );
 +
}
 +
}
 +
 +
ret = '<hr />' + ret.join( ', ' );
 +
return ret;
 +
}
 +
 
 +
function APIcontribsPreviewHTML(article, download, navpop) {
 +
return APIhistoryPreviewHTML(article, download, navpop, true);
 
}
 
}
  
function doSelectionPopup() {
+
function APIhistoryPreviewHTML(article, download, navpop, reallyContribs) {
// popup if the selection looks like [[foo|anything afterwards at all
+
try {
// or [[foo|bar]]text without ']]'
+
var jsobj=getJsObj(download.data);
// or [[foo|bar]]
+
var tz=jsobj.query.userinfo.options.timecorrection;
var sel=getEditboxSelection();
+
if( reallyContribs )
var open=sel.indexOf('[[');
+
var edits=jsobj.query.usercontribs;
var pipe=sel.indexOf('|');
+
else
var close=sel.indexOf(']]');
+
var edits=anyChild(jsobj.query.pages)['revisions'];
if (open == -1 || ( pipe == -1 && close == -1) ) { return; }
+
} catch (someError) {
if (pipe != -1 && open > pipe || close != -1 && open > close) { return; }
+
return 'History preview failed :-(';
if (getValueOf('popupOnEditSelection')=='boxpreview') {
 
return doSeparateSelectionPopup(sel);
 
}
 
var article=new Title(sel.substring(open+2, (pipe < 0) ? close : pipe)).urlString();
 
if (close > 0 && sel.substring(close+2).indexOf('[[') >= 0) {
 
return;  
 
}
 
var a=document.createElement('a');
 
a.href=pg.wiki.titlebase + article;
 
mouseOverWikiLink2(a);
 
if (a.navpopup) {
 
a.navpopup.addHook(function(){runStopPopupTimer(a.navpopup);}, 'unhide', 'after');
 
 
}
 
}
 +
var timeOffset = getTimeOffset(tz);
 +
Cookie.create('popTz', timeOffset, 1);
 +
 +
var ret=editPreviewTable(article, edits, reallyContribs, timeOffset);
 +
return ret;
 
}
 
}
  
function doSeparateSelectionPopup(str) {
+
 
var div=document.getElementById('selectionPreview');
+
//</NOLITE>
if (!div) {
+
// ENDFILE: querypreview.js
div = document.createElement('div');
+
// STARTFILE: debug.js
div.id='selectionPreview';
+
////////////////////////////////////////////////////////////////////
try {
+
// Debugging functions
var box=document.editform.wpTextbox1;
+
////////////////////////////////////////////////////////////////////
box.parentNode.insertBefore(div, box);
+
 
} catch (error) {
+
function log(){}; // dummy to stop errors
return;
+
function setupDebugging() {
 +
//<NOLITE>
 +
if (window.popupDebug) { // popupDebug is set from .version
 +
window.log=function(x) { //if(gMsg!='')gMsg += '\n'; gMsg+=time() + ' ' + x; };
 +
window.console.log(x);
 +
}
 +
window.errlog=function(x) {
 +
window.console.error(x);
 
}
 
}
 +
log('Initializing logger');
 +
} else {
 +
//</NOLITE>
 +
window.log = function(x) {};
 +
window.errlog = function(x) {};
 +
//<NOLITE>
 
}
 
}
div.innerHTML=wiki2html(str);
+
//</NOLITE>
div.ranSetupTooltipsAlready = false;
 
popTipsSoonFn('selectionPreview')();
 
 
}
 
}
//</NOLITE>
+
// ENDFILE: debug.js
// ENDFILE: selpop.js
+
// STARTFILE: images.js
// STARTFILE: navpopup.js
 
/**
 
  @fileoverview  Defines two classes: {@link Navpopup} and {@link Mousetracker}.
 
  
  <code>Navpopup</code> describes popups: when they appear, where, what
+
// load image of type Title.
  they look like and so on.
+
function loadImage(image, navpop) {
 +
if (typeof image.stripNamespace != 'function') { alert('loadImages bad'); }
 +
// API call to retrieve image info.
  
  <code>Mousetracker</code> "captures" the mouse using
+
if ( !getValueOf('popupImages') || !mw.config.get('wgEnableAPI') ) return;
  <code>document.onmousemove</code>.
+
if ( !isValidImageName(image) ) return false;
*/
+
 +
var art=image.urlString();
  
 +
var url=pg.wiki.apiwikibase + '?format=json&action=query';
 +
url += '&prop=imageinfo&iiprop=url|mime&iiurlwidth=' + getValueOf('popupImageSizeLarge');;
 +
url += '&titles=' + art;
 +
if (pg.flag.isIE) {
 +
url = url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
 +
}
  
/**
+
pendingNavpopTask(navpop);
  Creates a new Mousetracker.
+
var callback=function(d){
  @constructor
+
popupsInsertImage(navpop.idNumber, navpop, d);
  @class The Mousetracker class. This monitors mouse movements and manages associated hooks.
+
};
*/
+
var go = function(){
function Mousetracker() {
+
getPageWithCaching(url, callback, navpop);
/**
+
return true;
  Interval to regularly run the hooks anyway, in milliseconds.
+
}
  @type Integer
+
if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
*/
+
else { navpop.addHook(go, 'unhide', 'after', 'DOWNLOAD_IMAGE_QUERY_DATA'); }
this.loopDelay=400;
 
  
/**
 
  Timer for the loop.
 
  @type Timer
 
*/
 
this.timer=null;
 
 
/**
 
  Flag - are we switched on?
 
  @type Boolean
 
*/
 
this.active=false;
 
/**
 
  Flag - are we probably inaccurate, i.e. not reflecting the actual mouse position?
 
*/
 
this.dirty=true;
 
/**
 
  Array of hook functions.
 
  @private
 
  @type Array
 
*/
 
this.hooks=[];
 
 
}
 
}
  
/**
+
function popupsInsertImage(id, navpop, download) {
  Adds a hook, to be called when we get events.
+
log( "popupsInsertImage");
  @param {Function} f A function which is called as
+
try {
  <code>f(x,y)</code>. It should return <code>true</code> when it
+
var jsObj=getJsObj(download.data);
  wants to be removed, and <code>false</code> otherwise.
+
var imagepage=anyChild(jsObj.query.pages);
*/
+
if (typeof imagepage.imageinfo === 'undefined') return;
Mousetracker.prototype.addHook = function (f) {
+
var imageinfo = imagepage.imageinfo[0];
this.hooks.push(f);
+
} catch (someError) {
};
+
log( "popupsInsertImage failed :(" );
 
+
return;
/**
 
  Runs hooks, passing them the x
 
  and y coords of the mouse.  Hook functions that return true are
 
  passed to {@link Mousetracker#removeHooks} for removal.
 
  @private
 
*/
 
Mousetracker.prototype.runHooks = function () {
 
if (!this.hooks || !this.hooks.length) { return; }
 
//log('Mousetracker.runHooks; we got some hooks to run');
 
var remove=false;
 
var removeObj={};
 
// this method gets called a LOT -
 
// pre-cache some variables
 
var x=this.x, y=this.y, len = this.hooks.length;
 
 
 
for (var i=0; i<len; ++i) {
 
//~ run the hook function, and remove it if it returns true
 
if (this.hooks[i](x, y)===true) {
 
remove=true;
 
removeObj[i]=true;
 
}
 
 
}
 
}
if (remove) { this.removeHooks(removeObj); }
 
};
 
  
/**
+
var popupImage = document.getElementById("popupImg"+id);
  Removes hooks.
+
if (!popupImage) {
  @private
+
log( "could not find insertion point for image");
  @param {Object} removeObj An object whose keys are the index
+
return;
  numbers of functions for removal, with values that evaluate to true
 
*/
 
Mousetracker.prototype.removeHooks = function(removeObj) {
 
var newHooks=[];
 
var len = this.hooks.length;
 
for (var i=0; i<len; ++i) {
 
if (! removeObj[i]) { newHooks.push(this.hooks[i]); }
 
 
}
 
}
this.hooks=newHooks;
 
};
 
  
 +
popupImage.width=getValueOf('popupImageSize');
 +
popupImage.style.display='inline';
  
/**
+
// Set the source for the image.
  Event handler for mouse wiggles.
+
if( imageinfo.thumburl )
  We simply grab the event, set x and y and run the hooks.
+
popupImage.src=imageinfo.thumburl;
  This makes the cpu all hot and bothered :-(
+
else if( imageinfo.mime.indexOf("image") == 0 ){
  @private
+
popupImage.src=imageinfo.url;
  @param {Event} e Mousemove event
+
log( "a thumb could not be found, using original image" );
*/
+
} else log( "fullsize imagethumb, but not sure if it's an image");
Mousetracker.prototype.track=function (e) {
 
//~ Apparently this is needed in IE.
 
e = e || window.event;
 
var x, y;
 
if (e) {
 
if (e.pageX) { x=e.pageX; y=e.pageY; }
 
else if (typeof e.clientX!='undefined') {
 
var left, top, docElt = document.documentElement;
 
  
if (docElt) { left=docElt.scrollLeft; }
 
left = left || document.body.scrollLeft || document.scrollLeft || 0;
 
  
if (docElt) { top=docElt.scrollTop; }
+
var a=document.getElementById("popupImageLink"+id);
top = top || document.body.scrollTop || document.scrollTop || 0;
+
if (a === null) { return null; }
  
x=e.clientX + left;
+
// Determine the action of the surrouding imagelink.
y=e.clientY + top;
+
switch (getValueOf('popupThumbAction')) {
} else { return; }
+
case 'imagepage':
this.setPosition(x,y);
+
if (pg.current.article.namespaceId()!=pg.nsImageId) {
 +
a.href=imageinfo.descriptionurl;
 +
// FIXME: unreliable pg.idNumber
 +
popTipsSoonFn('popupImage' + id)();
 +
break;
 +
} // else fall through
 +
case 'sizetoggle':
 +
a.onclick=toggleSize;
 +
a.title=popupString('Toggle image size');
 +
return;
 +
case 'linkfull':
 +
a.href = imageinfo.url;
 +
a.title=popupString('Open full-size image');
 +
return;
 
}
 
}
};
 
  
/**
+
}
  Sets the x and y coordinates stored and takes appropriate action,
+
 
  running hooks as appropriate.
+
// Toggles the image between inline small and navpop fullwidth.
  @param {Integer} x, y Screen coordinates to set
+
// It's the same image, no actual sizechange occurs, only display width.
*/
+
function toggleSize() {
 +
var imgContainer=this;
 +
if (!imgContainer) { alert('imgContainer is null :/'); return;}
 +
img=imgContainer.firstChild;
 +
if (!img) { alert('img is null :/'); return;}
 +
if (!img.style.width || img.style.width=='') { img.style.width='100%'; }
 +
else { img.style.width=''; }
 +
}
 +
 
 +
// Returns one title of an image from wikiText.
 +
function getValidImageFromWikiText(wikiText) {
 +
// nb in pg.re.image we're interested in the second bracketed expression
 +
// this may change if the regex changes :-(
 +
//var match=pg.re.image.exec(wikiText);
 +
var matched=null;
 +
var match;
 +
// strip html comments, used by evil bots :-(
 +
var t = removeMatchesUnless(wikiText, RegExp('(<!--[\\s\\S]*?-->)'), 1,
 +
RegExp('^<!--[^[]*popup', 'i'));
  
Mousetracker.prototype.setPosition=function(x,y) {
+
while ( match = pg.re.image.exec(t) ) {
this.x = x;
+
// now find a sane image name - exclude templates by seeking {
this.y = y;
+
var m = match[2] || match[6];
if (this.dirty || this.hooks.length === 0) { this.dirty=false; return; }
+
if ( isValidImageName(m) ) {
if (typeof this.lastHook_x != 'number') { this.lastHook_x = -100; this.lastHook_y=-100; }
+
matched=m;
var diff = (this.lastHook_x - x)*(this.lastHook_y - y);
+
break;
diff = (diff >= 0) ? diff : -diff;
+
}
if ( diff > 1 ) {
 
this.lastHook_x=x;
 
this.lastHook_y=y;
 
if (this.dirty) { this.dirty = false; }
 
else { this.runHooks(); }
 
 
}
 
}
};
+
pg.re.image.lastIndex=0;
 +
if (!matched) { return null; }
 +
return mw.config.get('wgFormattedNamespaces')[pg.nsImageId]+':'+upcaseFirst(matched);
 +
}
  
/**
+
function removeMatchesUnless(str, re1, parencount, re2) {
  Sets things in motion, unless they are already that is, registering an event handler on <code>document.onmousemove</code>.
+
var split=str.parenSplit(re1);
  A half-hearted attempt is made to preserve the old event handler if there is one.
+
var c=parencount + 1;
*/
+
for (var i=0; i<split.length; ++i) {
Mousetracker.prototype.enable = function () {
+
if ( i%c === 0 || re2.test(split[i]) ) { continue; }
if (this.active) { return; }
+
split[i]='';
this.active=true;
+
}
//~ Save the current handler for mousemove events. This isn't too
+
return split.join('');
//~ robust, of course.
+
}
this.savedHandler=document.onmousemove;
+
 
//~ Gotta save @tt{this} again for the closure, and use apply for
+
//</NOLITE>
//~ the member function.
+
// ENDFILE: images.js
var savedThis=this;
+
// STARTFILE: namespaces.js
document.onmousemove=function (e) {savedThis.track.apply(savedThis, [e]);};
+
// Set up namespaces and other non-strings.js localization
if (this.loopDelay) { this.timer = setInterval(function() { //log('loop delay in mousetracker is working');
+
// (currently that means redirs too)
savedThis.runHooks();}, this.loopDelay); }
+
 
};
+
 
 +
function namespaceListToRegex(list) {return RegExp('^('+list.join('|').split(' ').join('[ _]')+'):');};
  
/**
+
function setNamespaces() {
  Disables the tracker, removing the event handler.
+
pg.nsSpecialId  = -1;
*/
+
pg.nsMainspaceId  = 0;
Mousetracker.prototype.disable = function () {
+
pg.nsImageId = 6;
if (!this.active) { return; }
+
pg.nsUserId   = 2;
if ($.isFunction(this.savedHandler)) {
+
pg.nsUsertalkId  = 3;
document.onmousemove=this.savedHandler;
+
pg.nsCategoryId  = 14;
} else { delete document.onmousemove; }
+
pg.nsTemplateId  = 10;
if (this.timer) { clearInterval(this.timer); }
+
}
this.active=false;
 
};
 
  
/**
+
 
  Creates a new Navpopup.
+
function setRedirs() {
  Gets a UID for the popup and
+
var r='redirect';
  @param init Contructor object. If <code>init.draggable</code> is true or absent, the popup becomes draggable.
+
var R='REDIRECT';
  @constructor
+
var redirLists={
  @class The Navpopup class. This generates popup hints, and does some management of them.
+
//<NOLITE>
*/
+
'ar':  [ R, 'تحويل' ],
function Navpopup(/*init*/) {
+
'be':  [ r, 'перанакіраваньне' ],
//alert('new Navpopup(init)');
+
'bg':  [ r, 'пренасочване', 'виж' ],
/** UID for each Navpopup instance.
+
'bs':  [ r, 'Preusmjeri', 'preusmjeri', 'PREUSMJERI' ],
Read-only.
+
'cs':  [ R, 'PŘESMĚRUJ' ],
@type integer
+
'cy':  [ r, 'ail-cyfeirio' ],
*/
+
'de':  [ R, 'WEITERLEITUNG' ],
this.uid=Navpopup.uid++;
+
'el':  [ R, 'ΑΝΑΚΑΤΕΥΘΥΝΣΗ'],
/**
+
'eo':  [ R, 'ALIDIREKTU', 'ALIDIREKTI' ],
  Read-only flag for current visibility of the popup.
+
'es':  [ R, 'REDIRECCIÓN' ],
  @type boolean
+
'et':  [ r, 'suuna' ],
  @private
+
'ga':  [ r, 'athsheoladh' ],
*/
+
'gl':  [ r, 'REDIRECCIÓN', 'REDIRECIONAMENTO'],
this.visible=false;
+
'he':  [ R, 'הפניה' ],
/** Flag to be set when we want to cancel a previous request to
+
'hu':  [ R, 'ÁTIRÁNYÍTÁS' ],
show the popup in a little while.
+
'is':  [ r, 'tilvísun', 'TILVÍSUN' ],
@private
+
'it':  [ R, 'RINVIA', 'Rinvia'],
@type boolean
+
'ja':  [ R, '転送' ],
*/
+
'mk':  [ r, 'пренасочување', 'види' ],
this.noshow=false;
+
'nds': [ r, 'wiederleiden' ],
/** Categorised list of hooks.
+
'nl':  [ R, 'DOORVERWIJZING' ],
@see #runHooks
+
'nn':  [ r, 'omdiriger' ],
@see #addHook
+
'pl':  [ R, 'PATRZ', 'PRZEKIERUJ', 'TAM' ],
@private
+
'pt':  [ R, 'redir' ],
@type Object
+
'ru':  [ R, 'ПЕРЕНАПРАВЛЕНИЕ', 'ПЕРЕНАПР' ],
*/
+
'sk':  [ r, 'presmeruj' ],
this.hooks={
+
'sr':  [ r, 'Преусмери', 'преусмери', 'ПРЕУСМЕРИ', 'Preusmeri', 'preusmeri', 'PREUSMERI' ],
'create': [],
+
'tt':  [ R, 'yünältü', 'перенаправление', 'перенапр' ],
'unhide': [],
+
'uk': [ R, 'ПЕРЕНАПРАВЛЕННЯ', 'ПЕРЕНАПР' ],
'hide': []
+
'vi': [ r, 'đổi' ],
 +
'zh': [ R, '重定向'] // no comma
 +
//</NOLITE>
 
};
 
};
/** list of unique IDs of hook functions, to avoid duplicates
+
var redirList=redirLists[ pg.wiki.lang ] || [r, R];
@private
+
// Mediawiki is very tolerant about what comes after the #redirect at the start
*/
+
pg.re.redirect=RegExp('^\\s*[#](' + redirList.join('|') + ').*?\\[{2}([^\\|\\]]*)(|[^\\]]*)?\\]{2}\\s*(.*)', 'i');
this.hookIds={};
+
}
/** List of downloads associated with the popup.
+
 
@private
+
function setInterwiki() {
@type Array
+
if (pg.wiki.wikimedia) {
*/
+
// From http://meta.wikimedia.org/wiki/List_of_Wikipedias
this.downloads=[];
+
pg.wiki.interwiki='aa|ab|ace|af|ak|als|am|an|ang|ar|arc|arz|as|ast|av|ay|az|ba|bar|bat-smg|bcl|be|be-x-old|bg|bh|bi|bjn|bm|bn|bo|bpy|br|bs|bug|bxr|ca|cbk-zam|cdo|ce|ceb|ch|cho|chr|chy|ckb|co|cr|crh|cs|csb|cu|cv|cy|da|de|diq|dsb|dv|dz|ee|el|eml|en|eo|es|et|eu|ext|fa|ff|fi|fiu-vro|fj|fo|fr|frp|frr|fur|fy|ga|gag|gan|gd|gl|glk|gn|got|gu|gv|ha|hak|haw|he|hi|hif|ho|hr|hsb|ht|hu|hy|hz|ia|id|ie|ig|ii|ik|ilo|io|is|it|iu|ja|jbo|jv|ka|kaa|kab|kbd|kg|ki|kj|kk|kl|km|kn|ko|koi|kr|krc|ks|ksh|ku|kv|kw|ky|la|lad|lb|lbe|lg|li|lij|lmo|ln|lo|lt|ltg|lv|map-bms|mdf|mg|mh|mhr|mi|mk|ml|mn|mo|mr|mrj|ms|mt|mus|mwl|my|myv|mzn|na|nah|nap|nds|nds-nl|ne|new|ng|nl|nn|no|nov|nrm|nv|ny|oc|om|or|os|pa|pag|pam|pap|pcd|pdc|pfl|pi|pih|pl|pms|pnb|pnt|ps|pt|qu|rm|rmy|rn|ro|roa-rup|roa-tara|ru|rue|rw|sa|sah|sc|scn|sco|sd|se|sg|sh|si|simple|sk|sl|sm|sn|so|sq|sr|srn|ss|st|stq|su|sv|sw|szl|ta|te|tet|tg|th|ti|tk|tl|tn|to|tpi|tr|ts|tt|tum|tw|ty|udm|ug|uk|ur|uz|ve|vec|vi|vls|vo|wa|war|wo|wuu|xal|xh|yi|yo|za|zea|zh|zh-classical|zh-min-nan|zh-yue|zu';
/** Number of uncompleted downloads.
+
pg.re.interwiki=RegExp('^'+pg.wiki.interwiki+':');
@type integer
+
} else {
*/
+
pg.wiki.interwiki=null;
this.pending=null;
+
pg.re.interwiki=RegExp('^$');
/** Tolerance in pixels when detecting whether the mouse has left the popup.
+
}
@type integer
+
}
*/
+
 
this.fuzz=5;
+
// return a regexp pattern matching all variants to write the given namespace
/** Flag to toggle running {@link #limitHorizontalPosition} to regulate the popup's position.
+
function nsRe(namespaceId) {
@type boolean
+
var imageNamespaceVariants = [];
*/
+
jQuery.each(mw.config.get('wgNamespaceIds'), function(_localizedNamespaceLc, _namespaceId) {
this.constrained=true;
+
if (_namespaceId!=namespaceId) return;
/** The popup width in pixels.
+
//todo: escape regexp fragments!
@private
+
_localizedNamespaceLc = upcaseFirst(_localizedNamespaceLc);
@type integer
+
imageNamespaceVariants.push(_localizedNamespaceLc.split(' ').join('[ _]'));
*/
+
imageNamespaceVariants.push(encodeURI(_localizedNamespaceLc));
this.width=0;
+
});
/** The popup width in pixels.
 
@private
 
@type integer
 
*/
 
this.height=0;
 
/** The main content DIV element.
 
@type HTMLDivElement
 
*/
 
this.mainDiv=null;
 
this.createMainDiv();
 
  
// if (!init || typeof init.popups_draggable=='undefined' || init.popups_draggable) {
+
return '(?:' + imageNamespaceVariants.join('|') + ')';
// this.makeDraggable(true);
 
// }
 
 
}
 
}
  
/**
+
function nsReImage() {
  A UID for each Navpopup. This constructor property is just a counter.
+
return nsRe(pg.nsImageId);
  @type integer
+
}
  @private
 
*/
 
Navpopup.uid=0;
 
  
/**
+
// ENDFILE: namespaces.js
  Retrieves the {@link #visible} attribute, indicating whether the popup is currently visible.
+
// STARTFILE: selpop.js
  @type boolean
+
//<NOLITE>
*/
+
function getEditboxSelection() {
Navpopup.prototype.isVisible=function() {
+
// see http://www.webgurusforum.com/8/12/0
return this.visible;
+
try {
};
+
var editbox=document.editform.wpTextbox1;
 
+
} catch (dang) { return; }
/**
+
// IE, Opera
  Repositions popup using CSS style.
+
if (document.selection) { return document.selection.createRange().text; }
  @private
+
// Mozilla
  @param {integer} x x-coordinate (px)
+
var selStart = editbox.selectionStart;
  @param {integer} y y-coordinate (px)
+
var selEnd = editbox.selectionEnd;
  @param {boolean} noLimitHor Don't call {@link #limitHorizontalPosition}
+
return (editbox.value).substring(selStart, selEnd);
*/
+
}
Navpopup.prototype.reposition= function (x,y, noLimitHor) {
+
 
log ('reposition('+x+','+y+','+noLimitHor+')');
+
function doSelectionPopup() {
if (typeof x != 'undefined' && x !== null) { this.left=x; }
+
// popup if the selection looks like [[foo|anything afterwards at all
if (typeof y != 'undefined' && y !== null) { this.top=y; }
+
// or [[foo|bar]]text without ']]'
if (typeof this.left != 'undefined' && typeof this.top != 'undefined') {
+
// or [[foo|bar]]
this.mainDiv.style.left=this.left + 'px';
+
var sel=getEditboxSelection();
this.mainDiv.style.top=this.top + 'px';
+
var open=sel.indexOf('[[');
 +
var pipe=sel.indexOf('|');
 +
var close=sel.indexOf(']]');
 +
if (open == -1 || ( pipe == -1 && close == -1) ) { return; }
 +
if (pipe != -1 && open > pipe || close != -1 && open > close) { return; }
 +
if (getValueOf('popupOnEditSelection')=='boxpreview') {
 +
return doSeparateSelectionPopup(sel);
 +
}
 +
var article=new Title(sel.substring(open+2, (pipe < 0) ? close : pipe)).urlString();
 +
if (close > 0 && sel.substring(close+2).indexOf('[[') >= 0) {  
 +
return;
 +
}
 +
var a=document.createElement('a');
 +
a.href=pg.wiki.titlebase + article;
 +
mouseOverWikiLink2(a);
 +
if (a.navpopup) {
 +
a.navpopup.addHook(function(){runStopPopupTimer(a.navpopup);}, 'unhide', 'after');
 
}
 
}
if (!noLimitHor) { this.limitHorizontalPosition(); }
+
}
//console.log('navpop'+this.uid+' - (left,top)=(' + this.left + ',' + this.top + '), css=('
 
//+ this.mainDiv.style.left + ',' + this.mainDiv.style.top + ')');
 
};
 
  
/**
+
function doSeparateSelectionPopup(str) {
  Prevents popups from being in silly locations. Hopefully.
+
var div=document.getElementById('selectionPreview');
  Should not be run if {@link #constrained} is true.
+
if (!div) {
  @private
+
div = document.createElement('div');
*/
+
div.id='selectionPreview';
Navpopup.prototype.limitHorizontalPosition=function() {
+
try { var box=document.editform.wpTextbox1; }
if (!this.constrained || this.tooWide) { return; }
+
catch (oopsie) { return; }
this.updateDimensions();
+
box.parentNode.insertBefore(div, box);
var x=this.left;
+
}
var w=this.width;
+
div.innerHTML=wiki2html(str);
var cWidth=document.body.clientWidth;
+
div.ranSetupTooltipsAlready = false;
 +
popTipsSoonFn('selectionPreview')();
 +
}
 +
//</NOLITE>
 +
// ENDFILE: selpop.js
 +
// STARTFILE: navpopup.js
 +
/**
 +
  @fileoverview  Defines two classes: {@link Navpopup} and {@link Mousetracker}.
  
 +
  <code>Navpopup</code> describes popups: when they appear, where, what
 +
  they look like and so on.
  
// log('limitHorizontalPosition: x='+x+
+
  <code>Mousetracker</code> "captures" the mouse using
// ', this.left=' + this.left +
+
  <code>document.onmousemove</code>.
// ', this.width=' + this.width +
+
*/
// ', cWidth=' + cWidth);
 
  
 
if ( (x+w) >= cWidth ||
 
( x > 0 &&
 
this.maxWidth &&
 
this.width < this.maxWidth &&
 
this.height > this.width &&
 
x > cWidth - this.maxWidth ) ) {
 
// This is a very nasty hack. There has to be a better way!
 
// We find the "natural" width of the div by positioning it at the far left
 
// then reset it so that it should be flush right (well, nearly)
 
this.mainDiv.style.left='-10000px';
 
this.mainDiv.style.width = this.maxWidth + 'px';
 
var naturalWidth=parseInt(this.mainDiv.offsetWidth, 10);
 
var newLeft=cWidth - naturalWidth - 1;
 
if (newLeft < 0) { newLeft = 0; this.tooWide=true; } // still unstable for really wide popups?
 
log ('limitHorizontalPosition: moving to ('+newLeft + ','+ this.top+');' + ' naturalWidth=' + naturalWidth + ', clientWidth=' + cWidth);
 
this.reposition(newLeft, null, true);
 
}
 
};
 
  
 
/**
 
/**
   Counter indicating the z-order of the "highest" popup.
+
   Creates a new Mousetracker.
  We start the z-index at 1000 so that popups are above everything
+
   @constructor
  else on the screen.
+
   @class The Mousetracker class. This monitors mouse movements and manages associated hooks.
   @private
 
   @type integer
 
 
*/
 
*/
Navpopup.highest=1000;
+
function Mousetracker() {
 +
/**
 +
  Interval to regularly run the hooks anyway, in milliseconds.
 +
  @type Integer
 +
*/
 +
this.loopDelay=400;
 +
 
 +
/**
 +
  Timer for the loop.
 +
  @type Timer
 +
*/
 +
this.timer=null;
 +
 
 +
/**
 +
  Flag - are we switched on?
 +
  @type Boolean
 +
*/
 +
this.active=false;
 +
/**
 +
  Flag - are we probably inaccurate, i.e. not reflecting the actual mouse position?
 +
*/
 +
this.dirty=true;
 +
/**
 +
  Array of hook functions.
 +
  @private
 +
  @type Array
 +
*/
 +
this.hooks=[];
 +
}
  
 
/**
 
/**
   Brings popup to the top of the z-order.
+
   Adds a hook, to be called when we get events.
   We increment the {@link #highest} property of the contructor here.
+
   @param {Function} f A function which is called as
   @private
+
  <code>f(x,y)</code>. It should return <code>true</code> when it
 +
   wants to be removed, and <code>false</code> otherwise.
 
*/
 
*/
Navpopup.prototype.raise = function () {
+
Mousetracker.prototype.addHook = function (f) {
this.mainDiv.style.zIndex=Navpopup.highest + 1;
+
this.hooks.push(f);
++Navpopup.highest;
 
 
};
 
};
  
 
/**
 
/**
   Shows the popup provided {@link #noshow} is not true.
+
   Runs hooks, passing them the x
   Updates the position, brings the popup to the top of the z-order and unhides it.
+
  and y coords of the mouse.  Hook functions that return true are
 +
  passed to {@link Mousetracker#removeHooks} for removal.
 +
   @private
 
*/
 
*/
Navpopup.prototype.show = function () {
+
Mousetracker.prototype.runHooks = function () {
//document.title+='s';
+
if (!this.hooks || !this.hooks.length) { return; }
if (this.noshow) { return; }
+
//log('Mousetracker.runHooks; we got some hooks to run');
//document.title+='t';
+
var remove=false;
this.reposition();
+
var removeObj={};
this.raise();
+
// this method gets called a LOT -
this.unhide();
+
// pre-cache some variables
 +
var x=this.x, y=this.y, len = this.hooks.length;
 +
 
 +
for (var i=0; i<len; ++i) {
 +
//~ run the hook function, and remove it if it returns true
 +
if (this.hooks[i](x, y)===true) {
 +
remove=true;
 +
removeObj[i]=true;
 +
}
 +
}
 +
if (remove) { this.removeHooks(removeObj); }
 
};
 
};
  
 
/**
 
/**
   Checks to see if the mouse pointer has
+
   Removes hooks.
   stabilised (checking every <code>time</code>/2 milliseconds) and runs the
+
   @private
  {@link #show} method if it has.
+
   @param {Object} removeObj An object whose keys are the index
   @param {integer} time The minimum time (ms) before the popup may be shown.
+
  numbers of functions for removal, with values that evaluate to true
 
*/
 
*/
Navpopup.prototype.showSoonIfStable = function (time) {
+
Mousetracker.prototype.removeHooks = function(removeObj) {
log ('showSoonIfStable, time='+time);
+
var newHooks=[];
if (this.visible) { return; }
+
var len = this.hooks.length;
this.noshow = false;
+
for (var i=0; i<len; ++i) {
 +
if (! removeObj[i]) { newHooks.push(this.hooks[i]); }
 +
}
 +
this.hooks=newHooks;
 +
};
  
//~ initialize these variables so that we never run @tt{show} after
 
//~ just half the time
 
this.stable_x = -10000; this.stable_y = -10000;
 
  
var stableShow = function() {
+
/**
log('stableShow called');
+
  Event handler for mouse wiggles.
var new_x = Navpopup.tracker.x, new_y = Navpopup.tracker.y;
+
  We simply grab the event, set x and y and run the hooks.
var dx = savedThis.stable_x - new_x, dy = savedThis.stable_y - new_y;
+
  This makes the cpu all hot and bothered :-(
var fuzz2 = 0; // savedThis.fuzz * savedThis.fuzz;
+
  @private
//document.title += '[' + [savedThis.stable_x,new_x, savedThis.stable_y,new_y, dx, dy, fuzz2].join(',') + '] ';
+
  @param {Event} e Mousemove event
if ( dx * dx <= fuzz2 && dy * dy <= fuzz2 ) {
+
*/
log ('mouse is stable');
+
Mousetracker.prototype.track=function (e) {
clearInterval(savedThis.showSoonStableTimer);
+
//~ Apparently this is needed in IE.
savedThis.reposition.apply(savedThis, [new_x + 2, new_y + 2]);
+
e = e || window.event;
savedThis.show.apply(savedThis, []);
+
var x, y;
savedThis.limitHorizontalPosition.apply(savedThis, []);
+
if (e) {
return;
+
if (e.pageX) { x=e.pageX; y=e.pageY; }
}
+
else if (typeof e.clientX!='undefined') {
savedThis.stable_x = new_x; savedThis.stable_y = new_y;
+
var left, top, docElt = window.document.documentElement;
};
+
 
var savedThis = this;
+
if (docElt) { left=docElt.scrollLeft; }
this.showSoonStableTimer = setInterval(stableShow, time/2);
+
left = left || window.document.body.scrollLeft || window.document.scrollLeft || 0;
};
+
 
 +
if (docElt) { top=docElt.scrollTop; }
 +
top = top || window.document.body.scrollTop || window.document.scrollTop || 0;
  
/**
+
x=e.clientX + left;
  Sets the {@link #noshow} flag and hides the popup. This should be called
+
y=e.clientY + top;
  when the mouse leaves the link before
+
} else { return; }
  (or after) it's actually been displayed.
+
this.setPosition(x,y);
*/
 
Navpopup.prototype.banish = function () {
 
log ('banish called');
 
// hide and prevent showing with showSoon in the future
 
this.noshow=true;
 
if (this.showSoonStableTimer) {
 
log('clearing showSoonStableTimer');
 
clearInterval(this.showSoonStableTimer);
 
 
}
 
}
this.hide();
 
 
};
 
};
  
 
/**
 
/**
   Runs hooks added with {@link #addHook}.
+
   Sets the x and y coordinates stored and takes appropriate action,
  @private
+
  running hooks as appropriate.
   @param {String} key Key name of the {@link #hooks} array - one of 'create', 'unhide', 'hide'
+
   @param {Integer} x, y Screen coordinates to set
  @param {String} when Controls exactly when the hook is run: either 'before' or 'after'
 
 
*/
 
*/
Navpopup.prototype.runHooks = function (key, when) {
+
 
if (!this.hooks[key]) { return; }
+
Mousetracker.prototype.setPosition=function(x,y) {
var keyHooks=this.hooks[key];
+
this.x = x;
var len=keyHooks.length;
+
this.y = y;
for (var i=0; i< len; ++i) {
+
if (this.dirty || this.hooks.length === 0) { this.dirty=false; return; }
if (keyHooks[i] && keyHooks[i].when == when) {
+
if (typeof this.lastHook_x != 'number') { this.lastHook_x = -100; this.lastHook_y=-100; }
if (keyHooks[i].hook.apply(this, [])) {
+
var diff = (this.lastHook_x - x)*(this.lastHook_y - y);
// remove the hook
+
diff = (diff >= 0) ? diff : -diff;
if (keyHooks[i].hookId) {
+
if ( diff > 1 ) {
delete this.hookIds[keyHooks[i].hookId];
+
this.lastHook_x=x;
}
+
this.lastHook_y=y;
keyHooks[i]=null;
+
if (this.dirty) { this.dirty = false; }
}
+
else { this.runHooks(); }
}
 
 
}
 
}
 +
}
 +
 +
/**
 +
  Sets things in motion, unless they are already that is, registering an event handler on <code>document.onmousemove</code>.
 +
  A half-hearted attempt is made to preserve the old event handler if there is one.
 +
*/
 +
Mousetracker.prototype.enable = function () {
 +
if (this.active) { return; }
 +
this.active=true;
 +
//~ Save the current handler for mousemove events. This isn't too
 +
//~ robust, of course.
 +
this.savedHandler=document.onmousemove;
 +
//~ Gotta save @tt{this} again for the closure, and use apply for
 +
//~ the member function.
 +
var savedThis=this;
 +
document.onmousemove=function (e) {savedThis.track.apply(savedThis, [e]);};
 +
if (this.loopDelay) { this.timer = setInterval(function() { //log('loop delay in mousetracker is working');
 +
savedThis.runHooks();}, this.loopDelay); }
 
};
 
};
  
 
/**
 
/**
   Adds a hook to the popup. Hook functions are run with <code>this</code> set to refer to the Navpopup instance, and no arguments.
+
   Disables the tracker, removing the event handler.
  @param {Function} hook The hook function. Functions that return true are deleted.
 
  @param {String} key Key name of the {@link #hooks} array - one of 'create', 'unhide', 'hide'
 
  @param {String} when Controls exactly when the hook is run: either 'before' or 'after'
 
  @param {String} uid A truthy string identifying the hook function; if it matches another hook in this position, it won't be added again.
 
 
*/
 
*/
Navpopup.prototype.addHook = function ( hook, key, when, uid ) {
+
Mousetracker.prototype.disable = function () {
when = when || 'after';
+
if (!this.active) { return; }
if (!this.hooks[key]) { return; }
+
if ($.isFunction(this.savedHandler)) {
// if uid is specified, don't add duplicates
+
document.onmousemove=this.savedHandler;
var hookId=null;
+
} else { delete document.onmousemove; }
if (uid) {
+
if (this.timer) { clearInterval(this.timer); }
hookId=[key,when,uid].join('|');
+
this.active=false;
if (this.hookIds[hookId]) {
 
return;
 
}
 
this.hookIds[hookId]=true;
 
}
 
this.hooks[key].push( {hook: hook, when: when, hookId: hookId} );
 
 
};
 
};
  
 
/**
 
/**
   Creates the main DIV element, which contains all the actual popup content.
+
   Creates a new Navpopup.
   Runs hooks with key 'create'.
+
  Gets a UID for the popup and
   @private
+
  @param init Contructor object. If <code>init.draggable</code> is true or absent, the popup becomes draggable.
 +
   @constructor
 +
   @class The Navpopup class. This generates popup hints, and does some management of them.
 
*/
 
*/
Navpopup.prototype.createMainDiv = function () {
+
function Navpopup(init) {
if (this.mainDiv) { return; }
+
//alert('new Navpopup(init)');
this.runHooks('create', 'before');
+
/** UID for each Navpopup instance.
var mainDiv=document.createElement('div');
+
Read-only.
 
+
@type integer
var savedThis=this;
+
*/
mainDiv.onclick=function(e) {savedThis.onclickHandler(e);};
+
this.uid=Navpopup.uid++;
mainDiv.className=(this.className) ? this.className : 'navpopup_maindiv';
+
/**
mainDiv.id=mainDiv.className + this.uid;
+
  Read-only flag for current visibility of the popup.
 
+
  @type boolean
mainDiv.style.position='absolute';
+
  @private
mainDiv.style.minWidth = '350px';
+
*/
mainDiv.style.display='none';
+
this.visible=false;
mainDiv.className='navpopup';
+
/** Flag to be set when we want to cancel a previous request to
 
+
show the popup in a little while.
// easy access to javascript object through DOM functions
+
@private
mainDiv.navpopup=this;
+
@type boolean
 
+
*/
this.mainDiv=mainDiv;
+
this.noshow=false;
document.body.appendChild(mainDiv);
+
/** Categorised list of hooks.
this.runHooks('create', 'after');
+
@see #runHooks
};
+
@see #addHook
/**
+
@private
  Calls the {@link #raise} method.
+
@type Object
  @private
+
*/
*/
+
this.hooks={
Navpopup.prototype.onclickHandler=function(/*e*/) {
+
'create': [],
this.raise();
+
'unhide': [],
};
+
'hide': []
/**
+
};
  Makes the popup draggable, using a {@link Drag} object.
+
/** list of unique IDs of hook functions, to avoid duplicates
  @private
+
@private
*/
+
*/
Navpopup.prototype.makeDraggable=function(handleName) {
+
this.hookIds={};
if (!this.mainDiv) { this.createMainDiv(); }
+
/** List of downloads associated with the popup.
var drag=new Drag();
+
@private
if (!handleName) {
+
@type Array
drag.startCondition=function(e) {
+
*/
try { if (!e.shiftKey) { return false; } } catch (err) { return false; }
+
this.downloads=[];
return true;
+
/** Number of uncompleted downloads.
};
+
@type integer
}
+
*/
var dragHandle;
+
this.pending=null;
if (handleName) dragHandle = document.getElementById(handleName);
+
/** Tolerance in pixels when detecting whether the mouse has left the popup.
if (!dragHandle) dragHandle = this.mainDiv;
+
@type integer
var np=this;
+
*/
drag.endHook=function(x,y) {
+
this.fuzz=5;
Navpopup.tracker.dirty=true;
+
/** Flag to toggle running {@link #limitHorizontalPosition} to regulate the popup's position.
np.reposition(x,y);
+
@type boolean
};
+
*/
drag.init(dragHandle,this.mainDiv);
+
this.constrained=true;
};
+
/** The popup width in pixels.
 +
@private
 +
@type integer
 +
*/
 +
this.width=0;
 +
/** The popup width in pixels.
 +
@private
 +
@type integer
 +
*/
 +
this.height=0;
 +
/** The main content DIV element.
 +
@type HTMLDivElement
 +
*/
 +
this.mainDiv=null;
 +
this.createMainDiv();
  
/** Hides the popup using CSS. Runs hooks with key 'hide'.
+
// if (!init || typeof init.popups_draggable=='undefined' || init.popups_draggable) {
Sets {@link #visible} appropriately. {@link #banish} should be called externally instead of this method.
+
// this.makeDraggable(true);
 +
// }
 +
}
  
@private
+
/**
*/
+
  A UID for each Navpopup. This constructor property is just a counter.
Navpopup.prototype.hide = function () {
+
  @type integer
this.runHooks('hide', 'before');
+
  @private
this.abortDownloads();
 
if (typeof this.visible != 'undefined' && this.visible) {
 
this.mainDiv.style.display='none';
 
this.visible=false;
 
}
 
this.runHooks('hide', 'after');
 
};
 
 
 
/** Shows the popup using CSS. Runs hooks with key 'unhide'.
 
Sets {@link #visible} appropriately.  {@link #show} should be called externally instead of this method.
 
@private
 
 
*/
 
*/
Navpopup.prototype.unhide = function () {
+
Navpopup.uid=0;
this.runHooks('unhide', 'before');
 
if (typeof this.visible != 'undefined' && !this.visible) {
 
this.mainDiv.style.display='inline';
 
this.visible=true;
 
}
 
this.runHooks('unhide', 'after');
 
};
 
  
 
/**
 
/**
   Sets the <code>innerHTML</code> attribute of the main div containing the popup content.
+
   Retrieves the {@link #visible} attribute, indicating whether the popup is currently visible.
   @param {String} html The HTML to set.
+
   @type boolean
 
*/
 
*/
Navpopup.prototype.setInnerHTML = function (html) {
+
Navpopup.prototype.isVisible=function() {
this.mainDiv.innerHTML = html;
+
return this.visible;
 
};
 
};
  
 
/**
 
/**
   Updates the {@link #width} and {@link #height} attributes with the CSS properties.
+
   Repositions popup using CSS style.
 
   @private
 
   @private
 +
  @param {integer} x x-coordinate (px)
 +
  @param {integer} y y-coordinate (px)
 +
  @param {boolean} noLimitHor Don't call {@link #limitHorizontalPosition}
 
*/
 
*/
Navpopup.prototype.updateDimensions = function () {
+
Navpopup.prototype.reposition= function (x,y, noLimitHor) {
this.width=parseInt(this.mainDiv.offsetWidth, 10);
+
log ('reposition('+x+','+y+','+noLimitHor+')');
this.height=parseInt(this.mainDiv.offsetHeight, 10);
+
if (typeof x != 'undefined' && x!==null) { this.left=x; }
 +
if (typeof y != 'undefined' && y!==null) { this.top=y; }
 +
if (typeof this.left != 'undefined' && typeof this.top != 'undefined') {
 +
this.mainDiv.style.left=this.left + 'px';
 +
this.mainDiv.style.top=this.top + 'px';
 +
}
 +
if (!noLimitHor) { this.limitHorizontalPosition(); }
 +
//console.log('navpop'+this.uid+' - (left,top)=(' + this.left + ',' + this.top + '), css=('
 +
//+ this.mainDiv.style.left + ',' + this.mainDiv.style.top + ')');
 
};
 
};
  
 
/**
 
/**
   Checks if the point (x,y) is within {@link #fuzz} of the
+
   Prevents popups from being in silly locations. Hopefully.
   {@link #mainDiv}.
+
   Should not be run if {@link #constrained} is true.
   @param {integer} x x-coordinate (px)
+
   @private
  @param {integer} y y-coordinate (px)
 
  @type boolean
 
 
*/
 
*/
Navpopup.prototype.isWithin = function(x,y) {
+
Navpopup.prototype.limitHorizontalPosition=function() {
//~ If we're not even visible, no point should be considered as
+
if (!this.constrained || this.tooWide) { return; }
//~ being within the popup.
 
if (!this.visible) { return false; }
 
 
this.updateDimensions();
 
this.updateDimensions();
var fuzz=this.fuzz || 0;
+
var x=this.left;
//~ Use a simple box metric here.
+
var w=this.width;
return (x+fuzz >= this.left && x-fuzz <= this.left + this.width &&
+
var cWidth=document.body.clientWidth;
y+fuzz >= this.top  && y-fuzz <= this.top + this.height);
+
 
 +
 
 +
// log('limitHorizontalPosition: x='+x+
 +
// ', this.left=' + this.left +
 +
// ', this.width=' + this.width +
 +
// ', cWidth=' + cWidth);
 +
 
 +
 
 +
if ( (x+w) >= cWidth ||
 +
( x > 0 && this.maxWidth && this.width < this.maxWidth && this.height > this.width
 +
  && x > cWidth - this.maxWidth ) ) {
 +
// This is a very nasty hack. There has to be a better way!
 +
// We find the "natural" width of the div by positioning it at the far left
 +
// then reset it so that it should be flush right (well, nearly)
 +
this.mainDiv.style.left='-10000px';
 +
this.mainDiv.style.width = this.maxWidth + 'px';
 +
var naturalWidth=parseInt(this.mainDiv.offsetWidth, 10);
 +
var newLeft=cWidth - naturalWidth - 1;
 +
if (newLeft < 0) { newLeft = 0; this.tooWide=true; } // still unstable for really wide popups?
 +
log ('limitHorizontalPosition: moving to ('+newLeft + ','+ this.top+');' + ' naturalWidth=' + naturalWidth + ', clientWidth=' + cWidth);
 +
this.reposition(newLeft, null, true);
 +
}
 
};
 
};
  
 
/**
 
/**
   Adds a download to {@link #downloads}.
+
   Counter indicating the z-order of the "highest" popup.
   @param {Downloader} download
+
  We start the z-index at 1000 so that popups are above everything
 +
  else on the screen.
 +
  @private
 +
   @type integer
 
*/
 
*/
Navpopup.prototype.addDownload=function(download) {
+
Navpopup.highest=1000;
if (!download) { return; }
+
 
this.downloads.push(download);
+
/**
 +
  Brings popup to the top of the z-order.
 +
  We increment the {@link #highest} property of the contructor here.
 +
  @private
 +
*/
 +
Navpopup.prototype.raise = function () {
 +
this.mainDiv.style.zIndex=Navpopup.highest + 1;
 +
++Navpopup.highest;
 
};
 
};
 +
 
/**
 
/**
   Aborts the downloads listed in {@link #downloads}.
+
   Shows the popup provided {@link #noshow} is not true.
   @see Downloader#abort
+
   Updates the position, brings the popup to the top of the z-order and unhides it.
 
*/
 
*/
Navpopup.prototype.abortDownloads=function() {
+
Navpopup.prototype.show = function () {
for(var i=0; i<this.downloads.length; ++i) {
+
//document.title+='s';
var d=this.downloads[i];
+
if (this.noshow) { return; }
if (d && d.abort) { d.abort(); }
+
//document.title+='t';
}
+
this.reposition();
this.downloads=[];
+
this.raise();
 +
this.unhide();
 
};
 
};
  
  
 
/**
 
/**
   A {@link Mousetracker} instance which is a property of the constructor (pseudo-global).
+
   Runs the {@link #show} method in a little while, unless we're
 +
  already visible.
 +
  @param {integer} time Delay in milliseconds
 +
  @see #showSoonIfStable
 
*/
 
*/
Navpopup.tracker=new Mousetracker();
+
Navpopup.prototype.showSoon = function (time) {
// ENDFILE: navpopup.js
+
if (this.visible) { return; }
// STARTFILE: diff.js
+
this.noshow=false;
//<NOLITE>
+
//~ We have to save the value of @tt{this} so that the closure below
/*
+
//~ works.
* Javascript Diff Algorithm
+
var savedThis=this;
*  By John Resig (http://ejohn.org/) and [[:en:User:Lupin]]
+
//this.start_x = Navpopup.tracker.x;
*
+
//this.start_y = Navpopup.tracker.y;
* More Info:
+
setTimeout(function () {
*  http://ejohn.org/projects/javascript-diff-algorithm/
+
if (Navpopup.tracker.active) {
*/
+
savedThis.reposition.apply(savedThis, [Navpopup.tracker.x + 2, Navpopup.tracker.y + 2]);
 +
}
 +
//~ Have to use apply to invoke his member function here
 +
savedThis.show.apply(savedThis, []);
 +
}, time);
 +
};
  
function delFmt(x) {
+
/**
if (!x.length) { return ''; }
+
  Checks to see if the mouse pointer has
return "<del class='popupDiff'>" + x.join('') +"</del>";
+
  stabilised (checking every <code>time</code>/2 milliseconds) and runs the
}
+
  {@link #show} method if it has. This method makes {@link #showSoon} redundant.
function insFmt(x) {
+
  @param {integer} time The minimum time (ms) before the popup may be shown.
if (!x.length) { return ''; }
+
*/
return "<ins class='popupDiff'>" + x.join('') +"</ins>";
+
Navpopup.prototype.showSoonIfStable = function (time) {
}
+
log ('showSoonIfStable, time='+time);
 +
if (this.visible) { return; }
 +
this.noshow = false;
 +
 
 +
//~ initialize these variables so that we never run @tt{show} after
 +
//~ just half the time
 +
this.stable_x = -10000; this.stable_y = -10000;
  
function countCrossings(a, b, i, eject) {
+
var stableShow = function() {
// count the crossings on the edge starting at b[i]
+
log('stableShow called');
if (!b[i].row && b[i].row !== 0) { return -1; }
+
var new_x = Navpopup.tracker.x, new_y = Navpopup.tracker.y;
var count=0;
+
var dx = savedThis.stable_x - new_x, dy = savedThis.stable_y - new_y;
for (var j=0; j<a.length; ++j) {
+
var fuzz2 = 0; // savedThis.fuzz * savedThis.fuzz;
if (!a[j].row && a[j].row !== 0) { continue; }
+
//document.title += '[' + [savedThis.stable_x,new_x, savedThis.stable_y,new_y, dx, dy, fuzz2].join(',') + '] ';
if ( (j-b[i].row)*(i-a[j].row) > 0) {
+
if ( dx * dx <= fuzz2 && dy * dy <= fuzz2 ) {
if(eject) { return true; }
+
log ('mouse is stable');
count++;
+
clearInterval(savedThis.showSoonStableTimer);
 +
savedThis.reposition.apply(savedThis, [new_x + 2, new_y + 2]);
 +
savedThis.show.apply(savedThis, []);
 +
return;
 
}
 
}
}
+
savedThis.stable_x = new_x; savedThis.stable_y = new_y;
return count;
+
};
}
+
var savedThis = this;
 +
this.showSoonStableTimer = setInterval(stableShow, time/2);
 +
};
  
function shortenDiffString(str, context) {
+
/**
var re=RegExp('(<del[\\s\\S]*?</del>|<ins[\\s\\S]*?</ins>)');
+
  Makes the popup unhidable until we call {@link #unstick}.
var splitted=str.parenSplit(re);
+
*/
var ret=[''];
+
Navpopup.prototype.stick=function() {
for (var i=0; i<splitted.length; i+=2) {
+
this.noshow=false;
if (splitted[i].length < 2*context) {
+
this.sticky=true;
ret[ret.length-1] += splitted[i];
+
};
if (i+1<splitted.length) { ret[ret.length-1] += splitted[i+1]; }
 
continue;
 
}
 
else {
 
if (i > 0) { ret[ret.length-1] += splitted[i].substring(0,context); }
 
if (i+1 < splitted.length) {
 
ret.push(splitted[i].substring(splitted[i].length-context) +
 
splitted[i+1]);
 
}
 
}
 
}
 
while (ret.length > 0 && !ret[0]) { ret = ret.slice(1); }
 
return ret;
 
}
 
  
 +
/**
 +
  Allows the popup to be hidden.
 +
*/
 +
Navpopup.prototype.unstick=function() {
 +
this.sticky=false;
 +
};
  
function diffString( o, n, simpleSplit ) {
+
/**
var splitRe=RegExp('([[]{2}|[\\]]{2}|[{]{2,3}|[}]{2,3}|[|]|=|<|>|[*:]+|\\s|\\b)');
+
  Sets the {@link #noshow} flag and hides the popup. This should be called
 
+
  when the mouse leaves the link before
// We need to split the strings o and n first, and entify() the parts
+
  (or after) it's actually been displayed.
//  individually, so that the HTML entities are never cut apart. (AxelBoldt)
+
*/
var out, i, oSplitted, nSplitted;
+
Navpopup.prototype.banish = function () {
if (simpleSplit) {  
+
log ('banish called');
oSplitted=o.split(/\b/);  
+
// hide and prevent showing with showSoon in the future
nSplitted=n.split(/\b/);
+
this.noshow=true;
} else {
+
if (this.showSoonStableTimer) {
oSplitted=o.parenSplit(splitRe);
+
log('clearing showSoonStableTimer');
nSplitted=n.parenSplit(splitRe);  
+
clearInterval(this.showSoonStableTimer);
 
}
 
}
for (i=0; i<oSplitted.length; ++i) {oSplitted[i]=oSplitted[i].entify();}
+
this.hide();
for (i=0; i<nSplitted.length; ++i) {nSplitted[i]=nSplitted[i].entify();}
+
};
 
out = diff (oSplitted, nSplitted);
 
var str = "";
 
var acc=[]; // accumulator for prettier output
 
  
// crossing pairings -- eg 'A B' vs 'B A' -- cause problems, so let's iron them out
+
/**
// this doesn't always do things optimally but it should be fast enough
+
  Runs hooks added with {@link #addHook}.
var maxOutputPair=0;
+
  @private
for (i=0; i<out.n.length; ++i) {
+
  @param {String} key Key name of the {@link #hooks} array - one of 'create', 'unhide', 'hide'
if ( out.n[i].paired ) {
+
  @param {String} when Controls exactly when the hook is run: either 'before' or 'after'
if( maxOutputPair > out.n[i].row ) {
+
*/
// tangle - delete pairing
+
Navpopup.prototype.runHooks = function (key, when) {
out.o[ out.n[i].row ]=out.o[ out.n[i].row ].text;
+
if (!this.hooks[key]) { return; }
out.n[i]=out.n[i].text;
+
var keyHooks=this.hooks[key];
 +
var len=keyHooks.length;
 +
for (var i=0; i< len; ++i) {
 +
if (keyHooks[i] && keyHooks[i].when == when) {
 +
if (keyHooks[i].hook.apply(this, [])) {
 +
// remove the hook
 +
if (keyHooks[i].hookId) {
 +
delete this.hookIds[keyHooks[i].hookId];
 +
}
 +
keyHooks[i]=null;
 +
}
 
}
 
}
if (maxOutputPair < out.n[i].row) { maxOutputPair = out.n[i].row; }
+
}
 +
};
 +
 
 +
/**
 +
  Adds a hook to the popup. Hook functions are run with <code>this</code> set to refer to the Navpopup instance, and no arguments.
 +
  @param {Function} hook The hook function. Functions that return true are deleted.
 +
  @param {String} key Key name of the {@link #hooks} array - one of 'create', 'unhide', 'hide'
 +
  @param {String} when Controls exactly when the hook is run: either 'before' or 'after'
 +
  @param {String} uid A truthy string identifying the hook function; if it matches another hook in this position, it won't be added again.
 +
*/
 +
Navpopup.prototype.addHook = function ( hook, key, when, uid ) {
 +
when = when || 'after';
 +
if (!this.hooks[key]) { return; }
 +
// if uid is specified, don't add duplicates
 +
var hookId=null;
 +
if (uid) {
 +
hookId=[key,when,uid].join('|');
 +
if (this.hookIds[hookId]) {
 +
return;
 
}
 
}
 +
this.hookIds[hookId]=true;
 
}
 
}
 +
this.hooks[key].push( {hook: hook, when: when, hookId: hookId} );
 +
};
  
// output the stuff preceding the first paired old line
+
/**
for (i=0; i<out.o.length && !out.o[i].paired; ++i) { acc.push( out.o[i] ); }
+
  Creates the main DIV element, which contains all the actual popup content.
str += delFmt(acc); acc=[];
+
  Runs hooks with key 'create'.
 +
  @private
 +
*/
 +
Navpopup.prototype.createMainDiv = function () {
 +
if (this.mainDiv) { return; }
 +
this.runHooks('create', 'before');
 +
var mainDiv=document.createElement('div');
 +
 
 +
var savedThis=this;
 +
mainDiv.onclick=function(e) {savedThis.onclickHandler(e);};
 +
mainDiv.className=(this.className) ? this.className : 'navpopup_maindiv';
 +
mainDiv.id=mainDiv.className + this.uid;
  
// main loop
+
mainDiv.style.position='absolute';
for ( i = 0; i < out.n.length; ++i ) {
+
mainDiv.style.display='none';
// output unpaired new "lines"
+
mainDiv.className='navpopup';
while ( i < out.n.length && !out.n[i].paired ) { acc.push( out.n[i++] ); }
 
str += insFmt(acc); acc=[];
 
if ( i < out.n.length ) { // this new "line" is paired with the (out.n[i].row)th old "line"
 
str += out.n[i].text;
 
// output unpaired old rows starting after this new line's partner
 
var m = out.n[i].row + 1;
 
while ( m < out.o.length && !out.o[m].paired ) { acc.push ( out.o[m++] ); }
 
str += delFmt(acc); acc=[];
 
}
 
}
 
return str;
 
}
 
  
// see http://developer.mozilla.org/en/docs/Core_JavaScript_1.5_Reference:Global_Objects:Object
+
// easy access to javascript object through DOM functions
// FIXME: use obj.hasOwnProperty instead of this kludge!
+
mainDiv.navpopup=this;
var jsReservedProperties=RegExp('^(constructor|prototype|__((define|lookup)[GS]etter)__' +
 
  '|eval|hasOwnProperty|propertyIsEnumerable' +
 
  '|to(Source|String|LocaleString)|(un)?watch|valueOf)$');
 
function diffBugAlert(word) {
 
if (!diffBugAlert.list[word]) {
 
diffBugAlert.list[word]=1;
 
alert('Bad word: '+word+'\n\nPlease report this bug.');
 
}
 
}
 
diffBugAlert.list={};
 
  
function makeDiffHashtable(src) {
+
this.mainDiv=mainDiv;
var ret={};
+
document.body.appendChild(mainDiv);
for ( var i = 0; i < src.length; i++ ) {
+
this.runHooks('create', 'after');
if ( jsReservedProperties.test(src[i]) ) { src[i] += '<!-- -->'; }
+
};
if ( !ret[ src[i] ] ) { ret[ src[i] ] = []; }
+
/**
try { ret[ src[i] ].push( i ); } catch (err) { diffBugAlert(src[i]); }
+
  Calls the {@link #raise} method.
 +
  @private
 +
*/
 +
Navpopup.prototype.onclickHandler=function(e) {
 +
this.raise();
 +
};
 +
/**
 +
  Makes the popup draggable, using a {@link Drag} object.
 +
  @private
 +
*/
 +
Navpopup.prototype.makeDraggable=function(handleName) {
 +
if (!this.mainDiv) { this.createMainDiv(); }
 +
var drag=new Drag();
 +
if (!handleName) {
 +
drag.startCondition=function(e) {
 +
try { if (!e.shiftKey) { return false; } } catch (err) { return false; }
 +
return true;
 +
};
 
}
 
}
return ret;
+
var dragHandle;
}
+
if (handleName) dragHandle = document.getElementById(handleName);
 +
if (!dragHandle) dragHandle = this.mainDiv;
 +
var np=this;
 +
drag.endHook=function(x,y) {
 +
Navpopup.tracker.dirty=true;
 +
np.reposition(x,y);
 +
};
 +
drag.init(dragHandle,this.mainDiv);
 +
};
  
function diff( o, n ) {
+
/** Hides the popup using CSS. Runs hooks with key 'hide'.
 +
Sets {@link #visible} appropriately. {@link #banish} should be called externally instead of this method.
  
// pass 1: make hashtable ns with new rows as keys
+
@private
var ns = makeDiffHashtable(n);
+
*/
 
+
Navpopup.prototype.hide = function () {
// pass 2: make hashtable os with old rows as keys
+
this.runHooks('hide', 'before');
var os = makeDiffHashtable(o);
+
this.abortDownloads();
 
+
if (this.sticky) { return; }
// pass 3: pair unique new rows and matching unique old rows
+
if (typeof this.visible != 'undefined' && this.visible) {
var i;
+
this.mainDiv.style.display='none';
for ( i in ns ) {
+
this.visible=false;
if ( ns[i].length == 1 && os[i] && os[i].length == 1 ) {
 
n[ ns[i][0] ] = { text: n[ ns[i][0] ], row: os[i][0], paired: true };
 
o[ os[i][0] ] = { text: o[ os[i][0] ], row: ns[i][0], paired: true };
 
}
 
 
}
 
}
 +
this.runHooks('hide', 'after');
 +
};
  
// pass 4: pair matching rows immediately following paired rows (not necessarily unique)
+
/** Shows the popup using CSS. Runs hooks with key 'unhide'.
for ( i = 0; i < n.length - 1; i++ ) {
+
Sets {@link #visible} appropriately.  {@link #show} should be called externally instead of this method.
if ( n[i].paired && ! n[i+1].paired && n[i].row + 1 < o.length && ! o[ n[i].row + 1 ].paired &&
+
@private
n[i+1] == o[ n[i].row + 1 ] ) {
+
*/
n[i+1] = { text: n[i+1], row: n[i].row + 1, paired: true };
+
Navpopup.prototype.unhide = function () {
o[n[i].row+1] = { text: o[n[i].row+1], row: i + 1, paired: true };
+
this.runHooks('unhide', 'before');
}
+
if (typeof this.visible != 'undefined' && !this.visible) {
 +
this.mainDiv.style.display='inline';
 +
this.visible=true;
 
}
 
}
 +
this.runHooks('unhide', 'after');
 +
};
  
// pass 5: pair matching rows immediately preceding paired rows (not necessarily unique)
+
/**
for ( i = n.length - 1; i > 0; i-- ) {
+
  Sets the <code>innerHTML</code> attribute of the main div containing the popup content.
if ( n[i].paired && ! n[i-1].paired && n[i].row > 0 && ! o[ n[i].row - 1 ].paired &&
+
  @param {String} html The HTML to set.
n[i-1] == o[ n[i].row - 1 ] ) {
+
*/
n[i-1] = { text: n[i-1], row: n[i].row - 1, paired: true };
+
Navpopup.prototype.setInnerHTML = function (html) {
o[n[i].row-1] = { text: o[n[i].row-1], row: i - 1, paired: true };
+
this.mainDiv.innerHTML = html;
}
+
};
}
 
  
return { o: o, n: n };
+
/**
}
+
  Updates the {@link #width} and {@link #height} attributes with the CSS properties.
//</NOLITE>
+
  @private
// ENDFILE: diff.js
+
*/
// STARTFILE: init.js
+
Navpopup.prototype.updateDimensions = function () {
function setSiteInfo() {
+
this.width=parseInt(this.mainDiv.offsetWidth, 10);
if (window.popupLocalDebug) {
+
this.height=parseInt(this.mainDiv.offsetHeight, 10);
pg.wiki.hostname = 'en.wikipedia.org';
+
};
} else {
 
pg.wiki.hostname = location.hostname; // use in preference to location.hostname for flexibility (?)
 
}
 
pg.wiki.wikimedia=RegExp('(wiki([pm]edia|source|books|news|quote|versity)|wiktionary|mediawiki)[.]org').test(pg.wiki.hostname);
 
pg.wiki.wikia=RegExp('[.]wikia[.]com$', 'i').test(pg.wiki.hostname);
 
pg.wiki.isLocal=RegExp('^localhost').test(pg.wiki.hostname);
 
pg.wiki.commons=( pg.wiki.wikimedia && pg.wiki.hostname != 'commons.wikimedia.org') ? 'commons.wikimedia.org' : null;
 
pg.wiki.lang = mw.config.get('wgContentLanguage');
 
var port = location.port ? ':' + location.port : '';
 
pg.wiki.sitebase = pg.wiki.hostname + port;
 
}
 
  
function setUserInfo() {
+
/**
var api = new mw.Api( {
+
  Checks if the point (x,y) is within {@link #fuzz} of the
    ajax: {
+
  {@link #mainDiv}.
        headers: { 'Api-User-Agent': pg.misc.userAgent }
+
  @param {integer} x x-coordinate (px)
    }
+
  @param {integer} y y-coordinate (px)
} );
+
  @type boolean
var params = {
+
*/
action: 'query',
+
Navpopup.prototype.isWithin = function(x,y) {
list: 'users',
+
//~ If we're not even visible, no point should be considered as
ususers: mw.config.get('wgUserName'),
+
//~ being within the popup.
usprop: 'rights'
+
if (!this.visible) { return false; }
};
+
this.updateDimensions();
   
+
var fuzz=this.fuzz || 0;
pg.user.canReview = false;
+
//~ Use a simple box metric here.
    if (getValueOf('popupReview')) {
+
return (x+fuzz >= this.left && x-fuzz <= this.left + this.width &&
api.get(params).done(function(data){
+
y+fuzz >= this.top  && y-fuzz <= this.top  + this.height);
var rights = data.query.users[0].rights;
+
};
pg.user.canReview = rights.indexOf('review') !== -1; // TODO: Should it be a getValueOf('ReviewRight') ?
 
});
 
    }
 
}
 
  
function setTitleBase() {
+
/**
var protocol = ( window.popupLocalDebug ? 'http:' : location.protocol );
+
  Adds a download to {@link #downloads}.
pg.wiki.articlePath = mw.config.get('wgArticlePath').replace(/\/\$1/, "");   // as in http://some.thing.com/wiki/Article
+
  @param {Downloader} download
pg.wiki.botInterfacePath = mw.config.get('wgScript');
+
*/
pg.wiki.APIPath = mw.config.get('wgScriptPath') +"/api.php";
+
Navpopup.prototype.addDownload=function(download) {
// default mediawiki setting is paths like http://some.thing.com/articlePath/index.php?title=foo
+
if (!download) { return; }
 +
this.downloads.push(download);
 +
};
 +
/**
 +
  Aborts the downloads listed in {@link #downloads}.
 +
  @see Downloader#abort
 +
*/
 +
Navpopup.prototype.abortDownloads=function() {
 +
for(var i=0; i<this.downloads.length; ++i) {
 +
var d=this.downloads[i];
 +
if (d && d.abort) { d.abort(); }
 +
}
 +
this.downloads=[];
 +
};
  
var titletail = pg.wiki.botInterfacePath + '?title=';
 
//var titletail2 = joinPath([pg.wiki.botInterfacePath, 'wiki.phtml?title=']);
 
  
// other sites may need to add code here to set titletail depending on how their urls work
+
/**
 +
  A {@link Mousetracker} instance which is a property of the constructor (pseudo-global).
 +
*/
 +
Navpopup.tracker=new Mousetracker();
 +
// ENDFILE: navpopup.js
 +
// STARTFILE: diff.js
 +
//<NOLITE>
 +
/*
 +
* Javascript Diff Algorithm
 +
*  By John Resig (http://ejohn.org/) and [[:en:User:Lupin]]
 +
*
 +
* More Info:
 +
*  http://ejohn.org/projects/javascript-diff-algorithm/
 +
*/
  
pg.wiki.titlebase  = protocol + '//' + pg.wiki.sitebase + titletail;
+
function delFmt(x) {
//pg.wiki.titlebase2  = protocol + '//' + joinPath([pg.wiki.sitebase, titletail2]);
+
if (!x.length) { return ''; }
pg.wiki.wikibase = protocol + '//' + pg.wiki.sitebase + pg.wiki.botInterfacePath;
+
return "<del class='popupDiff'>" + x.join('') +"</del>";
pg.wiki.apiwikibase = protocol + '//' + pg.wiki.sitebase + pg.wiki.APIPath;
+
}
pg.wiki.articlebase = protocol + '//' + pg.wiki.sitebase + pg.wiki.articlePath;
+
function insFmt(x) {
pg.wiki.commonsbase = protocol + '//' + pg.wiki.commons  + pg.wiki.botInterfacePath;
+
if (!x.length) { return ''; }
pg.wiki.apicommonsbase = protocol + '//' + pg.wiki.commons  + pg.wiki.APIPath;
+
return "<ins class='popupDiff'>" + x.join('') +"</ins>";
pg.re.basenames = RegExp( '^(' +
 
  map( literalizeRegex, [ pg.wiki.titlebase, //pg.wiki.titlebase2,
 
  pg.wiki.articlebase ]).join('|') + ')' );
 
 
}
 
}
  
 +
function countCrossings(a, b, i, eject) {
 +
// count the crossings on the edge starting at b[i]
 +
if (!b[i].row && b[i].row !== 0) { return -1; }
 +
var count=0;
 +
for (var j=0; j<a.length; ++j) {
 +
if (!a[j].row && a[j].row !== 0) { continue; }
 +
if ( (j-b[i].row)*(i-a[j].row) > 0) {
 +
if(eject) { return true; }
 +
count++;
 +
}
 +
}
 +
return count;
 +
}
  
//////////////////////////////////////////////////
+
function shortenDiffString(str, context) {
// Global regexps
+
var re=RegExp('(<del[\\s\\S]*?</del>|<ins[\\s\\S]*?</ins>)');
 
+
var splitted=str.parenSplit(re);
function setMainRegex() {
+
var ret=[''];
var reStart='[^:]*://';
+
for (var i=0; i<splitted.length; i+=2) {
var preTitles = literalizeRegex( mw.config.get('wgScriptPath') ) + '/(?:index[.]php|wiki[.]phtml)[?]title=';
+
if (splitted[i].length < 2*context) {
preTitles += '|' + literalizeRegex( pg.wiki.articlePath + '/' );
+
ret[ret.length-1] += splitted[i];
 
+
if (i+1<splitted.length) { ret[ret.length-1] += splitted[i+1]; }
var reEnd='(' + preTitles + ')([^&?#]*)[^#]*(?:#(.+))?';
+
continue;
pg.re.main = RegExp(reStart + literalizeRegex(pg.wiki.sitebase) + reEnd);
+
}
 +
else {
 +
if (i > 0) { ret[ret.length-1] += splitted[i].substring(0,context); }
 +
if (i+1 < splitted.length) {
 +
ret.push(splitted[i].substring(splitted[i].length-context) +
 +
splitted[i+1]);
 +
}
 +
}
 +
}
 +
while (ret.length > 0 && !ret[0]) { ret = ret.slice(1); }
 +
return ret;
 
}
 
}
  
function setRegexps() {
 
// TODO: We shoud use an api call to get the aliases for special pages, now it does not work for non-English wikipedias:
 
// E.g., https://ru.wikipedia.org/w/api.php?action=query&meta=siteinfo&siprop=specialpagealiases&formatversion=2
 
setMainRegex();
 
var sp=nsRe(pg.nsSpecialId);
 
pg.re.urlNoPopup=RegExp('((title=|/)' + sp + '(?:%3A|:)|section=[0-9]|^#$)') ;
 
pg.re.contribs  =RegExp('(title=|/)'  + sp + '(?:%3A|:)Contributions' + '(&target=|/|/' + nsRe(pg.nsUserId)+':)(.*)') ;
 
pg.re.email     =RegExp('(title=|/)'  + sp + '(?:%3A|:)EmailUser' + '(&target=|/|/(?:' + nsRe(pg.nsUserId)+':)?)(.*)') ;
 
pg.re.backlinks =RegExp('(title=|/)'  + sp + '(?:%3A|:)WhatLinksHere' + '(&target=|/)([^&]*)');
 
pg.re.specialdiff=RegExp('/'          + sp + '(?:%3A|:)Diff/([^?#]*)');
 
  
//<NOLITE>
+
function diffString( o, n, simpleSplit ) {
var im=nsReImage();
+
var splitRe=RegExp('([[]{2}|[\\]]{2}|[{]{2,3}|[}]{2,3}|[|]|=|<|>|[*:]+|\\s|\\b)');
// note: tries to get images in infobox templates too, e.g. movie pages, album pages etc
 
//   (^|\[\[)image: *([^|\]]*[^|\] ]) *
 
//   (^|\[\[)image: *([^|\]]*[^|\] ])([^0-9\]]*([0-9]+) *px)?
 
// $4 = 120 as in 120px
 
pg.re.image = RegExp('(^|\\[\\[)' + im + ': *([^|\\]]*[^|\\] ])' +
 
'([^0-9\\]]*([0-9]+) *px)?|(?:\\n *[|]?|[|]) *' +
 
'(' + getValueOf('popupImageVarsRegexp') + ')' +
 
' *= *(?:\\[\\[ *)?(?:' + im + ':)?' +
 
'([^|]*?)(?:\\]\\])? *[|]? *\\n', 'img') ;
 
pg.re.imageBracketCount = 6;
 
  
pg.re.category = RegExp('\\[\\[' +nsRe(pg.nsCategoryId) +
+
//  We need to split the strings o and n first, and entify() the parts
': *([^|\\]]*[^|\\] ]) *', 'i');
+
//  individually, so that the HTML entities are never cut apart. (AxelBoldt)
pg.re.categoryBracketCount = 1;
+
var out, i, oSplitted, nSplitted;
 +
if (simpleSplit) {
 +
oSplitted=o.split(/\b/);
 +
nSplitted=n.split(/\b/);
 +
} else {
 +
oSplitted=o.parenSplit(splitRe);
 +
nSplitted=n.parenSplit(splitRe);
 +
}
 +
for (i=0; i<oSplitted.length; ++i) {oSplitted[i]=oSplitted[i].entify();}
 +
for (i=0; i<nSplitted.length; ++i) {nSplitted[i]=nSplitted[i].entify();}
 +
 +
out = diff (oSplitted, nSplitted);
 +
var str = "";
 +
var acc=[]; // accumulator for prettier output
  
pg.re.ipUser=RegExp('^' +
+
// crossing pairings -- eg 'A B' vs 'B A' -- cause problems, so let's iron them out
// IPv6
+
// this doesn't always do things optimally but it should be fast enough
'(?::(?::|(?::[0-9A-Fa-f]{1,4}){1,7})|[0-9A-Fa-f]{1,4}(?::[0-9A-Fa-f]{1,4}){0,6}::|[0-9A-Fa-f]{1,4}(?::[0-9A-Fa-f]{1,4}){7})' +
+
var maxOutputPair=0;
// IPv4
+
for (i=0; i<out.n.length; ++i) {
'|(((25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9])\\.){3}' +
+
if ( out.n[i].paired ) {
'(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9]))$');
+
if( maxOutputPair > out.n[i].row ) {
 +
// tangle - delete pairing
 +
out.o[ out.n[i].row ]=out.o[ out.n[i].row ].text;
 +
out.n[i]=out.n[i].text;
 +
}
 +
if (maxOutputPair < out.n[i].row) { maxOutputPair = out.n[i].row; }
 +
}
 +
}
  
pg.re.stub= RegExp(getValueOf('popupStubRegexp'), 'im');
+
// output the stuff preceding the first paired old line
pg.re.disambig=RegExp(getValueOf('popupDabRegexp'), 'im');
+
for (i=0; i<out.o.length && !out.o[i].paired; ++i) { acc.push( out.o[i] ); }
 +
str += delFmt(acc); acc=[];
  
//</NOLITE>
+
// main loop
// FIXME replace with general parameter parsing function, this is daft
+
for ( i = 0; i < out.n.length; ++i ) {
pg.re.oldid=RegExp('[?&]oldid=([^&]*)');
+
// output unpaired new "lines"
pg.re.diff=RegExp('[?&]diff=([^&]*)');
+
while ( i < out.n.length && !out.n[i].paired ) { acc.push( out.n[i++] ); }
 +
str += insFmt(acc); acc=[];
 +
if ( i < out.n.length ) { // this new "line" is paired with the (out.n[i].row)th old "line"
 +
str += out.n[i].text;
 +
// output unpaired old rows starting after this new line's partner
 +
var m = out.n[i].row + 1;
 +
while ( m < out.o.length && !out.o[m].paired ) { acc.push ( out.o[m++] ); }
 +
str += delFmt(acc); acc=[];
 +
}
 +
}
 +
return str;
 
}
 
}
  
 +
// see http://developer.mozilla.org/en/docs/Core_JavaScript_1.5_Reference:Global_Objects:Object
 +
// FIXME: use obj.hasOwnProperty instead of this kludge!
 +
window.jsReservedProperties=RegExp('^(constructor|prototype|__((define|lookup)[GS]etter)__' +
 +
  '|eval|hasOwnProperty|propertyIsEnumerable' +
 +
  '|to(Source|String|LocaleString)|(un)?watch|valueOf)$');
 +
function diffBugAlert(word) {
 +
if (!diffBugAlert.list[word]) {
 +
diffBugAlert.list[word]=1;
 +
alert('Bad word: '+word+'\n\nPlease report this bug.');
 +
}
 +
}
 +
diffBugAlert.list={};
  
//////////////////////////////////////////////////
+
function makeDiffHashtable(src) {
// miscellany
+
var ret={};
 
+
for ( var i = 0; i < src.length; i++ ) {
function setupCache() {
+
if ( jsReservedProperties.test(src[i]) ) { src[i] += '<!-- -->'; }
// page caching
+
if ( !ret[ src[i] ] ) { ret[ src[i] ] = []; }
pg.cache.pages = [];
+
try { ret[ src[i] ].push( i ); } catch (err) { diffBugAlert(src[i]); }
 +
}
 +
return ret;
 
}
 
}
  
function setMisc() {
+
function diff( o, n ) {
pg.current.link=null;
 
pg.current.links=[];
 
pg.current.linksHash={};
 
  
setupCache();
+
// pass 1: make hashtable ns with new rows as keys
 +
var ns = makeDiffHashtable(n);
  
pg.timer.checkPopupPosition=null;
+
// pass 2: make hashtable os with old rows as keys
pg.counter.loop=0;
+
var os = makeDiffHashtable(o);
  
// ids change with each popup: popupImage0, popupImage1 etc
+
// pass 3: pair unique new rows and matching unique old rows
pg.idNumber=0;
+
var i;
 +
for ( i in ns ) {
 +
if ( ns[i].length == 1 && os[i] && os[i].length == 1 ) {
 +
n[ ns[i][0] ] = { text: n[ ns[i][0] ], row: os[i][0], paired: true };
 +
o[ os[i][0] ] = { text: o[ os[i][0] ], row: ns[i][0], paired: true };
 +
}
 +
}
  
// for myDecodeURI
+
// pass 4: pair matching rows immediately following paired rows (not necessarily unique)
pg.misc.decodeExtras = [
+
for ( i = 0; i < n.length - 1; i++ ) {
{from: '%2C', to: ',' },
+
if ( n[i].paired && ! n[i+1].paired && n[i].row + 1 < o.length && ! o[ n[i].row + 1 ].paired &&
{from: '_',   to: ' ' },
+
n[i+1] == o[ n[i].row + 1 ] ) {
{from: '%24', to: '$'},
+
n[i+1] = { text: n[i+1], row: n[i].row + 1, paired: true };
{from: '%26',  to: '&' } // no ,
+
o[n[i].row+1] = { text: o[n[i].row+1], row: i + 1, paired: true };
];
+
}
 +
}
  
pg.misc.userAgent = 'Navigation popups/1.0 (' + mw.config.get( 'wgServerName' ) +')';
+
// pass 5: pair matching rows immediately preceding paired rows (not necessarily unique)
}
+
for ( i = n.length - 1; i > 0; i-- ) {
 
+
if ( n[i].paired && ! n[i-1].paired && n[i].row > 0 && ! o[ n[i].row - 1 ].paired &&
// We need a callback since this might end up asynchronous because of
+
n[i-1] == o[ n[i].row - 1 ] ) {
// the mw.loader.using() call.
+
n[i-1] = { text: n[i-1], row: n[i].row - 1, paired: true };
function setupPopups( callback ) {
+
o[n[i].row-1] = { text: o[n[i].row-1], row: i - 1, paired: true };
if ( setupPopups.completed ) {
 
if ( $.isFunction( callback ) ) {
 
callback();
 
 
}
 
}
return;
 
 
}
 
}
// These dependencies are also enforced from the gadget,
 
// but not everyone loads this as a gadget, so double check
 
mw.loader.using( ['mediawiki.util', 'mediawiki.user', 'user.options'] ).then( function() {
 
// NB translatable strings should be set up first (strings.js)
 
// basics
 
setupDebugging();
 
setSiteInfo();
 
setTitleBase();
 
setOptions(); // see options.js
 
setUserInfo();
 
  
// namespaces etc
+
return { o: o, n: n };
setNamespaces();
+
}
setInterwiki();
+
//</NOLITE>
 +
// ENDFILE: diff.js
 +
// STARTFILE: init.js
 +
function setSiteInfo() {
 +
if (window.popupLocalDebug) {
 +
pg.wiki.hostname = 'en.wikipedia.org';
 +
} else {
 +
pg.wiki.hostname = location.hostname; // use in preference to location.hostname for flexibility (?)
 +
}
 +
pg.wiki.wikimedia=RegExp('(wiki([pm]edia|source|books|news|quote|versity)|wiktionary|mediawiki)[.]org').test(pg.wiki.hostname);
 +
pg.wiki.wikia=RegExp('[.]wikia[.]com$', 'i').test(pg.wiki.hostname);
 +
pg.wiki.isLocal=RegExp('^localhost').test(pg.wiki.hostname);
 +
pg.wiki.commons=( pg.wiki.wikimedia && pg.wiki.hostname != 'commons.wikimedia.org') ? 'commons.wikimedia.org' : null;
 +
pg.wiki.lang = mw.config.get('wgContentLanguage');
 +
var port = location.port ? ':' + location.port : '';
 +
pg.wiki.sitebase = pg.wiki.hostname + port;
 +
}
  
// regexps
+
function setTitleBase() {
setRegexps();
+
var protocol = ( window.popupLocalDebug ? 'http:' : location.protocol );
setRedirs();
+
pg.wiki.articlePath = mw.config.get('wgArticlePath').replace(/\/\$1/, "");  // as in http://some.thing.com/wiki/Article
 +
pg.wiki.botInterfacePath = mw.config.get('wgScript');
 +
pg.wiki.APIPath = mw.config.get('wgScriptPath') +"/api.php";
 +
// default mediawiki setting is paths like http://some.thing.com/articlePath/index.php?title=foo
  
// other stuff
+
var titletail = pg.wiki.botInterfacePath + '?title=';
setMisc();
+
//var titletail2 = joinPath([pg.wiki.botInterfacePath, 'wiki.phtml?title=']);
setupLivePreview();
 
  
// main deal here
+
// other sites may need to add code here to set titletail depending on how their urls work
setupTooltips();
 
log('In setupPopups(), just called setupTooltips()');
 
Navpopup.tracker.enable();
 
  
setupPopups.completed = true;
+
pg.wiki.titlebase  = protocol + '//' + pg.wiki.sitebase + titletail;
if ( $.isFunction( callback ) ) {
+
//pg.wiki.titlebase2  = protocol + '//' + joinPath([pg.wiki.sitebase, titletail2]);
callback();
+
pg.wiki.wikibase = protocol + '//' + pg.wiki.sitebase + pg.wiki.botInterfacePath;
}
+
pg.wiki.apiwikibase = protocol + '//' + pg.wiki.sitebase + pg.wiki.APIPath;
});
+
pg.wiki.articlebase = protocol + '//' + pg.wiki.sitebase + pg.wiki.articlePath;
 +
pg.wiki.commonsbase = protocol + '//' + pg.wiki.commons  + pg.wiki.botInterfacePath;
 +
pg.wiki.apicommonsbase = protocol + '//' + pg.wiki.commons  + pg.wiki.APIPath;
 +
pg.re.basenames = RegExp( '^(' +
 +
  map( literalizeRegex, [ pg.wiki.titlebase, //pg.wiki.titlebase2,
 +
  pg.wiki.articlebase ]).join('|') + ')' );
 
}
 
}
// ENDFILE: init.js
 
// STARTFILE: navlinks.js
 
//<NOLITE>
 
//////////////////////////////////////////////////
 
// navlinks... let the fun begin
 
//
 
  
function defaultNavlinkSpec() {
 
var str='';
 
str += '<b><<mainlink|shortcut= >></b>';
 
if (getValueOf('popupLastEditLink')) {
 
str += '*<<lastEdit|shortcut=/>>|<<lastContrib>>|<<sinceMe>>if(oldid){|<<oldEdit>>|<<diffCur>>}';
 
}
 
  
// user links
+
//////////////////////////////////////////////////
// contribs - log - count - email - block
+
// Global regexps
// count only if applicable; block only if popupAdminLinks
 
str += 'if(user){<br><<contribs|shortcut=c>>*<<userlog|shortcut=L|log>>';
 
str+='if(ipuser){*<<arin>>}if(wikimedia){*<<count|shortcut=#>>}';
 
str+='if(ipuser){}else{*<<email|shortcut=E>>}if(admin){*<<block|shortcut=b>>|<<blocklog|log>>}}';
 
  
// editing links
+
function setMainRegex() {
// talkpage  -> edit|new - history - un|watch - article|edit
+
var reStart='[^:]*://';
// other page -> edit - history - un|watch - talk|edit|new
+
var preTitles = literalizeRegex( mw.config.get('wgScriptPath') ) + '/(?:index[.]php|wiki[.]phtml)[?]title=';
var editstr='<<edit|shortcut=e>>';
+
preTitles += '|' + literalizeRegex( pg.wiki.articlePath + '/' );
var editOldidStr='if(oldid){<<editOld|shortcut=e>>|<<revert|shortcut=v|rv>>|<<edit|cur>>}else{' + editstr + '}';
 
var historystr='<<history|shortcut=h>>|<<editors|shortcut=E|>>';
 
var watchstr='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';
 
  
str+='<br>if(talk){' +
+
var reEnd='(' + preTitles + ')([^&?#]*)[^#]*(?:#(.+))?';
editOldidStr+'|<<new|shortcut=+>>' + '*' + historystr+'*'+watchstr + '*' +
+
pg.re.main = RegExp(reStart + literalizeRegex(pg.wiki.sitebase) + reEnd);
'<b><<article|shortcut=a>></b>|<<editArticle|edit>>' +
 
'}else{' + // not a talk page
 
editOldidStr + '*' + historystr + '*' + watchstr + '*' +
 
'<b><<talk|shortcut=t>></b>|<<editTalk|edit>>|<<newTalk|shortcut=+|new>>}';
 
 
 
// misc links
 
str += '<br><<whatLinksHere|shortcut=l>>*<<relatedChanges|shortcut=r>>*<<move|shortcut=m>>';
 
 
 
// admin links
 
str += 'if(admin){<br><<unprotect|unprotectShort>>|<<protect|shortcut=p>>|<<protectlog|log>>*' +
 
'<<undelete|undeleteShort>>|<<delete|shortcut=d>>|<<deletelog|log>>}';
 
return str;
 
 
}
 
}
  
function navLinksHTML (article, hint, params) { //oldid, rcid) {
+
function setRegexps() {
var str = '<span class="popupNavLinks">' + defaultNavlinkSpec() + '</span>';
+
setMainRegex();
// BAM
+
var sp=nsRe(pg.nsSpecialId);
return navlinkStringToHTML(str, article, params);
+
pg.re.urlNoPopup=RegExp('((title=|/)' + sp + '(?:%3A|:)|section=[0-9]|^#$)') ;
}
+
pg.re.contribs  =RegExp('(title=|/)'  + sp + '(?:%3A|:)Contributions' + '(&target=|/|/' + mw.config.get('wgFormattedNamespaces')[pg.nsUserId]+':)(.*)') ;
 +
pg.re.email     =RegExp('(title=|/)' + sp + '(?:%3A|:)EmailUser' + '(&target=|/|/(?:' + mw.config.get('wgFormattedNamespaces')[pg.nsUserId]+':)?)(.*)') ;
 +
pg.re.backlinks =RegExp('(title=|/)'  + sp + '(?:%3A|:)WhatLinksHere' + '(&target=|/)([^&]*)');
 +
pg.re.specialdiff=RegExp('/'          + sp + '(?:%3A|:)Diff/([^?#]*)');
  
function expandConditionalNavlinkString(s,article,z,recursionCount) {
+
//<NOLITE>
var oldid=z.oldid, rcid=z.rcid, diff=z.diff;
+
var im=nsReImage();
// nested conditionals (up to 10 deep) are ok, hopefully! (work from the inside out)
+
// note: tries to get images in infobox templates too, e.g. movie pages, album pages etc
if (typeof recursionCount!=typeof 0) { recursionCount=0; }
+
//   (^|\[\[)image: *([^|\]]*[^|\] ]) *
var conditionalSplitRegex=RegExp(
+
//   (^|\[\[)image: *([^|\]]*[^|\] ])([^0-9\]]*([0-9]+) *px)?
//(1 if \\( (2 2) \\)   {(3 3)}  (4  else   {(5 5)4)1)
+
// $4 = 120 as in 120px
'(;?\\s*if\\s*\\(\\s*([\\w]*)\\s*\\)\\s*\\{([^{}]*)\\}(\\s*else\\s*\\{([^{}]*?)\\}|))', 'i');
+
pg.re.image = RegExp('(^|\\[\\[)' + im + ': *([^|\\]]*[^|\\] ])' +
var splitted=s.parenSplit(conditionalSplitRegex);
+
'([^0-9\\]]*([0-9]+) *px)?|(?:\\n *[|]?|[|]) *' +
// $1: whole conditional
+
'(' + getValueOf('popupImageVarsRegexp') + ')' +
// $2: test condition
+
' *= *(?:\\[\\[ *)?(?:' + im + ':)?' +
// $3: true expansion
+
'([^|]*?)(?:\\]\\])? *[|]? *\\n', 'img') ;
// $4: else clause (possibly empty)
+
pg.re.imageBracketCount = 6;
// $5: false expansion (possibly null)
 
var numParens=5;
 
var ret = splitted[0];
 
for (var i=1; i<splitted.length; i=i+numParens+1) {
 
  
var testString=splitted[i+2-1];
+
pg.re.category = RegExp('\\[\\[' +nsRe(pg.nsCategoryId) +
var trueString=splitted[i+3-1];
+
': *([^|\\]]*[^|\\] ]) *', 'i');
var falseString=splitted[i+5-1];
+
pg.re.categoryBracketCount = 1;
if (typeof falseString=='undefined' || !falseString) { falseString=''; }
 
var testResult=null;
 
  
switch (testString) {
+
pg.re.ipUser=RegExp('^' +
case 'user':
+
// IPv6
testResult=(article.userName())?true:false;
+
'(?::(?::|(?::[0-9A-Fa-f]{1,4}){1,7})|[0-9A-Fa-f]{1,4}(?::[0-9A-Fa-f]{1,4}){0,6}::|[0-9A-Fa-f]{1,4}(?::[0-9A-Fa-f]{1,4}){7})' +
break;
+
// IPv4
case 'talk':
+
'|(((25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9])\\.){3}' +
testResult=(article.talkPage())?false:true; // talkPage converts _articles_ to talkPages
+
'(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9]))$');
break;
 
case 'admin':
 
testResult=getValueOf('popupAdminLinks')?true:false;
 
break;
 
case 'oldid':
 
testResult=(typeof oldid != 'undefined' && oldid)?true:false;
 
break;
 
case 'rcid':
 
testResult=(typeof rcid != 'undefined' && rcid)?true:false;
 
break;
 
case 'ipuser':
 
testResult=(article.isIpUser())?true:false;
 
break;
 
case 'mainspace_en':
 
testResult=isInMainNamespace(article) &&
 
pg.wiki.hostname=='en.wikipedia.org';
 
break;
 
case 'wikimedia':
 
testResult=(pg.wiki.wikimedia) ? true : false;
 
break;
 
case 'diff':
 
testResult=(typeof diff != 'undefined' && diff)?true:false;
 
break;
 
}
 
  
switch(testResult) {
+
pg.re.stub= RegExp(getValueOf('popupStubRegexp'), 'im');
case null: ret+=splitted[i];  break;
+
pg.re.disambig=RegExp(getValueOf('popupDabRegexp'), 'im');
case true: ret+=trueString;  break;
 
case false: ret+=falseString; break;
 
}
 
  
// append non-conditional string
+
//</NOLITE>
ret += splitted[i+numParens];
+
// FIXME replace with general parameter parsing function, this is daft
}
+
pg.re.oldid=RegExp('[?&]oldid=([^&]*)');
if (conditionalSplitRegex.test(ret) && recursionCount < 10) {
+
pg.re.diff=RegExp('[?&]diff=([^&]*)');
return expandConditionalNavlinkString(ret,article,z,recursionCount+1);
 
}
 
return ret;
 
 
}
 
}
  
function navlinkStringToArray(s, article, params) {
+
 
s=expandConditionalNavlinkString(s,article,params);
+
//////////////////////////////////////////////////
var splitted=s.parenSplit(RegExp('<<(.*?)>>'));
+
// miscellany
var ret=[];
+
 
for (var i=0; i<splitted.length; ++i) {
+
function setupCache() {
if (i%2) { // i odd, so s is a tag
+
// page caching
var t=new navlinkTag();
+
pg.cache.pages = [];
var ss=splitted[i].split('|');
 
t.id=ss[0];
 
for (var j=1; j<ss.length; ++j) {
 
var sss=ss[j].split('=');
 
if (sss.length>1) {
 
t[sss[0]]=sss[1];
 
}
 
else { // no assignment (no "="), so treat this as a title (overwriting the last one)
 
t.text=popupString(sss[0]);
 
}
 
}
 
t.article=article;
 
var oldid=params.oldid, rcid=params.rcid, diff=params.diff;
 
if (typeof oldid !== 'undefined' && oldid !== null) { t.oldid=oldid; }
 
if (typeof rcid !== 'undefined' && rcid !== null) { t.rcid=rcid; }
 
if (typeof diff !== 'undefined' && diff !== null) { t.diff=diff; }
 
if (!t.text && t.id !== 'mainlink') { t.text=popupString(t.id); }
 
ret.push(t);
 
}
 
else { // plain HTML
 
ret.push(splitted[i]);
 
}
 
}
 
return ret;
 
 
}
 
}
  
 +
function setMisc() {
 +
pg.current.link=null;
 +
pg.current.links=[];
 +
pg.current.linksHash={};
  
function navlinkSubstituteHTML(s) {
+
setupCache();
return s.split('*').join(getValueOf('popupNavLinkSeparator'))
 
.split('<menurow>').join('<li class="popup_menu_row">')
 
.split('</menurow>').join('</li>')
 
.split('<menu>').join('<ul class="popup_menu">')
 
.split('</menu>').join('</ul>');
 
  
}
+
pg.timer.checkPopupPosition=null;
 +
pg.counter.loop=0;
  
function navlinkDepth(magic,s) {
+
// ids change with each popup: popupImage0, popupImage1 etc
return s.split('<' + magic + '>').length - s.split('</' + magic + '>').length;
+
pg.idNumber=0;
}
 
  
 +
// for myDecodeURI
 +
pg.misc.decodeExtras = [
 +
{from: '%2C', to: ',' },
 +
{from: '_',  to: ' ' },
 +
{from: '%24', to: '$'},
 +
{from: '%26',  to: '&' } // no ,
 +
];
  
// navlinkString: * becomes the separator
 
// <<foo|bar=baz|fubar>> becomes a foo-link with attribute bar='baz'
 
//   and visible text 'fubar'
 
// if(test){...} and if(test){...}else{...} work too (nested ok)
 
 
function navlinkStringToHTML(s,article,params) {
 
//limitAlert(navlinkStringToHTML, 5, 'navlinkStringToHTML\n' + article + '\n' + (typeof article));
 
var p=navlinkStringToArray(s,article,params);
 
var html='';
 
var menudepth = 0; // nested menus not currently allowed, but doesn't do any harm to code for it
 
var menurowdepth = 0;
 
for (var i=0; i<p.length; ++i) {
 
if (typeof p[i] == typeof '') {
 
html+=navlinkSubstituteHTML(p[i]);
 
menudepth += navlinkDepth('menu', p[i]);
 
menurowdepth += navlinkDepth('menurow', p[i]);
 
// if (menudepth === 0) {
 
// tagType='span';
 
// } else if (menurowdepth === 0) {
 
// tagType='li';
 
// } else {
 
// tagType = null;
 
// }
 
} else if (typeof p[i].type != 'undefined' && p[i].type=='navlinkTag') {
 
if (menudepth > 0 && menurowdepth === 0) {
 
html += '<li class="popup_menu_item">' + p[i].html() + '</li>';
 
} else {
 
html+=p[i].html();
 
}
 
}
 
}
 
return html;
 
 
}
 
}
  
function navlinkTag() {
+
function leadingInteger(s){
this.type='navlinkTag';
+
var n=s.match(/^(\d*)/)[1];
 +
if (n) { return +n; }
 +
return null;
 
}
 
}
  
navlinkTag.prototype.html=function () {
+
function setBrowserHacks() {
this.getNewWin();
+
var useOriginal=false;
this.getPrintFunction();
+
// browser-specific hacks
var html='';
+
if (typeof window.opera != 'undefined') {
var opening, closing;
+
//if (leadingInteger(opera.version()) < 9)
var tagType='span';
+
{ useOriginal=true; } // v9 beta still seems to have buggy css
if (!tagType) {
+
setDefault('popupNavLinkSeparator', ' &#183; ');
opening = ''; closing = '';
+
} else if (navigator.appName=='Konqueror') {
} else {
+
setDefault('popupNavLinkSeparator', ' &bull; ');
opening = '<' + tagType + ' class="popup_' + this.id + '">';
+
pg.flag.isKonq=true;
closing = '</' + tagType + '>';
+
} else if ( navigator.vendor && navigator.vendor.toLowerCase().indexOf('apple computer')===0) {
 +
pg.flag.isSafari=true;
 +
var webkit=+navigator.userAgent.replace(RegExp('^.*AppleWebKit[/](\\d+).*', 'i'), '$1');
 +
if (webkit < 420) { useOriginal=true; }
 +
} else if (navigator.appName.indexOf("Microsoft")!=-1) {
 +
setDefault('popupNavLinkSeparator', ' &#183; ');
 +
var ver=+navigator.userAgent.replace(RegExp('^.*MSIE (\\d+).*'), '$1');
 +
pg.flag.isIE=true;
 +
pg.flag.IEVersion=ver;
 
}
 
}
if (typeof this.print!='function') {
+
if (pg.flag.isIE && pg.flag.IEVersion < 8) {
errlog ('Oh dear - invalid print function for a navlinkTag, id='+this.id);
+
useOriginal=true;
} else {
 
html=this.print(this);
 
if (typeof html != typeof '') {html='';}
 
else if (typeof this.shortcut!='undefined') html=addPopupShortcut(html, this.shortcut);
 
 
}
 
}
return opening + html + closing;
+
if ((pg.flag.isIE && pg.flag.IEVersion < 7) || pg.flag.isKonq || (pg.flag.isSafari && webkit < 420)) {
};
+
pg.flag.linksLikeIE6=true;
 
 
navlinkTag.prototype.getNewWin=function() {
 
getValueOf('popupLinksNewWindow');
 
if (typeof pg.option.popupLinksNewWindow[this.id] === 'undefined') { this.newWin=null; }
 
this.newWin=pg.option.popupLinksNewWindow[this.id];
 
};
 
 
 
navlinkTag.prototype.getPrintFunction=function() { //think about this some more
 
// this.id and this.article should already be defined
 
if (typeof this.id!=typeof '' || typeof this.article!=typeof {} ) { return; }
 
 
 
this.noPopup=1;
 
switch (this.id) {
 
case 'contribs': case 'history': case 'whatLinksHere':
 
case 'userPage': case 'monobook': case 'userTalk':
 
case 'talk': case 'article': case 'lastEdit':
 
this.noPopup=null;
 
 
}
 
}
switch (this.id) {
+
if (useOriginal && pg.structures.original) {
case 'email': case 'contribs':  case 'block': case 'unblock':
+
setDefault('popupStructure','original');
case 'userlog':  case 'userSpace': case 'deletedContribs':
 
this.article=this.article.userName();
 
 
}
 
}
 +
}
  
switch (this.id) {
+
// We need a callback since this might end up asynchronous because of
case 'userTalk': case 'newUserTalk': case 'editUserTalk':
+
// the mw.loader.using() call.
case 'userPage': case 'monobook': case 'editMonobook': case 'blocklog':
+
function setupPopups( callback ) {
this.article=this.article.userName(true);
+
mw.loader.using( 'mediawiki.user', function() {
/* fall through */
+
// NB translatable strings should be set up first (strings.js)
case 'pagelog': case 'deletelog': case 'protectlog':
+
// basics
delete this.oldid;
+
setupDebugging();
}
+
setSiteInfo();
 +
setTitleBase();
 +
setOptions(); // see options.js
  
if (this.id=='editMonobook' || this.id=='monobook') { this.article.append('/monobook.js'); }
+
// namespaces etc
 +
setNamespaces();
 +
setInterwiki();
  
if (this.id != 'mainlink') {
+
// regexps
// FIXME anchor handling should be done differently with Title object
+
setRegexps();
this.article=this.article.removeAnchor();
+
setRedirs();
// if (typeof this.text=='undefined') this.text=popupString(this.id);
 
}
 
  
switch (this.id) {
+
// other stuff
case 'undelete':      this.print=specialLink; this.specialpage='Undelete'; this.sep='/'; break;
+
setBrowserHacks();
case 'whatLinksHere':  this.print=specialLink; this.specialpage='Whatlinkshere'; break;
+
setMisc();
case 'relatedChanges': this.print=specialLink; this.specialpage='Recentchangeslinked'; break;
+
setupLivePreview();
case 'move':          this.print=specialLink; this.specialpage='Movepage'; break;
 
case 'contribs':      this.print=specialLink; this.specialpage='Contributions'; break;
 
case 'deletedContribs':this.print=specialLink; this.specialpage='Deletedcontributions'; break;
 
case 'email':          this.print=specialLink; this.specialpage='EmailUser'; this.sep='/'; break;
 
case 'block':          this.print=specialLink; this.specialpage='Blockip'; this.sep='&ip='; break;
 
case 'unblock':        this.print=specialLink; this.specialpage='Ipblocklist'; this.sep='&action=unblock&ip='; break;
 
case 'userlog':        this.print=specialLink; this.specialpage='Log'; this.sep='&user='; break;
 
case 'blocklog':      this.print=specialLink; this.specialpage='Log'; this.sep='&type=block&page='; break;
 
case 'pagelog':        this.print=specialLink; this.specialpage='Log'; this.sep='&page='; break;
 
case 'protectlog':    this.print=specialLink; this.specialpage='Log'; this.sep='&type=protect&page='; break;
 
case 'deletelog':      this.print=specialLink; this.specialpage='Log'; this.sep='&type=delete&page='; break;
 
case 'userSpace':      this.print=specialLink; this.specialpage='PrefixIndex'; this.sep='&namespace=2&prefix='; break;
 
case 'search':        this.print=specialLink; this.specialpage='Search'; this.sep='&fulltext=Search&search='; break;
 
case 'thank':          this.print=specialLink; this.specialpage='Thanks'; this.sep='/'; this.article.value = this.diff; break;
 
case 'unwatch': case 'watch':
 
this.print=magicWatchLink; this.action=this.id+'&autowatchlist=1&autoimpl=' + popupString('autoedit_version') + '&actoken='+autoClickToken(); break;
 
case 'history': case 'historyfeed':
 
case 'unprotect': case 'protect':
 
this.print=wikiLink; this.action=this.id; break;
 
  
case 'delete':
+
// main deal here
this.print=wikiLink; this.action='delete';
+
setupTooltips();
if (this.article.namespaceId()==pg.nsImageId) {
+
Navpopup.tracker.enable();
var img=this.article.stripNamespace();
 
this.action+='&image='+img;
 
}
 
break;
 
  
case 'markpatrolled':
+
setupPopups.completed = true;
case 'edit': // editOld should keep the oldid, but edit should not.
+
if ( $.isFunction( callback ) ) {
delete this.oldid;
+
callback();
/* fall through */
 
case 'view': case 'purge': case 'render':
 
this.print=wikiLink;
 
this.action=this.id; break;
 
case 'raw':
 
this.print=wikiLink; this.action='raw'; break;
 
case 'new':
 
this.print=wikiLink; this.action='edit&section=new'; break;
 
case 'mainlink':
 
if (typeof this.text=='undefined') { this.text=this.article.toString().entify(); }
 
if (getValueOf('popupSimplifyMainLink') && isInStrippableNamespace(this.article)) {
 
// only show the /subpage part of the title text
 
var s=this.text.split('/'); this.text=s[s.length-1];
 
if (this.text==='' && s.length > 1) { this.text=s[s.length-2]; }
 
 
}
 
}
this.print=titledWikiLink;
+
});
if (typeof this.title==='undefined' && pg.current.link && typeof pg.current.link.href !== 'undefined') {
+
}
this.title=safeDecodeURI((pg.current.link.originalTitle)?pg.current.link.originalTitle:this.article);
+
// ENDFILE: init.js
if (typeof this.oldid !== 'undefined' && this.oldid) {
+
// STARTFILE: navlinks.js
this.title=tprintf('Revision %s of %s', [this.oldid, this.title]);
+
//<NOLITE>
}
+
//////////////////////////////////////////////////
}
+
// navlinks... let the fun begin
this.action='view'; break;
 
case 'userPage':
 
case 'article':
 
case 'monobook':
 
case 'editMonobook':
 
case 'editArticle':
 
delete this.oldid;
 
//alert(this.id+'\n'+this.article + '\n'+ typeof this.article);
 
this.article=this.article.articleFromTalkOrArticle();
 
//alert(this.id+'\n'+this.article + '\n'+ typeof this.article);
 
this.print=wikiLink;
 
if (this.id.indexOf('edit')===0) {
 
this.action='edit';
 
} else { this.action='view';}
 
break;
 
case 'userTalk':
 
case 'talk':
 
this.article=this.article.talkPage();
 
delete this.oldid;
 
this.print=wikiLink;
 
this.action='view'; break;
 
case 'arin':
 
this.print=arinLink; break;
 
case 'count':
 
this.print=editCounterLink; break;
 
case 'google':
 
this.print=googleLink; break;
 
case 'editors':
 
this.print=editorListLink; break;
 
case 'globalsearch':
 
this.print=globalSearchLink; break;
 
case 'lastEdit':
 
this.print=titledDiffLink;
 
this.title=popupString('Show the last edit');
 
this.from='prev'; this.to='cur'; break;
 
case 'oldEdit':
 
this.print=titledDiffLink;
 
this.title=popupString('Show the edit made to get revision') + ' ' + this.oldid;
 
this.from='prev'; this.to=this.oldid; break;
 
case 'editOld':
 
this.print=wikiLink; this.action='edit'; break;
 
case 'undo':
 
this.print=wikiLink; this.action='edit&undo='; break;
 
case 'revert':
 
this.print=wikiLink; this.action='revert'; break;
 
case 'nullEdit':
 
this.print=wikiLink; this.action='nullEdit'; break;
 
case 'diffCur':
 
this.print=titledDiffLink;
 
this.title=tprintf('Show changes since revision %s', [this.oldid]);
 
this.from=this.oldid; this.to='cur'; break;
 
case 'editUserTalk':
 
case 'editTalk':
 
delete this.oldid;
 
this.article=this.article.talkPage();
 
this.action='edit'; this.print=wikiLink; break;
 
case 'newUserTalk':
 
case 'newTalk':
 
this.article=this.article.talkPage();
 
this.action='edit&section=new'; this.print=wikiLink; break;
 
case 'lastContrib':
 
case 'sinceMe':
 
this.print=magicHistoryLink;
 
break;
 
case 'togglePreviews':
 
this.text=popupString(pg.option.simplePopups ? 'enable previews' : 'disable previews');
 
/* fall through */
 
case 'disablePopups': case 'purgePopups':
 
this.print=popupMenuLink;
 
break;
 
default:
 
this.print=function () {return 'Unknown navlink type: '+this.id+'';};
 
}
 
};
 
 
//
 
//
//  end navlinks
 
//////////////////////////////////////////////////
 
//</NOLITE>
 
// ENDFILE: navlinks.js
 
// STARTFILE: shortcutkeys.js
 
//<NOLITE>
 
function popupHandleKeypress(evt) {
 
var keyCode = window.event ? window.event.keyCode : ( evt.keyCode ? evt.keyCode : evt.which);
 
if (!keyCode || !pg.current.link || !pg.current.link.navpopup) { return; }
 
if (keyCode==27) { // escape
 
killPopup();
 
return false; // swallow keypress
 
}
 
  
var letter=String.fromCharCode(keyCode);
+
function defaultNavlinkSpec() {
var links=pg.current.link.navpopup.mainDiv.getElementsByTagName('A');
+
var str='';
var startLink=0;
+
str += '<b><<mainlink|shortcut= >></b>';
var i,j;
+
if (getValueOf('popupLastEditLink')) {
 
+
str += '*<<lastEdit|shortcut=/>>|<<lastContrib>>|<<sinceMe>>if(oldid){|<<oldEdit>>|<<diffCur>>}';
if (popupHandleKeypress.lastPopupLinkSelected) {
 
for (i=0; i<links.length; ++i) {
 
if (links[i]==popupHandleKeypress.lastPopupLinkSelected) { startLink=i; }
 
}
 
}
 
for (j=0; j<links.length; ++j) {
 
i=(startLink + j + 1) % links.length;
 
if (links[i].getAttribute('popupkey')==letter) {
 
if (evt && evt.preventDefault) evt.preventDefault();
 
links[i].focus();
 
popupHandleKeypress.lastPopupLinkSelected=links[i];
 
return false; // swallow keypress
 
}
 
 
}
 
}
  
// pass keypress on
+
// user links
if (document.oldPopupOnkeypress) { return document.oldPopupOnkeypress(evt); }
+
// contribs - log - count - email - block
return true;
+
// count only if applicable; block only if popupAdminLinks
}
+
str += 'if(user){<br><<contribs|shortcut=c>>*<<userlog|shortcut=L|log>>';
 +
str+='if(ipuser){*<<arin>>}if(wikimedia){*<<count|shortcut=#>>}';
 +
str+='if(ipuser){}else{*<<email|shortcut=E>>}if(admin){*<<block|shortcut=b>>|<<blocklog|log>>}}';
  
function addPopupShortcuts() {
+
// editing links
if (document.onkeypress!=popupHandleKeypress) {
+
// talkpage  -> edit|new - history - un|watch - article|edit
document.oldPopupOnkeypress=document.onkeypress;
+
// other page -> edit - history - un|watch - talk|edit|new
}
+
var editstr='<<edit|shortcut=e>>';
document.onkeypress=popupHandleKeypress;
+
var editOldidStr='if(oldid){<<editOld|shortcut=e>>|<<revert|shortcut=v|rv>>|<<edit|cur>>}else{' + editstr + '}'
}
+
var historystr='<<history|shortcut=h>>if(mainspace_en){|<<editors|shortcut=E|>>}';
 +
var watchstr='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';
  
function rmPopupShortcuts() {
+
str+='<br>if(talk){' +
popupHandleKeypress.lastPopupLinkSelected=null;
+
editOldidStr+'|<<new|shortcut=+>>' + '*' + historystr+'*'+watchstr + '*' +
try {
+
'<b><<article|shortcut=a>></b>|<<editArticle|edit>>' +
if (document.oldPopupOnkeypress && document.oldPopupOnkeypress==popupHandleKeypress) {
+
'}else{' + // not a talk page
// panic
+
editOldidStr + '*' + historystr + '*' + watchstr + '*' +
document.onkeypress=null; //function () {};
+
'<b><<talk|shortcut=t>></b>|<<editTalk|edit>>|<<newTalk|shortcut=+|new>>'
return;
+
+ '}';
}
 
document.onkeypress=document.oldPopupOnkeypress;
 
} catch (nasties) { /* IE goes here */ }
 
}
 
  
 +
// misc links
 +
str += '<br><<whatLinksHere|shortcut=l>>*<<relatedChanges|shortcut=r>>*<<move|shortcut=m>>';
  
function addLinkProperty(html, property) {
+
// admin links
// take "<a href=...>...</a> and add a property
+
str += 'if(admin){<br><<unprotect|unprotectShort>>|<<protect|shortcut=p>>|<<protectlog|log>>*' +
// not sophisticated at all, easily broken
+
'<<undelete|undeleteShort>>|<<delete|shortcut=d>>|<<deletelog|log>>}';
var i=html.indexOf('>');
+
return str;
if (i<0) { return html; }
 
return html.substring(0,i) + ' ' + property + html.substring(i);
 
 
}
 
}
  
function addPopupShortcut(html, key) {
+
function navLinksHTML (article, hint, params) { //oldid, rcid) {
if (!getValueOf('popupShortcutKeys')) { return html; }
+
var str = '<span class="popupNavLinks">' + defaultNavlinkSpec() + '</span>';
var ret= addLinkProperty(html, 'popupkey="'+key+'"');
+
// BAM
if (key==' ') { key=popupString('spacebar'); }
+
return navlinkStringToHTML(str, article, params);
return ret.replace(RegExp('^(.*?)(title=")(.*?)(".*)$', 'i'),'$1$2$3 ['+key+']$4');
 
 
}
 
}
//</NOLITE>
 
// ENDFILE: shortcutkeys.js
 
// STARTFILE: diffpreview.js
 
//<NOLITE>
 
//lets jump through hoops to find the rev ids we need to retrieve
 
function loadDiff(article, oldid, diff, navpop) {
 
navpop.diffData={ oldRev: {}, newRev: {} };
 
mw.loader.using( 'mediawiki.api' ).then( function() {
 
var api = new mw.Api( {
 
    ajax: {
 
        headers: { 'Api-User-Agent': pg.misc.userAgent }
 
    }
 
} );
 
var params = {
 
action: 'compare',
 
prop: 'ids|title'
 
};
 
if(article.title){
 
params.fromtitle = article.title;
 
}
 
 
switch (diff) {
 
case 'cur':
 
switch ( oldid ) {
 
case null:
 
case '':
 
case 'prev':
 
// this can only work if we have the title
 
// cur -> prev
 
params.torelative = 'prev';
 
break;
 
default:
 
params.fromrev = oldid;
 
params.torelative = 'cur';
 
break;
 
}
 
break;
 
case 'prev':
 
if( oldid ) {
 
params.fromrev = oldid;
 
} else {
 
params.fromtitle;
 
}
 
params.torelative = 'prev';
 
break;
 
case 'next':
 
params.fromrev = oldid || 0;
 
params.torelative = 'next';
 
break;
 
default:
 
params.fromrev = oldid || 0;
 
params.torev = diff || 0;
 
break;
 
}
 
 
api.get( params ).then( function( data ) {
 
navpop.diffData.oldRev.revid = data.compare.fromrevid;
 
navpop.diffData.newRev.revid = data.compare.torevid;
 
 
addReviewLink(navpop, 'popupMiscTools');
 
  
var go = function() {
+
function expandConditionalNavlinkString(s,article,z,recursionCount) {
pendingNavpopTask(navpop);
+
var oldid=z.oldid, rcid=z.rcid, diff=z.diff;
var url=pg.wiki.apiwikibase + '?format=json&formatversion=2&action=query&';
+
// nested conditionals (up to 10 deep) are ok, hopefully! (work from the inside out)
+
if (typeof recursionCount!=typeof 0) { recursionCount=0; }
url += 'revids=' + navpop.diffData.oldRev.revid + '|' + navpop.diffData.newRev.revid;
+
var conditionalSplitRegex=RegExp(
url += '&prop=revisions&rvprop=ids|timestamp|content';
+
//(1 if \\( (2 2) \\)   {(3 3)}  (4  else   {(5 5)}  4)1)
+
'(;?\\s*if\\s*\\(\\s*([\\w]*)\\s*\\)\\s*\\{([^{}]*)\\}(\\s*else\\s*\\{([^{}]*?)\\}|))', 'i');
getPageWithCaching(url, doneDiff, navpop);
+
var splitted=s.parenSplit(conditionalSplitRegex);
+
// $1: whole conditional
return true; // remove hook once run
+
// $2: test condition
};
+
// $3: true expansion
if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
+
// $4: else clause (possibly empty)
else { navpop.addHook(go, 'unhide', 'before', 'DOWNLOAD_DIFFS'); }
+
// $5: false expansion (possibly null)
} );
+
var numParens=5;
} );
+
var ret = splitted[0];
}
+
for (var i=1; i<splitted.length; i=i+numParens+1) {
  
// Put a "mark patrolled" link to an element target
+
var testString=splitted[i+2-1];
// TODO: Allow patrol a revision, as well as a diff
+
var trueString=splitted[i+3-1];
function addReviewLink (navpop, target) {
+
var falseString=splitted[i+5-1];
if (! pg.user.canReview) return;
+
if (typeof falseString=='undefined' || !falseString) { falseString=''; }
// If 'newRev' is older than 'oldRev' than it could be confusing, so we do not show the review link.
+
var testResult=null;
if (navpop.diffData.newRev.revid <= navpop.diffData.oldRev.revid) return;
 
var api = new mw.Api( {
 
    ajax: {
 
        headers: { 'Api-User-Agent': pg.misc.userAgent }
 
    }
 
} );
 
var params = {
 
action: 'query',
 
prop: 'info|flagged',
 
revids: navpop.diffData.oldRev.revid,
 
formatversion : 2
 
};
 
api.get (params).then(function(data){
 
var stable_revid = data.query.pages[0].flagged && data.query.pages[0].flagged.stable_revid || 0;
 
// The diff can be reviewed if the old version is the last reviewed version
 
// TODO: Other possible conditions that we may want to implement instead of this one:
 
//  * old version is patrolled and the new version is not patrolled
 
//  * old version is patrolled and the new version is more recent than the last reviewed version
 
if (stable_revid == navpop.diffData.oldRev.revid) {
 
var a = document.createElement('a');
 
a.innerHTML = popupString('mark patrolled');
 
a.title=popupString('markpatrolledHint');
 
a.onclick = function() {
 
var params = {
 
action: 'review',
 
revid: navpop.diffData.newRev.revid,
 
comment: tprintf('defaultpopupReviewedSummary', [navpop.diffData.oldRev.revid, navpop.diffData.newRev.revid])
 
};
 
api.postWithToken('csrf',params).done(function(){
 
a.style.display = "none";
 
// TODO: Update current page and other already constructed popups
 
} ).fail(function(){
 
alert(popupString('Could not marked this edit as patrolled'));
 
});
 
};
 
setPopupHTML(a, target, navpop.idNumber,null,true);
 
}
 
});
 
}
 
  
function doneDiff(download) {
+
switch (testString) {
if (!download.owner || !download.owner.diffData) { return; }
+
case 'user':
var navpop=download.owner;
+
testResult=(article.userName())?true:false;
completedNavpopTask(navpop);
+
break;
+
case 'talk':
var pages, revisions=[];
+
testResult=(article.talkPage())?false:true; // talkPage converts _articles_ to talkPages
try{
+
break;
// Process the downloads
+
case 'admin':
pages = getJsObj(download.data).query.pages;
+
testResult=getValueOf('popupAdminLinks')?true:false;
for(var i=0; i < pages.length; i++ ) {
+
break;
revisions = revisions.concat(pages[i].revisions);
+
case 'oldid':
}
+
testResult=(typeof oldid != 'undefined' && oldid)?true:false;
for(i=0; i< revisions.length; i++){
+
break;
if(revisions[i].revid == navpop.diffData.oldRev.revid) {
+
case 'rcid':
navpop.diffData.oldRev.revision = revisions[i];
+
testResult=(typeof rcid != 'undefined' && rcid)?true:false;
} else if (revisions[i].revid == navpop.diffData.newRev.revid) {
+
break;
navpop.diffData.newRev.revision = revisions[i];
+
case 'ipuser':
}
+
testResult=(article.isIpUser())?true:false;
}
+
break;
} catch(someError) {
+
case 'mainspace_en':
errlog( 'Could not get diff' );
+
testResult=isInMainNamespace(article) &&
}
+
pg.wiki.hostname=='en.wikipedia.org';
+
break;
insertDiff(navpop);
+
case 'wikimedia':
}
+
testResult=(pg.wiki.wikimedia) ? true : false;
 
+
break;
function rmBoringLines(a,b,context) {
+
case 'diff':
 +
testResult=(typeof diff != 'undefined' && diff)?true:false;
 +
break;
 +
}
  
if (typeof context == 'undefined') { context=2; }
+
switch(testResult) {
// this is fairly slow... i think it's quicker than doing a word-based diff from the off, though
+
case null: ret+=splitted[i]; break;
var aa=[], aaa=[];
+
case true: ret+=trueString;   break;
var bb=[], bbb=[];
+
case false: ret+=falseString; break;
var i, j;
 
 
 
// first, gather all disconnected nodes in a and all crossing nodes in a and b
 
for (i=0; i<a.length; ++i ) {
 
if(!a[i].paired) { aa[i]=1; }
 
else if (countCrossings(b,a,i, true)) {
 
aa[i]=1;
 
bb[ a[i].row ] = 1;
 
 
}
 
}
}
 
  
// pick up remaining disconnected nodes in b
+
// append non-conditional string
for (i=0; i<b.length; ++i ) {
+
ret += splitted[i+numParens];
if (bb[i]==1) { continue; }
 
if(!b[i].paired) { bb[i]=1; }
 
 
}
 
}
 
+
if (conditionalSplitRegex.test(ret) && recursionCount < 10) {
// another pass to gather context: we want the neighbours of included nodes which are not yet included
+
return expandConditionalNavlinkString(ret,article,z,recursionCount+1);
// we have to add in partners of these nodes, but we don't want to add context for *those* nodes in the next pass
 
for (i=0; i<b.length; ++i) {
 
if ( bb[i] == 1 ) {
 
for (j=Math.max(0,i-context); j < Math.min(b.length, i+context); ++j) {
 
if ( !bb[j] ) { bb[j] = 1; aa[ b[j].row ] = 0.5; }
 
}
 
}
 
 
}
 
}
 +
return ret;
 +
}
  
for (i=0; i<a.length; ++i) {
+
function navlinkStringToArray(s, article, params) {
if ( aa[i] == 1 ) {
+
s=expandConditionalNavlinkString(s,article,params);
for (j=Math.max(0,i-context); j < Math.min(a.length, i+context); ++j) {
+
var splitted=s.parenSplit(RegExp('<<(.*?)>>'));
if ( !aa[j] ) { aa[j] = 1; bb[ a[j].row ] = 0.5; }
+
var ret=[];
 +
for (var i=0; i<splitted.length; ++i) {
 +
if (i%2) { // i odd, so s is a tag
 +
var t=new navlinkTag();
 +
var ss=splitted[i].split('|');
 +
t.id=ss[0];
 +
for (var j=1; j<ss.length; ++j) {
 +
var sss=ss[j].split('=');
 +
if (sss.length>1) {
 +
t[sss[0]]=sss[1];
 +
}
 +
else { // no assignment (no "="), so treat this as a title (overwriting the last one)
 +
t.text=popupString(sss[0]);
 +
}
 
}
 
}
 +
t.article=article;
 +
var oldid=params.oldid, rcid=params.rcid, diff=params.diff;
 +
if (typeof oldid != 'undefined' && oldid != null) { t.oldid=oldid; }
 +
if (typeof rcid != 'undefined' && rcid != null) { t.rcid=rcid; }
 +
if (typeof diff != 'undefined' && diff != null) { t.diff=diff; }
 +
if (!t.text && t.id != 'mainlink') { t.text=popupString(t.id); }
 +
ret.push(t);
 
}
 
}
}
+
else { // plain HTML
 
+
ret.push(splitted[i]);
for (i=0; i<bb.length; ++i) {
 
if (bb[i] > 0) { // it's a row we need
 
if (b[i].paired) { bbb.push(b[i].text); } // joined; partner should be in aa
 
else {
 
bbb.push(b[i]);
 
}
 
 
}
 
}
 
}
 
}
for (i=0; i<aa.length; ++i) {
+
return ret;
if (aa[i] > 0) { // it's a row we need
 
if (a[i].paired) { aaa.push(a[i].text); } // joined; partner should be in aa
 
else {
 
aaa.push(a[i]);
 
}
 
}
 
}
 
 
 
return { a: aaa, b: bbb};
 
 
}
 
}
  
function stripOuterCommonLines(a,b,context) {
 
var i=0;
 
while (i<a.length && i < b.length && a[i]==b[i]) { ++i; }
 
var j=a.length-1; var k=b.length-1;
 
while ( j>=0 && k>=0 && a[j]==b[k] ) { --j; --k; }
 
  
return { a: a.slice(Math.max(0,i - 1 - context), Math.min(a.length+1, j + context+1)),
+
function navlinkSubstituteHTML(s) {
b: b.slice(Math.max(0,i - 1 - context), Math.min(b.length+1, k + context+1)) };
+
return s.split('*').join(getValueOf('popupNavLinkSeparator'))
}
+
.split('<menurow>').join('<li class="popup_menu_row">')
 +
.split('</menurow>').join('</li>')
 +
.split('<menu>').join('<ul class="popup_menu">')
 +
.split('</menu>').join('</ul>');
  
function insertDiff(navpop) {
+
}
// for speed reasons, we first do a line-based diff, discard stuff that seems boring, then do a word-based diff
+
 
// FIXME: sometimes this gives misleading diffs as distant chunks are squashed together
+
function navlinkDepth(magic,s) {
var oldlines = navpop.diffData.oldRev.revision.content.split('\n');
+
return s.split('<' + magic + '>').length - s.split('</' + magic + '>').length;
var newlines = navpop.diffData.newRev.revision.content.split('\n');
+
}
var inner=stripOuterCommonLines(oldlines,newlines,getValueOf('popupDiffContextLines'));
 
oldlines=inner.a; newlines=inner.b;
 
var truncated=false;
 
getValueOf('popupDiffMaxLines');
 
if (oldlines.length > pg.option.popupDiffMaxLines || newlines.length > pg.option.popupDiffMaxLines) {
 
// truncate
 
truncated=true;
 
inner=stripOuterCommonLines(oldlines.slice(0,pg.option.popupDiffMaxLines),
 
newlines.slice(0,pg.option.popupDiffMaxLines),
 
pg.option.popupDiffContextLines);
 
oldlines=inner.a; newlines=inner.b;
 
}
 
  
var lineDiff=diff(oldlines, newlines);
 
var lines2=rmBoringLines(lineDiff.o, lineDiff.n);
 
var oldlines2=lines2.a; var newlines2=lines2.b;
 
  
var simpleSplit = !String.prototype.parenSplit.isNative;
+
// navlinkString: * becomes the separator
var html='<hr />';
+
// <<foo|bar=baz|fubar>> becomes a foo-link with attribute bar='baz'
if (getValueOf('popupDiffDates')) {
+
//   and visible text 'fubar'
html += diffDatesTable(navpop);
+
// if(test){...} and if(test){...}else{...} work too (nested ok)
html += '<hr />';
 
}
 
html += shortenDiffString(
 
diffString(oldlines2.join('\n'), newlines2.join('\n'), simpleSplit),
 
getValueOf('popupDiffContextCharacters') ).join('<hr />');
 
setPopupTipsAndHTML(html.split('\n').join('<br>') +
 
(truncated ? '<hr /><b>'+popupString('Diff truncated for performance reasons')+'</b>' : '') ,
 
'popupPreview', navpop.idNumber);
 
}
 
  
function diffDatesTable( navpop ) {
+
function navlinkStringToHTML(s,article,params) {
var html='<table class="popup_diff_dates">';
+
//limitAlert(navlinkStringToHTML, 5, 'navlinkStringToHTML\n' + article + '\n' + (typeof article));
html += diffDatesTableRow( navpop.diffData.newRev.revision, tprintf('New revision'));
+
var p=navlinkStringToArray(s,article,params);
html += diffDatesTableRow( navpop.diffData.oldRev.revision, tprintf('Old revision'));
+
var html='';
html += '</table>';
+
var menudepth = 0; // nested menus not currently allowed, but doesn't do any harm to code for it
return html;
+
var menurowdepth = 0;
}
+
var wrapping = null;
function diffDatesTableRow( revision, label ) {
+
for (var i=0; i<p.length; ++i) {
var txt='';
+
if (typeof p[i] == typeof '') {
var lastModifiedDate = new Date(revision.timestamp);
+
html+=navlinkSubstituteHTML(p[i]);
var datePrint=getValueOf('popupDiffDatePrinter');
+
menudepth += navlinkDepth('menu', p[i]);
if (typeof lastModifiedDate[datePrint] == 'function') {
+
menurowdepth += navlinkDepth('menurow', p[i]);
var d2 = adjustDate(lastModifiedDate, getTimeOffset());
+
// if (menudepth === 0) {
txt = dayFormat(d2, true) + ' ' + timeFormat(d2, true);
+
// tagType='span';
} else {
+
// } else if (menurowdepth === 0) {
txt = tprintf('Invalid %s %s', ['popupDiffDatePrinter', datePrint]);
+
// tagType='li';
}
+
// } else {
var revlink = generalLink({url: mw.config.get('wgScript') + '?oldid='+revision.revid,
+
// tagType = null;
  text: label, title: label});
+
// }
return simplePrintf('<tr><td>%s</td><td>%s</td></tr>', [ revlink, txt ]);
+
} else if (typeof p[i].type != 'undefined' && p[i].type=='navlinkTag') {
 +
if (menudepth > 0 && menurowdepth === 0) {
 +
html += '<li class="popup_menu_item">' + p[i].html() + '</li>';
 +
} else {
 +
html+=p[i].html();
 +
}
 +
}
 +
}
 +
return html;
 
}
 
}
//</NOLITE>
 
// ENDFILE: diffpreview.js
 
// STARTFILE: links.js
 
//<NOLITE>
 
/////////////////////
 
// LINK GENERATION //
 
/////////////////////
 
  
// titledDiffLink --> titledWikiLink --> generalLink
+
function navlinkTag() {
// wikiLink   --> titledWikiLink --> generalLink
+
this.type='navlinkTag';
// editCounterLink --> generalLink
 
 
 
// TODO Make these functions return Element objects, not just raw HTML strings.
 
 
 
function titledDiffLink(l) { // article, text, title, from, to) {
 
return titledWikiLink({article: l.article, action: l.to + '&oldid=' + l.from,
 
newWin: l.newWin,
 
noPopup: l.noPopup,
 
text: l.text, title: l.title,
 
/* hack: no oldid here */
 
actionName: 'diff'});
 
 
}
 
}
  
 +
navlinkTag.prototype.html=function () {
 +
this.getNewWin();
 +
this.getPrintFunction();
 +
var html='';
 +
var opening, closing;
 +
var tagType='span';
 +
if (!tagType) {
 +
opening = ''; closing = '';
 +
} else {
 +
opening = '<' + tagType + ' class="popup_' + this.id + '">';
 +
closing = '</' + tagType + '>';
 +
}
 +
if (typeof this.print!='function') {
 +
errlog ('Oh dear - invalid print function for a navlinkTag, id='+this.id);
 +
} else {
 +
html=this.print(this);
 +
if (typeof html != typeof '') {html='';}
 +
else if (typeof this.shortcut!='undefined') html=addPopupShortcut(html, this.shortcut);
 +
}
 +
return opening + html + closing;
 +
};
  
function wikiLink(l) {
+
navlinkTag.prototype.getNewWin=function() {
//{article:article, action:action, text:text, oldid, newid}) {
+
getValueOf('popupLinksNewWindow');
if (! (typeof l.article == typeof {} &&
+
if (typeof pg.option.popupLinksNewWindow[this.id] === 'undefined') { this.newWin=null; }
typeof l.action == typeof '' &&
+
this.newWin=pg.option.popupLinksNewWindow[this.id];
typeof l.text==typeof '')) return null;
+
}
if (typeof l.oldid == 'undefined') { l.oldid=null; }
+
 
var savedOldid = l.oldid;
+
navlinkTag.prototype.getPrintFunction=function() { //think about this some more
if (!/^(edit|view|revert|render)$|^raw/.test(l.action)) { l.oldid=null; }
+
// this.id and this.article should already be defined
var hint=popupString(l.action + 'Hint'); // revertHint etc etc etc
+
if (typeof this.id!=typeof '' || typeof this.article!=typeof {} ) { return; }
var oldidData=[l.oldid, safeDecodeURI(l.article)];
+
var html='';
var revisionString = tprintf('revision %s of %s', oldidData);
+
var a,t;
log('revisionString='+revisionString);
+
 
switch (l.action) {
+
this.noPopup=1;
case 'edit&section=new': hint = popupString('newSectionHint');  break;
+
switch (this.id) {
case 'edit&undo=':
+
case 'contribs': case 'history': case 'whatLinksHere':
if (l.diff && l.diff != 'prev' && savedOldid ) {
+
case 'userPage': case 'monobook': case 'userTalk':
  l.action += l.diff + '&undoafter=' + savedOldid;
+
case 'talk': case 'article': case 'lastEdit':
} else if (savedOldid) {
+
this.noPopup=null;
  l.action += savedOldid;
+
}
}
+
switch (this.id) {
hint = popupString('undoHint');
+
case 'email': case 'contribs':  case 'block': case 'unblock':
break;
+
case 'userlog':   case 'userSpace': case 'deletedContribs':
case 'raw&ctype=text/css': hint=popupString('rawHint'); break;
+
this.article=this.article.userName();
case 'revert':
 
var p=parseParams(pg.current.link.href);
 
l.action='edit&autoclick=wpSave&actoken=' + autoClickToken() + '&autoimpl=' + popupString('autoedit_version') + '&autosummary=' + revertSummary(l.oldid, p.diff);
 
if (p.diff=='prev') {
 
l.action += '&direction=prev';
 
revisionString = tprintf('the revision prior to revision %s of %s', oldidData);
 
}
 
if (getValueOf('popupRevertSummaryPrompt')) { l.action += '&autosummaryprompt=true'; }
 
if (getValueOf('popupMinorReverts')) { l.action += '&autominor=true'; }
 
log('revisionString is now '+revisionString);
 
break;
 
case 'nullEdit':
 
l.action='edit&autoclick=wpSave&actoken=' + autoClickToken() + '&autoimpl=' + popupString('autoedit_version') + '&autosummary=null';
 
break;
 
case 'historyfeed':
 
l.action='history&feed=rss';
 
break;
 
case 'markpatrolled':
 
l.action='markpatrolled&rcid='+l.rcid;
 
 
}
 
}
  
if (hint) {
+
switch (this.id) {
if (l.oldid) {
+
case 'userTalk': case 'newUserTalk': case 'editUserTalk':
hint = simplePrintf(hint, [revisionString]);
+
case 'userPage': case 'monobook': case 'editMonobook': case 'blocklog':
}
+
this.article=this.article.userName(true);
else {
+
// fall through; no break
hint = simplePrintf(hint, [safeDecodeURI(l.article)]);
+
case 'pagelog': case 'deletelog': case 'protectlog':
}
+
delete this.oldid;
}
 
else {
 
hint = safeDecodeURI(l.article + '&action=' + l.action) + (l.oldid) ? '&oldid='+l.oldid : '';
 
 
}
 
}
  
return titledWikiLink({article: l.article, action: l.action, text: l.text, newWin:l.newWin,
+
if (this.id=='editMonobook' || this.id=='monobook') { this.article.append('/monobook.js'); }
title: hint, oldid: l.oldid, noPopup: l.noPopup, onclick: l.onclick});
 
}
 
  
function revertSummary(oldid, diff) {
+
if (this.id != 'mainlink') {
var ret='';
+
// FIXME anchor handling should be done differently with Title object
if (diff == 'prev') {
+
this.article=this.article.removeAnchor();
ret=getValueOf('popupQueriedRevertToPreviousSummary');
+
// if (typeof this.text=='undefined') this.text=popupString(this.id);
} else { ret = getValueOf('popupQueriedRevertSummary'); }
+
}
return ret + '&autorv=' + oldid;
 
}
 
  
function titledWikiLink(l) {
+
switch (this.id) {
// possible properties of argument:
+
case 'undelete':      this.print=specialLink; this.specialpage='Undelete'; this.sep='/'; break;
// article, action, text, title, oldid, actionName, className, noPopup
+
case 'whatLinksHere':  this.print=specialLink; this.specialpage='Whatlinkshere'; break;
// oldid = null is fine here
+
case 'relatedChanges': this.print=specialLink; this.specialpage='Recentchangeslinked'; break;
 +
case 'move':          this.print=specialLink; this.specialpage='Movepage'; break;
 +
case 'contribs':      this.print=specialLink; this.specialpage='Contributions'; break;
 +
case 'deletedContribs':this.print=specialLink; this.specialpage='Deletedcontributions'; break;
 +
case 'email':          this.print=specialLink; this.specialpage='EmailUser'; this.sep='/'; break;
 +
case 'block':          this.print=specialLink; this.specialpage='Blockip'; this.sep='&ip='; break;
 +
case 'unblock':        this.print=specialLink; this.specialpage='Ipblocklist'; this.sep='&action=unblock&ip='; break;
 +
case 'userlog':        this.print=specialLink; this.specialpage='Log'; this.sep='&user='; break;
 +
case 'blocklog':      this.print=specialLink; this.specialpage='Log'; this.sep='&type=block&page='; break;
 +
case 'pagelog':        this.print=specialLink; this.specialpage='Log'; this.sep='&page='; break;
 +
case 'protectlog':    this.print=specialLink; this.specialpage='Log'; this.sep='&type=protect&page='; break;
 +
case 'deletelog':      this.print=specialLink; this.specialpage='Log'; this.sep='&type=delete&page='; break;
 +
case 'userSpace':      this.print=specialLink; this.specialpage='PrefixIndex'; this.sep='&namespace=2&prefix='; break;
 +
case 'search':         this.print=specialLink; this.specialpage='Search'; this.sep='&fulltext=Search&search='; break;
 +
case 'unwatch': case 'watch':
 +
this.print=magicWatchLink; this.action=this.id+'&autowatchlist=1&autoimpl=' + popupString('autoedit_version') + '&actoken='+autoClickToken(); break;
 +
case 'history': case 'historyfeed':
 +
case 'unprotect': case 'protect':
 +
this.print=wikiLink; this.action=this.id; break;
  
// article and action are mandatory args
+
case 'delete':
 
+
this.print=wikiLink; this.action='delete';
if (typeof l.article == 'undefined' || typeof l.action=='undefined') {
+
if (this.article.namespaceId()==pg.nsImageId) {
errlog('got undefined article or action in titledWikiLink');
+
var img=this.article.stripNamespace();
return null;
+
this.action+='&image='+img;
 
}
 
}
 +
break;
  
var base = pg.wiki.titlebase +  l.article.urlString();
+
case 'markpatrolled':
var url=base;
+
case 'edit': // editOld should keep the oldid, but edit should not.
 
+
delete this.oldid; // fall through
if (typeof l.actionName=='undefined' || !l.actionName) { l.actionName='action'; }
+
case 'view': case 'purge': case 'render':
 
+
this.print=wikiLink;
// no need to add &action=view, and this confuses anchors
+
this.action=this.id; break;
if (l.action != 'view') { url = base + '&' + l.actionName + '=' + l.action; }
+
case 'raw':
 
+
this.print=wikiLink; this.action='raw&ctype=text/css'; break;
if (typeof l.oldid!='undefined' && l.oldid) { url+='&oldid='+l.oldid; }
+
case 'new':
 
+
this.print=wikiLink; this.action='edit&section=new'; break;
var cssClass=pg.misc.defaultNavlinkClassname;
+
case 'mainlink':
if (typeof l.className!='undefined' && l.className) { cssClass=l.className; }
+
if (typeof this.text=='undefined') { this.text=this.article.toString().entify(); }
 
+
if (getValueOf('popupSimplifyMainLink') && isInStrippableNamespace(this.article)) {
return generalNavLink({url: url, newWin: l.newWin,
+
var s=this.text.split('/'); this.text=s[s.length-1];
title: (typeof l.title != 'undefined') ? l.title : null,
+
if (this.text=='' && s.length > 1) { this.text=s[s.length-2]; }
text: (typeof l.text!='undefined')?l.text:null,
 
className: cssClass, noPopup:l.noPopup, onclick:l.onclick});
 
}
 
 
 
pg.fn.getLastContrib = function getLastContrib(wikipage, newWin) {
 
getHistoryInfo(wikipage, function(x) {
 
processLastContribInfo(x, {page: wikipage, newWin: newWin});
 
});
 
};
 
 
 
function processLastContribInfo(info, stuff) {
 
if(!info.edits || !info.edits.length) { alert('Popups: an odd thing happened. Please retry.'); return; }
 
if(!info.firstNewEditor) {
 
alert(tprintf('Only found one editor: %s made %s edits', [info.edits[0].editor,info.edits.length]));
 
return;
 
 
}
 
}
var newUrl=pg.wiki.titlebase + new Title(stuff.page).urlString() + '&diff=cur&oldid='+info.firstNewEditor.oldid;
+
this.print=titledWikiLink;
displayUrl(newUrl, stuff.newWin);
+
if (typeof this.title=='undefined' && pg.current.link && typeof pg.current.link.href != 'undefined') {
}
+
this.title=safeDecodeURI((pg.current.link.originalTitle)?pg.current.link.originalTitle:this.article);
 
+
if (typeof this.oldid != 'undefined' && this.oldid) {
pg.fn.getDiffSinceMyEdit = function getDiffSinceMyEdit(wikipage, newWin) {
+
this.title=tprintf('Revision %s of %s', [this.oldid, this.title]);
getHistoryInfo(wikipage, function(x){
+
}
processDiffSinceMyEdit(x, {page: wikipage, newWin: newWin});
 
});
 
};
 
 
 
function processDiffSinceMyEdit(info, stuff) {
 
if(!info.edits || !info.edits.length) { alert('Popups: something fishy happened. Please try again.'); return; }
 
var friendlyName=stuff.page.split('_').join(' ');
 
if(!info.myLastEdit) {
 
alert(tprintf('Couldn\'t find an edit by %s\nin the last %s edits to\n%s',
 
  [info.userName, getValueOf('popupHistoryLimit'), friendlyName]));
 
return;
 
 
}
 
}
if(info.myLastEdit.index === 0) {
+
this.action='view'; break;
alert(tprintf("%s seems to be the last editor to the page %s", [info.userName, friendlyName]));
+
case 'userPage':
return;
+
case 'article':
}
+
case 'monobook':
var newUrl=pg.wiki.titlebase + new Title(stuff.page).urlString() + '&diff=cur&oldid='+ info.myLastEdit.oldid;
+
case 'editMonobook':
displayUrl(newUrl, stuff.newWin);
+
case 'editArticle':
}
+
delete this.oldid;
 
+
//alert(this.id+'\n'+this.article + '\n'+ typeof this.article);
function displayUrl(url, newWin){
+
this.article=this.article.articleFromTalkOrArticle();
if(newWin) { window.open(url); }
+
//alert(this.id+'\n'+this.article + '\n'+ typeof this.article);
else { document.location=url; }
+
this.print=wikiLink;
}
+
if (this.id.indexOf('edit')==0) {
 
+
this.action='edit';
pg.fn.purgePopups = function purgePopups() {
+
} else { this.action='view';}
processAllPopups(true);
+
break;
setupCache(); // deletes all cached items (not browser cached, though...)
+
case 'userTalk':
pg.option={};
+
case 'talk':
abortAllDownloads();
+
this.article=this.article.talkPage();
};
+
delete this.oldid;
 
+
this.print=wikiLink;
function processAllPopups(nullify, banish) {
+
this.action='view'; break;
for (var i=0; pg.current.links && i<pg.current.links.length; ++i) {
+
case 'arin':
if (!pg.current.links[i].navpopup) { continue; }
+
this.print=arinLink; break;
if (nullify || banish) pg.current.links[i].navpopup.banish();
+
case 'count':
pg.current.links[i].simpleNoMore=false;
+
this.print=editCounterLink; break;
if (nullify) pg.current.links[i].navpopup=null;
+
case 'google':
 +
this.print=googleLink; break;
 +
case 'editors':
 +
this.print=editorListLink; break;
 +
case 'globalsearch':
 +
this.print=globalSearchLink; break;
 +
case 'lastEdit':
 +
this.print=titledDiffLink;
 +
this.title=popupString('Show the last edit');
 +
this.from='prev'; this.to='cur'; break;
 +
case 'oldEdit':
 +
this.print=titledDiffLink;
 +
this.title=popupString('Show the edit made to get revision') + ' ' + this.oldid;
 +
this.from='prev'; this.to=this.oldid; break;
 +
case 'editOld':
 +
this.print=wikiLink; this.action='edit'; break;
 +
case 'undo':
 +
this.print=wikiLink; this.action='edit&undo='; break;
 +
case 'markpatrolled':
 +
this.print=wikiLink; this.action='markpatrolled';
 +
case 'revert':
 +
this.print=wikiLink; this.action='revert'; break;
 +
case 'nullEdit':
 +
this.print=wikiLink; this.action='nullEdit'; break;
 +
case 'diffCur':
 +
this.print=titledDiffLink;
 +
this.title=tprintf('Show changes since revision %s', [this.oldid]);
 +
this.from=this.oldid; this.to='cur'; break;
 +
case 'editUserTalk':
 +
case 'editTalk':
 +
delete this.oldid;
 +
this.article=this.article.talkPage();
 +
this.action='edit'; this.print=wikiLink; break;
 +
case 'newUserTalk':
 +
case 'newTalk':
 +
this.article=this.article.talkPage();
 +
this.action='edit&section=new'; this.print=wikiLink; break;
 +
case 'lastContrib':
 +
case 'sinceMe':
 +
this.print=magicHistoryLink;
 +
break;
 +
case 'togglePreviews':
 +
this.text=popupString(pg.option.simplePopups ? 'enable previews' : 'disable previews');
 +
case 'disablePopups': case 'purgePopups':
 +
this.print=popupMenuLink;
 +
break;
 +
default:
 +
this.print=function () {return 'Unknown navlink type: '+this.id+''};
 
}
 
}
}
 
 
pg.fn.disablePopups = function disablePopups(){
 
processAllPopups(false, true);
 
setupTooltips(null, true);
 
};
 
 
pg.fn.togglePreviews = function togglePreviews() {
 
processAllPopups(true, true);
 
pg.option.simplePopups=!pg.option.simplePopups;
 
abortAllDownloads();
 
 
};
 
};
 +
//
 +
//  end navlinks
 +
//////////////////////////////////////////////////
 +
//</NOLITE>
 +
// ENDFILE: navlinks.js
 +
// STARTFILE: shortcutkeys.js
 +
//<NOLITE>
 +
function popupHandleKeypress(evt) {
 +
var keyCode = window.event ? window.event.keyCode : ( evt.keyCode ? evt.keyCode : evt.which);
 +
if (!keyCode || !pg.current.link || !pg.current.link.navpopup) { return; }
 +
if (keyCode==27) { // escape
 +
killPopup();
 +
return false; // swallow keypress
 +
}
  
function magicWatchLink(l) {
+
var letter=String.fromCharCode(keyCode);
//Yuck!! Would require a thorough redesign to add this as a click event though ...
+
var links=pg.current.link.navpopup.mainDiv.getElementsByTagName('A');
l.onclick = simplePrintf( 'pg.fn.modifyWatchlist(\'%s\',\'%s\');return false;', [l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), this.id] );
+
var startLink=0;
return wikiLink(l);
+
var i,j;
}
 
  
pg.fn.modifyWatchlist = function modifyWatchlist(title, action) {
+
if (popupHandleKeypress.lastPopupLinkSelected) {
var reqData = {
+
for (i=0; i<links.length; ++i) {
'action': 'watch',
+
if (links[i]==popupHandleKeypress.lastPopupLinkSelected) { startLink=i; }
'formatversion': 2,
+
}
'titles': title,
+
}
'uselang': mw.config.get('wgUserLanguage')
+
for (j=0; j<links.length; ++j) {
};
+
i=(startLink + j + 1) % links.length;
if ( action === 'unwatch' ) reqData.unwatch = true;
+
if (links[i].getAttribute('popupkey')==letter) {
 
+
if (evt && evt.preventDefault) evt.preventDefault();
var api = new mw.Api( {
+
links[i].focus();
    ajax: {
+
popupHandleKeypress.lastPopupLinkSelected=links[i];
        headers: { 'Api-User-Agent': pg.misc.userAgent }
+
return false; // swallow keypress
    }
+
}
} );
 
// Load the Addedwatchtext or Removedwatchtext message and show it
 
var mwTitle = mw.Title.newFromText( title );
 
var messageName;
 
if ( mwTitle && mwTitle.getNamespaceId() > 0 && mwTitle.getNamespaceId() % 2 === 1 ) {
 
messageName = action === 'watch' ? 'addedwatchtext-talk' : 'removedwatchtext-talk';
 
} else {
 
messageName = action === 'watch' ? 'addedwatchtext' : 'removedwatchtext';
 
 
}
 
}
$.when(
 
api.postWithToken( 'watch', reqData ),
 
mw.loader.using( [ 'mediawiki.api', 'mediawiki.jqueryMsg' ] ).then( function () {
 
return api.loadMessagesIfMissing( [ messageName ] );
 
} )
 
).done( function () {
 
mw.notify( mw.message( messageName, title ).parseDom() );
 
} );
 
};
 
  
function magicHistoryLink(l) {
+
// pass keypress on
// FIXME use onclick change href trick to sort this out instead of window.open
+
if (document.oldPopupOnkeypress) { return document.oldPopupOnkeypress(evt); }
 +
return true;
 +
}
  
var jsUrl='', title='', onClick='';
+
function addPopupShortcuts() {
switch(l.id) {
+
if (document.onkeypress!=popupHandleKeypress) {
case 'lastContrib':
+
document.oldPopupOnkeypress=document.onkeypress;
onClick=simplePrintf('pg.fn.getLastContrib(\'%s\',%s)',
 
[l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), l.newWin]);
 
title=popupString('lastContribHint');
 
break;
 
case 'sinceMe':
 
onClick=simplePrintf('pg.fn.getDiffSinceMyEdit(\'%s\',%s)',
 
[l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), l.newWin]);
 
title=popupString('sinceMeHint');
 
break;
 
 
}
 
}
jsUrl = 'javascript:' + onClick; // jshint ignore:line
+
document.onkeypress=popupHandleKeypress;
onClick += ';return false;';
 
 
 
return generalNavLink({url: jsUrl, newWin: false, // can't have new windows with JS links, I think
 
title: title, text: l.text, noPopup: l.noPopup, onclick: onClick });
 
 
}
 
}
  
function popupMenuLink(l) {
+
function rmPopupShortcuts() {
var jsUrl=simplePrintf('javascript:pg.fn.%s()', [l.id]); // jshint ignore:line
+
popupHandleKeypress.lastPopupLinkSelected=null;
var title=popupString(simplePrintf('%sHint', [l.id]));
+
try {
var onClick=simplePrintf('pg.fn.%s();return false;', [l.id]);
+
if (document.oldPopupOnkeypress && document.oldPopupOnkeypress==popupHandleKeypress) {
return generalNavLink({url: jsUrl, newWin:false, title:title, text:l.text, noPopup:l.noPopup, onclick: onClick});
+
// panic
}
+
document.onkeypress=null; //function () {};
 
+
return;
function specialLink(l) {
 
// properties: article, specialpage, text, sep
 
if (typeof l.specialpage=='undefined'||!l.specialpage) return null;
 
var base = pg.wiki.titlebase +  mw.config.get('wgFormattedNamespaces')[pg.nsSpecialId]+':'+l.specialpage;
 
if (typeof l.sep == 'undefined' || l.sep === null) l.sep='&target=';
 
var article=l.article.urlString({keepSpaces: l.specialpage=='Search'});
 
var hint=popupString(l.specialpage+'Hint');
 
switch (l.specialpage) {
 
case 'Log':
 
switch (l.sep) {
 
case '&user=': hint=popupString('userLogHint'); break;
 
case '&type=block&page=': hint=popupString('blockLogHint'); break;
 
case '&page=': hint=popupString('pageLogHint'); break;
 
case '&type=protect&page=': hint=popupString('protectLogHint'); break;
 
case '&type=delete&page=': hint=popupString('deleteLogHint'); break;
 
default: log('Unknown log type, sep=' + l.sep); hint='Missing hint (FIXME)';
 
 
}
 
}
break;
+
document.onkeypress=document.oldPopupOnkeypress;
case 'PrefixIndex': article += '/'; break;
+
} catch (nasties) { /* IE goes here */ }
}
 
if (hint) hint = simplePrintf(hint, [safeDecodeURI(l.article)]);
 
else hint = safeDecodeURI(l.specialpage+':'+l.article) ;
 
 
 
var url = base + l.sep + article;
 
return generalNavLink({url: url, title: hint, text: l.text, newWin:l.newWin, noPopup:l.noPopup});
 
 
}
 
}
  
function generalLink(l) {
 
// l.url, l.text, l.title, l.newWin, l.className, l.noPopup, l.onclick
 
if (typeof l.url=='undefined') return null;
 
  
// only quotation marks in the url can screw us up now... I think
+
function addLinkProperty(html, property) {
var url=l.url.split('"').join('%22');
+
// take "<a href=...>...</a> and add a property
 +
// not sophisticated at all, easily broken
 +
var i=html.indexOf('>');
 +
if (i<0) { return html; }
 +
return html.substring(0,i) + ' ' + property + html.substring(i);
 +
}
  
var ret='<a href="' + url + '"';
+
function addPopupShortcut(html, key) {
if (typeof l.title!='undefined' && l.title) { ret += ' title="' + pg.escapeQuotesHTML(l.title) + '"'; }
+
if (!getValueOf('popupShortcutKeys')) { return html; }
if (typeof l.onclick!='undefined' && l.onclick) { ret += ' onclick="' + pg.escapeQuotesHTML(l.onclick) + '"'; }
+
var ret= addLinkProperty(html, 'popupkey="'+key+'"');
if (l.noPopup) { ret += ' noPopup=1'; }
+
if (key==' ') { key=popupString('spacebar'); }
var newWin;
+
return ret.replace(RegExp('^(.*?)(title=")(.*?)(".*)$', 'i'),'$1$2$3 ['+key+']$4');
if (typeof l.newWin=='undefined' || l.newWin === null) { newWin=getValueOf('popupNewWindows'); }
 
else { newWin=l.newWin; }
 
if (newWin) { ret += ' target="_blank"'; }
 
if (typeof l.className!='undefined'&&l.className) { ret+=' class="'+l.className+'"'; }
 
ret += '>';
 
if (typeof l.text==typeof '') { ret+= l.text; }
 
ret +='</a>';
 
return ret;
 
 
}
 
}
 +
//</NOLITE>
 +
// ENDFILE: shortcutkeys.js
 +
// STARTFILE: diffpreview.js
 +
//<NOLITE>
 +
function loadDiff(article, oldid, diff, navpop) {
 +
navpop.diffData={};
 +
var oldRev, newRev;
 +
switch (diff) {
 +
case 'cur':
 +
switch ( oldid ) {
 +
case null:
 +
case '':
 +
case 'prev':
 +
// eg newmessages diff link
 +
oldRev='0&direction=prev';
 +
newRev=0;
 +
break;
 +
default:
 +
oldRev = oldid;
 +
newRev = ( oldid || 0 ) + '&direction=cur';
 +
}
 +
break;
 +
case 'prev':
 +
oldRev = ( oldid || 0 ) + '&direction=prev'; newRev = oldid; break;
 +
case 'next':
 +
oldRev = oldid; newRev = oldid + '&direction=next';
 +
break;
 +
default:
 +
oldRev = oldid || 0; newRev = diff || 0; break;
 +
}
 +
oldRev = oldRev || 0;
 +
newRev = newRev || 0;
  
function appendParamsToLink(linkstr, params) {
+
var go = function() {
var sp=linkstr.parenSplit(RegExp('(href="[^"]+?)"', 'i'));
+
pendingNavpopTask(navpop);
if (sp.length<2) return null;
+
getWiki(article, doneDiffNew, newRev, navpop);
var ret=sp.shift() + sp.shift();
+
 
ret += '&' + params + '"';
+
pendingNavpopTask(navpop);
ret += sp.join('');
+
getWiki(article, doneDiffOld, oldRev, navpop);
return ret;
+
 
 +
var tz = Cookie.read('popTz');
 +
if ( mw.config.get('wgEnableAPI') && getValueOf('popupAdjustDiffDates') && tz===null) {
 +
pendingNavpopTask(navpop);
 +
getPageWithCaching(pg.wiki.apiwikibase + '?format=json&action=query&meta=userinfo&uiprop=options',
 +
  function(d) {
 +
  completedNavpopTask(navpop);
 +
  setTimecorrectionCookie(d);
 +
  if (diffDownloadsComplete(navpop)) { insertDiff(navpop); }
 +
  },  navpop);
 +
}
 +
return true; // remove hook once run
 +
}
 +
if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
 +
else { navpop.addHook(go, 'unhide', 'before', 'DOWNLOAD_DIFFS'); }
 
}
 
}
  
function changeLinkTargetLink(x) { // newTarget, text, hint, summary, clickButton, minor, title (optional) {
+
function setTimecorrectionCookie(d) {
if (x.newTarget) {
+
try {
log ('changeLinkTargetLink: newTarget=' + x.newTarget);
+
var jsobj=getJsObj(d.data);
}
+
var tz=jsobj.query.userinfo.options.timecorrection;
if (x.oldTarget !== decodeURIComponent( x.oldTarget ) ) {
+
} catch (someError) {
log ('This might be an input problem: ' + x.oldTarget );
+
logerr( 'setTimecorretion failed' );
 +
return;
 
}
 
}
 +
Cookie.create( 'popTz', getTimeOffset(tz), 1);
 +
}
 +
 +
function doneDiff(download, isOld) {
 +
if (!download.owner || !download.owner.diffData) { return; }
 +
var navpop=download.owner;
 +
var label= (isOld) ? 'Old' : 'New';
 +
var otherLabel=(isOld) ? 'New' : 'Old';
 +
navpop.diffData[label]=download;
 +
completedNavpopTask(download.owner);
 +
if (diffDownloadsComplete(navpop)) { insertDiff(navpop); }
 +
}
 +
 +
function diffDownloadsComplete(navpop) {
 +
if ( Cookie.read('popTz')===null) { return false; }
 +
return navpop.diffData.Old && navpop.diffData.New;
 +
}
 +
 +
function doneDiffNew(download) { doneDiff(download, false); }
 +
function doneDiffOld(download) { doneDiff(download, true);  }
  
// FIXME: first character of page title as well as namespace should be case insensitive
+
function rmBoringLines(a,b,context) {
// eg [[category:X1]] and [[Category:X1]] are equivalent
 
// this'll break if charAt(0) is nasty
 
var cA=literalizeRegex(x.oldTarget);
 
var chs=cA.charAt(0).toUpperCase();
 
chs='['+chs + chs.toLowerCase()+']';
 
var currentArticleRegexBit=chs+cA.substring(1);
 
currentArticleRegexBit=currentArticleRegexBit
 
.split(RegExp('(?:[_ ]+|%20)', 'g')).join('(?:[_ ]+|%20)')
 
.split('\\(').join('(?:%28|\\()')
 
.split('\\)').join('(?:%29|\\))'); // why does this need to match encoded strings ? links in the document ?
 
// leading and trailing space should be ignored, and anchor bits optional:
 
currentArticleRegexBit = '\\s*(' + currentArticleRegexBit + '(?:#[^\\[\\|]*)?)\\s*';
 
// e.g. Computer (archaic) -> \s*([Cc]omputer[_ ](?:%2528|\()archaic(?:%2528|\)))\s*
 
  
// autoedit=s~\[\[([Cc]ad)\]\]~[[Computer-aided%20design|$1]]~g;s~\[\[([Cc]AD)[|]~[[Computer-aided%20design|~g
+
if (typeof context == 'undefined') { context=2; }
 +
// this is fairly slow... i think it's quicker than doing a word-based diff from the off, though
 +
var aa=[], aaa=[];
 +
var bb=[], bbb=[];
 +
var i, j;
  
var title=x.title || mw.config.get('wgPageName').split('_').join(' ');
+
// first, gather all disconnected nodes in a and all crossing nodes in a and b
var lk=titledWikiLink({article: new Title(title), newWin:x.newWin,
+
for (i=0; i<a.length; ++i ) {
action:  'edit',
+
if(!a[i].paired) { aa[i]=1; }
text: x.text,
+
else if (countCrossings(b,a,i, true)) {
title:  x.hint,
+
aa[i]=1;
className: 'popup_change_title_link'
+
bb[ a[i].row ] = 1;
});
+
}
var cmd='';
 
if (x.newTarget) {
 
// escape '&' and other nasties
 
var t=x.newTarget;
 
var s=literalizeRegex(x.newTarget);
 
cmd += 's~\\[\\['+currentArticleRegexBit+'\\]\\]~[['+t+'|$1]]~g;';
 
cmd += 's~\\[\\['+currentArticleRegexBit+'[|]~[['+t+'|~g;';
 
cmd += 's~\\[\\['+s + '\\|' + s + '\\]\\]~[[' + t + ']]~g';
 
} else {
 
cmd += 's~\\[\\['+currentArticleRegexBit+'\\]\\]~$1~g;';
 
cmd += 's~\\[\\['+currentArticleRegexBit+'[|](.*?)\\]\\]~$2~g';
 
 
}
 
}
// Build query
 
cmd = 'autoedit=' + encodeURIComponent ( cmd );
 
cmd += '&autoclick='+ encodeURIComponent( x.clickButton ) + '&actoken=' + encodeURIComponent( autoClickToken() );
 
cmd += ( x.minor === null ) ? '' : '&autominor='+ encodeURIComponent( x.minor );
 
cmd += ( x.watch === null ) ? '' : '&autowatch='+ encodeURIComponent( x.watch );
 
cmd += '&autosummary='+encodeURIComponent(x.summary);
 
cmd += '&autoimpl='+encodeURIComponent( popupString('autoedit_version') );
 
return appendParamsToLink(lk, cmd);
 
}
 
  
 +
// pick up remaining disconnected nodes in b
 +
for (i=0; i<b.length; ++i ) {
 +
if (bb[i]==1) { continue; }
 +
if(!b[i].paired) { bb[i]=1; }
 +
}
  
function redirLink(redirMatch, article) {
+
// another pass to gather context: we want the neighbours of included nodes which are not yet included
// NB redirMatch is in wikiText
+
// we have to add in partners of these nodes, but we don't want to add context for *those* nodes in the next pass
var ret='';
+
for (i=0; i<b.length; ++i) {
 +
if ( bb[i] == 1 ) {
 +
for (j=max(0,i-context); j < min(b.length, i+context); ++j) {
 +
if ( !bb[j] ) { bb[j] = 1; aa[ b[j].row ] = 0.5; }
 +
}
 +
}
 +
}
  
if (getValueOf('popupAppendRedirNavLinks') && getValueOf('popupNavLinks')) {
+
for (i=0; i<a.length; ++i) {
ret += '<hr />';
+
if ( aa[i] == 1 ) {
if (getValueOf('popupFixRedirs') && typeof autoEdit != 'undefined' && autoEdit) {
+
for (j=max(0,i-context); j < min(a.length, i+context); ++j) {
log('redirLink: newTarget=' + redirMatch);
+
if ( !aa[j] ) { aa[j] = 1; bb[ a[j].row ] = 0.5; }
ret += addPopupShortcut(changeLinkTargetLink({
+
}
newTarget: redirMatch,
 
text: popupString('Redirects'),
 
hint: popupString('Fix this redirect'),
 
summary: simplePrintf(getValueOf('popupFixRedirsSummary'),[article.toString(), redirMatch]),
 
oldTarget: article.toString(),
 
clickButton: getValueOf('popupRedirAutoClick'),
 
minor: true,
 
watch: getValueOf('popupWatchRedirredPages')
 
}), 'R');
 
ret += popupString(' to ');
 
 
}
 
}
else ret += popupString('Redirects') + popupString(' to ');
 
return ret;
 
 
}
 
}
  
else return '<br> ' + popupString('Redirects') + popupString(' to ') +
+
for (i=0; i<bb.length; ++i) {
titledWikiLink({article: new Title().fromWikiText(redirMatch), action: 'view',  /* FIXME: newWin */
+
if (bb[i] > 0) { // it's a row we need
  text: safeDecodeURI(redirMatch), title: popupString('Bypass redirect')});
+
if (b[i].paired) { bbb.push(b[i].text); } // joined; partner should be in aa
}
+
else {
 +
bbb.push(b[i]);
 +
}
 +
}
 +
}
 +
for (i=0; i<aa.length; ++i) {
 +
if (aa[i] > 0) { // it's a row we need
 +
if (a[i].paired) { aaa.push(a[i].text); } // joined; partner should be in aa
 +
else {
 +
aaa.push(a[i]);
 +
}
 +
}
 +
}
  
function arinLink(l) {
+
return { a: aaa, b: bbb};
if (!saneLinkCheck(l)) { return null; }
 
if ( ! l.article.isIpUser() || ! pg.wiki.wikimedia) return null;
 
 
 
var uN=l.article.userName();
 
 
 
return generalNavLink({url:'http://ws.arin.net/cgi-bin/whois.pl?queryinput=' + encodeURIComponent(uN), newWin:l.newWin,
 
title: tprintf('Look up %s in ARIN whois database', [uN]),
 
text: l.text, noPopup:1});
 
 
}
 
}
  
function toolDbName(cookieStyle) {
+
function stripOuterCommonLines(a,b,context) {
var ret = mw.config.get('wgDBname');
+
var i=0;
if (!cookieStyle) { ret+= '_p'; }
+
while (i<a.length && i < b.length && a[i]==b[i]) { ++i; }
return ret;
+
var j=a.length-1; var k=b.length-1;
}
+
while ( j>=0 && k>=0 && a[j]==b[k] ) { --j; --k; }
  
function saneLinkCheck(l) {
+
return { a: a.slice(max(0,i - 1 - context), min(a.length+1, j + context+1)),
if (typeof l.article != typeof {} || typeof l.text != typeof '') { return false; }
+
b: b.slice(max(0,i - 1 - context), min(b.length+1, k + context+1)) };
return true;
 
 
}
 
}
function editCounterLink(l) {
 
if(!saneLinkCheck(l)) return null;
 
if (! pg.wiki.wikimedia) return null;
 
var uN=l.article.userName();
 
var tool=getValueOf('popupEditCounterTool');
 
var url;
 
var defaultToolUrl='//tools.wmflabs.org/supercount/index.php?user=$1&project=$2.$3';
 
  
switch(tool) {
+
function insertDiff(navpop) {
case 'custom':
+
// for speed reasons, we first do a line-based diff, discard stuff that seems boring, then do a word-based diff
url=simplePrintf(getValueOf('popupEditCounterUrl'), [ encodeURIComponent(uN), toolDbName() ]);
+
// FIXME: sometimes this gives misleading diffs as distant chunks are squashed together
break;
+
var oldlines=navpop.diffData.Old.data.split('\n');
case 'soxred':  // no longer available
+
var newlines=navpop.diffData.New.data.split('\n');
case 'kate':    // no longer available
+
var inner=stripOuterCommonLines(oldlines,newlines,getValueOf('popupDiffContextLines'));
case 'interiot':// no longer available
+
oldlines=inner.a; newlines=inner.b;
/* fall through */
+
var truncated=false;
case 'supercount':
+
getValueOf('popupDiffMaxLines');
default:
+
if (oldlines.length > pg.option.popupDiffMaxLines || newlines.length > pg.option.popupDiffMaxLines) {
var theWiki=pg.wiki.hostname.split('.');
+
// truncate
url=simplePrintf(defaultToolUrl, [ encodeURIComponent(uN), theWiki[0], theWiki[1] ]);
+
truncated=true;
 +
inner=stripOuterCommonLines(oldlines.slice(0,pg.option.popupDiffMaxLines),
 +
newlines.slice(0,pg.option.popupDiffMaxLines),
 +
pg.option.popupDiffContextLines);
 +
oldlines=inner.a; newlines=inner.b;
 
}
 
}
return generalNavLink({url:url, title: tprintf('editCounterLinkHint', [uN]),
 
newWin:l.newWin, text: l.text, noPopup:1});
 
}
 
  
 +
var lineDiff=diff(oldlines, newlines);
 +
var lines2=rmBoringLines(lineDiff.o, lineDiff.n);
 +
var oldlines2=lines2.a; var newlines2=lines2.b;
  
function globalSearchLink(l) {
+
var simpleSplit = !String.prototype.parenSplit.isNative;
if(!saneLinkCheck(l)) return null;
+
var html='<hr />';
 
+
if (getValueOf('popupDiffDates')) {
var base='http://vs.aka-online.de/cgi-bin/globalwpsearch.pl?timeout=120&search=';
+
html += diffDatesTable(navpop.diffData.Old, navpop.diffData.New);
var article=l.article.urlString({keepSpaces:true});
+
html += '<hr />';
 
+
}
return generalNavLink({url:base + article, newWin:l.newWin,
+
html += shortenDiffString(
title: tprintf('globalSearchHint', [safeDecodeURI(l.article)]),
+
diffString(oldlines2.join('\n'), newlines2.join('\n'), simpleSplit),
text: l.text, noPopup:1});
+
getValueOf('popupDiffContextCharacters') ).join('<hr />');
 +
setPopupTipsAndHTML(html.split('\n').join('<br>') +
 +
(truncated ? '<hr /><b>'+popupString('Diff truncated for performance reasons')+'</b>' : '') ,
 +
'popupPreview', navpop.idNumber);
 
}
 
}
  
function googleLink(l) {
+
function diffDatesTable( oldDl, newDl ) {
if(!saneLinkCheck(l)) return null;
+
var html='<table class="popup_diff_dates">';
 
+
html += diffDatesTableRow( newDl, tprintf('New revision'));
var base='https://www.google.com/search?q=';
+
html += diffDatesTableRow( oldDl, tprintf('Old revision'));
var article=l.article.urlString({keepSpaces:true});
+
html += '</table>';
 
+
return html;
return generalNavLink({url:base + '%22' + article + '%22', newWin:l.newWin,
 
title: tprintf('googleSearchHint', [safeDecodeURI(l.article)]),
 
text: l.text, noPopup:1});
 
 
}
 
}
 
+
function diffDatesTableRow( dl, label ) {
function editorListLink(l) {
+
var txt='';
if(!saneLinkCheck(l)) return null;
+
if (!dl) {
var article= l.article.articleFromTalkPage() || l.article;
+
txt=popupString('Something went wrong :-(');
var url='https://xtools.wmflabs.org/articleinfo/' +
+
} else if (!dl.lastModified) {
encodeURI( pg.wiki.hostname ) + '/' +
+
txt= (/^\s*$/.test(dl.data)) ?
article.urlString() +
+
popupString('Empty revision, maybe non-existent') : popupString('Unknown date');
'?uselang=' + mw.config.get('wgUserLanguage');
+
} else {
return generalNavLink({url:url,
+
var datePrint=getValueOf('popupDiffDatePrinter');
title: tprintf('editorListHint', [article]),
+
if (typeof dl.lastModified[datePrint] == 'function') {
newWin:l.newWin, text: l.text, noPopup:1});
+
if (getValueOf('popupAdjustDiffDates')) {
 +
var off;
 +
if (off=Cookie.read('popTz')) {
 +
var d2=adjustDate(dl.lastModified, off);
 +
txt = dayFormat(d2, true) + ' ' + timeFormat(d2, true);
 +
}
 +
} else {
 +
txt = dl.lastModified[datePrint]();
 +
}
 +
} else {
 +
txt = tprintf('Invalid %s %s', ['popupDiffDatePrinter', datePrint]);
 +
}
 +
}
 +
var revlink = generalLink({url: dl.url.replace(/&.*?(oldid=[0-9]+(?:&direction=[^&]*)?).*/, '&$1'),
 +
  text: label, title: label});
 +
return simplePrintf('<tr><td>%s</td><td>%s</td></tr>', [ revlink, txt ]);
 
}
 
}
 +
//</NOLITE>
 +
// ENDFILE: diffpreview.js
 +
// STARTFILE: links.js
 +
//<NOLITE>
 +
/////////////////////
 +
// LINK GENERATION //
 +
/////////////////////
  
function generalNavLink(l) {
+
// titledDiffLink --> titledWikiLink --> generalLink
l.className = (l.className === null) ? 'popupNavLink' : l.className;
+
// wikiLink   --> titledWikiLink --> generalLink
return generalLink(l);
+
// editCounterLink --> generalLink
}
 
  
//////////////////////////////////////////////////
+
function titledDiffLink(l) { // article, text, title, from, to) {
// magic history links
+
return titledWikiLink({article: l.article, action: l.to + '&oldid=' + l.from,
//
+
newWin: l.newWin,
 
+
noPopup: l.noPopup,
function getHistoryInfo(wikipage, whatNext) {
+
text: l.text, title: l.title,
log('getHistoryInfo');
+
/* hack: no oldid here */
getHistory(wikipage, whatNext ? function(d){whatNext(processHistory(d));} : processHistory);
+
actionName: 'diff'});
 
}
 
}
  
// FIXME eliminate pg.idNumber ... how? :-(
 
  
function getHistory(wikipage, onComplete) {
+
function wikiLink(l) {
log('getHistory');
+
//{article:article, action:action, text:text, oldid, newid}) {
var url = pg.wiki.apiwikibase + '?format=json&formatversion=2&action=query&prop=revisions&titles=' +
+
if (! (typeof l.article == typeof {}
new Title(wikipage).urlString() + '&rvlimit=' + getValueOf('popupHistoryLimit');
+
  && typeof l.action == typeof '' && typeof l.text==typeof '')) return null;
log('getHistory: url='+url);
+
if (typeof l.oldid == 'undefined') { l.oldid=null; }
return startDownload(url, pg.idNumber+'history', onComplete);
+
var savedOldid = l.oldid;
}
+
if (!/^(edit|view|revert|render)$|^raw/.test(l.action)) { l.oldid=null; }
 
+
var hint=popupString(l.action + 'Hint'); // revertHint etc etc etc
function processHistory(download) {
+
var oldidData=[l.oldid, safeDecodeURI(l.article)];
var jsobj = getJsObj(download.data);
+
var revisionString = tprintf('revision %s of %s', oldidData);
try {
+
log('revisionString='+revisionString);
var revisions = anyChild(jsobj.query.pages).revisions;
+
switch (l.action) {
var edits=[];
+
case 'edit&section=new': hint = popupString('newSectionHint');  break;
for (var i=0; i<revisions.length; ++i) {
+
case 'edit&undo=':
edits.push({ oldid: revisions[i].revid, editor: revisions[i].user });
+
if (l.diff && l.diff != 'prev' && savedOldid ) {
 +
  l.action += l.diff + '&undoafter=' + savedOldid;
 +
} else if (savedOldid) {
 +
  l.action += savedOldid;
 
}
 
}
log('processed ' + edits.length + ' edits');
+
hint = popupString('undoHint');
return finishProcessHistory( edits, mw.config.get('wgUserName') );
+
break;
} catch (someError) {
+
case 'raw&ctype=text/css': hint=popupString('rawHint'); break;
log('Something went wrong with JSON business');
+
case 'revert':
return finishProcessHistory([]);
+
if ( !mw.config.get('wgEnableAPI') ) {
}
+
alert( 'This function of navigation popups now requires a MediaWiki ' +
}
+
'installation with the API enabled.');
 
+
break;
 
 
function finishProcessHistory(edits, userName) {
 
var histInfo={};
 
 
 
histInfo.edits=edits;
 
histInfo.userName=userName;
 
 
 
for (var i=0; i<edits.length; ++i) {
 
if (typeof histInfo.myLastEdit === 'undefined' && userName && edits[i].editor==userName) {
 
histInfo.myLastEdit={index: i, oldid: edits[i].oldid, previd: (i === 0 ? null : edits[i-1].oldid)};
 
 
}
 
}
if (typeof histInfo.firstNewEditor === 'undefined' && edits[i].editor != edits[0].editor) {
+
var p=parseParams(pg.current.link.href);
histInfo.firstNewEditor={index:i, oldid:edits[i].oldid, previd: (i === 0 ? null : edits[i-1].oldid)};
+
l.action='edit&autoclick=wpSave&actoken=' + autoClickToken() + '&autoimpl=' + popupString('autoedit_version') + '&autosummary=' + revertSummary(l.oldid, p.diff);
 +
if (p.diff=='prev') {
 +
l.action += '&direction=prev';
 +
revisionString = tprintf('the revision prior to revision %s of %s', oldidData);
 
}
 
}
 +
if (getValueOf('popupRevertSummaryPrompt')) { l.action += '&autosummaryprompt=true'; }
 +
if (getValueOf('popupMinorReverts')) { l.action += '&autominor=true'; }
 +
log('revisionString is now '+revisionString);
 +
break;
 +
case 'nullEdit':
 +
l.action='edit&autoclick=wpSave&actoken=' + autoClickToken() + '&autoimpl=' + popupString('autoedit_version') + '&autosummary=null';
 +
break;
 +
case 'historyfeed':
 +
l.action='history&feed=rss';
 +
break;
 +
case 'markpatrolled':
 +
l.action='markpatrolled&rcid='+l.rcid;
 
}
 
}
//pg.misc.historyInfo=histInfo;
 
return histInfo;
 
}
 
//</NOLITE>
 
// ENDFILE: links.js
 
// STARTFILE: options.js
 
//////////////////////////////////////////////////
 
// options
 
  
// check for existing value, else use default
+
if (hint) {
function defaultize(x) {
+
if (l.oldid) {
if (pg.option[x]===null || typeof pg.option[x]=='undefined') {
+
hint = simplePrintf(hint, [revisionString]);
if (typeof window[x] != 'undefined' ) pg.option[x]=window[x];
+
}
else pg.option[x]=pg.optionDefault[x];
+
else {
 +
hint = simplePrintf(hint, [safeDecodeURI(l.article)]);
 +
}
 +
}
 +
else {
 +
hint = safeDecodeURI(l.article + '&action=' + l.action) + (l.oldid) ? '&oldid='+l.oldid : '';
 
}
 
}
}
 
  
function newOption(x, def) {
+
return titledWikiLink({article: l.article, action: l.action, text: l.text, newWin:l.newWin,
pg.optionDefault[x]=def;
+
title: hint, oldid: l.oldid, noPopup: l.noPopup, onclick: l.onclick});
 
}
 
}
  
function setDefault(x, def) {
+
function revertSummary(oldid, diff) {
return newOption(x, def);
+
var ret='';
 +
if (diff == 'prev') {
 +
ret=getValueOf('popupQueriedRevertToPreviousSummary');
 +
} else { ret = getValueOf('popupQueriedRevertSummary'); }
 +
return ret + '&autorv=' + oldid;
 
}
 
}
  
function getValueOf(varName) {
+
function titledWikiLink(l) {
defaultize(varName);
+
// possible properties of argument:
return pg.option[varName];
+
// article, action, text, title, oldid, actionName, className, noPopup
}
+
// oldid = null is fine here
  
/*eslint-disable */
+
// article and action are mandatory args
function useDefaultOptions() { // for testing
 
for (var p in pg.optionDefault) {
 
pg.option[p]=pg.optionDefault[p];
 
if (typeof window[p]!='undefined') { delete window[p]; }
 
}
 
}
 
/*eslint-enable */
 
  
function setOptions() {
+
if (typeof l.article == 'undefined' || typeof l.action=='undefined') {
// user-settable parameters and defaults
+
errlog('got undefined article or action in titledWikiLink');
var userIsSysop = false;
+
return null;
if ( mw.config.get('wgUserGroups') ) {
 
for ( var g = 0; g < mw.config.get('wgUserGroups').length; ++g ) {
 
if ( mw.config.get('wgUserGroups')[g] == "sysop" )
 
userIsSysop = true;
 
}
 
 
}
 
}
  
// Basic options
+
var base = pg.wiki.titlebase +  l.article.urlString();
newOption('popupDelay',              0.5);
+
var url=base;
newOption('popupHideDelay',          0.5);
 
newOption('simplePopups',            false);
 
newOption('popupStructure',          'shortmenus');  // see later - default for popupStructure is 'original' if simplePopups is true
 
newOption('popupActionsMenu',        true);
 
newOption('popupSetupMenu',          true);
 
newOption('popupAdminLinks',          userIsSysop);
 
newOption('popupShortcutKeys',        false);
 
newOption('popupHistoricalLinks',    true);
 
newOption('popupOnlyArticleLinks',    true);
 
newOption('removeTitles',            true);
 
newOption('popupMaxWidth',            350);
 
newOption('popupSimplifyMainLink',    true);
 
newOption('popupAppendRedirNavLinks', true);
 
newOption('popupTocLinks',            false);
 
newOption('popupSubpopups',          true);
 
newOption('popupDragHandle',          false /* 'popupTopLinks'*/);
 
newOption('popupLazyPreviews',        true);
 
newOption('popupLazyDownloads',      true);
 
newOption('popupAllDabsStubs',        false);
 
newOption('popupDebugging',          false);
 
newOption('popupActiveNavlinks',      true);
 
newOption('popupModifier',            false); // ctrl, shift, alt or meta
 
newOption('popupModifierAction',      'enable'); // or 'disable'
 
newOption('popupDraggable',          true);
 
newOption('popupReview',              false);
 
  
//<NOLITE>
+
if (typeof l.actionName=='undefined' || !l.actionName) { l.actionName='action'; }
// images
 
newOption('popupImages',                true);
 
newOption('imagePopupsForImages',        true);
 
newOption('popupNeverGetThumbs',        false);
 
//newOption('popupImagesToggleSize',      true);
 
newOption('popupThumbAction',            'imagepage'); //'sizetoggle');
 
newOption('popupImageSize',              60);
 
newOption('popupImageSizeLarge',        200);
 
  
// redirs, dabs, reversion
+
// no need to add &action=view, and this confuses anchors
newOption('popupFixRedirs',            false);
+
if (l.action != 'view') { url = base + '&' + l.actionName + '=' + l.action; }
newOption('popupRedirAutoClick',        'wpDiff');
 
newOption('popupFixDabs',              false);
 
newOption('popupDabsAutoClick',        'wpDiff');
 
newOption('popupRevertSummaryPrompt',  false);
 
newOption('popupMinorReverts',          false);
 
newOption('popupRedlinkRemoval',        false);
 
newOption('popupRedlinkAutoClick',      'wpDiff');
 
newOption('popupWatchDisambiggedPages', null);
 
newOption('popupWatchRedirredPages',    null);
 
newOption('popupDabWiktionary',        'last');
 
  
// navlinks
+
if (typeof l.oldid!='undefined' && l.oldid) { url+='&oldid='+l.oldid; }
newOption('popupNavLinks',          true);
 
newOption('popupNavLinkSeparator',  ' &sdot; ');
 
newOption('popupLastEditLink',      true);
 
newOption('popupEditCounterTool',  'supercount');
 
newOption('popupEditCounterUrl',    '');
 
//</NOLITE>
 
  
// previews etc
+
var cssClass=pg.misc.defaultNavlinkClassname;
newOption('popupPreviews',            true);
+
if (typeof l.className!='undefined' && l.className) { cssClass=l.className; }
newOption('popupSummaryData',          true);
 
newOption('popupMaxPreviewSentences',  5);
 
newOption('popupMaxPreviewCharacters', 600);
 
newOption('popupLastModified',        true);
 
newOption('popupPreviewKillTemplates', true);
 
newOption('popupPreviewRawTemplates',  true);
 
newOption('popupPreviewFirstParOnly',  true);
 
newOption('popupPreviewCutHeadings',  true);
 
newOption('popupPreviewButton',        false);
 
newOption('popupPreviewButtonEvent',  'click');
 
  
//<NOLITE>
+
return generalNavLink({url: url, newWin: l.newWin,
// diffs
+
title: (typeof l.title != 'undefined') ? l.title : null,
newOption('popupPreviewDiffs',         true);
+
text: (typeof l.text!='undefined')?l.text:null,
newOption('popupDiffMaxLines',         100);
+
className: cssClass, noPopup:l.noPopup, onclick:l.onclick});
newOption('popupDiffContextLines',     2);
+
}
newOption('popupDiffContextCharacters', 40);
 
newOption('popupDiffDates',             true);
 
newOption('popupDiffDatePrinter',      'toLocaleString');
 
  
// edit summaries. God, these are ugly.
+
function getLastContrib(wikipage, newWin) {
newOption('popupReviewedSummary',         popupString('defaultpopupReviewedSummary') );
+
getHistoryInfo(wikipage, function(x){processLastContribInfo(x,{page: wikipage, newWin: newWin})});
newOption('popupFixDabsSummary',          popupString('defaultpopupFixDabsSummary') );
+
}
newOption('popupExtendedRevertSummary',   popupString('defaultpopupExtendedRevertSummary') );
+
function processLastContribInfo(info, stuff) {
newOption('popupRevertSummary',            popupString('defaultpopupRevertSummary') );
+
if(!info.edits || !info.edits.length) { alert('Popups: an odd thing happened. Please retry.'); return; }
newOption('popupRevertToPreviousSummary', popupString('defaultpopupRevertToPreviousSummary') );
+
if(!info.firstNewEditor) {
newOption('popupQueriedRevertSummary',           popupString('defaultpopupQueriedRevertSummary') );
+
alert(tprintf('Only found one editor: %s made %s edits', [info.edits[0].editor,info.edits.length]));
newOption('popupQueriedRevertToPreviousSummary', popupString('defaultpopupQueriedRevertToPreviousSummary') );
+
return;
newOption('popupFixRedirsSummary',        popupString('defaultpopupFixRedirsSummary') );
+
}
newOption('popupRedlinkSummary',          popupString('defaultpopupRedlinkSummary') );
+
var newUrl=pg.wiki.titlebase + new Title(stuff.page).urlString() + '&diff=cur&oldid='+info.firstNewEditor.oldid;
newOption('popupRmDabLinkSummary',         popupString('defaultpopupRmDabLinkSummary') );
+
displayUrl(newUrl, stuff.newWin);
//</NOLITE>
+
}
// misc
+
function getDiffSinceMyEdit(wikipage, newWin) {
newOption('popupHistoryLimit',        50);
+
getHistoryInfo(wikipage, function(x){processDiffSinceMyEdit(x,{page: wikipage, newWin: newWin})});
//<NOLITE>
+
}
newOption('popupFilters',             [popupFilterStubDetect,     popupFilterDisambigDetect,
+
function processDiffSinceMyEdit(info, stuff) {
      popupFilterPageSize,      popupFilterCountLinks,
+
if(!info.edits || !info.edits.length) { alert('Popups: something fishy happened. Please try again.'); return; }
      popupFilterCountImages,    popupFilterCountCategories,
+
var friendlyName=stuff.page.split('_').join(' ');
      popupFilterLastModified]);
+
if(!info.myLastEdit) {
newOption('extraPopupFilters',        []);
+
alert(tprintf('Couldn\'t find an edit by %s\nin the last %s edits to\n%s',
newOption('popupOnEditSelection', 'cursor');
+
  [info.userName, getValueOf('popupHistoryLimit'), friendlyName]));
newOption('popupPreviewHistory',     true);
+
return;
newOption('popupImageLinks',          true);
+
}
newOption('popupCategoryMembers',    true);
+
if(info.myLastEdit.index==0) {
newOption('popupUserInfo',            true);
+
alert(tprintf("%s seems to be the last editor to the page %s", [info.userName, friendlyName]));
newOption('popupHistoryPreviewLimit', 25);
+
return;
newOption('popupContribsPreviewLimit',25);
+
}
newOption('popupRevDelUrl',          '//en.wikipedia.org/wiki/Wikipedia:Revision_deletion');
+
var newUrl=pg.wiki.titlebase + new Title(stuff.page).urlString() + '&diff=cur&oldid='+ info.myLastEdit.oldid;
newOption('popupShowGender',          true);
+
displayUrl(newUrl, stuff.newWin);
//</NOLITE>
+
}
 
+
function displayUrl(url, newWin){
// new windows
+
if(newWin) { window.open(url); }
newOption('popupNewWindows',     false);
+
else { document.location=url; }
newOption('popupLinksNewWindow', {'lastContrib': true, 'sinceMe': true});
+
}
 
+
 
// regexps
+
function purgePopups() {
newOption('popupDabRegexp', '(\\{\\{\\s*disambig(?!uation needed)|disambig(uation|)\\s*\\}\\}|disamb\\s*\\}\\}|dab\\s*\\}\\})|\\{\\{\\s*(((geo|hn|road?|school|number)dis)|[234][lc][acw]|(road|ship)index)(\\s*[|][^}]*)?\\s*[}][}]|is a .*disambiguation.*page');
+
processAllPopups(true);
newOption('popupAnchorRegexp', 'anchors?'); //how to identify an anchors template
+
setupCache(); // deletes all cached items (not browser cached, though...)
newOption('popupStubRegexp', '(sect)?stub[}][}]|This .*-related article is a .*stub');
+
pg.option={};
newOption('popupImageVarsRegexp', 'image|image_(?:file|skyline|name|flag|seal)|cover|badge|logo');
+
abortAllDownloads();
 +
}
 +
 
 +
function processAllPopups(nullify, banish) {
 +
for (var i=0; pg.current.links && i<pg.current.links.length; ++i) {
 +
if (!pg.current.links[i].navpopup) { continue; }
 +
(nullify || banish) && pg.current.links[i].navpopup.banish();
 +
pg.current.links[i].simpleNoMore=false;
 +
nullify && (pg.current.links[i].navpopup=null);
 +
}
 +
}
 +
 
 +
function disablePopups(){
 +
processAllPopups(false, true);
 +
setupTooltips(null, true);
 
}
 
}
// ENDFILE: options.js
 
// STARTFILE: strings.js
 
//<NOLITE>
 
//////////////////////////////////////////////////
 
// Translatable strings
 
//////////////////////////////////////////////////
 
//
 
// See instructions at
 
// https://en.wikipedia.org/wiki/Wikipedia:Tools/Navigation_popups/Translation
 
  
pg.string = {
+
function togglePreviews() {
/////////////////////////////////////
+
processAllPopups(true, true);
// summary data, searching etc.
+
pg.option.simplePopups=!pg.option.simplePopups;
/////////////////////////////////////
+
abortAllDownloads();
'article': 'article',
+
}
'category': 'category',
+
 
'categories': 'categories',
+
function magicWatchLink(l) {
'image': 'image',
+
//Yuck!! Would require a thorough redesign to add this as a click event though ...
'images': 'images',
+
l.onclick = simplePrintf( 'modifyWatchlist(\'%s\',\'%s\');return false;', [l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), this.id] );
'stub': 'stub',
+
return wikiLink(l);
'section stub': 'section stub',
+
}
'Empty page': 'Empty page',
+
 
'kB': 'kB',
+
function modifyWatchlist(title, action) {
'bytes': 'bytes',
+
var reqData = {
'day': 'day',
+
'action': 'watch',
'days': 'days',
+
'format': 'json',
'hour': 'hour',
+
'title': title,
'hours': 'hours',
+
'token': mw.user.tokens.get('watchToken'),
'minute': 'minute',
+
'uselang': mw.config.get('wgUserLanguage')
'minutes': 'minutes',
+
};
'second': 'second',
+
if (action==='unwatch') reqData.unwatch = '';
'seconds': 'seconds',
+
 
'week': 'week',
+
jQuery.ajax({
'weeks': 'weeks',
+
url: mw.util.wikiScript('api'),
'search': 'search',
+
dataType: 'json',
'SearchHint': 'Find English Wikipedia articles containing %s',
+
type: 'POST',
'web': 'web',
+
data: reqData,
'global': 'global',
+
success: function( data, textStatus, xhr ) {
'globalSearchHint': 'Search across Wikipedias in different languages for %s',
+
mw.util.jsMessage( data.watch.message, 'watch' );
'googleSearchHint': 'Google for %s',
+
}
/////////////////////////////////////
+
});
// article-related actions and info
+
}
// (some actions also apply to user pages)
+
 
/////////////////////////////////////
+
function magicHistoryLink(l) {
'actions': 'actions', ///// view articles and view talk
+
// FIXME use onclick change href trick to sort this out instead of window.open
'popupsMenu': 'popups',
+
 
'togglePreviewsHint': 'Toggle preview generation in popups on this page',
+
var jsUrl='', title='';
'enable previews': 'enable previews',
+
switch(l.id) {
'disable previews': 'disable previews',
+
case 'lastContrib':
'toggle previews': 'toggle previews',
+
jsUrl=simplePrintf('javascript:getLastContrib(\'%s\',%s)',
'show preview': 'show preview',
+
[l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), l.newWin]);
'reset': 'reset',
+
title=popupString('lastContribHint');
'more...': 'more...',
+
break;
'disable': 'disable popups',
+
case 'sinceMe':
'disablePopupsHint': 'Disable popups on this page. Reload page to re-enable.',
+
jsUrl=simplePrintf('javascript:getDiffSinceMyEdit(\'%s\',%s)',
'historyfeedHint': 'RSS feed of recent changes to this page',
+
[l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), l.newWin]);
'purgePopupsHint': 'Reset popups, clearing all cached popup data.',
+
title=popupString('sinceMeHint');
'PopupsHint': 'Reset popups, clearing all cached popup data.',
+
break;
'spacebar': 'space',
+
}
'view': 'view',
+
 
'view article': 'view article',
+
return generalNavLink({url: jsUrl, newWin: false, // can't have new windows with JS links, I think
'viewHint': 'Go to %s',
+
title: title, text: l.text, noPopup: l.noPopup});
'talk': 'talk',
+
}
'talk page': 'talk page',
+
 
'this&nbsp;revision': 'this&nbsp;revision',
+
function popupMenuLink(l) {
'revision %s of %s': 'revision %s of %s',
+
var jsUrl=simplePrintf('javascript:%s()', [l.id]);
'Revision %s of %s': 'Revision %s of %s',
+
var title=popupString(simplePrintf('%sHint', [l.id]));
'the revision prior to revision %s of %s': 'the revision prior to revision %s of %s',
+
return generalNavLink({url: jsUrl, newWin:false, title:title, text:l.text, noPopup:l.noPopup});
'Toggle image size': 'Click to toggle image size',
+
}
'del': 'del', ///// delete, protect, move
+
 
'delete': 'delete',
+
function specialLink(l) {
'deleteHint': 'Delete %s',
+
// properties: article, specialpage, text, sep
'undeleteShort': 'un',
+
if (typeof l.specialpage=='undefined'||!l.specialpage) return null;
'UndeleteHint': 'Show the deletion history for %s',
+
var base = pg.wiki.titlebase +  mw.config.get('wgFormattedNamespaces')[pg.nsSpecialId]+':'+l.specialpage;
'protect': 'protect',
+
if (typeof l.sep == 'undefined' || l.sep===null) l.sep='&target=';
'protectHint': 'Restrict editing rights to %s',
+
var article=l.article.urlString({keepSpaces: l.specialpage=='Search'});
'unprotectShort': 'un',
+
var hint=popupString(l.specialpage+'Hint');
'unprotectHint': 'Allow %s to be edited by anyone again',
+
switch (l.specialpage) {
'send thanks': 'send thanks',
+
case 'Log':
'ThanksHint': 'Send a thank you notification to this user',
+
switch (l.sep) {
'move': 'move',
+
case '&user=': hint=popupString('userLogHint'); break;
'move page': 'move page',
+
case '&type=block&page=': hint=popupString('blockLogHint'); break;
'MovepageHint': 'Change the title of %s',
+
case '&page=': hint=popupString('pageLogHint'); break;
'edit': 'edit',   ///// edit articles and talk
+
case '&type=protect&page=': hint=popupString('protectLogHint'); break;
'edit article': 'edit article',
+
case '&type=delete&page=': hint=popupString('deleteLogHint'); break;
'editHint': 'Change the content of %s',
+
default: log('Unknown log type, sep=' + l.sep); hint='Missing hint (FIXME)';
'edit talk': 'edit talk',
+
}
'new': 'new',
+
break;
'new topic': 'new topic',
+
case 'PrefixIndex': article += '/'; break;
'newSectionHint': 'Start a new section on %s',
+
}
'null edit': 'null edit',
+
if (hint) hint = simplePrintf(hint, [safeDecodeURI(l.article)]);
'nullEditHint': 'Submit an edit to %s, making no changes ',
+
else hint = safeDecodeURI(l.specialpage+':'+l.article) ;
'hist': 'hist',   ///// history, diffs, editors, related
+
 
'history': 'history',
+
var url = base + l.sep + article;
'historyHint': 'List the changes made to %s',
+
return generalNavLink({url: url, title: hint, text: l.text, newWin:l.newWin, noPopup:l.noPopup});
'last': 'prev', // For labelling the previous revision in history pages; the key is "last" for backwards compatibility
+
}
'lastEdit': 'lastEdit',
+
 
'mark patrolled': 'mark patrolled',
+
function generalLink(l) {
'markpatrolledHint': 'Mark this edit as patrolled',
+
// l.url, l.text, l.title, l.newWin, l.className, l.noPopup, l.onclick
'Could not marked this edit as patrolled': 'Could not marked this edit as patrolled',
+
if (typeof l.url=='undefined') return null;
'show last edit': 'most recent edit',
+
 
'Show the last edit': 'Show the effects of the most recent change',
+
// only quotation marks in the url can screw us up now... I think
'lastContrib': 'lastContrib',
+
var url=l.url.split('"').join('%22');
'last set of edits': 'latest edits',
+
 
'lastContribHint': 'Show the net effect of changes made by the last editor',
+
var ret='<a href="' + url + '"';
'cur': 'cur',
+
if (typeof l.title!='undefined' && l.title) { ret += ' title="' + pg.escapeQuotesHTML(l.title) + '"'; }
'diffCur': 'diffCur',
+
if (typeof l.onclick!='undefined' && l.onclick) { ret += ' onclick="' + pg.escapeQuotesHTML(l.onclick) + '"'; }
'Show changes since revision %s': 'Show changes since revision %s',
+
if (l.noPopup) { ret += ' noPopup=1'; }
'%s old': '%s old', // as in 4 weeks old
+
var newWin;
'oldEdit': 'oldEdit',
+
if (typeof l.newWin=='undefined' || l.newWin===null) { newWin=getValueOf('popupNewWindows'); }
'purge': 'purge',
+
else { newWin=l.newWin; }
'purgeHint': 'Demand a fresh copy of %s',
+
if (newWin) { ret += ' target="_blank"'; }
'raw': 'source',
+
if (typeof l.className!='undefined'&&l.className) { ret+=' class="'+l.className+'"'; }
'rawHint': 'Download the source of %s',
+
ret += '>';
'render': 'simple',
+
if (typeof l.text==typeof '') { ret+= l.text; }
'renderHint': 'Show a plain HTML version of %s',
+
ret +='</a>';
'Show the edit made to get revision': 'Show the edit made to get revision',
+
return ret;
'sinceMe': 'sinceMe',
+
}
'changes since mine': 'diff my edit',
+
 
'sinceMeHint': 'Show changes since my last edit',
+
function appendParamsToLink(linkstr, params) {
'Couldn\'t find an edit by %s\nin the last %s edits to\n%s': 'Couldn\'t find an edit by %s\nin the last %s edits to\n%s',
+
var sp=linkstr.parenSplit(RegExp('(href="[^"]+?)"', 'i'));
'eds': 'eds',
+
if (sp.length<2) return null;
'editors': 'editors',
+
var ret=sp.shift() + sp.shift();
'editorListHint': 'List the users who have edited %s',
+
ret += '&' + params + '"';
'related': 'related',
+
ret += sp.join('');
'relatedChanges': 'relatedChanges',
+
return ret;
'related changes': 'related changes',
+
}
'RecentchangeslinkedHint': 'Show changes in articles related to %s',
+
 
'editOld': 'editOld',   ///// edit old version, or revert
+
function changeLinkTargetLink(x) { // newTarget, text, hint, summary, clickButton, minor, title (optional) {
'rv': 'rv',
+
if (x.newTarget) {
'revert': 'revert',
+
log ('changeLinkTargetLink: newTarget=' + x.newTarget);
'revertHint': 'Revert to %s',
+
}
'defaultpopupReviewedSummary': 'Accepted by reviewing the [[Special:diff/%s/%s|difference]] between this version and previously accepted version using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
if (x.oldTarget !== decodeURIComponent( x.oldTarget ) ) {
'defaultpopupRedlinkSummary': 'Removing link to empty page [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
log ('This might be an input problem: ' + x.oldTarget );
'defaultpopupFixDabsSummary': 'Disambiguate [[%s]] to [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
}
'defaultpopupFixRedirsSummary': 'Redirect bypass from [[%s]] to [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
 
'defaultpopupExtendedRevertSummary': 'Revert to revision dated %s by %s, oldid %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
// FIXME: first character of page title as well as namespace should be case insensitive
'defaultpopupRevertToPreviousSummary': 'Revert to the revision prior to revision %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
// eg [[category:foo]] and [[Category:Foo]] are equivalent
'defaultpopupRevertSummary': 'Revert to revision %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
// this'll break if charAt(0) is nasty
'defaultpopupQueriedRevertToPreviousSummary': 'Revert to the revision prior to revision $1 dated $2 by $3 using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
var cA=literalizeRegex(x.oldTarget);
'defaultpopupQueriedRevertSummary': 'Revert to revision $1 dated $2 by $3 using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
var chs=cA.charAt(0).toUpperCase();
'defaultpopupRmDabLinkSummary': 'Remove link to dab page [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
+
chs='['+chs + chs.toLowerCase()+']';
'Redirects': 'Redirects', // as in Redirects to ...
+
var currentArticleRegexBit=chs+cA.substring(1);
' to ': ' to ',   // as in Redirects to ...
+
currentArticleRegexBit=currentArticleRegexBit
'Bypass redirect': 'Bypass redirect',
+
.split(RegExp('(?:[_ ]+|%20)', 'g')).join('(?:[_ ]+|%20)')
'Fix this redirect': 'Fix this redirect',
+
.split('\\(').join('(?:%28|\\()')
'disambig': 'disambig',   ///// add or remove dab etc.
+
.split('\\)').join('(?:%29|\\))'); // why does this need to match encoded strings ? links in the document ?
'disambigHint': 'Disambiguate this link to [[%s]]',
+
// leading and trailing space should be ignored, and anchor bits optional:
'Click to disambiguate this link to:': 'Click to disambiguate this link to:',
+
currentArticleRegexBit = '\\s*(' + currentArticleRegexBit + '(?:#[^\\[\\|]*)?)\\s*';
'remove this link': 'remove this link',
+
// e.g. Computer (archaic) -> \s*([Cc]omputer[_ ](?:%2528|\()archaic(?:%2528|\)))\s*
'remove all links to this page from this article': 'remove all links to this page from this article',
+
 
'remove all links to this disambig page from this article': 'remove all links to this disambig page from this article',
+
// autoedit=s~\[\[([Cc]ad)\]\]~[[Computer-aided%20design|$1]]~g;s~\[\[([Cc]AD)[|]~[[Computer-aided%20design|~g
'mainlink': 'mainlink',   ///// links, watch, unwatch
+
 
'wikiLink': 'wikiLink',
+
var title=x.title || mw.config.get('wgPageName').split('_').join(' ');
'wikiLinks': 'wikiLinks',
+
var lk=titledWikiLink({article: new Title(title), newWin:x.newWin,
'links here': 'links here',
+
action: 'edit',
'whatLinksHere': 'whatLinksHere',
+
text: x.text,
'what links here': 'what links here',
+
title:   x.hint,
'WhatlinkshereHint': 'List the pages that are hyperlinked to %s',
+
className: 'popup_change_title_link'
'unwatchShort': 'un',
+
});
'watchThingy': 'watch',  // called watchThingy because {}.watch is a function
+
var cmd='';
'watchHint': 'Add %s to my watchlist',
+
if (x.newTarget) {
'unwatchHint': 'Remove %s from my watchlist',
+
// escape '&' and other nasties
'Only found one editor: %s made %s edits': 'Only found one editor: %s made %s edits',
+
var t=x.newTarget;
'%s seems to be the last editor to the page %s': '%s seems to be the last editor to the page %s',
+
var s=literalizeRegex(x.newTarget);
'rss': 'rss',
+
cmd += 's~\\[\\['+currentArticleRegexBit+'\\]\\]~[['+t+'|$1]]~g;';
/////////////////////////////////////
+
cmd += 's~\\[\\['+currentArticleRegexBit+'[|]~[['+t+'|~g;';
// diff previews
+
cmd += 's~\\[\\['+s + '\\|' + s + '\\]\\]~[[' + t + ']]~g';
/////////////////////////////////////
+
} else {
'Diff truncated for performance reasons': 'Diff truncated for performance reasons',
+
cmd += 's~\\[\\['+currentArticleRegexBit+'\\]\\]~$1~g;';
'Old revision': 'Old revision',
+
cmd += 's~\\[\\['+currentArticleRegexBit+'[|](.*?)\\]\\]~$2~g';
'New revision': 'New revision',
+
}
'Something went wrong :-(': 'Something went wrong :-(',
+
// Build query
'Empty revision, maybe non-existent': 'Empty revision, maybe non-existent',
+
cmd = 'autoedit=' + encodeURIComponent ( cmd );
'Unknown date': 'Unknown date',
+
cmd += '&autoclick='+ encodeURIComponent( x.clickButton ) + '&actoken=' + encodeURIComponent( autoClickToken() );
/////////////////////////////////////
+
cmd += ( x.minor == null ) ? '' : '&autominor='+ encodeURIComponent( x.minor );
// other special previews
+
cmd += ( x.watch == null ) ? '' : '&autowatch='+ encodeURIComponent( x.watch );
/////////////////////////////////////
+
cmd += '&autosummary='+encodeURIComponent(x.summary);
'Empty category': 'Empty category',
+
cmd += '&autoimpl='+encodeURIComponent( popupString('autoedit_version') );
'Category members (%s shown)': 'Category members (%s shown)',
+
return appendParamsToLink(lk, cmd);
'No image links found': 'No image links found',
+
}
'File links': 'File links',
+
 
'No image found': 'No image found',
+
 
'Image from Commons': 'Image from Commons',
+
function redirLink(redirMatch, article) {
'Description page': 'Description page',
+
// NB redirMatch is in wikiText
'Alt text:': 'Alt text:',
+
var ret='';
'revdel':'Hidden revision',
+
 
/////////////////////////////////////
+
if (getValueOf('popupAppendRedirNavLinks') && getValueOf('popupNavLinks')) {
// user-related actions and info
+
ret += '<hr />';
/////////////////////////////////////
+
if (getValueOf('popupFixRedirs') && typeof autoEdit != 'undefined' && autoEdit) {
'user': 'user',   ///// user page, talk, email, space
+
log('redirLink: newTarget=' + redirMatch);
'user&nbsp;page': 'user&nbsp;page',
+
ret += addPopupShortcut(
'user talk': 'user talk',
+
changeLinkTargetLink(
'edit user talk': 'edit user talk',
+
{newTarget: redirMatch, text: popupString('Redirects'),
'leave comment': 'leave comment',
+
hint: popupString('Fix this redirect'),
'email': 'email',
+
summary: simplePrintf(getValueOf('popupFixRedirsSummary'),
'email user': 'email user',
+
  [article.toString(), redirMatch ]),
'EmailuserHint': 'Send an email to %s',
+
oldTarget: article.toString(),
'space': 'space', // short form for userSpace link
+
clickButton: getValueOf('popupRedirAutoClick'), minor: true,
'PrefixIndexHint': 'Show pages in the userspace of %s',
+
watch: getValueOf('popupWatchRedirredPages')})
'count': 'count', ///// contributions, log
+
, 'R');
'edit counter': 'edit counter',
+
ret += popupString(' to ');
'editCounterLinkHint': 'Count the contributions made by %s',
+
}
'contribs': 'contribs',
+
else ret += popupString('Redirects') + popupString(' to ');
'contributions': 'contributions',
+
return ret;
'deletedContribs': 'deleted contributions',
+
}
'DeletedcontributionsHint': 'List deleted edits made by %s',
+
 
'ContributionsHint': 'List the contributions made by %s',
+
else return '<br> ' + popupString('Redirects') + popupString(' to ') +
'log': 'log',
+
titledWikiLink({article: new Title().fromWikiText(redirMatch), action: 'view', /* FIXME: newWin */
'user log': 'user log',
+
  text: safeDecodeURI(redirMatch), title: popupString('Bypass redirect')});
'userLogHint': 'Show %s\'s user log',
+
}
'arin': 'ARIN lookup', ///// ARIN lookup, block user or IP
+
 
'Look up %s in ARIN whois database': 'Look up %s in the ARIN whois database',
+
function arinLink(l) {
'unblockShort': 'un',
+
if (!saneLinkCheck(l)) { return null; }
'block': 'block',
+
if ( ! l.article.isIpUser() || ! pg.wiki.wikimedia) return null;
'block user': 'block user',
+
 
'IpblocklistHint': 'Unblock %s',
+
var uN=l.article.userName();
'BlockipHint': 'Prevent %s from editing',
+
 
'block log': 'block log',
+
return generalNavLink({url:'http://ws.arin.net/cgi-bin/whois.pl?queryinput=' + encodeURIComponent(uN), newWin:l.newWin,
'blockLogHint': 'Show the block log for %s',
+
title: tprintf('Look up %s in ARIN whois database', [uN]),
'protectLogHint': 'Show the protection log for %s',
+
text: l.text, noPopup:1});
'pageLogHint': 'Show the page log for %s',
+
}
'deleteLogHint': 'Show the deletion log for %s',
+
 
'Invalid %s %s': 'The option %s is invalid: %s',
+
function toolDbName(cookieStyle) {
'No backlinks found': 'No backlinks found',
+
var ret = mw.config.get('wgDBname');
' and more': ' and more',
+
if (!cookieStyle) { ret+= '_p'; }
'undo': 'undo',
+
return ret;
'undoHint': 'undo this edit',
+
}
'Download preview data': 'Download preview data',
+
 
'Invalid or IP user': 'Invalid or IP user',
+
function saneLinkCheck(l) {
'Not a registered username': 'Not a registered username',
+
if (typeof l.article != typeof {} || typeof l.text != typeof '') { return false; }
'BLOCKED': 'BLOCKED',
+
return true;
' edits since: ': ' edits since: ',
+
}
'last edit on ': 'last edit on ',
+
function editCounterLink(l) {
/////////////////////////////////////
+
if(!saneLinkCheck(l)) return null;
// Autoediting
+
if (! pg.wiki.wikimedia) return null;
/////////////////////////////////////
+
var uN=l.article.userName();
'Enter a non-empty edit summary or press cancel to abort': 'Enter a non-empty edit summary or press cancel to abort',
+
var tool=getValueOf('popupEditCounterTool');
'Failed to get revision information, please edit manually.\n\n': 'Failed to get revision information, please edit manually.\n\n',
+
var url;
'The %s button has been automatically clicked. Please wait for the next page to load.': 'The %s button has been automatically clicked. Please wait for the next page to load.',
+
var soxredToolUrl='//tools.wmflabs.org/supercount/index.php?user=$1&project=$2.$3';
'Could not find button %s. Please check the settings in your javascript file.': 'Could not find button %s. Please check the settings in your javascript file.',
 
/////////////////////////////////////
 
// Popups setup
 
/////////////////////////////////////
 
'Open full-size image': 'Open full-size image',
 
'zxy': 'zxy',
 
'autoedit_version': 'np20140416'
 
};
 
  
 
+
switch(tool) {
function popupString(str) {
+
case 'custom':
if (typeof popupStrings != 'undefined' && popupStrings && popupStrings[str]) { return popupStrings[str]; }
+
url=simplePrintf(getValueOf('popupEditCounterUrl'), [ encodeURIComponent(uN), toolDbName() ]);
if (pg.string[str]) { return pg.string[str]; }
+
break;
return str;
+
case 'kate':
}
+
case 'interiot':
 
+
default:
 
+
var theWiki=pg.wiki.hostname.split('.');
function tprintf(str,subs) {
+
url=simplePrintf(soxredToolUrl, [ encodeURIComponent(uN), theWiki[0], theWiki[1] ]);
if (typeof subs != typeof []) { subs = [subs]; }
+
}
return simplePrintf(popupString(str), subs);
+
return generalNavLink({url:url, title: tprintf('editCounterLinkHint', [uN]),
}
+
newWin:l.newWin, text: l.text, noPopup:1});
 
+
}
//</NOLITE>
+
 
// ENDFILE: strings.js
+
 
// STARTFILE: run.js
+
function globalSearchLink(l) {
////////////////////////////////////////////////////////////////////
+
if(!saneLinkCheck(l)) return null;
// Run things
+
 
////////////////////////////////////////////////////////////////////
+
var base='http://vs.aka-online.de/cgi-bin/globalwpsearch.pl?timeout=120&search=';
 
+
var article=l.article.urlString({keepSpaces:true});
 
+
 
// For some reason popups requires a fully loaded page jQuery.ready(...) causes problems for some.
+
return generalNavLink({url:base + article, newWin:l.newWin,
// The old addOnloadHook did something similar to the below
+
title: tprintf('globalSearchHint', [safeDecodeURI(l.article)]),
if (document.readyState=="complete")
+
text: l.text, noPopup:1});
autoEdit(); //will setup popups
+
}
else
+
 
$( window ).on( 'load', autoEdit );
+
function googleLink(l) {
 
+
if(!saneLinkCheck(l)) return null;
 
+
 
// Support for MediaWiki's live preview, VisualEditor's saves and Echo's flyout.
+
var base='http://www.google.com/search?q=';
( function () {
+
var article=l.article.urlString({keepSpaces:true});
var once = true;
+
 
function dynamicContentHandler( $content ) {
+
return generalNavLink({url:base + '%22' + article + '%22', newWin:l.newWin,
// Try to detect the hook fired on initial page load and disregard
+
title: tprintf('googleSearchHint', [safeDecodeURI(l.article)]),
// it, we already hook to onload (possibly to different parts of
+
text: l.text, noPopup:1});
// page - it's configurable) and running twice might be bad. Ugly…
+
}
if ( $content.attr( 'id' ) == 'mw-content-text' ) {
+
 
if ( once ) {
+
function editorListLink(l) {
once = false;
+
if(!saneLinkCheck(l)) return null;
return;
+
var article= l.article.articleFromTalkPage() || l.article;
}
+
var theWiki=pg.wiki.hostname.split('.');
}
+
var base='//tools.wmflabs.org/xtools/articleinfo/index.php?&uselang=' + mw.config.get('wgUserLanguage') +
+
'lang=' + theWiki[0] + '&wiki=' + theWiki[1] + '&begin=&end=&article='
function registerHooksForVisibleNavpops () {
+
return generalNavLink({url:base+article.urlString(),
for (var i=0; pg.current.links && i<pg.current.links.length; ++i) {
+
title: tprintf('editorListHint', [article]),
var navpop = pg.current.links[i].navpopup;
+
newWin:l.newWin, text: l.text, noPopup:1});
if (!navpop || !navpop.isVisible()) { continue; }
+
}
+
 
Navpopup.tracker.addHook(posCheckerHook(navpop));
+
function generalNavLink(l) {
}
+
l.className = (l.className==null) ? 'popupNavLink' : l.className;
}
+
return generalLink(l);
+
}
function doIt () {
+
 
registerHooksForVisibleNavpops();
+
//////////////////////////////////////////////////
$content.each( function () {
+
// magic history links
this.ranSetupTooltipsAlready = false;
+
//
setupTooltips( this );
+
 
} );
+
function getHistoryInfo(wikipage, whatNext) {
 +
log('getHistoryInfo');
 +
getHistory(wikipage, whatNext ? function(d){whatNext(processHistory(d));} : processHistory);
 +
}
 +
 
 +
// FIXME eliminate pg.idNumber ... how? :-(
 +
 
 +
function getHistory(wikipage, onComplete) {
 +
log('getHistory');
 +
if( !mw.config.get('wgEnableAPI') ) {
 +
alert( 'This function of navigation popups now requires a MediaWiki ' +
 +
'installation with the API enabled.');
 +
return false;
 +
}
 +
var url = pg.wiki.apiwikibase + '?format=json&action=query&prop=revisions&titles=' +
 +
new Title(wikipage).urlString() + '&rvlimit=' + getValueOf('popupHistoryLimit');
 +
log('getHistory: url='+url);
 +
if (pg.flag.isIE) {
 +
url = url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
 +
}
 +
return startDownload(url, pg.idNumber+'history', onComplete);
 +
}
 +
 
 +
function processHistory(download) {
 +
var jsobj = getJsObj(download.data);
 +
try {
 +
window.x=jsobj;
 +
var p=jsobj['query']['pages']
 +
for (var pageid in p) {
 +
var revisions=p[pageid]['revisions'];
 +
// we only get the first one
 +
break;
 +
}
 +
} catch (someError) {
 +
log('Something went wrong with JSON business');
 +
return finishProcessHistory([]);
 +
}
 +
var edits=[];
 +
for (var i=0; i<revisions.length; ++i) {
 +
edits.push({ oldid: revisions[i]['revid'], editor: revisions[i]['user'] });
 +
}
 +
log('processed ' + edits.length + ' edits');
 +
return finishProcessHistory( edits, mw.config.get('wgUserName') );
 +
}
 +
 
 +
 
 +
function finishProcessHistory(edits, userName) {
 +
var histInfo={};
 +
 
 +
histInfo.edits=edits;
 +
histInfo.userName=userName;
 +
 
 +
for (var i=0; i<edits.length; ++i) {
 +
if (typeof histInfo.myLastEdit == 'undefined' && userName && edits[i].editor==userName) {
 +
histInfo.myLastEdit={index: i, oldid: edits[i].oldid, previd: (i==0 ? null : edits[i-1].oldid)};
 +
}
 +
if (typeof histInfo.firstNewEditor == 'undefined' && edits[i].editor != edits[0].editor) {
 +
histInfo.firstNewEditor={index:i, oldid:edits[i].oldid, previd: (i==0 ? null : edits[i-1].oldid)};
 +
}
 +
}
 +
//pg.misc.historyInfo=histInfo;
 +
return histInfo;
 +
}
 +
//</NOLITE>
 +
// ENDFILE: links.js
 +
// STARTFILE: options.js
 +
//////////////////////////////////////////////////
 +
// options
 +
 
 +
// check for cookies and existing value, else use default
 +
function defaultize(x) {
 +
var val=null;
 +
if (x!='popupCookies') {
 +
defaultize('popupCookies');
 +
if (pg.option.popupCookies && (val=Cookie.read(x))) {
 +
pg.option[x]=val;
 +
return;
 +
}
 +
}
 +
if (pg.option[x]===null || typeof pg.option[x]=='undefined') {
 +
if (typeof window[x] != 'undefined' ) pg.option[x]=window[x];
 +
else pg.option[x]=pg.optionDefault[x];
 +
}
 +
}
 +
 
 +
function newOption(x, def) {
 +
pg.optionDefault[x]=def;
 +
}
 +
 
 +
function setDefault(x, def) {
 +
return newOption(x, def);
 +
}
 +
 
 +
function getValueOf(varName) {
 +
defaultize(varName);
 +
return pg.option[varName];
 +
}
 +
 
 +
function useDefaultOptions() { // for testing
 +
for (var p in pg.optionDefault) {
 +
pg.option[p]=pg.optionDefault[p];
 +
if (typeof window[p]!='undefined') { delete window[p]; }
 +
}
 +
}
 +
 
 +
function setOptions() {
 +
// user-settable parameters and defaults
 +
var userIsSysop = false;
 +
if ( mw.config.get('wgUserGroups') ) {
 +
for ( var g = 0; g < mw.config.get('wgUserGroups').length; ++g ) {
 +
if ( mw.config.get('wgUserGroups')[g] == "sysop" )
 +
userIsSysop = true
 +
}
 +
}
 +
 
 +
// Basic options
 +
newOption('popupDelay',              0.5);
 +
newOption('popupHideDelay',          0.5);
 +
newOption('simplePopups',            false);
 +
newOption('popupStructure',          'shortmenus');  // see later - default for popupStructure is 'original' if simplePopups is true
 +
newOption('popupActionsMenu',        true);
 +
newOption('popupSetupMenu',          true);
 +
newOption('popupAdminLinks',          userIsSysop);
 +
newOption('popupShortcutKeys',        false);
 +
newOption('popupHistoricalLinks',    true);
 +
newOption('popupOnlyArticleLinks',    true);
 +
newOption('removeTitles',            true);
 +
newOption('popupMaxWidth',            350);
 +
newOption('popupInitialWidth',        false); // integer or false
 +
newOption('popupSimplifyMainLink',    true);
 +
newOption('popupAppendRedirNavLinks', true);
 +
newOption('popupTocLinks',            false);
 +
newOption('popupSubpopups',          true);
 +
newOption('popupDragHandle',          false /* 'popupTopLinks'*/);
 +
newOption('popupLazyPreviews',        true);
 +
newOption('popupLazyDownloads',      true);
 +
newOption('popupAllDabsStubs',        false);
 +
newOption('popupDebugging',          false);
 +
newOption('popupAdjustDiffDates',    true);
 +
newOption('popupActiveNavlinks',      true);
 +
newOption('popupModifier',            false); // ctrl, shift, alt or meta
 +
newOption('popupModifierAction',      'enable'); // or 'disable'
 +
newOption('popupDraggable',          true);
 +
 
 +
//<NOLITE>
 +
// images
 +
newOption('popupImages',                true);
 +
newOption('imagePopupsForImages',        true);
 +
newOption('popupNeverGetThumbs',        false);
 +
//newOption('popupImagesToggleSize',      true);
 +
newOption('popupThumbAction',            'imagepage'); //'sizetoggle');
 +
newOption('popupImageSize',              60);
 +
newOption('popupImageSizeLarge',        200);
 +
 
 +
// redirs, dabs, reversion
 +
newOption('popupFixRedirs',            false);
 +
newOption('popupRedirAutoClick',        'wpDiff');
 +
newOption('popupFixDabs',              false);
 +
newOption('popupDabsAutoClick',        'wpDiff');
 +
newOption('popupRevertSummaryPrompt',  false);
 +
newOption('popupMinorReverts',          false);
 +
newOption('popupRedlinkRemoval',        false);
 +
newOption('popupWatchDisambiggedPages', null);
 +
newOption('popupWatchRedirredPages',    null);
 +
newOption('popupDabWiktionary',        'last');
 +
 
 +
// navlinks
 +
newOption('popupNavLinks',          true);
 +
newOption('popupNavLinkSeparator',  ' &sdot; ');
 +
newOption('popupLastEditLink',      true);
 +
newOption('popupEditCounterTool',  'soxred');
 +
newOption('popupEditCounterUrl',    '');
 +
newOption('popupExtraUserMenu',    '');
 +
//</NOLITE>
 +
 
 +
// previews etc
 +
newOption('popupPreviews',            true);
 +
newOption('popupSummaryData',          true);
 +
newOption('popupMaxPreviewSentences',  5);
 +
newOption('popupMaxPreviewCharacters', 600);
 +
newOption('popupLastModified',        true);
 +
newOption('popupPreviewKillTemplates', true);
 +
newOption('popupPreviewRawTemplates',  true);
 +
newOption('popupPreviewFirstParOnly',  true);
 +
newOption('popupPreviewCutHeadings',  true);
 +
newOption('popupPreviewButton',        false);
 +
newOption('popupPreviewButtonEvent',  'click');
 +
 
 +
//<NOLITE>
 +
// diffs
 +
newOption('popupPreviewDiffs',          true);
 +
newOption('popupDiffMaxLines',          100);
 +
newOption('popupDiffContextLines',      2);
 +
newOption('popupDiffContextCharacters', 40);
 +
newOption('popupDiffDates',            true);
 +
newOption('popupDiffDatePrinter',      'toLocaleString');
 +
 
 +
// edit summaries. God, these are ugly.
 +
newOption('popupFixDabsSummary',          popupString('defaultpopupFixDabsSummary') );
 +
newOption('popupExtendedRevertSummary',    popupString('defaultpopupExtendedRevertSummary') );
 +
newOption('popupTimeOffset',              null);
 +
newOption('popupRevertSummary',            popupString('defaultpopupRevertSummary') );
 +
newOption('popupRevertToPreviousSummary',  popupString('defaultpopupRevertToPreviousSummary') );
 +
newOption('popupQueriedRevertSummary',            popupString('defaultpopupQueriedRevertSummary') );
 +
newOption('popupQueriedRevertToPreviousSummary',  popupString('defaultpopupQueriedRevertToPreviousSummary') );
 +
newOption('popupFixRedirsSummary',        popupString('defaultpopupFixRedirsSummary') );
 +
newOption('popupRedlinkSummary',          popupString('defaultpopupRedlinkSummary') );
 +
newOption('popupRmDabLinkSummary',        popupString('defaultpopupRmDabLinkSummary') );
 +
//</NOLITE>
 +
// misc
 +
newOption('popupCookies',            false);
 +
newOption('popupHistoryLimit',        50);
 +
//<NOLITE>
 +
newOption('popupFilters',            [popupFilterStubDetect,    popupFilterDisambigDetect,
 +
      popupFilterPageSize,      popupFilterCountLinks,
 +
      popupFilterCountImages,    popupFilterCountCategories,
 +
      popupFilterLastModified]);
 +
newOption('extraPopupFilters',        []);
 +
newOption('popupOnEditSelection', 'cursor');
 +
newOption('popupPreviewHistory',      true);
 +
newOption('popupImageLinks',          true);
 +
newOption('popupCategoryMembers',    true);
 +
newOption('popupUserInfo',            true);
 +
newOption('popupHistoryPreviewLimit', 25);
 +
newOption('popupContribsPreviewLimit',25);
 +
newOption('popupRevDelUrl',          '//en.wikipedia.org/wiki/Wikipedia:Revision_deletion');
 +
//</NOLITE>
 +
 
 +
// new windows
 +
newOption('popupNewWindows',    false);
 +
newOption('popupLinksNewWindow', {'lastContrib': true, 'sinceMe': true});
 +
 
 +
// regexps
 +
newOption('popupDabRegexp', '(\\{\\{\\s*disambig(?!uation needed)|disambig(uation|)\\s*\\}\\}|disamb\\s*\\}\\}|dab\\s*\\}\\})|\\{\\{\\s*(((geo|hn|road?|school|number)dis)|[234][lc][acw]|(road|ship)index)(\\s*[|][^}]*)?\\s*[}][}]|is a .*disambiguation.*page');
 +
newOption('popupAnchorRegexp', 'anchors?'); //how to identify an anchors template
 +
newOption('popupStubRegexp', '(sect)?stub[}][}]|This .*-related article is a .*stub');
 +
newOption('popupImageVarsRegexp', 'image|image_(?:file|skyline|name|flag|seal)|cover|badge|logo');
 +
}
 +
// ENDFILE: options.js
 +
// STARTFILE: strings.js
 +
//<NOLITE>
 +
//////////////////////////////////////////////////
 +
// Translatable strings
 +
//////////////////////////////////////////////////
 +
//
 +
// See instructions at
 +
// http://en.wikipedia.org/wiki/Wikipedia:Tools/Navigation_popups/Translation
 +
 
 +
pg.string = {
 +
/////////////////////////////////////
 +
// summary data, searching etc.
 +
/////////////////////////////////////
 +
'article': 'article',
 +
'category': 'category',
 +
'categories': 'categories',
 +
'image': 'image',
 +
'images': 'images',
 +
'stub': 'stub',
 +
'section stub': 'section stub',
 +
'Empty page': 'Empty page',
 +
'kB': 'kB',
 +
'bytes': 'bytes',
 +
'day': 'day',
 +
'days': 'days',
 +
'hour': 'hour',
 +
'hours': 'hours',
 +
'minute': 'minute',
 +
'minutes': 'minutes',
 +
'second': 'second',
 +
'seconds': 'seconds',
 +
'week': 'week',
 +
'weeks': 'weeks',
 +
'search': 'search',
 +
'SearchHint': 'Find English Wikipedia articles containing %s',
 +
'web': 'web',
 +
'global': 'global',
 +
'globalSearchHint': 'Search across Wikipedias in different languages for %s',
 +
'googleSearchHint': 'Google for %s',
 +
/////////////////////////////////////
 +
// article-related actions and info
 +
// (some actions also apply to user pages)
 +
/////////////////////////////////////
 +
'actions': 'actions', ///// view articles and view talk
 +
'popupsMenu': 'popups',
 +
'togglePreviewsHint': 'Toggle preview generation in popups on this page',
 +
'enable previews': 'enable previews',
 +
'disable previews': 'disable previews',
 +
'toggle previews': 'toggle previews',
 +
'show preview': 'show preview',
 +
'reset': 'reset',
 +
'more...': 'more...',
 +
'disable': 'disable popups',
 +
'disablePopupsHint': 'Disable popups on this page. Reload page to re-enable.',
 +
'historyfeedHint': 'RSS feed of recent changes to this page',
 +
'purgePopupsHint': 'Reset popups, clearing all cached popup data.',
 +
'PopupsHint': 'Reset popups, clearing all cached popup data.',
 +
'spacebar': 'space',
 +
'view': 'view',
 +
'view article': 'view article',
 +
'viewHint': 'Go to %s',
 +
'talk': 'talk',
 +
'talk page': 'talk page',
 +
'this&nbsp;revision': 'this&nbsp;revision',
 +
'revision %s of %s': 'revision %s of %s',
 +
'Revision %s of %s': 'Revision %s of %s',
 +
'the revision prior to revision %s of %s': 'the revision prior to revision %s of %s',
 +
'Toggle image size': 'Click to toggle image size',
 +
'del': 'del', ///// delete, protect, move
 +
'delete': 'delete',
 +
'deleteHint': 'Delete %s',
 +
'undeleteShort': 'un',
 +
'UndeleteHint': 'Show the deletion history for %s',
 +
'protect': 'protect',
 +
'protectHint': 'Restrict editing rights to %s',
 +
'unprotectShort': 'un',
 +
'unprotectHint': 'Allow %s to be edited by anyone again',
 +
'move': 'move',
 +
'move page': 'move page',
 +
'MovepageHint': 'Change the title of %s',
 +
'edit': 'edit',   ///// edit articles and talk
 +
'edit article': 'edit article',
 +
'editHint': 'Change the content of %s',
 +
'edit talk': 'edit talk',
 +
'new': 'new',
 +
'new topic': 'new topic',
 +
'newSectionHint': 'Start a new section on %s',
 +
'null edit': 'null edit',
 +
'nullEditHint': 'Submit an edit to %s, making no changes ',
 +
'hist': 'hist',   ///// history, diffs, editors, related
 +
'history': 'history',
 +
'historyHint': 'List the changes made to %s',
 +
'last': 'last',
 +
'lastEdit': 'lastEdit',
 +
'mark patrolled': 'mark patrolled',
 +
'markpatrolledHint': 'Mark this edit as patrolled',
 +
'show last edit': 'most recent edit',
 +
'Show the last edit': 'Show the effects of the most recent change',
 +
'lastContrib': 'lastContrib',
 +
'last set of edits': 'latest edits',
 +
'lastContribHint': 'Show the net effect of changes made by the last editor',
 +
'cur': 'cur',
 +
'diffCur': 'diffCur',
 +
'Show changes since revision %s': 'Show changes since revision %s',
 +
'%s old': '%s old', // as in 4 weeks old
 +
'oldEdit': 'oldEdit',
 +
'purge': 'purge',
 +
'purgeHint': 'Demand a fresh copy of %s',
 +
'raw': 'source',
 +
'rawHint': 'Download the source of %s',
 +
'render': 'simple',
 +
'renderHint': 'Show a plain HTML version of %s',
 +
'Show the edit made to get revision': 'Show the edit made to get revision',
 +
'sinceMe': 'sinceMe',
 +
'changes since mine': 'diff my edit',
 +
'sinceMeHint': 'Show changes since my last edit',
 +
'Couldn\'t find an edit by %s\nin the last %s edits to\n%s': 'Couldn\'t find an edit by %s\nin the last %s edits to\n%s',
 +
'eds': 'eds',
 +
'editors': 'editors',
 +
'editorListHint': 'List the users who have edited %s',
 +
'related': 'related',
 +
'relatedChanges': 'relatedChanges',
 +
'related changes': 'related changes',
 +
'RecentchangeslinkedHint': 'Show changes in articles related to %s',
 +
'editOld': 'editOld',   ///// edit old version, or revert
 +
'rv': 'rv',
 +
'revert': 'revert',
 +
'revertHint': 'Revert to %s',
 +
'defaultpopupRedlinkSummary': 'Removing link to empty page [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupFixDabsSummary': 'Disambiguate [[%s]] to [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupFixRedirsSummary': 'Redirect bypass from [[%s]] to [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupExtendedRevertSummary': 'Revert to revision dated %s by %s, oldid %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupRevertToPreviousSummary': 'Revert to the revision prior to revision %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupRevertSummary': 'Revert to revision %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupQueriedRevertToPreviousSummary': 'Revert to the revision prior to revision $1 dated $2 by $3 using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupQueriedRevertSummary': 'Revert to revision $1 dated $2 by $3 using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'defaultpopupRmDabLinkSummary': 'Remove link to dab page [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
 +
'Redirects': 'Redirects', // as in Redirects to ...
 +
' to ': ' to ',   // as in Redirects to ...
 +
'Bypass redirect': 'Bypass redirect',
 +
'Fix this redirect': 'Fix this redirect',
 +
'disambig': 'disambig',   ///// add or remove dab etc.
 +
'disambigHint': 'Disambiguate this link to [[%s]]',
 +
'Click to disambiguate this link to:': 'Click to disambiguate this link to:',
 +
'remove this link': 'remove this link',
 +
'remove all links to this page from this article': 'remove all links to this page from this article',
 +
'remove all links to this disambig page from this article': 'remove all links to this disambig page from this article',
 +
'mainlink': 'mainlink',   ///// links, watch, unwatch
 +
'wikiLink': 'wikiLink',
 +
'wikiLinks': 'wikiLinks',
 +
'links here': 'links here',
 +
'whatLinksHere': 'whatLinksHere',
 +
'what links here': 'what links here',
 +
'WhatlinkshereHint': 'List the pages that are hyperlinked to %s',
 +
'unwatchShort': 'un',
 +
'watchThingy': 'watch',  // called watchThingy because {}.watch is a function
 +
'watchHint': 'Add %s to my watchlist',
 +
'unwatchHint': 'Remove %s from my watchlist',
 +
'Only found one editor: %s made %s edits': 'Only found one editor: %s made %s edits',
 +
'%s seems to be the last editor to the page %s': '%s seems to be the last editor to the page %s',
 +
'rss': 'rss',
 +
/////////////////////////////////////
 +
// diff previews
 +
/////////////////////////////////////
 +
'Diff truncated for performance reasons': 'Diff truncated for performance reasons',
 +
'Old revision': 'Old revision',
 +
'New revision': 'New revision',
 +
'Something went wrong :-(': 'Something went wrong :-(',
 +
'Empty revision, maybe non-existent': 'Empty revision, maybe non-existent',
 +
'Unknown date': 'Unknown date',
 +
/////////////////////////////////////
 +
// other special previews
 +
/////////////////////////////////////
 +
'Empty category': 'Empty category',
 +
'Category members (%s shown)': 'Category members (%s shown)',
 +
'No image links found': 'No image links found',
 +
'File links': 'File links',
 +
'No image found': 'No image found',
 +
'Image from Commons': 'Image from Commons',
 +
'Description page': 'Description page',
 +
'Alt text:': 'Alt text:',
 +
'revdel':'Hidden revision',
 +
/////////////////////////////////////
 +
// user-related actions and info
 +
/////////////////////////////////////
 +
'user': 'user',   ///// user page, talk, email, space
 +
'user&nbsp;page': 'user&nbsp;page',
 +
'user talk': 'user talk',
 +
'edit user talk': 'edit user talk',
 +
'leave comment': 'leave comment',
 +
'email': 'email',
 +
'email user': 'email user',
 +
'EmailuserHint': 'Send an email to %s',
 +
'space': 'space', // short form for userSpace link
 +
'PrefixIndexHint': 'Show pages in the userspace of %s',
 +
'count': 'count', ///// contributions, log
 +
'edit counter': 'edit counter',
 +
'editCounterLinkHint': 'Count the contributions made by %s',
 +
'contribs': 'contribs',
 +
'contributions': 'contributions',
 +
'deletedContribs': 'deleted contributions',
 +
'DeletedcontributionsHint': 'List deleted edits made by %s',
 +
'ContributionsHint': 'List the contributions made by %s',
 +
'log': 'log',
 +
'user log': 'user log',
 +
'userLogHint': 'Show %s\'s user log',
 +
'arin': 'ARIN lookup', ///// ARIN lookup, block user or IP
 +
'Look up %s in ARIN whois database': 'Look up %s in the ARIN whois database',
 +
'unblockShort': 'un',
 +
'block': 'block',
 +
'block user': 'block user',
 +
'IpblocklistHint': 'Unblock %s',
 +
'BlockipHint': 'Prevent %s from editing',
 +
'block log': 'block log',
 +
'blockLogHint': 'Show the block log for %s',
 +
'protectLogHint': 'Show the protection log for %s',
 +
'pageLogHint': 'Show the page log for %s',
 +
'deleteLogHint': 'Show the deletion log for %s',
 +
'Invalid %s %s': 'The option %s is invalid: %s',
 +
'No backlinks found': 'No backlinks found',
 +
' and more': ' and more',
 +
'undo': 'undo',
 +
'undoHint': 'undo this edit',
 +
'Download preview data': 'Download preview data',
 +
'Invalid or IP user': 'Invalid or IP user',
 +
'Not a registered username': 'Not a registered username',
 +
'BLOCKED': 'BLOCKED',
 +
' edits since: ': ' edits since: ',
 +
/////////////////////////////////////
 +
// Autoediting
 +
/////////////////////////////////////
 +
'Enter a non-empty edit summary or press cancel to abort': 'Enter a non-empty edit summary or press cancel to abort',
 +
'Failed to get revision information, please edit manually.\n\n': 'Failed to get revision information, please edit manually.\n\n',
 +
'The %s button has been automatically clicked. Please wait for the next page to load.': 'The %s button has been automatically clicked. Please wait for the next page to load.',
 +
'Could not find button %s. Please check the settings in your javascript file.': 'Could not find button %s. Please check the settings in your javascript file.',
 +
/////////////////////////////////////
 +
// Popups setup
 +
/////////////////////////////////////
 +
'Open full-size image': 'Open full-size image',
 +
'zxy': 'zxy',
 +
'autoedit_version': 'np20140416'
 +
};
 +
 
 +
 
 +
function popupString(str) {
 +
if (typeof popupStrings != 'undefined' && popupStrings && popupStrings[str]) { return popupStrings[str]; }
 +
if (pg.string[str]) { return pg.string[str]; }
 +
return str;
 +
}
 +
 
 +
 
 +
function tprintf(str,subs) {
 +
if (typeof subs != typeof []) { subs = [subs]; }
 +
return simplePrintf(popupString(str), subs);
 +
}
 +
 
 +
//</NOLITE>
 +
// ENDFILE: strings.js
 +
 
 +
 
 +
////////////////////////////////////////////////////////////////////
 +
// Run things
 +
////////////////////////////////////////////////////////////////////
 +
 
 +
 
 +
// For some reason popups requires a fully loaded page jQuery.ready(...) causes problems for some.
 +
// The old addOnloadHook did something similar to the below
 +
if (document.readyState=="complete")
 +
autoEdit(); //will setup popups
 +
else
 +
$( window ).on( 'load', autoEdit );
 +
 
 +
 
 +
// Support for MediaWiki's live preview, VisualEditor's saves and Echo's flyout.
 +
( function () {
 +
var once = true;
 +
function dynamicContentHandler( $content ) {
 +
// Try to detect the hook fired on initial page load and disregard
 +
// it, we already hook to onload (possibly to different parts of
 +
// page - it's configurable) and running twice might be bad. Ugly…
 +
if ( $content.attr( 'id' ) == 'mw-content-text' ) {
 +
if ( once ) {
 +
once = false;
 +
return;
 +
}
 +
}
 +
 
 +
function doIt () {
 +
$content.each( function () {
 +
this.ranSetupTooltipsAlready = false;
 +
setupTooltips( this );
 +
} );
 +
};
 +
 
 +
if ( !setupPopups.completed ) {
 +
setupPopups( doIt );
 +
} else {
 +
doIt();
 
}
 
}
 
setupPopups( doIt );
 
 
}
 
}
  
Line 7,169: Line 7,737:
 
});
 
});
 
} )();
 
} )();
 
});
 
// ENDFILE: run.js
 

Revision as of 18:11, 5 October 2019

// STARTFILE: main.js
// **********************************************************************
// **                                                                  **
// **             changes to this file affect many users.              **
// **           please discuss on the talk page before editing         **
// **                                                                  **
// **********************************************************************
// **                                                                  **
// ** if you do edit this file, be sure that your editor recognizes it **
// ** as utf8, or the weird and wonderful characters in the namespaces **
// **   below will be completely broken. You can check with the show   **
// **            changes button before submitting the edit.            **
// **                      test: مدیا מיוחד Мэдыя                      **
// **                                                                  **
// **********************************************************************

//////////////////////////////////////////////////
// Globals
//

// Trying to shove as many of these as possible into the pg (popup globals) object
function pg(){} // dummy to stop errors
window.pg = {
	re: {},               // regexps
	ns: {},               // namespaces
	string: {},           // translatable strings
	wiki: {},             // local site info
	misc: {},             // YUCK PHOOEY
	option: {},           // options, see newOption etc
	optionDefault: {},    // default option values
	flag: {},             // misc flags
	cache: {},            // page and image cache
	structures: {},       // navlink structures
	timer: {},            // all sorts of timers (too damn many)
	counter: {},          // .. and all sorts of counters
	current: {},          // state info
	endoflist: null
};
window.pop = {	              // wrap various functions in here
	init: {},
	util: {},
	endoflist: null
};
function popupsReady() {
	if (!window.pg) { return false; }
	if (!pg.flag) { return false; }
	if (!pg.flag.finishedLoading) { return false; }
	return true;
}

/// Local Variables: ///
/// mode:c ///
/// End: ///
// ENDFILE: main.js
// STARTFILE: actions.js
function setupTooltips(container, remove, force, popData) {
	log('setupTooltips, container='+container+', remove='+remove);
	if (!container) {
//<NOLITE>
		// the main initial call
		if (getValueOf('popupOnEditSelection') && window.doSelectionPopup && document && document.editform && document.editform.wpTextbox1) {
			document.editform.wpTextbox1.onmouseup=doSelectionPopup;
		}
//</NOLITE>
		// article/content is a structure-dependent thing
		container = defaultPopupsContainer();
	}

	if (!remove && !force && container.ranSetupTooltipsAlready) { return; }
	container.ranSetupTooltipsAlready = !remove;

	var anchors;
	anchors=container.getElementsByTagName('A');
	setupTooltipsLoop(anchors, 0, 250, 100, remove, popData);
}

function defaultPopupsContainer() {
	if (getValueOf('popupOnlyArticleLinks')) {
		return document.getElementById('mw_content') || 
			document.getElementById('content') ||
			document.getElementById('article') || document;
	}
	return  document;
}

function setupTooltipsLoop(anchors,begin,howmany,sleep, remove, popData) {
	log(simplePrintf('setupTooltipsLoop(%s,%s,%s,%s,%s)', arguments));
	var finish=begin+howmany;
	var loopend = min(finish, anchors.length);
	var j=loopend - begin;
	log ('setupTooltips: anchors.length=' + anchors.length + ', begin=' + begin +
		 ', howmany=' + howmany + ', loopend=' + loopend + ', remove=' + remove);
	var doTooltip= remove ? removeTooltip : addTooltip;
	// try a faster (?) loop construct
	if (j > 0) {
		do {
			var a=anchors[loopend - j];
			if (typeof a==='undefined' || !a || !a.href) {
				log('got null anchor at index ' + loopend - j);
				continue;
			}
			doTooltip(a, popData);
		} while (--j);
	}
	if (finish < anchors.length) {
		setTimeout(function() {
				setupTooltipsLoop(anchors,finish,howmany,sleep,remove,popData);},
			sleep);
	} else {
		if ( !remove && ! getValueOf('popupTocLinks')) { rmTocTooltips(); }
		pg.flag.finishedLoading=true;
	}
}

// eliminate popups from the TOC
// This also kills any onclick stuff that used to be going on in the toc
function rmTocTooltips() {
	var toc=document.getElementById('toc');
	if (toc) {
		var tocLinks=toc.getElementsByTagName('A');
		var tocLen = tocLinks.length;
		for (j=0; j<tocLen; ++j) {
			removeTooltip(tocLinks[j], true);
		}
	}
}

function addTooltip(a, popData) {
	if ( !isPopupLink(a) ) { return; }
	a.onmouseover=mouseOverWikiLink;
	a.onmouseout= mouseOutWikiLink;
	a.onmousedown = killPopup;
	a.hasPopup = true;
	a.popData = popData;
}

function removeTooltip(a) {
	if ( !a.hasPopup ) { return; }
	a.onmouseover = null;
	a.onmouseout = null;
	if (a.originalTitle) { a.title = a.originalTitle; }
	a.hasPopup=false;
}

function removeTitle(a) {
	if (!a.originalTitle) {
		a.originalTitle=a.title;
	}
		a.title='';
}

function restoreTitle(a) {
	if ( a.title || !a.originalTitle ) { return; }
	a.title = a.originalTitle;
}

function registerHooks(np) {
	var popupMaxWidth=getValueOf('popupMaxWidth');

	if (typeof popupMaxWidth === 'number') {
		var setMaxWidth = function () {
			np.mainDiv.style.maxWidth = popupMaxWidth + 'px';
			np.maxWidth = popupMaxWidth;

			try {
				// hack for IE
				// see http://www.svendtofte.com/code/max_width_in_ie/
				// use setExpression as documented here on msdn: http://tinyurl dot com/dqljn
	
				if (np.mainDiv.style.setExpression) {
					np.mainDiv.style.setExpression(
						'width', 'document.body.clientWidth > ' +
						popupMaxWidth + ' ? "' +popupMaxWidth + 'px": "auto"');
				}
			}
			catch (errors) {
				errlog( "Running on IE8 are we not?: " + errors );
			}
		};
		np.addHook(setMaxWidth, 'unhide', 'before');
	}
//<NOLITE>
	if (window.addPopupShortcuts && window.rmPopupShortcuts) {
		np.addHook(addPopupShortcuts, 'unhide', 'after');
		np.addHook(rmPopupShortcuts, 'hide', 'before');
	}
//</NOLITE>
}


function mouseOverWikiLink(evt) {
	if (!window.popupsReady || !window.popupsReady()) { return; }
	if (!evt && window.event) {evt=window.event;}
	return mouseOverWikiLink2(this, evt);
}

function footnoteTarget(a) {
	var aTitle=Title.fromAnchor(a);
	// We want ".3A" rather than "%3A" or "?" here, so use the anchor property directly
	var anch = aTitle.anchor;
	if ( ! /^(cite_note-|_note-|endnote)/.test(anch) ) { return false; }

	var lTitle=Title.fromURL(location.href);
	if ( lTitle.toString(true) !== aTitle.toString(true) ) { return false; }

	var el=document.getElementById(anch);
	while ( el && typeof el.nodeName === 'string') {
		var nt = el.nodeName.toLowerCase();
		if ( nt === 'li' ) { return el; }
		else if ( nt === 'body' ) { return false; }
		else if ( el.parentNode ) { el=el.parentNode; }
		else { return false; }
	}
	return false;
}

function footnotePreview(x, navpop) {
	setPopupHTML('<hr />' + x.innerHTML, 'popupPreview',  navpop.idNumber, getValueOf('popupSubpopups')
		? function() { setupTooltips(document.getElementById('popupPreview' + navpop.idNumber)); }
		: null);
}

// var modid=0;
// if(!window.opera) { window.opera={postError: console.log}; }

function modifierKeyHandler(a) {
	return function(evt) {
		//		opera.postError('modifierKeyHandler called' + (++modid));
		//		opera.postError(''+evt + modid);
		//		for (var i in evt) {
		//			opera.postError('' + modid + ' ' + i + ' ' + evt[i]);
		//		}
		//		opera.postError(''+evt.ctrlKey + modid);
		var mod=getValueOf('popupModifier');
		if (!mod) { return true; }

		if (!evt && window.event) {evt=window.event;}
		//		opera.postError('And now....'+modid);
		//		opera.postError(''+evt+modid);
		//		opera.postError(''+evt.ctrlKey+modid);

		var modPressed = modifierPressed(evt);
		var action = getValueOf('popupModifierAction');

		// FIXME: probable bug - modifierPressed should be modPressed below?
		if ( action === 'disable' && modifierPressed ) { return true; }
		if ( action === 'enable' && !modifierPressed ) { return true; }

		mouseOverWikiLink2(a, evt);
	};
}

function modifierPressed(evt) {
		var mod=getValueOf('popupModifier');
		if (!mod) { return false; }

		if (!evt && window.event) {evt=window.event;}
//		opera.postError('And now....'+modid);
//		opera.postError(''+evt+modid);
//		opera.postError(''+evt.ctrlKey+modid);

		return ( evt && mod && evt[mod.toLowerCase() + 'Key'] );

}

function dealWithModifier(a,evt) {
	if (!getValueOf('popupModifier')) { return false; }
	var action = getValueOf('popupModifierAction');
	if ( action ==  'enable' && !modifierPressed(evt) ||
	     action == 'disable' &&  modifierPressed(evt) ) {
		// if the modifier is needed and not pressed, listen for it until
		// we mouseout of this link.
		restoreTitle(a);
		var addHandler='addEventListener';
		var rmHandler='removeEventListener';
		var on='';
		if (!document.addEventListener) {
			addHandler='attachEvent';
			rmHandler='detachEvent';
			on='on';
		}
		if (!document[addHandler]) { // forget it
			return;
		}

		a.modifierKeyHandler=modifierKeyHandler(a);

		switch (action) {
		case 'enable':
			document[addHandler](on+'keydown', a.modifierKeyHandler, false);
			a[addHandler](on+'mouseout', function() {
					document[rmHandler](on+'keydown',
								a.modifierKeyHandler, false);
				}, true);
			break;
		case 'disable':
			document[addHandler](on+'keyup', a.modifierKeyHandler, false);
		}

		return true;
	}
	return false;
}

function mouseOverWikiLink2(a, evt) {
	if (dealWithModifier(a,evt)) { return; }
	if ( getValueOf('removeTitles') ) { removeTitle(a); }
	if ( a==pg.current.link && a.navpopup && a.navpopup.isVisible() ) { return; }
	pg.current.link=a;

	if (getValueOf('simplePopups') && pg.option.popupStructure===null) {
		// reset *default value* of popupStructure
		setDefault('popupStructure', 'original');
	}

	var article=(new Title()).fromAnchor(a);
	// set global variable (ugh) to hold article (wikipage)
	pg.current.article = article;

	if (!a.navpopup) {
		// FIXME: this doesn't behave well if you mouse out of a popup
		// directly into a link with the same href
		if (pg.current.linksHash[a.href] && false) {
			a.navpopup = pg.current.linksHash[a.href];
		}
		else {
			a.navpopup=newNavpopup(a, article);
			pg.current.linksHash[a.href] = a.navpopup;
			pg.current.links.push(a);
		}
	}
	if (a.navpopup.pending===null || a.navpopup.pending!==0) {
		// either fresh popups or those with unfinshed business are redone from scratch
		simplePopupContent(a, article);
	}
	a.navpopup.showSoonIfStable(a.navpopup.delay);

	getValueOf('popupInitialWidth');

	clearInterval(pg.timer.checkPopupPosition);
	pg.timer.checkPopupPosition=setInterval(checkPopupPosition, 600);

	if(getValueOf('simplePopups')) {
		if (getValueOf('popupPreviewButton') && !a.simpleNoMore) {
			var d=document.createElement('div');
			d.className='popupPreviewButtonDiv';
			var s=document.createElement('span');
			d.appendChild(s);
			s.className='popupPreviewButton';
			s['on' + getValueOf('popupPreviewButtonEvent')] = function() {
				a.simpleNoMore=true;
				nonsimplePopupContent(a,article);
			};
			s.innerHTML=popupString('show preview');
			setPopupHTML(d, 'popupPreview', a.navpopup.idNumber);
		}
		return;
	}

	if (a.navpopup.pending!==0 ) {
		nonsimplePopupContent(a, article);
	}
}

// simplePopupContent: the content that is shown even when simplePopups is true
function simplePopupContent(a, article) {
	/* FIXME hack */ a.navpopup.hasPopupMenu=false;
	a.navpopup.setInnerHTML(popupHTML(a));
	fillEmptySpans({navpopup:a.navpopup});

	if (getValueOf('popupDraggable'))
	{
		var dragHandle = getValueOf('popupDragHandle') || null;
		if (dragHandle && dragHandle != 'all') {
			dragHandle += a.navpopup.idNumber;
		}
		setTimeout(function(){a.navpopup.makeDraggable(dragHandle);}, 150);
	}

//<NOLITE>
	if (getValueOf('popupRedlinkRemoval') && a.className=='new') {
		setPopupHTML('<br>'+popupRedlinkHTML(article), 'popupRedlink', a.navpopup.idNumber);
	}
//</NOLITE>
}

function debugData(navpopup) {
	if(getValueOf('popupDebugging') && navpopup.idNumber) {
		setPopupHTML('idNumber='+navpopup.idNumber + ', pending=' + navpopup.pending,
				 'popupError', navpopup.idNumber);
	}
}

function newNavpopup(a, article) {
	var navpopup = new Navpopup();
	navpopup.fuzz=5;
	navpopup.delay=getValueOf('popupDelay')*1000;
	// increment global counter now
	navpopup.idNumber = ++pg.idNumber;
	navpopup.parentAnchor = a;
	navpopup.parentPopup = (a.popData && a.popData.owner);
	navpopup.article = article;
	registerHooks(navpopup);
	return navpopup;
}


function nonsimplePopupContent(a, article) {
	var diff=null, history=null;
	var params=parseParams(a.href);
	var oldid=(typeof params.oldid=='undefined' ? null : params.oldid);
//<NOLITE>
	if(getValueOf('popupPreviewDiffs') && window.loadDiff) {
		diff=params.diff;
	}
	if(getValueOf('popupPreviewHistory')) {
		history=(params.action=='history');
	}
//</NOLITE>
	a.navpopup.pending=0;
	var x;
	
	if (x=footnoteTarget(a)) {
		footnotePreview(x, a.navpopup);
//<NOLITE>
	} else if ( diff || diff === 0 ) {
		loadDiff(article, oldid, diff, a.navpopup);
	} else if ( history ) {
		loadAPIPreview('history', article, a.navpopup);
	} else if ( pg.re.contribs.test(a.href) ) {
		loadAPIPreview('contribs', article, a.navpopup);
	} else if ( pg.re.backlinks.test(a.href) ) {
		loadAPIPreview('backlinks', article, a.navpopup);
	} else if ( // FIXME should be able to get all preview combinations with options
		article.namespaceId()==pg.nsImageId &&
		( getValueOf('imagePopupsForImages') || ! anchorContainsImage(a) )
		) {
		loadAPIPreview('imagepagepreview', article, a.navpopup);
		loadImage(article, a.navpopup);
//</NOLITE>
	} else {
		if (article.namespaceId() == pg.nsCategoryId &&
				getValueOf('popupCategoryMembers')) {
			loadAPIPreview('category', article, a.navpopup);
		} else if ((article.namespaceId() == pg.nsUserId || article.namespaceId() == pg.nsUsertalkId) &&
				getValueOf('popupUserInfo')) {
			loadAPIPreview('userinfo', article, a.navpopup);
		}
		startArticlePreview(article, oldid, a.navpopup);
	}
}

function pendingNavpopTask(navpop) {
	if (navpop && navpop.pending===null) { navpop.pending=0; }
	++navpop.pending;
	debugData(navpop);
}

function completedNavpopTask(navpop) {
	if (navpop && navpop.pending) { --navpop.pending; }
	debugData(navpop);
}

function startArticlePreview(article, oldid, navpop) {
	navpop.redir=0;
	loadPreview(article, oldid, navpop);
}

function loadPreview(article, oldid, navpop) {
	pendingNavpopTask(navpop);
	if (!navpop.redir) { navpop.originalArticle=article; }
	if (!navpop.visible && getValueOf('popupLazyDownloads')) {
		var id=(navpop.redir) ? 'DOWNLOAD_PREVIEW_REDIR_HOOK' : 'DOWNLOAD_PREVIEW_HOOK';
		navpop.addHook(function() {
				getWiki(article, insertPreview, oldid, navpop);
				return true; }, 'unhide', 'before', id);
	} else {
		getWiki(article, insertPreview, oldid, navpop);
	}
}

function loadPreviewFromRedir(redirMatch, navpop) {
	// redirMatch is a regex match
	var target = new Title().fromWikiText(redirMatch[2]);
	// overwrite (or add) anchor from original target
	// mediawiki does overwrite; eg [[User:Lupin/foo3#Done]]
	if ( navpop.article.anchor ) { target.anchor = navpop.article.anchor; }
	var trailingRubbish=redirMatch[4];
	navpop.redir++;
	navpop.redirTarget=target;
//<NOLITE>
	if (window.redirLink) {
		var warnRedir = redirLink(target, navpop.article);
		setPopupHTML(warnRedir, 'popupWarnRedir', navpop.idNumber);
	}
//</NOLITE>
	navpop.article=target;
	fillEmptySpans({redir: true, redirTarget: target, navpopup:navpop});
	return loadPreview(target, null,  navpop);
}

function insertPreview(download) {
	if (!download.owner) { return; }

	var redirMatch = pg.re.redirect.exec(download.data);
	if (download.owner.redir===0 && redirMatch) {
		completedNavpopTask(download.owner);
		loadPreviewFromRedir(redirMatch, download.owner);
		return;
	}

	if (download.owner.visible || !getValueOf('popupLazyPreviews')) {
		insertPreviewNow(download);
	} else {
		var id=(download.owner.redir) ? 'PREVIEW_REDIR_HOOK' : 'PREVIEW_HOOK';
		download.owner.addHook( function(){insertPreviewNow(download); return true;},
					'unhide', 'after', id );
	}
}

function insertPreviewNow(download) {
	if (!download.owner) { return; }
	var wikiText=download.data;
	var navpop=download.owner;
	completedNavpopTask(navpop);
	var art=navpop.redirTarget || navpop.originalArticle;

//<NOLITE>
	makeFixDabs(wikiText, navpop);
	if (getValueOf('popupSummaryData') && window.getPageInfo) {
		var info=getPageInfo(wikiText, download);
		setPopupTrailer(getPageInfo(wikiText, download), navpop.idNumber);
	}

	var imagePage='';
	if (art.namespaceId()==pg.nsImageId) { imagePage=art.toString(); }
	else { imagePage=getValidImageFromWikiText(wikiText); }
	if(imagePage) { loadImage(Title.fromWikiText(imagePage), navpop); }
//</NOLITE>

	if (getValueOf('popupPreviews')) { insertArticlePreview(download, art, navpop); }

}

function insertArticlePreview(download, art, navpop) {
	if (download && typeof download.data == typeof ''){
		if (art.namespaceId()==pg.nsTemplateId && getValueOf('popupPreviewRawTemplates')) {
			// FIXME compare/consolidate with diff escaping code for wikitext
			var h='<hr /><span style="font-family: monospace;">' + download.data.entify().split('\\n').join('<br />\\n') + '</span>';
			setPopupHTML(h, 'popupPreview', navpop.idNumber);
		}
		else {
			var p=prepPreviewmaker(download.data, art, navpop);
			p.showPreview();
		}
	}
}

function prepPreviewmaker(data, article, navpop) {
	// deal with tricksy anchors
	var d=anchorize(data, article.anchorString());
	var urlBase=joinPath([pg.wiki.articlebase, article.urlString()]);
	var p=new Previewmaker(d, urlBase, navpop);
	return p;
}


// Try to imitate the way mediawiki generates HTML anchors from section titles
function anchorize(d, anch) {
	if (!anch) { return d; }
	var anchRe=RegExp('(?:=+\\s*' + literalizeRegex(anch).replace(/[_ ]/g, '[_ ]') + '\\s*=+|\\{\\{\\s*'+getValueOf('popupAnchorRegexp')+'\\s*(?:\\|[^|}]*)*?\\s*'+literalizeRegex(anch)+'\\s*(?:\\|[^}]*)?\}\})');
	var match=d.match(anchRe);
	if(match && match.length > 0 && match[0]) { return d.substring(d.indexOf(match[0])); }

	// now try to deal with == foo [[bar|baz]] boom == -> #foo_baz_boom
	var lines=d.split('\n');
	for (var i=0; i<lines.length; ++i) {
		lines[i]=lines[i].replace(RegExp('[[]{2}([^|\\]]*?[|])?(.*?)[\\]]{2}', 'g'), '$2')
			.replace(/'''([^'])/g, '$1').replace(RegExp("''([^'])", 'g'), '$1');
		if (lines[i].match(anchRe)) {
			return d.split('\n').slice(i).join('\n').replace(RegExp('^[^=]*'), '');
		}
	}
	return d;
}

function killPopup() {
	if (getValueOf('popupShortcutKeys') && window.rmPopupShortcuts) { rmPopupShortcuts(); }
	if (!pg) { return; }
	pg.current.link && pg.current.link.navpopup && pg.current.link.navpopup.banish();
	pg.current.link=null;
	abortAllDownloads();
	if (pg.timer.checkPopupPosition !== null) {
		clearInterval(pg.timer.checkPopupPosition);
		pg.timer.checkPopupPosition=null;
	}
	return true; // preserve default action
}
// ENDFILE: actions.js
// STARTFILE: domdrag.js
/**
   @fileoverview
   The {@link Drag} object, which enables objects to be dragged around.

   <pre>
   *************************************************
   dom-drag.js
   09.25.2001
   www.youngpup.net
   **************************************************
   10.28.2001 - fixed minor bug where events
   sometimes fired off the handle, not the root.
   *************************************************
   Pared down, some hooks added by [[User:Lupin]]

   Copyright Aaron Boodman.
   Saying stupid things daily since March 2001.
   </pre>
*/

/**
   Creates a new Drag object. This is used to make various DOM elements draggable.
   @constructor
*/
function Drag () {
	/**
	   Condition to determine whether or not to drag. This function should take one parameter, an Event.
	   To disable this, set it to <code>null</code>.
	   @type Function
	*/
	this.startCondition = null;
	/**
	   Hook to be run when the drag finishes. This is passed the final coordinates of
	   the dragged object (two integers, x and y). To disables this, set it to <code>null</code>.
	   @type Function
	*/
	this.endHook = null;
}

/**
   Gets an event in a cross-browser manner.
   @param {Event} e
   @private
*/
Drag.prototype.fixE = function(e) {
	if (typeof e == 'undefined') { e = window.event; }
	if (typeof e.layerX == 'undefined') { e.layerX = e.offsetX; }
	if (typeof e.layerY == 'undefined') { e.layerY = e.offsetY; }
	return e;
};
/**
   Initialises the Drag instance by telling it which object you want to be draggable, and what you want to drag it by.
   @param {DOMElement} o The "handle" by which <code>oRoot</code> is dragged.
   @param {DOMElement} oRoot The object which moves when <code>o</code> is dragged, or <code>o</code> if omitted.
*/
Drag.prototype.init = function(o, oRoot) {
	var dragObj	  = this;
	this.obj = o;
	o.onmousedown	= function(e) { dragObj.start.apply( dragObj, [e]); };
	o.dragging	   = false;
	o.popups_draggable	  = true;
	o.hmode		  = true;
	o.vmode		  = true;

	o.root = oRoot && oRoot !== null ? oRoot : o ;

	if (isNaN(parseInt(o.root.style.left, 10))) { o.root.style.left   = "0px"; }
	if (isNaN(parseInt(o.root.style.top,  10))) { o.root.style.top	= "0px"; }

	o.root.onthisStart  = function(){};
	o.root.onthisEnd	= function(){};
	o.root.onthis	   = function(){};
};

/**
   Starts the drag.
   @private
   @param {Event} e
*/
Drag.prototype.start = function(e) {
	var o = this.obj; // = this;
	e = this.fixE(e);
	if (this.startCondition && !this.startCondition(e)) { return; }
	var y = parseInt(o.vmode ? o.root.style.top  : o.root.style.bottom, 10);
	var x = parseInt(o.hmode ? o.root.style.left : o.root.style.right,  10);
	o.root.onthisStart(x, y);

	o.lastMouseX	= e.clientX;
	o.lastMouseY	= e.clientY;

	var dragObj	  = this;
	o.onmousemoveDefault	= document.onmousemove;
	o.dragging			  = true;
	document.onmousemove	= function(e) { dragObj.drag.apply( dragObj, [e] ); };
	document.onmouseup	  = function(e) { dragObj.end.apply( dragObj, [e] ); };
	return false;
};
/**
   Does the drag.
   @param {Event} e
   @private
*/
Drag.prototype.drag = function(e) {
	e = this.fixE(e);
	var o = this.obj;

	var ey	= e.clientY;
	var ex	= e.clientX;
	var y = parseInt(o.vmode ? o.root.style.top  : o.root.style.bottom, 10);
	var x = parseInt(o.hmode ? o.root.style.left : o.root.style.right,  10 );
	var nx, ny;

	nx = x + ((ex - o.lastMouseX) * (o.hmode ? 1 : -1));
	ny = y + ((ey - o.lastMouseY) * (o.vmode ? 1 : -1));

	this.obj.root.style[o.hmode ? "left" : "right"] = nx + "px";
	this.obj.root.style[o.vmode ? "top" : "bottom"] = ny + "px";
	this.obj.lastMouseX	= ex;
	this.obj.lastMouseY	= ey;

	this.obj.root.onthis(nx, ny);
	return false;
};

/**
   Ends the drag.
   @private
*/
Drag.prototype.end = function()  {
	document.onmousemove=this.obj.onmousemoveDefault;
	document.onmouseup   = null;
	this.obj.dragging	= false;
	if (this.endHook) {
		this.endHook( parseInt(this.obj.root.style[this.obj.hmode ? "left" : "right"], 10),
				  parseInt(this.obj.root.style[this.obj.vmode ? "top" : "bottom"], 10));
	}
};
// ENDFILE: domdrag.js
// STARTFILE: structures.js
//<NOLITE>
pg.structures.original={};
pg.structures.original.popupLayout=function () {
	return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle',
		'popupData', 'popupOtherLinks',
		'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks',
				   'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
		'popupMiscTools', ['popupRedlink'],
		'popupPrePreviewSep', 'popupPreview', 'popupSecondPreview', 'popupPreviewMore', 'popupPostPreview', 'popupFixDab'];
};
pg.structures.original.popupRedirSpans=function () {
	return ['popupRedir', 'popupWarnRedir', 'popupRedirTopLinks',
		'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'];
};
pg.structures.original.popupTitle=function (x) {
	log ('defaultstructure.popupTitle');
	if (!getValueOf('popupNavLinks')) {
		return navlinkStringToHTML('<b><<mainlink>></b>',x.article,x.params);
	}
	return '';
};
pg.structures.original.popupTopLinks=function (x) {
	log ('defaultstructure.popupTopLinks');
	if (getValueOf('popupNavLinks')) { return navLinksHTML(x.article, x.hint, x.params); }
	return '';
};
pg.structures.original.popupImage=function(x) {
	log ('original.popupImage, x.article='+x.article+', x.navpop.idNumber='+x.navpop.idNumber);
	return imageHTML(x.article, x.navpop.idNumber);
};
pg.structures.original.popupRedirTitle=pg.structures.original.popupTitle;
pg.structures.original.popupRedirTopLinks=pg.structures.original.popupTopLinks;


function copyStructure(oldStructure, newStructure) {
	pg.structures[newStructure]={};
	for (var prop in pg.structures[oldStructure]) {
		pg.structures[newStructure][prop]=pg.structures[oldStructure][prop];
	}
}

copyStructure('original', 'nostalgia');
pg.structures.nostalgia.popupTopLinks=function(x)  {
	var str='';
	str += '<b><<mainlink|shortcut= >></b>';

	// user links
	// contribs - log - count - email - block
	// count only if applicable; block only if popupAdminLinks
	str += 'if(user){<br><<contribs|shortcut=c>>';
	str+='if(wikimedia){*<<count|shortcut=#>>}';
	str+='if(ipuser){}else{*<<email|shortcut=E>>}if(admin){*<<block|shortcut=b>>}}';

	// editing links
	// talkpage   -> edit|new - history - un|watch - article|edit
	// other page -> edit - history - un|watch - talk|edit|new
	var editstr='<<edit|shortcut=e>>';
	var editOldidStr='if(oldid){<<editOld|shortcut=e>>|<<revert|shortcut=v|rv>>|<<edit|cur>>}else{'
	+ editstr + '}';
	var historystr='<<history|shortcut=h>>';
	var watchstr='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';

	str+='<br>if(talk){' +
		editOldidStr+'|<<new|shortcut=+>>' + '*' + historystr+'*'+watchstr + '*' +
		'<b><<article|shortcut=a>></b>|<<editArticle|edit>>' +
		'}else{' + // not a talk page
		editOldidStr + '*' + historystr + '*' + watchstr + '*' +
		'<b><<talk|shortcut=t>></b>|<<editTalk|edit>>|<<newTalk|shortcut=+|new>>'
		+ '}';

	// misc links
	str += '<br><<whatLinksHere|shortcut=l>>*<<relatedChanges|shortcut=r>>';
	str += 'if(admin){<br>}else{*}<<move|shortcut=m>>';

	// admin links
	str += 'if(admin){*<<unprotect|unprotectShort>>|<<protect|shortcut=p>>*' +
	'<<undelete|undeleteShort>>|<<delete|shortcut=d>>}';
	return navlinkStringToHTML(str, x.article, x.params);
};
pg.structures.nostalgia.popupRedirTopLinks=pg.structures.nostalgia.popupTopLinks;

/** -- fancy -- **/
copyStructure('original', 'fancy');
pg.structures.fancy.popupTitle=function (x) {
	return navlinkStringToHTML('<font size=+0><<mainlink>></font>',x.article,x.params);
};
pg.structures.fancy.popupTopLinks=function(x) {
	var hist='<<history|shortcut=h|hist>>|<<lastEdit|shortcut=/|last>>if(mainspace_en){|<<editors|shortcut=E|eds>>}';
	var watch='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';
	var move='<<move|shortcut=m|move>>';
	return navlinkStringToHTML('if(talk){' +
				   '<<edit|shortcut=e>>|<<new|shortcut=+|+>>*' + hist + '*' +
				   '<<article|shortcut=a>>|<<editArticle|edit>>' + '*' + watch + '*' + move +
				   '}else{<<edit|shortcut=e>>*' + hist +
				   '*<<talk|shortcut=t|>>|<<editTalk|edit>>|<<newTalk|shortcut=+|new>>' +
				   '*' + watch + '*' + move+'}<br>', x.article, x.params);
};
pg.structures.fancy.popupOtherLinks=function(x) {
	var admin='<<unprotect|unprotectShort>>|<<protect|shortcut=p>>*<<undelete|undeleteShort>>|<<delete|shortcut=d|del>>';
	var user='<<contribs|shortcut=c>>if(wikimedia){|<<count|shortcut=#|#>>}';
	user+='if(ipuser){|<<arin>>}else{*<<email|shortcut=E|'+
	popupString('email')+'>>}if(admin){*<<block|shortcut=b>>}';

	var normal='<<whatLinksHere|shortcut=l|links here>>*<<relatedChanges|shortcut=r|related>>';
	return navlinkStringToHTML('<br>if(user){' + user + '*}if(admin){'+admin+'if(user){<br>}else{*}}' + normal,
				   x.article, x.params);
};
pg.structures.fancy.popupRedirTitle=pg.structures.fancy.popupTitle;
pg.structures.fancy.popupRedirTopLinks=pg.structures.fancy.popupTopLinks;
pg.structures.fancy.popupRedirOtherLinks=pg.structures.fancy.popupOtherLinks;


/** -- fancy2 -- **/
// hack for [[User:MacGyverMagic]]
copyStructure('fancy', 'fancy2');
pg.structures.fancy2.popupTopLinks=function(x) { // hack out the <br> at the end and put one at the beginning
	return '<br>'+pg.structures.fancy.popupTopLinks(x).replace(RegExp('<br>$','i'),'');
};
pg.structures.fancy2.popupLayout=function () { // move toplinks to after the title
	return ['popupError', 'popupImage', 'popupTitle', 'popupData', 'popupTopLinks', 'popupOtherLinks',
		'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
		'popupMiscTools', ['popupRedlink'],
		'popupPrePreviewSep', 'popupPreview', 'popupSecondPreview', 'popupPreviewMore', 'popupPostPreview', 'popupFixDab'];
};

/** -- menus -- **/
copyStructure('original', 'menus');
pg.structures.menus.popupLayout=function () {
	return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle', 'popupOtherLinks',
		'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
		'popupData', 'popupMiscTools', ['popupRedlink'],
		'popupPrePreviewSep', 'popupPreview', 'popupSecondPreview', 'popupPreviewMore', 'popupPostPreview', 'popupFixDab'];
};
function toggleSticky(uid) {
	var popDiv=document.getElementById('navpopup_maindiv'+uid);
	if (!popDiv) { return; }
	if (!popDiv.navpopup.sticky) { popDiv.navpopup.stick(); }
	else {
		popDiv.navpopup.unstick();
		popDiv.navpopup.hide();
	}
}
pg.structures.menus.popupTopLinks = function (x, shorter) {
	// FIXME maybe this stuff should be cached
	var s=[];
	var dropdiv='<div class="popup_drop">';
	var enddiv='</div>';
	var endspan='</span>';
	var hist='<<history|shortcut=h>>';
	if (!shorter) { hist = '<menurow>' + hist +
			'|<<historyfeed|rss>>if(mainspace_en){|<<editors|shortcut=E>>}</menurow>'; }
	var lastedit='<<lastEdit|shortcut=/|show last edit>>';
	var jsHistory='<<lastContrib|last set of edits>><<sinceMe|changes since mine>>';
	var linkshere='<<whatLinksHere|shortcut=l|what links here>>';
	var related='<<relatedChanges|shortcut=r|related changes>>';
	var search='<menurow><<search|shortcut=s>>if(wikimedia){|<<globalsearch|shortcut=g|global>>}' +
	'|<<google|shortcut=G|web>></menurow>';
	var watch='<menurow><<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>></menurow>';
	var protect='<menurow><<unprotect|unprotectShort>>|' +
	'<<protect|shortcut=p>>|<<protectlog|log>></menurow>';
	var del='<menurow><<undelete|undeleteShort>>|<<delete|shortcut=d>>|' +
	'<<deletelog|log>></menurow>';
	var move='<<move|shortcut=m|move page>>';
	var nullPurge='<menurow><<nullEdit|shortcut=n|null edit>>|<<purge|shortcut=P>></menurow>';
	var viewOptions='<menurow><<view|shortcut=v>>|<<render|shortcut=S>>|<<raw>></menurow>';
	var editRow='if(oldid){' +
	'<menurow><<edit|shortcut=e>>|<<editOld|shortcut=e|this&nbsp;revision>></menurow>' +
	'<menurow><<revert|shortcut=v>>|<<undo>></menurow>' + '}else{<<edit|shortcut=e>>}';
	var markPatrolled='if(rcid){<<markpatrolled|mark patrolled>>}';
	var newTopic='if(talk){<<new|shortcut=+|new topic>>}';
	var protectDelete='if(admin){' + protect + del + '}';

	if (getValueOf('popupActionsMenu')) {
		s.push( '<<mainlink>>*' + dropdiv + menuTitle('actions'));
	} else {
		s.push( dropdiv + '<<mainlink>>');
	}
	s.push( '<menu>');
	s.push( editRow + markPatrolled + newTopic + hist + lastedit );
	if (!shorter) { s.push(jsHistory); }
	s.push( move + linkshere + related);
	if (!shorter) { s.push(nullPurge + search); }
	if (!shorter) { s.push(viewOptions); }
	s.push('<hr />' + watch + protectDelete);
	s.push('<hr />' +
		   'if(talk){<<article|shortcut=a|view article>><<editArticle|edit article>>}' +
		   'else{<<talk|shortcut=t|talk page>><<editTalk|edit talk>>' +
		   '<<newTalk|shortcut=+|new topic>>}</menu>' + enddiv);

	// user menu starts here
	var email='<<email|shortcut=E|email user>>';
	var contribs=	'if(wikimedia){<menurow>}<<contribs|shortcut=c|contributions>>if(wikimedia){</menurow>}' +
	'if(admin){<menurow><<deletedContribs>></menurow>}';


	s.push('if(user){*' + dropdiv + menuTitle('user'));
	s.push('<menu>'); +
	s.push('<menurow><<userPage|shortcut=u|user&nbsp;page>>|<<userSpace|space>></menurow>');
	s.push('<<userTalk|shortcut=t|user talk>><<editUserTalk|edit user talk>>' +
		   '<<newUserTalk|shortcut=+|leave comment>>');
	if(!shorter) { s.push( 'if(ipuser){<<arin>>}else{' + email + '}' ); }
	else { s.push( 'if(ipuser){}else{' + email + '}' ); }
	s.push('<hr />' + contribs + '<<userlog|shortcut=L|user log>>');
	s.push('if(wikimedia){<<count|shortcut=#|edit counter>>}');
	s.push('if(admin){<menurow><<unblock|unblockShort>>|<<block|shortcut=b|block user>></menurow>}');
	s.push('<<blocklog|shortcut=B|block log>>' + getValueOf('popupExtraUserMenu'));
	s.push('</menu>'  + enddiv + '}');

	// popups menu starts here
	if (getValueOf('popupSetupMenu') && !x.navpop.hasPopupMenu /* FIXME: hack */) {
		x.navpop.hasPopupMenu=true;
		s.push('*' + dropdiv + menuTitle('popupsMenu') + '<menu>');
		s.push('<<togglePreviews|toggle previews>>');
		s.push('<<purgePopups|reset>>');
		s.push('<<disablePopups|disable>>');
		s.push('</menu>'+enddiv);
	}
	return navlinkStringToHTML(s.join(''), x.article, x.params);
};

function menuTitle(s) {
	return '<a href="#" noPopup=1>' + popupString(s) + '</a>';
}

pg.structures.menus.popupRedirTitle=pg.structures.menus.popupTitle;
pg.structures.menus.popupRedirTopLinks=pg.structures.menus.popupTopLinks;

copyStructure('menus', 'shortmenus');
pg.structures.shortmenus.popupTopLinks=function(x) {
	return pg.structures.menus.popupTopLinks(x,true);
};
pg.structures.shortmenus.popupRedirTopLinks=pg.structures.shortmenus.popupTopLinks;

copyStructure('shortmenus', 'dabshortmenus');
pg.structures.dabshortmenus.popupLayout=function () {
	return ['popupError', 'popupImage', 'popupTopLinks', 'popupTitle', 'popupOtherLinks',
		'popupRedir', ['popupWarnRedir', 'popupRedirTopLinks', 'popupRedirTitle', 'popupRedirData', 'popupRedirOtherLinks'],
		'popupData', 'popupMiscTools', ['popupRedlink'], 'popupFixDab',
		'popupPrePreviewSep', 'popupPreview', 'popupSecondPreview', 'popupPreviewMore', 'popupPostPreview'];
};

copyStructure('menus', 'dabmenus');
pg.structures.dabmenus.popupLayout=pg.structures.dabshortmenus.popupLayout;


//</NOLITE>
pg.structures.lite={};
pg.structures.lite.popupLayout=function () {
	return ['popupTitle', 'popupPreview' ];
};
pg.structures.lite.popupTitle=function (x) {
	log (x.article + ': structures.lite.popupTitle');
	//return navlinkStringToHTML('<b><<mainlink>></b>',x.article,x.params);
	return '<div><span class="popup_mainlink"><b>' + x.article.toString() + '</b></span></div>';
};
// ENDFILE: structures.js
// STARTFILE: autoedit.js
//<NOLITE>
function substitute(data,cmdBody) {
	// alert('sub\nfrom: '+cmdBody.from+'\nto: '+cmdBody.to+'\nflags: '+cmdBody.flags);
	var fromRe=RegExp(cmdBody.from, cmdBody.flags);
	return data.replace(fromRe, cmdBody.to);
}

function execCmds(data, cmdList) {
	for (var i=0; i<cmdList.length; ++i) {
		data=cmdList[i].action(data, cmdList[i]);
	}
	return data;
}

function parseCmd(str) {
	// returns a list of commands
	if (!str.length) { return []; }
	var p=false;
	switch (str.charAt(0)) {
	case 's':
		p=parseSubstitute(str);
		break;
	default:
		return false;
	}
	if (p) { return [p].concat(parseCmd(p.remainder)); }
	return false;
}

function unEscape(str, sep) {
	return str.split('\\\\').join('\\').split('\\'+sep).join(sep).split('\\n').join('\n');
}


function parseSubstitute(str) {
	// takes a string like s/a/b/flags;othercmds and parses it

	var from,to,flags,tmp;

	if (str.length<4) { return false; }
	var sep=str.charAt(1);
	str=str.substring(2);

	tmp=skipOver(str,sep);
	if (tmp) { from=tmp.segment; str=tmp.remainder; }
	else { return false; }

	tmp=skipOver(str,sep);
	if (tmp) { to=tmp.segment; str=tmp.remainder; }
	else { return false; }

	flags='';
	if (str.length) {
		tmp=skipOver(str,';') || skipToEnd(str, ';');
		if (tmp) {flags=tmp.segment; str=tmp.remainder; }
	}

	return {action: substitute, from: from, to: to, flags: flags, remainder: str};

}

function skipOver(str,sep) {
	var endSegment=findNext(str,sep);
	if (endSegment<0) { return false; }
	var segment=unEscape(str.substring(0,endSegment), sep);
	return {segment: segment, remainder: str.substring(endSegment+1)};
}

function skipToEnd(str,sep) {
	return {segment: str, remainder: ''};
}

function findNext(str, ch) {
	for (var i=0; i<str.length; ++i) {
		if (str.charAt(i)=='\\') { i+=2; }
		if (str.charAt(i)==ch) { return i; }
	}
	return -1;
}

function setCheckbox(param, box) {
	var val=mw.util.getParamValue(param);
	if (val!==null) {
		switch (val) {
		case '1': case 'yes': case 'true':
			box.checked=true;
			break;
		case '0': case 'no':  case 'false':
			box.checked=false;
		}
	}
}

function autoEdit() {
	if (!setupPopups.completed) { setupPopups(); }
	if (!mw.config.get('wgEnableAPI') || mw.util.getParamValue('autoimpl') !== popupString('autoedit_version') ) { return false; }
	if (mw.util.getParamValue('autowatchlist') && mw.util.getParamValue('actoken')===autoClickToken()) {
		modifyWatchlist(mw.util.getParamValue('title'), mw.util.getParamValue('action'));
	}
	if (!document.editform) { return false; }
	if (window.autoEdit.alreadyRan) { return false; }
	window.autoEdit.alreadyRan=true;
	var cmdString=mw.util.getParamValue('autoedit');
	if (cmdString) {
		try {
			var editbox=document.editform.wpTextbox1;
		} catch (dang) { return; }
		var cmdList=parseCmd(cmdString);
		var input=editbox.value;
		var output=execCmds(input, cmdList);
		editbox.value=output;
		// wikEd user script compatibility
		if (typeof(wikEdUseWikEd) != 'undefined') {
			if (wikEdUseWikEd === true) {
				WikEdUpdateFrame();
			}
		}
	}
	setCheckbox('autominor', document.editform.wpMinoredit);
	setCheckbox('autowatch', document.editform.wpWatchthis);

	var rvid = mw.util.getParamValue('autorv');
	if (rvid) {
		var url=pg.wiki.apiwikibase + '?action=query&format=json&prop=revisions&revids='+rvid;
		startDownload(url, null, autoEdit2);
	} else { autoEdit2(); }
}

function autoEdit2(d) {
	var summary=mw.util.getParamValue('autosummary');
	var summaryprompt=mw.util.getParamValue('autosummaryprompt');
	var summarynotice='';
	if (d && d.data && mw.util.getParamValue('autorv')) {
		var s = getRvSummary(summary, d.data);
		if (s===false) {
			summaryprompt=true;
			summarynotice=popupString('Failed to get revision information, please edit manually.\n\n');
			summary = simplePrintf(summary, [mw.util.getParamValue('autorv'), '(unknown)', '(unknown)']);
		} else { summary = s; }
	}
	if (summaryprompt) {
		var txt= summarynotice +
			popupString('Enter a non-empty edit summary or press cancel to abort');
		var response=prompt(txt, summary);
		if (response) { summary=response; }
		else { return; }
	}
	if (summary) { document.editform.wpSummary.value=summary; }
	// Attempt to avoid possible premature clicking of the save button
	// (maybe delays in updates to the DOM are to blame?? or a red herring)
	setTimeout(autoEdit3, 100);
}

function autoClickToken() {
	return mw.user.sessionId();
}

function autoEdit3() {
	if( mw.util.getParamValue('actoken') != autoClickToken()) { return; }

	var btn=mw.util.getParamValue('autoclick');
	if (btn) {
		if (document.editform && document.editform[btn]) {
			var button=document.editform[btn];
			var msg=tprintf('The %s button has been automatically clicked. Please wait for the next page to load.',
					[ button.value ]);
			bannerMessage(msg);
			document.title='('+document.title+')';
			button.click();
		} else {
			alert(tprintf('Could not find button %s. Please check the settings in your javascript file.',
					  [ btn ]));
		}
	}
}

function bannerMessage(s) {
	var headings=document.getElementsByTagName('h1');
	if (headings) {
		var div=document.createElement('div');
		div.innerHTML='<font size=+1><b>' + s + '</b></font>';
		headings[0].parentNode.insertBefore(div, headings[0]);
	}
}

function getRvSummary(template, json) {
	try {
		var o=getJsObj(json);
		var edit = anyChild(o.query.pages).revisions[0];
	} catch (badness) {return false;}
	var timestamp = edit.timestamp.split(/[A-Z]/g).join(' ').replace(/^ *| *$/g, '');
	return simplePrintf(template, [edit.revid, timestamp, edit.user]);
}

//</NOLITE>
// ENDFILE: autoedit.js
// STARTFILE: downloader.js
/**
   @fileoverview
   {@link Downloader}, a xmlhttprequest wrapper, and helper functions.
*/

/**
   Creates a new Downloader
   @constructor
   @class The Downloader class. Create a new instance of this class to download stuff.
   @param {String} url The url to download. This can be omitted and supplied later.
*/
function Downloader(url) {
	// Source: http://jibbering.com/2002/4/httprequest.html
	/** xmlhttprequest object which we're wrapping */
	this.http = false;

	/*@cc_on @*/
	/*@if (@_jscript_version >= 5)
	// JScript gives us Conditional compilation,
	// we can cope with old IE versions.
	// and security blocked creation of the objects.
	try {
	this.http = new ActiveXObject("Msxml2.XMLHTTP");
	} catch (e) {
	try {
	this.http = new ActiveXObject("Microsoft.XMLHTTP");
	} catch (E) {
	// this.http = false;
	}
	}
	@end @*/

	if (! this.http && typeof XMLHttpRequest!='undefined') { this.http = new XMLHttpRequest(); }
	/**
		The url to download
		@type String
	*/
	this.url = url;
	/**
		A universally unique ID number
		@type integer
	*/
	this.id=null;
	/**
		Modification date, to be culled from the incoming headers
		@type Date
		@private
	*/
	this.lastModified = null;
	/**
		What to do when the download completes successfully
		@type Function
		@private
	*/
	this.callbackFunction = null;
	/**
		What to do on failure
		@type Function
		@private
	*/
	this.onFailure = null;
	/**
		Flag set on <code>abort</code>
		@type boolean
	*/
	this.aborted = false;
	/**
	   HTTP method. See http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html for possibilities.
	   @type String
	*/
	this.method='GET';
	/**
		Async flag.
		@type boolean
	*/
	this.async=true;
}

new Downloader();

/** Submits the http request. */
Downloader.prototype.send = function (x) {
	if (!this.http) { return null; }
	return this.http.send(x);
};
/** Aborts the download, setting the <code>aborted</code> field to true.  */
Downloader.prototype.abort = function () {
	if (!this.http) { return null; }
	this.aborted=true;
	return this.http.abort();
};
/** Returns the downloaded data. */
Downloader.prototype.getData = function () {if (!this.http) { return null; } return this.http.responseText;};
/** Prepares the download. */
Downloader.prototype.setTarget = function () {
	if (!this.http) { return null; }
	this.http.open(this.method, this.url, this.async);
};
/** Gets the state of the download. */
Downloader.prototype.getReadyState=function () {if (!this.http) { return null; } return this.http.readyState;};

pg.misc.downloadsInProgress = { };

/** Starts the download.
	Note that setTarget {@link Downloader#setTarget} must be run first
*/
Downloader.prototype.start=function () {
	if (!this.http) { return; }
	pg.misc.downloadsInProgress[this.id] = this;
	this.http.send(null);
};

/** Gets the 'Last-Modified' date from the download headers.
	Should be run after the download completes.
	Returns <code>null</code> on failure.
	@return {Date}
*/
Downloader.prototype.getLastModifiedDate=function () {
	if(!this.http) { return null; }
	var lastmod=null;
	try {
		lastmod=this.http.getResponseHeader('Last-Modified');
	} catch (err) {}
	if (lastmod) { return new Date(lastmod); }
	return null;
};

/** Sets the callback function.
	@param {Function} f callback function, called as <code>f(this)</code> on success
*/
Downloader.prototype.setCallback = function (f) {
	if(!this.http) { return; }
	this.http.onreadystatechange = f;
};

Downloader.prototype.getStatus = function() { if (!this.http) { return null; } return this.http.status; };

//////////////////////////////////////////////////
// helper functions

/** Creates a new {@link Downloader} and prepares it for action.
	@param {String} url The url to download
	@param {integer} id The ID of the {@link Downloader} object
	@param {Function} callback The callback function invoked on success
	@return {String/Downloader} the {@link Downloader} object created, or 'ohdear' if an unsupported browser
*/
function newDownload(url, id, callback, onfailure) {
	var d=new Downloader(url);
	if (!d.http) { return 'ohdear'; }
	d.id=id;
	d.setTarget();
	if (!onfailure) {
		onfailure=2;
	}
	var f = function () {
		if (d.getReadyState() == 4) {
			delete pg.misc.downloadsInProgress[this.id];
			try {
				if ( d.getStatus() == 200 ) {
					d.data=d.getData();
					d.lastModified=d.getLastModifiedDate();
					callback(d);
				} else if (typeof onfailure == typeof 1) {
					if (onfailure > 0) {
						// retry
						newDownload(url, id, callback, onfailure - 1);
					}
				} else if ($.isFunction(onfailure)) {
					onfailure(d,url,id,callback);
				}
			} catch (somerr) { /* ignore it */ }
		}
	};
	d.setCallback(f);
	return d;
}
/** Simulates a download from cached data.
	The supplied data is put into a {@link Downloader} as if it had downloaded it.
	@param {String} url The url.
	@param {integer} id The ID.
	@param {Function} callback The callback, which is invoked immediately as <code>callback(d)</code>,
	where <code>d</code> is the new {@link Downloader}.
	@param {String} data The (cached) data.
	@param {Date} lastModified The (cached) last modified date.
*/
function fakeDownload(url, id, callback, data, lastModified, owner) {
	var d=newDownload(url,callback);
	d.owner=owner;
	d.id=id; d.data=data;
	d.lastModified=lastModified;
	return callback(d);
}

/**
   Starts a download.
   @param {String} url The url to download
   @param {integer} id The ID of the {@link Downloader} object
   @param {Function} callback The callback function invoked on success
   @return {String/Downloader} the {@link Downloader} object created, or 'ohdear' if an unsupported browser
*/
function startDownload(url, id, callback) {
	var d=newDownload(url, id, callback);
	if (typeof d == typeof '' ) { return d; }
	d.start();
	return d;
}

/**
   Aborts all downloads which have been started.
*/
function abortAllDownloads() {
	for ( var x in pg.misc.downloadsInProgress ) {
		try {
			pg.misc.downloadsInProgress[x].aborted=true;
			pg.misc.downloadsInProgress[x].abort();
			delete pg.misc.downloadsInProgress[x];
		} catch (e) { }
	}
}
// ENDFILE: downloader.js
// STARTFILE: livepreview.js
// TODO: location is often not correct (eg relative links in previews)

/**
 * InstaView - a Mediawiki to HTML converter in JavaScript
 * Version 0.6.1
 * Copyright (C) Pedro Fayolle 2005-2006
 * http://en.wikipedia.org/wiki/User:Pilaf
 * Distributed under the BSD license
 *
 * Changelog:
 *
 * 0.6.1
 * - Fixed problem caused by \r characters
 * - Improved inline formatting parser
 *
 * 0.6
 * - Changed name to InstaView
 * - Some major code reorganizations and factored out some common functions
 * - Handled conversion of relative links (i.e. [[/foo]])
 * - Fixed misrendering of adjacent definition list items
 * - Fixed bug in table headings handling
 * - Changed date format in signatures to reflect Mediawiki's
 * - Fixed handling of [[:Image:...]]
 * - Updated MD5 function (hopefully it will work with UTF-8)
 * - Fixed bug in handling of links inside images
 *
 * To do:
 * - Better support for math tags
 * - Full support for <nowiki>
 * - Parser-based (as opposed to RegExp-based) inline wikicode handling (make it one-pass and bullet-proof)
 * - Support for templates (through AJAX)
 * - Support for coloured links (AJAX)
 */


var Insta = {};

function setupLivePreview() {

	// options
	Insta.conf =
	{
		baseUrl: '',

		user: {},

		wiki: {
		lang: pg.wiki.lang,
		interwiki: pg.wiki.interwiki,
		default_thumb_width: 180
		},

		paths: {
		articles: pg.wiki.articlePath + '/',
		// Only used for Insta previews with images. (not in popups)
		math: '/math/',
		images: '//upload.wikimedia.org/wikipedia/en/', // FIXME ( window.getImageUrlStart ? getImageUrlStart(pg.wiki.hostname) : ''),
		images_fallback: '//upload.wikimedia.org/wikipedia/commons/',
		},

		locale: {
		user: mw.config.get('wgFormattedNamespaces')[pg.nsUserId],
		image: mw.config.get('wgFormattedNamespaces')[pg.nsImageId],
		category: mw.config.get('wgFormattedNamespaces')[pg.nsCategoryId],
		// shouldn't be used in popup previews, i think
		months: ['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec']
		}
	};

	// options with default values or backreferences
	with (Insta.conf) {
	user.name = user.name || 'Wikipedian';
	user.signature = '[['+locale.user+':'+user.name+'|'+user.name+']]';
	//paths.images = '//upload.wikimedia.org/wikipedia/' + wiki.lang + '/';
	}

	// define constants
	Insta.BLOCK_IMAGE = new RegExp('^\\[\\[(?:File|Image|'+Insta.conf.locale.image+
        '):.*?\\|.*?(?:frame|thumbnail|thumb|none|right|left|center)', 'i');

}


Insta.dump = function(from, to)
{
	if (typeof from == 'string') { from = document.getElementById(from); }
	if (typeof to == 'string') { to = document.getElementById(to); }
	to.innerHTML = this.convert(from.value);
};

Insta.convert = function(wiki)
{
	var ll = (typeof wiki == 'string')? wiki.replace(/\r/g,'').split(/\n/): wiki, // lines of wikicode
		o  = '', // output
		p  = 0,	 // para flag
		$r;	 // result of passing a regexp to $()

	// some shorthands
	function remain() { return ll.length; }
	function sh() { return ll.shift(); } // shift
	function ps(s) { o += s; } // push

	// similar to C's printf, uses ? as placeholders, ?? to escape question marks
	function f()
	{
		var i=1, a=arguments,  f=a[0], o='', c, p;
		for (; i<a.length; i++) {
			if ((p=f.indexOf('?'))+1) {
				// allow character escaping
				i -= c = f.charAt(p+1)=='?' ? 1 : 0;
				o += f.substring(0,p) + (c ? '?' : a[i]);
				f = f.substr(p+1+c);
			} else { break; }
		}
		return o+f;
	}

	function html_entities(s) {
		return s.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;");
	}
	
	// Wiki text parsing to html is a nightmare.
	// The below functions deliberately don't escape the ampersand since this would make it more difficult,
	// and we don't absolutely need to for how we need it.
	// This means that any unescaped ampersands in wikitext will remain unescaped and can cause invalid HTML.
	// Browsers should all be able to handle it though.
	// We also escape significant wikimarkup characters to prevent further matching on the processed text
	function htmlescape_text(s) {
		return s.replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/:/g,"&#58;").replace(/\[/g,"&#91;").replace(/]/g,"&#93;");
	}
	function htmlescape_attr(s) {
		return htmlescape_text(s).replace(/'/g,"&#39;").replace(/"/g,"&quot;");
	}

	function max(a,b) { return (a>b)?a:b; }
	function min(a,b) { return (a<b)?a:b; }

	// return the first non matching character position between two strings
	function str_imatch(a, b)
	{
		for (var i=0, l=min(a.length, b.length); i<l; i++) {
			if (a.charAt(i)!=b.charAt(i)) { break; }
		}
		return i;
	}

	// compare current line against a string or regexp
	// if passed a string it will compare only the first string.length characters
	// if passed a regexp the result is stored in $r
	function $(c) { return (typeof c == 'string') ? (ll[0].substr(0,c.length)==c) : ($r = ll[0].match(c)); }

	function $$(c) { return ll[0]==c; } // compare current line against a string
	function _(p) { return ll[0].charAt(p); } // return char at pos p

	function endl(s) { ps(s); sh(); }

	function parse_list()
	{
		var prev='';

		while (remain() && $(/^([*#:;]+)(.*)$/)) {

			var l_match = $r;

			sh();

			var ipos = str_imatch(prev, l_match[1]);

			// close uncontinued lists
			for (var i=prev.length-1; i >= ipos; i--) {

				var pi = prev.charAt(i);

				if (pi=='*') { ps('</ul>'); }
				else if (pi=='#') { ps('</ol>'); }
				// close a dl only if the new item is not a dl item (:, ; or empty)
				else switch (l_match[1].charAt(i)) { case'':case'*':case'#': ps('</dl>') }
			}

			// open new lists
			for (var i=ipos; i<l_match[1].length; i++) {

				var li = l_match[1].charAt(i);

				if (li=='*') { ps('<ul>'); }
				else if (li=='#') { ps('<ol>'); }
				// open a new dl only if the prev item is not a dl item (:, ; or empty)
				else { switch(prev.charAt(i)) { case'':case'*':case'#': ps('<dl>') } }
			}

			switch (l_match[1].charAt(l_match[1].length-1)) {

				case '*': case '#':
					ps('<li>' + parse_inline_nowiki(l_match[2])); break

				case ';':
					ps('<dt>');

					var dt_match;

					// handle ;dt :dd format
					if (dt_match = l_match[2].match(/(.*?)(:.*?)$/)) {

						ps(parse_inline_nowiki(dt_match[1]))
						ll.unshift(dt_match[2])

					} else ps(parse_inline_nowiki(l_match[2]))

					break;

				case ':':
					ps('<dd>' + parse_inline_nowiki(l_match[2]))
			}

			prev=l_match[1]
		}

		// close remaining lists
		for (var i=prev.length-1; i>=0; i--)
			ps(f('</?>', (prev.charAt(i)=='*')? 'ul': ((prev.charAt(i)=='#')? 'ol': 'dl')))
	}

	function parse_table()
	{
		endl(f('<table>', $(/^\{\|( .*)$/)? $r[1]: ''))

		for (;remain();) if ($('|')) switch (_(1)) {
			case '}': endl('</table>'); return
			case '-': endl(f('<tr>', $(/\|-*(.*)/)[1])); break
			default: parse_table_data()
		}
		else if ($('!')) parse_table_data()
		else sh()
	}

	function parse_table_data()
	{
		var td_line, match_i

		// 1: "|+", '|' or '+'
		// 2: ??
		// 3: attributes ??
		// TODO: finish commenting this regexp
		var td_match = sh().match(/^(\|\+|\||!)((?:([^[|]*?)\|(?!\|))?(.*))$/)

		if (td_match[1] == '|+') ps('<caption');
		else ps('<t' + ((td_match[1]=='|')?'d':'h'))

		if (typeof td_match[3] != 'undefined') {

			//ps(' ' + td_match[3])
			match_i = 4

		} else match_i = 2

		ps('>')

		if (td_match[1] != '|+') {

			// use || or !! as a cell separator depending on context
			// NOTE: when split() is passed a regexp make sure to use non-capturing brackets
			td_line = td_match[match_i].split((td_match[1] == '|')? '||': /(?:\|\||!!)/)

			ps(parse_inline_nowiki(td_line.shift()))

			while (td_line.length) ll.unshift(td_match[1] + td_line.pop())

		} else ps(td_match[match_i])

		var tc = 0, td = []

		for (;remain(); td.push(sh()))
		if ($('|')) {
			if (!tc) break // we're at the outer-most level (no nested tables), skip to td parse
			else if (_(1)=='}') tc--
		}
		else if (!tc && $('!')) break
		else if ($('{|')) tc++

		if (td.length) ps(Insta.convert(td))
	}

	function parse_pre()
	{
		ps('<pre>')
		do endl(parse_inline_nowiki(ll[0].substring(1)) + "\n"); while (remain() && $(' '))
		ps('</pre>')
	}

	function parse_block_image()
	{
		ps(parse_image(sh()))
	}

	function parse_image(str)
	{
//<NOLITE>
		// get what's in between "[[Image:" and "]]"
		var tag = str.substring(str.indexOf(':') + 1, str.length - 2);

		var width;
		var attr = [], filename, caption = '';
		var thumb=0, frame=0, center=0;
		var align='';

		if (tag.match(/\|/)) {
			// manage nested links
			var nesting = 0;
			var last_attr;
			for (var i = tag.length-1; i > 0; i--) {
				if (tag.charAt(i) == '|' && !nesting) {
					last_attr = tag.substr(i+1);
					tag = tag.substring(0, i);
					break;
				} else switch (tag.substr(i-1, 2)) {
					case ']]':
						nesting++;
						i--;
						break;
					case '[[':
						nesting--;
						i--;
				}
			}

			attr = tag.split(/\s*\|\s*/);
			attr.push(last_attr);
			filename = attr.shift();

			var w_match;

			for (;attr.length; attr.shift())
			if (w_match = attr[0].match(/^(\d*)(?:[px]*\d*)?px$/)) width = w_match[1]
			else switch(attr[0]) {
				case 'thumb':
				case 'thumbnail':
					thumb=true;
				case 'frame':
					frame=true;
					break;
				case 'none':
				case 'right':
				case 'left':
					center=false;
					align=attr[0];
					break;
				case 'center':
					center=true;
					align='none';
					break;
				default:
					if (attr.length == 1) caption = attr[0];
			}

		} else filename = tag;


		var o='';

		if (frame) {

			if (align=='') align = 'right';

			o += f("<div class='thumb t?'>", align);

			if (thumb) {
				if (!width) width = Insta.conf.wiki.default_thumb_width;

				o += f("<div style='width:?px;'>?", 2+width*1, make_image(filename, caption, width)) +
					f("<div class='thumbcaption'><div class='magnify' style='float:right'><a href='?' title='Enlarge'></a></div>?</div>",
						htmlescape_attr(Insta.conf.paths.articles + Insta.conf.locale.image + ':' + filename),
						parse_inline_nowiki(caption)
					)
			} else {
				o += '<div>' + make_image(filename, caption) + f("<div class='thumbcaption'>?</div>", parse_inline_nowiki(caption))
			}

			o += '</div></div>';

		} else if (align != '') {
			o += f("<div class='float?'><span>?</span></div>", align, make_image(filename, caption, width));
		} else {
			return make_image(filename, caption, width);
		}

		return center? f("<div class='center'>?</div>", o): o;
//</NOLITE>
	}

	function parse_inline_nowiki(str)
	{
		var start, lastend=0
		var substart=0, nestlev=0, open, close, subloop;
		var html='';

		while (-1 != (start = str.indexOf('<nowiki>', substart))) {
			html += parse_inline_wiki(str.substring(lastend, start));
			start += 8;
			substart = start;
			subloop = true;
			do {
				open = str.indexOf('<nowiki>', substart);
				close = str.indexOf('</nowiki>', substart);
				if (close<=open || open==-1) {
					if (close==-1) {
						return html + html_entities(str.substr(start));
					}
					substart = close+9;
					if (nestlev) {
						nestlev--;
					} else {
						lastend = substart;
						html += html_entities(str.substring(start, lastend-9));
						subloop = false;
					}
				} else {
					substart = open+8;
					nestlev++;
				}
			} while (subloop)
		}

		return html + parse_inline_wiki(str.substr(lastend));
	}

	function make_image(filename, caption, width)
	{
//<NOLITE>
		// uppercase first letter in file name
		filename = filename.charAt(0).toUpperCase() + filename.substr(1);
		// replace spaces with underscores
		filename = filename.replace(/ /g, '_');

		caption = strip_inline_wiki(caption);

		var md5 = hex_md5(filename);

		var source = md5.charAt(0) + '/' + md5.substr(0,2) + '/' + filename;

		if (width) width = "width='" + width + "px'";

		var img = "<img onerror=\""+pg.escapeQuotesHTML("this.onerror=null;this.src='"+pg.jsescape(Insta.conf.paths.images_fallback + source)+"'")+"\" src=\""+pg.escapeQuotesHTML(Insta.conf.paths.images + source)+"\" "+(caption!='' ? "alt=\"" + pg.escapeQuotesHTML(caption) + "\"" : '')+" "+width+">";

		return f("<a class='image' ? href=\"?\">?</a>", (caption!='')? "title=\"" + pg.escapeQuotesHTML(caption) + "\"" : '', pg.escapeQuotesHTML(Insta.conf.paths.articles + Insta.conf.locale.image + ':' + filename), img);
//</NOLITE>
	}

	function parse_inline_images(str)
	{
//<NOLITE>
		var start, substart=0, nestlev=0;
		var loop, close, open, wiki, html;

		while (-1 != (start=str.indexOf('[[', substart))) {
			if(str.substr(start+2).match(RegExp('^(Image|File|' + Insta.conf.locale.image + '):','i'))) {
				loop=true;
				substart=start;
				do {
					substart+=2;
					close=str.indexOf(']]',substart);
					open=str.indexOf('[[',substart);
					if (close<=open||open==-1) {
						if (close==-1) return str;
						substart=close;
						if (nestlev) {
							nestlev--;
						} else {
							wiki=str.substring(start,close+2);
							html=parse_image(wiki);
							str=str.replace(wiki,html);
							substart=start+html.length;
							loop=false;
						}
					} else {
						substart=open;
						nestlev++;
					}
				} while (loop)

			} else break;
		}

//</NOLITE>
		return str;
	}

	// the output of this function doesn't respect the FILO structure of HTML
	// but since most browsers can handle it I'll save myself the hassle
	function parse_inline_formatting(str)
	{
		var em,st,i,li,o='';
		while ((i=str.indexOf("''",li))+1) {
			o += str.substring(li,i);
			li=i+2;
			if (str.charAt(i+2)=="'") {
				li++;
				st=!st;
				o+=st?'<strong>':'</strong>';
			} else {
				em=!em;
				o+=em?'<em>':'</em>';
			}
		}
		return o+str.substr(li);
	}

	function parse_inline_wiki(str)
	{
		var aux_match;

		str = parse_inline_images(str);
		str = parse_inline_formatting(str);

		// math
		while (aux_match = str.match(/<(?:)math>(.*?)<\/math>/i)) {
			var math_md5 = hex_md5(aux_match[1]);
			str = str.replace(aux_match[0], f("<img src='?.png'>", Insta.conf.paths.math+math_md5));
		}

		// Build a Mediawiki-formatted date string
		var date = new Date;
		var minutes = date.getUTCMinutes();
		if (minutes < 10) minutes = '0' + minutes;
		var date = f("?:?, ? ? ? (UTC)", date.getUTCHours(), minutes, date.getUTCDate(), Insta.conf.locale.months[date.getUTCMonth()], date.getUTCFullYear());

		// text formatting
		return str.
			// signatures
			replace(/~{5}(?!~)/g, date).
			replace(/~{4}(?!~)/g, Insta.conf.user.name+' '+date).
			replace(/~{3}(?!~)/g, Insta.conf.user.name).

			// [[:Category:...]], [[:Image:...]], etc...
			replace(RegExp('\\[\\[:((?:'+Insta.conf.locale.category+'|Image|File|'+Insta.conf.locale.image+'|'+Insta.conf.wiki.interwiki+'):[^|]*?)\\]\\](\w*)','gi'), function($0,$1,$2){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($1) + htmlescape_text($2));}).
			// remove straight category and interwiki tags
			replace(RegExp('\\[\\[(?:'+Insta.conf.locale.category+'|'+Insta.conf.wiki.interwiki+'):.*?\\]\\]','gi'),'').

			// [[:Category:...|Links]], [[:Image:...|Links]], etc...
			replace(RegExp('\\[\\[:((?:'+Insta.conf.locale.category+'|Image|File|'+Insta.conf.locale.image+'|'+Insta.conf.wiki.interwiki+'):.*?)\\|([^\\]]+?)\\]\\](\\w*)','gi'), function($0,$1,$2,$3){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($2) + htmlescape_text($3));}).

			// [[/Relative links]]
			replace(/\[\[(\/[^|]*?)\]\]/g, function($0,$1){return f("<a href='?'>?</a>", Insta.conf.baseUrl + htmlescape_attr($1), htmlescape_text($1)); }).

			// [[/Replaced|Relative links]]
			replace(/\[\[(\/.*?)\|(.+?)\]\]/g, function($0,$1,$2){return f("<a href='?'>?</a>", Insta.conf.baseUrl + htmlescape_attr($1), htmlescape_text($2)); }).

			// [[Common links]]
			replace(/\[\[([^|]*?)\]\](\w*)/g, function($0,$1,$2){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($1) + htmlescape_text($2)); }).

			// [[Replaced|Links]]
			replace(/\[\[(.*?)\|([^\]]+?)\]\](\w*)/g, function($0,$1,$2,$3){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1), htmlescape_text($2) + htmlescape_text($3)); }).

			// [[Stripped:Namespace|Namespace]]
			replace(/\[\[([^\]]*?:)?(.*?)( *\(.*?\))?\|\]\]/g, function($0,$1,$2,$3){return f("<a href='?'>?</a>", Insta.conf.paths.articles + htmlescape_attr($1) + htmlescape_attr($2) + htmlescape_attr($3), htmlescape_text($2)); }).

			// External links
			replace(/\[(https?|news|ftp|mailto|gopher|irc):(\/*)([^\]]*?) (.*?)\]/g, function($0,$1,$2,$3,$4){return f("<a class='external' href='?:?'>?</a>", htmlescape_attr($1), htmlescape_attr($2) + htmlescape_attr($3), htmlescape_text($4)); }).
			replace(/\[http:\/\/(.*?)\]/g, function($0,$1){return f("<a class='external' href='http://?'>[#]</a>", htmlescape_attr($1)); }).
			replace(/\[(news|ftp|mailto|gopher|irc):(\/*)(.*?)\]/g, function($0,$1,$2,$3,$4){return f("<a class='external' href='?:?'>?:?</a>", htmlescape_attr($1), htmlescape_attr($2) + htmlescape_attr($3), htmlescape_text($1), htmlescape_text($2) + htmlescape_text($3)); }).
			replace(/(^| )(https?|news|ftp|mailto|gopher|irc):(\/*)([^ $]*[^.,!?;: $])/g, function($0,$1,$2,$3,$4){return f("?<a class='external' href='?:?'>?:?</a>", htmlescape_text($1), htmlescape_attr($2), htmlescape_attr($3) + htmlescape_attr($4), htmlescape_text($2), htmlescape_text($3) + htmlescape_text($4)); }).

			replace('__NOTOC__','').
			replace('__NOEDITSECTION__','');
	}
/*
*/
	function strip_inline_wiki(str)
	{
		return str
			.replace(/\[\[[^\]]*\|(.*?)\]\]/g,'$1')
			.replace(/\[\[(.*?)\]\]/g,'$1')
			.replace(/''(.*?)''/g,'$1');
	}

	// begin parsing
	for (;remain();) if ($(/^(={1,6})(.*)\1(.*)$/)) {
		p=0
		endl(f('<h?>?</h?>?', $r[1].length, parse_inline_nowiki($r[2]), $r[1].length, $r[3]))

	} else if ($(/^[*#:;]/)) {
		p=0
		parse_list()

	} else if ($(' ')) {
		p=0
		parse_pre()

	} else if ($('{|')) {
		p=0
		parse_table()

	} else if ($(/^----+$/)) {
		p=0
		endl('<hr />')

	} else if ($(Insta.BLOCK_IMAGE)) {
		p=0
		parse_block_image()

	} else {

		// handle paragraphs
		if ($$('')) {
			if (p = (remain()>1 && ll[1]==(''))) endl('<p><br>')
		} else {
			if(!p) {
				ps('<p>')
				p=1
			}
			ps(parse_inline_nowiki(ll[0]) + ' ')
		}

		sh();
	}

	return o
};

window.wiki2html=function(txt,baseurl) {
	Insta.conf.baseUrl=baseurl;
	return Insta.convert(txt);
};
// ENDFILE: livepreview.js
// STARTFILE: pageinfo.js
//<NOLITE>
function popupFilterPageSize(data) {
	return formatBytes(data.length);
}

function popupFilterCountLinks(data) {
	var num=countLinks(data);
	return String(num) + '&nbsp;' + ((num!=1)?popupString('wikiLinks'):popupString('wikiLink'));
}

function popupFilterCountImages(data) {
	var num=countImages(data);
	return String(num) + '&nbsp;' + ((num!=1)?popupString('images'):popupString('image'));
}

function popupFilterCountCategories(data) {
	var num=countCategories(data);
	return String(num) + '&nbsp;' + ((num!=1)?popupString('categories'):popupString('category'));
}


function popupFilterLastModified(data,download) {
	var lastmod=download.lastModified;
	var now=new Date();
	var age=now-lastmod;
	if (lastmod && getValueOf('popupLastModified')) {
		return (tprintf('%s old', [formatAge(age)])).replace(RegExp(' ','g'), '&nbsp;');
	}
	return '';
}

function formatAge(age) {
	// coerce into a number
	var a=0+age, aa=a;

	var seclen  = 1000;
	var minlen  = 60*seclen;
	var hourlen = 60*minlen;
	var daylen  = 24*hourlen;
	var weeklen = 7*daylen;

	var numweeks = (a-a%weeklen)/weeklen; a = a-numweeks*weeklen; var sweeks = addunit(numweeks, 'week');
	var numdays  = (a-a%daylen)/daylen;   a = a-numdays*daylen;   var sdays  = addunit(numdays, 'day');
	var numhours = (a-a%hourlen)/hourlen; a = a-numhours*hourlen; var shours = addunit(numhours,'hour');
	var nummins  = (a-a%minlen)/minlen;   a = a-nummins*minlen;   var smins  = addunit(nummins, 'minute');
	var numsecs  = (a-a%seclen)/seclen;   a = a-numsecs*seclen;   var ssecs  = addunit(numsecs, 'second');

	if (aa > 4*weeklen) { return sweeks; }
	if (aa > weeklen)   { return sweeks + ' ' + sdays; }
	if (aa > daylen)	{ return sdays  + ' ' + shours; }
	if (aa > 6*hourlen) { return shours; }
	if (aa > hourlen)   { return shours + ' ' + smins; }
	if (aa > 10*minlen) { return smins; }
	if (aa > minlen)	{ return smins  + ' ' + ssecs; }
	return ssecs;
}

function addunit(num,str) { return '' + num + ' ' + ((num!=1) ? popupString(str+'s') : popupString(str)) ;}

function runPopupFilters(list, data, download) {
	var ret=[];
	for (var i=0; i<list.length; ++i) {
		if (list[i] && typeof list[i] == 'function') {
			var s=list[i](data, download, download.owner.article);
			if (s) { ret.push(s); }
		}
	}
	return ret;
}

function getPageInfo(data, download) {
	if (!data || data.length === 0) { return popupString('Empty page'); }

	var popupFilters=getValueOf('popupFilters') || [];
	var extraPopupFilters = getValueOf('extraPopupFilters') || [];
	var pageInfoArray = runPopupFilters(popupFilters.concat(extraPopupFilters), data, download);

	var pageInfo=pageInfoArray.join(', ');
	if (pageInfo !== '' ) { pageInfo = upcaseFirst(pageInfo); }
	return pageInfo;
}


// this could be improved!
function countLinks(wikiText) { return wikiText.split('[[').length - 1; }

// if N = # matches, n = # brackets, then
// String.parenSplit(regex) intersperses the N+1 split elements
// with Nn other elements. So total length is
// L= N+1 + Nn = N(n+1)+1. So N=(L-1)/(n+1).

function countImages(wikiText) {
	return (wikiText.parenSplit(pg.re.image).length - 1) / (pg.re.imageBracketCount + 1);
}

function countCategories(wikiText) {
	return (wikiText.parenSplit(pg.re.category).length - 1) / (pg.re.categoryBracketCount + 1);
}

function popupFilterStubDetect(data, download, article)	 {
	var counts=stubCount(data, article);
	if (counts.real) { return popupString('stub'); }
	if (counts.sect) { return popupString('section stub'); }
	return '';
}

function popupFilterDisambigDetect(data, download, article) {
	if (getValueOf('popupOnlyArticleDabStub') && article.namespace()) { return ''; }
	return (isDisambig(data, article)) ? popupString('disambig') : '';
}

function formatBytes(num) {
	return (num > 949) ? (Math.round(num/100)/10+popupString('kB')) : (num +'&nbsp;' + popupString('bytes')) ;
}
//</NOLITE>
// ENDFILE: pageinfo.js
// STARTFILE: titles.js
/**
   @fileoverview Defines the {@link Title} class, and associated crufty functions.

   <code>Title</code> deals with article titles and their various
   forms.  {@link Stringwrapper} is the parent class of
   <code>Title</code>, which exists simply to make things a little
   neater.

*/

/**
   Creates a new Stringwrapper.
   @constructor

   @class the Stringwrapper class. This base class is not really
   useful on its own; it just wraps various common string operations.
*/
function Stringwrapper() {
	/**
	   Wrapper for this.toString().indexOf()
	   @param {String} x
	   @type integer
	*/
	this.indexOf=function(x){return this.toString().indexOf(x);};
	/**
	   Returns this.value.
	   @type String
	*/
	this.toString=function(){return this.value;};
	/**
	   Wrapper for {@link String#parenSplit} applied to this.toString()
	   @param {RegExp} x
	   @type Array
	*/
	this.parenSplit=function(x){return this.toString().parenSplit(x);};
	/**
	   Wrapper for this.toString().substring()
	   @param {String} x
	   @param {String} y (optional)
	   @type String
	*/
	this.substring=function(x,y){
		if (typeof y=='undefined') { return this.toString().substring(x); }
		return this.toString().substring(x,y);
	};
	/**
	   Wrapper for this.toString().split()
	   @param {String} x
	   @type Array
	*/
	this.split=function(x){return this.toString().split(x);};
	/**
	   Wrapper for this.toString().replace()
	   @param {String} x
	   @param {String} y
	   @type String
	*/
	this.replace=function(x,y){ return this.toString().replace(x,y); };
}


/**
   Creates a new <code>Title</code>.
   @constructor

   @class The Title class. Holds article titles and converts them into
   various forms. Also deals with anchors, by which we mean the bits
   of the article URL after a # character, representing locations
   within an article.

   @param {String} value The initial value to assign to the
   article. This must be the canonical title (see {@link
   Title#value}. Omit this in the constructor and use another function
   to set the title if this is unavailable.
*/
function Title(val) {
	/**
	   The canonical article title. This must be in UTF-8 with no
	   entities, escaping or nasties. Also, underscores should be
	   replaced with spaces.
	   @type String
	   @private
	*/
	this.value=null;
	/**
	   The canonical form of the anchor. This should be exactly as
	   it appears in the URL, i.e. with the .C3.0A bits in.
	   @type String
	*/
	this.anchor='';

	this.setUtf(val);
}
Title.prototype=new Stringwrapper();
/**
   Returns the canonical representation of the article title, optionally without anchor.
   @param {boolean} omitAnchor
   @fixme Decide specs for anchor
   @return String The article title and the anchor.
*/
Title.prototype.toString=function(omitAnchor) {
	return this.value + ( (!omitAnchor && this.anchor) ? '#' + this.anchorString() : '' );
};
Title.prototype.anchorString=function() {
	if (!this.anchor) { return ''; }
	var split=this.anchor.parenSplit(/((?:[.][0-9A-F]{2})+)/);
	var len=split.length;
	for (var j=1; j<len; j+=2) {
		// FIXME s/decodeURI/decodeURIComponent/g ?
		split[j]=decodeURIComponent(split[j].split('.').join('%')).split('_').join(' ');
	}
	return split.join('');
};
Title.prototype.urlAnchor=function() {
	var split=this.anchor.parenSplit('/((?:[%][0-9A-F]{2})+)/');
	var len=split.length;
	for (var j=1; j<len; j+=2) {
		split[j]=split[j].split('%').join('.');
	}
	return split.join('');
};
Title.prototype.anchorFromUtf=function(str) {
	this.anchor=encodeURIComponent(str.split(' ').join('_'))
	.split('%3A').join(':').split("'").join('%27').split('%').join('.');
};
Title.fromURL=function(h) {
	return new Title().fromURL(h);
};
Title.prototype.fromURL=function(h) {
	if (typeof h != 'string') {
		this.value=null;
		return this;
	}

	// NOTE : playing with decodeURI, encodeURI, escape, unescape,
	// we seem to be able to replicate the IE borked encoding

	// IE doesn't do this new-fangled utf-8 thing.
	// and it's worse than that.
	// IE seems to treat the query string differently to the rest of the url
	// the query is treated as bona-fide utf8, but the first bit of the url is pissed around with

	// we fix up & for all browsers, just in case.
	var splitted=h.split('?');
	splitted[0]=splitted[0].split('&').join('%26');

	if (pg.flag.linksLikeIE6) {
		splitted[0]=encodeURI(decode_utf8(splitted[0]));
	}

	h=splitted.join('?');

	var contribs=pg.re.contribs.exec(h);
	if (contribs !== null) {
		if (contribs[1]=='title=') { contribs[3]=contribs[3].split('+').join(' '); }
		var u=new Title(contribs[3]);
		this.setUtf(this.decodeNasties(mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' + u.stripNamespace()));
		return this;
	}

	var email=pg.re.email.exec(h);
	if (email !== null) {
		this.setUtf(this.decodeNasties(mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' + new Title(email[3]).stripNamespace()));
		return this;
	}

	var backlinks=pg.re.backlinks.exec(h);
	if (backlinks) {
		this.setUtf(this.decodeNasties(new Title(backlinks[3])));
		return this;
	}

	//A dummy title object for a Special:Diff link.
	var specialdiff=pg.re.specialdiff.exec(h);
	if (specialdiff) {
		this.setUtf(this.decodeNasties(new Title(mw.config.get('wgFormattedNamespaces')[pg.nsSpecialId] + ':Diff')));
		return this;
	}

	// no more special cases to check --
	// hopefully it's not a disguised user-related or specially treated special page
	var m=pg.re.main.exec(h);
	if(m===null) { this.value=null; }
	else {
		var fromBotInterface = /[?](.+[&])?title=/.test(h);
		if (fromBotInterface) {
			m[2]=m[2].split('+').join('_');
		}
		var extracted = m[2] + (m[3] ? '#' + m[3] : '');
		if (pg.flag.isSafari && /%25[0-9A-Fa-f]{2}/.test(extracted)) {
			// Fix Safari issue
			// Safari sometimes encodes % as %25 in UTF-8 encoded strings like %E5%A3 -> %25E5%25A3.
			this.setUtf(decodeURIComponent(unescape(extracted)));
		} else {
			this.setUtf(this.decodeNasties(extracted));
		}
	}
	return this;
};
Title.prototype.decodeNasties=function(txt) {
	var ret= this.decodeEscapes(decodeURI(txt));
	ret = ret.replace(/[_ ]*$/, '');
	return ret;
};
Title.prototype.decodeEscapes=function(txt) {
	var split=txt.parenSplit(/((?:[%][0-9A-Fa-f]{2})+)/);
	var len=split.length;
	for (var i=1; i<len; i=i+2) {
		// FIXME is decodeURIComponent better?
		split[i]=unescape(split[i]);
	}
	return split.join('');
};
Title.fromAnchor=function(a) {
	return new Title().fromAnchor(a);
};
Title.prototype.fromAnchor=function(a) {
	if (!a) { this.value=null; return this; }
	return this.fromURL(a.href);
};
Title.fromWikiText=function(txt) {
	return new Title().fromWikiText(txt);
};
Title.prototype.fromWikiText=function(txt) {
	// FIXME - testing needed
	if (!pg.flag.linksLikeIE6) { txt=myDecodeURI(txt); }
	this.setUtf(txt);
	return this;
};
Title.prototype.hintValue=function(){
	if(!this.value) { return ''; }
	return safeDecodeURI(this.value);
};
//<NOLITE>
Title.prototype.toUserName=function(withNs) {
	if (this.namespaceId() != pg.nsUserId && this.namespaceId() != pg.nsUsertalkId) {
		this.value=null;
		return;
	}
	this.value = (withNs ? mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' : '') + this.stripNamespace().split('/')[0];
};
Title.prototype.userName=function(withNs) {
	var t=(new Title(this.value));
	t.toUserName(withNs);
	if (t.value) { return t; }
	return null;
};
Title.prototype.toTalkPage=function() {
	// convert article to a talk page, or if we can't, return null
	// In other words: return null if this ALREADY IS a talk page
	// and return the corresponding talk page otherwise
	//
	// Per http://www.mediawiki.org/wiki/Manual:Namespace#Subject_and_talk_namespaces
	// * All discussion namespaces have odd-integer indices
	// * The discussion namespace index for a specific namespace with index n is n + 1
	if (this.value===null) { return null; }
	
	var namespaceId = this.namespaceId();
	if (namespaceId>=0 && namespaceId % 2 == 0) //non-special and subject namespace
	{
		var localizedNamespace = mw.config.get('wgFormattedNamespaces')[namespaceId+1];
		if (typeof localizedNamespace!=='undefined')
		{
			if (localizedNamespace==='') return this.value = this.stripNamespace();
			this.value = localizedNamespace.split(' ').join('_') + ':' + this.stripNamespace();
			return this.value;
		}
	}

	this.value=null;
	return null;
};
//</NOLITE>
// Return canonical, localized namespace
Title.prototype.namespace=function() {
	return mw.config.get('wgFormattedNamespaces')[this.namespaceId()];
};
Title.prototype.namespaceId=function() {
	var n=this.value.indexOf(':');
	if (n<0) { return 0; } //mainspace
	var namespaceId = mw.config.get('wgNamespaceIds')[this.value.substring(0,n).split(' ').join('_').toLowerCase()];
	if (typeof namespaceId=='undefined') return 0; //mainspace
	return namespaceId;
};
//<NOLITE>
Title.prototype.talkPage=function() {
	var t=new Title(this.value);
	t.toTalkPage();
	if (t.value) { return t; }
	return null;
};
Title.prototype.isTalkPage=function() {
	if (this.talkPage()===null) { return true; }
	return false;
};
Title.prototype.toArticleFromTalkPage=function() {
	//largely copy/paste from toTalkPage above.
	if (this.value===null) { return null; }
	
	var namespaceId = this.namespaceId();
	if (namespaceId>=0 && namespaceId % 2 == 1) //non-special and talk namespace
	{
		var localizedNamespace = mw.config.get('wgFormattedNamespaces')[namespaceId-1];
		if (typeof localizedNamespace!=='undefined')
		{
			if (localizedNamespace==='') return this.value = this.stripNamespace();
			this.value = localizedNamespace.split(' ').join('_') + ':' + this.stripNamespace();
			return this.value;
		}
	}

	this.value=null;
	return null;
};
Title.prototype.articleFromTalkPage=function() {
	var t=new Title(this.value);
	t.toArticleFromTalkPage();
	if (t.value) { return t; }
	return null;
};
Title.prototype.articleFromTalkOrArticle=function() {
	var t=new Title(this.value);
	if ( t.toArticleFromTalkPage() ) { return t; }
	return this;
};
Title.prototype.isIpUser=function() {
	return pg.re.ipUser.test(this.userName());
};
//</NOLITE>
Title.prototype.stripNamespace=function(){ // returns a string, not a Title
	var n=this.value.indexOf(':');
	if (n<0) { return this.value; }
	var namespaceId = this.namespaceId();
	if (namespaceId===pg.nsMainspaceId) return this.value;
	return this.value.substring(n+1);
};
Title.prototype.setUtf=function(value){
	if (!value) { this.value=''; return; }
	var anch=value.indexOf('#');
	if(anch < 0) { this.value=value.split('_').join(' '); this.anchor=''; return; }
	this.value=value.substring(0,anch).split('_').join(' ');
	this.anchor=value.substring(anch+1);
	this.ns=null; // wait until namespace() is called
};
Title.prototype.setUrl=function(urlfrag) {
	var anch=urlfrag.indexOf('#');
	this.value=safeDecodeURI(urlfrag.substring(0,anch));
	this.anchor=value.substring(anch+1);
};
Title.prototype.append=function(x){
	this.setUtf(this.value + x);
};
Title.prototype.urlString=function(x) {
	x || ( x={} );
	var v=this.toString(true);
	if (!x.omitAnchor && this.anchor) { v+= '#' + this.urlAnchor(); }
	if (!x.keepSpaces) { v=v.split(' ').join('_'); }
	return encodeURI(v).split('&').join('%26').split('?').join('%3F').split('+').join('%2B');
};
Title.prototype.removeAnchor=function() {
	return new Title(this.toString(true));
};
Title.prototype.toUrl=function() {
	return pg.wiki.titlebase + this.urlString();
};


function paramValue(param, url) {
	var s=url.parenSplit(RegExp('[?&]' + literalizeRegex(param) + '=([^?&]*)'));
	if (!url) { return null; }
	return s[1] || null;
}

function parseParams(url) {
	var specialDiff = pg.re.specialdiff.exec(url);
	if (specialDiff!==null)
	{
		var split= specialDiff[1].split('/');
		if (split.length==1) return {oldid:split[0], diff: 'prev'};
		else if (split.length==2) return {oldid: split[0], diff: split[1]};
	}

	var ret={};
	if (url.indexOf('?')==-1) { return ret; }
	url = url.split('#')[0];
	var s=url.split('?').slice(1).join();
	var t=s.split('&');
	for (var i=0; i<t.length; ++i) {
		var z=t[i].split('=');
		z.push(null);
		ret[z[0]]=z[1];
	}
	//Diff revision with no oldid is interpreted as a diff to the previous revision by MediaWiki
	if (ret.diff && typeof(ret.oldid)==='undefined')
	{
		ret.oldid = "prev";
	}
	//Documentation seems to say something different, but oldid can also accept prev/next, and Echo is emitting such URLs. Simple fixup during parameter decoding:
	if (ret.oldid && (ret.oldid==='prev' || ret.oldid==='next' || ret.oldid==='cur'))
	{
		var helper = ret.diff;
		ret.diff = ret.oldid;
		ret.oldid = helper;
	}
	return ret;
}

// all sorts of stuff here
// FIXME almost everything needs to be rewritten

function oldidFromAnchor(a) { return paramValue('oldid', a.href); }
//function diffFromAnchor(a) { return paramValue('diff', a.href); }


function wikiMarkupToAddressFragment (str) { // for images
	var ret = safeDecodeURI(str);
	ret = ret.split(' ').join('_');
	ret = encodeURI(ret);
	return ret;
}

// (a) myDecodeURI (first standard decodeURI, then pg.re.urlNoPopup)
// (b) change spaces to underscores
// (c) encodeURI (just the straight one, no pg.re.urlNoPopup)

function myDecodeURI (str) {
	var ret;
	// FIXME decodeURIComponent??
	try { ret=decodeURI(str.toString()); }
	catch (summat) { return str; }
	for (var i=0; i<pg.misc.decodeExtras.length; ++i) {
		var from=pg.misc.decodeExtras[i].from;
		var to=pg.misc.decodeExtras[i].to;
		ret=ret.split(from).join(to);
	}
	return ret;
}

function safeDecodeURI(str) { var ret=myDecodeURI(str); return ret || str; }

///////////
// TESTS //
///////////

//<NOLITE>
function isIpUser(user) {return pg.re.ipUser.test(user);}

function isDisambig(data, article) {
	if (!getValueOf('popupAllDabsStubs') && article.namespace()) { return false; }
	return ! article.isTalkPage() && pg.re.disambig.test(data);
}

function stubCount(data, article) {
	if (!getValueOf('popupAllDabsStubs') && article.namespace()) { return false; }
	var sectStub=0;
	var realStub=0;
	if (pg.re.stub.test(data)) {
		var s=data.parenSplit(pg.re.stub);
		for (var i=1; i<s.length; i=i+2) {
			if (s[i]) { ++sectStub; }
			else { ++realStub; }
		}
	}
	return { real: realStub, sect: sectStub };
}

function isValidImageName(str){ // extend as needed...
	return ( str.indexOf('{') == -1 );
}

function isInStrippableNamespace(article) {
	//I believe that this method means to return whether the given article is in a namspace without subpages. Meaning, it's broken.
	return ( article.namespace() !== '' );
}

function isInMainNamespace(article) { return !isInStrippableNamespace(article); }

function anchorContainsImage(a) {
	// iterate over children of anchor a
	// see if any are images
	if (a===null) { return false; }
	kids=a.childNodes;
	for (var i=0; i<kids.length; ++i) { if (kids[i].nodeName=='IMG') { return true; } }
	return false;
}
//</NOLITE>
function isPopupLink(a) {
	// NB for performance reasons, TOC links generally return true
	// they should be stripped out later

	if (!markNopopupSpanLinks.done) { markNopopupSpanLinks(); }
	if (a.inNopopupSpan) { return false; }

	// FIXME is this faster inline?
	if (a.onmousedown || a.getAttribute('nopopup')) { return false; }
	var h=a.href;
	if (h===document.location.href+'#') { return false; }
	if (!pg.re.basenames.test(h)) { return false; }
	if (!pg.re.urlNoPopup.test(h)) { return true;	}
	return (
		(pg.re.email.test(h) || pg.re.contribs.test(h) || pg.re.backlinks.test(h) || pg.re.specialdiff.test(h)) &&
		h.indexOf('&limit=') == -1 );
}

function markNopopupSpanLinks() {
	if( !getValueOf('popupOnlyArticleLinks'))
		fixVectorMenuPopups();

	var s = $('.nopopups').toArray();
	for (var i=0; i<s.length; ++i) {
		var as=s[i].getElementsByTagName('a');
		for (var j=0; j<as.length; ++j) {
			as[j].inNopopupSpan=true;
		}
	}
	
	markNopopupSpanLinks.done=true;
}

function fixVectorMenuPopups() {
	$('div.vectorMenu h3:first a:first').prop('inNopopupSpan', true);
}
// ENDFILE: titles.js
// STARTFILE: cookies.js
//<NOLITE>
//////////////////////////////////////////////////
// Cookie handling
// from http://www.quirksmode.org/js/cookies.html

var Cookie= {
	create: function(name,value,days)
	{
		var expires;
		if (days)
		{
			var date = new Date();
			date.setTime(date.getTime()+(days*24*60*60*1000));
			expires = "; expires="+date.toGMTString();
		}
		else { expires = ""; }
		document.cookie = name+"="+value+expires+"; path=/";
	},

	read: function(name)
	{
		var nameEQ = name + "=";
		var ca = document.cookie.split(';');
		for(var i=0;i < ca.length;i++)
		{
			var c = ca[i];
			while (c.charAt(0)==' ') { c = c.substring(1,c.length); }
			if (c.indexOf(nameEQ) === 0) { return c.substring(nameEQ.length,c.length); }
		}
		return null;
	},

	erase: function(name)
	{
		Cookie.create(name,"",-1);
	}
};
//</NOLITE>
// ENDFILE: cookies.js
// STARTFILE: getpage.js
//////////////////////////////////////////////////
// Wiki-specific downloading
//

// Schematic for a getWiki call
//
//   getWiki->-getPageWithCaching
//					|
//	   false		|		  true
// getPage<-[findPictureInCache]->-onComplete(a fake download)
//   \.
//	 (async)->addPageToCache(download)->-onComplete(download)


/** @todo {document}
	@param {Title} article
	@param {Function} onComplete
	@param {integer} oldid
	@param {Navapopup} owner
*/
function getWiki(article, onComplete, oldid, owner) {
	// set ctype=text/css to get around opera gzip bug
	var url = pg.wiki.titlebase;
	if (article.namespaceId() >= 0)
		url += article.removeAnchor().urlString();
	if (oldid || oldid===0 || oldid==='0')
		url += '&oldid='+oldid;
	url += '&action=raw&ctype=text/css&maxage=0&smaxage=0';

	getPageWithCaching(url, onComplete, owner);
}

// check cache to see if page exists

function getPageWithCaching(url, onComplete, owner) {
	log('getPageWithCaching, url='+url);
	var i=findInPageCache(url);
	if (i > -1) {
		var d=fakeDownload(url, owner.idNumber, onComplete,
				   pg.cache.pages[i].data, pg.cache.pages[i].lastModified,
				   owner);
	} else {
		var d=getPage(url, onComplete, owner);
		if (d && owner && owner.addDownload) {
			owner.addDownload(d);
			d.owner=owner;
		}
	}
}

function getPage(url, onComplete, owner) {
	log('getPage');
	var callback= function (d) { if (!d.aborted) {addPageToCache(d); onComplete(d);} };
	return startDownload(url, owner.idNumber, callback);
}

function findInPageCache(url) {
	for (var i=0; i<pg.cache.pages.length; ++i) {
		if (url==pg.cache.pages[i].url) { return i; }
	}
	return -1;
}

function addPageToCache(download) {
	log('addPageToCache '+download.url);
	var page = {url: download.url, data: download.data, lastModified: download.lastModified};
	return pg.cache.pages.push(page);
}
// ENDFILE: getpage.js
// STARTFILE: md5-2.2alpha.js
//<NOLITE>
/*
 * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
 * Digest Algorithm, as defined in RFC 1321.
 * Version 2.2-alpha Copyright (C) Paul Johnston 1999 - 2005
 * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
 * Distributed under the BSD License
 * See http://pajhome.org.uk/crypt/md5 for more info.
 */

/*
 * Configurable variables. You may need to tweak these to be compatible with
 * the server-side, but the defaults work in most cases.
 */
var hexcase = 0;   /* hex output format. 0 - lowercase; 1 - uppercase		 */
var b64pad  = ""; /* base-64 pad character. "=" for strict RFC compliance   */

/*
 * These are the functions you'll usually want to call
 * They take string arguments and return either hex or base-64 encoded strings
 */
function hex_md5(s)	{ return rstr2hex(rstr_md5(str2rstr_utf8(s))); }
function b64_md5(s)	{ return rstr2b64(rstr_md5(str2rstr_utf8(s))); }
function any_md5(s, e) { return rstr2any(rstr_md5(str2rstr_utf8(s)), e); }
function hex_hmac_md5(k, d)
  { return rstr2hex(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
function b64_hmac_md5(k, d)
  { return rstr2b64(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
function any_hmac_md5(k, d, e)
  { return rstr2any(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d)), e); }

/*
 * Perform a simple self-test to see if the VM is working
 */
function md5_vm_test()
{
  return hex_md5("abc") == "900150983cd24fb0d6963f7d28e17f72";
}

/*
 * Calculate the MD5 of a raw string
 */
function rstr_md5(s)
{
  return binl2rstr(binl_md5(rstr2binl(s), s.length * 8));
}

/*
 * Calculate the HMAC-MD5, of a key and some data (raw strings)
 */
function rstr_hmac_md5(key, data)
{
  var bkey = rstr2binl(key);
  if(bkey.length > 16) bkey = binl_md5(bkey, key.length * 8);

  var ipad = Array(16), opad = Array(16);
  for(var i = 0; i < 16; i++)
  {
	ipad[i] = bkey[i] ^ 0x36363636;
	opad[i] = bkey[i] ^ 0x5C5C5C5C;
  }

  var hash = binl_md5(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
  return binl2rstr(binl_md5(opad.concat(hash), 512 + 128));
}

/*
 * Convert a raw string to a hex string
 */
function rstr2hex(input)
{
  var hex_tab = hexcase ? "0123456789ABCDEF" : "0123456789abcdef";
  var output = "";
  var x;
  for(var i = 0; i < input.length; i++)
  {
	x = input.charCodeAt(i);
	output += hex_tab.charAt((x >>> 4) & 0x0F)
	   +  hex_tab.charAt( x		   & 0x0F);
  }
  return output;
}

/*
 * Convert a raw string to a base-64 string
 */
function rstr2b64(input)
{
  var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
  var output = "";
  var len = input.length;
  for(var i = 0; i < len; i += 3)
  {
	var triplet = (input.charCodeAt(i) << 16)
		| (i + 1 < len ? input.charCodeAt(i+1) << 8 : 0)
		| (i + 2 < len ? input.charCodeAt(i+2)		: 0);
	for(var j = 0; j < 4; j++)
	{
	  if(i * 8 + j * 6 > input.length * 8) output += b64pad;
	  else output += tab.charAt((triplet >>> 6*(3-j)) & 0x3F);
	}
  }
  return output;
}

/*
 * Convert a raw string to an arbitrary string encoding
 */
function rstr2any(input, encoding)
{
  var divisor = encoding.length;
  var remainders = Array();
  var i, q, x, quotient;

  /* Convert to an array of 16-bit big-endian values, forming the dividend */
  var dividend = Array(input.length / 2);
  for(i = 0; i < dividend.length; i++)
  {
	dividend[i] = (input.charCodeAt(i * 2) << 8) | input.charCodeAt(i * 2 + 1);
  }

  /*
   * Repeatedly perform a long division. The binary array forms the dividend,
   * the length of the encoding is the divisor. Once computed, the quotient
   * forms the dividend for the next step. We stop when the dividend is zero.
   * All remainders are stored for later use.
   */
  while(dividend.length > 0)
  {
	quotient = Array();
	x = 0;
	for(i = 0; i < dividend.length; i++)
	{
	  x = (x << 16) + dividend[i];
	  q = Math.floor(x / divisor);
	  x -= q * divisor;
	  if(quotient.length > 0 || q > 0)
	quotient[quotient.length] = q;
	}
	remainders[remainders.length] = x;
	dividend = quotient;
  }

  /* Convert the remainders to the output string */
  var output = "";
  for(i = remainders.length - 1; i >= 0; i--)
	output += encoding.charAt(remainders[i]);

  return output;
}

/*
 * Encode a string as utf-8.
 * For efficiency, this assumes the input is valid utf-16.
 */
function str2rstr_utf8(input)
{
  var output = "";
  var i = -1;
  var x, y;

  while(++i < input.length)
  {
	/* Decode utf-16 surrogate pairs */
	x = input.charCodeAt(i);
	y = i + 1 < input.length ? input.charCodeAt(i + 1) : 0;
	if(0xD800 <= x && x <= 0xDBFF && 0xDC00 <= y && y <= 0xDFFF)
	{
	  x = 0x10000 + ((x & 0x03FF) << 10) + (y & 0x03FF);
	  i++;
	}

	/* Encode output as utf-8 */
	if(x <= 0x7F)
	  output += String.fromCharCode(x);
	else if(x <= 0x7FF)
	  output += String.fromCharCode(0xC0 | ((x >>> 6 ) & 0x1F),
					0x80 | ( x		   & 0x3F));
	else if(x <= 0xFFFF)
	  output += String.fromCharCode(0xE0 | ((x >>> 12) & 0x0F),
					0x80 | ((x >>> 6 ) & 0x3F),
					0x80 | ( x		   & 0x3F));
	else if(x <= 0x1FFFFF)
	  output += String.fromCharCode(0xF0 | ((x >>> 18) & 0x07),
					0x80 | ((x >>> 12) & 0x3F),
					0x80 | ((x >>> 6 ) & 0x3F),
					0x80 | ( x		   & 0x3F));
  }
  return output;
}

/*
 * Encode a string as utf-16
 */
function str2rstr_utf16le(input)
{
  var output = "";
  for(var i = 0; i < input.length; i++)
	output += String.fromCharCode( input.charCodeAt(i)		  & 0xFF,
				  (input.charCodeAt(i) >>> 8) & 0xFF);
  return output;
}

function str2rstr_utf16be(input)
{
  var output = "";
  for(var i = 0; i < input.length; i++)
	output += String.fromCharCode((input.charCodeAt(i) >>> 8) & 0xFF,
				   input.charCodeAt(i)		  & 0xFF);
  return output;
}

/*
 * Convert a raw string to an array of little-endian words
 * Characters >255 have their high-byte silently ignored.
 */
function rstr2binl(input)
{
  var output = Array(input.length >> 2);
  for(var i = 0; i < output.length; i++)
	output[i] = 0;
  for(var i = 0; i < input.length * 8; i += 8)
	output[i>>5] |= (input.charCodeAt(i / 8) & 0xFF) << (i%32);
  return output;
}

/*
 * Convert an array of little-endian words to a string
 */
function binl2rstr(input)
{
  var output = "";
  for(var i = 0; i < input.length * 32; i += 8)
	output += String.fromCharCode((input[i>>5] >>> (i % 32)) & 0xFF);
  return output;
}

/*
 * Calculate the MD5 of an array of little-endian words, and a bit length.
 */
function binl_md5(x, len)
{
  /* append padding */
  x[len >> 5] |= 0x80 << ((len) % 32);
  x[(((len + 64) >>> 9) << 4) + 14] = len;

  var a =  1732584193;
  var b = -271733879;
  var c = -1732584194;
  var d =  271733878;

  for(var i = 0; i < x.length; i += 16)
  {
	var olda = a;
	var oldb = b;
	var oldc = c;
	var oldd = d;

	a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936);
	d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586);
	c = md5_ff(c, d, a, b, x[i+ 2], 17,	 606105819);
	b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330);
	a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897);
	d = md5_ff(d, a, b, c, x[i+ 5], 12,	 1200080426);
	c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341);
	b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983);
	a = md5_ff(a, b, c, d, x[i+ 8], 7 ,	 1770035416);
	d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417);
	c = md5_ff(c, d, a, b, x[i+10], 17, -42063);
	b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162);
	a = md5_ff(a, b, c, d, x[i+12], 7 ,	 1804603682);
	d = md5_ff(d, a, b, c, x[i+13], 12, -40341101);
	c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290);
	b = md5_ff(b, c, d, a, x[i+15], 22,	 1236535329);

	a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510);
	d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632);
	c = md5_gg(c, d, a, b, x[i+11], 14,	 643717713);
	b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302);
	a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691);
	d = md5_gg(d, a, b, c, x[i+10], 9 ,	 38016083);
	c = md5_gg(c, d, a, b, x[i+15], 14, -660478335);
	b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848);
	a = md5_gg(a, b, c, d, x[i+ 9], 5 ,	 568446438);
	d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690);
	c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961);
	b = md5_gg(b, c, d, a, x[i+ 8], 20,	 1163531501);
	a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467);
	d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784);
	c = md5_gg(c, d, a, b, x[i+ 7], 14,	 1735328473);
	b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734);

	a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558);
	d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463);
	c = md5_hh(c, d, a, b, x[i+11], 16,	 1839030562);
	b = md5_hh(b, c, d, a, x[i+14], 23, -35309556);
	a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060);
	d = md5_hh(d, a, b, c, x[i+ 4], 11,	 1272893353);
	c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632);
	b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640);
	a = md5_hh(a, b, c, d, x[i+13], 4 ,	 681279174);
	d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222);
	c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979);
	b = md5_hh(b, c, d, a, x[i+ 6], 23,	 76029189);
	a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487);
	d = md5_hh(d, a, b, c, x[i+12], 11, -421815835);
	c = md5_hh(c, d, a, b, x[i+15], 16,	 530742520);
	b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651);

	a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844);
	d = md5_ii(d, a, b, c, x[i+ 7], 10,	 1126891415);
	c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905);
	b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055);
	a = md5_ii(a, b, c, d, x[i+12], 6 ,	 1700485571);
	d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606);
	c = md5_ii(c, d, a, b, x[i+10], 15, -1051523);
	b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799);
	a = md5_ii(a, b, c, d, x[i+ 8], 6 ,	 1873313359);
	d = md5_ii(d, a, b, c, x[i+15], 10, -30611744);
	c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380);
	b = md5_ii(b, c, d, a, x[i+13], 21,	 1309151649);
	a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070);
	d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379);
	c = md5_ii(c, d, a, b, x[i+ 2], 15,	 718787259);
	b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551);

	a = safe_add(a, olda);
	b = safe_add(b, oldb);
	c = safe_add(c, oldc);
	d = safe_add(d, oldd);
  }
  return Array(a, b, c, d);
}

/*
 * These functions implement the four basic operations the algorithm uses.
 */
function md5_cmn(q, a, b, x, s, t)
{
  return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s),b);
}
function md5_ff(a, b, c, d, x, s, t)
{
  return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t);
}
function md5_gg(a, b, c, d, x, s, t)
{
  return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t);
}
function md5_hh(a, b, c, d, x, s, t)
{
  return md5_cmn(b ^ c ^ d, a, b, x, s, t);
}
function md5_ii(a, b, c, d, x, s, t)
{
  return md5_cmn(c ^ (b | (~d)), a, b, x, s, t);
}

/*
 * Add integers, wrapping at 2^32. This uses 16-bit operations internally
 * to work around bugs in some JS interpreters.
 */
function safe_add(x, y)
{
  var lsw = (x & 0xFFFF) + (y & 0xFFFF);
  var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
  return (msw << 16) | (lsw & 0xFFFF);
}

/*
 * Bitwise rotate a 32-bit number to the left.
 */
function bit_rol(num, cnt)
{
  return (num << cnt) | (num >>> (32 - cnt));
}
//</NOLITE>
// ENDFILE: md5-2.2alpha.js
// STARTFILE: parensplit.js
//////////////////////////////////////////////////
// parenSplit

// String.prototype.parenSplit should do what ECMAscript says String.prototype.split does,
// interspersing paren matches (regex capturing groups) between the split elements.
// i.e. 'abc'.split(/(b)/)) should return ['a','b','c'], not ['a','c']

if (String('abc'.split(/(b)/))!='a,b,c') {
	// broken String.split, e.g. konq, IE
	String.prototype.parenSplit=function (re) {
		re=nonGlobalRegex(re);
		var s=this;
		var m=re.exec(s);
		var ret=[];
		while (m && s) {
			// without the following loop, we have
			// 'ab'.parenSplit(/a|(b)/) != 'ab'.split(/a|(b)/)
			for(var i=0; i<m.length; ++i) {
				if (typeof m[i]=='undefined') m[i]='';
			}
			ret.push(s.substring(0,m.index));
			ret = ret.concat(m.slice(1));
			s=s.substring(m.index + m[0].length);
			m=re.exec(s);
		}
		ret.push(s);
		return ret;
	};
} else {
	String.prototype.parenSplit=function (re) { return this.split(re); };
	String.prototype.parenSplit.isNative=true;
}

function nonGlobalRegex(re) {
	var s=re.toString();
	flags='';
	for (var j=s.length; s.charAt(j) != '/'; --j) {
		if (s.charAt(j) != 'g') { flags += s.charAt(j); }
	}
	var t=s.substring(1,j);
	return RegExp(t,flags);
}
// ENDFILE: parensplit.js
// STARTFILE: tools.js
// IE madness with encoding
// ========================
//
// suppose throughout that the page is in utf8, like wikipedia
//
// if a is an anchor DOM element and a.href should consist of
//
// http://host.name.here/wiki/foo?bar=baz
//
// then IE gives foo as "latin1-encoded" utf8; we have foo = decode_utf8(decodeURI(foo_ie))
// but IE gives bar=baz correctly as plain utf8
//
// ---------------------------------
//
// IE's xmlhttp doesn't understand utf8 urls. Have to use encodeURI here.
//
// ---------------------------------
//
// summat else

// Source: http://aktuell.de.selfhtml.org/artikel/javascript/utf8b64/utf8.htm

//<NOLITE>
function encode_utf8(rohtext) {
	// dient der Normalisierung des Zeilenumbruchs
	rohtext = rohtext.replace(/\r\n/g,"\n");
	var utftext = "";
	for(var n=0; n<rohtext.length; n++)
	{
		// ermitteln des Unicodes des  aktuellen Zeichens
		var c=rohtext.charCodeAt(n);
		// alle Zeichen von 0-127 => 1byte
		if (c<128)
			utftext += String.fromCharCode(c);
		// alle Zeichen von 127 bis 2047 => 2byte
		else if((c>127) && (c<2048)) {
			utftext += String.fromCharCode((c>>6)|192);
			utftext += String.fromCharCode((c&63)|128);}
		// alle Zeichen von 2048 bis 66536 => 3byte
		else {
			utftext += String.fromCharCode((c>>12)|224);
			utftext += String.fromCharCode(((c>>6)&63)|128);
			utftext += String.fromCharCode((c&63)|128);}
	}
	return utftext;
}

function getJsObj(json) {
	try {
		var json_ret = JSON.parse(json);
	} catch (someError) {
		errlog('Something went wrong with getJsobj, json='+json);
		return 1;
	}
	if( json_ret['warnings'] ) {
		for( var w=0; w < json_ret['warnings'].length; w++ ) {
			log( json_ret['warnings'][w]['*'] );
		}
	} else if ( json_ret['error'] ) {
		errlog( json_ret['error'].code + ': ' + json_ret['error'].info );
	}
	return json_ret;
}

function anyChild(obj) {
	for (var p in obj) {
		return obj[p];
	}
	return null;
}

//</NOLITE>

function decode_utf8(utftext) {
	var plaintext = ""; var i=0, c=0, c1=0, c2=0;
	// while-Schleife, weil einige Zeichen uebersprungen werden
	while(i<utftext.length)
	{
		c = utftext.charCodeAt(i);
		if (c<128) {
			plaintext += String.fromCharCode(c);
			i++;}
		else if((c>191) && (c<224)) {
			c2 = utftext.charCodeAt(i+1);
			plaintext += String.fromCharCode(((c&31)<<6) | (c2&63));
			i+=2;}
		else {
			c2 = utftext.charCodeAt(i+1); c3 = utftext.charCodeAt(i+2);
			plaintext += String.fromCharCode(((c&15)<<12) | ((c2&63)<<6) | (c3&63));
			i+=3;}
	}
	return plaintext;
}


function upcaseFirst(str) {
	if (typeof str != typeof '' || str=='') return '';
	return str.charAt(0).toUpperCase() + str.substring(1);
}


function findInArray(arr, foo) {
	if (!arr || !arr.length) { return -1; }
	var len=arr.length;
	for (var i=0; i<len; ++i) { if (arr[i]==foo) { return i; } }
	return -1;
}

function nextOne (array, value) {
	// NB if the array has two consecutive entries equal
	//	then this will loop on successive calls
	var i=findInArray(array, value);
	if (i<0) { return null; }
	return array[i+1];
}

function literalizeRegex(str){
	return str.replace(RegExp('([-.|()\\\\+?*^${}\\[\\]])', 'g'), '\\$1');
}

String.prototype.entify=function() {
	//var shy='&shy;';
	return this.split('&').join('&amp;').split('<').join('&lt;').split('>').join('&gt;'/*+shy*/).split('"').join('&quot;');
};

function findThis(array, value) {
	if (typeof array.length == 'undefined') { return null; }
	for (var i=0; i<array.length; ++i) {
		if (array[i]==value) { return i; }
	}
	return null;
}

function removeNulls(list) {
	var ret=[];
	for (var i=0; i<list.length; ++i) {
		if (list[i]) {
			ret.push(list[i]);
		}
	}
	return ret;
}
function joinPath(list) {
	return removeNulls(list).join('/');
}


function simplePrintf(str, subs) {
	if (!str || !subs) { return str; }
	var ret=[];
	var s=str.parenSplit(/(%s|\$[0-9]+)/);
	var i=0;
	do {
		ret.push(s.shift());
		if ( !s.length ) { break; }
		var cmd=s.shift();
		if (cmd == '%s') {
			if ( i < subs.length ) { ret.push(subs[i]); } else { ret.push(cmd); }
			++i;
		} else {
			var j=parseInt( cmd.replace('$', ''), 10 ) - 1;
			if ( j > -1 && j < subs.length ) { ret.push(subs[j]); } else { ret.push(cmd); }
		}
	} while (s.length > 0);
	return ret.join('');
}

function max(a,b){return a<b ? b : a;}
function min(a,b){return a>b ? b : a;}

function isString(x) { return (typeof x === 'string' || x instanceof String); }
//function isNumber(x) { return (typeof x === 'number' || x instanceof Number); }
function isRegExp(x) { return x instanceof RegExp; }
function isArray (x) { return x instanceof Array; }
function isObject(x) { return x instanceof Object; }
function isFunction(x) {
	return !isRegExp(x) && ($.isFunction(x) || x instanceof Function);
}

function repeatString(s,mult) {
	var ret='';
	for (var i=0; i<mult; ++i) { ret += s; }
	return ret;
}

function zeroFill(s, min) {
	min = min || 2;
	var t=s.toString();
	return repeatString('0', min - t.length) + t;
}

function map(f, o) {
	if (isArray(o)) { return map_array(f,o); }
	return map_object(f,o);
}
function map_array(f,o) {
	var ret=[];
	for (var i=0; i<o.length; ++i) {
		ret.push(f(o[i]));
	}
	return ret;
}
function map_object(f,o) {
	var ret={};
	for (var i in o) { ret[o]=f(o[i]); }
	return ret;
}

pg.escapeQuotesHTML = function ( text ) {	var re = new RegExp( '&', "g" );	text = text.replace( re, "&amp;" );	re = new RegExp( '"', "g" );	text = text.replace( re, "&quot;" );	re = new RegExp( '<', "g" );	text = text.replace( re, "&lt;" );	re = new RegExp( '>', "g" );	text = text.replace( re, "&gt;" );	return text;}

pg.jsescape = function(s)
{
	if (typeof s !== "string") throw "Invalid type in pg.jsescape";
	var res = "";
	//this can be optimized by copying substrings instead of char by char!
	for (var i=0; i<s.length; i++)
	{
		var c = s[i];
	switch (c)
	{
	  case '\b': res += '\\b'; continue;
	  case '\f': res += '\\f'; continue;
	  case '\n': res += '\\n'; continue;
	  case '\0': res += '\\0'; continue;
	  case '\r': res += '\\r'; continue;
	  case '\t': res += '\\t'; continue;
	  case '\v': res += '\\v'; continue;
	  case '\\': res += '\\\\'; continue;
	  case '\"':  res += '\\\"'; continue;
	  case '\'':  res += '\\\''; continue;
		   continue;
	  default:
			if (c < ' ' || c==='<' || c==='>' || c==="'")
			{
					var unicodeChar = c.charCodeAt(0).toString(16).toUpperCase();
					res += "\\u" + (unicodeChar.length>1?"00":"000") + unicodeChar;
			}
			else
			{
				res += c;
			}
	}
  }
  return res;
}

// ENDFILE: tools.js
// STARTFILE: dab.js
//<NOLITE>
//////////////////////////////////////////////////
// Dab-fixing code
//


function retargetDab(newTarget, oldTarget, friendlyCurrentArticleName, titleToEdit) {
	log('retargetDab: newTarget='+newTarget + ' oldTarget=' + oldTarget);
	return changeLinkTargetLink(
	{newTarget: newTarget,
			text: newTarget.split(' ').join('&nbsp;'),
			hint: tprintf('disambigHint', [newTarget]),
			summary: simplePrintf(
					getValueOf('popupFixDabsSummary'), [friendlyCurrentArticleName, newTarget ]),
			clickButton: getValueOf('popupDabsAutoClick'), minor: true, oldTarget: oldTarget,
			watch: getValueOf('popupWatchDisambiggedPages'),
			title: titleToEdit});
}

function listLinks(wikitext, oldTarget, titleToEdit) {
	// mediawiki strips trailing spaces, so we do the same
	// testcase: http://en.wikipedia.org/w/index.php?title=Radial&oldid=97365633
	var reg=RegExp('\\[\\[([^|]*?) *(\\||\\]\\])', 'gi');
	var ret=[];
	var splitted=wikitext.parenSplit(reg);
	// ^[a-z]+ should match interwiki links, hopefully (case-insensitive)
	// and ^[a-z]* should match those and [[:Category...]] style links too
	var omitRegex=RegExp('^[a-z]*:|^[Ss]pecial:|^[Ii]mage|^[Cc]ategory');
	var friendlyCurrentArticleName= oldTarget.toString();
	var wikPos = getValueOf('popupDabWiktionary');

	for (var i=1; i<splitted.length; i=i+3) {
		if (typeof splitted[i] == typeof 'string' && splitted[i].length>0 && !omitRegex.test(splitted[i])) {
			ret.push( retargetDab(splitted[i], oldTarget, friendlyCurrentArticleName, titleToEdit) );
		} /* if */
	} /* for loop */

	ret = rmDupesFromSortedList(ret.sort());

	if (wikPos) {
		var wikTarget='wiktionary:' +
			friendlyCurrentArticleName.replace( RegExp('^(.+)\\s+[(][^)]+[)]\\s*$'), '$1' );

		var meth;
		if (wikPos.toLowerCase() == 'first') { meth = 'unshift'; }
		else { meth = 'push'; }

		ret[meth]( retargetDab(wikTarget, oldTarget, friendlyCurrentArticleName, titleToEdit) );
	}

	ret.push(changeLinkTargetLink(
	{ newTarget: null,
			text: popupString('remove this link').split(' ').join('&nbsp;'),
			hint: popupString("remove all links to this disambig page from this article"),
			clickButton: "wpDiff", oldTarget: oldTarget,
			summary: simplePrintf(getValueOf('popupRmDabLinkSummary'), [friendlyCurrentArticleName]),
			watch: getValueOf('popupWatchDisambiggedPages'),
			title: titleToEdit
			}));
	return ret;
}

function rmDupesFromSortedList(list) {
	var ret=[];
	for (var i=0; i<list.length; ++i) {
		if (ret.length===0 || list[i]!=ret[ret.length-1]) { ret.push(list[i]); }
	}
	return ret;
}

function makeFixDab(data, navpop) {
	// grab title from parent popup if there is one; default exists in changeLinkTargetLink
	var titleToEdit=(navpop.parentPopup && navpop.parentPopup.article.toString());
	var list=listLinks(data, navpop.originalArticle, titleToEdit);
	if (list.length===0) { log('listLinks returned empty list'); return null; }
	var html='<hr />' + popupString('Click to disambiguate this link to:') + '<br>';
	html+=list.join(', ');
	return html;
}


function makeFixDabs(wikiText, navpop) {
	if (getValueOf('popupFixDabs') && isDisambig(wikiText, navpop.article) &&
		Title.fromURL(location.href).namespaceId() != pg.nsSpecialId &&
		navpop.article.talkPage() ) {
		setPopupHTML(makeFixDab(wikiText, navpop), 'popupFixDab', navpop.idNumber);
	}
}

function popupRedlinkHTML(article) {
	return changeLinkTargetLink(
		{ newTarget: null, text: popupString('remove this link').split(' ').join('&nbsp;'),
			hint: popupString("remove all links to this page from this article"),
			clickButton: "wpDiff",
			oldTarget: article.toString(),
			summary: simplePrintf(getValueOf('popupRedlinkSummary'), [article.toString()])});
}
//</NOLITE>
// ENDFILE: dab.js
// STARTFILE: htmloutput.js

function appendPopupContent(obj, elementId, popupId, onSuccess) {
	return setPopupHTML(obj, elementId, popupId, onSuccess, true);
}

// this has to use a timer loop as we don't know if the DOM element exists when we want to set the text
function setPopupHTML (str, elementId, popupId, onSuccess, append) {
	if (elementId=='popupPreview') {
	}
	if (typeof popupId === 'undefined') {
		//console.error('popupId is not defined in setPopupHTML, html='+str.substring(0,100));
		popupId = pg.idNumber;
	}

	var popupElement=document.getElementById(elementId+popupId);
	if (popupElement) {
		if (!append) { popupElement.innerHTML=''; }
		if (isString(str)) {
			popupElement.innerHTML+=str;
		} else {
			popupElement.appendChild(str);
		}
		if (onSuccess) { onSuccess(); }
		setTimeout(checkPopupPosition, 100);
		return true;
	} else {
		// call this function again in a little while...
		setTimeout(function(){
				setPopupHTML(str,elementId,popupId,onSuccess);
			}, 600);
	}
	return null;
}

//<NOLITE>
function setPopupTrailer(str,id) {return setPopupHTML(str, 'popupData', id);}
//</NOLITE>


function fillEmptySpans(args) { return fillEmptySpans2(args); }

// args.navpopup is mandatory
// optional: args.redir, args.redirTarget
// FIXME: ye gods, this is ugly stuff
function fillEmptySpans2(args) { // if redir is present and true then redirTarget is mandatory
	var redir=true;
	if (typeof args != 'object' || typeof args.redir == 'undefined' || !args.redir) { redir=false; }
	var a=args.navpopup.parentAnchor;

	var article, hint=null, oldid=null, params={};
	if (redir && typeof args.redirTarget == typeof {}) {
		article=args.redirTarget;
		//hint=article.hintValue();
	} else {
		article=(new Title()).fromAnchor(a);
		hint=a.originalTitle || article.hintValue();
		params=parseParams(a.href);
		oldid=(getValueOf('popupHistoricalLinks')) ? params.oldid : null;
		rcid=params.rcid;
	}
	var x={ article:article, hint: hint, oldid: oldid, rcid: rcid, navpop:args.navpopup, params:params };

	var structure=pg.structures[getValueOf('popupStructure')];
	if (typeof structure != 'object') {
		setPopupHTML('popupError', 'Unknown structure (this should never happen): '+
				 pg.option.popupStructure, args.navpopup.idNumber);
		return;
	}
	var spans=flatten(pg.misc.layout);
	var numspans = spans.length;
	var redirs=pg.misc.redirSpans;

	for (var i=0; i<numspans; ++i) {
		var f=findThis(redirs, spans[i]);
		//log('redir='+redir+', f='+f+', spans[i]='+spans[i]);
		if ( (f!==null && !redir) || (f===null && redir) ) {
			//log('skipping this set of the loop');
			continue;
		}
		var structurefn=structure[spans[i]];
		var setfn = setPopupHTML;
		if (getValueOf('popupActiveNavlinks') && 
			(spans[i].indexOf('popupTopLinks')==0 || spans[i].indexOf('popupRedirTopLinks')==0)
				) {
			setfn = setPopupTipsAndHTML;
		}
		switch (typeof structurefn) {
		case 'function':
			//log('running '+spans[i]+'({article:'+x.article+', hint:'+x.hint+', oldid: '+x.oldid+'})');
			setfn(structurefn(x), spans[i], args.navpopup.idNumber);
			break;
		case 'string':
			setfn(structurefn, spans[i], args.navpopup.idNumber);
			break;
		default:
			errlog('unknown thing with label '+spans[i]);
			break;
		}
	}
}

// flatten an array
function flatten(list, start) {
	var ret=[];
	if (typeof start == 'undefined') { start=0; }
	for (var i=start; i<list.length; ++i) {
		if (typeof list[i] == typeof []) {
			return ret.concat(flatten(list[i])).concat(flatten(list, i+1));
		}
		else { ret.push(list[i]); }
	}
	return ret;
}

// Generate html for whole popup
function popupHTML (a) {
	getValueOf('popupStructure');
	var structure=pg.structures[pg.option.popupStructure];
	if (typeof structure != 'object') {
		//return 'Unknown structure: '+pg.option.popupStructure;
		// override user choice
		pg.option.popupStructure=pg.optionDefault.popupStructure;
		return popupHTML(a);
	}
	if (typeof structure.popupLayout != 'function') { return 'Bad layout'; }
	pg.misc.layout=structure.popupLayout();
	if ($.isFunction(structure.popupRedirSpans)) { pg.misc.redirSpans=structure.popupRedirSpans(); }
	else { pg.misc.redirSpans=[]; }
	return makeEmptySpans(pg.misc.layout, a.navpopup);
}

function makeEmptySpans (list, navpop) {
	var ret='';
	for (var i=0; i<list.length; ++i) {
		if (typeof list[i] == typeof '') {
			ret += emptySpanHTML(list[i], navpop.idNumber, 'div');
		} else if (typeof list[i] == typeof [] && list[i].length > 0 ) {
			ret = ret.parenSplit(RegExp('(</[^>]*?>$)')).join(makeEmptySpans(list[i], navpop));
		} else if (typeof list[i] == typeof {} && list[i].nodeType ) {
			ret += emptySpanHTML(list[i].name, navpop.idNumber, list[i].nodeType);
		}
	}
	return ret;
}


function emptySpanHTML(name, id, tag, classname) {
	tag = tag || 'span';
	if (!classname) { classname = emptySpanHTML.classAliases[name]; }
	classname = classname || name;
	if (name == getValueOf('popupDragHandle')) { classname += ' popupDragHandle'; }
	return simplePrintf('<%s id="%s" class="%s"></%s>', [tag, name + id, classname, tag]);
}
emptySpanHTML.classAliases={ 'popupSecondPreview': 'popupPreview' };

// generate html for popup image
// <a id="popupImageLinkn"><img id="popupImagen">
// where n=idNumber
function imageHTML(article, idNumber) {
	return simplePrintf('<a id="popupImageLink$1">' +
				'<img align="right" valign="top" id="popupImg$1" style="display: none;"></img>' +
				'</a>', [ idNumber ]);
}

function popTipsSoonFn(id, when, popData) {
	when || ( when=250 );
	var popTips=function(){ setupTooltips(document.getElementById(id), false, true, popData); };
	return function() { setTimeout( popTips, when, popData ); };
}

function setPopupTipsAndHTML(html, divname, idnumber, popData) {
	setPopupHTML(html, divname, idnumber,
			 getValueOf('popupSubpopups') ? 
			 popTipsSoonFn(divname + idnumber, null, popData) : 
			 null);
}
// ENDFILE: htmloutput.js
// STARTFILE: mouseout.js
//////////////////////////////////////////////////
// fuzzy checks

function fuzzyCursorOffMenus(x,y, fuzz, parent) {
	if (!parent) { return null; }
	var uls=parent.getElementsByTagName('ul');
	for (var i=0; i<uls.length; ++i) {
		if (uls[i].className=='popup_menu') {
			if (uls[i].offsetWidth > 0) return false;
		} // else {document.title+='.';}
	}
	return true;
}

function checkPopupPosition () { // stop the popup running off the right of the screen
	// FIXME avoid pg.current.link
	pg.current.link && pg.current.link.navpopup &&
		pg.current.link.navpopup.limitHorizontalPosition();
}

function mouseOutWikiLink () {
	if (!window.popupsReady || !window.popupsReady()) { return; }
	//console ('mouseOutWikiLink');
	var a=this;
	if (a.navpopup==null) return;
	if ( ! a.navpopup.isVisible() ) {
		a.navpopup.banish();
		return;
	}
		restoreTitle(a);
	Navpopup.tracker.addHook(posCheckerHook(a.navpopup));
}

function posCheckerHook(navpop) {
	return function() {
		if (!navpop.isVisible()) { return true; /* remove this hook */ }
		if (Navpopup.tracker.dirty) {
			return false;
		}
		var x=Navpopup.tracker.x, y=Navpopup.tracker.y;
		var mouseOverNavpop = navpop.isWithin(x,y,navpop.fuzz, navpop.mainDiv) ||
			!fuzzyCursorOffMenus(x,y,navpop.fuzz, navpop.mainDiv);

		// FIXME it'd be prettier to do this internal to the Navpopup objects
		var t=getValueOf('popupHideDelay');
		if (t) { t = t * 1000; }
		if (!t) {
			if(!mouseOverNavpop) {
				if(navpop.parentAnchor) {
					restoreTitle( navpop.parentAnchor );
				}
				navpop.banish();
				return true; /* remove this hook */
			}
			return false;
		}
		// we have a hide delay set
		var d=+(new Date());
		if ( !navpop.mouseLeavingTime ) {
			navpop.mouseLeavingTime = d;
			return false;
		}
		if ( mouseOverNavpop ) {
			navpop.mouseLeavingTime=null;
			return false;
		}
		if (d - navpop.mouseLeavingTime > t) {
			navpop.mouseLeavingTime=null;
			navpop.banish(); return true; /* remove this hook */
		}
		return false;
	};
}

function runStopPopupTimer(navpop) {
	// at this point, we should have left the link but remain within the popup
	// so we call this function again until we leave the popup.
	if (!navpop.stopPopupTimer) {
		navpop.stopPopupTimer=setInterval(posCheckerHook(navpop), 500);
		navpop.addHook(function(){clearInterval(navpop.stopPopupTimer);},
				   'hide', 'before');
	}
}
// ENDFILE: mouseout.js
// STARTFILE: previewmaker.js
/**
   @fileoverview
   Defines the {@link Previewmaker} object, which generates short previews from wiki markup.
*/

/**
   Creates a new Previewmaker
   @constructor
   @class The Previewmaker class. Use an instance of this to generate short previews from Wikitext.
   @param {String} wikiText The Wikitext source of the page we wish to preview.
   @param {String} baseUrl The url we should prepend when creating relative urls.
   @param {Navpopup} owner The navpop associated to this preview generator
*/
function Previewmaker(wikiText, baseUrl, owner) {
	/** The wikitext which is manipulated to generate the preview. */
	this.originalData=wikiText;
	this.setData();
	this.baseUrl=baseUrl;
	this.owner=owner;
	this.maxCharacters=getValueOf('popupMaxPreviewCharacters');
	this.maxSentences=getValueOf('popupMaxPreviewSentences');
}
Previewmaker.prototype.setData=function() {
	var maxSize=max(10000, 2*this.maxCharacters);
	this.data=this.originalData.substring(0,maxSize);
};
/** Remove HTML comments
	@private
*/
Previewmaker.prototype.killComments = function () {
	// this also kills one trailing newline, eg [[diamyo]]
	this.data=this.data.replace(RegExp('<!--[\\s\\S]*?-->\\n?', 'g'), ''); 
};
/**
   @private
*/
Previewmaker.prototype.killDivs = function () {
	// say goodbye, divs (can be nested, so use * not *?)
	this.data=this.data.replace(RegExp('< *div[^>]* *>[\\s\\S]*?< */ *div *>',
					   'gi'), '');
};
/**
   @private
*/
Previewmaker.prototype.killGalleries = function () {
	this.data=this.data.replace(RegExp('< *gallery[^>]* *>[\\s\\S]*?< */ *gallery *>',
					   'gi'), '');
};
/**
   @private
*/
Previewmaker.prototype.kill = function(opening, closing, subopening, subclosing, repl) {
	var oldk=this.data;
	var k=this.killStuff(this.data, opening, closing, subopening, subclosing, repl);
	while (k.length < oldk.length) {
		oldk=k;
		k=this.killStuff(k, opening, closing, subopening, subclosing, repl);
	}
	this.data=k;
};
/**
   @private
*/
Previewmaker.prototype.killStuff = function (txt, opening, closing, subopening, subclosing, repl) {
	var op=this.makeRegexp(opening);
	var cl=this.makeRegexp(closing, '^');
	var sb=subopening ? this.makeRegexp(subopening, '^') : null;
	var sc=subclosing ? this.makeRegexp(subclosing, '^') : cl;
	if (!op || !cl) {
		alert('Navigation Popups error: op or cl is null! something is wrong.');
		return;
	}
	if (!op.test(txt)) { return txt; }
	var ret='';
	var opResult = op.exec(txt);
	ret = txt.substring(0,opResult.index);
	txt=txt.substring(opResult.index+opResult[0].length);
	var depth = 1;
	while (txt.length > 0) {
		var removal=0;
		if (depth==1 && cl.test(txt)) {
			depth--;
			removal=cl.exec(txt)[0].length;
		} else if (depth > 1 && sc.test(txt)) {
			depth--;
			removal=sc.exec(txt)[0].length;
		}else if (sb && sb.test(txt)) {
			depth++;
			removal=sb.exec(txt)[0].length;
		}
		if ( !removal ) { removal = 1; }
		txt=txt.substring(removal);
		if (depth==0) { break; }
	}
	return ret + (repl || '') + txt;
};
/**
   @private
*/
Previewmaker.prototype.makeRegexp = function (x, prefix, suffix) {
	prefix = prefix || '';
	suffix = suffix || '';
	var reStr='';
	var flags='';
	if (isString(x)) {
		reStr=prefix + literalizeRegex(x) + suffix;
	} else if (isRegExp(x)) {
		var s=x.toString().substring(1);
		var sp=s.split('/');
		flags=sp[sp.length-1];
		sp[sp.length-1]='';
		s=sp.join('/');
		s=s.substring(0,s.length-1);
		reStr= prefix + s + suffix;
	} else {
		log ('makeRegexp failed');
	}

	log ('makeRegexp: got reStr=' + reStr + ', flags=' + flags);
	return RegExp(reStr, flags);
};
/**
   @private
*/
Previewmaker.prototype.killBoxTemplates = function () {

	// taxobox removal... in fact, there's a saudiprincebox_begin, so let's be more general
	// also, have float_begin, ... float_end
	this.kill(RegExp('[{][{][^{}\\s|]*?(float|box)[_ ](begin|start)', 'i'),	/[}][}]\s*/, '{{');

	// infoboxes etc
	// from [[User:Zyxw/popups.js]]: kill frames too
	this.kill(RegExp('[{][{][^{}\\s|]*?(infobox|elementbox|frame)[_ ]', 'i'), /[}][}]\s*/, '{{');

};
/**
   @private
*/
Previewmaker.prototype.killTemplates = function () {
	this.kill('{{', '}}', '{', '}', ' ');
};
/**
   @private
*/
Previewmaker.prototype.killTables = function () {
	// tables are bad, too
	// this can be slow, but it's an inprovement over a browser hang
	// torture test: [[Comparison_of_Intel_Central_Processing_Units]]
	this.kill('{|', /[|]}\s*/, '{|');
	this.kill(/<table.*?>/i, /<\/table.*?>/i, /<table.*?>/i);
	// remove lines starting with a pipe for the hell of it (?)
	this.data=this.data.replace(RegExp('^[|].*$', 'mg'), '');
};
/**
   @private
*/
Previewmaker.prototype.killImages = function () {
	var forbiddenNamespaceAliases = [];
	jQuery.each(mw.config.get('wgNamespaceIds'), function(_localizedNamespaceLc, _namespaceId) {
		if (_namespaceId!=pg.nsImageId && _namespaceId!=pg.nsCategoryId) return;
		forbiddenNamespaceAliases.push(_localizedNamespaceLc.split(' ').join('[ _]')); //todo: escape regexp fragments!
	});
	
	// images and categories are a nono
	this.kill(RegExp('[[][[]\\s*(' + forbiddenNamespaceAliases.join('|') + ')\\s*:', 'i'),
		  /\]\]\s*/, '[', ']');
};
/**
   @private
*/
Previewmaker.prototype.killHTML = function () {
	// kill <ref ...>...</ref>
	this.kill(/<ref\b[^/>]*?>/i, /<\/ref>/i);

	// let's also delete entire lines starting with <. it's worth a try.
	this.data=this.data.replace(RegExp('(^|\\n) *<.*', 'g'), '\n');

	// and those pesky html tags, but not <nowiki> or <blockquote>
	var splitted=this.data.parenSplit(/(<[\w\W]*?(?:>|$|(?=<)))/);
	var len=splitted.length;
	for (var i=1; i<len; i=i+2) {
		switch (splitted[i]) {
		case '<nowiki>':
		case '</nowiki>':
		case '<blockquote>':
		case '</blockquote>':
			break;
		default:
			splitted[i]='';
		}
	}
	this.data=splitted.join('');
};
/**
   @private
*/
Previewmaker.prototype.killChunks = function() { // heuristics alert
	// chunks of italic text? you crazy, man?
	var italicChunkRegex=new RegExp
	("((^|\\n)\\s*:*\\s*''[^']([^']|'''|'[^']){20}(.|\\n[^\\n])*''[.!?\\s]*\\n)+", 'g');
	// keep stuff separated, though, so stick in \n (fixes [[Union Jack]]?
	this.data=this.data.replace(italicChunkRegex, '\n');
};
/**
   @private
*/
Previewmaker.prototype.mopup = function () {
	// we simply *can't* be doing with horizontal rules right now
	this.data=this.data.replace(RegExp('^-{4,}','mg'),'');

	// no indented lines
	this.data=this.data.replace(RegExp('(^|\\n) *:[^\\n]*','g'), '');

	// replace __TOC__, __NOTOC__ and whatever else there is
	// this'll probably do
	this.data=this.data.replace(RegExp('^__[A-Z_]*__ *$', 'gmi'),'');
};
/**
   @private
*/
Previewmaker.prototype.firstBit = function () {
	// dont't be givin' me no subsequent paragraphs, you hear me?
	/// first we "normalize" section headings, removing whitespace after, adding before
	var d=this.data;

	if (getValueOf('popupPreviewCutHeadings')) {
		this.data=this.data.replace(RegExp('\\s*(==+[^=]*==+)\\s*', 'g'), '\n\n$1 ');
		/// then we want to get rid of paragraph breaks whose text ends badly
		this.data=this.data.replace(RegExp('([:;]) *\\n{2,}', 'g'), '$1\n');

		this.data=this.data.replace(RegExp('^[\\s\\n]*'), '');
		stuff=(RegExp('^([^\\n]|\\n[^\\n\\s])*')).exec(this.data);
		if (stuff) { d = stuff[0]; }
		if (!getValueOf('popupPreviewFirstParOnly')) { d = this.data; }

		/// now put \n\n after sections so that bullets and numbered lists work
		d=d.replace(RegExp('(==+[^=]*==+)\\s*', 'g'), '$1\n\n');
	}


	// Split sentences. Superfluous sentences are RIGHT OUT.
	// note: exactly 1 set of parens here needed to make the slice work
	d = d.parenSplit(RegExp('([!?.]+["'+"'"+']*\\s)','g'));
	// leading space is bad, mmkay?
	d[0]=d[0].replace(RegExp('^\\s*'), '');

	var notSentenceEnds=RegExp('([^.][a-z][.] *[a-z]|etc|sic|Dr|Mr|Mrs|Ms|St|no|op|cit|\\[[^\\]]*|\\s[A-Zvclm])$', 'i');
	d = this.fixSentenceEnds(d, notSentenceEnds);

	this.fullLength=d.join('').length;
	var maxChars=getValueOf('popupMaxPreviewCharacters') + this.extraCharacters;
	var n=this.maxSentences;
	var dd=this.firstSentences(d,n); 

	do {
		dd=this.firstSentences(d,n); --n;
	} while ( dd.length > this.maxCharacters && n != 0 );

	this.data = dd;
};
/**
   @private
*/
Previewmaker.prototype.fixSentenceEnds = function(strs, reg) {
	// take an array of strings, strs
	// join strs[i] to strs[i+1] & strs[i+2] if strs[i] matches regex reg

	var abbrevRe=/\b[a-z][^a-z]*$/i;

	for (var i=0; i<strs.length-2; ++i) {
		if (reg.test(strs[i])) {
			a=[];
			for (var j=0; j<strs.length; ++j) {
				if (j<i)   a[j]=strs[j];
				if (j==i)  a[i]=strs[i]+strs[i+1]+strs[i+2];
				if (j>i+2) a[j-2]=strs[j];
			}
			return this.fixSentenceEnds(a,reg);
		}
		// BUGGY STUFF - trying to fix up [[S. C. Johnson & Son]] preview
		if (false && abbrevRe.test(strs[i])) {
			var j=i, buf='';
			do {
				buf=buf+strs[i]+strs[i+1];
				i=i+2;
			} while (i<strs.length-2 && abbrevRe.test(strs[i]));
			strs[i]=buf+strs[i];
			var a=(j?strs.slice(0,j-1):[]).concat(strs.slice(i));
			return this.fixSentenceEnds(a,reg);
		}
	}
	return strs;
};
/**
   @private
*/
Previewmaker.prototype.firstSentences = function(strs, howmany) {
	var t=strs.slice(0, 2*howmany);
	return t.join('');
};
/**
   @private
*/
Previewmaker.prototype.killBadWhitespace = function() {
	// also cleans up isolated '''', eg [[Suntory Sungoliath]]
	this.data=this.data.replace(RegExp('^ *\'+ *$', 'gm'), '');
};
/**
   Runs the various methods to generate the preview.
   The preview is stored in the <code>html</html> field.
   @private
*/
Previewmaker.prototype.makePreview = function() {
	if (this.owner.article.namespaceId()!=pg.nsTemplateId &&
				this.owner.article.namespaceId()!=pg.nsImageId ) {
		this.killComments();
		this.killDivs();
		this.killGalleries();
		this.killBoxTemplates();

		if (getValueOf('popupPreviewKillTemplates')) {
			this.killTemplates();
		} else {
			this.killMultilineTemplates();
		}
		this.killTables();
		this.killImages();
		this.killHTML();
		this.killChunks();
		this.mopup();

		this.firstBit();
		this.killBadWhitespace();
	}
	else
	{
		this.killHTML();
	}
	this.html=wiki2html(this.data, this.baseUrl); // needs livepreview
	this.fixHTML();
	this.stripLongTemplates();
};
/**
   @private
*/
Previewmaker.prototype.esWiki2HtmlPart = function(data) {
  var reLinks = /(?:\[\[([^|\]]*)(?:\|([^|\]]*))*]]([a-z]*))/gi; //match a wikilink
  reLinks.lastIndex = 0; //reset regex

  var match;
  var result = "";
  var postfixIndex = 0;
  while ((match = reLinks.exec(data)) != null) //match all wikilinks
  {
	//FIXME: the way that link is built here isn't perfect. It is clickable, but popups preview won't recognize it in some cases.
	result += pg.escapeQuotesHTML(data.substring(postfixIndex, match.index)) + 
			  '<a href="'+Insta.conf.paths.articles+pg.escapeQuotesHTML(match[1])+'">'+pg.escapeQuotesHTML((match[2]?match[2]:match[1])+match[3])+"</a>";
	postfixIndex = reLinks.lastIndex;
  }
  //append the rest
  result += pg.escapeQuotesHTML(data.substring(postfixIndex));
  
  return result;
};
Previewmaker.prototype.editSummaryPreview=function() {
	var reAes   = /\/\* *(.*?) *\*\//g; //match the first section marker
	reAes.lastIndex = 0; //reset regex
	
	var match;
	
	match = reAes.exec(this.data);
	if (match)
	{
		//we have a section link. Split it, process it, combine it.
		var prefix = this.data.substring(0,match.index-1);
		var section = match[1];
		var postfix = this.data.substring(reAes.lastIndex);
		
		var start = "<span class='autocomment'>";
		var end = "</span>";
		if (prefix.length>0) start = this.esWiki2HtmlPart(prefix) + " " + start + "- ";
		if (postfix.length>0) end = ": " + end + this.esWiki2HtmlPart(postfix);
		

		var t=new Title().fromURL(this.baseUrl);
		t.anchorFromUtf(section);
		var sectionLink = Insta.conf.paths.articles + pg.escapeQuotesHTML(t.toString(true)) + '#' + pg.escapeQuotesHTML(t.anchor);
		return start + '<a href="'+sectionLink+'">&rarr;</a> '+pg.escapeQuotesHTML(section) + end;
	}
	
	//else there's no section link, htmlify the whole thing.
	return this.esWiki2HtmlPart(this.data);
};

//<NOLITE>
/** Test function for debugging preview problems one step at a time.
 */
function previewSteps(txt) {
	try {
		txt=txt || document.editform.wpTextbox1.value;
	} catch (err) {
		if (pg.cache.pages.length > 0) {
			txt=pg.cache.pages[pg.cache.pages.length-1].data;
		} else {
			alert('provide text or use an edit page');
		}
	}
	txt=txt.substring(0,10000);
	var base=pg.wiki.articlebase + Title.fromURL(document.location.href).urlString();
	var p=new Previewmaker(txt, base, pg.current.link.navpopup);
	if (this.owner.article.namespaceId() != pg.nsTemplateId) {
		p.killComments(); if (!confirm('done killComments(). Continue?\n---\n' + p.data)) { return; }
		p.killDivs(); if (!confirm('done killDivs(). Continue?\n---\n' + p.data)) { return; }
		p.killGalleries(); if (!confirm('done killGalleries(). Continue?\n---\n' + p.data)) { return; }
		p.killBoxTemplates(); if (!confirm('done killBoxTemplates(). Continue?\n---\n' + p.data)) { return; }

		if (getValueOf('popupPreviewKillTemplates')) {
			p.killTemplates(); if (!confirm('done killTemplates(). Continue?\n---\n' + p.data)) { return; }
		} else {
			p.killMultilineTemplates(); if (!confirm('done killMultilineTemplates(). Continue?\n---\n' + p.data)) { return; }
		}

		p.killTables(); if (!confirm('done killTables(). Continue?\n---\n' + p.data)) { return; }
		p.killImages(); if (!confirm('done killImages(). Continue?\n---\n' + p.data)) { return; }
		p.killHTML(); if (!confirm('done killHTML(). Continue?\n---\n' + p.data)) { return; }
		p.killChunks(); if (!confirm('done killChunks(). Continue?\n---\n' + p.data)) { return; }
		p.mopup(); if (!confirm('done mopup(). Continue?\n---\n' + p.data)) { return; }

		p.firstBit(); if (!confirm('done firstBit(). Continue?\n---\n' + p.data)) { return; }
		p.killBadWhitespace(); if (!confirm('done killBadWhitespace(). Continue?\n---\n' + p.data)) { return; }
	}

	p.html=wiki2html(p.data, base); // needs livepreview
	p.fixHTML(); if (!confirm('done fixHTML(). Continue?\n---\n' + p.html)) { return; }
	p.stripLongTemplates(); if (!confirm('done stripLongTemplates(). Continue?\n---\n' + p.html)) { return; }
	alert('finished preview - end result follows.\n---\n' + p.html);
}
//</NOLITE>

/**
   Works around livepreview bugs.
   @private
*/
Previewmaker.prototype.fixHTML = function() {
	if(!this.html) return;

  var ret = this.html;

	// fix question marks in wiki links
	// maybe this'll break some stuff :-(
	ret=ret.replace(RegExp('\(<a href="' + pg.wiki.articlePath + '/[^"]*\)[?]\(.*?"\)', 'g'), '$1%3F$2');
	ret=ret.replace(RegExp('\(<a href=\'' + pg.wiki.articlePath + '/[^\']*\)[?]\(.*?\'\)', 'g'), '$1%3F$2');
	// FIXME fix up % too
	
	
	this.html=ret;
};
/**
   Generates the preview and displays it in the current popup.

   Does nothing if the generated preview is invalid or consists of whitespace only.
   Also activates wikilinks in the preview for subpopups if the popupSubpopups option is true.
*/
Previewmaker.prototype.showPreview = function () {
	this.makePreview();
	if (typeof this.html != typeof '') return;
	if (RegExp('^\\s*$').test(this.html)) return;
	setPopupHTML('<hr />', 'popupPrePreviewSep', this.owner.idNumber);
	setPopupTipsAndHTML(this.html, 'popupPreview', this.owner.idNumber, { owner: this.owner });
	var more = (this.fullLength > this.data.length) ? this.moreLink() : '';
	setPopupHTML(more, 'popupPreviewMore', this.owner.idNumber);
};
/**
   @private
*/
Previewmaker.prototype.moreLink=function() {
	var a=document.createElement('a');
	a.className='popupMoreLink';
	a.innerHTML=popupString('more...');
	var savedThis=this;
	a.onclick=function() {
		savedThis.maxCharacters+=2000;
		savedThis.maxSentences+=20;
		savedThis.setData();
		savedThis.showPreview();
	}
	return a;
}

/**
   @private
*/
Previewmaker.prototype.stripLongTemplates = function() {
	// operates on the HTML!
	this.html=this.html.replace(RegExp('^.{0,1000}[{][{][^}]*?(<(p|br)( /)?>\\s*){2,}([^{}]*?[}][}])?', 'gi'), '');
	this.html=this.html.split('\n').join(' '); // workaround for <pre> templates
	this.html=this.html.replace(RegExp('[{][{][^}]*<pre>[^}]*[}][}]','gi'), '');
};
/**
   @private
*/
Previewmaker.prototype.killMultilineTemplates = function() {
	this.kill('{{{', '}}}');
	this.kill(RegExp('\\s*[{][{][^{}]*\\n'), '}}', '{{');
};
// ENDFILE: previewmaker.js
// STARTFILE: querypreview.js
function loadAPIPreview(queryType, article, navpop) {
	var art=new Title(article).urlString();
	var url=pg.wiki.apiwikibase + '?format=json&action=query&';
	var htmlGenerator=function(a,d){alert('invalid html generator');};
	switch (queryType) {
	case 'history':
		url += 'meta=userinfo&uiprop=options&titles=' + art + '&prop=revisions&rvlimit=' +
			getValueOf('popupHistoryPreviewLimit');
		htmlGenerator=APIhistoryPreviewHTML;
		break;
	case 'category':
		url += 'list=categorymembers&rawcontinue=&cmtitle=' + art;
		htmlGenerator=APIcategoryPreviewHTML;
		break;
	case 'userinfo':
	  var username = new Title( article ).userName();
		var usernameart = encodeURIComponent( username );
		if (pg.re.ipUser.test(username)) {
			url += 'list=blocks&bkprop=range&bkip=' + usernameart;
		} else {
			url += 'list=users&usprop=blockinfo|groups|editcount|registration&ususers=' + usernameart + "&meta=globaluserinfo&guiprop=groups|unattached&guiuser="+ usernameart;
		}
		htmlGenerator=APIuserInfoPreviewHTML;
		break;
	case 'contribs':
		var usernameart = encodeURIComponent( new Title( article ).userName() );
		url += 'list=usercontribs&meta=userinfo&uiprop=options&ucuser=' + usernameart +
			'&uclimit=' + getValueOf('popupContribsPreviewLimit');
		htmlGenerator=APIcontribsPreviewHTML;
		break;
	case 'imagepagepreview':
		var trail='';
		if (getValueOf('popupImageLinks')) { trail = '&list=imageusage&iutitle=' + art; }
		url += 'titles=' + art + '&prop=revisions|imageinfo&rvprop=content' + trail;
		htmlGenerator=APIimagepagePreviewHTML;
		break;
	case 'backlinks':
		url += 'list=backlinks&rawcontinue=&bltitle=' + art;
		htmlGenerator=APIbacklinksPreviewHTML;
		break;
	}
	pendingNavpopTask(navpop);
	if( !mw.config.get('wgEnableAPI') ) {
		/* The API is not available */
		htmlGenerator=function(a,d){
			return 'This function of navigation popups now requires a MediaWiki ' + 
			'installation with the <a href="http://www.mediawiki.org/wiki/API">API</a> enabled.'; };
	}
	var callback=function(d){
		log( "callback of API functions was hit" );
		showAPIPreview(queryType, htmlGenerator(article,d,navpop), navpop.idNumber, navpop, d);
	};
	if (pg.flag.isIE) {
		url = url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
	}
	var go = function(){
		getPageWithCaching(url, callback, navpop);
		return true;
	}
	if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
	else { navpop.addHook(go, 'unhide', 'before', 'DOWNLOAD_'+queryType+'_QUERY_DATA'); }
}

function linkList(list) {
	list.sort(function(x,y) { return (x==y ? 0 : (x<y ? -1 : 1)); });
	var buf=[];
	for (var i=0; i<list.length; ++i) {
		buf.push(wikiLink({article: new Title(list[i]),
				   text:	list[i].split(' ').join('&nbsp;'),
				   action:  'view'}));
	}
	return buf.join(', ');
}

function getTimeOffset(tz) {
	if( tz ) {
		if( tz.indexOf('|') > -1 ) {
			// New format
			return parseInt(tz.split('|')[1],10);
		} else if ( tz.indexOf(':') > -1 ) {
			// Old format
			return( parseInt(tz,10)*60 + parseInt(tz.split(':')[1],10) );
		}
	}
	return 0;
}

function editPreviewTable(article, h, reallyContribs, timeOffset) {
	var html=['<table>'];
	var day=null;
	var curart=article;
	for (var i=0; i<h.length; ++i) {
		if (reallyContribs) { 
			var page=h[i]['title']; curart = new Title(page);
		}
		var minor=typeof h[i]['minor']=='undefined' ? '' : '<b>m </b>';
		var editDate=adjustDate(getDateFromTimestamp(h[i].timestamp), timeOffset);
		var thisDay = dayFormat(editDate);
		var thisTime = timeFormat(editDate);
		if (thisDay==day) { thisDay=''; }
		else { day=thisDay; }
		if (thisDay) {
			html.push( '<tr><td colspan=3><span class="popup_history_date">' +
				  thisDay+'</span></td></tr>' );
		}
		html.push('<tr class="popup_history_row_' + ( (i%2) ? 'odd' : 'even') + '">');
		html.push('<td>(<a href="' + pg.wiki.titlebase + new Title(curart).urlString() +
			'&diff=prev&oldid=' + h[i]['revid'] + '">' + popupString('last') + '</a>)</td>');
		html.push('<td>' +
			'<a href="' + pg.wiki.titlebase + new Title(curart).urlString() +
			'&oldid=' + h[i]['revid'] + '">' + thisTime + '</a></td>');
		var col3url='', col3txt='';
		if (!reallyContribs) {
			var user=h[i]['user'];
			if( typeof h[i]['userhidden'] == "undefined" ) {
				if( pg.re.ipUser.test(user) ) {
					col3url=pg.wiki.titlebase + mw.config.get('wgFormattedNamespaces')[pg.nsSpecialId] + ':Contributions&target=' + new Title(user).urlString();
				} else {
					col3url=pg.wiki.titlebase + mw.config.get('wgFormattedNamespaces')[pg.nsUserId] + ':' + new Title(user).urlString();
				}
				col3txt=pg.escapeQuotesHTML(user);
			} else {
				col3url=getValueOf('popupRevDelUrl');
				col3txt=pg.escapeQuotesHTML( popupString('revdel'));
			}
		} else {
			col3url=pg.wiki.titlebase + curart.urlString();
			col3txt=pg.escapeQuotesHTML(page);
		}
		html.push('<td>' + (reallyContribs ? minor : '') +
			'<a href="' + col3url + '">' + col3txt + '</a></td>');
		var comment='';
		var c=h[i].comment || h[i]['*'];
		if (c) {
			comment=new Previewmaker(c, new Title(curart).toUrl()).editSummaryPreview();
		} else if (typeof h[i]['commenthidden'] != "undefined" ) {
			comment=popupString('revdel');
		}
		html.push('<td>' + (!reallyContribs ? minor : '') + comment + '</td>');
		html.push('</tr>');
		html=[html.join('')];
	}
	html.push('</table>');
	return html.join('');
}

function getDateFromTimestamp(t) {
	var s=t.split(/[^0-9]/);
	switch(s.length) {
	case 0: return null;
	case 1: return new Date(s[0]);
	case 2: return new Date(s[0], s[1]-1);
	case 3: return new Date(s[0], s[1]-1, s[2]);
	case 4: return new Date(s[0], s[1]-1, s[2], s[3]);
	case 5: return new Date(s[0], s[1]-1, s[2], s[3], s[4]);
	case 6: return new Date(s[0], s[1]-1, s[2], s[3], s[4], s[5]);
	default: return new Date(s[0], s[1]-1, s[2], s[3], s[4], s[5], s[6]);
	}
}

function adjustDate(d, offset) {
	// offset is in minutes
	var o=offset * 60 * 1000;
	return new Date( +d + o);
}

function dayFormat(editDate, utc) {
	if (utc) { return map(zeroFill, [editDate.getUTCFullYear(), editDate.getUTCMonth()+1, editDate.getUTCDate()]).join('-'); }
	return map(zeroFill, [editDate.getFullYear(), editDate.getMonth()+1, editDate.getDate()]).join('-');
}

function timeFormat(editDate, utc) {
	if (utc) { return map(zeroFill, [editDate.getUTCHours(), editDate.getUTCMinutes(), editDate.getUTCSeconds()]).join(':'); }
	return map(zeroFill, [editDate.getHours(), editDate.getMinutes(), editDate.getSeconds()]).join(':');
}

function showAPIPreview(queryType, html, id, navpop, download) {
	// DJ: done
	var target='popupPreview';
	switch (queryType) {
	case 'imagelinks':
	case 'category':
	case 'userinfo':
		target='popupPostPreview'; break;
	}
	setPopupTipsAndHTML(html, target, id);
	completedNavpopTask(navpop);
}

function APIbacklinksPreviewHTML(article, download, navpop) {
	try {
		var jsObj=getJsObj(download.data);
		   var list=jsObj.query.backlinks;
	} catch (someError) { return 'backlinksPreviewHTML went wonky'; }
	var html=[];
	if (!list) { return popupString('No backlinks found'); }
	for ( var i=0; i < list.length; i++ ) {
		var t=new Title(list[i]['title']);
		html.push('<a href="' + pg.wiki.titlebase + t.urlString() + '">' + t + '</a>');
	}
	html=html.join(', ');
	if (jsObj['query-continue'] && jsObj['query-continue'].backlinks && jsObj['query-continue'].backlinks.blcontinue) {
		html += popupString(' and more');
	}
	return html;
}

function APIsharedImagePagePreviewHTML(obj) {
	log( "APIsharedImagePagePreviewHTML" );
	var popupid = obj['requestid'];
	if( obj['query'] && obj['query']['pages'] )
	{
		var page=anyChild(obj['query']['pages']);
		var content=(page && page.revisions ) ? page.revisions[0]['*'] : null;
		if( content ) 
		{
			/* Not entirely safe, but the best we can do */
			var p=new Previewmaker(content, pg.current.link.navpopup.article, pg.current.link.navpopup);
			p.makePreview();
			setPopupHTML( p.html, "popupSecondPreview", popupid );
		}
	}
}

function APIimagepagePreviewHTML(article, download, navpop) {
	try {
		var jsObj=getJsObj(download.data);
		var page=anyChild(jsObj.query.pages);
		var content=(page && page.revisions ) ? page.revisions[0]['*'] : null;
	} catch (someError) {
		return 'API imagepage preview failed :(';
	}
	var ret='';
	var alt='';
	try{alt=navpop.parentAnchor.childNodes[0].alt;} catch(e){}
	if (alt) {
		ret = ret + '<hr /><b>' + popupString('Alt text:') + '</b> ' + pg.escapeQuotesHTML(alt);
	}
	if (content) {
		var p=prepPreviewmaker(content, article, navpop);
		p.makePreview();
		if (p.html) { ret += '<hr />' + p.html; }
	}
	if (content!==null && getValueOf('popupSummaryData')) {
		var info=getPageInfo(content, download);
		log(info);
		setPopupTrailer(info, navpop.idNumber);
	}
	if (page && page.imagerepository == "shared" ) {
		var art=new Title(article);
		var encart = encodeURIComponent( "File:" + art.stripNamespace() );
		var shared_url =  pg.wiki.apicommonsbase + '?format=json&callback=APIsharedImagePagePreviewHTML' +
							'&requestid=' + navpop.idNumber +
							'&action=query&prop=revisions&rvprop=content&titles=' + encart;
		if (pg.flag.isIE) {
			shared_url = shared_url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
		}

		ret = ret +'<hr />' + popupString( 'Image from Commons') +
				': <a href="' + pg.wiki.commonsbase + '?title=' + encart + '">' +
				popupString( 'Description page') + '</a>';
		mw.loader.load( shared_url );
	}
	showAPIPreview('imagelinks', APIimagelinksPreviewHTML(article,download), navpop.idNumber, download);
	return ret;
}

function APIimagelinksPreviewHTML(article, download) {
	try {
		var jsobj=getJsObj(download.data);
		var list=jsobj.query.imageusage;
		if (!list) { return popupString('No image links found'); }
	} catch(someError) { return 'Image links preview generation failed :('; }
	var ret=[];
	for (var i=0; i < list.length; i++) {
		ret.push(list[i]['title']);
	}
	if (ret.length === 0) { return popupString('No image links found'); }
	return '<h2>' + popupString('File links') + '</h2>' + linkList(ret);
}

function APIcategoryPreviewHTML(article, download) {
	try{
		var jsobj=getJsObj(download.data);
		var list=jsobj.query.categorymembers;
	} catch(someError) { return 'Category preview failed :('; }
	var ret=[];
	for (var p=0; p < list.length; p++) { 
	   ret.push(list[p]['title']); 
	}
	if (ret.length === 0) { return popupString('Empty category'); }
	ret = '<h2>' + tprintf('Category members (%s shown)', [ret.length]) + '</h2>' +linkList(ret);
	if (jsobj['query-continue'] && jsobj['query-continue'].categorymembers && jsobj['query-continue'].categorymembers.cmcontinue) {
		ret += popupString(' and more');
	}
	return ret;
}

function APIuserInfoPreviewHTML(article, download) {
	var ret=[];
	try{
		var queryobj=getJsObj(download.data).query;
	} catch(someError) { return 'Userinfo preview failed :('; }
	
	var user=anyChild(queryobj.users);
	if (user) {
		var globaluserinfo=queryobj.globaluserinfo;
		if (user.invalid == '') {
			ret.push( popupString( 'Invalid user') );
		} else if (user.missing == '') {
			ret.push( popupString( 'Not a registered username') );
		}
		if( user.blockedby )
			ret.push('<b>' + popupString('BLOCKED') + '</b>');
		if( globaluserinfo && (globaluserinfo.locked != null || globaluserinfo.hidden != null) ) {
			var lockedSulAccountIsAttachedToThis = true;
			for( var i=0; globaluserinfo.unattached && i < globaluserinfo.unattached.length; i++) {
				if ( globaluserinfo.unattached[i].wiki===mw.config.get('wgDBname') ) { lockedSulAccountIsAttachedToThis=false; break; }
			}
			if (lockedSulAccountIsAttachedToThis) {
				if (globaluserinfo.locked != null) ret.push('<b><i>' + popupString('LOCKED') + '</i></b>');
				if (globaluserinfo.hidden != null) ret.push('<b><i>' + popupString('HIDDEN') + '</i></b>');
			}
		}
		for( var i=0; (user.groups && i < user.groups.length); i++) {
			switch (user.groups[i]) {
				case '*':
				case 'user':
				case 'autoconfirmed':
					break;
				default:
					ret.push( pg.escapeQuotesHTML(user.groups[i]) );
			}
		}
		for( var i=0; (globaluserinfo && globaluserinfo.groups && i < globaluserinfo.groups.length); i++) {
			ret.push( '<i>'+pg.escapeQuotesHTML(globaluserinfo.groups[i])+'</i>' );
		}
		if( user.editcount || user.registration )
			ret.push( pg.escapeQuotesHTML((user.editcount?user.editcount:'') + popupString(' edits since: ') + (user.registration?dayFormat(getDateFromTimestamp(user.registration)):'')) );
	}
	
	if (queryobj.blocks) {
		ret.push( popupString( 'IP user') ); //we only request list=blocks for IPs
		for (var i=0; i<queryobj.blocks.length; i++) {
			ret.push('<b>' + popupString(queryobj.blocks[i].rangestart===queryobj.blocks[i].rangeend ? 'BLOCKED' : 'RANGEBLOCKED') + '</b>' );
		}
	}
	
	ret = '<hr />' + ret.join( ', ' );
	return ret;
}

function APIcontribsPreviewHTML(article, download, navpop) {
	return APIhistoryPreviewHTML(article, download, navpop, true);
}

function APIhistoryPreviewHTML(article, download, navpop, reallyContribs) {
	try {
		var jsobj=getJsObj(download.data);
		var tz=jsobj.query.userinfo.options.timecorrection;
		if( reallyContribs )
			var edits=jsobj.query.usercontribs;
		else 
			var edits=anyChild(jsobj.query.pages)['revisions'];
	} catch (someError) {
		return 'History preview failed :-(';
	}
	var timeOffset = getTimeOffset(tz);
	Cookie.create('popTz', timeOffset, 1);

	var ret=editPreviewTable(article, edits, reallyContribs, timeOffset);
	return ret;
}


//</NOLITE>
// ENDFILE: querypreview.js
// STARTFILE: debug.js
////////////////////////////////////////////////////////////////////
// Debugging functions
////////////////////////////////////////////////////////////////////

function log(){}; // dummy to stop errors
function setupDebugging() {
//<NOLITE>
	if (window.popupDebug) { // popupDebug is set from .version
		window.log=function(x) { //if(gMsg!='')gMsg += '\n'; gMsg+=time() + ' ' + x; };
			window.console.log(x);
		}
		window.errlog=function(x) {
			window.console.error(x);
		}
		log('Initializing logger');
	} else {
//</NOLITE>
		window.log = function(x) {};
		window.errlog = function(x) {};
//<NOLITE>
	}
//</NOLITE>
}
// ENDFILE: debug.js
// STARTFILE: images.js

// load image of type Title.
function loadImage(image, navpop) {
	if (typeof image.stripNamespace != 'function') { alert('loadImages bad'); }
	// API call to retrieve image info.

	if ( !getValueOf('popupImages') || !mw.config.get('wgEnableAPI') ) return;
	if ( !isValidImageName(image) ) return false;
	
	var art=image.urlString();

	var url=pg.wiki.apiwikibase + '?format=json&action=query';
	url += '&prop=imageinfo&iiprop=url|mime&iiurlwidth=' + getValueOf('popupImageSizeLarge');;
	url += '&titles=' + art;
	if (pg.flag.isIE) {
		url = url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
	}

	pendingNavpopTask(navpop);
	var callback=function(d){
		popupsInsertImage(navpop.idNumber, navpop, d);
	};
	var go = function(){
		getPageWithCaching(url, callback, navpop);
		return true;
	}
	if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
	else { navpop.addHook(go, 'unhide', 'after', 'DOWNLOAD_IMAGE_QUERY_DATA'); }

}

function popupsInsertImage(id, navpop, download) {
	log( "popupsInsertImage");
	try {
		var jsObj=getJsObj(download.data);
		var imagepage=anyChild(jsObj.query.pages);
		if (typeof imagepage.imageinfo === 'undefined') return;
		var imageinfo = imagepage.imageinfo[0];
	} catch (someError) {
		log( "popupsInsertImage failed :(" );
		return;
	}

	var popupImage = document.getElementById("popupImg"+id);
	if (!popupImage) {
		log( "could not find insertion point for image");
		return;
	}

	popupImage.width=getValueOf('popupImageSize');
	popupImage.style.display='inline';

	// Set the source for the image.
	if( imageinfo.thumburl )
		popupImage.src=imageinfo.thumburl;
	else if( imageinfo.mime.indexOf("image") == 0 ){
		popupImage.src=imageinfo.url;
		log( "a thumb could not be found, using original image" );
	} else log( "fullsize imagethumb, but not sure if it's an image");


	var a=document.getElementById("popupImageLink"+id);
	if (a === null) { return null; }

	// Determine the action of the surrouding imagelink.
	switch (getValueOf('popupThumbAction')) {
	case 'imagepage':
		if (pg.current.article.namespaceId()!=pg.nsImageId) {
			a.href=imageinfo.descriptionurl;
			// FIXME: unreliable pg.idNumber
			popTipsSoonFn('popupImage' + id)();
			break;
		} // else fall through
	case 'sizetoggle':
		a.onclick=toggleSize;
		a.title=popupString('Toggle image size');
		return;
	case 'linkfull':
		a.href = imageinfo.url;
		a.title=popupString('Open full-size image');
		return;
	}

}

// Toggles the image between inline small and navpop fullwidth.
// It's the same image, no actual sizechange occurs, only display width.
function toggleSize() {
	var imgContainer=this;
	if (!imgContainer) { alert('imgContainer is null :/'); return;}
	img=imgContainer.firstChild;
	if (!img) { alert('img is null :/'); return;}
	if (!img.style.width || img.style.width=='') { img.style.width='100%'; }
	else { img.style.width=''; }
}

// Returns one title of an image from wikiText.
function getValidImageFromWikiText(wikiText) {
	// nb in pg.re.image we're interested in the second bracketed expression
	// this may change if the regex changes :-(
	//var match=pg.re.image.exec(wikiText);
	var matched=null;
	var match;
	// strip html comments, used by evil bots :-(
	var t = removeMatchesUnless(wikiText, RegExp('(<!--[\\s\\S]*?-->)'), 1,
					RegExp('^<!--[^[]*popup', 'i'));

	while ( match = pg.re.image.exec(t) ) {
		// now find a sane image name - exclude templates by seeking {
		var m = match[2] || match[6];
		if ( isValidImageName(m) ) {
			matched=m;
			break;
		}
	}
	pg.re.image.lastIndex=0;
	if (!matched) { return null; }
	return mw.config.get('wgFormattedNamespaces')[pg.nsImageId]+':'+upcaseFirst(matched);
}

function removeMatchesUnless(str, re1, parencount, re2) {
	var split=str.parenSplit(re1);
	var c=parencount + 1;
	for (var i=0; i<split.length; ++i) {
	if ( i%c === 0 || re2.test(split[i]) ) { continue; }
	split[i]='';
	}
	return split.join('');
}

//</NOLITE>
// ENDFILE: images.js
// STARTFILE: namespaces.js
// Set up namespaces and other non-strings.js localization
// (currently that means redirs too)


function namespaceListToRegex(list) {return RegExp('^('+list.join('|').split(' ').join('[ _]')+'):');};

function setNamespaces() {
	pg.nsSpecialId   = -1;
	pg.nsMainspaceId  = 0;
	pg.nsImageId	 = 6;
	pg.nsUserId	  = 2;
	pg.nsUsertalkId  = 3;
	pg.nsCategoryId  = 14;
	pg.nsTemplateId  = 10;
}


function setRedirs() {
	var r='redirect';
	var R='REDIRECT';
	var redirLists={
//<NOLITE>
		'ar':  [ R, 'تحويل' ],
		'be':  [ r, 'перанакіраваньне' ],
		'bg':  [ r, 'пренасочване', 'виж' ],
		'bs':  [ r, 'Preusmjeri', 'preusmjeri', 'PREUSMJERI' ],
		'cs':  [ R, 'PŘESMĚRUJ' ],
		'cy':  [ r, 'ail-cyfeirio' ],
		'de':  [ R, 'WEITERLEITUNG' ],
		'el':  [ R, 'ΑΝΑΚΑΤΕΥΘΥΝΣΗ'],
		'eo':  [ R, 'ALIDIREKTU', 'ALIDIREKTI' ],
		'es':  [ R, 'REDIRECCIÓN' ],
		'et':  [ r, 'suuna' ],
		'ga':  [ r, 'athsheoladh' ],
		'gl':  [ r, 'REDIRECCIÓN', 'REDIRECIONAMENTO'],
		'he':  [ R, 'הפניה' ],
		'hu':  [ R, 'ÁTIRÁNYÍTÁS' ],
		'is':  [ r, 'tilvísun', 'TILVÍSUN' ],
		'it':  [ R, 'RINVIA', 'Rinvia'],
		'ja':  [ R, '転送' ],
		'mk':  [ r, 'пренасочување', 'види' ],
		'nds': [ r, 'wiederleiden' ],
		'nl':  [ R, 'DOORVERWIJZING' ],
		'nn':  [ r, 'omdiriger' ],
		'pl':  [ R, 'PATRZ', 'PRZEKIERUJ', 'TAM' ],
		'pt':  [ R, 'redir' ],
		'ru':  [ R, 'ПЕРЕНАПРАВЛЕНИЕ', 'ПЕРЕНАПР' ],
		'sk':  [ r, 'presmeruj' ],
		'sr':  [ r, 'Преусмери', 'преусмери', 'ПРЕУСМЕРИ', 'Preusmeri', 'preusmeri', 'PREUSMERI' ],
		'tt':  [ R, 'yünältü', 'перенаправление', 'перенапр' ],
		'uk':  [ R, 'ПЕРЕНАПРАВЛЕННЯ', 'ПЕРЕНАПР' ],
		'vi':  [ r, 'đổi' ],
		'zh':  [ R, '重定向'] // no comma
//</NOLITE>
	};
	var redirList=redirLists[ pg.wiki.lang ] || [r, R];
	// Mediawiki is very tolerant about what comes after the #redirect at the start
	pg.re.redirect=RegExp('^\\s*[#](' + redirList.join('|') + ').*?\\[{2}([^\\|\\]]*)(|[^\\]]*)?\\]{2}\\s*(.*)', 'i');
}

function setInterwiki() {
	if (pg.wiki.wikimedia) {
		// From http://meta.wikimedia.org/wiki/List_of_Wikipedias
		pg.wiki.interwiki='aa|ab|ace|af|ak|als|am|an|ang|ar|arc|arz|as|ast|av|ay|az|ba|bar|bat-smg|bcl|be|be-x-old|bg|bh|bi|bjn|bm|bn|bo|bpy|br|bs|bug|bxr|ca|cbk-zam|cdo|ce|ceb|ch|cho|chr|chy|ckb|co|cr|crh|cs|csb|cu|cv|cy|da|de|diq|dsb|dv|dz|ee|el|eml|en|eo|es|et|eu|ext|fa|ff|fi|fiu-vro|fj|fo|fr|frp|frr|fur|fy|ga|gag|gan|gd|gl|glk|gn|got|gu|gv|ha|hak|haw|he|hi|hif|ho|hr|hsb|ht|hu|hy|hz|ia|id|ie|ig|ii|ik|ilo|io|is|it|iu|ja|jbo|jv|ka|kaa|kab|kbd|kg|ki|kj|kk|kl|km|kn|ko|koi|kr|krc|ks|ksh|ku|kv|kw|ky|la|lad|lb|lbe|lg|li|lij|lmo|ln|lo|lt|ltg|lv|map-bms|mdf|mg|mh|mhr|mi|mk|ml|mn|mo|mr|mrj|ms|mt|mus|mwl|my|myv|mzn|na|nah|nap|nds|nds-nl|ne|new|ng|nl|nn|no|nov|nrm|nv|ny|oc|om|or|os|pa|pag|pam|pap|pcd|pdc|pfl|pi|pih|pl|pms|pnb|pnt|ps|pt|qu|rm|rmy|rn|ro|roa-rup|roa-tara|ru|rue|rw|sa|sah|sc|scn|sco|sd|se|sg|sh|si|simple|sk|sl|sm|sn|so|sq|sr|srn|ss|st|stq|su|sv|sw|szl|ta|te|tet|tg|th|ti|tk|tl|tn|to|tpi|tr|ts|tt|tum|tw|ty|udm|ug|uk|ur|uz|ve|vec|vi|vls|vo|wa|war|wo|wuu|xal|xh|yi|yo|za|zea|zh|zh-classical|zh-min-nan|zh-yue|zu';
		pg.re.interwiki=RegExp('^'+pg.wiki.interwiki+':');
	} else {
		pg.wiki.interwiki=null;
		pg.re.interwiki=RegExp('^$');
	}
}

// return a regexp pattern matching all variants to write the given namespace
function nsRe(namespaceId) {
	var imageNamespaceVariants = [];
	jQuery.each(mw.config.get('wgNamespaceIds'), function(_localizedNamespaceLc, _namespaceId) {
		if (_namespaceId!=namespaceId) return;
		//todo: escape regexp fragments!
		_localizedNamespaceLc = upcaseFirst(_localizedNamespaceLc);
		imageNamespaceVariants.push(_localizedNamespaceLc.split(' ').join('[ _]'));
		imageNamespaceVariants.push(encodeURI(_localizedNamespaceLc));
	});

	return '(?:' + imageNamespaceVariants.join('|') + ')';
}

function nsReImage() {
	return nsRe(pg.nsImageId);
}

// ENDFILE: namespaces.js
// STARTFILE: selpop.js
//<NOLITE>
function getEditboxSelection() {
	// see http://www.webgurusforum.com/8/12/0
	try {
		var editbox=document.editform.wpTextbox1;
	} catch (dang) { return; }
	// IE, Opera
	if (document.selection) { return document.selection.createRange().text; }
	// Mozilla
	var selStart = editbox.selectionStart;
	var selEnd = editbox.selectionEnd;
	return (editbox.value).substring(selStart, selEnd);
}

function doSelectionPopup() {
	// popup if the selection looks like [[foo|anything afterwards at all
	// or [[foo|bar]]text without ']]'
	// or [[foo|bar]]
	var sel=getEditboxSelection();
	var open=sel.indexOf('[[');
	var pipe=sel.indexOf('|');
	var close=sel.indexOf(']]');
	if (open == -1 || ( pipe == -1 && close == -1) ) { return; }
	if (pipe != -1 && open > pipe || close != -1 && open > close) { return; }
	if (getValueOf('popupOnEditSelection')=='boxpreview') {
		return doSeparateSelectionPopup(sel);
	}
	var article=new Title(sel.substring(open+2, (pipe < 0) ? close : pipe)).urlString();
	if (close > 0 && sel.substring(close+2).indexOf('[[') >= 0) { 
		return; 
	}
	var a=document.createElement('a');
	a.href=pg.wiki.titlebase + article;
	mouseOverWikiLink2(a);
	if (a.navpopup) {
		a.navpopup.addHook(function(){runStopPopupTimer(a.navpopup);}, 'unhide', 'after');
	}
}

function doSeparateSelectionPopup(str) {
	var div=document.getElementById('selectionPreview');
	if (!div) {
		div = document.createElement('div');
		div.id='selectionPreview';
		try { var box=document.editform.wpTextbox1; }
		catch (oopsie) { return; }
		box.parentNode.insertBefore(div, box);
	}
	div.innerHTML=wiki2html(str);
	div.ranSetupTooltipsAlready = false;
	popTipsSoonFn('selectionPreview')();
}
//</NOLITE>
// ENDFILE: selpop.js
// STARTFILE: navpopup.js
/**
   @fileoverview  Defines two classes: {@link Navpopup} and {@link Mousetracker}.

   <code>Navpopup</code> describes popups: when they appear, where, what
   they look like and so on.

   <code>Mousetracker</code> "captures" the mouse using
   <code>document.onmousemove</code>.
*/


/**
   Creates a new Mousetracker.
   @constructor
   @class The Mousetracker class. This monitors mouse movements and manages associated hooks.
*/
function Mousetracker() {
	/**
	   Interval to regularly run the hooks anyway, in milliseconds.
	   @type Integer
	*/
	this.loopDelay=400;

	/**
	   Timer for the loop.
	   @type Timer
	*/
	this.timer=null;

	/**
	   Flag - are we switched on?
	   @type Boolean
	*/
	this.active=false;
	/**
	   Flag - are we probably inaccurate, i.e. not reflecting the actual mouse position?
	*/
	this.dirty=true;
	/**
	   Array of hook functions.
	   @private
	   @type Array
	*/
	this.hooks=[];
}

/**
   Adds a hook, to be called when we get events.
   @param {Function} f A function which is called as
   <code>f(x,y)</code>. It should return <code>true</code> when it
   wants to be removed, and <code>false</code> otherwise.
*/
Mousetracker.prototype.addHook = function (f) {
	this.hooks.push(f);
};

/**
   Runs hooks, passing them the x
   and y coords of the mouse.  Hook functions that return true are
   passed to {@link Mousetracker#removeHooks} for removal.
   @private
*/
Mousetracker.prototype.runHooks = function () {
	if (!this.hooks || !this.hooks.length) { return; }
	//log('Mousetracker.runHooks; we got some hooks to run');
	var remove=false;
	var removeObj={};
	// this method gets called a LOT -
	// pre-cache some variables
	var x=this.x, y=this.y, len = this.hooks.length;

	for (var i=0; i<len; ++i) {
		//~ run the hook function, and remove it if it returns true
		if (this.hooks[i](x, y)===true) {
			remove=true;
			removeObj[i]=true;
		}
	}
	if (remove) { this.removeHooks(removeObj); }
};

/**
   Removes hooks.
   @private
   @param {Object} removeObj An object whose keys are the index
   numbers of functions for removal, with values that evaluate to true
*/
Mousetracker.prototype.removeHooks = function(removeObj) {
	var newHooks=[];
	var len = this.hooks.length;
	for (var i=0; i<len; ++i) {
		if (! removeObj[i]) { newHooks.push(this.hooks[i]); }
	}
	this.hooks=newHooks;
};


/**
   Event handler for mouse wiggles.
   We simply grab the event, set x and y and run the hooks.
   This makes the cpu all hot and bothered :-(
   @private
   @param {Event} e Mousemove event
*/
Mousetracker.prototype.track=function (e) {
	//~ Apparently this is needed in IE.
	e = e || window.event;
	var x, y;
	if (e) {
		if (e.pageX) { x=e.pageX; y=e.pageY; }
		else if (typeof e.clientX!='undefined') {
			var left, top, docElt = window.document.documentElement;

			if (docElt) { left=docElt.scrollLeft; }
			left = left || window.document.body.scrollLeft || window.document.scrollLeft || 0;

			if (docElt) { top=docElt.scrollTop; }
			top = top || window.document.body.scrollTop || window.document.scrollTop || 0;

			x=e.clientX + left;
			y=e.clientY + top;
		} else { return; }
		this.setPosition(x,y);
	}
};

/**
   Sets the x and y coordinates stored and takes appropriate action,
   running hooks as appropriate.
   @param {Integer} x, y Screen coordinates to set
*/

Mousetracker.prototype.setPosition=function(x,y) {
	this.x = x;
	this.y = y;
	if (this.dirty || this.hooks.length === 0) { this.dirty=false; return; }
	if (typeof this.lastHook_x != 'number') { this.lastHook_x = -100; this.lastHook_y=-100; }
	var diff = (this.lastHook_x - x)*(this.lastHook_y - y);
	diff = (diff >= 0) ? diff : -diff;
	if ( diff > 1 ) {
		this.lastHook_x=x;
		this.lastHook_y=y;
		if (this.dirty) { this.dirty = false; }
		else { this.runHooks(); }
	}
}

/**
   Sets things in motion, unless they are already that is, registering an event handler on <code>document.onmousemove</code>.
   A half-hearted attempt is made to preserve the old event handler if there is one.
*/
Mousetracker.prototype.enable = function () {
	if (this.active) { return; }
	this.active=true;
	//~ Save the current handler for mousemove events. This isn't too
	//~ robust, of course.
	this.savedHandler=document.onmousemove;
	//~ Gotta save @tt{this} again for the closure, and use apply for
	//~ the member function.
	var savedThis=this;
	document.onmousemove=function (e) {savedThis.track.apply(savedThis, [e]);};
	if (this.loopDelay) { this.timer = setInterval(function() { //log('loop delay in mousetracker is working');
									savedThis.runHooks();}, this.loopDelay); }
};

/**
   Disables the tracker, removing the event handler.
*/
Mousetracker.prototype.disable = function () {
	if (!this.active) { return; }
	if ($.isFunction(this.savedHandler)) {
		document.onmousemove=this.savedHandler;
	} else { delete document.onmousemove; }
	if (this.timer) { clearInterval(this.timer); }
	this.active=false;
};

/**
   Creates a new Navpopup.
   Gets a UID for the popup and
   @param init Contructor object. If <code>init.draggable</code> is true or absent, the popup becomes draggable.
   @constructor
   @class The Navpopup class. This generates popup hints, and does some management of them.
*/
function Navpopup(init) {
	//alert('new Navpopup(init)');
	/** UID for each Navpopup instance.
		Read-only.
		@type integer
	*/
	this.uid=Navpopup.uid++;
	/**
	   Read-only flag for current visibility of the popup.
	   @type boolean
	   @private
	*/
	this.visible=false;
	/** Flag to be set when we want to cancel a previous request to
		show the popup in a little while.
		@private
		@type boolean
	*/
	this.noshow=false;
	/** Categorised list of hooks.
		@see #runHooks
		@see #addHook
		@private
		@type Object
	*/
	this.hooks={
		'create': [],
		'unhide': [],
		'hide': []
	};
	/** list of unique IDs of hook functions, to avoid duplicates
		@private
	*/
	this.hookIds={};
	/** List of downloads associated with the popup.
		@private
		@type Array
	*/
	this.downloads=[];
	/** Number of uncompleted downloads.
		@type integer
	*/
	this.pending=null;
	/** Tolerance in pixels when detecting whether the mouse has left the popup.
		@type integer
	*/
	this.fuzz=5;
	/** Flag to toggle running {@link #limitHorizontalPosition} to regulate the popup's position.
		@type boolean
	*/
	this.constrained=true;
	/** The popup width in pixels.
		@private
		@type integer
	*/
	this.width=0;
	/** The popup width in pixels.
		@private
		@type integer
	*/
	this.height=0;
	/** The main content DIV element.
		@type HTMLDivElement
	*/
	this.mainDiv=null;
	this.createMainDiv();

//	if (!init || typeof init.popups_draggable=='undefined' || init.popups_draggable) {
//		this.makeDraggable(true);
//	}
}

/**
   A UID for each Navpopup. This constructor property is just a counter.
   @type integer
   @private
*/
Navpopup.uid=0;

/**
   Retrieves the {@link #visible} attribute, indicating whether the popup is currently visible.
   @type boolean
*/
Navpopup.prototype.isVisible=function() {
	return this.visible;
};

/**
   Repositions popup using CSS style.
   @private
   @param {integer} x x-coordinate (px)
   @param {integer} y y-coordinate (px)
   @param {boolean} noLimitHor Don't call {@link #limitHorizontalPosition}
*/
Navpopup.prototype.reposition= function (x,y, noLimitHor) {
	log ('reposition('+x+','+y+','+noLimitHor+')');
	if (typeof x != 'undefined' && x!==null) { this.left=x; }
	if (typeof y != 'undefined' && y!==null) { this.top=y; }
	if (typeof this.left != 'undefined' && typeof this.top != 'undefined') {
		this.mainDiv.style.left=this.left + 'px';
		this.mainDiv.style.top=this.top + 'px';
	}
	if (!noLimitHor) { this.limitHorizontalPosition(); }
	//console.log('navpop'+this.uid+' - (left,top)=(' + this.left + ',' + this.top + '), css=('
	//+ this.mainDiv.style.left + ',' + this.mainDiv.style.top + ')');
};

/**
   Prevents popups from being in silly locations. Hopefully.
   Should not be run if {@link #constrained} is true.
   @private
*/
Navpopup.prototype.limitHorizontalPosition=function() {
	if (!this.constrained || this.tooWide) { return; }
	this.updateDimensions();
	var x=this.left;
	var w=this.width;
	var cWidth=document.body.clientWidth;


//	log('limitHorizontalPosition: x='+x+
//			', this.left=' + this.left +
//			', this.width=' + this.width +
//			', cWidth=' + cWidth);


	if ( (x+w) >= cWidth ||
		 ( x > 0 && this.maxWidth && this.width < this.maxWidth && this.height > this.width
		   && x > cWidth - this.maxWidth ) ) {
		// This is a very nasty hack. There has to be a better way!
		// We find the "natural" width of the div by positioning it at the far left
		// then reset it so that it should be flush right (well, nearly)
		this.mainDiv.style.left='-10000px';
		this.mainDiv.style.width = this.maxWidth + 'px';
		var naturalWidth=parseInt(this.mainDiv.offsetWidth, 10);
		var newLeft=cWidth - naturalWidth - 1;
		if (newLeft < 0) { newLeft = 0; this.tooWide=true; } // still unstable for really wide popups?
		log ('limitHorizontalPosition: moving to ('+newLeft + ','+ this.top+');' + ' naturalWidth=' + naturalWidth + ', clientWidth=' + cWidth);
		this.reposition(newLeft, null, true);
	}
};

/**
   Counter indicating the z-order of the "highest" popup.
   We start the z-index at 1000 so that popups are above everything
   else on the screen.
   @private
   @type integer
*/
Navpopup.highest=1000;

/**
   Brings popup to the top of the z-order.
   We increment the {@link #highest} property of the contructor here.
   @private
*/
Navpopup.prototype.raise = function () {
	this.mainDiv.style.zIndex=Navpopup.highest + 1;
	++Navpopup.highest;
};

/**
   Shows the popup provided {@link #noshow} is not true.
   Updates the position, brings the popup to the top of the z-order and unhides it.
*/
Navpopup.prototype.show = function () {
	//document.title+='s';
	if (this.noshow) { return; }
	//document.title+='t';
	this.reposition();
	this.raise();
	this.unhide();
};


/**
   Runs the {@link #show} method in a little while, unless we're
   already visible.
   @param {integer} time Delay in milliseconds
   @see #showSoonIfStable
*/
Navpopup.prototype.showSoon = function (time) {
	if (this.visible) { return; }
	this.noshow=false;
	//~ We have to save the value of @tt{this} so that the closure below
	//~ works.
	var savedThis=this;
	//this.start_x = Navpopup.tracker.x;
	//this.start_y = Navpopup.tracker.y;
	setTimeout(function () {
		if (Navpopup.tracker.active) {
			savedThis.reposition.apply(savedThis, [Navpopup.tracker.x + 2, Navpopup.tracker.y + 2]);
		}
		//~ Have to use apply to invoke his member function here
		savedThis.show.apply(savedThis, []);
	}, time);
};

/**
   Checks to see if the mouse pointer has
   stabilised (checking every <code>time</code>/2 milliseconds) and runs the
   {@link #show} method if it has. This method makes {@link #showSoon} redundant.
   @param {integer} time The minimum time (ms) before the popup may be shown.
*/
Navpopup.prototype.showSoonIfStable = function (time) {
	log ('showSoonIfStable, time='+time);
	if (this.visible) { return; }
	this.noshow = false;

	//~ initialize these variables so that we never run @tt{show} after
	//~ just half the time
	this.stable_x = -10000; this.stable_y = -10000;

	var stableShow = function() {
		log('stableShow called');
		var new_x = Navpopup.tracker.x, new_y = Navpopup.tracker.y;
		var dx = savedThis.stable_x - new_x, dy = savedThis.stable_y - new_y;
		var fuzz2 = 0; // savedThis.fuzz * savedThis.fuzz;
		//document.title += '[' + [savedThis.stable_x,new_x, savedThis.stable_y,new_y, dx, dy, fuzz2].join(',') + '] ';
		if ( dx * dx <= fuzz2 && dy * dy <= fuzz2 ) {
			log ('mouse is stable');
			clearInterval(savedThis.showSoonStableTimer);
			savedThis.reposition.apply(savedThis, [new_x + 2, new_y + 2]);
			savedThis.show.apply(savedThis, []);
			return;
		}
		savedThis.stable_x = new_x; savedThis.stable_y = new_y;
	};
	var savedThis = this;
	this.showSoonStableTimer = setInterval(stableShow, time/2);
};

/**
   Makes the popup unhidable until we call {@link #unstick}.
*/
Navpopup.prototype.stick=function() {
	this.noshow=false;
	this.sticky=true;
};

/**
   Allows the popup to be hidden.
*/
Navpopup.prototype.unstick=function() {
	this.sticky=false;
};

/**
   Sets the {@link #noshow} flag and hides the popup. This should be called
   when the mouse leaves the link before
   (or after) it's actually been displayed.
*/
Navpopup.prototype.banish = function () {
	log ('banish called');
	// hide and prevent showing with showSoon in the future
	this.noshow=true;
	if (this.showSoonStableTimer) {
		log('clearing showSoonStableTimer');
		clearInterval(this.showSoonStableTimer);
	}
	this.hide();
};

/**
   Runs hooks added with {@link #addHook}.
   @private
   @param {String} key Key name of the {@link #hooks} array - one of 'create', 'unhide', 'hide'
   @param {String} when Controls exactly when the hook is run: either 'before' or 'after'
*/
Navpopup.prototype.runHooks = function (key, when) {
	if (!this.hooks[key]) { return; }
	var keyHooks=this.hooks[key];
	var len=keyHooks.length;
	for (var i=0; i< len; ++i) {
		if (keyHooks[i] && keyHooks[i].when == when) {
			if (keyHooks[i].hook.apply(this, [])) {
				// remove the hook
				if (keyHooks[i].hookId) {
					delete this.hookIds[keyHooks[i].hookId];
				}
				keyHooks[i]=null;
			}
		}
	}
};

/**
   Adds a hook to the popup. Hook functions are run with <code>this</code> set to refer to the Navpopup instance, and no arguments.
   @param {Function} hook The hook function. Functions that return true are deleted.
   @param {String} key Key name of the {@link #hooks} array - one of 'create', 'unhide', 'hide'
   @param {String} when Controls exactly when the hook is run: either 'before' or 'after'
   @param {String} uid A truthy string identifying the hook function; if it matches another hook in this position, it won't be added again.
*/
Navpopup.prototype.addHook = function ( hook, key, when, uid ) {
	when = when || 'after';
	if (!this.hooks[key]) { return; }
	// if uid is specified, don't add duplicates
	var hookId=null;
	if (uid) {
		hookId=[key,when,uid].join('|');
		if (this.hookIds[hookId]) {
			return;
		}
		this.hookIds[hookId]=true;
	}
	this.hooks[key].push( {hook: hook, when: when, hookId: hookId} );
};

/**
   Creates the main DIV element, which contains all the actual popup content.
   Runs hooks with key 'create'.
   @private
*/
Navpopup.prototype.createMainDiv = function () {
	if (this.mainDiv) { return; }
	this.runHooks('create', 'before');
	var mainDiv=document.createElement('div');

	var savedThis=this;
	mainDiv.onclick=function(e) {savedThis.onclickHandler(e);};
	mainDiv.className=(this.className) ? this.className : 'navpopup_maindiv';
	mainDiv.id=mainDiv.className + this.uid;

	mainDiv.style.position='absolute';
	mainDiv.style.display='none';
	mainDiv.className='navpopup';

	// easy access to javascript object through DOM functions
	mainDiv.navpopup=this;

	this.mainDiv=mainDiv;
	document.body.appendChild(mainDiv);
	this.runHooks('create', 'after');
};
/**
   Calls the {@link #raise} method.
   @private
*/
Navpopup.prototype.onclickHandler=function(e) {
	this.raise();
};
/**
   Makes the popup draggable, using a {@link Drag} object.
   @private
*/
Navpopup.prototype.makeDraggable=function(handleName) {
	if (!this.mainDiv) { this.createMainDiv(); }
	var drag=new Drag();
	if (!handleName) {
		drag.startCondition=function(e) {
		try { if (!e.shiftKey) { return false; } } catch (err) { return false; }
		return true;
		};
	}
	var dragHandle;
	if (handleName) dragHandle = document.getElementById(handleName);
	if (!dragHandle) dragHandle = this.mainDiv;
	var np=this;
	drag.endHook=function(x,y) {
		Navpopup.tracker.dirty=true;
		np.reposition(x,y);
	};
	drag.init(dragHandle,this.mainDiv);
};

/** Hides the popup using CSS. Runs hooks with key 'hide'.
	Sets {@link #visible} appropriately.	 {@link #banish} should be called externally instead of this method.

	@private
*/
Navpopup.prototype.hide = function () {
	this.runHooks('hide', 'before');
	this.abortDownloads();
	if (this.sticky) { return; }
	if (typeof this.visible != 'undefined' && this.visible) {
		this.mainDiv.style.display='none';
		this.visible=false;
	}
	this.runHooks('hide', 'after');
};

/** Shows the popup using CSS. Runs hooks with key 'unhide'.
	Sets {@link #visible} appropriately.   {@link #show} should be called externally instead of this method.
	@private
*/
Navpopup.prototype.unhide = function () {
	this.runHooks('unhide', 'before');
	if (typeof this.visible != 'undefined' && !this.visible) {
		this.mainDiv.style.display='inline';
		this.visible=true;
	}
	this.runHooks('unhide', 'after');
};

/**
   Sets the <code>innerHTML</code> attribute of the main div containing the popup content.
   @param {String} html The HTML to set.
*/
Navpopup.prototype.setInnerHTML = function (html) {
	this.mainDiv.innerHTML = html;
};

/**
   Updates the {@link #width} and {@link #height} attributes with the CSS properties.
   @private
*/
Navpopup.prototype.updateDimensions = function () {
	this.width=parseInt(this.mainDiv.offsetWidth, 10);
	this.height=parseInt(this.mainDiv.offsetHeight, 10);
};

/**
   Checks if the point (x,y) is within {@link #fuzz} of the
   {@link #mainDiv}.
   @param {integer} x x-coordinate (px)
   @param {integer} y y-coordinate (px)
   @type boolean
*/
Navpopup.prototype.isWithin = function(x,y) {
	//~ If we're not even visible, no point should be considered as
	//~ being within the popup.
	if (!this.visible) { return false; }
	this.updateDimensions();
	var fuzz=this.fuzz || 0;
	//~ Use a simple box metric here.
	return (x+fuzz >= this.left && x-fuzz <= this.left + this.width &&
		y+fuzz >= this.top  && y-fuzz <= this.top  + this.height);
};

/**
   Adds a download to {@link #downloads}.
   @param {Downloader} download
*/
Navpopup.prototype.addDownload=function(download) {
	if (!download) { return; }
	this.downloads.push(download);
};
/**
   Aborts the downloads listed in {@link #downloads}.
   @see Downloader#abort
*/
Navpopup.prototype.abortDownloads=function() {
	for(var i=0; i<this.downloads.length; ++i) {
		var d=this.downloads[i];
		if (d && d.abort) { d.abort(); }
	}
	this.downloads=[];
};


/**
   A {@link Mousetracker} instance which is a property of the constructor (pseudo-global).
*/
Navpopup.tracker=new Mousetracker();
// ENDFILE: navpopup.js
// STARTFILE: diff.js
//<NOLITE>
/*
 * Javascript Diff Algorithm
 *  By John Resig (http://ejohn.org/) and [[:en:User:Lupin]]
 *
 * More Info:
 *  http://ejohn.org/projects/javascript-diff-algorithm/
 */

function delFmt(x) {
	if (!x.length) { return ''; }
	return "<del class='popupDiff'>" + x.join('') +"</del>";
}
function insFmt(x) {
	if (!x.length) { return ''; }
	return "<ins class='popupDiff'>" + x.join('') +"</ins>";
}

function countCrossings(a, b, i, eject) {
	// count the crossings on the edge starting at b[i]
	if (!b[i].row && b[i].row !== 0) { return -1; }
	var count=0;
	for (var j=0; j<a.length; ++j) {
		if (!a[j].row && a[j].row !== 0) { continue; }
		if ( (j-b[i].row)*(i-a[j].row) > 0) {
			if(eject) { return true; }
			count++;
		}
	}
	return count;
}

function shortenDiffString(str, context) {
	var re=RegExp('(<del[\\s\\S]*?</del>|<ins[\\s\\S]*?</ins>)');
	var splitted=str.parenSplit(re);
	var ret=[''];
	for (var i=0; i<splitted.length; i+=2) {
		if (splitted[i].length < 2*context) {
			ret[ret.length-1] += splitted[i];
			if (i+1<splitted.length) { ret[ret.length-1] += splitted[i+1]; }
			continue;
		}
		else {
			if (i > 0) { ret[ret.length-1] += splitted[i].substring(0,context); }
			if (i+1 < splitted.length) {
				ret.push(splitted[i].substring(splitted[i].length-context) +
					 splitted[i+1]);
			}
		}
	}
	while (ret.length > 0 && !ret[0]) { ret = ret.slice(1); }
	return ret;
}


function diffString( o, n, simpleSplit ) {
	var splitRe=RegExp('([[]{2}|[\\]]{2}|[{]{2,3}|[}]{2,3}|[|]|=|<|>|[*:]+|\\s|\\b)');

	//  We need to split the strings o and n first, and entify() the parts
	//  individually, so that the HTML entities are never cut apart. (AxelBoldt)
	var out, i, oSplitted, nSplitted;
	if (simpleSplit) { 
		oSplitted=o.split(/\b/); 
		nSplitted=n.split(/\b/); 
	} else { 
		oSplitted=o.parenSplit(splitRe); 
		nSplitted=n.parenSplit(splitRe); 
	}
	for (i=0; i<oSplitted.length; ++i) {oSplitted[i]=oSplitted[i].entify();}
	for (i=0; i<nSplitted.length; ++i) {nSplitted[i]=nSplitted[i].entify();}
		
	out = diff (oSplitted, nSplitted);
	var str = "";
	var acc=[]; // accumulator for prettier output

	// crossing pairings -- eg 'A B' vs 'B A' -- cause problems, so let's iron them out
	// this doesn't always do things optimally but it should be fast enough
	var maxOutputPair=0;
	for (i=0; i<out.n.length; ++i) {
		if ( out.n[i].paired ) {
		if( maxOutputPair > out.n[i].row ) {
			// tangle - delete pairing
			out.o[ out.n[i].row ]=out.o[ out.n[i].row ].text;
			out.n[i]=out.n[i].text;
		}
		if (maxOutputPair < out.n[i].row) { maxOutputPair = out.n[i].row; }
		}
	}

	// output the stuff preceding the first paired old line
	for (i=0; i<out.o.length && !out.o[i].paired; ++i) { acc.push( out.o[i] ); }
	str += delFmt(acc); acc=[];

	// main loop
	for ( i = 0; i < out.n.length; ++i ) {
		// output unpaired new "lines"
		while ( i < out.n.length && !out.n[i].paired ) { acc.push( out.n[i++] ); }
		str += insFmt(acc); acc=[];
		if ( i < out.n.length ) { // this new "line" is paired with the (out.n[i].row)th old "line"
			str += out.n[i].text;
			// output unpaired old rows starting after this new line's partner
			var m = out.n[i].row + 1;
			while ( m < out.o.length && !out.o[m].paired ) { acc.push ( out.o[m++] ); }
			str += delFmt(acc); acc=[];
		}
	}
	return str;
}

// see http://developer.mozilla.org/en/docs/Core_JavaScript_1.5_Reference:Global_Objects:Object
// FIXME: use obj.hasOwnProperty instead of this kludge!
window.jsReservedProperties=RegExp('^(constructor|prototype|__((define|lookup)[GS]etter)__' +
				   '|eval|hasOwnProperty|propertyIsEnumerable' +
				   '|to(Source|String|LocaleString)|(un)?watch|valueOf)$');
function diffBugAlert(word) {
	if (!diffBugAlert.list[word]) {
		diffBugAlert.list[word]=1;
		alert('Bad word: '+word+'\n\nPlease report this bug.');
	}
}
diffBugAlert.list={};

function makeDiffHashtable(src) {
	var ret={};
	for ( var i = 0; i < src.length; i++ ) {
		if ( jsReservedProperties.test(src[i]) ) { src[i] += '<!-- -->'; }
		if ( !ret[ src[i] ] ) { ret[ src[i] ] = []; }
		try { ret[ src[i] ].push( i ); } catch (err) { diffBugAlert(src[i]); }
	}
	return ret;
}

function diff( o, n ) {

	// pass 1: make hashtable ns with new rows as keys
	var ns = makeDiffHashtable(n);

	// pass 2: make hashtable os with old rows as keys
	var os = makeDiffHashtable(o);

	// pass 3: pair unique new rows and matching unique old rows
	var i;
	for ( i in ns ) {
		if ( ns[i].length == 1 && os[i] && os[i].length == 1 ) {
			n[ ns[i][0] ] = { text: n[ ns[i][0] ], row: os[i][0], paired: true };
			o[ os[i][0] ] = { text: o[ os[i][0] ], row: ns[i][0], paired: true };
		}
	}

	// pass 4: pair matching rows immediately following paired rows (not necessarily unique)
	for ( i = 0; i < n.length - 1; i++ ) {
		if ( n[i].paired && ! n[i+1].paired && n[i].row + 1 < o.length && ! o[ n[i].row + 1 ].paired &&
			 n[i+1] == o[ n[i].row + 1 ] ) {
			n[i+1] = { text: n[i+1], row: n[i].row + 1, paired: true };
			o[n[i].row+1] = { text: o[n[i].row+1], row: i + 1, paired: true };
		}
	}

	// pass 5: pair matching rows immediately preceding paired rows (not necessarily unique)
	for ( i = n.length - 1; i > 0; i-- ) {
		if ( n[i].paired && ! n[i-1].paired && n[i].row > 0 && ! o[ n[i].row - 1 ].paired &&
			 n[i-1] == o[ n[i].row - 1 ] ) {
			n[i-1] = { text: n[i-1], row: n[i].row - 1, paired: true };
			o[n[i].row-1] = { text: o[n[i].row-1], row: i - 1, paired: true };
		}
	}

	return { o: o, n: n };
}
//</NOLITE>
// ENDFILE: diff.js
// STARTFILE: init.js
function setSiteInfo() {
	if (window.popupLocalDebug) {
		pg.wiki.hostname = 'en.wikipedia.org';
	} else {
		pg.wiki.hostname = location.hostname; // use in preference to location.hostname for flexibility (?)
	}
	pg.wiki.wikimedia=RegExp('(wiki([pm]edia|source|books|news|quote|versity)|wiktionary|mediawiki)[.]org').test(pg.wiki.hostname);
	pg.wiki.wikia=RegExp('[.]wikia[.]com$', 'i').test(pg.wiki.hostname);
	pg.wiki.isLocal=RegExp('^localhost').test(pg.wiki.hostname);
	pg.wiki.commons=( pg.wiki.wikimedia && pg.wiki.hostname != 'commons.wikimedia.org') ? 'commons.wikimedia.org' : null;
	pg.wiki.lang = mw.config.get('wgContentLanguage');
	var port = location.port ? ':' + location.port : '';
	pg.wiki.sitebase = pg.wiki.hostname + port;
}

function setTitleBase() {
	var protocol = ( window.popupLocalDebug ? 'http:' : location.protocol );
	pg.wiki.articlePath = mw.config.get('wgArticlePath').replace(/\/\$1/, "");   // as in http://some.thing.com/wiki/Article
	pg.wiki.botInterfacePath = mw.config.get('wgScript');
	pg.wiki.APIPath = mw.config.get('wgScriptPath') +"/api.php";
	// default mediawiki setting is paths like http://some.thing.com/articlePath/index.php?title=foo

	var titletail = pg.wiki.botInterfacePath + '?title=';
	//var titletail2 = joinPath([pg.wiki.botInterfacePath, 'wiki.phtml?title=']);

	// other sites may need to add code here to set titletail depending on how their urls work

	pg.wiki.titlebase   = protocol + '//' + pg.wiki.sitebase + titletail;
	//pg.wiki.titlebase2  = protocol + '//' + joinPath([pg.wiki.sitebase, titletail2]);
	pg.wiki.wikibase	= protocol + '//' + pg.wiki.sitebase + pg.wiki.botInterfacePath;
	pg.wiki.apiwikibase = protocol + '//' + pg.wiki.sitebase + pg.wiki.APIPath;
	pg.wiki.articlebase = protocol + '//' + pg.wiki.sitebase + pg.wiki.articlePath;
	pg.wiki.commonsbase = protocol + '//' + pg.wiki.commons  + pg.wiki.botInterfacePath;
	pg.wiki.apicommonsbase = protocol + '//' + pg.wiki.commons  + pg.wiki.APIPath;
	pg.re.basenames = RegExp( '^(' +
				  map( literalizeRegex, [ pg.wiki.titlebase, //pg.wiki.titlebase2,
							  pg.wiki.articlebase ]).join('|') + ')' );
}


//////////////////////////////////////////////////
// Global regexps

function setMainRegex() {
	var reStart='[^:]*://';
	var preTitles = literalizeRegex( mw.config.get('wgScriptPath') ) + '/(?:index[.]php|wiki[.]phtml)[?]title=';
	preTitles += '|' + literalizeRegex( pg.wiki.articlePath + '/' );

	var reEnd='(' + preTitles + ')([^&?#]*)[^#]*(?:#(.+))?';
	pg.re.main = RegExp(reStart + literalizeRegex(pg.wiki.sitebase) + reEnd);
}

function setRegexps() {
	setMainRegex();
	var sp=nsRe(pg.nsSpecialId);
	pg.re.urlNoPopup=RegExp('((title=|/)' + sp + '(?:%3A|:)|section=[0-9]|^#$)') ;
	pg.re.contribs  =RegExp('(title=|/)'  + sp + '(?:%3A|:)Contributions' + '(&target=|/|/' + mw.config.get('wgFormattedNamespaces')[pg.nsUserId]+':)(.*)') ;
	pg.re.email	    =RegExp('(title=|/)'  + sp + '(?:%3A|:)EmailUser'	 + '(&target=|/|/(?:' + mw.config.get('wgFormattedNamespaces')[pg.nsUserId]+':)?)(.*)') ;
	pg.re.backlinks =RegExp('(title=|/)'  + sp + '(?:%3A|:)WhatLinksHere' + '(&target=|/)([^&]*)');
	pg.re.specialdiff=RegExp('/'          + sp + '(?:%3A|:)Diff/([^?#]*)');

//<NOLITE>
	var im=nsReImage();
	// note: tries to get images in infobox templates too, e.g. movie pages, album pages etc
	//					  (^|\[\[)image: *([^|\]]*[^|\] ]) *
	//					  (^|\[\[)image: *([^|\]]*[^|\] ])([^0-9\]]*([0-9]+) *px)?
	//														$4 = 120 as in 120px
	pg.re.image = RegExp('(^|\\[\\[)' + im + ': *([^|\\]]*[^|\\] ])' +
				 '([^0-9\\]]*([0-9]+) *px)?|(?:\\n *[|]?|[|]) *' +
				 '(' + getValueOf('popupImageVarsRegexp') + ')' +
				 ' *= *(?:\\[\\[ *)?(?:' + im + ':)?' +
				 '([^|]*?)(?:\\]\\])? *[|]? *\\n', 'img') ;
	pg.re.imageBracketCount = 6;

	pg.re.category = RegExp('\\[\\[' +nsRe(pg.nsCategoryId) +
				': *([^|\\]]*[^|\\] ]) *', 'i');
	pg.re.categoryBracketCount = 1;

	pg.re.ipUser=RegExp('^' +
				// IPv6
				'(?::(?::|(?::[0-9A-Fa-f]{1,4}){1,7})|[0-9A-Fa-f]{1,4}(?::[0-9A-Fa-f]{1,4}){0,6}::|[0-9A-Fa-f]{1,4}(?::[0-9A-Fa-f]{1,4}){7})' +
				// IPv4
				'|(((25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9])\\.){3}' +
				'(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9]))$');

	pg.re.stub= RegExp(getValueOf('popupStubRegexp'), 'im');
	pg.re.disambig=RegExp(getValueOf('popupDabRegexp'), 'im');

//</NOLITE>
	// FIXME replace with general parameter parsing function, this is daft
	pg.re.oldid=RegExp('[?&]oldid=([^&]*)');
	pg.re.diff=RegExp('[?&]diff=([^&]*)');
}


//////////////////////////////////////////////////
// miscellany

function setupCache() {
	// page caching
	pg.cache.pages = [];
}

function setMisc() {
	pg.current.link=null;
	pg.current.links=[];
	pg.current.linksHash={};

	setupCache();

	pg.timer.checkPopupPosition=null;
	pg.counter.loop=0;

	// ids change with each popup: popupImage0, popupImage1 etc
	pg.idNumber=0;

	// for myDecodeURI
	pg.misc.decodeExtras = [
		{from: '%2C', to: ',' },
		{from: '_',   to: ' ' },
		{from: '%24', to: '$'},
		{from: '%26',   to: '&' } // no ,
		];

}

function leadingInteger(s){
	var n=s.match(/^(\d*)/)[1];
	if (n) { return +n; }
	return null;
}

function setBrowserHacks() {
	var useOriginal=false;
	// browser-specific hacks
	if (typeof window.opera != 'undefined') {
		//if (leadingInteger(opera.version()) < 9)
		{ useOriginal=true; } // v9 beta still seems to have buggy css
		setDefault('popupNavLinkSeparator', ' &#183; ');
	} else if (navigator.appName=='Konqueror') {
		setDefault('popupNavLinkSeparator', ' &bull; ');
		pg.flag.isKonq=true;
	} else if ( navigator.vendor && navigator.vendor.toLowerCase().indexOf('apple computer')===0) {
		pg.flag.isSafari=true;
		var webkit=+navigator.userAgent.replace(RegExp('^.*AppleWebKit[/](\\d+).*', 'i'), '$1');
		if (webkit < 420) { useOriginal=true; }
	} else if (navigator.appName.indexOf("Microsoft")!=-1) {
		setDefault('popupNavLinkSeparator', ' &#183; ');
		var ver=+navigator.userAgent.replace(RegExp('^.*MSIE (\\d+).*'), '$1');
		pg.flag.isIE=true;
		pg.flag.IEVersion=ver;
	}
	if (pg.flag.isIE && pg.flag.IEVersion < 8) {
		useOriginal=true;
	}
	if ((pg.flag.isIE && pg.flag.IEVersion < 7) || pg.flag.isKonq || (pg.flag.isSafari && webkit < 420)) {
		pg.flag.linksLikeIE6=true;
	}
	if (useOriginal && pg.structures.original) {
		setDefault('popupStructure','original');
	}
}

// We need a callback since this might end up asynchronous because of
// the mw.loader.using() call.
function setupPopups( callback ) {
	mw.loader.using( 'mediawiki.user', function() {
		// NB translatable strings should be set up first (strings.js)
		// basics
		setupDebugging();
		setSiteInfo();
		setTitleBase();
		setOptions(); // see options.js

		// namespaces etc
		setNamespaces();
		setInterwiki();

		// regexps
		setRegexps();
		setRedirs();

		// other stuff
		setBrowserHacks();
		setMisc();
		setupLivePreview();

		// main deal here
		setupTooltips();
		Navpopup.tracker.enable();

		setupPopups.completed = true;
		if ( $.isFunction( callback ) ) {
			callback();
		}
	});
}
// ENDFILE: init.js
// STARTFILE: navlinks.js
//<NOLITE>
//////////////////////////////////////////////////
// navlinks... let the fun begin
//

function defaultNavlinkSpec() {
	var str='';
	str += '<b><<mainlink|shortcut= >></b>';
	if (getValueOf('popupLastEditLink')) {
		str += '*<<lastEdit|shortcut=/>>|<<lastContrib>>|<<sinceMe>>if(oldid){|<<oldEdit>>|<<diffCur>>}';
	}

	// user links
	// contribs - log - count - email - block
	// count only if applicable; block only if popupAdminLinks
	str += 'if(user){<br><<contribs|shortcut=c>>*<<userlog|shortcut=L|log>>';
	str+='if(ipuser){*<<arin>>}if(wikimedia){*<<count|shortcut=#>>}';
	str+='if(ipuser){}else{*<<email|shortcut=E>>}if(admin){*<<block|shortcut=b>>|<<blocklog|log>>}}';

	// editing links
	// talkpage   -> edit|new - history - un|watch - article|edit
	// other page -> edit - history - un|watch - talk|edit|new
	var editstr='<<edit|shortcut=e>>';
	var editOldidStr='if(oldid){<<editOld|shortcut=e>>|<<revert|shortcut=v|rv>>|<<edit|cur>>}else{' + editstr + '}'
		var historystr='<<history|shortcut=h>>if(mainspace_en){|<<editors|shortcut=E|>>}';
	var watchstr='<<unwatch|unwatchShort>>|<<watch|shortcut=w|watchThingy>>';

	str+='<br>if(talk){' +
		editOldidStr+'|<<new|shortcut=+>>' + '*' + historystr+'*'+watchstr + '*' +
		'<b><<article|shortcut=a>></b>|<<editArticle|edit>>' +
		'}else{' + // not a talk page
		editOldidStr + '*' + historystr + '*' + watchstr + '*' +
		'<b><<talk|shortcut=t>></b>|<<editTalk|edit>>|<<newTalk|shortcut=+|new>>'
		+ '}';

	// misc links
	str += '<br><<whatLinksHere|shortcut=l>>*<<relatedChanges|shortcut=r>>*<<move|shortcut=m>>';

	// admin links
	str += 'if(admin){<br><<unprotect|unprotectShort>>|<<protect|shortcut=p>>|<<protectlog|log>>*' +
		'<<undelete|undeleteShort>>|<<delete|shortcut=d>>|<<deletelog|log>>}';
	return str;
}

function navLinksHTML (article, hint, params) { //oldid, rcid) {
	var str = '<span class="popupNavLinks">' + defaultNavlinkSpec() + '</span>';
	// BAM
	return navlinkStringToHTML(str, article, params);
}

function expandConditionalNavlinkString(s,article,z,recursionCount) {
	var oldid=z.oldid, rcid=z.rcid, diff=z.diff;
	// nested conditionals (up to 10 deep) are ok, hopefully! (work from the inside out)
	if (typeof recursionCount!=typeof 0) { recursionCount=0; }
	var conditionalSplitRegex=RegExp(
		//(1	 if	\\(	(2	2)	\\)	  {(3	3)}  (4   else	  {(5	 5)}  4)1)
		'(;?\\s*if\\s*\\(\\s*([\\w]*)\\s*\\)\\s*\\{([^{}]*)\\}(\\s*else\\s*\\{([^{}]*?)\\}|))', 'i');
	var splitted=s.parenSplit(conditionalSplitRegex);
	// $1: whole conditional
	// $2: test condition
	// $3: true expansion
	// $4: else clause (possibly empty)
	// $5: false expansion (possibly null)
	var numParens=5;
	var ret = splitted[0];
	for (var i=1; i<splitted.length; i=i+numParens+1) {

		var testString=splitted[i+2-1];
		var trueString=splitted[i+3-1];
		var falseString=splitted[i+5-1];
		if (typeof falseString=='undefined' || !falseString) { falseString=''; }
		var testResult=null;

		switch (testString) {
		case 'user':
			testResult=(article.userName())?true:false;
			break;
		case 'talk':
			testResult=(article.talkPage())?false:true; // talkPage converts _articles_ to talkPages
			break;
		case 'admin':
			testResult=getValueOf('popupAdminLinks')?true:false;
			break;
		case 'oldid':
			testResult=(typeof oldid != 'undefined' && oldid)?true:false;
			break;
		case 'rcid':
			testResult=(typeof rcid != 'undefined' && rcid)?true:false;
			break;
		case 'ipuser':
			testResult=(article.isIpUser())?true:false;
			break;
		case 'mainspace_en':
			testResult=isInMainNamespace(article) &&
				pg.wiki.hostname=='en.wikipedia.org';
			break;
		case 'wikimedia':
			testResult=(pg.wiki.wikimedia) ? true : false;
			break;
		case 'diff':
			testResult=(typeof diff != 'undefined' && diff)?true:false;
			break;
		}

		switch(testResult) {
		case null: ret+=splitted[i];  break;
		case true: ret+=trueString;   break;
		case false: ret+=falseString; break;
		}

		// append non-conditional string
		ret += splitted[i+numParens];
	}
	if (conditionalSplitRegex.test(ret) && recursionCount < 10) {
		return expandConditionalNavlinkString(ret,article,z,recursionCount+1);
	}
	return ret;
}

function navlinkStringToArray(s, article, params) {
	s=expandConditionalNavlinkString(s,article,params);
	var splitted=s.parenSplit(RegExp('<<(.*?)>>'));
	var ret=[];
	for (var i=0; i<splitted.length; ++i) {
		if (i%2) { // i odd, so s is a tag
			var t=new navlinkTag();
			var ss=splitted[i].split('|');
			t.id=ss[0];
			for (var j=1; j<ss.length; ++j) {
				var sss=ss[j].split('=');
				if (sss.length>1) {
					t[sss[0]]=sss[1];
				}
				else { // no assignment (no "="), so treat this as a title (overwriting the last one)
					t.text=popupString(sss[0]);
				}
			}
			t.article=article;
			var oldid=params.oldid, rcid=params.rcid, diff=params.diff;
			if (typeof oldid != 'undefined' && oldid != null) { t.oldid=oldid; }
			if (typeof rcid != 'undefined' && rcid != null) { t.rcid=rcid; }
			if (typeof diff != 'undefined' && diff != null) { t.diff=diff; }
			if (!t.text && t.id != 'mainlink') { t.text=popupString(t.id); }
			ret.push(t);
		}
		else { // plain HTML
			ret.push(splitted[i]);
		}
	}
	return ret;
}


function navlinkSubstituteHTML(s) {
	return s.split('*').join(getValueOf('popupNavLinkSeparator'))
		.split('<menurow>').join('<li class="popup_menu_row">')
		.split('</menurow>').join('</li>')
		.split('<menu>').join('<ul class="popup_menu">')
		.split('</menu>').join('</ul>');

}

function navlinkDepth(magic,s) {
	return s.split('<' + magic + '>').length - s.split('</' + magic + '>').length;
}


// navlinkString: * becomes the separator
//				<<foo|bar=baz|fubar>> becomes a foo-link with attribute bar='baz'
//									  and visible text 'fubar'
//				if(test){...} and if(test){...}else{...} work too (nested ok)

function navlinkStringToHTML(s,article,params) {
	//limitAlert(navlinkStringToHTML, 5, 'navlinkStringToHTML\n' + article + '\n' + (typeof article));
	var p=navlinkStringToArray(s,article,params);
	var html='';
	var menudepth = 0; // nested menus not currently allowed, but doesn't do any harm to code for it
	var menurowdepth = 0;
	var wrapping = null;
	for (var i=0; i<p.length; ++i) {
		if (typeof p[i] == typeof '') {
			html+=navlinkSubstituteHTML(p[i]);
			menudepth += navlinkDepth('menu', p[i]);
			menurowdepth += navlinkDepth('menurow', p[i]);
//			if (menudepth === 0) {
//				tagType='span';
//			} else if (menurowdepth === 0) {
//				tagType='li';
//			} else {
//				tagType = null;
//			}
		} else if (typeof p[i].type != 'undefined' && p[i].type=='navlinkTag') {
			if (menudepth > 0 && menurowdepth === 0) {
				html += '<li class="popup_menu_item">' + p[i].html() + '</li>';
			} else {
				html+=p[i].html();
			}
		}
	}
	return html;
}

function navlinkTag() {
	this.type='navlinkTag';
}

navlinkTag.prototype.html=function () {
	this.getNewWin();
	this.getPrintFunction();
	var html='';
	var opening, closing;
	var tagType='span';
	if (!tagType) {
		opening = ''; closing = '';
	} else {
		opening = '<' + tagType + ' class="popup_' + this.id + '">';
		closing = '</' + tagType + '>';
	}
	if (typeof this.print!='function') {
		errlog ('Oh dear - invalid print function for a navlinkTag, id='+this.id);
	} else {
		html=this.print(this);
		if (typeof html != typeof '') {html='';}
		else if (typeof this.shortcut!='undefined') html=addPopupShortcut(html, this.shortcut);
	}
	return opening + html + closing;
};

navlinkTag.prototype.getNewWin=function() {
	getValueOf('popupLinksNewWindow');
	if (typeof pg.option.popupLinksNewWindow[this.id] === 'undefined') { this.newWin=null; }
	this.newWin=pg.option.popupLinksNewWindow[this.id];
}

navlinkTag.prototype.getPrintFunction=function() { //think about this some more
	// this.id and this.article should already be defined
	if (typeof this.id!=typeof '' || typeof this.article!=typeof {} ) { return; }
	var html='';
	var a,t;

	this.noPopup=1;
	switch (this.id) {
	case 'contribs': case 'history': case 'whatLinksHere':
	case 'userPage': case 'monobook': case 'userTalk':
	case 'talk': case 'article': case 'lastEdit':
		this.noPopup=null;
	}
	switch (this.id) {
	case 'email':	 case 'contribs':  case 'block':	 case 'unblock':
	case 'userlog':   case 'userSpace': case 'deletedContribs':
		this.article=this.article.userName();
	}

	switch (this.id) {
	case 'userTalk': case 'newUserTalk': case 'editUserTalk':
	case 'userPage': case 'monobook': case 'editMonobook': case 'blocklog':
		this.article=this.article.userName(true);
	// fall through; no break
	case 'pagelog': case 'deletelog': case 'protectlog':
	delete this.oldid;
	}

	if (this.id=='editMonobook' || this.id=='monobook') { this.article.append('/monobook.js'); }

	if (this.id != 'mainlink') {
		// FIXME anchor handling should be done differently with Title object
		this.article=this.article.removeAnchor();
		// if (typeof this.text=='undefined') this.text=popupString(this.id);
	}

	switch (this.id) {
	case 'undelete':       this.print=specialLink; this.specialpage='Undelete'; this.sep='/'; break;
	case 'whatLinksHere':  this.print=specialLink; this.specialpage='Whatlinkshere'; break;
	case 'relatedChanges': this.print=specialLink; this.specialpage='Recentchangeslinked'; break;
	case 'move':           this.print=specialLink; this.specialpage='Movepage'; break;
	case 'contribs':       this.print=specialLink; this.specialpage='Contributions'; break;
	case 'deletedContribs':this.print=specialLink; this.specialpage='Deletedcontributions'; break;
	case 'email':          this.print=specialLink; this.specialpage='EmailUser'; this.sep='/'; break;
	case 'block':          this.print=specialLink; this.specialpage='Blockip'; this.sep='&ip='; break;
	case 'unblock':        this.print=specialLink; this.specialpage='Ipblocklist'; this.sep='&action=unblock&ip='; break;
	case 'userlog':        this.print=specialLink; this.specialpage='Log'; this.sep='&user='; break;
	case 'blocklog':       this.print=specialLink; this.specialpage='Log'; this.sep='&type=block&page='; break;
	case 'pagelog':        this.print=specialLink; this.specialpage='Log'; this.sep='&page='; break;
	case 'protectlog':     this.print=specialLink; this.specialpage='Log'; this.sep='&type=protect&page='; break;
	case 'deletelog':      this.print=specialLink; this.specialpage='Log'; this.sep='&type=delete&page='; break;
	case 'userSpace':      this.print=specialLink; this.specialpage='PrefixIndex'; this.sep='&namespace=2&prefix='; break;
	case 'search':         this.print=specialLink; this.specialpage='Search'; this.sep='&fulltext=Search&search='; break;
	case 'unwatch': case 'watch':
	this.print=magicWatchLink; this.action=this.id+'&autowatchlist=1&autoimpl=' + popupString('autoedit_version') + '&actoken='+autoClickToken(); break;
	case 'history': case 'historyfeed': 
	case 'unprotect': case 'protect':
	this.print=wikiLink; this.action=this.id; break;

	case 'delete':
	this.print=wikiLink; this.action='delete';
	if (this.article.namespaceId()==pg.nsImageId) {
		var img=this.article.stripNamespace();
		this.action+='&image='+img;
	}
	break;

	case 'markpatrolled':
	case 'edit': // editOld should keep the oldid, but edit should not.
	delete this.oldid; // fall through
	case 'view': case 'purge': case 'render':
	this.print=wikiLink;
	this.action=this.id; break;
	case 'raw':
	this.print=wikiLink; this.action='raw&ctype=text/css'; break;
	case 'new':
	this.print=wikiLink; this.action='edit&section=new'; break;
	case 'mainlink':
	if (typeof this.text=='undefined') { this.text=this.article.toString().entify(); }
	if (getValueOf('popupSimplifyMainLink') && isInStrippableNamespace(this.article)) {
		var s=this.text.split('/'); this.text=s[s.length-1];
		if (this.text=='' && s.length > 1) { this.text=s[s.length-2]; }
	}
	this.print=titledWikiLink;
	if (typeof this.title=='undefined' && pg.current.link && typeof pg.current.link.href != 'undefined') {
		this.title=safeDecodeURI((pg.current.link.originalTitle)?pg.current.link.originalTitle:this.article);
		if (typeof this.oldid != 'undefined' && this.oldid) {
		this.title=tprintf('Revision %s of %s', [this.oldid, this.title]);
		}
	}
	this.action='view'; break;
	case 'userPage':
	case 'article':
	case 'monobook':
	case 'editMonobook':
	case 'editArticle':
	delete this.oldid;
	//alert(this.id+'\n'+this.article + '\n'+ typeof this.article);
	this.article=this.article.articleFromTalkOrArticle();
	//alert(this.id+'\n'+this.article + '\n'+ typeof this.article);
	this.print=wikiLink;
	if (this.id.indexOf('edit')==0) {
		this.action='edit';
	} else { this.action='view';}
	break;
	case 'userTalk':
	case 'talk':
	this.article=this.article.talkPage();
	delete this.oldid;
	this.print=wikiLink;
	this.action='view'; break;
	case 'arin':
	this.print=arinLink; break;
	case 'count':
	this.print=editCounterLink; break;
	case 'google':
	this.print=googleLink; break;
	case 'editors':
	this.print=editorListLink; break;
	case 'globalsearch':
	this.print=globalSearchLink; break;
	case 'lastEdit':
	this.print=titledDiffLink;
	this.title=popupString('Show the last edit');
	this.from='prev'; this.to='cur'; break;
	case 'oldEdit':
	this.print=titledDiffLink;
	this.title=popupString('Show the edit made to get revision') + ' ' + this.oldid;
	this.from='prev'; this.to=this.oldid; break;
	case 'editOld':
	this.print=wikiLink; this.action='edit'; break;
	case 'undo':
	this.print=wikiLink; this.action='edit&undo='; break;
	case 'markpatrolled':
	this.print=wikiLink; this.action='markpatrolled';
	case 'revert':
	this.print=wikiLink; this.action='revert'; break;
	case 'nullEdit':
	this.print=wikiLink; this.action='nullEdit'; break;
	case 'diffCur':
	this.print=titledDiffLink;
	this.title=tprintf('Show changes since revision %s', [this.oldid]);
	this.from=this.oldid; this.to='cur'; break;
	case 'editUserTalk':
	case 'editTalk':
	delete this.oldid;
	this.article=this.article.talkPage();
	this.action='edit'; this.print=wikiLink; break;
	case 'newUserTalk':
	case 'newTalk':
	this.article=this.article.talkPage();
	this.action='edit&section=new'; this.print=wikiLink; break;
	case 'lastContrib':
	case 'sinceMe':
	this.print=magicHistoryLink;
	break;
	case 'togglePreviews':
	this.text=popupString(pg.option.simplePopups ? 'enable previews' : 'disable previews');
	case 'disablePopups': case 'purgePopups':
	this.print=popupMenuLink;
	break;
	default:
	this.print=function () {return 'Unknown navlink type: '+this.id+''};
	}
};
//
//  end navlinks
//////////////////////////////////////////////////
//</NOLITE>
// ENDFILE: navlinks.js
// STARTFILE: shortcutkeys.js
//<NOLITE>
function popupHandleKeypress(evt) {
	var keyCode = window.event ? window.event.keyCode : ( evt.keyCode ? evt.keyCode : evt.which);
	if (!keyCode || !pg.current.link || !pg.current.link.navpopup) { return; }
	if (keyCode==27) { // escape
		killPopup();
		return false; // swallow keypress
	}

	var letter=String.fromCharCode(keyCode);
	var links=pg.current.link.navpopup.mainDiv.getElementsByTagName('A');
	var startLink=0;
	var i,j;

	if (popupHandleKeypress.lastPopupLinkSelected) {
		for (i=0; i<links.length; ++i) {
			if (links[i]==popupHandleKeypress.lastPopupLinkSelected) { startLink=i; }
		}
	}
	for (j=0; j<links.length; ++j) {
		i=(startLink + j + 1) % links.length;
		if (links[i].getAttribute('popupkey')==letter) {
			if (evt && evt.preventDefault) evt.preventDefault();
			links[i].focus();
			popupHandleKeypress.lastPopupLinkSelected=links[i];
			return false; // swallow keypress
		}
	}

	// pass keypress on
	if (document.oldPopupOnkeypress) { return document.oldPopupOnkeypress(evt); }
	return true;
}

function addPopupShortcuts() {
	if (document.onkeypress!=popupHandleKeypress) {
		document.oldPopupOnkeypress=document.onkeypress;
	}
	document.onkeypress=popupHandleKeypress;
}

function rmPopupShortcuts() {
	popupHandleKeypress.lastPopupLinkSelected=null;
	try {
		if (document.oldPopupOnkeypress && document.oldPopupOnkeypress==popupHandleKeypress) {
			// panic
			document.onkeypress=null; //function () {};
			return;
		}
		document.onkeypress=document.oldPopupOnkeypress;
	} catch (nasties) { /* IE goes here */ }
}


function addLinkProperty(html, property) {
	// take "<a href=...>...</a> and add a property
	// not sophisticated at all, easily broken
	var i=html.indexOf('>');
	if (i<0) { return html; }
	return html.substring(0,i) + ' ' + property + html.substring(i);
}

function addPopupShortcut(html, key) {
	if (!getValueOf('popupShortcutKeys')) { return html; }
	var ret= addLinkProperty(html, 'popupkey="'+key+'"');
	if (key==' ') { key=popupString('spacebar'); }
	return ret.replace(RegExp('^(.*?)(title=")(.*?)(".*)$', 'i'),'$1$2$3 ['+key+']$4');
}
//</NOLITE>
// ENDFILE: shortcutkeys.js
// STARTFILE: diffpreview.js
//<NOLITE>
function loadDiff(article, oldid, diff, navpop) {
	navpop.diffData={};
	var oldRev, newRev;
	switch (diff) {
	case 'cur':
		switch ( oldid ) {
		case null:
		case '':
		case 'prev':
			// eg newmessages diff link
			oldRev='0&direction=prev';
			newRev=0;
			break;
		default:
			oldRev = oldid;
			newRev = ( oldid || 0 ) + '&direction=cur';
		}
		break;
	case 'prev':
		oldRev = ( oldid || 0 ) + '&direction=prev'; newRev = oldid; break;
	case 'next':
		oldRev = oldid; newRev = oldid + '&direction=next';
		break;
	default:
		oldRev = oldid || 0; newRev = diff || 0; break;
	}
	oldRev = oldRev || 0;
	newRev = newRev || 0;

	var go = function() {
		pendingNavpopTask(navpop);
		getWiki(article, doneDiffNew, newRev, navpop);

		pendingNavpopTask(navpop);
		getWiki(article, doneDiffOld, oldRev, navpop);

		var tz = Cookie.read('popTz');
		if ( mw.config.get('wgEnableAPI') && getValueOf('popupAdjustDiffDates') && tz===null) {
			pendingNavpopTask(navpop);
			getPageWithCaching(pg.wiki.apiwikibase + '?format=json&action=query&meta=userinfo&uiprop=options',
					   function(d) {
						   completedNavpopTask(navpop);
						   setTimecorrectionCookie(d);
						   if (diffDownloadsComplete(navpop)) { insertDiff(navpop); }
					   },  navpop);
		}
		return true; // remove hook once run
	}
	if (navpop.visible || !getValueOf('popupLazyDownloads')) { go(); }
	else { navpop.addHook(go, 'unhide', 'before', 'DOWNLOAD_DIFFS'); }
}

function setTimecorrectionCookie(d) {
	try {
		var jsobj=getJsObj(d.data);
		var tz=jsobj.query.userinfo.options.timecorrection;
	} catch (someError) {
		logerr( 'setTimecorretion failed' );
		return;
	}
	Cookie.create( 'popTz', getTimeOffset(tz), 1);
}

function doneDiff(download, isOld) {
	if (!download.owner || !download.owner.diffData) { return; }
	var navpop=download.owner;
	var label= (isOld) ? 'Old' : 'New';
	var otherLabel=(isOld) ? 'New' : 'Old';
	navpop.diffData[label]=download;
	completedNavpopTask(download.owner);
	if (diffDownloadsComplete(navpop)) { insertDiff(navpop); }
}

function diffDownloadsComplete(navpop) {
	if ( Cookie.read('popTz')===null) { return false; }
	return navpop.diffData.Old && navpop.diffData.New;
}

function doneDiffNew(download) { doneDiff(download, false); }
function doneDiffOld(download) { doneDiff(download, true);  }

function rmBoringLines(a,b,context) {

	if (typeof context == 'undefined') { context=2; }
	// this is fairly slow... i think it's quicker than doing a word-based diff from the off, though
	var aa=[], aaa=[];
	var bb=[], bbb=[];
	var i, j;

	// first, gather all disconnected nodes in a and all crossing nodes in a and b
	for (i=0; i<a.length; ++i ) {
		if(!a[i].paired) { aa[i]=1; }
		else if (countCrossings(b,a,i, true)) {
			aa[i]=1;
			bb[ a[i].row ] = 1;
		}
	}

	// pick up remaining disconnected nodes in b
	for (i=0; i<b.length; ++i ) {
		if (bb[i]==1) { continue; }
		if(!b[i].paired) { bb[i]=1; }
	}

	// another pass to gather context: we want the neighbours of included nodes which are not yet included
	// we have to add in partners of these nodes, but we don't want to add context for *those* nodes in the next pass
	for (i=0; i<b.length; ++i) {
		if ( bb[i] == 1 ) {
			for (j=max(0,i-context); j < min(b.length, i+context); ++j) {
				if ( !bb[j] ) { bb[j] = 1; aa[ b[j].row ] = 0.5; }
			}
		}
	}

	for (i=0; i<a.length; ++i) {
		if ( aa[i] == 1 ) {
			for (j=max(0,i-context); j < min(a.length, i+context); ++j) {
				if ( !aa[j] ) { aa[j] = 1; bb[ a[j].row ] = 0.5; }
			}
		}
	}

	for (i=0; i<bb.length; ++i) {
		if (bb[i] > 0) { // it's a row we need
			if (b[i].paired) { bbb.push(b[i].text); } // joined; partner should be in aa
			else {
				bbb.push(b[i]);
			}
		}
	}
	for (i=0; i<aa.length; ++i) {
		if (aa[i] > 0) { // it's a row we need
			if (a[i].paired) { aaa.push(a[i].text); } // joined; partner should be in aa
			else {
				aaa.push(a[i]);
			}
		}
	}

	return { a: aaa, b: bbb};
}

function stripOuterCommonLines(a,b,context) {
	var i=0;
	while (i<a.length && i < b.length && a[i]==b[i]) { ++i; }
	var j=a.length-1; var k=b.length-1;
	while ( j>=0 && k>=0 && a[j]==b[k] ) { --j; --k; }

	return { a: a.slice(max(0,i - 1 - context), min(a.length+1, j + context+1)),
				b: b.slice(max(0,i - 1 - context), min(b.length+1, k + context+1)) };
}

function insertDiff(navpop) {
	// for speed reasons, we first do a line-based diff, discard stuff that seems boring, then do a word-based diff
	// FIXME: sometimes this gives misleading diffs as distant chunks are squashed together
	var oldlines=navpop.diffData.Old.data.split('\n');
	var newlines=navpop.diffData.New.data.split('\n');
	var inner=stripOuterCommonLines(oldlines,newlines,getValueOf('popupDiffContextLines'));
	oldlines=inner.a; newlines=inner.b;
	var truncated=false;
	getValueOf('popupDiffMaxLines');
	if (oldlines.length > pg.option.popupDiffMaxLines || newlines.length > pg.option.popupDiffMaxLines) {
		// truncate
		truncated=true;
		inner=stripOuterCommonLines(oldlines.slice(0,pg.option.popupDiffMaxLines),
						newlines.slice(0,pg.option.popupDiffMaxLines),
						pg.option.popupDiffContextLines);
		oldlines=inner.a; newlines=inner.b;
	}

	var lineDiff=diff(oldlines, newlines);
	var lines2=rmBoringLines(lineDiff.o, lineDiff.n);
	var oldlines2=lines2.a; var newlines2=lines2.b;

	var simpleSplit = !String.prototype.parenSplit.isNative;
	var html='<hr />';
	if (getValueOf('popupDiffDates')) {
		html += diffDatesTable(navpop.diffData.Old, navpop.diffData.New);
		html += '<hr />';
	}
	html += shortenDiffString(
		diffString(oldlines2.join('\n'), newlines2.join('\n'), simpleSplit),
		getValueOf('popupDiffContextCharacters') ).join('<hr />');
	setPopupTipsAndHTML(html.split('\n').join('<br>') +
			 (truncated ? '<hr /><b>'+popupString('Diff truncated for performance reasons')+'</b>' : '') ,
				'popupPreview', navpop.idNumber);
}

function diffDatesTable( oldDl, newDl ) {
	var html='<table class="popup_diff_dates">';
	html += diffDatesTableRow( newDl, tprintf('New revision'));
	html += diffDatesTableRow( oldDl, tprintf('Old revision'));
	html += '</table>';
	return html;
}
function diffDatesTableRow( dl, label ) {
	var txt='';
	if (!dl) {
		txt=popupString('Something went wrong :-(');
	} else if (!dl.lastModified) {
		txt= (/^\s*$/.test(dl.data)) ?
			popupString('Empty revision, maybe non-existent') : popupString('Unknown date');
	} else {
		var datePrint=getValueOf('popupDiffDatePrinter');
		if (typeof dl.lastModified[datePrint] == 'function') {
			if (getValueOf('popupAdjustDiffDates')) {
				var off;
				if (off=Cookie.read('popTz')) {
					var d2=adjustDate(dl.lastModified, off);
					txt = dayFormat(d2, true) + ' ' + timeFormat(d2, true);
				}
			} else {
				txt = dl.lastModified[datePrint]();
			}
		} else {
			txt = tprintf('Invalid %s %s', ['popupDiffDatePrinter', datePrint]);
		}
	}
	var revlink = generalLink({url: dl.url.replace(/&.*?(oldid=[0-9]+(?:&direction=[^&]*)?).*/, '&$1'),
				   text: label, title: label});
	return simplePrintf('<tr><td>%s</td><td>%s</td></tr>', [ revlink, txt ]);
}
//</NOLITE>
// ENDFILE: diffpreview.js
// STARTFILE: links.js
//<NOLITE>
/////////////////////
// LINK GENERATION //
/////////////////////

// titledDiffLink --> titledWikiLink --> generalLink
// wikiLink	   --> titledWikiLink --> generalLink
// editCounterLink --> generalLink

function titledDiffLink(l) { // article, text, title, from, to) {
	return titledWikiLink({article: l.article, action: l.to + '&oldid=' + l.from,
				newWin: l.newWin,
				noPopup: l.noPopup,
				text: l.text, title: l.title,
				/* hack: no oldid here */
				actionName: 'diff'});
}


function wikiLink(l) {
	//{article:article, action:action, text:text, oldid, newid}) {
	if (! (typeof l.article == typeof {}
		   && typeof l.action == typeof '' && typeof l.text==typeof '')) return null;
	if (typeof l.oldid == 'undefined') { l.oldid=null; }
	var savedOldid = l.oldid;
	if (!/^(edit|view|revert|render)$|^raw/.test(l.action)) { l.oldid=null; }
	var hint=popupString(l.action + 'Hint'); // revertHint etc etc etc
	var oldidData=[l.oldid, safeDecodeURI(l.article)];
	var revisionString = tprintf('revision %s of %s', oldidData);
	log('revisionString='+revisionString);
	switch (l.action) {
	case 'edit&section=new': hint = popupString('newSectionHint');  break;
	case 'edit&undo=':
		if (l.diff && l.diff != 'prev' && savedOldid ) {
		  l.action += l.diff + '&undoafter=' + savedOldid;
		} else if (savedOldid) {
		  l.action += savedOldid;
		}
		hint = popupString('undoHint');
		break;
	case 'raw&ctype=text/css': hint=popupString('rawHint'); break;
	case 'revert':
		if ( !mw.config.get('wgEnableAPI') ) {
			alert( 'This function of navigation popups now requires a MediaWiki ' + 
			'installation with the API enabled.');
			break;
		}
		var p=parseParams(pg.current.link.href);
		l.action='edit&autoclick=wpSave&actoken=' + autoClickToken() + '&autoimpl=' + popupString('autoedit_version') + '&autosummary=' + revertSummary(l.oldid, p.diff);
		if (p.diff=='prev') {
			l.action += '&direction=prev';
			revisionString = tprintf('the revision prior to revision %s of %s', oldidData);
		}
		if (getValueOf('popupRevertSummaryPrompt')) { l.action += '&autosummaryprompt=true'; }
		if (getValueOf('popupMinorReverts')) { l.action += '&autominor=true'; }
		log('revisionString is now '+revisionString);
		break;
	case 'nullEdit':
		l.action='edit&autoclick=wpSave&actoken=' + autoClickToken() + '&autoimpl=' + popupString('autoedit_version') + '&autosummary=null';
		break;
	case 'historyfeed':
		l.action='history&feed=rss';
		break;
	case 'markpatrolled':
		l.action='markpatrolled&rcid='+l.rcid;
	}

	if (hint) {
		if (l.oldid) {
			hint = simplePrintf(hint, [revisionString]);
		}
		else {
			hint = simplePrintf(hint, [safeDecodeURI(l.article)]);
		}
	}
	else {
		hint = safeDecodeURI(l.article + '&action=' + l.action) + (l.oldid) ? '&oldid='+l.oldid : '';
	}

	return titledWikiLink({article: l.article, action: l.action, text: l.text, newWin:l.newWin,
				title: hint, oldid: l.oldid, noPopup: l.noPopup, onclick: l.onclick});
}

function revertSummary(oldid, diff) {
	var ret='';
	if (diff == 'prev') {
		ret=getValueOf('popupQueriedRevertToPreviousSummary');
	} else { ret = getValueOf('popupQueriedRevertSummary'); }
	return ret + '&autorv=' + oldid;
}

function titledWikiLink(l) {
	// possible properties of argument:
	// article, action, text, title, oldid, actionName, className, noPopup
	// oldid = null is fine here

	// article and action are mandatory args

	if (typeof l.article == 'undefined' || typeof l.action=='undefined') {
		errlog('got undefined article or action in titledWikiLink');
		return null;
	}

	var base = pg.wiki.titlebase +  l.article.urlString();
	var url=base;

	if (typeof l.actionName=='undefined' || !l.actionName) { l.actionName='action'; }

	// no need to add &action=view, and this confuses anchors
	if (l.action != 'view') { url = base + '&' + l.actionName + '=' + l.action; }

	if (typeof l.oldid!='undefined' && l.oldid) { url+='&oldid='+l.oldid; }

	var cssClass=pg.misc.defaultNavlinkClassname;
	if (typeof l.className!='undefined' && l.className) { cssClass=l.className; }

	return generalNavLink({url: url, newWin: l.newWin,
				title: (typeof l.title != 'undefined') ? l.title : null,
				text: (typeof l.text!='undefined')?l.text:null,
				className: cssClass, noPopup:l.noPopup, onclick:l.onclick});
}

function getLastContrib(wikipage, newWin) {
	getHistoryInfo(wikipage, function(x){processLastContribInfo(x,{page: wikipage, newWin: newWin})});
}
function processLastContribInfo(info, stuff) {
	if(!info.edits || !info.edits.length) { alert('Popups: an odd thing happened. Please retry.'); return; }
	if(!info.firstNewEditor) {
		alert(tprintf('Only found one editor: %s made %s edits', [info.edits[0].editor,info.edits.length]));
		return;
	}
	var newUrl=pg.wiki.titlebase + new Title(stuff.page).urlString() + '&diff=cur&oldid='+info.firstNewEditor.oldid;
	displayUrl(newUrl, stuff.newWin);
}
function getDiffSinceMyEdit(wikipage, newWin) {
	getHistoryInfo(wikipage, function(x){processDiffSinceMyEdit(x,{page: wikipage, newWin: newWin})});
}
function processDiffSinceMyEdit(info, stuff) {
	if(!info.edits || !info.edits.length) { alert('Popups: something fishy happened. Please try again.'); return; }
	var friendlyName=stuff.page.split('_').join(' ');
	if(!info.myLastEdit) {
		alert(tprintf('Couldn\'t find an edit by %s\nin the last %s edits to\n%s',
				  [info.userName, getValueOf('popupHistoryLimit'), friendlyName]));
		return;
	}
	if(info.myLastEdit.index==0) {
		alert(tprintf("%s seems to be the last editor to the page %s", [info.userName, friendlyName]));
		return;
	}
	var newUrl=pg.wiki.titlebase + new Title(stuff.page).urlString() + '&diff=cur&oldid='+ info.myLastEdit.oldid;
	displayUrl(newUrl, stuff.newWin);
}
function displayUrl(url, newWin){
	if(newWin) { window.open(url); }
	else { document.location=url; }
}

function purgePopups() {
	processAllPopups(true);
	setupCache(); // deletes all cached items (not browser cached, though...)
	pg.option={};
	abortAllDownloads();
}

function processAllPopups(nullify, banish) {
	for (var i=0; pg.current.links && i<pg.current.links.length; ++i) {
		if (!pg.current.links[i].navpopup) { continue; }
		(nullify || banish) && pg.current.links[i].navpopup.banish();
		pg.current.links[i].simpleNoMore=false;
		nullify && (pg.current.links[i].navpopup=null);
	}
}

function disablePopups(){
	processAllPopups(false, true);
	setupTooltips(null, true);
}

function togglePreviews() {
	processAllPopups(true, true);
	pg.option.simplePopups=!pg.option.simplePopups;
	abortAllDownloads();
}

function magicWatchLink(l) {
	//Yuck!! Would require a thorough redesign to add this as a click event though ...
	l.onclick = simplePrintf( 'modifyWatchlist(\'%s\',\'%s\');return false;', [l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), this.id] );
	return wikiLink(l);
}

function modifyWatchlist(title, action) {
		var reqData = {
			'action': 'watch',
			'format': 'json',
			'title': title,
			'token': mw.user.tokens.get('watchToken'),
			'uselang': mw.config.get('wgUserLanguage')
		};
		if (action==='unwatch') reqData.unwatch = '';

		jQuery.ajax({
			url: mw.util.wikiScript('api'),
			dataType: 'json',
			type: 'POST',
			data: reqData,
			success: function( data, textStatus, xhr ) {
				mw.util.jsMessage( data.watch.message, 'watch' );
			}
		});
}

function magicHistoryLink(l) {
	// FIXME use onclick change href trick to sort this out instead of window.open

	var jsUrl='', title='';
	switch(l.id) {
	case 'lastContrib':
		jsUrl=simplePrintf('javascript:getLastContrib(\'%s\',%s)',
			[l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), l.newWin]);
		title=popupString('lastContribHint');
		break;
	case 'sinceMe':
		jsUrl=simplePrintf('javascript:getDiffSinceMyEdit(\'%s\',%s)',
			[l.article.toString(true).split("\\").join("\\\\").split("'").join("\\'"), l.newWin]);
		title=popupString('sinceMeHint');
		break;
	}

	return generalNavLink({url: jsUrl, newWin: false, // can't have new windows with JS links, I think
				title: title, text: l.text, noPopup: l.noPopup});
}

function popupMenuLink(l) {
	var jsUrl=simplePrintf('javascript:%s()', [l.id]);
	var title=popupString(simplePrintf('%sHint', [l.id]));
	return generalNavLink({url: jsUrl, newWin:false, title:title, text:l.text, noPopup:l.noPopup});
}

function specialLink(l) {
	// properties: article, specialpage, text, sep
	if (typeof l.specialpage=='undefined'||!l.specialpage) return null;
	var base = pg.wiki.titlebase +  mw.config.get('wgFormattedNamespaces')[pg.nsSpecialId]+':'+l.specialpage;
	if (typeof l.sep == 'undefined' || l.sep===null) l.sep='&target=';
	var article=l.article.urlString({keepSpaces: l.specialpage=='Search'});
	var hint=popupString(l.specialpage+'Hint');
	switch (l.specialpage) {
	case 'Log':
		switch (l.sep) {
		case '&user=': hint=popupString('userLogHint'); break;
		case '&type=block&page=': hint=popupString('blockLogHint'); break;
		case '&page=': hint=popupString('pageLogHint'); break;
		case '&type=protect&page=': hint=popupString('protectLogHint'); break;
		case '&type=delete&page=': hint=popupString('deleteLogHint'); break;
		default: log('Unknown log type, sep=' + l.sep); hint='Missing hint (FIXME)';
		}
		break;
	case 'PrefixIndex': article += '/'; break;
	}
	if (hint) hint = simplePrintf(hint, [safeDecodeURI(l.article)]);
	else hint = safeDecodeURI(l.specialpage+':'+l.article) ;

	var url = base + l.sep + article;
	return generalNavLink({url: url, title: hint, text: l.text, newWin:l.newWin, noPopup:l.noPopup});
}

function generalLink(l) {
	// l.url, l.text, l.title, l.newWin, l.className, l.noPopup, l.onclick
	if (typeof l.url=='undefined') return null;

	// only quotation marks in the url can screw us up now... I think
	var url=l.url.split('"').join('%22');

	var ret='<a href="' + url + '"';
	if (typeof l.title!='undefined' && l.title) { ret += ' title="' + pg.escapeQuotesHTML(l.title) + '"'; }
	if (typeof l.onclick!='undefined' && l.onclick) { ret += ' onclick="' + pg.escapeQuotesHTML(l.onclick) + '"'; }
	if (l.noPopup) { ret += ' noPopup=1'; }
	var newWin;
	if (typeof l.newWin=='undefined' || l.newWin===null) { newWin=getValueOf('popupNewWindows'); }
	else { newWin=l.newWin; }
	if (newWin) { ret += ' target="_blank"'; }
	if (typeof l.className!='undefined'&&l.className) { ret+=' class="'+l.className+'"'; }
	ret += '>';
	if (typeof l.text==typeof '') { ret+= l.text; }
	ret +='</a>';
	return ret;
}

function appendParamsToLink(linkstr, params) {
	var sp=linkstr.parenSplit(RegExp('(href="[^"]+?)"', 'i'));
	if (sp.length<2) return null;
	var ret=sp.shift() + sp.shift();
	ret += '&' + params + '"';
	ret += sp.join('');
	return ret;
}

function changeLinkTargetLink(x) { // newTarget, text, hint, summary, clickButton, minor, title (optional) {
	if (x.newTarget) {
		log ('changeLinkTargetLink: newTarget=' + x.newTarget);
	}
	if (x.oldTarget !== decodeURIComponent( x.oldTarget ) ) {
		log ('This might be an input problem: ' + x.oldTarget );
	}

	// FIXME: first character of page title as well as namespace should be case insensitive
	// eg [[category:foo]] and [[Category:Foo]] are equivalent
	// this'll break if charAt(0) is nasty
	var cA=literalizeRegex(x.oldTarget);
	var chs=cA.charAt(0).toUpperCase();
	chs='['+chs + chs.toLowerCase()+']';
	var currentArticleRegexBit=chs+cA.substring(1);
	currentArticleRegexBit=currentArticleRegexBit
		.split(RegExp('(?:[_ ]+|%20)', 'g')).join('(?:[_ ]+|%20)')
		.split('\\(').join('(?:%28|\\()')
		.split('\\)').join('(?:%29|\\))'); // why does this need to match encoded strings ? links in the document ?
	// leading and trailing space should be ignored, and anchor bits optional:
	currentArticleRegexBit = '\\s*(' + currentArticleRegexBit + '(?:#[^\\[\\|]*)?)\\s*';
	// e.g. Computer (archaic) -> \s*([Cc]omputer[_ ](?:%2528|\()archaic(?:%2528|\)))\s*

	// autoedit=s~\[\[([Cc]ad)\]\]~[[Computer-aided%20design|$1]]~g;s~\[\[([Cc]AD)[|]~[[Computer-aided%20design|~g

	var title=x.title || mw.config.get('wgPageName').split('_').join(' ');
	var lk=titledWikiLink({article: new Title(title), newWin:x.newWin,
						action:  'edit',
						text:	x.text,
						title:   x.hint,
						className: 'popup_change_title_link'
						});
	var cmd='';
	if (x.newTarget) {
		// escape '&' and other nasties
		var t=x.newTarget;
		var s=literalizeRegex(x.newTarget);
		cmd += 's~\\[\\['+currentArticleRegexBit+'\\]\\]~[['+t+'|$1]]~g;';
		cmd += 's~\\[\\['+currentArticleRegexBit+'[|]~[['+t+'|~g;';
		cmd += 's~\\[\\['+s + '\\|' + s + '\\]\\]~[[' + t + ']]~g';
	} else {
		cmd += 's~\\[\\['+currentArticleRegexBit+'\\]\\]~$1~g;';
		cmd += 's~\\[\\['+currentArticleRegexBit+'[|](.*?)\\]\\]~$2~g';
	}
	// Build query
	cmd = 'autoedit=' + encodeURIComponent ( cmd );
	cmd += '&autoclick='+ encodeURIComponent( x.clickButton ) + '&actoken=' + encodeURIComponent( autoClickToken() );
	cmd += ( x.minor == null ) ? '' : '&autominor='+ encodeURIComponent( x.minor );
	cmd += ( x.watch == null ) ? '' : '&autowatch='+ encodeURIComponent( x.watch );
	cmd += '&autosummary='+encodeURIComponent(x.summary);
	cmd += '&autoimpl='+encodeURIComponent( popupString('autoedit_version') );
	return appendParamsToLink(lk, cmd);
}


function redirLink(redirMatch, article) {
	// NB redirMatch is in wikiText
	var ret='';

	if (getValueOf('popupAppendRedirNavLinks') && getValueOf('popupNavLinks')) {
		ret += '<hr />';
		if (getValueOf('popupFixRedirs') && typeof autoEdit != 'undefined' && autoEdit) {
			log('redirLink: newTarget=' + redirMatch);
			ret += addPopupShortcut(
				changeLinkTargetLink(
				{newTarget: redirMatch, text: popupString('Redirects'),
					hint: popupString('Fix this redirect'),
						summary: simplePrintf(getValueOf('popupFixRedirsSummary'),
									  [article.toString(), redirMatch ]),
						oldTarget: article.toString(),
						clickButton: getValueOf('popupRedirAutoClick'), minor: true,
						watch: getValueOf('popupWatchRedirredPages')})
				, 'R');
			ret += popupString(' to ');
		}
		else ret += popupString('Redirects') + popupString(' to ');
		return ret;
	}

	else return '<br> ' + popupString('Redirects') + popupString(' to ') +
			 titledWikiLink({article: new Title().fromWikiText(redirMatch), action: 'view',  /* FIXME: newWin */
							  text: safeDecodeURI(redirMatch), title: popupString('Bypass redirect')});
}

function arinLink(l) {
	if (!saneLinkCheck(l)) { return null; }
	if ( ! l.article.isIpUser() || ! pg.wiki.wikimedia) return null;

	var uN=l.article.userName();

	return generalNavLink({url:'http://ws.arin.net/cgi-bin/whois.pl?queryinput=' + encodeURIComponent(uN), newWin:l.newWin,
				title: tprintf('Look up %s in ARIN whois database', [uN]),
				text: l.text, noPopup:1});
}

function toolDbName(cookieStyle) {
	var ret = mw.config.get('wgDBname');
	if (!cookieStyle) { ret+= '_p'; }
	return ret;
}

function saneLinkCheck(l) {
	if (typeof l.article != typeof {} || typeof l.text != typeof '') { return false; }
	return true;
}
function editCounterLink(l) {
	if(!saneLinkCheck(l)) return null;
	if (! pg.wiki.wikimedia) return null;
	var uN=l.article.userName();
	var tool=getValueOf('popupEditCounterTool');
	var url;
	var soxredToolUrl='//tools.wmflabs.org/supercount/index.php?user=$1&project=$2.$3';

	switch(tool) {
	case 'custom':
		url=simplePrintf(getValueOf('popupEditCounterUrl'), [ encodeURIComponent(uN), toolDbName() ]);
		break;
	case 'kate':
	case 'interiot':
	default:
		var theWiki=pg.wiki.hostname.split('.');
		url=simplePrintf(soxredToolUrl, [ encodeURIComponent(uN), theWiki[0], theWiki[1] ]);
	}
	return generalNavLink({url:url, title: tprintf('editCounterLinkHint', [uN]),
				newWin:l.newWin, text: l.text, noPopup:1});
}


function globalSearchLink(l) {
	if(!saneLinkCheck(l)) return null;

	var base='http://vs.aka-online.de/cgi-bin/globalwpsearch.pl?timeout=120&search=';
	var article=l.article.urlString({keepSpaces:true});

	return generalNavLink({url:base + article, newWin:l.newWin,
				title: tprintf('globalSearchHint', [safeDecodeURI(l.article)]),
				text: l.text, noPopup:1});
}

function googleLink(l) {
	if(!saneLinkCheck(l)) return null;

	var base='http://www.google.com/search?q=';
	var article=l.article.urlString({keepSpaces:true});

	return generalNavLink({url:base + '%22' + article + '%22', newWin:l.newWin,
				title: tprintf('googleSearchHint', [safeDecodeURI(l.article)]),
				text: l.text, noPopup:1});
}

function editorListLink(l) {
	if(!saneLinkCheck(l)) return null;
	var article= l.article.articleFromTalkPage() || l.article;
		var theWiki=pg.wiki.hostname.split('.');
		var base='//tools.wmflabs.org/xtools/articleinfo/index.php?&uselang=' + mw.config.get('wgUserLanguage') +
			'lang=' + theWiki[0] + '&wiki=' + theWiki[1] + '&begin=&end=&article='
	return generalNavLink({url:base+article.urlString(),
				title: tprintf('editorListHint', [article]),
				newWin:l.newWin, text: l.text, noPopup:1});
}

function generalNavLink(l) {
	l.className = (l.className==null) ? 'popupNavLink' : l.className;
	return generalLink(l);
}

//////////////////////////////////////////////////
// magic history links
//

function getHistoryInfo(wikipage, whatNext) {
	log('getHistoryInfo');
	getHistory(wikipage, whatNext ? function(d){whatNext(processHistory(d));} : processHistory);
}

// FIXME eliminate pg.idNumber ... how? :-(

function getHistory(wikipage, onComplete) {
	log('getHistory');
	if( !mw.config.get('wgEnableAPI') ) {
		alert( 'This function of navigation popups now requires a MediaWiki ' + 
			'installation with the API enabled.');
		return false;
	}
	var url = pg.wiki.apiwikibase + '?format=json&action=query&prop=revisions&titles=' +
			new Title(wikipage).urlString() + '&rvlimit=' + getValueOf('popupHistoryLimit');
	log('getHistory: url='+url);
	if (pg.flag.isIE) {
		url = url + '&*'; //to circumvent https://bugzilla.wikimedia.org/show_bug.cgi?id=28840
	}
	return startDownload(url, pg.idNumber+'history', onComplete);
}

function processHistory(download) {
	var jsobj = getJsObj(download.data);
	try {
		window.x=jsobj;
		var p=jsobj['query']['pages']
		for (var pageid in p) {
			var revisions=p[pageid]['revisions'];
			// we only get the first one
			break;
		}
	} catch (someError) {
		log('Something went wrong with JSON business');
		return finishProcessHistory([]);
	}
	var edits=[];
	for (var i=0; i<revisions.length; ++i) {
		edits.push({ oldid: revisions[i]['revid'], editor: revisions[i]['user'] });
	}
	log('processed ' + edits.length + ' edits');
	return finishProcessHistory( edits, mw.config.get('wgUserName') );
}


function finishProcessHistory(edits, userName) {
	var histInfo={};

	histInfo.edits=edits;
	histInfo.userName=userName;

	for (var i=0; i<edits.length; ++i) {
		if (typeof histInfo.myLastEdit == 'undefined' && userName && edits[i].editor==userName) {
			histInfo.myLastEdit={index: i, oldid: edits[i].oldid, previd: (i==0 ? null : edits[i-1].oldid)};
		}
		if (typeof histInfo.firstNewEditor == 'undefined' && edits[i].editor != edits[0].editor) {
			histInfo.firstNewEditor={index:i, oldid:edits[i].oldid, previd: (i==0 ? null : edits[i-1].oldid)};
		}
	}
	//pg.misc.historyInfo=histInfo;
	return histInfo;
}
//</NOLITE>
// ENDFILE: links.js
// STARTFILE: options.js
//////////////////////////////////////////////////
// options

// check for cookies and existing value, else use default
function defaultize(x) {
	var val=null;
	if (x!='popupCookies') {
		defaultize('popupCookies');
		if (pg.option.popupCookies && (val=Cookie.read(x))) {
			pg.option[x]=val;
			return;
		}
	}
	if (pg.option[x]===null || typeof pg.option[x]=='undefined') {
		if (typeof window[x] != 'undefined' ) pg.option[x]=window[x];
		else pg.option[x]=pg.optionDefault[x];
	}
}

function newOption(x, def) {
	pg.optionDefault[x]=def;
}

function setDefault(x, def) {
	return newOption(x, def);
}

function getValueOf(varName) {
	defaultize(varName);
	return pg.option[varName];
}

function useDefaultOptions() { // for testing
	for (var p in pg.optionDefault) {
		pg.option[p]=pg.optionDefault[p];
		if (typeof window[p]!='undefined') { delete window[p]; }
	}
}

function setOptions() {
	// user-settable parameters and defaults
	var userIsSysop = false;
	if ( mw.config.get('wgUserGroups') ) {
		for ( var g = 0; g < mw.config.get('wgUserGroups').length; ++g ) {
			if ( mw.config.get('wgUserGroups')[g] == "sysop" )
				userIsSysop = true
		}
	}

	// Basic options
	newOption('popupDelay',               0.5);
	newOption('popupHideDelay',           0.5);
	newOption('simplePopups',             false);
	newOption('popupStructure',           'shortmenus');   // see later - default for popupStructure is 'original' if simplePopups is true
	newOption('popupActionsMenu',         true);
	newOption('popupSetupMenu',           true);
	newOption('popupAdminLinks',          userIsSysop);
	newOption('popupShortcutKeys',        false);
	newOption('popupHistoricalLinks',     true);
	newOption('popupOnlyArticleLinks',    true);
	newOption('removeTitles',             true);
	newOption('popupMaxWidth',            350);
	newOption('popupInitialWidth',        false); // integer or false
	newOption('popupSimplifyMainLink',    true);
	newOption('popupAppendRedirNavLinks', true);
	newOption('popupTocLinks',            false);
	newOption('popupSubpopups',           true);
	newOption('popupDragHandle',          false /* 'popupTopLinks'*/);
	newOption('popupLazyPreviews',        true);
	newOption('popupLazyDownloads',       true);
	newOption('popupAllDabsStubs',        false);
	newOption('popupDebugging',           false);
	newOption('popupAdjustDiffDates',     true);
	newOption('popupActiveNavlinks',      true);
	newOption('popupModifier',            false); // ctrl, shift, alt or meta
	newOption('popupModifierAction',      'enable'); // or 'disable'
	newOption('popupDraggable',           true);

//<NOLITE>
	// images
	newOption('popupImages',                 true);
	newOption('imagePopupsForImages',        true);
	newOption('popupNeverGetThumbs',         false);
	//newOption('popupImagesToggleSize',       true);
	newOption('popupThumbAction',            'imagepage'); //'sizetoggle');
	newOption('popupImageSize',              60);
	newOption('popupImageSizeLarge',         200);

	// redirs, dabs, reversion
	newOption('popupFixRedirs',             false);
	newOption('popupRedirAutoClick',        'wpDiff');
	newOption('popupFixDabs',               false);
	newOption('popupDabsAutoClick',         'wpDiff');
	newOption('popupRevertSummaryPrompt',   false);
	newOption('popupMinorReverts',          false);
	newOption('popupRedlinkRemoval',        false);
	newOption('popupWatchDisambiggedPages', null);
	newOption('popupWatchRedirredPages',    null);
	newOption('popupDabWiktionary',         'last');

	// navlinks
	newOption('popupNavLinks',          true);
	newOption('popupNavLinkSeparator',  ' &sdot; ');
	newOption('popupLastEditLink',      true);
	newOption('popupEditCounterTool',   'soxred');
	newOption('popupEditCounterUrl',    '');
	newOption('popupExtraUserMenu',     '');
//</NOLITE>

	// previews etc
	newOption('popupPreviews',             true);
	newOption('popupSummaryData',          true);
	newOption('popupMaxPreviewSentences',  5);
	newOption('popupMaxPreviewCharacters', 600);
	newOption('popupLastModified',         true);
	newOption('popupPreviewKillTemplates', true);
	newOption('popupPreviewRawTemplates',  true);
	newOption('popupPreviewFirstParOnly',  true);
	newOption('popupPreviewCutHeadings',   true);
	newOption('popupPreviewButton',        false);
	newOption('popupPreviewButtonEvent',   'click');

//<NOLITE>
	// diffs
	newOption('popupPreviewDiffs',          true);
	newOption('popupDiffMaxLines',          100);
	newOption('popupDiffContextLines',      2);
	newOption('popupDiffContextCharacters', 40);
	newOption('popupDiffDates',             true);
	newOption('popupDiffDatePrinter',       'toLocaleString');

	// edit summaries. God, these are ugly.
	newOption('popupFixDabsSummary',           popupString('defaultpopupFixDabsSummary') );
	newOption('popupExtendedRevertSummary',    popupString('defaultpopupExtendedRevertSummary') );
	newOption('popupTimeOffset',               null);
	newOption('popupRevertSummary',            popupString('defaultpopupRevertSummary') );
	newOption('popupRevertToPreviousSummary',  popupString('defaultpopupRevertToPreviousSummary') );
	newOption('popupQueriedRevertSummary',            popupString('defaultpopupQueriedRevertSummary') );
	newOption('popupQueriedRevertToPreviousSummary',  popupString('defaultpopupQueriedRevertToPreviousSummary') );
	newOption('popupFixRedirsSummary',         popupString('defaultpopupFixRedirsSummary') );
	newOption('popupRedlinkSummary',           popupString('defaultpopupRedlinkSummary') );
	newOption('popupRmDabLinkSummary',         popupString('defaultpopupRmDabLinkSummary') );
//</NOLITE>
	// misc
	newOption('popupCookies',             false);
	newOption('popupHistoryLimit',        50);
//<NOLITE>
	newOption('popupFilters',             [popupFilterStubDetect,     popupFilterDisambigDetect,
					       popupFilterPageSize,       popupFilterCountLinks,
					       popupFilterCountImages,    popupFilterCountCategories,
					       popupFilterLastModified]);
	newOption('extraPopupFilters',        []);
	newOption('popupOnEditSelection', 'cursor');
	newOption('popupPreviewHistory',      true);
	newOption('popupImageLinks',          true);
	newOption('popupCategoryMembers',     true);
	newOption('popupUserInfo',            true);
	newOption('popupHistoryPreviewLimit', 25);
	newOption('popupContribsPreviewLimit',25);
	newOption('popupRevDelUrl',          '//en.wikipedia.org/wiki/Wikipedia:Revision_deletion');
//</NOLITE>

	// new windows
	newOption('popupNewWindows',     false);
	newOption('popupLinksNewWindow', {'lastContrib': true, 'sinceMe': true});

	// regexps
	newOption('popupDabRegexp', '(\\{\\{\\s*disambig(?!uation needed)|disambig(uation|)\\s*\\}\\}|disamb\\s*\\}\\}|dab\\s*\\}\\})|\\{\\{\\s*(((geo|hn|road?|school|number)dis)|[234][lc][acw]|(road|ship)index)(\\s*[|][^}]*)?\\s*[}][}]|is a .*disambiguation.*page');
	newOption('popupAnchorRegexp', 'anchors?'); //how to identify an anchors template
	newOption('popupStubRegexp', '(sect)?stub[}][}]|This .*-related article is a .*stub');
	newOption('popupImageVarsRegexp', 'image|image_(?:file|skyline|name|flag|seal)|cover|badge|logo');
}
// ENDFILE: options.js
// STARTFILE: strings.js
//<NOLITE>
//////////////////////////////////////////////////
// Translatable strings
//////////////////////////////////////////////////
//
// See instructions at
// http://en.wikipedia.org/wiki/Wikipedia:Tools/Navigation_popups/Translation

pg.string = {
	/////////////////////////////////////
	// summary data, searching etc.
	/////////////////////////////////////
	'article': 'article',
	'category': 'category',
	'categories': 'categories',
	'image': 'image',
	'images': 'images',
	'stub': 'stub',
	'section stub': 'section stub',
	'Empty page': 'Empty page',
	'kB': 'kB',
	'bytes': 'bytes',
	'day': 'day',
	'days': 'days',
	'hour': 'hour',
	'hours': 'hours',
	'minute': 'minute',
	'minutes': 'minutes',
	'second': 'second',
	'seconds': 'seconds',
	'week': 'week',
	'weeks': 'weeks',
	'search': 'search',
	'SearchHint': 'Find English Wikipedia articles containing %s',
	'web': 'web',
	'global': 'global',
	'globalSearchHint': 'Search across Wikipedias in different languages for %s',
	'googleSearchHint': 'Google for %s',
	/////////////////////////////////////
	// article-related actions and info
	// (some actions also apply to user pages)
	/////////////////////////////////////
	'actions': 'actions',		 ///// view articles and view talk
	'popupsMenu': 'popups',
	'togglePreviewsHint': 'Toggle preview generation in popups on this page',
	'enable previews': 'enable previews',
	'disable previews': 'disable previews',
	'toggle previews': 'toggle previews',
	'show preview': 'show preview',
	'reset': 'reset',
	'more...': 'more...',
	'disable': 'disable popups',
	'disablePopupsHint': 'Disable popups on this page. Reload page to re-enable.',
	'historyfeedHint': 'RSS feed of recent changes to this page',
	'purgePopupsHint': 'Reset popups, clearing all cached popup data.',
	'PopupsHint': 'Reset popups, clearing all cached popup data.',
	'spacebar': 'space',
	'view': 'view',
	'view article': 'view article',
	'viewHint': 'Go to %s',
	'talk': 'talk',
	'talk page': 'talk page',
	'this&nbsp;revision': 'this&nbsp;revision',
	'revision %s of %s': 'revision %s of %s',
	'Revision %s of %s': 'Revision %s of %s',
	'the revision prior to revision %s of %s': 'the revision prior to revision %s of %s',
	'Toggle image size': 'Click to toggle image size',
	'del': 'del',				 ///// delete, protect, move
	'delete': 'delete',
	'deleteHint': 'Delete %s',
	'undeleteShort': 'un',
	'UndeleteHint': 'Show the deletion history for %s',
	'protect': 'protect',
	'protectHint': 'Restrict editing rights to %s',
	'unprotectShort': 'un',
	'unprotectHint': 'Allow %s to be edited by anyone again',
	'move': 'move',
	'move page': 'move page',
	'MovepageHint': 'Change the title of %s',
	'edit': 'edit',			   ///// edit articles and talk
	'edit article': 'edit article',
	'editHint': 'Change the content of %s',
	'edit talk': 'edit talk',
	'new': 'new',
	'new topic': 'new topic',
	'newSectionHint': 'Start a new section on %s',
	'null edit': 'null edit',
	'nullEditHint': 'Submit an edit to %s, making no changes ',
	'hist': 'hist',			   ///// history, diffs, editors, related
	'history': 'history',
	'historyHint': 'List the changes made to %s',
	'last': 'last',
	'lastEdit': 'lastEdit',
	'mark patrolled': 'mark patrolled',
	'markpatrolledHint': 'Mark this edit as patrolled',
	'show last edit': 'most recent edit',
	'Show the last edit': 'Show the effects of the most recent change',
	'lastContrib': 'lastContrib',
	'last set of edits': 'latest edits',
	'lastContribHint': 'Show the net effect of changes made by the last editor',
	'cur': 'cur',
	'diffCur': 'diffCur',
	'Show changes since revision %s': 'Show changes since revision %s',
	'%s old': '%s old', // as in 4 weeks old
	'oldEdit': 'oldEdit',
	'purge': 'purge',
	'purgeHint': 'Demand a fresh copy of %s',
	'raw': 'source',
	'rawHint': 'Download the source of %s',
	'render': 'simple',
	'renderHint': 'Show a plain HTML version of %s',
	'Show the edit made to get revision': 'Show the edit made to get revision',
	'sinceMe': 'sinceMe',
	'changes since mine': 'diff my edit',
	'sinceMeHint': 'Show changes since my last edit',
	'Couldn\'t find an edit by %s\nin the last %s edits to\n%s': 'Couldn\'t find an edit by %s\nin the last %s edits to\n%s',
	'eds': 'eds',
	'editors': 'editors',
	'editorListHint': 'List the users who have edited %s',
	'related': 'related',
	'relatedChanges': 'relatedChanges',
	'related changes': 'related changes',
	'RecentchangeslinkedHint': 'Show changes in articles related to %s',
	'editOld': 'editOld',		  ///// edit old version, or revert
	'rv': 'rv',
	'revert': 'revert',
	'revertHint': 'Revert to %s',
	'defaultpopupRedlinkSummary': 'Removing link to empty page [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupFixDabsSummary': 'Disambiguate [[%s]] to [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupFixRedirsSummary': 'Redirect bypass from [[%s]] to [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupExtendedRevertSummary': 'Revert to revision dated %s by %s, oldid %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupRevertToPreviousSummary': 'Revert to the revision prior to revision %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupRevertSummary': 'Revert to revision %s using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupQueriedRevertToPreviousSummary': 'Revert to the revision prior to revision $1 dated $2 by $3 using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupQueriedRevertSummary': 'Revert to revision $1 dated $2 by $3 using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'defaultpopupRmDabLinkSummary': 'Remove link to dab page [[%s]] using [[:en:Wikipedia:Tools/Navigation_popups|popups]]',
	'Redirects': 'Redirects', // as in Redirects to ...
	' to ': ' to ',		   // as in Redirects to ...
	'Bypass redirect': 'Bypass redirect',
	'Fix this redirect': 'Fix this redirect',
	'disambig': 'disambig',		  ///// add or remove dab etc.
	'disambigHint': 'Disambiguate this link to [[%s]]',
	'Click to disambiguate this link to:': 'Click to disambiguate this link to:',
	'remove this link': 'remove this link',
	'remove all links to this page from this article': 'remove all links to this page from this article',
	'remove all links to this disambig page from this article': 'remove all links to this disambig page from this article',
	'mainlink': 'mainlink',		  ///// links, watch, unwatch
	'wikiLink': 'wikiLink',
	'wikiLinks': 'wikiLinks',
	'links here': 'links here',
	'whatLinksHere': 'whatLinksHere',
	'what links here': 'what links here',
	'WhatlinkshereHint': 'List the pages that are hyperlinked to %s',
	'unwatchShort': 'un',
	'watchThingy': 'watch',  // called watchThingy because {}.watch is a function
	'watchHint': 'Add %s to my watchlist',
	'unwatchHint': 'Remove %s from my watchlist',
	'Only found one editor: %s made %s edits': 'Only found one editor: %s made %s edits',
	'%s seems to be the last editor to the page %s': '%s seems to be the last editor to the page %s',
	'rss': 'rss',
	/////////////////////////////////////
	// diff previews
	/////////////////////////////////////
	'Diff truncated for performance reasons': 'Diff truncated for performance reasons',
	'Old revision': 'Old revision',
	'New revision': 'New revision',
	'Something went wrong :-(': 'Something went wrong :-(',
	'Empty revision, maybe non-existent': 'Empty revision, maybe non-existent',
	'Unknown date': 'Unknown date',
	/////////////////////////////////////
	// other special previews
	/////////////////////////////////////
	'Empty category': 'Empty category',
	'Category members (%s shown)': 'Category members (%s shown)',
	'No image links found': 'No image links found',
	'File links': 'File links',
	'No image found': 'No image found',
	'Image from Commons': 'Image from Commons',
	'Description page': 'Description page',
	'Alt text:': 'Alt text:',
	'revdel':'Hidden revision',
	/////////////////////////////////////
	// user-related actions and info
	/////////////////////////////////////
	'user': 'user',			   ///// user page, talk, email, space
	'user&nbsp;page': 'user&nbsp;page',
	'user talk': 'user talk',
	'edit user talk': 'edit user talk',
	'leave comment': 'leave comment',
	'email': 'email',
	'email user': 'email user',
	'EmailuserHint': 'Send an email to %s',
	'space': 'space', // short form for userSpace link
	'PrefixIndexHint': 'Show pages in the userspace of %s',
	'count': 'count',			 ///// contributions, log
	'edit counter': 'edit counter',
	'editCounterLinkHint': 'Count the contributions made by %s',
	'contribs': 'contribs',
	'contributions': 'contributions',
	'deletedContribs': 'deleted contributions',
	'DeletedcontributionsHint': 'List deleted edits made by %s',
	'ContributionsHint': 'List the contributions made by %s',
	'log': 'log',
	'user log': 'user log',
	'userLogHint': 'Show %s\'s user log',
	'arin': 'ARIN lookup',			 ///// ARIN lookup, block user or IP
	'Look up %s in ARIN whois database': 'Look up %s in the ARIN whois database',
	'unblockShort': 'un',
	'block': 'block',
	'block user': 'block user',
	'IpblocklistHint': 'Unblock %s',
	'BlockipHint': 'Prevent %s from editing',
	'block log': 'block log',
	'blockLogHint': 'Show the block log for %s',
	'protectLogHint': 'Show the protection log for %s',
	'pageLogHint': 'Show the page log for %s',
	'deleteLogHint': 'Show the deletion log for %s',
	'Invalid %s %s': 'The option %s is invalid: %s',
	'No backlinks found': 'No backlinks found',
	' and more': ' and more',
	'undo': 'undo',
	'undoHint': 'undo this edit',
	'Download preview data': 'Download preview data',
	'Invalid or IP user': 'Invalid or IP user',
	'Not a registered username': 'Not a registered username',
	'BLOCKED': 'BLOCKED',
	' edits since: ': ' edits since: ',
	/////////////////////////////////////
	// Autoediting
	/////////////////////////////////////
	'Enter a non-empty edit summary or press cancel to abort': 'Enter a non-empty edit summary or press cancel to abort',
	'Failed to get revision information, please edit manually.\n\n': 'Failed to get revision information, please edit manually.\n\n',
	'The %s button has been automatically clicked. Please wait for the next page to load.': 'The %s button has been automatically clicked. Please wait for the next page to load.',
	'Could not find button %s. Please check the settings in your javascript file.': 'Could not find button %s. Please check the settings in your javascript file.',
	/////////////////////////////////////
	// Popups setup
	/////////////////////////////////////
	'Open full-size image': 'Open full-size image',
	'zxy': 'zxy',
	'autoedit_version': 'np20140416'
};


function popupString(str) {
	if (typeof popupStrings != 'undefined' && popupStrings && popupStrings[str]) { return popupStrings[str]; }
	if (pg.string[str]) { return pg.string[str]; }
	return str;
}


function tprintf(str,subs) {
	if (typeof subs != typeof []) { subs = [subs]; }
	return simplePrintf(popupString(str), subs);
}

//</NOLITE>
// ENDFILE: strings.js


////////////////////////////////////////////////////////////////////
// Run things
////////////////////////////////////////////////////////////////////


// For some reason popups requires a fully loaded page jQuery.ready(...) causes problems for some.
// The old addOnloadHook did something similar to the below
if (document.readyState=="complete")
	autoEdit(); //will setup popups
else
	$( window ).on( 'load', autoEdit );


// Support for MediaWiki's live preview, VisualEditor's saves and Echo's flyout.
( function () {
	var once = true;
	function dynamicContentHandler( $content ) {
		// Try to detect the hook fired on initial page load and disregard
		// it, we already hook to onload (possibly to different parts of
		// page - it's configurable) and running twice might be bad. Ugly…
		if ( $content.attr( 'id' ) == 'mw-content-text' ) {
			if ( once ) {
				once = false;
				return;
			}
		}

		function doIt () {
			$content.each( function () {
				this.ranSetupTooltipsAlready = false;
				setupTooltips( this );
			} );
		};

		if ( !setupPopups.completed ) {
			setupPopups( doIt );
		} else {
			doIt();
		}
	}

	// This hook is also fired after page load.
	mw.hook( 'wikipage.content' ).add( dynamicContentHandler );

	mw.hook( 'ext.echo.overlay.beforeShowingOverlay' ).add( function($overlay){
		dynamicContentHandler( $overlay.find(".mw-echo-state") );
	});
} )();