Welcome to the DFO World Wiki. With many major updates since the release of DFO, many items are missing. Visit Item Database Project to learn more.
Please remember to click "show preview" before saving the page.
Thanks for the updated logo snafuPop!

Difference between revisions of "MediaWiki:Common.js"

From DFO World Wiki
Jump to: navigation, search
m
m
Line 497: Line 497:
 
}
 
}
 
});
 
});
//
+
/* Ajax batch delete thingy, version [0.1.1b]
/* http://dev.wikia.com/wiki/AjaxBatchDelete/code.2.js
+
Originally from: http://en.wikipedia.org/wiki/User:Splarka/ajaxbatchdelete.js
* Ajax Batch Delete V2
+
* @description Delete listed multiple pages
+
Notes:
* Based on and faster than the original
+
* It is a bit verbose, after debugging perhaps some output should be removed.
* http://dev.wikia.com/wiki/AjaxBatchDelete
+
* It waits 1 second after every delete before starting the next.
* Does not need to go to Special:BlankPage to use
+
* Can be aborted by simply deleting the contents of the textarea, or leaving the page.
* Includes the option to protect after deleting
+
* Stops when it hits a blank line.
* Includes the option to grab a whole category's contents
+
* Nonfatal errors (skip to next line):
* @author Ozuzanna
+
** Bad character or malformed line
 +
** Bad token
 +
** Unexpected response
 +
* Pauses in execution can be added with a blank line.
 +
 +
To do:
 +
* Cache the token if two the same?
 +
** Please note the delete token is not guaranteed to be static, but currently it always is.
 
*/
 
*/
 
   
 
   
;(function($, mw) {
+
addOnloadHook(function() {
 +
  addPortletLink('p-tb','/wiki/Special:BlankPage?blankspecial=ajaxbd','Batch Delete');
 +
});
 
   
 
   
if ($('#t-bd').length)
+
if(wgCanonicalSpecialPageName && wgCanonicalSpecialPageName.toLowerCase() == 'blankpage' && queryString('blankspecial') == 'ajaxbd') {
   return;
+
  document.title = 'Ajax Batch Deletion';
 +
   addOnloadHook(abdForm);
 +
}
 
   
 
   
var ug = mw.config.get("wgUserGroups").join(' ');
+
function abdForm() {
if (ug.indexOf('sysop') >= 0) {
+
  addPortletLink('p-tb','/wiki/Special:Log/delete?user=' + encodeURIComponent(wgUserName),'My delete log');
 
   
 
   
   var FormHTML = '\
+
   //subvert this Special: page to our own needs.
  <form method="" name="" class="WikiaForm "> \
+
  var con = document.getElementById('content') || document.getElementById('mw_content');
    <fieldset> \
+
  var bcon = document.getElementById('bodyContent') || document.getElementById('mw_contentholder');
      <p>Reason for deleting: \
+
   var fh = getElementsByClassName(con,'h1','firstHeading')[0];
        <input type="text" id="delete-reason" value="" /> \
+
   while(fh.firstChild) fh.removeChild(fh.firstChild)
        <br/> \
+
  fh.appendChild(document.createTextNode('Ajax Batch Deletion'));
        <label for="protect-check">Protect for admin only? <input type="checkbox" id="protect-check" /></label> \
+
   for(var i=0;i<bcon.childNodes.length;i++) {
      </p> \
+
    bcur = bcon.childNodes[i];
      <p>Put the name of each page you want to delete on a <b>separate line</b></p>. \
+
    if(bcur.id != 'siteSub' && bcur.id != 'contentSub' && bcur.className != 'visualClear') {
        <textarea style="height: 20em; width: 80%;" id="text-mass-delete"/> \
+
      while(bcur.firstChild) bcur.removeChild(bcur.firstChild)
<div id="text-error-output" style="height:10em; width: 80%; margin: 5px auto 0px auto; color: #000; background-color: #ffbfbf; height: 150px; border: 1px solid black; font-weight: bold; overflow: scroll">Any errors encountered will appear below<br/></div> \
+
      if(bcur.nodeType == 3) bcur.nodeValue = '';
    </fieldset> \
+
    }
   </form>',
 
   token = mw.user.tokens.get('editToken'),
 
   delay = window.batchDeleteDelay || 1000;
 
 
  //Support for Monobook
 
  if (mw.config.get('skin') === 'monobook') {
 
   
 
  }
 
  else {
 
  mw.util.addPortletLink('p-tb', '#', 'Batch Delete', 't-bd');
 
 
   }
 
   }
 
   
 
   
   $('#t-bd').click(function () {
+
   //generate content
    $.showCustomModal('Ajax Batch Delete', FormHTML, {
+
  var form = document.createElement('form');
      id: 'form-mass-delete',
+
  form.appendChild(document.createTextNode('List of pages to delete:'));
      width: 500,
+
  form.appendChild(document.createElement('p'));
      buttons: [{ 
+
  form.setAttribute('action','javascript:void(0);');
          message: 'Cancel',
+
  var txt = document.createElement('textarea');
          handler: function() {
+
    txt.style.height = '20em';
            $('#form-mass-delete').closeModal();
+
    txt.style.width = '50%';
          }
+
    txt.setAttribute('id','abd-textarea');
      }, {
+
  form.appendChild(txt);
          message: 'Add category contents',
+
  form.appendChild(document.createElement('p'));
          defaultButton: true,
+
  var lab1 = document.createElement('label');
          handler: function() {
+
    lab1.setAttribute('for','abd-reason')
            addCategoryContents();
+
    lab1.appendChild(document.createTextNode('Deletion reason: '));
          }
+
  form.appendChild(lab1);
      }, {
+
  var inp1 = document.createElement('input');
          id: 'startButton',
+
    inp1.style.width = '20em';
          message: 'Initiate',
+
    inp1.setAttribute('type','text');
          defaultButton: true,
+
    inp1.setAttribute('id','abd-reason');
          handler: function () {
+
  form.appendChild(inp1);
            init();  
+
  form.appendChild(document.createElement('p'));
          }
+
  var sub1 = document.createElement('input');
      }]
+
    sub1.setAttribute('type','button');
    });
+
    sub1.setAttribute('id','abd-startbutton');
   });
+
    sub1.setAttribute('value','start');
 +
    sub1.setAttribute('onclick','abdStart()');
 +
  form.appendChild(sub1);
 +
  bcon.appendChild(form);
 +
  var pre = document.createElement('pre');
 +
  pre.setAttribute('id','abd-output');
 +
   bcon.appendChild(pre);
 +
}
 
   
 
   
    function init() {
+
function abdStart() {
      var txt = document.getElementById('text-mass-delete'),
+
  document.getElementById('abd-startbutton').setAttribute('disabled','disabled');
      deleteReason = document.getElementById('delete-reason').value,
+
  var out = document.getElementById('abd-output');
       pages = txt.value.split('\n'),
+
  var txt = document.getElementById('abd-textarea');
       currentPage = pages[0];
+
  var deletes = txt.value.split('\n');
 +
  var page = deletes[0];
 +
  if(page == '') {
 +
    out.appendChild(document.createTextNode('* Done! Nothing left to do, or next line is blank.\n'));
 +
    document.getElementById('abd-startbutton').removeAttribute('disabled');
 +
  } else {
 +
    var badchars = /(\#|\<|\>|\[|\]|\{|\}|\|)/;
 +
    if(badchars.test(page)) {
 +
       out.appendChild(document.createTextNode('! Illegal characters detected, skipping:' + page + '\n'));
 +
      setTimeout('abdStart()',1000);
 +
    } else {
 +
       out.appendChild(document.createTextNode('> Attempting to delete [[' + page + ']]\n'));
 +
      abdGetToken(page);
 +
    }
 +
  }
 +
  deletes = deletes.slice(1,deletes.length);
 +
  txt.value = deletes.join('\n');
 +
}
 
   
 
   
      if (!deleteReason) {
+
function abdGetToken(page) {
        alert('Please state a reason!');
+
  var out = document.getElementById('abd-output');
        return;
+
  out.appendChild(document.createTextNode(' > Fetching delete token for [[' + page + ']]\n'));
      }
+
  var url = wgScriptPath + '/api.php?action=query&prop=info&indexpageids=1&intoken=delete&format=json&titles=' + encodeURIComponent(page);
 +
  var req = sajax_init_object();
 +
  req.open('GET', url, true);
 +
  req.onreadystatechange = function() {
 +
    if(req.readyState == 4 && req.status == 200) {
 +
      eval("abdDelete(" + req.responseText + ",'" + req.responseText.replace(/\'/g,"`") + "','" + page + "')");
 +
    }
 +
  }
 +
  req.send(null);
 +
}
 
   
 
   
      document.getElementById('startButton').setAttribute('disabled','disabled');
+
function abdDelete(obj,txt,page) {
 +
  var out = document.getElementById('abd-output');
 +
  if(obj['error']) {
 +
    out.appendChild(document.createTextNode(' ! Api error: ' + obj['error']['code'] + ' - ' + obj['error']['info'] + '\n'));
 +
    return;
 +
  }
 +
  if(!obj['query'] || !obj['query']['pageids'] || !obj['query']['pages'][obj['query']['pageids'][0]] || !obj['query']['pages'][obj['query']['pageids'][0]]['deletetoken']) {
 +
    out.appendChild(document.createTextNode(' ? Unexpected response: ' + txt + '\n'));
 +
    return;
 +
  }
 +
  var token = obj['query']['pages'][obj['query']['pageids'][0]]['deletetoken'];
 +
  out.appendChild(document.createTextNode('  > Token found, attempting delete\n'));
 +
  var reason = document.getElementById('abd-reason').value;
 
   
 
   
      if (!currentPage) {
+
  var params = 'action=delete&format=json&token=' + encodeURIComponent(token) + '&title=' + encodeURIComponent(page) + '&reason=' + encodeURIComponent(reason);
        document.getElementById('startButton').removeAttribute("disabled");
+
  var url = wgScriptPath + '/api.php';
        $.showCustomModal('Finished!', 'Nothing left to do, or next line is blank.', {
 
          id: 'mass-delete-complete',
 
          width: 200,
 
          buttons: [{
 
              message: 'Close',
 
              defaultButton: true,
 
              handler: function() {
 
                $('#mass-delete-complete').closeModal();
 
              }
 
          }]
 
        });
 
      }
 
      else {
 
              process(currentPage,deleteReason); 
 
      }
 
      pages = pages.slice(1,pages.length);
 
      txt.value = pages.join('\n');
 
  }
 
 
   
 
   
    function addCategoryContents() {
+
  var req = sajax_init_object();
      var category = prompt('Please enter the category name (no category prefix):');
+
  req.open('POST', url, true);
      new mw.Api().get({
+
  req.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
      action: 'query',
+
  req.setRequestHeader('Content-length', params.length);
      list: 'categorymembers',
+
  req.setRequestHeader('Connection', 'close');
      cmtitle: "Category:"+category,
+
  req.onreadystatechange = function() {
      cmlimit: 5000
+
    if(req.readyState == 4 && req.status == 200) {
      })
+
      eval("abdDeleteAftermath(" + req.responseText + ",'" + req.responseText.replace(/\'/g,"`") + "')");
      .done(function(d) {
+
    }
        if (!d.error) {
+
  }
          var data = d.query;
+
  req.send(params);
 +
}
 
   
 
   
  for (var i in data.categorymembers) {
+
function abdDeleteAftermath(obj,txt) {
            $('#text-mass-delete').append(data.categorymembers[i].title+'\n');
+
  var out = document.getElementById('abd-output');
          }
+
  if(obj['error']) {
        }
+
    out.appendChild(document.createTextNode('  ! Api error: ' + obj['error']['code'] + ' - ' + obj['error']['info'] + '\n'));
        else {
+
  } else if(obj['delete'] && obj['delete']['title']) {
          $('#text-error-output').append('Failed to get contents of '+ category +' : '+ d.error.code +'<br/>');
+
    out.appendChild(document.createTextNode('   > Page [[' + obj['delete']['title'] + ']] deleted\n'));
        }
+
  } else {
      })
+
    out.appendChild(document.createTextNode('   ? Unexpected response: ' + txt + '\n'));
      .fail(function() {
+
    return;
        $('#text-error-output').append('Failed to get contents of '+ category +'!<br/>');
+
  }
      });
+
  setTimeout('abdStart()',1000);
    }  
+
}
 
   
 
   
    function process(page,reason) {
+
function queryString(p) {
      new mw.Api().post({
+
  var re = RegExp('[&?]' + p + '=([^&]*)');
      format: 'json',
+
  var matches;
      action: 'delete',
+
  if (matches = re.exec(document.location)) {
      watchlist: 'nochange',
+
    try {  
      title: page,
+
      return decodeURI(matches[1]);
      reason: reason,
+
    } catch (e) {
      token: token
 
      })
 
      .done(function(d) {
 
        if (!d.error) {
 
          console.log('Deletion of '+page+' successful!');
 
          if (document.getElementById('protect-check').checked) {
 
            new mw.Api().post({
 
            format: 'json',
 
            action: 'protect',
 
            expiry: 'infinite',
 
            protections: 'create=sysop',
 
            watchlist: 'nochange',
 
            title: page,
 
            reason: reason,
 
            token: token
 
            })
 
            .done(function(d) {
 
              if (!d.error) {
 
                console.log('Protection of '+page+' successful!');
 
              }
 
              else {
 
                console.log('Failed to protect '+page+': '+ d.error.code);
 
        $('#text-error-output').append('Failed to protect '+page+': '+d.error.code+'<br/>');
 
              }
 
            })
 
            .fail(function() {
 
              console.log('Failed to protect '+page+': unknownerror');
 
      $('#text-error-output').append('Failed to protect '+page+': unknownerror<br/>');
 
            });
 
          }
 
        }
 
else {
 
          console.log('Failed to delete '+page+': '+ d.error.code);
 
          $('#text-error-output').append('Failed to delete '+page+': '+d.error.code+'<br/>');
 
        }
 
      })
 
      .fail(function() {
 
        console.log('Failed to delete '+page+': unknownerror');
 
        $('#text-error-output').append('Failed to delete '+page+': unknownerror<br/>');
 
      });
 
      setTimeout(init,delay);
 
 
     }
 
     }
 
   }
 
   }
}) (this.jQuery, this.mediaWiki);
+
  return null;
//
+
}

Revision as of 07:59, 1 March 2015

/** Collapsible tables *********************************************************
  *
  *  Description: Allows tables to be collapsed, showing only the header. See
  *               [[Wikipedia:NavFrame]].
  *  Maintainers: [[User:R. Koot]]
  */
 
 var autoCollapse = 2;
 var collapseCaption = "hide";
 var expandCaption = "show";
 
 function collapseTable( tableIndex )
 {
     var Button = document.getElementById( "collapseButton" + tableIndex );
     var Table = document.getElementById( "collapsibleTable" + tableIndex );
 
     if ( !Table || !Button ) {
         return false;
     }
 
     var Rows = Table.rows;
 
     if ( Button.firstChild.data == collapseCaption ) {
         for ( var i = 1; i < Rows.length; i++ ) {
             Rows[i].style.display = "none";
         }
         Button.firstChild.data = expandCaption;
     } else {
         for ( var i = 1; i < Rows.length; i++ ) {
             Rows[i].style.display = Rows[0].style.display;
         }
         Button.firstChild.data = collapseCaption;
     }
 }
 
 function createCollapseButtons()
 {
     var tableIndex = 0;
     var NavigationBoxes = new Object();
     var Tables = document.getElementsByTagName( "table" );
 
     for ( var i = 0; i < Tables.length; i++ ) {
         if ( hasClass( Tables[i], "collapsible" ) ) {
 
             /* only add button and increment count if there is a header row to work with */
             var HeaderRow = Tables[i].getElementsByTagName( "tr" )[0];
             if (!HeaderRow) continue;
             var Header = HeaderRow.getElementsByTagName( "th" )[0];
             if (!Header) continue;
 
             NavigationBoxes[ tableIndex ] = Tables[i];
             Tables[i].setAttribute( "id", "collapsibleTable" + tableIndex );
 
             var Button     = document.createElement( "span" );
             var ButtonLink = document.createElement( "a" );
             var ButtonText = document.createTextNode( collapseCaption );
 
             Button.style.styleFloat = "right";
             Button.style.cssFloat = "right";
             Button.style.fontWeight = "normal";
             Button.style.textAlign = "right";
             Button.style.width = "6em";
 
             ButtonLink.style.color = Header.style.color;
             ButtonLink.setAttribute( "id", "collapseButton" + tableIndex );
             ButtonLink.setAttribute( "href", "javascript:collapseTable(" + tableIndex + ");" );
             ButtonLink.appendChild( ButtonText );
 
             Button.appendChild( document.createTextNode( "[" ) );
             Button.appendChild( ButtonLink );
             Button.appendChild( document.createTextNode( "]" ) );
 
             Header.insertBefore( Button, Header.childNodes[0] );
             tableIndex++;
         }
     }
 
     for ( var i = 0;  i < tableIndex; i++ ) {
         if ( hasClass( NavigationBoxes[i], "collapsed" ) || ( tableIndex >= autoCollapse && hasClass( NavigationBoxes[i], "autocollapse" ) ) ) {
             collapseTable( i );
         }
     }
 }
 
 addOnloadHook( createCollapseButtons );
 
 /** Dynamic Navigation Bars (experimental) *************************************
  *
  *  Description: See [[Wikipedia:NavFrame]].
  *  Maintainers: UNMAINTAINED
  */
 
  // set up the words in your language
  var NavigationBarHide = '[' + collapseCaption + ']';
  var NavigationBarShow = '[' + expandCaption + ']';
 
  // shows and hides content and picture (if available) of navigation bars
  // Parameters:
  //     indexNavigationBar: the index of navigation bar to be toggled
  function toggleNavigationBar(indexNavigationBar)
  {
     var NavToggle = document.getElementById("NavToggle" + indexNavigationBar);
     var NavFrame = document.getElementById("NavFrame" + indexNavigationBar);
 
     if (!NavFrame || !NavToggle) {
         return false;
     }
 
     // if shown now
     if (NavToggle.firstChild.data == NavigationBarHide) {
         for (
                 var NavChild = NavFrame.firstChild;
                 NavChild != null;
                 NavChild = NavChild.nextSibling
             ) {
             if ( hasClass( NavChild, 'NavPic' ) ) {
                 NavChild.style.display = 'none';
             }
             if ( hasClass( NavChild, 'NavContent') ) {
                 NavChild.style.display = 'none';
             }
         }
     NavToggle.firstChild.data = NavigationBarShow;
 
     // if hidden now
     } else if (NavToggle.firstChild.data == NavigationBarShow) {
         for (
                 var NavChild = NavFrame.firstChild;
                 NavChild != null;
                 NavChild = NavChild.nextSibling
             ) {
             if (hasClass(NavChild, 'NavPic')) {
                 NavChild.style.display = 'block';
             }
             if (hasClass(NavChild, 'NavContent')) {
                 NavChild.style.display = 'block';
             }
         }
     NavToggle.firstChild.data = NavigationBarHide;
     }
  }
 
  // adds show/hide-button to navigation bars
  function createNavigationBarToggleButton()
  {
     var indexNavigationBar = 0;
     // iterate over all < div >-elements 
     var divs = document.getElementsByTagName("div");
     for(
             var i=0; 
             NavFrame = divs[i]; 
             i++
         ) {
         // if found a navigation bar
         if (hasClass(NavFrame, "NavFrame")) {
 
             indexNavigationBar++;
             var NavToggle = document.createElement("a");
             NavToggle.className = 'NavToggle';
             NavToggle.setAttribute('id', 'NavToggle' + indexNavigationBar);
             NavToggle.setAttribute('href', 'javascript:toggleNavigationBar(' + indexNavigationBar + ');');
 
             var NavToggleText = document.createTextNode(NavigationBarHide);
             for (
                  var NavChild = NavFrame.firstChild;
                  NavChild != null;
                  NavChild = NavChild.nextSibling
                 ) {
                 if ( hasClass( NavChild, 'NavPic' ) || hasClass( NavChild, 'NavContent' ) ) {
                     if (NavChild.style.display == 'none') {
                         NavToggleText = document.createTextNode(NavigationBarShow);
                         break;
                     }
                 }
             }
 
             NavToggle.appendChild(NavToggleText);
             // Find the NavHead and attach the toggle link (Must be this complicated because Moz's firstChild handling is borked)
             for(
               var j=0; 
               j < NavFrame.childNodes.length; 
               j++
             ) {
               if (hasClass(NavFrame.childNodes[j], "NavHead")) {
                 NavFrame.childNodes[j].appendChild(NavToggle);
               }
             }
             NavFrame.setAttribute('id', 'NavFrame' + indexNavigationBar);
         }
     }
  }
 
  addOnloadHook( createNavigationBarToggleButton );
 
 
/* Test if an element has a certain class **************************************
  *
  * Description: Uses regular expressions and caching for better performance.
  * Maintainers: [[User:Mike Dillon]], [[User:R. Koot]], [[User:SG]]
  */
 
 var hasClass = (function () {
     var reCache = {};
     return function (element, className) {
         return (reCache[className] ? reCache[className] : (reCache[className] = new RegExp("(?:\\s|^)" + className + "(?:\\s|$)"))).test(element.className);
     };
 })();

/* Mouse-over boxes (Work-in-progress) */
function initmouseoverbox()
{
	var affectedSpans = new Array();
	var affectedSpansi = 0;
	var spans = document.getElementsByTagName("span");
	for (i = 0; i < spans.length; i++)
	{
		if (spans[i].getAttribute("class") == null)
			continue;
		var classes = spans[i].getAttribute("class").split(" ");
		for (c = 0; c < classes.length; c++)
			if (classes[c] == "mouseover-wrapper")
			{
				affectedSpans[affectedSpansi] = i;
				affectedSpansi++;
				break;
			}
	}

	var content = null;
	var mouseover = null;
	var innerDivs = null;
	for (i = 0; i < affectedSpansi; i++)
	{
		innerDivs = spans[affectedSpans[i]].getElementsByTagName("div");
		for (c = 0; c < innerDivs.length; c++)
			if (innerDivs[c].getAttribute("id") == "mouseover-content")
				content = innerDivs[c];
			else if (innerDivs[c].getAttribute("id") == "mouseover-trigger")
				trigger = innerDivs[c];
		content.style.display = "none";

		/* Replace <p>s inside the trigger with <br />s */
		var ps = trigger.getElementsByTagName("p");
		var anchor = trigger.getElementsByTagName("a");
		var parent = null;
		var innerHTML = null;
		var e1 = null;
		var e2 = null;
		for (d = 0; d < anchor.length; d++)
		{
			anchor[d].setAttribute("title", "");
		}
		for (c = 0; c < ps.length; c++)
		{
			var parent = ps[c].parentNode;
			var innerHTML = ps[c].innerHTML;

			e1 = document.createElement("br");
			parent.replaceChild(e1, ps[c]);
			e2 = document.createElement("br");
			e2.setAttribute("class", "mouseover-p");
			parent.insertBefore(e2, e1.nextSibling);
			e1 = document.createTextNode(innerHTML);
			parent.insertBefore(e1, e2.nextSibling);
			e2 = document.createElement("br");
			parent.insertBefore(e2, e1.nextSibling);
			e1 = document.createElement("br");
			e1.setAttribute("class", "mouseover-p");
			parent.insertBefore(e1, e2.nextSibling);
		}

		trigger.setAttribute("onmouseover", "mouseoverbox(" + i + ", this, event);");
		trigger.setAttribute("onmousemove", "mouseoverbox(" + i + ", this, event, true);");
		trigger.setAttribute("onmouseout", "mouseoverbox(" + i + ", this);");
	}
}
addOnloadHook(initmouseoverbox);

var mouseoverboxes = new Array();
function mouseoverFade(id, fadelength, lastTick) // Had to make it global becuase of setTimeout
{
	if (lastTick !== undefined)
	{
		var curTick = new Date().getTime();
		var elapsedTicks = curTick - lastTick;

		if (mouseoverboxes[id].fadeState == 1 && mouseoverboxes[id].style.display != "block")
			mouseoverboxes[id].style.display = "block";

		if (mouseoverboxes[id].fadeTimeLeft <= elapsedTicks)
		{
			mouseoverboxes[id].style.opacity = mouseoverboxes[id].fadeState == 1 ? 1 : 0;
			if (mouseoverboxes[id].fadeState == -1)
				mouseoverboxes[id].style.display = "none";
			mouseoverboxes[id].fadeState = mouseoverboxes[id].fadeState == 1 ? 2 : -2;
			return;
		}

		mouseoverboxes[id].fadeTimeLeft -= elapsedTicks;
		var newOpVal = mouseoverboxes[id].fadeTimeLeft/fadelength;
		if (mouseoverboxes[id].fadeState == 1)
			newOpVal = 1 - newOpVal;

		mouseoverboxes[id].style.opacity = newOpVal;

		setTimeout("mouseoverFade(" + id + ", " + fadelength + ", " + curTick + ")", 20);
		return;
	}
	if (mouseoverboxes[id].fadeState == undefined)
		if (mouseoverboxes[id].style.opacity == undefined || mouseoverboxes[id].style.opacity == "" || mouseoverboxes[id].style.opacity == 1)
			mouseoverboxes[id].fadeState = 2;
		else
			mouseoverboxes[id].fadeState = -2;

	if (mouseoverboxes[id].fadeState == 1 || mouseoverboxes[id].fadeState == -1)
	{
		mouseoverboxes[id].fadeState = mouseoverboxes[id].fadeState == 1 ? -1 : 1;
		mouseoverboxes[id].fadeTimeLeft = fadelength - mouseoverboxes[id].fadeTimeLeft;
	}
	else
	{
		mouseoverboxes[id].fadeState = mouseoverboxes[id].fadeState == 2 ? -1 : 1;
		mouseoverboxes[id].fadeTimeLeft = fadelength;
		setTimeout("mouseoverFade(" + id + ", " + fadelength + ", " + new Date().getTime() + ")", 20);
	}  
}
function mouseoverbox(id, obj, event, update)
{
	function updatexy(id, event)
	{
		var x, y;
		switch (mouseoverboxes[id].settings["anchor"])
		{
			case 1:
				y = (-mouseoverboxes[id].offsetHeight) + (mouseoverboxes[id].settings["mody"] == 0 ? -10 : mouseoverboxes[id].settings["mody"]);
				x = (-mouseoverboxes[id].offsetWidth) + (mouseoverboxes[id].settings["modx"] == 0 ? -10 : mouseoverboxes[id].settings["modx"]);
				break;
			case 2:
				y = (-mouseoverboxes[id].offsetHeight) + (mouseoverboxes[id].settings["mody"] == 0 ? -10 : mouseoverboxes[id].settings["mody"]);
				x = (-(mouseoverboxes[id].offsetWidth/2)) + mouseoverboxes[id].settings["modx"];
				break;
			case 3:
				y = (-mouseoverboxes[id].offsetHeight) + (mouseoverboxes[id].settings["mody"] == 0 ? -10 : mouseoverboxes[id].settings["mody"]);
				x = mouseoverboxes[id].settings["modx"] == 0 ? 10 : mouseoverboxes[id].settings["modx"];
				break;
			case 4:
				y = (-(mouseoverboxes[id].offsetHeight/2)) + mouseoverboxes[id].settings["mody"];
				x = (-mouseoverboxes[id].offsetWidth) + (mouseoverboxes[id].settings["modx"] == 0 ? -10 : mouseoverboxes[id].settings["modx"]);
				break;
			case 5:
				y = (-(mouseoverboxes[id].offsetHeight/2)) + mouseoverboxes[id].settings["mody"];
				x = (-(mouseoverboxes[id].offsetWidth/2)) + mouseoverboxes[id].settings["modx"];
				break;
			case 6:
				y = (-(mouseoverboxes[id].offsetHeight/2)) + mouseoverboxes[id].settings["mody"];
				x = mouseoverboxes[id].settings["modx"] == 0 ? 10 : mouseoverboxes[id].settings["modx"];
				break;
			case 7:
				y = mouseoverboxes[id].settings["mody"] == 0 ? 10 : mouseoverboxes[id].settings["mody"];
				x = (-mouseoverboxes[id].offsetWidth) + (mouseoverboxes[id].settings["modx"] == 0 ? -10 : mouseoverboxes[id].settings["modx"]);
				break;
			case 8:
				y = mouseoverboxes[id].settings["mody"] == 0 ? 10 : mouseoverboxes[id].settings["mody"];
				x = (-(mouseoverboxes[id].offsetWidth/2)) + mouseoverboxes[id].settings["modx"];
				break;
			case 9:
			default:
				y = mouseoverboxes[id].settings["mody"] == 0 ? 20 : mouseoverboxes[id].settings["mody"];
				x = mouseoverboxes[id].settings["modx"] == 0 ? -1 : mouseoverboxes[id].settings["modx"];
				break;
		}

		mouseoverboxes[id].style.top = (event.clientY + y) + "px";
		mouseoverboxes[id].style.left = (event.clientX + x) + "px";
	}

	if (mouseoverboxes[id] === undefined)
	{
		mouseoverboxes[id] = document.createElement("div");

		var settings = new Array();
		/* Defaults */
		settings["style"] = "";
		settings["anchor"] = 9;
		settings["modx"] = 0;
		settings["mody"] = 0;
		settings["fade-length"] = 200.0;

		var divs = obj.parentNode.getElementsByTagName("div");
		for (i = 0; i < divs.length; i++)
			if (divs[i].getAttribute("id") == "mouseover-custom")
			{
				var spans = divs[i].getElementsByTagName("span");
				for (c = 0; c < spans.length; c++)
					switch (spans[c].getAttribute("id"))
					{
						case "mouseover-custom-style":
							if (spans[c].innerHTML != "")
								settings["style"] = spans[c].innerHTML;
							break;
						case "mouseover-custom-anchor":
							if (spans[c].innerHTML != "")
								settings["anchor"] = parseInt(spans[c].innerHTML);
							break;
						case "mouseover-custom-modx":
							if (spans[c].innerHTML != "")
								settings["modx"] = parseInt(spans[c].innerHTML);
							break;
						case "mouseover-custom-mody":
							if (spans[c].innerHTML != "")
								settings["mody"] = parseInt(spans[c].innerHTML);
							break;
						case "mouseover-custom-fade-length":
							if (spans[c].innerHTML != "")
								settings["fade-length"] = parseInt(spans[c].innerHTML);
							break;
					}
				break;
			}
			else if (divs[i].getAttribute("id") == "mouseover-content")
				mouseoverboxes[id].innerHTML = divs[i].innerHTML;

		mouseoverboxes[id].setAttribute("class", "mouseover-box");
		mouseoverboxes[id].style.cssText = settings["style"];
		mouseoverboxes[id].style.display = "none";
		mouseoverboxes[id].style.position = "fixed";
		mouseoverboxes[id].style.opacity = 0;
		mouseoverboxes[id].settings = settings;
		document.getElementById("bodyContent").appendChild(mouseoverboxes[id]);
	}

	if (event !== undefined)
	{
		updatexy(id, event);
		if (mouseoverboxes[id].style.display != "block")
			mouseoverboxes[id].style.display = "block";
		if (update == true)
			return;
	}

	mouseoverFade(id, mouseoverboxes[id].settings["fade-length"]);
}

/** 
 * DuplicateFiles
 * Looks for Duplicate Files.
 *
 * Originally by Pcj
 * http://community.wikia.com/wiki/Forum:Search_for_duplicate_files
 * 
 * Requires jQuery 
 *
 * Create a page with "<div id="mw-dupimages"></div>" and it will generate the duplicate files. 
 */


function findDupImages(gf) {
	$('#dupImagesProgress').show();
	var indicator = stylepath + '/common/progress-wheel.gif',
	dil = [],
	output = "",
	url = "/api.php?action=query&generator=allimages&prop=duplicatefiles&gailimit=500&format=json";
 
	if (!($('#dupImagesProgress').length)) {
		$("#mw-dupimages").prepend('<span style="float: right;" id="dupImagesProgress" title="In progress..."><img src="' + indicator + '" style="vertical-align: baseline;" border="0" alt="In progress..." /></span>');
	}
 
	if (gf) {
		url += "&gaifrom=" + gf;
	}
 
	$.getJSON(url, function (data) {
		if (data.query) {
			var pages = data.query.pages;
			for (pageID in pages) {
				dils = "," + dil.join();
				if (dils.indexOf("," + pages[pageID].title) == -1 && pages[pageID].title.indexOf("File::") == -1 && pages[pageID].duplicatefiles) {
					output += "<h3><a href='/wiki/" + pages[pageID].title + "'>" + pages[pageID].title + "</a></h3>\n<ul>\n";
					for (var x = 0; x < pages[pageID].duplicatefiles.length; x++) {
						output += "<li><a href='/wiki/File:" + pages[pageID].duplicatefiles[x].name + "'>File:" + pages[pageID].duplicatefiles[x].name + "</a></li>\n";
						dil.push("File:" + pages[pageID].duplicatefiles[x].name.replace(/_/g, " "));
					}
					output += "</ul>\n\n"
				}
			}
			$("#mw-dupimages").append(output);
			$('#dupImagesProgress').hide();
 
			if (data["query-continue"]) setTimeout("findDupImages('" + encodeURIComponent(data["query-continue"].allimages.gaifrom).replace(/'/g, "%27") + "');", 2000);
		}
	});
}
 
$(function () {
	if ($("#mw-dupimages").length) {
		findDupImages();
	}
});
/* Ajax batch delete thingy, version [0.1.1b]
Originally from: http://en.wikipedia.org/wiki/User:Splarka/ajaxbatchdelete.js
 
Notes:
* It is a bit verbose, after debugging perhaps some output should be removed.
* It waits 1 second after every delete before starting the next.
* Can be aborted by simply deleting the contents of the textarea, or leaving the page.
* Stops when it hits a blank line.
* Nonfatal errors (skip to next line):
** Bad character or malformed line
** Bad token
** Unexpected response
* Pauses in execution can be added with a blank line.
 
To do:
* Cache the token if two the same?
** Please note the delete token is not guaranteed to be static, but currently it always is.
*/
 
addOnloadHook(function() {
  addPortletLink('p-tb','/wiki/Special:BlankPage?blankspecial=ajaxbd','Batch Delete');
});
 
if(wgCanonicalSpecialPageName && wgCanonicalSpecialPageName.toLowerCase() == 'blankpage' && queryString('blankspecial') == 'ajaxbd') {
  document.title = 'Ajax Batch Deletion';
  addOnloadHook(abdForm);
}
 
function abdForm() {
  addPortletLink('p-tb','/wiki/Special:Log/delete?user=' + encodeURIComponent(wgUserName),'My delete log');
 
  //subvert this Special: page to our own needs.
  var con = document.getElementById('content') || document.getElementById('mw_content');
  var bcon = document.getElementById('bodyContent') || document.getElementById('mw_contentholder');
  var fh = getElementsByClassName(con,'h1','firstHeading')[0];
  while(fh.firstChild) fh.removeChild(fh.firstChild)
  fh.appendChild(document.createTextNode('Ajax Batch Deletion'));
  for(var i=0;i<bcon.childNodes.length;i++) {
    bcur = bcon.childNodes[i];
    if(bcur.id != 'siteSub' && bcur.id != 'contentSub' && bcur.className != 'visualClear') {
      while(bcur.firstChild) bcur.removeChild(bcur.firstChild)
      if(bcur.nodeType == 3) bcur.nodeValue = '';
    }
  }
 
  //generate content
  var form = document.createElement('form');
   form.appendChild(document.createTextNode('List of pages to delete:'));
   form.appendChild(document.createElement('p'));
   form.setAttribute('action','javascript:void(0);');
   var txt = document.createElement('textarea');
    txt.style.height = '20em';
    txt.style.width = '50%';
    txt.setAttribute('id','abd-textarea');
   form.appendChild(txt);
   form.appendChild(document.createElement('p'));
   var lab1 = document.createElement('label');
    lab1.setAttribute('for','abd-reason')
    lab1.appendChild(document.createTextNode('Deletion reason: '));
   form.appendChild(lab1);
   var inp1 = document.createElement('input');
    inp1.style.width = '20em';
    inp1.setAttribute('type','text');
    inp1.setAttribute('id','abd-reason');
   form.appendChild(inp1);
   form.appendChild(document.createElement('p'));
   var sub1 = document.createElement('input');
    sub1.setAttribute('type','button');
    sub1.setAttribute('id','abd-startbutton');
    sub1.setAttribute('value','start');
    sub1.setAttribute('onclick','abdStart()');
   form.appendChild(sub1);
  bcon.appendChild(form);
  var pre = document.createElement('pre');
   pre.setAttribute('id','abd-output');
  bcon.appendChild(pre);
}
 
function abdStart() {
  document.getElementById('abd-startbutton').setAttribute('disabled','disabled');
  var out = document.getElementById('abd-output');
  var txt = document.getElementById('abd-textarea');
  var deletes = txt.value.split('\n');
  var page = deletes[0];
  if(page == '') {
    out.appendChild(document.createTextNode('* Done! Nothing left to do, or next line is blank.\n'));
    document.getElementById('abd-startbutton').removeAttribute('disabled');
  } else {
    var badchars = /(\#|\<|\>|\[|\]|\{|\}|\|)/;
    if(badchars.test(page)) {
      out.appendChild(document.createTextNode('! Illegal characters detected, skipping:' + page + '\n'));
      setTimeout('abdStart()',1000);
    } else {
      out.appendChild(document.createTextNode('> Attempting to delete [[' + page + ']]\n'));
      abdGetToken(page);
    }
  }
  deletes = deletes.slice(1,deletes.length);
  txt.value = deletes.join('\n');
}
 
function abdGetToken(page) {
  var out = document.getElementById('abd-output');
  out.appendChild(document.createTextNode(' > Fetching delete token for [[' + page + ']]\n'));
  var url = wgScriptPath + '/api.php?action=query&prop=info&indexpageids=1&intoken=delete&format=json&titles=' + encodeURIComponent(page);
  var req = sajax_init_object();
  req.open('GET', url, true);
  req.onreadystatechange = function() {
    if(req.readyState == 4 && req.status == 200) {
      eval("abdDelete(" + req.responseText + ",'" + req.responseText.replace(/\'/g,"`") + "','" + page + "')");
    }
  }
  req.send(null);
}
 
function abdDelete(obj,txt,page) {
  var out = document.getElementById('abd-output');
  if(obj['error']) {
    out.appendChild(document.createTextNode(' ! Api error: ' + obj['error']['code'] + ' - ' + obj['error']['info'] + '\n'));
    return;
  }
  if(!obj['query'] || !obj['query']['pageids'] || !obj['query']['pages'][obj['query']['pageids'][0]] || !obj['query']['pages'][obj['query']['pageids'][0]]['deletetoken']) {
    out.appendChild(document.createTextNode('  ? Unexpected response: ' + txt + '\n'));
    return;
  }
  var token = obj['query']['pages'][obj['query']['pageids'][0]]['deletetoken'];
  out.appendChild(document.createTextNode('  > Token found, attempting delete\n'));
  var reason = document.getElementById('abd-reason').value;
 
  var params = 'action=delete&format=json&token=' + encodeURIComponent(token) + '&title=' + encodeURIComponent(page) + '&reason=' + encodeURIComponent(reason);
  var url = wgScriptPath + '/api.php';
 
  var req = sajax_init_object();
  req.open('POST', url, true);
  req.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
  req.setRequestHeader('Content-length', params.length);
  req.setRequestHeader('Connection', 'close');
  req.onreadystatechange = function() {
    if(req.readyState == 4 && req.status == 200) {
      eval("abdDeleteAftermath(" + req.responseText + ",'" + req.responseText.replace(/\'/g,"`") + "')");
    }
  }
  req.send(params);
}
 
function abdDeleteAftermath(obj,txt) {
  var out = document.getElementById('abd-output');
  if(obj['error']) {
    out.appendChild(document.createTextNode('   ! Api error: ' + obj['error']['code'] + ' - ' + obj['error']['info'] + '\n'));
  } else if(obj['delete'] && obj['delete']['title']) {
    out.appendChild(document.createTextNode('   > Page [[' + obj['delete']['title'] + ']] deleted\n'));
  } else {
    out.appendChild(document.createTextNode('   ? Unexpected response: ' + txt + '\n'));
    return;
  }
  setTimeout('abdStart()',1000);
}
 
function queryString(p) {
  var re = RegExp('[&?]' + p + '=([^&]*)');
  var matches;
  if (matches = re.exec(document.location)) {
    try { 
      return decodeURI(matches[1]);
    } catch (e) {
    }
  }
  return null;
}