User:Cacycle/wikiWatch.js

In this article we are going to address the topic of User:Cacycle/wikiWatch.js and explore its different facets. User:Cacycle/wikiWatch.js is a topic of great relevance in today's society, since it impacts various aspects of daily life. Throughout this article we will analyze its origin, its evolution over time and its influence in different areas. In addition, we will examine the different perspectives that exist around User:Cacycle/wikiWatch.js, with the aim of offering a complete and enriching vision on this topic. Through a multidimensional approach, we aim to offer our readers a broad and detailed vision of User:Cacycle/wikiWatch.js, with the purpose of encouraging reflection and debate around this topic that is so relevant today.
// <pre><nowiki>

// ==UserScript==
// @name        wikiWatch
// @namespace   http://en.wikipedia.orghttps://wikifreehand.com/en/User:Cacycle/
// @description A MediaWiki watchlist, recent changes, and user contributions page tool that sorts into namespaces, adds unwatch links, and auto-expands entries
// @include     /Special:
// @exclude
//
// @homepage    http://en.wikipedia.orghttps://wikifreehand.com/en/User:Cacycle/wikiWatch
// @source      http://en.wikipedia.orghttps://wikifreehand.com/en/User:Cacycle/wikiWatch.js
// @author      Cacycle (http://en.wikipedia.orghttps://wikifreehand.com/en/User:Cacycle)
// @license     Released into the public domain
// @version     0.9.5
// ==/UserScript==


//
// add wrapped wikiWatch code to body in order to access and use page scripts and variables under Greasemonkey
//

// start of global code wrapper
if (typeof(wwInstalledFlag) == 'undefined') { window.wwInstalledFlag = false; }

window.WWWrapper = function() {

	window.wwInstalledFlag = true;
	if (typeof(wwSetupFlag) == 'undefined') { window.wwSetupFlag = false; }
	var wwOpenPopup;
	var wwEditDivs = ;


//
// WWMain: wikiWatch setup
//

	window.WWMain = function() {

		WWRemoveEventListener(window, 'load', WWMain, false);

// check if this has already been run, either as a wiki gadget, wiki user script, or a Greasemonkey user script
		if (wwSetupFlag == true) {
			return;
		}
		wwSetupFlag = true;

// customizable definitions

// number of most recent entries in short display
		var wwRecentShort = wwRecentShort || 15;

// texts
		var WWText = WWText || {
			'WWRecentAll':       'All',
			'WWRecentAllToggle': '<span class="WWToggle" title="Toggle the display of all entries">(<a style="cursor: pointer;" onclick="javascript:{document.getElementById(\'WWSectionBlock-1-{day}\').style.display=\'block\';document.getElementById(\'WWSectionBlock-2-{day}\').style.display=\'none\';}">most recent only</a>)</span>',
			'WWRecentTop':       'All',
			'WWRecentTopToggle': '<span class="WWToggle" title="Toggle the display of all entries">(<a style="cursor: pointer;" onclick="document.getElementById(\'WWSectionBlock-2-{day}\').style.display=\'block\';document.getElementById(\'WWSectionBlock-1-{day}\').style.display=\'none\';">show all</a>)</span>',
			'WWUnwatch':         'unwatch'
		};

// css styles
		var WWCss = WWCss ||
			'.WWToggle         { font-weight: normal; }' +
			'.WWSectionFooter  { margin-left: 0.5em; }' +
			'.WWDayHeading     { background-color: #c0c0c0; border: gray solid 1px; margin: 1.25em 0 0.75em 0 !important; padding: 0 2em; }' +
			'.WWSectionHeading { margin: 0 0 0.5em 0; padding: 0 0 0.75em 0.5em; }' +
			'.WWSectionBlock   { margin: 0 0 0.75em 0; }' +
			'.WWSection        { border: gray solid 1px; padding: 0 0.5em 0.75em 0.5em; }' +
			'.WWSection > H5   { padding-top: 0.3em; padding-bottom: 0; }' +

			'.WWPopup          { display: block; padding: 0 1em 1.5em 2em; position: absolute; left: 17em; right: 1em; border: 1px solid black; border-top: none; border-right: 1px solid gray; }' +
			'.WWPopup TT       { display: none; }' +
			'.WWDebug          { display: none; position: static; margin-top: 3em; margin-left: 10em; zIndex: 1000000;}' +

//			'TD':                'text-indent: -2em; padding-left: 2em;',

			'.WWSection-1, .WWSection-1 DIV { background-color: #f8fcff; }' +
			'.WWSection-2, .WWSection-2 DIV { background-color: #f8fcff; }' +

			'.WWSection0, .WWSection0 DIV   { background-color: #ffffff; }' +
			'.WWSection1, .WWSection1 DIV   { background-color: #f2f2f2; }' +

			'.WWSection2, .WWSection2 DIV   { background-color: #e5e5e5; }' +
			'.WWSection3, .WWSection3 DIV   { background-color: #d0d0d0; }' +

// colors: HSB 40°-5%-100% (light), 40°-5%-95% (dark), 40° steps
			'.WWSection4, .WWSection4 DIV   { background-color: #fffbf2; }' +
			'.WWSection5, .WWSection5 DIV   { background-color: #f2eee6; }' +

			'.WWSection6, .WWSection6 DIV   { background-color: #fbfff2; }' +
			'.WWSection7, .WWSection7 DIV   { background-color: #eef2e6; }' +

			'.WWSection8, .WWSection8 DIV   { background-color: #f2fff2; }' +
			'.WWSection9, .WWSection9 DIV   { background-color: #e6f2e6; }' +

			'.WWSection10, .WWSection10 DIV { background-color: #f2fffb; }' +
			'.WWSection11, .WWSection11 DIV { background-color: #e6f2ee; }' +

			'.WWSection12, .WWSection12 DIV { background-color: #f2fbff; }' +
			'.WWSection13, .WWSection13 DIV { background-color: #e6eef2; }' +

			'.WWSection14, .WWSection14 DIV { background-color: #f2f2ff; }' +
			'.WWSection15, .WWSection15 DIV { background-color: #e6e6f2; }' +

			'.WWSection16, .WWSection16 DIV { background-color: #fbf2ff; }' +
			'.WWSection17, .WWSection17 DIV { background-color: #eee6f2; }' +

			'.WWSection18, .WWSection18 DIV { background-color: #fff2fb; }' +
			'.WWSection19, .WWSection19 DIV { background-color: #f2e6ee; }' +

			'.WWSection20, .WWSection20 DIV { background-color: #fff2f2; }' +
			'.WWSection21, .WWSection21 DIV { background-color: #f2e6e6; }' +

			'.WWSection22, .WWSection22 DIV { background-color: #fff2fb; }' +
			'.WWSection23, .WWSection23 DIV { background-color: #f2e6ee; }'
		;

// add stylesheet definitions
		var styles = new WWStyleSheet();
		styles.addRules(WWCss);

// detect pagetype and get the smallest container to replace for compatibility
		var wwWatchlist = false;
		var wwRecentchanges = false;
		var wwContributions = false;
		var container = document.body;
		if (mw.config.get('wgCanonicalSpecialPageName') == 'Watchlist') {
			wwWatchlist = true;
			var watchlistElement = document.getElementById('watchlist-message');
			if (watchlistElement != null) {
				container = watchlistElement.parentNode;
			}
		}
		else if (mw.config.get('wgCanonicalSpecialPageName') == 'Recentchanges') {
			wwRecentchanges = true;
			watchlistElement = document.getElementById('recentchangestext');
			if (watchlistElement != null) {
				container = watchlistElement.parentNode;
			}
		}
		else if (mw.config.get('wgCanonicalSpecialPageName') == 'Contributions') {
			wwContributions = true;
			watchlistElement = document.getElementById('contentSub');
			if (watchlistElement != null) {
				container = watchlistElement.parentNode;
			}
		}

		var regExpMatch;

// get watchlist block
		var sectionBlockIds = ;

// h4: watchlist, recent changes; ul, p: contributions
		var regExpWatchlist = new RegExp('(<select\\b*?\\bid="?namespace"?*>(.|\\s)*?)(<(h4|ul)\\b*>(.|\\s)*?)((<p>.*?</p>)?<div\\b*?\\bclass="?printfooter"?*>)', 'i');

// replace watchlist container
		container.innerHTML = container.innerHTML.replace(regExpWatchlist,
			function (p, p1, p2, p3, p4, p5, p6) {
				var namespaceBlock = p1;
				var watchlistBlock = p3;
				var pageBottom = p6;

// add unwatch links
				if ( (wwWatchlist == true) || (wwContributions == true) ) {

//                                                 12 link                         2 action3                  3     4     41
					watchlistBlock = watchlistBlock.replace(/((<a href=\"?*?&amp;action=)history(\b*\"?*>)+(<\/a>))/gi, '$1, $2unwatch$3' + WWText + '$4');
				}

// get namespaces
				var namespaceName = ;
				var namespaceNumber = ;
				var namespaceHash = ;
				var ns = 0;
				var namespaceSelect = namespaceBlock;

// get namespace select
				if ( (regExpMatch = /(<select\b*?\bid=\"?namespace\"?*>(.|\s)*?<\/select>)/i.exec(namespaceSelect) ) != null) {
					namespaceSelect = regExpMatch;
				}

// cycle through namespace options
				var regExpNamespace = new RegExp('<option\\b*?\\bvalue="?(\\d+)"?*>((.|\\s)*?)</option>', 'gi');
				while ( (regExpMatch = regExpNamespace.exec(namespaceSelect) ) != null) {
					namespaceNumber = regExpMatch;
					namespaceName = regExpMatch;
					if (namespaceNumber == 0) {
						namespaceName = namespaceName.substr(0, 1).toUpperCase() + namespaceName.substr(1);
					}
					namespaceHash ] = ns;
					ns ++;
				}
				var namespaceMax = ns;
				namespaceName = WWText;
				namespaceName = WWText;
				var namespaceSection = ;

// get day block
				var day = 0;
				var sorted = '';
				var regExpDays;

// contributions has no h4 but ul
				if (wwContributions == true) {
					regExpDays = new RegExp('((()))\\s*(<ul\\b*>)\\s*((.|\\s)*?)\\s*(</ul>)', 'gi');
				}

// watchlist and recent changes have h4 and optionally ul
				else {
					regExpDays = new RegExp('(<h4\\b*>\\s*((.|\\s)*?)\\s*</h4>)\\s*(<ul\\b*>)?\\s*((.|\\s)*?)\\s*(</ul>)?\\s*(?=(<h4\\b*>|$))', 'gi');
				}

// cycle through days or whole block
				while ( (regExpMatch = regExpDays.exec(watchlistBlock) ) != null) {
					var dayHeading = regExpMatch;
					var dayContainerTop = regExpMatch || '';
					var dayContainer = regExpMatch;
					var dayContainerBottom = regExpMatch || '';
					var mostRecent = 0;

// initialize and clear for +=
					for (var ns = -2; ns <= namespaceMax; ns ++) {
						namespaceSection = '';
					}
					namespaceSection = dayContainer;

// add day heading
					if (dayHeading != '') {
						sorted += '<div class="WWDayHeading">' + dayHeading + '</div>';
					}

// define regexps outside the loops
					var regExpArticles = new RegExp('((<(table|li)\\b*?>(.|\\s)*?</\\3>)\\s*(<div\\b*>(.|\\s)*?</div>)?)', 'gi');
					var regExpArticleName = new RegExp('<a\\b*?\\btitle=(\\"(*)\\"|(*))*>(\\2|\\4)</a>', 'gi');

// get article block
					while ( (regExpMatch = regExpArticles.exec(dayContainer) ) != null) {
						var articleEditsBlock = regExpMatch;
						var articleBlock = regExpMatch;
						var editsBlock = regExpMatch;

// get article name
						while ( (regExpMatch = regExpArticleName.exec(articleBlock) ) != null) {
							var articleName = regExpMatch;

// get article namespace
							regExpMatch = /^(.*?):/.exec(articleName);
							var namespaceIndex = 0;
							if (regExpMatch != null) {
								namespaceIndex = namespaceHash ];
							}

// append article block to section
							if (namespaceIndex != null) {
								namespaceSection += articleEditsBlock;

// add to most recent section
								if (mostRecent < wwRecentShort) {
									namespaceSection += articleEditsBlock;
									mostRecent ++;
								}
							}
						}

// finish articles
					}

// finish sections
					for (var ns = -2; ns <= namespaceMax; ns ++) {

// sort the two main namespaces last
						if (ns == 0) {
							ns = 2;
						}
						else if (ns == namespaceMax) {
							ns = 0;
						}
						else if (ns == 2) {
							break;
						}

// skip empty sections
						if (namespaceSection == '') {
							continue;
						}

// rename all and top section elements
						if ( (ns == -2) || (ns == -1) ) {
							namespaceSection = namespaceSection.replace(/(<*?\bid=(\"?)RC\d+)\2/g, '$1.' + ns + '$2');
							namespaceSection = namespaceSection.replace(/(<*?=\"?javascript:toggleVisibility\(\'RC\d+)(\',\s*\'RC\d+)(\',\s*\'RC\d+)(\')/g, '$1.' + ns + '$2.' + ns + '$3.' + ns + '$4');
						}

						var sectionStyle = '';
						var headerToggle = '';
						var footerToggle = '';
						if (ns == -1) {
							var topToggle = WWText.replace(/\{day\}/g, day)
							headerToggle += ' ' + topToggle;
							footerToggle = '<div class="WWSectionFooter">' + topToggle + '</div>';
							if (wwRecentchanges == true) {
								sectionStyle = ' style="display: none;"';
							}
						}
						else if (ns == -2) {
							var allToggle = WWText.replace(/\{day\}/g, day)
							headerToggle = ' ' + allToggle;
							footerToggle = '<div class="WWSectionFooter">' + allToggle + '</div>';
							if (wwRecentchanges == false) {
								sectionStyle = ' style="display: none;"';
							}
						}

// add sections to sorted watchlist
						sorted += '<div class="WWSectionBlock"' + sectionStyle + ' id="WWSectionBlock' + ns + '-' + day + '"><div class="WWSection' + ns + '"><div class="WWSection"><h5 class="WWSectionHeading">' + namespaceName + headerToggle + '</h5>' + dayContainerTop + namespaceSection + dayContainerBottom + footerToggle + '</div></div></div>';

// remember section block names
						sectionBlockIds.push('WWSectionBlock' + ns + '-' + day);
					}

// finish day
					day ++;
				}

// return changed document.body
				return(namespaceBlock + '<div id="WWSorted" class="WWSorted">' + sorted + '</div>' + pageBottom);
			}
		);

// ceate list of expandable entries for closing popups
		var divs = document.getElementsByTagName('div');
		for (var i = 0; i < divs.length; i ++) {
			if (divs.id.match(/^RCI+/) != null) {
				wwEditDivs.push(divs);
			}
		}

// regexp to identify article links from link href
		var regExpStr = '^' + mw.config.get('wgServer') + mw.config.get('wgArticlePath') + '$';
		regExpStr = regExpStr.replace(/\$\d/, '*?');
		regExpStr = regExpStr.replace(/(file:\/\/)(:)(\/)/, '$1(/$2)?$3');
		regExpStr = regExpStr.replace(/\/()/g, '\\/');
		regExpStr = regExpStr.replace(/\./g, '\\.');
		var regExpArticleLink = new RegExp(regExpStr);

// get all page links
		var bodyContent = document.getElementById('bodyContent');
		if (bodyContent == null) {
			return;
		}
		var links = bodyContent.getElementsByTagName('a');

// cycle through all links
		var linkNumber = 0;
		for (var i = 0; i < links.length; i ++) {

// check if this is an article link
			if (links.innerHTML.replace(/\"/g, '') != links.title) {
				continue;
			}
			if (links.href.match(regExpArticleLink) == null) {
				continue;
			}

// add mouseover event to link
			WWAddEventListener(links, 'mouseover', WWOpenPopup, false);

// remove title
			links.title = '';

// find the container table element
			var mainTable = links;
				while (mainTable != null) {
				if (mainTable.nodeName == 'TABLE') {
					break;
				}
				mainTable = mainTable.parentNode;
			}

// check if next sibling is an edits div
			if (mainTable == null) {
				continue;
			}

			var editsDiv = mainTable.nextSibling;
			while (editsDiv != null) {
				if (editsDiv.nodeType != 3) {
					break;
				}
				editsDiv = editsDiv.nextSibling;
			}

// add link id for easy access of related edits div
			if (editsDiv == null) {
				continue;
			}
			if (editsDiv.nodeName != 'DIV') {
				continue;
			}
			var editsDivId = editsDiv.id;
			if (editsDivId.match(/^RCI+/) != null) {
				links.id = 'WW' + linkNumber + '-' + editsDivId;
				linkNumber ++;
			}
		}

// close popup when leaving a section
		for (var i = 0; i <	sectionBlockIds.length; i ++) {
			WWAddEventListener( document.getElementById( sectionBlockIds ), 'mouseout', WWClosePopupsHandler, false );
		}

		return;
	}


//
// WWToggleVisibility: replacement for toggleVisibility
//

	function WWToggleVisibility(levelId, otherId, linkId) {

		var thisLevel = document.getElementById(levelId);
		var otherLevel = document.getElementById(otherId);
		var linkLevel = document.getElementById(linkId);

		if (thisLevel.className == 'WWPopup') {
			thisLevel.style.display = 'none';
			thisLevel.className = 'WWStandardExpand';
		}

		if (thisLevel.style.display == 'none')  {
			thisLevel.style.display = 'block';
			otherLevel.style.display = 'none';
			linkLevel.style.display = 'inline';
		}
		else {
			thisLevel.style.display = 'none';
			otherLevel.style.display = 'inline';
			linkLevel.style.display = 'none';
		}

		WWClosePopups(levelId);

		return;
	}
	var toggleVisibility = WWToggleVisibility;


//
// WWOpenPopup: open detailed edits preview
//

	function WWOpenPopup(event) {

// MS IE compatibility fix
		event = WWEvent(event);
		if (event == null) {
			return;
		}

// get link id with div RCI id
		var lnk = event.target
		var levelId;
		if (lnk == null) {
			return;
		}
		if (lnk.nodeName != 'A') {
			return;
		}

// only close popups for single-entry links
		if (lnk.id == '') {
			WWClosePopups();
		}

// popup edits
		else {
			levelId = lnk.id.replace(/^WW\d+-/, '');
			if (levelId != '') {
				var thisLevel = document.getElementById(levelId);

// show edits div as popup
				thisLevel.className = 'WWPopup';
				thisLevel.style.display = 'block';
				wwOpenPopup = thisLevel;

// down arrow back to right arrow
				var regExpMatch = /^RCI(+)/.exec(levelId);
				if (regExpMatch != null) {

					var idNumber = regExpMatch;
					var otherLevel = document.getElementById('RCM' + idNumber);
					var linkLevel = document.getElementById('RCL' + idNumber);

					otherLevel.style.display = 'inline';
					otherLevel.blur();
					linkLevel.style.display = 'none';
				}
				WWClosePopups(levelId);
			}
		}
		return;
	}


//
// WWClosePopups: close all edit popups
//

	function WWClosePopups(levelId) {

		if (wwOpenPopup != null) {
			if (wwOpenPopup.id != levelId) {
				wwOpenPopup.className = '';
				wwOpenPopup.style.display = 'none';
				wwOpenPopup = null;
			}
		}
		for (var i = 0; i < wwEditDivs.length; i ++) {
			if (wwEditDivs.className == 'WWPopup') {
				if (wwEditDivs.id != levelId) {
					wwEditDivs.className = '';
					wwEditDivs.style.display = 'none';
				}
			}
		}
		return;
	}


//
// WWClosePopupsHandler: close all edit popups when leaving the section
//

	function WWClosePopupsHandler(event) {

// MS IE compatibility fix
		event = WWEvent(event);
		if (event == null) {
			return;
		}

		event.stopPropagation();
		if (event.relatedTarget != null) {
			if (event.relatedTarget.id != '') {
				if (event.relatedTarget.id.match(/\bRCI\d+\.+/) == null) {
					WWClosePopups();
				}
			}
		}
		return;
	}


//
// WWEvent: MS IE compatibility fix for event object
//

	function WWEvent(event) {

		var eventAlt;
		if (window.event != null) {
			eventAlt = window.event;
			if (eventAlt != null) {
				event = eventAlt;
				event.stopPropagation = function() {
					event.cancelBubble = true;
				};
				event.preventDefault = function() {
					event.returnValue = false;
				};
				event.target = event.srcElement;
				if (event.type == 'mouseout') {
			    event.relatedTarget = event.toElement;
				}
				else if (event.type == 'mouseover') {
			    event.relatedTarget = event.fromElement;
				}
			}
		}
		return(event);
	}


//
// WWAddEventListener: wrapper for addEventListener (http://ejohn.org/projects/flexible-javascript-events/)
//

	function WWAddEventListener(domElement, eventType, eventHandler, useCapture) {

		if (domElement != null) {
			if (domElement.attachEvent != null) {
				domElement = eventHandler;
				domElement = function() {
					domElement(window.event);
				}
				domElement.attachEvent('on' + eventType, domElement );
			}
			else {
				domElement.addEventListener(eventType, eventHandler, useCapture);
			}
		}
		return;
	}


//
// WWRemoveEventListener: wrapper for removeEventListener
//

	function WWRemoveEventListener(domElement, eventType, eventHandler, useCapture) {

		if (domElement.detachEvent != null) {
			domElement.detachEvent('on' + eventType, domElement);
			domElement = null;
		}
		else {
			domElement.removeEventListener(eventType, eventHandler, useCapture);
		}
		return;
	}


//
// WWStyleSheet: create a new style sheet object
//

	function WWStyleSheet(contextObj) {

		if (contextObj == null) {
			contextObj = document;
		}
		this.styleElement = null;

// MS IE compatibility
		if (contextObj.createStyleSheet) {
			this.styleElement = contextObj.createStyleSheet();
		}

// standards compliant browsers
		else {
			this.styleElement = contextObj.createElement('style');
			this.styleElement.from = 'text/css';
			var insert = contextObj.getElementsByTagName('head');
			if (insert != null) {
				this.styleElement.appendChild(contextObj.createTextNode(''));
				insert.appendChild(this.styleElement);
			}
		}


//
// WWStyleSheet.addRules: add all rules at once, much faster
//

		this.addRules = function(rules) {

// MS IE compatibility
			if (this.styleElement.innerHTML == null) {
				this.styleElement.cssText = rules;
			}

// via innerHTML
			else {
				this.styleElement.innerHTML = rules;
			}
			return;
		}
	}


//
// WWDebug: print the value of variables, use either a single value or a description followed by a value
//

	function WWDebug(objectName, object) {

// create debug textarea and add to debug wrapper
		if (window.WWDebugTextarea == null) {
			window.WWDebugTextarea = document.createElement('textarea');
			WWDebugTextarea.id = 'WWDebug';
			WWDebugTextarea.className = 'WWDebug';
			WWDebugTextarea.rows = 20;
			WWDebugTextarea = document.body.insertBefore(WWDebugTextarea, document.body.firstChild);
		}

		WWDebugTextarea.style.display = 'block';
		if (objectName == null) {
			WWDebugTextarea.value = '';
		}
		else {
			if (object == null) {
				WWDebugTextarea.value = objectName + '\n' + WWDebugTextarea.value;
			}
			else {
				WWDebugTextarea.value = objectName + ': ' + object + '\n' + WWDebugTextarea.value;
			}
		}
		return;
	}
	var WWD = WWDebug;

// schedule setup routine
	if ( (typeof(mw.config.get('wgCanonicalNamespace')) != 'undefined') && (typeof(mw.config.get('wgCanonicalSpecialPageName')) != 'undefined') ) {
		if (
			(mw.config.get('wgCanonicalNamespace') == 'Special') && (
				(mw.config.get('wgCanonicalSpecialPageName') == 'Watchlist') ||
				(mw.config.get('wgCanonicalSpecialPageName') == 'Recentchanges') ||
				(mw.config.get('wgCanonicalSpecialPageName') == 'Contributions')
			)
		) {
			WWAddEventListener(window, 'load', WWMain, false);
		}
	}

// close global wrapper
	return;
}

// append wrapper to head if run under Greasemonkey
if (window.parent == window) {
	if (typeof(GM_getValue) == 'function') {
		if ( (document.getElementById('WWHeadScript') == null) && (window.wwInstalledFlag == false) ) {
			window.wwInstalledFlag = true;
			var script = document.createElement('script');
			script.id = 'WWHeadScript';
			script.type = 'text/javascript';
			script.text = 'WWHeadWrapper = ' + WWWrapper.toString() + '; WWHeadWrapper();';
			document.getElementsByTagName('head').appendChild(script);
		}
	}

// otherwise run installation directly
	else {
		WWWrapper();
	}
}

// </nowiki></pre>