MediaWiki:Common.js

Fram Wikiwordbōc

Note: After publishing, you may have to bypass your browser's cache to see the changes.

  • Firefox / Safari: Hold Shift while clicking Reload, or press either Ctrl-F5 or Ctrl-R (⌘-R on a Mac)
  • Google Chrome: Press Ctrl-Shift-R (⌘-Shift-R on a Mac)
  • Internet Explorer / Edge: Hold Ctrl while clicking Refresh, or press Ctrl-F5
  • Opera: Press Ctrl-F5.
// Conversion stuff
mw.loader.load('//ang.wikipedia.org/w/index.php?title=User:PiRSquared17/spellingstyle.js&action=raw&ctype=text/javascript');

// Pillaged code from en.wiktionary.org/wiki/MediaWiki:Gadget-legacy.js that will hopefully get us nice things
/*</pre>
=== DOM creation ===
<pre>*/
/**
 * Create a new DOM node for the current document.
 *    Basic usage:  var mySpan = newNode('span', "Hello World!")
 *    Supports attributes and event handlers*: var mySpan = newNode('span', {style:"color: red", focus: function(){alert(this)}, id:"hello"}, "World, Hello!")
 *    Also allows nesting to create trees: var myPar = newNode('p', newNode('b',{style:"color: blue"},"Hello"), mySpan)
 *
 * *event handlers, there are some issues with IE6 not registering event handlers on some nodes that are not yet attached to the DOM,
 * it may be safer to add event handlers later manually.
**/
var newNode = window.newNode = function newNode(tagname) {
	var node = document.createElement(tagname);

	for (var i = 1; i < arguments.length; ++i){
		if (typeof arguments[i] === 'string') { // text
			node.appendChild(document.createTextNode(arguments[i]));
		} else if (typeof arguments[i] === 'object') {
			if (arguments[i].nodeName) { //If it is a DOM Node
				node.appendChild(arguments[i]);
			} else { // Attributes (hopefully)
				for (var j in arguments[i]){
					if (j === 'class') { //Classname different because...
						node.className = arguments[i][j];
					} else if (j === 'style') { //Style is special
						node.style.cssText = arguments[i][j];
					} else if (typeof arguments[i][j] === 'function') { //Basic event handlers
						newNode.addEventHandler(node, j, arguments[i][j]);
					} else {
						node.setAttribute(j, arguments[i][j]); //Normal attributes
					}
				}
			}
		}
	}

	node.addEventHandler = function(eventName, handler) {
		newNode.addEventHandler(this, eventName, handler);
	};

	return node;
};

newNode.addEventHandler = function(node, eventName, handler) {
	try{ node.addEventListener(eventName,handler,false); //W3C
	}catch(e){try{ node.attachEvent('on'+eventName,handler,"Language"); //MSIE
	}catch(e){ node['on'+eventName]=handler; }} //Legacy
};


/*</pre>
== Visibility toggling ==
<pre>*/
var VisibilityToggles = window.VisibilityToggles = {
	// toggles[category] = [[show, hide],...]; statuses[category] = [true, false,...]; buttons = <li>
	toggles: {}, statuses: {}, buttons: null,

	// Add a new toggle, adds a Show/Hide category button in the toolbar,
	// and will call show_function and hide_function once on register, and every alternate click.
	register: function (category, show_function, hide_function) {
		var id = 0;
		if (!this.toggles[category]) {
			this.toggles[category] = [];
			this.statuses[category] = [];
		} else {
			id = this.toggles[category].length;
		}
		this.toggles[category].push([show_function, hide_function]);
		this.statuses[category].push(this.currentStatus(category));
		this.addGlobalToggle(category);

		(this.statuses[category][id] ? show_function : hide_function)();

		return function () {
			var statuses = VisibilityToggles.statuses[category];
			statuses[id] = !statuses[id];
			VisibilityToggles.checkGlobalToggle(category);
			return (statuses[id] ? show_function : hide_function)();
		};
	},

	// Add a new global toggle to the side bar
	addGlobalToggle: function(category) {
		if (document.getElementById('p-visibility-'+category))
			return;
		if (!this.buttons) {
			this.buttons = newNode('ul');
			var collapsed = jQuery.cookie("vector-nav-p-visibility") === "false",
				toolbox = newNode('div', {'class': "portal portlet "+(collapsed?"collapsed":"expanded"), 'id': 'p-visibility'},
					newNode('h3', 'Visibility'),
					newNode('div', {'class': "pBody body"}, collapsed ? void(0) : {'style':'display:block;'}, this.buttons)
				);
			var sidebar = document.getElementById('mw-panel') || document.getElementById('column-one');
			var insert = null;
			if ((insert = (document.getElementById('p-lang') || document.getElementById('p-feedback'))))
				sidebar.insertBefore(toolbox, insert);
			else
				sidebar.appendChild(toolbox);
		}
		var status = this.currentStatus(category);
		var newToggle = newNode('li', newNode('a', {
			id: 'p-visibility-' + category, 
			style: 'cursor: pointer',
			href: '#visibility-' + category,
			click: function(e)
			{
				VisibilityToggles.toggleGlobal(category); 
				if (e && e.preventDefault)
					e.preventDefault();
				else 
					window.event.returnValue = false;
				return false; 
			}},
			(status ? 'Hȳdan ' : 'Ēowan ') + category));
		for (var i = 0; i < this.buttons.childNodes.length; i++) {
			if (this.buttons.childNodes[i].id < newToggle.id) {
				this.buttons.insertBefore(newToggle, this.buttons.childNodes[i]);
				return;
			}
		}
		this.buttons.appendChild(newToggle);
	},

	// Update the toggle-all buttons when all things are toggled one way
	checkGlobalToggle: function(category) {
		var statuses = this.statuses[category];
		var status = statuses[0];
		for (var i = 1; i < statuses.length; i++) {
			if (status != statuses[i])
				return;
		}
		document.getElementById('p-visibility-' + category).innerHTML = (status ? 'Hȳdan ' : 'Ēowan ') + category;
	},

	// Toggle all un-toggled elements when the global button is clicked
	toggleGlobal: function(category) {
		var status = document.getElementById('p-visibility-' + category).innerHTML.indexOf('Ēowan ') === 0;
		for (var i = 0; i < this.toggles[category].length; i++ ) {
			if (this.statuses[category][i] != status) {
				this.toggles[category][i][status ? 0 : 1]();
				this.statuses[category][i] = status;
			}
		}
		document.getElementById('p-visibility-' + category).innerHTML = (status ? 'Hȳdan ' : 'Ēowan ') + category;
		var current = jQuery.cookie('Visibility');
		if (!current)
			current = ";";
		current = current.replace(';' + category + ';', ';');
		if (status)
			current = current + category + ";";
		window.setCookie('Visibility', current);
	},

	currentStatus: function(category) {
		if (location.hash.toLowerCase().split('_')[0] == '#' + category.toLowerCase())
			return true;
		if (location.href.search(/[?](.*&)?hidecats=/) > 0)
		{
			var hidecats = location.href;
			hidecats = hidecats.replace(/^[^?]+[?]((?!hidecats=)[^&]*&)*hidecats=/, '');
			hidecats = hidecats.replace(/&.*/, '');
			hidecats = hidecats.split(',');
			for (var i = 0; i < hidecats.length; ++i)
				if (hidecats[i] == category || hidecats[i] == 'all')
					return false;
				else if (hidecats[i] == '!' + category || hidecats[i] == 'none')
					return true;
		}
		if (jQuery.cookie('WiktionaryPreferencesShowNav') == 'true')
			return true;
		if ((jQuery.cookie('Visibility') || "").indexOf(';' + category + ';') >= 0)
			return true;
		// TODO check category-specific cookies
		return false;
	}
};

/*</pre>
=== NavBars ===
<pre>*/
var NavigationBarHide = 'hȳdan ▲';
var NavigationBarShow = 'ēowan ▼';

function NavToggleCategory(navFrame)
{
	var table = navFrame.getElementsByTagName('table')[0];
	if (table && table.className == "translations")
		return "translations";

	var heading = navFrame.previousSibling;
	while (heading) {
		if (/[hH][4-6]/.test(heading.nodeName)) {
			if (heading.getElementsByTagName('span')[1])
				heading = heading.getElementsByTagName('span')[0];
			return jQuery(heading).text().toLowerCase()
				// jQuery's .text() is inconsistent about whitespace:
				.replace(/^\s+|\s+$/g, '').replace(/\s+/g, ' ')
				// remove numbers added by the "Auto-number headings" pref:
				.replace(/^[1-9][0-9.]+ ?/, '');
		}
		else if (/[hH][1-3]/.test(heading.nodeName)) 
			break;
		heading = heading.previousSibling;
	}
	return "other boxes";
}

function createNavToggle(navFrame){
	var navHead, navToggle, navContent;
	for (var j=0; j < navFrame.childNodes.length; j++) {
		var div = navFrame.childNodes[j];
		if (/(^|[^a-zA-Z0-9_\-])NavHead(?![a-zA-Z0-9_\-])/.test(div.className))
			navHead = div;
		if (/(^|[^a-zA-Z0-9_\-])NavContent(?![a-zA-Z0-9_\-])/.test(div.className))
			navContent = div;
	}
	if (!navHead || !navContent)
		return;
	// Step 1, don't react when a subitem is clicked.
	for (var i=0; i<navHead.childNodes.length; i++) {
		var child = navHead.childNodes[i];
		if (child.nodeName == "A") {
			child.onclick = function (e) {
				if (e && e.stopPropagation)
					e.stopPropagation();
				else
					window.event.cancelBubble = true;
			};
		}
	}
	// Step 2, toggle visibility when bar is clicked.
	// NOTE This function was chosen due to some funny behaviour in Safari.
	navToggle = newNode('a', {href: 'javascript:(function(){})()'}, '');
	navHead.insertBefore(newNode('span', {'class': 'NavToggle'}, '[', navToggle, ']'), navHead.firstChild);

	navHead.style.cursor = "pointer";
	navHead.onclick = VisibilityToggles.register(NavToggleCategory(navFrame),
		function show() {
			navToggle.innerHTML = NavigationBarHide;
			if (navContent)
				navContent.style.display = "block";
		},
		function hide() {
			navToggle.innerHTML = NavigationBarShow;
			if (navContent)
				navContent.style.display = "none";
		});
}

jQuery(document).ready( function ()
{
	var divs = jQuery(".NavFrame");
	for (var i=0; i < divs.length; i++) {
		// XXX: some templates use a class of NavFrame for the style, but for legacy reasons, are not NavFrames
		// if (divs[i].className == "NavFrame") {
		createNavToggle(divs[i]);
		// }
	}
});

// Insular script stuff
( function ( mw, $ ) {
    var clicked = false;
    $(window).on( 'load', function () {
        console.log($( '.uls-settings-trigger' ));
        if ( $.cookie( 'insular' ) === 'true' ) {
            $( '*[style*="Junicode"]' ).addClass( 'insular' );
        }
        var $ulsTrigger = $( '.uls-settings-trigger' );
        //$.webfonts.repository.languages.ang[2] = 'Junicode-insular';
        mw.webfonts.preferences.setFont = function ( language, font ) {
            if ( font === 'Junicode-insular' ) {
                this.registry.fonts[language] = 'Junicode';
                var $element = $( 'body' );

                // Set the font of this element if it's not excluded
                $element.not( $.fn.webfonts.defaults.exclude ).addClass( 'insular' );

                // Set the font of this element's children if they are not excluded.
                // font-family of <input>, <textarea> and <button> must be changed explicitly.
                $element.find( 'textarea, input, button' )
                    .not( $.fn.webfonts.defaults.exclude )
                    .addClass( 'insular' );

                $( '*[style*="Junicode"]' ).addClass( 'insular' );
                $.cookie( 'insular', 'true' );
            } else if ( font === 'Junicode' ) {
                this.registry.fonts[language] = font;
                $( '.insular' ).removeClass( 'insular' );
                $.cookie( 'insular', 'false' );
            } else {
                this.registry.fonts[language] = font;
                $( '.insular' ).removeClass( 'insular' );
                $.cookie( 'insular', 'false' );
            }
        }
        clicked = true;
    } );
}( mediaWiki, jQuery ) );