User:Slakr/twinklexfd-new.js

/** * +-+ * |                 === WARNING: GLOBAL GADGET FILE ===                    | * |               Changes to this page affect many users. | * |          Please discuss changes at WT:TW before editing. | * +-+ * * Imported from github. * All changes should be made in the repository, otherwise they will be lost. * * To update this script from github, you must have a local repository set up. Then * follow the instructions at. * * -- * * This is AzaToth's Twinkle, the popular script sidekick for newbies, admins, and * every Wikipedian in between. Visit WP:TW for more information. */

//

( function ( $, undefined ) { // Wrap with anonymous function

var Twinkle = {}; window.Twinkle = Twinkle; // allow global access

// for use by custom modules (normally empty) Twinkle.initCallbacks = []; Twinkle.addInitCallback = function twinkleAddInitCallback( func ) { Twinkle.initCallbacks.push( func ); };

Twinkle.defaultConfig = {}; /** * Twinkle.defaultConfig.twinkle and Twinkle.defaultConfig.friendly * * This holds the default set of preferences used by Twinkle. (The |friendly| object holds preferences stored in the FriendlyConfig object.) * It is important that all new preferences added here, especially admin-only ones, are also added to * |Twinkle.config.sections| in twinkleconfig.js, so they are configurable via the Twinkle preferences panel. * For help on the actual preferences, see the comments in twinkleconfig.js. */ Twinkle.defaultConfig.twinkle = { // General summaryAd: " (TW)", deletionSummaryAd: " (TW)", protectionSummaryAd: " (TW)", userTalkPageMode: "window", dialogLargeFont: false, // Fluff (revert and rollback) openTalkPage: [ "agf", "norm", "vand" ], openTalkPageOnAutoRevert: false, markRevertedPagesAsMinor: [ "vand" ], watchRevertedPages: [ "agf", "norm", "vand", "torev" ], offerReasonOnNormalRevert: true, confirmOnFluff: false, showRollbackLinks: [ "diff", "others" ], // DI (twinkleimage) notifyUserOnDeli: true, deliWatchPage: "default", deliWatchUser: "default", // PROD watchProdPages: true, prodReasonDefault: "", logProdPages: false, prodLogPageName: "PROD log", // CSD speedyPromptOnG7: false, watchSpeedyPages: [ "g3", "g5", "g10", "g11", "g12" ], markSpeedyPagesAsPatrolled: true, // these next two should probably be identical by default notifyUserOnSpeedyDeletionNomination:   [ "db", "g1", "g2", "g3", "g4", "g10", "g11", "g12", "a1", "a2", "a3", "a5", "a7", "a9", "a10", "f1", "f2", "f3", "f7", "f9", "f10", "u3", "t2", "t3", "p1", "p2" ], welcomeUserOnSpeedyDeletionNotification: [ "db", "g1", "g2", "g3", "g4", "g10", "g11", "g12", "a1", "a2", "a3", "a5", "a7", "a9", "a10", "f1", "f2", "f3", "f7", "f9", "f10", "u3", "t2", "t3", "p1", "p2" ], promptForSpeedyDeletionSummary: [ "db", "g1", "g2", "g3", "g4", "g6", "g7", "g8", "g10", "g11", "g12", "a1", "a2", "a3", "a5", "a7", "a9", "a10", "f2", "f4", "f7", "f8", "f10", "t2", "t3", "p1", "p2" ], openUserTalkPageOnSpeedyDelete: [ "db", "g1", "g2", "g3", "g4", "g5", "g10", "g11", "g12", "a1", "a3", "a7", "a9", "a10", "f3", "f7", "f9", "u3", "t2", "p1" ], deleteTalkPageOnDelete: false, deleteSysopDefaultToTag: false, speedyWindowHeight: 500, speedyWindowWidth: 800, logSpeedyNominations: false, speedyLogPageName: "CSD log", noLogOnSpeedyNomination: [ "u1" ], // Unlink unlinkNamespaces: [ "0", "100" ], // Warn defaultWarningGroup: "1", showSharedIPNotice: true, watchWarnings: true, blankTalkpageOnIndefBlock: false, // XfD xfdWatchDiscussion: "default", xfdWatchList: "no", xfdWatchPage: "default", xfdWatchUser: "default", // Hidden preferences revertMaxRevisions: 50, batchdeleteChunks: 50, batchDeleteMinCutOff: 5, batchMax: 5000, batchProtectChunks: 50, batchProtectMinCutOff: 5, batchundeleteChunks: 50, batchUndeleteMinCutOff: 5, deliChunks: 500, deliMax: 5000, proddeleteChunks: 50 };

// now some skin dependent config. if ( mw.config.get( "skin" ) === "vector" ) { Twinkle.defaultConfig.twinkle.portletArea = "right-navigation"; Twinkle.defaultConfig.twinkle.portletId  = "p-twinkle"; Twinkle.defaultConfig.twinkle.portletName = "TW"; Twinkle.defaultConfig.twinkle.portletType = "menu"; Twinkle.defaultConfig.twinkle.portletNext = "p-search"; } else { Twinkle.defaultConfig.twinkle.portletArea = null; Twinkle.defaultConfig.twinkle.portletId  = "p-cactions"; Twinkle.defaultConfig.twinkle.portletName = null; Twinkle.defaultConfig.twinkle.portletType = null; Twinkle.defaultConfig.twinkle.portletNext = null; }

Twinkle.defaultConfig.friendly = { // Tag groupByDefault: true, watchTaggedPages: true, markTaggedPagesAsMinor: false, markTaggedPagesAsPatrolled: true, tagArticleSortOrder: "cat", customTagList: [], // Welcome topWelcomes: false, watchWelcomes: true, welcomeHeading: "Welcome", insertHeadings: true, insertUsername: true, insertSignature: true, // sign welcome templates, where appropriate quickWelcomeMode: "norm", quickWelcomeTemplate: "welcome", customWelcomeList: [], // Talkback markTalkbackAsMinor: true, insertTalkbackSignature: true, // always sign talkback templates talkbackHeading: "Talkback", adminNoticeHeading: "Notice", mailHeading: "You've got mail!", // Shared markSharedIPAsMinor: true };

Twinkle.getPref = function twinkleGetPref( name ) { var result; if ( typeof Twinkle.prefs === "object" && typeof Twinkle.prefs.twinkle === "object" ) { // look in Twinkle.prefs (twinkleoptions.js) result = Twinkle.prefs.twinkle[name]; } else if ( typeof window.TwinkleConfig === "object" ) { // look in TwinkleConfig result = window.TwinkleConfig[name]; }

if ( result === undefined ) { return Twinkle.defaultConfig.twinkle[name]; }	return result; };

Twinkle.getFriendlyPref = function twinkleGetFriendlyPref(name) { var result; if ( typeof Twinkle.prefs === "object" && typeof Twinkle.prefs.friendly === "object" ) { // look in Twinkle.prefs (twinkleoptions.js) result = Twinkle.prefs.friendly[ name ]; } else if ( typeof window.FriendlyConfig === "object" ) { // look in FriendlyConfig result = window.FriendlyConfig[ name ]; }

if ( result === undefined ) { return Twinkle.defaultConfig.friendly[ name ]; }	return result; };

/** * **************** twAddPortlet **************** * * Adds a portlet menu to one of the navigation areas on the page. * This is necessarily quite a hack since skins, navigation areas, and * portlet menu types all work slightly different. * * Available navigation areas depend on the skin used. * Monobook: * "column-one", outer div class "portlet", inner div class "pBody". Existing portlets: "p-cactions", "p-personal", "p-logo", "p-navigation", "p-search", "p-interaction", "p-tb", "p-coll-print_export" * Special layout of p-cactions and p-personal through specialized styles. * Vector: * "mw-panel", outer div class "portal", inner div class "body". Existing portlets/elements: "p-logo", "p-navigation", "p-interaction", "p-tb", "p-coll-print_export" * "left-navigation", outer div class "vectorTabs" or "vectorMenu", inner div class "" or "menu". Existing portlets: "p-namespaces", "p-variants" (menu) * "right-navigation", outer div class "vectorTabs" or "vectorMenu", inner div class "" or "menu". Existing portlets: "p-views", "p-cactions" (menu), "p-search" * Special layout of p-personal portlet (part of "head") through specialized styles. * Modern: * "mw_contentwrapper" (top nav), outer div class "portlet", inner div class "pBody". Existing portlets or elements: "p-cactions", "mw_content" * "mw_portlets" (sidebar), outer div class "portlet", inner div class "pBody". Existing portlets: "p-navigation", "p-search", "p-interaction", "p-tb", "p-coll-print_export" * * @param String navigation -- id of the target navigation area (skin dependant, on vector either of "left-navigation", "right-navigation", or "mw-panel") * @param String id -- id of the portlet menu to create, preferably start with "p-". * @param String text -- name of the portlet menu to create. Visibility depends on the class used. * @param String type -- type of portlet. Currently only used for the vector non-sidebar portlets, pass "menu" to make this portlet a drop down menu. * @param Node nextnodeid -- the id of the node before which the new item should be added, should be another item in the same list, or undefined to place it at the end. * * @return Node -- the DOM node of the new item (a DIV element) or null */ function twAddPortlet( navigation, id, text, type, nextnodeid ) {	//sanity checks, and get required DOM nodes var root = document.getElementById( navigation ); if ( !root ) { return null; }

var item = document.getElementById( id ); if ( item ) { if ( item.parentNode && item.parentNode === root ) { return item; }		return null; }

var nextnode; if ( nextnodeid ) { nextnode = document.getElementById(nextnodeid); }

//verify/normalize input type = ( skin === "vector" && type === "menu" && ( navigation === "left-navigation" || navigation === "right-navigation" )) ? "menu" : ""; var outerDivClass; var innerDivClass; switch ( skin ) {		case "vector": if ( navigation !== "portal" && navigation !== "left-navigation" && navigation !== "right-navigation" ) { navigation = "mw-panel"; }			outerDivClass = ( navigation === "mw-panel" ) ? "portal" : ( type === "menu" ? "vectorMenu extraMenu" : "vectorTabs extraMenu" ); innerDivClass = ( navigation === "mw-panel" ) ? "body" : ( type === "menu" ? "menu" : "" ); break; case "modern": if ( navigation !== "mw_portlets" && navigation !== "mw_contentwrapper" ) { navigation = "mw_portlets"; }			outerDivClass = "portlet"; innerDivClass = "pBody"; break; default: navigation = "column-one"; outerDivClass = "portlet"; innerDivClass = "pBody"; break; }

// Build the DOM elements. var outerDiv = document.createElement( "div" ); outerDiv.className = outerDivClass + " emptyPortlet"; outerDiv.id = id; if ( type === "menu" ) { // Fix drop-down arrow image in Vector skin outerDiv.style.backgroundImage = 'url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABYAAAAQCAMAAAAlM38UAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAA9QTFRFsbGxmpqa3d3deXl58/n79CzHcQAAAAV0Uk5T/////wD7tg5TAAAAMklEQVR42mJgwQoYBkqYiZEZAhiZUFRDxWGicEPA4nBRhNlAcYQokpVMDEwD6kuAAAMAyGMFQVv5ldcAAAAASUVORK5CYII=")'; outerDiv.style.backgroundPosition = "right 60%"; }	if ( nextnode && nextnode.parentNode === root ) { root.insertBefore( outerDiv, nextnode ); } else { root.appendChild( outerDiv ); }

var h5 = document.createElement( "h5" ); if ( type === "menu" ) { var span = document.createElement( "span" ); span.appendChild( document.createTextNode( text ) ); h5.appendChild( span );

var a = document.createElement( "a" ); a.href = "#"; span = document.createElement( "span" ); span.appendChild( document.createTextNode( text ) ); a.appendChild( span ); h5.appendChild( a ); } else { h5.appendChild( document.createTextNode( text ) ); }	outerDiv.appendChild( h5 );

var innerDiv = document.createElement( "div" ); // Not strictly necessary with type vectorTabs, or other skins. innerDiv.className = innerDivClass; outerDiv.appendChild(innerDiv);

var ul = document.createElement( "ul" ); innerDiv.appendChild( ul );

return outerDiv; }

/** * **************** twAddPortletLink **************** * Builds a portlet menu if it doesn't exist yet, and add the portlet link. * @param task: Either a URL for the portlet link or a function to execute. */ function twAddPortletLink( task, text, id, tooltip ) {	if ( Twinkle.getPref("portletArea") !== null ) { twAddPortlet( Twinkle.getPref( "portletArea" ), Twinkle.getPref( "portletId" ), Twinkle.getPref( "portletName" ), Twinkle.getPref( "portletType" ), Twinkle.getPref( "portletNext" )); }	var link = mw.util.addPortletLink( Twinkle.getPref( "portletId" ), typeof task === "string" ? task : "#", text, id, tooltip ); if ($.isFunction(task)) $(link).click(function (ev) { task; ev.preventDefault; }); return link; }

// Check if account is experienced enough to use Twinkle var twinkleUserAuthorized = userIsInGroup( "autoconfirmed" ) || userIsInGroup( "confirmed" );

/* **************************************** *** twinklexfd.js: XFD module **************************************** * Mode of invocation:    Tab ("XFD") * Active on:             Existing, non-special pages, except for file pages with no local (non-Commons) file which are not redirects * Config directives in:  TwinkleConfig */

Twinkle.xfd = function twinklexfd { // Disable on: // * special pages // * non-existent pages // * files on Commons, whether there is a local page or not (unneeded local pages of files on Commons are eligible for CSD F2) // * file pages without actual files (these are eligible for CSD G8) if ( mw.config.get('wgNamespaceNumber') < 0 || !mw.config.get('wgArticleId') || (mw.config.get('wgNamespaceNumber') === 6 && (document.getElementById('mw-sharedupload') || (!document.getElementById('mw-imagepage-section-filehistory') && !Wikipedia.isPageRedirect))) ) { return; }	twAddPortletLink( Twinkle.xfd.callback, "XFD", "tw-xfd", "Nominate for deletion" ); };

Twinkle.xfd.num2order = function twinklexfdNum2order( num ) { switch( num ) { case 1: return ''; case 2: return '2nd'; case 3: return '3rd'; default: return num + 'th'; } };

Twinkle.xfd.currentRationale = null;

// error callback on Status object Twinkle.xfd.printRationale = function twinklexfdPrintRationale { if (Twinkle.xfd.currentRationale) { var p = document.createElement("p"); p.textContent = "Your deletion rationale is provided below, which you can copy and paste into a new XFD dialog if you wish to try again:"; var pre = document.createElement("pre"); pre.className = "toccolours"; pre.style.marginTop = "0"; pre.textContent = Twinkle.xfd.currentRationale; p.appendChild(pre); Status.root.appendChild(p); // only need to print the rationale once Twinkle.xfd.currentRationale = null; } };

Twinkle.xfd.callback = function twinklexfdCallback { if (!twinkleUserAuthorized) { alert("Your account is too new to use Twinkle."); return; }

var Window = new SimpleWindow( 600, 350 ); Window.setTitle( "Nominate for deletion (XfD)" ); Window.setScriptName( "Twinkle" ); Window.addFooterLink( "About deletion discussions", "WP:XFD" ); Window.addFooterLink( "Twinkle help", "WP:TW/DOC#xfd" );

var form = new QuickForm( Twinkle.xfd.callback.evaluate ); var categories = form.append( {			type: 'select',			name: 'category',			label: 'Deletion discussion venue:',			tooltip: 'When activated, a default choice is made, based on what namespace you are in. This default should be the most appropriate',			event: Twinkle.xfd.callback.change_category		} ); categories.append( {			type: 'option',			label: 'AfD (Articles for deletion)',			selected: mw.config.get('wgNamespaceNumber') === 0, // Main namespace			value: 'afd'		} ); categories.append( {			type: 'option',			label: 'TfD (Templates for discussion)',			selected: mw.config.get('wgNamespaceNumber') === 10, // Template namespace			value: 'tfd'		} ); categories.append( {			type: 'option',			label: 'FfD (Files for deletion)/PUF (Possibly unfree files)',			selected: mw.config.get('wgNamespaceNumber') === 6, // File namespace			value: 'ffd'		} ); categories.append( {			type: 'option',			label: 'CfD (Categories for discussion)',			selected: mw.config.get('wgNamespaceNumber') === 14, // Category namespace			value: 'cfd'		} ); categories.append( {			type: 'option',			label: 'CfD/S (Categories for speedy renaming)',			value: 'cfds'		} ); categories.append( {			type: 'option',			label: 'MfD (Miscellany for deletion)',			selected: [ 0, 6, 10, 14 ].indexOf( mw.config.get('wgNamespaceNumber') ) === -1,			value: 'mfd'		} ); categories.append( {			type: 'option',			label: 'RfD (Redirects for discussion)',			selected: Wikipedia.isPageRedirect,			value: 'rfd'		} ); //categories.append( {	//		type: 'option',	//		label: 'SfD (Stub types for deletion)',	//		disabled: true,	//		value: 'sfd'	//	} ); form.append( {			type: 'checkbox',			list: [				{					label: 'Notify page creator if possible',					value: 'notify',					name: 'notify',					tooltip: "A notification template will be placed on the creator's talk page if this is true.",					checked: true				}			]		}	); form.append( {			type: 'field',			label:'Work area',			name: 'work_area'		} ); form.append( { type:'submit' } );

var result = form.render; Window.setContent( result ); Window.display;

// We must init the controls var evt = document.createEvent( "Event" ); evt.initEvent( 'change', true, true ); result.category.dispatchEvent( evt ); };

Twinkle.xfd.previousNotify = true;

Twinkle.xfd.callback.change_category = function twinklexfdCallbackChangeCategory(e) { var value = e.target.value; var form = e.target.form; var old_area; var childNodes = form.childNodes; for( var i = 0; i < childNodes.length; ++i ) { var node = childNodes[i]; if (node instanceof Element &&		   node.getAttribute( 'name' ) === 'work_area') { old_area = node; break; }	}	var work_area = null;

var oldreasontextbox = form.getElementsByTagName('textarea')[0]; var oldreason = (oldreasontextbox ? oldreasontextbox.value : '');

switch( value ) { case 'afd': work_area = new QuickForm.element( {				type: 'field',				label: 'Articles for deletion',				name: 'work_area'			} ); work_area.append( {				type: 'checkbox',				list: [						{							label: 'Wrap deletion tag with ',							value: 'noinclude',							name: 'noinclude',							tooltip: 'Will wrap the deletion tag in &lt;noinclude&gt; tags, so that it won\'t transclude. This option is not normally required.'						}					]		} ); var afd_category = work_area.append( {				type:'select',				name:'xfdcat',				label:'Choose what category this nomination belongs in:'			} );

afd_category.append( { type:'option', label:'Unknown', value:'?', selected:true } ); afd_category.append( { type:'option', label:'Media and music', value:'M' } ); afd_category.append( { type:'option', label:'Organisation, corporation, or product', value:'O' } ); afd_category.append( { type:'option', label:'Biographical', value:'B' } ); afd_category.append( { type:'option', label:'Society topics', value:'S' } ); afd_category.append( { type:'option', label:'Web or internet', value:'W' } ); afd_category.append( { type:'option', label:'Games or sports', value:'G' } ); afd_category.append( { type:'option', label:'Science and technology', value:'T' } ); afd_category.append( { type:'option', label:'Fiction and the arts', value:'F' } ); afd_category.append( { type:'option', label:'Places and transportation', value:'P' } ); afd_category.append( { type:'option', label:'Indiscernible or unclassifiable topic', value:'I' } ); afd_category.append( { type:'option', label:'Debate not yet sorted', value:'U' } );

work_area.append( {				type: 'textarea',				name: 'xfdreason',				label: 'Reason: ',				value: oldreason			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; case 'tfd': work_area = new QuickForm.element( {				type: 'field',				label: 'Templates for discussion',				name: 'work_area'			} ); var linknode = document.createElement('a'); linknode.setAttribute("href", mw.util.getUrl("WP:SFD") ); linknode.appendChild(document.createTextNode('WP:SFD')); work_area.append( {				type: 'div',				label: [ 'Stub types and userboxes are not eligible for TfD. Stub types go to ', linknode, ', and userboxes go to MfD.' ]			} ); work_area.append( {				type: 'checkbox',				list: [						{							label: 'Inline deletion tag',							value: 'tfdinline',							name: 'tfdinline',							tooltip: 'Use to tag the page instead of . Good for inline templates (those that appear amongst the words of text).',							checked: false						},						{							label: 'Wrap deletion tag with (for substituted templates only)',							value: 'noinclude',							name: 'noinclude',							tooltip: 'Will wrap the deletion tag in &lt;noinclude&gt; tags, so that it won\'t get substituted along with the template.'						}					]		} ); work_area.append( {				type: 'textarea',				name: 'xfdreason',				label: 'Reason: ',				value: oldreason			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; case 'mfd': work_area = new QuickForm.element( {				type: 'field',				label: 'Miscellany for deletion',				name: 'work_area'			} ); work_area.append( {				type: 'checkbox',				list: [						{							label: 'Wrap deletion tag with ',							value: 'noinclude',							name: 'noinclude',							tooltip: 'Will wrap the deletion tag in &lt;noinclude&gt; tags, so that it won\'t transclude. Select this option for userboxes.'						}					]		} ); if (mw.config.get('wgNamespaceNumber') === 2 /* User: */ || mw.config.get('wgNamespaceNumber') === 3 /* User talk: */) { work_area.append( {				type: 'checkbox',				list: [						{							label: 'Also notify owner of userspace if they are not the page creator',							value: 'notifyuserspace',							name: 'notifyuserspace',							tooltip: 'If the user in whose userspace this page is located, is not the page creator (for example, the page is a rescued article stored as a userspace draft), notify the userspace owner as well.',							checked: true						}					]			} ); }		work_area.append( {				type: 'textarea',				name: 'xfdreason',				label: 'Reason: ',				value: oldreason			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; case 'ffd': work_area = new QuickForm.element( {				type: 'field',				label: 'Files for deletion',				name: 'work_area'			} ); work_area.append( {				type: 'checkbox',				name: 'puf',				list: [					{						label: 'Possibly unfree file',						value: 'puf',						tooltip: 'File has disputed source or licensing information'					}				]			} ); work_area.append( {				type: 'textarea',				name: 'xfdreason',				label: 'Reason: ',				value: oldreason			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; case 'cfd': work_area = new QuickForm.element( {				type: 'field',				label: 'Categories for discussion',				name: 'work_area'			} ); var cfd_category = work_area.append( {				type: 'select',				label: 'Choose type of action wanted: ',				name: 'xfdcat',				event: function(e) {					var value = e.target.value;					var target = e.target.form.xfdtarget;					// update enabled status					if( value === 'cfd' ) {						target.disabled = true;					} else {						target.disabled = false;					}					// update label					if( value === 'cfs' ) {						target.previousSibling.textContent = "Target categories: ";					} else if( value === 'cfc' ) {						target.previousSibling.textContent = "Target article: ";					} else {						target.previousSibling.textContent = "Target category: ";					}					// add/remove extra input box					if( value === 'cfs' && $(target.parentNode).find("input[name='xfdtarget2']").length === 0 ) {						var xfdtarget2 = document.createElement("input");						xfdtarget2.setAttribute("name", "xfdtarget2");						xfdtarget2.setAttribute("type", "text"); target.parentNode.appendChild(xfdtarget2); } else { $(target.parentNode).find("input[name='xfdtarget2']").remove; }				}			} );		cfd_category.append( { type: 'option', label: 'Deletion', value: 'cfd', selected: true } );		cfd_category.append( { type: 'option', label: 'Merge', value: 'cfm' } );		cfd_category.append( { type: 'option', label: 'Renaming', value: 'cfr' } );		cfd_category.append( { type: 'option', label: 'Split', value: 'cfs' } );		cfd_category.append( { type: 'option', label: 'Convert into article', value: 'cfc' } );

work_area.append( {				type: 'input',				name: 'xfdtarget',				label: 'Target page: ',				disabled: true,				value: ''			} ); work_area.append( {				type: 'textarea',				name: 'xfdreason',				label: 'Reason: ',				value: oldreason			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; case 'cfds': work_area = new QuickForm.element( {				type: 'field',				label: 'Categories for speedy renaming',				name: 'work_area'			} ); var cfds_category = work_area.append( {				type: 'select',				label: 'C2 sub-criterion: ',				name: 'xfdcat',				tooltip: 'See WP:CFDS for full explanations.',				event: function(e) {					var value = e.target.value;					var target = e.target.form.xfdtarget;					if( value === 'cfd' ) {						target.disabled = true;					} else {						target.disabled = false;					}				}			} ); cfds_category.append( { type: 'option', label: 'C2A: Typographic and spelling fixes', value: 'C2A', selected: true } ); cfds_category.append( { type: 'option', label: 'C2B: Naming conventions and disambiguation', value: 'C2B' } ); cfds_category.append( { type: 'option', label: 'C2C: Consistency with names of similar categories', value: 'C2C' } ); cfds_category.append( { type: 'option', label: 'C2D: Rename to match article name', value: 'C2D' } );

work_area.append( {				type: 'input',				name: 'xfdtarget',				label: 'New name: ',				value: ''			} ); work_area.append( {				type: 'textarea',				name: 'xfdreason',				label: 'Reason: ',				value: oldreason			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; case 'rfd': work_area = new QuickForm.element( {				type: 'field',				label: 'Redirects for discussion',				name: 'work_area'			} ); work_area.append( {				type: 'textarea',				name: 'xfdreason',				label: 'Reason: ',				value: oldreason			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; default: work_area = new QuickForm.element( {				type: 'field',				label: 'Nothing for anything',				name: 'work_area'			} ); work_area = work_area.render; old_area.parentNode.replaceChild( work_area, old_area ); break; }

// No creator notification for CFDS if (value === "cfds") { Twinkle.xfd.previousNotify = form.notify.checked; form.notify.checked = false; form.notify.disabled = true; } else { form.notify.checked = Twinkle.xfd.previousNotify; form.notify.disabled = false; } };

Twinkle.xfd.callbacks = { afd: { main: function(apiobj) { var xmlDoc = apiobj.responseXML; var titles = $(xmlDoc).find('allpages p');

// There has been no earlier entries with this prefix, just go on. if( titles.length <= 0 ) { apiobj.params.numbering = apiobj.params.number = ''; } else { var number = 0; for( var i = 0; i < titles.length; ++i ) { var title = titles[i].getAttribute('title');

// First, simple test, is there an instance with this exact name? if( title === 'Wikipedia:Articles for deletion/' + mw.config.get('wgPageName') ) { number = Math.max( number, 1 ); continue; }

var order_re = new RegExp( '^' +						RegExp.escape( 'Wikipedia:Articles for deletion/' + mw.config.get('wgPageName'), true ) +						'\\s*\\(\\s*(\\d+)(?:(?:th|nd|rd|st) nom(?:ination)?)?\\s*\\)\\s*$'); var match = order_re.exec( title );

// No match; A non-good value if( !match ) { continue; }

// A match, set number to the max of current number = Math.max( number, Number(match[1]) ); }				apiobj.params.number = Twinkle.xfd.num2order( parseInt( number, 10 ) + 1); apiobj.params.numbering = number > 0 ? ' (' + apiobj.params.number + ' nomination)' : ''; }			apiobj.params.discussionpage = 'Wikipedia:Articles for deletion/' + mw.config.get('wgPageName') + apiobj.params.numbering;

Status.info( "Next discussion page", "" + apiobj.params.discussionpage + "" );

// Updating data for the action completed event Wikipedia.actionCompleted.redirect = apiobj.params.discussionpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to the discussion page";

// Tagging article var wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Adding deletion tag to article"); wikipedia_page.setFollowRedirect(true); // should never be needed, but if the article is moved, we would want to follow the redirect wikipedia_page.setCallbackParameters(apiobj.params); wikipedia_page.load(Twinkle.xfd.callbacks.afd.taggingArticle); },		// Tagging needs to happen before everything else: this means we can check if there is an AfD tag already on the page taggingArticle: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters; var statelem = pageobj.getStatusElement;

// Check for existing AfD tag, for the benefit of new page patrollers var textNoAfd = text.replace(/\{\{\s*(Article for deletion\/dated|AfDM)\s*(\|(?:\{\{[^{}]*\}\}|[^{}])*)?\}\}\s*/g, ""); if (text !== textNoAfd) { if (confirm("An AfD tag was found on this article. Maybe someone beat you to it. \nClick OK to replace the current AfD tag (not recommended), or Cancel to abandon your nomination.")) { text = textNoAfd; } else { statelem.error("Article already tagged with AfD tag, and you chose to abort"); window.location.reload; return; }			}

// Now we know we want to go ahead with it, trigger the other AJAX requests

// Starting discussion page var wikipedia_page = new Wikipedia.page(params.discussionpage, "Creating article deletion discussion page"); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.afd.discussionPage);

// Today's list var date = new Date; wikipedia_page = new Wikipedia.page('Wikipedia:Articles for deletion/Log/' + date.getUTCFullYear + ' ' +				date.getUTCMonthName + ' ' + date.getUTCDate, "Adding discussion to today's list"); wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.afd.todaysList);

// Notification to first contributor if (params.usertalk) { var thispage = new Wikipedia.page(mw.config.get('wgPageName')); thispage.setCallbackParameters(params); thispage.lookupCreator(Twinkle.xfd.callbacks.afd.userNotification); }

// Remove some tags that should always be removed on AfD. text = text.replace(/\{\{\s*(dated prod|dated prod blp|Prod blp\/dated|Proposed deletion\/dated|prod2|Proposed deletion endorsed|New unreviewed article|Userspace draft)\s*(\|(?:\{\{[^{}]*\}\}|[^{}])*)?\}\}\s*/ig, ""); // Then, test if there are speedy deletion-related templates on the article. var textNoSd = text.replace(/\{\{\s*(db(-\w*)?|delete|(?:hang|hold)[\- ]?on)\s*(\|(?:\{\{[^{}]*\}\}|[^{}])*)?\}\}\s*/ig, ""); if (text !== textNoSd && confirm("A speedy deletion tag was found on this page. Should it be removed?")) { text = textNoSd; }

pageobj.setPageText((params.noinclude ? " \n") + text); pageobj.setEditSummary("Nominated for deletion; see " + params.discussionpage + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchPage')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('nocreate'); pageobj.save; },		discussionPage: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

pageobj.setPageText("\n"); pageobj.setEditSummary("Creating deletion discussion page for " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('createonly'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki },		todaysList: function(pageobj) { var old_text = pageobj.getPageText + "\n"; // MW strips trailing blanks, but we like them, so we add a fake one var params = pageobj.getCallbackParameters; var statelem = pageobj.getStatusElement;

var text = old_text.replace( /(<\!-- Add new entries to the TOP of the following list -->\n+)/, "$1\n"); if( text === old_text ) { statelem.error( 'failed to find target spot for the discussion' ); return; }			pageobj.setPageText(text); pageobj.setEditSummary("Adding " + params.discussionpage + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchList')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; },		userNotification: function(pageobj) { var params = pageobj.getCallbackParameters; var initialContrib = pageobj.getCreator; var usertalkpage = new Wikipedia.page('User talk:' + initialContrib, "Notifying initial contributor (" + initialContrib + ")"); var notifytext = "\n" + mw.config.get('wgPageName') + ( params.numbering !== '' ? ' ~";			usertalkpage.setAppendText(notifytext);			usertalkpage.setEditSummary("Notification: listing at articles for deletion of " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd'));			usertalkpage.setCreateOption('recreate');			switch (Twinkle.getPref('xfdWatchUser')) {				case 'yes':					usertalkpage.setWatchlist(true);					break;				case 'no':					usertalkpage.setWatchlistFromPreferences(false);					break;				default:					usertalkpage.setWatchlistFromPreferences(true);					break;			}			usertalkpage.setFollowRedirect(true);			usertalkpage.append;		}	},

tfd: { taggingTemplate: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

pageobj.setPageText((params.noinclude ? " \n") + text); pageobj.setEditSummary("Nominated for deletion; see " + params.logpage + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchPage')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('nocreate'); pageobj.save; },		todaysList: function(pageobj) { var old_text = pageobj.getPageText; var params = pageobj.getCallbackParameters; var statelem = pageobj.getStatusElement;

var text = old_text.replace( '-->', "-->\n" + mw.config.get('wgTitle') + ""); if( text === old_text ) { statelem.error( 'failed to find target spot for the discussion' ); return; }			pageobj.setPageText(text); pageobj.setEditSummary("Adding Template:" + mw.config.get('wgTitle') + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki },		userNotification: function(pageobj) { var initialContrib = pageobj.getCreator; var usertalkpage = new Wikipedia.page('User talk:' + initialContrib, "Notifying initial contributor (" + initialContrib + ")"); var notifytext = "\n" + mw.config.get('wgTitle') + " ~"; usertalkpage.setAppendText(notifytext); usertalkpage.setEditSummary("Notification: nomination at templates for discussion of " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); usertalkpage.setCreateOption('recreate'); switch (Twinkle.getPref('xfdWatchUser')) { case 'yes': usertalkpage.setWatchlist(true); break; case 'no': usertalkpage.setWatchlistFromPreferences(false); break; default: usertalkpage.setWatchlistFromPreferences(true); break; }			usertalkpage.setFollowRedirect(true); usertalkpage.append; }	},

mfd: { main: function(apiobj) { var xmlDoc = apiobj.responseXML; var titles = $(xmlDoc).find('allpages p');

// There has been no earlier entries with this prefix, just go on. if( titles.length <= 0 ) { apiobj.params.numbering = apiobj.params.number = ''; numbering = number = ''; } else { var number = 0; for( var i = 0; i < titles.length; ++i ) { var title = titles[i].getAttribute('title');

// First, simple test, is there an instance with this exact name? if( title === 'Wikipedia:Miscellany for deletion/' + mw.config.get('wgPageName') ) { number = Math.max( number, 1 ); continue; }

var order_re = new RegExp( '^' +							RegExp.escape( 'Wikipedia:Miscellany for deletion/' + mw.config.get('wgPageName'), true ) +							'\\s*\\(\\s*(\\d+)(?:(?:th|nd|rd|st) nom(?:ination)?)?\\s*\\)\\s*$' ); var match = order_re.exec( title );

// No match; A non-good value if( !match ) { continue; }

// A match, set number to the max of current number = Math.max( number, Number(match[1]) ); }				apiobj.params.number = Twinkle.xfd.num2order( parseInt( number, 10 ) + 1); apiobj.params.numbering = number > 0 ? ' (' + apiobj.params.number + ' nomination)' : ''; }			apiobj.params.discussionpage = "Wikipedia:Miscellany for deletion/" + mw.config.get('wgPageName') + apiobj.params.numbering;

apiobj.statelem.info( "next in order is " + apiobj.params.discussionpage + '');

// Tagging page var wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Tagging page with deletion tag"); wikipedia_page.setFollowRedirect(true); // should never be needed, but if the page is moved, we would want to follow the redirect wikipedia_page.setCallbackParameters(apiobj.params); wikipedia_page.load(Twinkle.xfd.callbacks.mfd.taggingPage);

// Updating data for the action completed event Wikipedia.actionCompleted.redirect = apiobj.params.discussionpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to the discussion page";

// Discussion page wikipedia_page = new Wikipedia.page(apiobj.params.discussionpage, "Creating deletion discussion page"); wikipedia_page.setCallbackParameters(apiobj.params); wikipedia_page.load(Twinkle.xfd.callbacks.mfd.discussionPage);

// Today's list wikipedia_page = new Wikipedia.page("Wikipedia:Miscellany for deletion", "Adding discussion to today's list"); //wikipedia_page.setPageSection(2); // pageSection has been disabled - the API seems to throw up with nonexistent edit conflicts // it can be turned on again once the problem is fixed, to save bandwidth //wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(apiobj.params); wikipedia_page.load(Twinkle.xfd.callbacks.mfd.todaysList);

// Notification to first contributor, and notification to owner of userspace (if applicable and required) if (apiobj.params.usertalk) { var thispage = new Wikipedia.page(mw.config.get('wgPageName')); thispage.setCallbackParameters(apiobj.params); thispage.lookupCreator(Twinkle.xfd.callbacks.mfd.userNotification); }		},		taggingPage: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

pageobj.setPageText((params.noinclude ? " " : "") + "\n" : ('mfdx|' + params.number + "}}\n")) + (params.noinclude ? " " : "") + text);			pageobj.setEditSummary("Nominated for deletion; see " + params.discussionpage + "." + Twinkle.getPref('summaryAd'));			switch (Twinkle.getPref('xfdWatchPage')) {				case 'yes':					pageobj.setWatchlist(true);					break;				case 'no':					pageobj.setWatchlistFromPreferences(false);					break;				default:					pageobj.setWatchlistFromPreferences(true);					break;			}			pageobj.setCreateOption('nocreate');			pageobj.save;		},		discussionPage: function(pageobj) {			var text = pageobj.getPageText;			var params = pageobj.getCallbackParameters;

pageobj.setPageText("\n"); pageobj.setEditSummary("Creating deletion discussion page for " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('createonly'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki },		todaysList: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters; var statelem = pageobj.getStatusElement;

var date = new Date; var date_header = "===" + date.getUTCMonthName + ' ' + date.getUTCDate + ', ' + date.getUTCFullYear + "===\n"; var date_header_regex = new RegExp( "(===\\s*" + date.getUTCMonthName + '\\s+' + date.getUTCDate + ',\\s+' + date.getUTCFullYear + "\\s*===)" ); var new_data = "";

if( date_header_regex.test( text ) ) { // we have a section already statelem.info( 'Found today\'s section, proceeding to add new entry' ); text = text.replace( date_header_regex, "$1\n" + new_data ); } else { // we need to create a new section statelem.info( 'No section for today found, proceeding to create one' ); text = text.replace("===", date_header + new_data + "\n\n==="); }

pageobj.setPageText(text); pageobj.setEditSummary("Adding " + params.discussionpage + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchList')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; },		userNotification: function(pageobj) { var initialContrib = pageobj.getCreator; var params = pageobj.getCallbackParameters;

// Really notify the creator Twinkle.xfd.callbacks.mfd.userNotificationMain(params, initialContrib, "Notifying initial contributor");

// Also notify the user who owns the subpage if they are not the creator if (params.notifyuserspace) { var userspaceOwner = ((mw.config.get('wgTitle').indexOf('/') === -1) ? mw.config.get('wgTitle') : mw.config.get('wgTitle').substring(0, mw.config.get('wgTitle').indexOf('/'))); if (userspaceOwner !== initialContrib) { Twinkle.xfd.callbacks.mfd.userNotificationMain(params, userspaceOwner, "Notifying owner of userspace"); }			}		},		userNotificationMain: function(params, initialContrib, actionName) {			var usertalkpage = new Wikipedia.page('User talk:' + initialContrib, actionName + " (" + initialContrib + ")"); var notifytext = "\n" + mw.config.get('wgPageName') + ( params.numbering !== '' ? ' ~";			usertalkpage.setAppendText(notifytext);			usertalkpage.setEditSummary("Notification: listing at miscellany for deletion of " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd'));			usertalkpage.setCreateOption('recreate');			switch (Twinkle.getPref('xfdWatchUser')) {				case 'yes':					usertalkpage.setWatchlist(true);					break;				case 'no':					usertalkpage.setWatchlistFromPreferences(false);					break;				default:					usertalkpage.setWatchlistFromPreferences(true);					break;			}			usertalkpage.setFollowRedirect(true);			usertalkpage.append;		}	},

ffd: { main: function(pageobj) { // this is coming in from lookupCreator...! var params = pageobj.getCallbackParameters; var initialContrib = pageobj.getCreator; params.uploader = initialContrib;

// Adding discussion wikipedia_page = new Wikipedia.page(params.logpage, "Adding discussion to today's list"); wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.ffd.todaysList);

// Notification to first contributor if(params.usertalk) { var usertalkpage = new Wikipedia.page('User talk:' + initialContrib, "Notifying initial contributor (" + initialContrib + ")"); var notifytext = "\n" + mw.config.get('wgTitle') + ""; usertalkpage.setAppendText(notifytext); usertalkpage.setEditSummary("Notification: listing at files for deletion of " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); usertalkpage.setCreateOption('recreate'); switch (Twinkle.getPref('xfdWatchUser')) { case 'yes': usertalkpage.setWatchlist(true); break; case 'no': usertalkpage.setWatchlistFromPreferences(false); break; default: usertalkpage.setWatchlistFromPreferences(true); break; }				usertalkpage.setFollowRedirect(true); usertalkpage.append; }		},		taggingImage: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

text = text.replace(/\{\{(mtc|(copy |move )?to ?commons|move to wikimedia commons|copy to wikimedia commons)[^}]*}}/gi, "");

pageobj.setPageText("\n" + text); pageobj.setEditSummary("Nominated for deletion; see " + params.logpage + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchPage')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); // it might be possible for a file to exist without a description page pageobj.save; },		todaysList: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

// add date header if the log is found to be empty (a bot should do this automatically, but it sometimes breaks down) if (!pageobj.exists) { text = ""; }

pageobj.setPageText(text + "\n" + mw.config.get('wgTitle') + " ~"); pageobj.setEditSummary("Adding " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki }	},

puf: { taggingImage: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

text = text.replace(/\{\{(mtc|(copy |move )?to ?commons|move to wikimedia commons|copy to wikimedia commons)[^}]*}}/gi, "");

pageobj.setPageText("\n" + text); pageobj.setEditSummary("Listed at possibly unfree files: " + params.logpage + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchPage')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); // it might be possible for a file to exist without a description page pageobj.save; },		todaysList: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

pageobj.setPageText(text + "\n ~"); pageobj.setEditSummary("Adding " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki },		userNotification: function(pageobj) { var initialContrib = pageobj.getCreator; var usertalkpage = new Wikipedia.page('User talk:' + initialContrib, "Notifying initial contributor (" + initialContrib + ")"); var notifytext = "\n" + mw.config.get('wgTitle') + " ~"; usertalkpage.setAppendText(notifytext); usertalkpage.setEditSummary("Notification: listing at possibly unfree files of " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); usertalkpage.setCreateOption('recreate'); switch (Twinkle.getPref('xfdWatchUser')) { case 'yes': usertalkpage.setWatchlist(true); break; case 'no': usertalkpage.setWatchlistFromPreferences(false); break; default: usertalkpage.setWatchlistFromPreferences(true); break; }			usertalkpage.setFollowRedirect(true); usertalkpage.append; }	},

cfd: { taggingCategory: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

var added_data = ""; var editsummary = ""; switch( params.xfdcat ) { case 'cfd': added_data = ""; editsummary = "Category being considered for deletion; see " + params.logpage + "."; break; case 'cfm': added_data = "" + params.target + ""; editsummary = "Category being considered for merging; see " + params.logpage + "."; break; case 'cfr': added_data = "" + params.target + ""; editsummary = "Category being considered for renaming; see " + params.logpage + "."; break; case 'cfs': added_data = "" + params.target + ""; editsummary = "Category being considered for splitting; see " + params.logpage + "."; break; case 'cfc': added_data = "" + params.target + ""; editsummary = "Category being considered for conversion to an article; see " + params.logpage + "."; break; default: alert("twinklexfd in taggingCategory: unknown CFD action"); break; }

pageobj.setPageText(added_data + "\n" + text); pageobj.setEditSummary(editsummary + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchPage')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); // since categories can be populated without an actual page at that title pageobj.save; },		todaysList: function(pageobj) { var old_text = pageobj.getPageText; var params = pageobj.getCallbackParameters; var statelem = pageobj.getStatusElement;

var added_data = ""; var editsummary = ""; switch( params.xfdcat ) { case 'cfd': added_data = "" + mw.config.get('wgTitle') + ""; editsummary = "Added delete nomination of " + mw.config.get('wgPageName') + "."; break; case 'cfm': added_data = "" + mw.config.get('wgTitle') + ""; editsummary = "Added merge nomination of " + mw.config.get('wgPageName') + "."; break; case 'cfr': added_data = "" + mw.config.get('wgTitle') + ""; editsummary = "Added rename nomination of " + mw.config.get('wgPageName') + "."; break; case 'cfs': added_data = "" + mw.config.get('wgTitle') + ""; editsummary = "Added split nomination of " + mw.config.get('wgPageName') + "."; break; case 'cfc': added_data = "" + mw.config.get('wgTitle') + ""; editsummary = "Added convert nomination of " + mw.config.get('wgPageName') + "."; break; default: alert("twinklexfd in todaysList: unknown CFD action"); break; }

text = old_text.replace( 'below this line -->', "below this line -->\n" + added_data ); if( text === old_text ) { statelem.error( 'failed to find target spot for the discussion' ); return; }

pageobj.setPageText(text); pageobj.setEditSummary(editsummary + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki },		userNotification: function(pageobj) { var initialContrib = pageobj.getCreator; var params = pageobj.getCallbackParameters; var usertalkpage = new Wikipedia.page('User talk:' + initialContrib, "Notifying initial contributor (" + initialContrib + ")"); var notifytext = "\n" + mw.config.get('wgPageName') + " ~"; usertalkpage.setAppendText(notifytext); usertalkpage.setEditSummary("Notification: listing at categories for discussion of " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); usertalkpage.setCreateOption('recreate'); switch (Twinkle.getPref('xfdWatchUser')) { case 'yes': usertalkpage.setWatchlist(true); break; case 'no': usertalkpage.setWatchlistFromPreferences(false); break; default: usertalkpage.setWatchlistFromPreferences(true); break; }			usertalkpage.setFollowRedirect(true); usertalkpage.append; }	},

cfds: { taggingCategory: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

pageobj.setPageText("" + params.target + "\n" + text); pageobj.setEditSummary("Nominated for speedy renaming; see Categories for discussion/Speedy." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchPage')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); // since categories can be populated without an actual page at that title pageobj.save; },		addToList: function(pageobj) { var old_text = pageobj.getPageText; var params = pageobj.getCallbackParameters; var statelem = pageobj.getStatusElement;

var newcatname = (/^Category:/.test(params.target) ? params.target : ("Category:" + params.target)); text = old_text.replace( 'BELOW THIS LINE -->', "BELOW THIS LINE -->\n* " + mw.config.get('wgPageName') + " to " +				newcatname + "\u00A0\u2013 " + params.xfdcat + (params.reason ? (": " + params.reason) : ".") + " ~" ); // U+00A0 NO-BREAK SPACE; U+2013 EN RULE if( text === old_text ) { statelem.error( 'failed to find target spot for the discussion' ); return; }

pageobj.setPageText(text); pageobj.setEditSummary("Adding " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki }	},

rfd: { // This is a callback from an API request, which gets the target of the redirect findTargetCallback: function(apiobj) { var xmlDoc = apiobj.responseXML; var target = $(xmlDoc).find('redirects r').first.attr('to'); if( !target ) { apiobj.statelem.error( "This page is currently not a redirect, aborting" ); return; }			apiobj.params.target = target; Twinkle.xfd.callbacks.rfd.main(apiobj.params); },		main: function(params) { var date = new Date; params.logpage = 'Wikipedia:Redirects for discussion/Log/' + date.getUTCFullYear + ' ' + date.getUTCMonthName + ' ' + date.getUTCDate;

// Tagging redirect var wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Adding deletion tag to redirect"); wikipedia_page.setFollowRedirect(false); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.rfd.taggingRedirect);

// Updating data for the action completed event Wikipedia.actionCompleted.redirect = params.logpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to today's log";

// Adding discussion wikipedia_page = new Wikipedia.page(params.logpage, "Adding discussion to today's log"); wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.rfd.todaysList);

// Notifying initial contributor if (params.usertalk) { var thispage = new Wikipedia.page(mw.config.get('wgPageName')); thispage.setCallbackParameters(params); thispage.lookupCreator(Twinkle.xfd.callbacks.rfd.userNotification); }		},		taggingRedirect: function(pageobj) { var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

pageobj.setPageText("\n" + text); pageobj.setEditSummary("Listed for discussion at " + params.logpage + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchPage')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('nocreate'); pageobj.save; },		todaysList: function(pageobj) { var old_text = pageobj.getPageText; var params = pageobj.getCallbackParameters; var statelem = pageobj.getStatusElement;

var text = old_text.replace( /(<\!-- Add new entries directly below this line -->)/, "$1\n ~\n" ); if( text === old_text ) { statelem.error( 'failed to find target spot for the discussion' ); return; }

pageobj.setPageText(text); pageobj.setEditSummary("Adding " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); switch (Twinkle.getPref('xfdWatchDiscussion')) { case 'yes': pageobj.setWatchlist(true); break; case 'no': pageobj.setWatchlistFromPreferences(false); break; default: pageobj.setWatchlistFromPreferences(true); break; }			pageobj.setCreateOption('recreate'); pageobj.save; Twinkle.xfd.currentRationale = null; // any errors from now on do not need to print the rationale, as it is safely saved on-wiki },		userNotification: function(pageobj) { var initialContrib = pageobj.getCreator; var usertalkpage = new Wikipedia.page('User talk:' + initialContrib, "Notifying initial contributor (" + initialContrib + ")"); var notifytext = "\n" + mw.config.get('wgPageName') + " ~"; usertalkpage.setAppendText(notifytext); usertalkpage.setEditSummary("Notification: listing at redirects for discussion of " + mw.config.get('wgPageName') + "." + Twinkle.getPref('summaryAd')); usertalkpage.setCreateOption('recreate'); switch (Twinkle.getPref('xfdWatchUser')) { case 'yes': usertalkpage.setWatchlist(true); break; case 'no': usertalkpage.setWatchlistFromPreferences(false); break; default: usertalkpage.setWatchlistFromPreferences(true); break; }			usertalkpage.setFollowRedirect(true); usertalkpage.append; }	} };

Twinkle.xfd.callback.evaluate = function(e) { mw.config.set('wgPageName', mw.config.get('wgPageName').replace(/_/g, ' ')); // for queen/king/whatever and country!

var type = e.target.category.value; var usertalk = e.target.notify.checked; var reason = e.target.xfdreason.value; var xfdcat, xfdtarget, xfdtarget2, puf, noinclude, tfdinline, notifyuserspace; if( type === "afd" || type === "cfd" || type === "cfds" ) { xfdcat = e.target.xfdcat.value; }	if( type === "cfd" || type === "cfds" ) { xfdtarget = e.target.xfdtarget.value; if (e.target.xfdtarget2) { xfdtarget2 = e.target.xfdtarget2.value; }	}	if( type === 'ffd' ) { puf = e.target.puf.checked; }	if( type === "afd" || type === "mfd" || type === "tfd" ) { noinclude = e.target.noinclude.checked; }	if( type === 'tfd' ) { tfdinline = e.target.tfdinline.checked; }	if( type === 'mfd' ) { notifyuserspace = e.target.notifyuserspace && e.target.notifyuserspace.checked; }

SimpleWindow.setButtonsEnabled( false ); Status.init( e.target );

Twinkle.xfd.currentRationale = reason; Status.onError(Twinkle.xfd.printRationale);

if( !type ) { Status.error( 'Error', 'no action given' ); return; }

var query, wikipedia_page, wikipedia_api, logpage, params; var date = new Date; switch( type ) {

case 'afd': // AFD query = { 'action': 'query', 'list': 'allpages', 'apprefix': 'Articles for deletion/' + mw.config.get('wgPageName'), 'apnamespace': 4, 'apfilterredir': 'nonredirects', 'aplimit': userIsInGroup( 'sysop' ) ? 5000 : 500		};		wikipedia_api = new Wikipedia.api( 'Tagging article with deletion tag', query, Twinkle.xfd.callbacks.afd.main ); wikipedia_api.params = { usertalk:usertalk, reason:reason, noinclude:noinclude, xfdcat:xfdcat }; wikipedia_api.post; break;

case 'tfd': // TFD Wikipedia.addCheckpoint;

logpage = 'Wikipedia:Templates for discussion/Log/' + date.getUTCFullYear + ' ' + date.getUTCMonthName + ' ' + date.getUTCDate;

// Tagging template wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Tagging template with deletion tag"); wikipedia_page.setFollowRedirect(true); // should never be needed, but if the page is moved, we would want to follow the redirect wikipedia_page.setCallbackParameters({ tfdinline: tfdinline, logpage: logpage, noinclude: noinclude }); wikipedia_page.load(Twinkle.xfd.callbacks.tfd.taggingTemplate);

// Updating data for the action completed event Wikipedia.actionCompleted.redirect = logpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to today's log";

// Adding discussion wikipedia_page = new Wikipedia.page(logpage, "Adding discussion to today's log"); wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters({ reason: reason }); wikipedia_page.load(Twinkle.xfd.callbacks.tfd.todaysList);

// Notification to first contributor if (usertalk) { var thispage = new Wikipedia.page(mw.config.get('wgPageName')); thispage.lookupCreator(Twinkle.xfd.callbacks.tfd.userNotification); }

Wikipedia.removeCheckpoint; break;

case 'mfd': // MFD query = { 'action': 'query', 'list': 'allpages', 'apprefix': 'Miscellany for deletion/' + mw.config.get('wgPageName'), 'apnamespace': 4, 'apfilterredir': 'nonredirects', 'aplimit': userIsInGroup( 'sysop' ) ? 5000 : 500		};		wikipedia_api = new Wikipedia.api( "Looking for prior nominations of this page", query, Twinkle.xfd.callbacks.mfd.main ); wikipedia_api.params = { usertalk: usertalk, notifyuserspace: notifyuserspace, reason: reason, noinclude: noinclude, xfdcat: xfdcat }; wikipedia_api.post; break;

case 'ffd': // FFD var dateString = date.getUTCFullYear + ' ' + date.getUTCMonthName + ' ' + date.getUTCDate; logpage = 'Wikipedia:Files for deletion/' + dateString; params = { usertalk: usertalk, reason: reason, date: dateString, logpage: logpage };

Wikipedia.addCheckpoint; if( puf ) { params.logpage = logpage = 'Wikipedia:Possibly unfree files/' + dateString;

// Updating data for the action completed event Wikipedia.actionCompleted.redirect = logpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to today's list";

// Tagging file wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Tagging file with PUF tag"); wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.puf.taggingImage);

// Adding discussion wikipedia_page = new Wikipedia.page(params.logpage, "Adding discussion to today's list"); wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.puf.todaysList);

// Notification to first contributor if (usertalk) { wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName')); wikipedia_page.setCallbackParameters(params); wikipedia_page.lookupCreator(Twinkle.xfd.callbacks.puf.userNotification); }

Wikipedia.removeCheckpoint;

} else { // Updating data for the action completed event Wikipedia.actionCompleted.redirect = logpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to the discussion page";

// Tagging file wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Adding deletion tag to file page"); wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.ffd.taggingImage);

// Contributor specific edits wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName')); wikipedia_page.setCallbackParameters(params); wikipedia_page.lookupCreator(Twinkle.xfd.callbacks.ffd.main); }		Wikipedia.removeCheckpoint; break;

case 'cfd': Wikipedia.addCheckpoint;

if( xfdtarget ) { xfdtarget = xfdtarget.replace( /^\:?Category\:/i, '' ); } else { xfdtarget = ''; }

if( xfdtarget2 ) { xfdtarget2 = xfdtarget2.replace( /^\:?Category\:/i, '' ); }

logpage = 'Wikipedia:Categories for discussion/Log/' + date.getUTCFullYear + ' ' + date.getUTCMonthName + ' ' + date.getUTCDate;

params = { reason: reason, xfdcat: xfdcat, target: xfdtarget, target2: xfdtarget2, logpage: logpage };

// Updating data for the action completed event Wikipedia.actionCompleted.redirect = logpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to today's log";

// Tagging category wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Tagging category with deletion tag"); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.cfd.taggingCategory);

// Adding discussion to list wikipedia_page = new Wikipedia.page(logpage, "Adding discussion to today's list"); //wikipedia_page.setPageSection(2); // pageSection has been disabled - the API seems to throw up with nonexistent edit conflicts // it can be turned on again once the problem is fixed, to save bandwidth //wikipedia_page.setFollowRedirect(true); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.cfd.todaysList);

// Notification to first contributor if (usertalk) { wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName')); wikipedia_page.setCallbackParameters(params); wikipedia_page.lookupCreator(Twinkle.xfd.callbacks.cfd.userNotification); }

Wikipedia.removeCheckpoint; break;

case 'cfds': xfdtarget = xfdtarget.replace( /^\:?Category\:/, '' );

logpage = "Wikipedia:Categories for discussion/Speedy"; params = { reason: reason, xfdcat: xfdcat, target: xfdtarget };

// Updating data for the action completed event Wikipedia.actionCompleted.redirect = logpage; Wikipedia.actionCompleted.notice = "Nomination completed, now redirecting to the discussion page";

// Tagging category wikipedia_page = new Wikipedia.page(mw.config.get('wgPageName'), "Tagging category with rename tag"); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.cfds.taggingCategory);

// Adding discussion to list wikipedia_page = new Wikipedia.page(logpage, "Adding discussion to the list"); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.xfd.callbacks.cfds.addToList);

break;

case 'rfd': params = { usertalk: usertalk, reason: reason }; if (document.getElementById("softredirect")) { // For soft redirects, skip straight to the callback params.target = document.getElementById("softredirect").textContent.replace(/^\:+/, ""); Twinkle.xfd.callbacks.rfd.main(params); } else { // Find current target of redirect query = { 'action': 'query', 'titles': mw.config.get('wgPageName'), 'redirects': true };			wikipedia_api = new Wikipedia.api( "Finding target of redirect", query, Twinkle.xfd.callbacks.rfd.findTargetCallback ); wikipedia_api.params = params; wikipedia_api.post; }		break; default: alert("twinklexfd: unknown XFD discussion venue"); break; } }; /** * General initialization code */

var scriptpathbefore = mw.util.wikiScript( "index" ) + "?title="; var scriptpathafter = "&action=raw&ctype=text/javascript&happy=yes";

// Retrieve the user's Twinkle preferences $.ajax({	url: scriptpathbefore + "User:" + encodeURIComponent( mw.config.get( "wgUserName" )) + "/twinkleoptions.js" + scriptpathafter,	dataType: "text",	error: function { jsMsg( "Could not load twinkleoptions.js" ); },	success: function ( optionsText ) {

// Quick pass if user has no options if ( optionsText === "" ) { return; }

// Twinkle options are basically a JSON object with some comments. Strip those: optionsText = optionsText.replace( /(?:^(?:\/\/[^\n]*\n)*\n*|(?:\/\/[^\n]*(?:\n|$))*$)/g, "" );

// First version of options had some boilerplate code to make it eval-able -- strip that too. This part may become obsolete down the line. if ( optionsText.lastIndexOf( "window.Twinkle.prefs = ", 0 ) === 0 ) { optionsText = optionsText.replace( /(?:^window.Twinkle.prefs = |;\n*$)/g, "" ); }

try { var options = JSON.parse( optionsText );

// Assuming that our options evolve, we will want to transform older versions: //if ( options.optionsVersion === undefined ) { // ...			// options.optionsVersion = 1; //}			//if ( options.optionsVersion === 1 ) { // ...			// options.optionsVersion = 2; //}			// At the same time, twinkleconfig.js needs to be adapted to write a higher version number into the options.

if ( options ) { Twinkle.prefs = options; }		}		catch ( e ) { jsMsg("Could not parse twinkleoptions.js"); }	},	complete: function { $( Twinkle.load ); } });

// Developers: you can import custom Twinkle modules here // For example, mw.loader.load(scriptpathbefore + "User:UncleDouggie/morebits-test.js" + scriptpathafter);

Twinkle.load = function { // Don't activate on special pages other than "Contributions" so that they load faster, especially the watchlist. // Also, Twinkle is incompatible with Internet Explorer versions 8 or lower, so don't load there either. if ( (mw.config.get('wgNamespaceNumber') === -1 && mw.config.get('wgCanonicalSpecialPageName') !== "Contributions" && mw.config.get('wgCanonicalSpecialPageName') !== "Prefixindex") ||		($.client.profile.name === 'msie' && $.client.profile.versionNumber < 9) ) { return; }

Twinkle.xfd;

// Run the initialization callbacks for any custom modules $( Twinkle.initCallbacks ).each(function ( k, v ) { v; }); Twinkle.addInitCallback = function ( func ) { func; };

// Increases text size in Twinkle dialogs, if so configured if ( Twinkle.getPref( "dialogLargeFont" ) ) { mw.util.addCSS( ".morebits-dialog-content, .morebits-dialog-footerlinks { font-size: 100% !important; } " +			".morebits-dialog input, .morebits-dialog select, .morebits-dialog-content button { font-size: inherit !important; }" ); } };

} ( jQuery )); // End wrap with anonymous function

//