User:94rain/Twinkle/modules/twinklespeedy.js

//

(function($) {

/* **************************************** *** twinklespeedy.js: CSD module **************************************** * Mode of invocation:    Tab ("CSD") * Active on:             Non-special, existing pages * Config directives in:  TwinkleConfig * * NOTE FOR DEVELOPERS: *  If adding a new criterion, add it to the appropriate places at the top of *   twinkleconfig.js. Also check out the default values of the CSD preferences *  in twinkle.js, and add your new criterion to those if you think it would be *   good. */

Twinkle.speedy = function twinklespeedy { // Disable on: // * special pages // * non-existent pages if (mw.config.get('wgNamespaceNumber') < 0 || !mw.config.get('wgArticleId')) { return; }

Twinkle.addPortletLink(Twinkle.speedy.callback, 'CSD', 'tw-csd', Morebits.userIsInGroup('sysop') ? 'Delete page according to WP:CSD' : 'Request speedy deletion according to WP:CSD'); };

// This function is run when the CSD tab/header link is clicked Twinkle.speedy.callback = function twinklespeedyCallback { Twinkle.speedy.initDialog(Morebits.userIsInGroup('sysop') ? Twinkle.speedy.callback.evaluateSysop : Twinkle.speedy.callback.evaluateUser, true); };

// Used by unlink feature Twinkle.speedy.dialog = null;

// The speedy criteria list can be in one of several modes Twinkle.speedy.mode = { sysopSingleSubmit: 1, // radio buttons, no subgroups, submit when "Submit" button is clicked sysopRadioClick: 2, // radio buttons, no subgroups, submit when a radio button is clicked sysopMultipleSubmit: 3, // check boxes, subgroups, "Submit" button already present sysopMultipleRadioClick: 4, // check boxes, subgroups, need to add a "Submit" button userMultipleSubmit: 5, // check boxes, subgroups, "Submit" button already pressent userMultipleRadioClick: 6, // check boxes, subgroups, need to add a "Submit" button userSingleSubmit: 7, // radio buttons, subgroups, submit when "Submit" button is clicked userSingleRadioClick: 8, // radio buttons, subgroups, submit when a radio button is clicked

// are we in "delete page" mode? // (sysops can access both "delete page" [sysop] and "tag page only" [user] modes) isSysop: function twinklespeedyModeIsSysop(mode) { return mode === Twinkle.speedy.mode.sysopSingleSubmit || mode === Twinkle.speedy.mode.sysopMultipleSubmit || mode === Twinkle.speedy.mode.sysopRadioClick || mode === Twinkle.speedy.mode.sysopMultipleRadioClick; },	// do we have a "Submit" button once the form is created? hasSubmitButton: function twinklespeedyModeHasSubmitButton(mode) { return mode === Twinkle.speedy.mode.sysopSingleSubmit || mode === Twinkle.speedy.mode.sysopMultipleSubmit || mode === Twinkle.speedy.mode.sysopMultipleRadioClick || mode === Twinkle.speedy.mode.userMultipleSubmit || mode === Twinkle.speedy.mode.userMultipleRadioClick || mode === Twinkle.speedy.mode.userSingleSubmit; },	// is db-multiple the outcome here? isMultiple: function twinklespeedyModeIsMultiple(mode) { return mode === Twinkle.speedy.mode.userMultipleSubmit || mode === Twinkle.speedy.mode.sysopMultipleSubmit || mode === Twinkle.speedy.mode.userMultipleRadioClick || mode === Twinkle.speedy.mode.sysopMultipleRadioClick; } };

// Prepares the speedy deletion dialog and displays it Twinkle.speedy.initDialog = function twinklespeedyInitDialog(callbackfunc) { var dialog; var isSysop = Morebits.userIsInGroup('sysop'); Twinkle.speedy.dialog = new Morebits.simpleWindow(Twinkle.getPref('speedyWindowWidth'), Twinkle.getPref('speedyWindowHeight')); dialog = Twinkle.speedy.dialog; dialog.setTitle('Choose criteria for speedy deletion'); dialog.setScriptName('Twinkle'); dialog.addFooterLink('Speedy deletion policy', 'WP:CSD'); dialog.addFooterLink('Twinkle help', 'WP:TW/DOC#speedy');

var form = new Morebits.quickForm(callbackfunc, Twinkle.getPref('speedySelectionStyle') === 'radioClick' ? 'change' : null); if (isSysop) { form.append({			type: 'checkbox',			list: [

{					label: 'Tag page only, don\'t delete', value: 'tag_only', name: 'tag_only', tooltip: 'If you just want to tag the page, instead of deleting it now', checked: Twinkle.getPref('deleteSysopDefaultToTag'), event: function(event) { var cForm = event.target.form; var cChecked = event.target.checked; // enable/disable talk page checkbox if (cForm.talkpage) { cForm.talkpage.disabled = cChecked; cForm.talkpage.checked = !cChecked && Twinkle.getPref('deleteTalkPageOnDelete'); }						// enable/disable redirects checkbox cForm.redirects.disabled = cChecked; cForm.redirects.checked = !cChecked; // enable/disable delete multiple cForm.delmultiple.disabled = cChecked; cForm.delmultiple.checked = false; // enable/disable notify checkbox cForm.notify.disabled = !cChecked; cForm.notify.checked = cChecked; // enable/disable deletion notification checkbox cForm.warnusertalk.disabled = cChecked; cForm.warnusertalk.checked = !cChecked; // enable/disable multiple cForm.multiple.disabled = !cChecked; cForm.multiple.checked = false;

Twinkle.speedy.callback.modeChanged(cForm);

event.stopPropagation; }				}			]		});

var deleteOptions = form.append({			type: 'div',			name: 'delete_options'		}); deleteOptions.append({			type: 'header',			label: 'Delete-related options'		}); if (mw.config.get('wgNamespaceNumber') % 2 === 0 && (mw.config.get('wgNamespaceNumber') !== 2 || (/\//).test(mw.config.get('wgTitle')))) { // hide option for user pages, to avoid accidentally deleting user talk page deleteOptions.append({				type: 'checkbox',				list: [					{						label: 'Also delete talk page',						value: 'talkpage',						name: 'talkpage',						tooltip: "This option deletes the page's talk page in addition. If you choose the F8 (moved to Commons) criterion, this option is ignored and the talk page is *not* deleted.",						checked: Twinkle.getPref('deleteTalkPageOnDelete'),						disabled: Twinkle.getPref('deleteSysopDefaultToTag'),						event: function(event) {							event.stopPropagation;						}					}				]			}); }		deleteOptions.append({			type: 'checkbox',			list: [				{					label: 'Also delete all redirects',					value: 'redirects',					name: 'redirects',					tooltip: 'This option deletes all incoming redirects in addition. Avoid this option for procedural (e.g. move/merge) deletions.',					checked: Twinkle.getPref('deleteRedirectsOnDelete'),					disabled: Twinkle.getPref('deleteSysopDefaultToTag'),					event: function(event) {						event.stopPropagation;					}				}			]		}); deleteOptions.append({			type: 'checkbox',			list: [				{					label: 'Delete under multiple criteria',					value: 'delmultiple',					name: 'delmultiple',					tooltip: 'When selected, you can select several criteria that apply to the page. For example, G11 and A7 are a common combination for articles.',					event: function(event) {						Twinkle.speedy.callback.modeChanged(event.target.form);						event.stopPropagation;					}				}			]		}); deleteOptions.append({			type: 'checkbox',			list: [				{					label: 'Notify page creator of page deletion',					value: 'warnusertalk',					name: 'warnusertalk',					tooltip: 'A notification template will be placed on the talk page of the creator, IF you have a notification enabled in your Twinkle preferences ' +						'for the criterion you choose AND this box is checked. The creator may be welcomed as well.',					checked: $('#delete-reason').length < 1,					event: function(event) {						event.stopPropagation;					}				}			]		}); }

var tagOptions = form.append({		type: 'div',		name: 'tag_options'	});

if (isSysop) { tagOptions.append({			type: 'header',			label: 'Tag-related options'		}); }

tagOptions.append({		type: 'checkbox',		list: [			{				label: 'Notify page creator if possible',				value: 'notify',				name: 'notify',				tooltip: 'A notification template will be placed on the talk page of the creator, IF you have a notification enabled in your Twinkle preferences ' +						'for the criterion you choose AND this box is checked. The creator may be welcomed as well.',				checked: !isSysop || Twinkle.getPref('deleteSysopDefaultToTag'),				disabled: isSysop && !Twinkle.getPref('deleteSysopDefaultToTag'),				event: function(event) {					event.stopPropagation;				}			}		]	}); tagOptions.append({		type: 'checkbox',		list: [			{				label: 'Tag with multiple criteria',				value: 'multiple',				name: 'multiple',				tooltip: 'When selected, you can select several criteria that apply to the page. For example, G11 and A7 are a common combination for articles.',				disabled: isSysop && !Twinkle.getPref('deleteSysopDefaultToTag'),				event: function(event) {					Twinkle.speedy.callback.modeChanged(event.target.form);					event.stopPropagation;				}			}		]	});

form.append({		type: 'div',		name: 'work_area',		label: 'Failed to initialize the CSD module. Please try again, or tell the Twinkle developers about the issue.'	});

if (Twinkle.getPref('speedySelectionStyle') !== 'radioClick') { form.append({ type: 'submit' }); }

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

Twinkle.speedy.callback.modeChanged(result); };

Twinkle.speedy.callback.getMode = function twinklespeedyCallbackGetMode(form) { var mode = Twinkle.speedy.mode.userSingleSubmit; if (form.tag_only && !form.tag_only.checked) { if (form.delmultiple.checked) { mode = Twinkle.speedy.mode.sysopMultipleSubmit; } else { mode = Twinkle.speedy.mode.sysopSingleSubmit; }	} else { if (form.multiple.checked) { mode = Twinkle.speedy.mode.userMultipleSubmit; } else { mode = Twinkle.speedy.mode.userSingleSubmit; }	}	if (Twinkle.getPref('speedySelectionStyle') === 'radioClick') { mode++; }

return mode; };

Twinkle.speedy.callback.modeChanged = function twinklespeedyCallbackModeChanged(form) { var namespace = mw.config.get('wgNamespaceNumber');

// first figure out what mode we're in	var mode = Twinkle.speedy.callback.getMode(form); var isSysopMode = Twinkle.speedy.mode.isSysop(mode);

if (isSysopMode) { $('[name=delete_options]').show; $('[name=tag_options]').hide; } else { $('[name=delete_options]').hide; $('[name=tag_options]').show; }

var work_area = new Morebits.quickForm.element({		type: 'div',		name: 'work_area'	});

if (mode === Twinkle.speedy.mode.userMultipleRadioClick || mode === Twinkle.speedy.mode.sysopMultipleRadioClick) { var evaluateType = isSysopMode ? 'evaluateSysop' : 'evaluateUser';

work_area.append({			type: 'div',			label: 'When finished choosing criteria, click:'		}); work_area.append({			type: 'button',			name: 'submit-multiple',			label: 'Submit Query',			event: function(event) {				Twinkle.speedy.callback[evaluateType](event);				event.stopPropagation;			}		}); }

var radioOrCheckbox = Twinkle.speedy.mode.isMultiple(mode) ? 'checkbox' : 'radio';

if (isSysopMode && !Twinkle.speedy.mode.isMultiple(mode)) { work_area.append({ type: 'header', label: 'Custom rationale' }); work_area.append({ type: radioOrCheckbox, name: 'csd', list: Twinkle.speedy.generateCsdList(Twinkle.speedy.customRationale, mode) }); }

if (namespace % 2 === 1 && namespace !== 3) { // show db-talk on talk pages, but not user talk pages work_area.append({ type: 'header', label: 'Talk pages' }); work_area.append({ type: radioOrCheckbox, name: 'csd', list: Twinkle.speedy.generateCsdList(Twinkle.speedy.talkList, mode) }); }

if (!mw.config.get('wgIsRedirect')) { switch (namespace) { case 1198: // Translation case 1199: // Translation talk work_area.append({ type: 'header', label: 'Translations' }); work_area.append({ type: radioOrCheckbox, name: 'csd', list: Twinkle.speedy.generateCsdList(Twinkle.speedy.translationList, mode) }); break;

default: break; }	} else { if (namespace === 2 || namespace === 3) { work_area.append({ type: 'header', label: 'User pages' }); work_area.append({ type: radioOrCheckbox, name: 'csd', list: Twinkle.speedy.generateCsdList(Twinkle.speedy.userList, mode) }); }		work_area.append({ type: 'header', label: 'Redirects' }); work_area.append({ type: radioOrCheckbox, name: 'csd', list: Twinkle.speedy.generateCsdList(Twinkle.speedy.redirectList, mode) }); }

var generalCriteria = Twinkle.speedy.generalList;

// custom rationale lives under general criteria when tagging if (!isSysopMode) { generalCriteria = Twinkle.speedy.customRationale.concat(generalCriteria); }	work_area.append({ type: 'header', label: 'General criteria' }); work_area.append({ type: radioOrCheckbox, name: 'csd', list: Twinkle.speedy.generateCsdList(generalCriteria, mode) });

var old_area = Morebits.quickForm.getElements(form, 'work_area')[0]; form.replaceChild(work_area.render, old_area);

// if sysop, check if CSD is already on the page and fill in custom rationale if (isSysopMode && $('#delete-reason').length) { var customOption = $('input[name=csd][value=reason]')[0]; if (customOption) { if (Twinkle.getPref('speedySelectionStyle') !== 'radioClick') { // force listeners to re-init customOption.click; customOption.parentNode.appendChild(customOption.subgroup); }			customOption.subgroup.querySelector('input').value = decodeURIComponent($('#delete-reason').text).replace(/\+/g, ' '); }	} };

Twinkle.speedy.generateCsdList = function twinklespeedyGenerateCsdList(list, mode) { // mode switches var isSysopMode = Twinkle.speedy.mode.isSysop(mode); var multiple = Twinkle.speedy.mode.isMultiple(mode); var hasSubmitButton = Twinkle.speedy.mode.hasSubmitButton(mode); var pageNamespace = mw.config.get('wgNamespaceNumber');

var openSubgroupHandler = function(e) { $(e.target.form).find('input').prop('disabled', true); $(e.target.form).children.css('color', 'gray'); $(e.target).parent.css('color', 'black').find('input').prop('disabled', false); $(e.target).parent.find('input:text')[0].focus; e.stopPropagation; };	var submitSubgroupHandler = function(e) { var evaluateType = Twinkle.speedy.mode.isSysop(mode) ? 'evaluateSysop' : 'evaluateUser'; Twinkle.speedy.callback[evaluateType](e); e.stopPropagation; };

return $.map(list, function(critElement) {		var criterion = $.extend({}, critElement);

if (multiple) { if (criterion.hideWhenMultiple) { return null; }			if (criterion.hideSubgroupWhenMultiple) { criterion.subgroup = null; }		} else { if (criterion.hideWhenSingle) { return null; }			if (criterion.hideSubgroupWhenSingle) { criterion.subgroup = null; }		}

if (isSysopMode) { if (criterion.hideWhenSysop) { return null; }			if (criterion.hideSubgroupWhenSysop) { criterion.subgroup = null; }		} else { if (criterion.hideWhenUser) { return null; }			if (criterion.hideSubgroupWhenUser) { criterion.subgroup = null; }		}

if (mw.config.get('wgIsRedirect') && criterion.hideWhenRedirect) { return null; }

if (criterion.showInNamespaces && criterion.showInNamespaces.indexOf(pageNamespace) < 0) { return null; }		if (criterion.hideInNamespaces && criterion.hideInNamespaces.indexOf(pageNamespace) > -1) { return null; }

if (criterion.subgroup && !hasSubmitButton) { if (Array.isArray(criterion.subgroup)) { criterion.subgroup = criterion.subgroup.concat({					type: 'button',					name: 'submit',					label: 'Submit Query',					event: submitSubgroupHandler				}); } else { criterion.subgroup = [ criterion.subgroup, {						type: 'button', name: 'submit', // ends up being called "csd.submit" so this is OK						label: 'Submit Query', event: submitSubgroupHandler }				];			}			// FIXME: does this do anything? criterion.event = openSubgroupHandler; }

return criterion; }); };

Twinkle.speedy.customRationale = [ {		label: 'Custom rationale' + (Morebits.userIsInGroup('sysop') ? ' (custom deletion reason)' : ' using template'), value: 'reason', tooltip: ' is short for "delete because". At least one of the other deletion criteria must still apply to the page, and you must make mention of this in your rationale. This is not a "catch-all" for when you can\'t find any criteria that fit.', subgroup: { name: 'reason_1', type: 'input', label: 'Rationale: ', size: 60 },		hideWhenMultiple: true } ];

Twinkle.speedy.translationList = [ {		label: 'not a translation', value: 'not a translation', tooltip: '' },	{		label: 'not a translation, original text', value: 'not a translation, original text', tooltip: '' },	{		label: 'not a translation, please mass delete', value: 'not a translation, please mass delete', tooltip: '' },	{		label: 'not a documentation', value: 'not a documentation', tooltip: '' } ];

Twinkle.speedy.generalList = [ {		label: 'Test or nonsense', value: 'test page', tooltip: 'Does not apply to user page', hideInNamespaces: [ 2 ] // Not applicable in userspace },	{		label: 'Out of project scope', value: 'Out of project scope', tooltip: '' },	{		label: 'Spam or promotional purposes', value: 'Spam or promotional purposes', tooltip: '' },	{		label: 'Spam or nonsense', value: 'Spam or nonsense', tooltip: '' },	{		label: 'Vandalism', value: 'Vandalism', tooltip: '' },	{		label: 'Cross-wiki vandalism', value: 'Cross-wiki vandalism', tooltip: '' } ];

Twinkle.speedy.redirectList = [ {		label: 'Redirects to invalid targets, such as nonexistent targets, redirect loops, and bad titles', value: 'redirnone', tooltip: 'This excludes any page that is useful to the project, and in particular: deletion discussions that are not logged elsewhere, user and user talk pages, talk page archives, plausible redirects that can be changed to valid targets, and file pages or talk pages for files that exist on Wikimedia Commons.', hideWhenMultiple: true } ];

Twinkle.speedy.normalizeHash = { 'reason': 'db', 'test page': 'test page', 'Out of project scope': 'Out of project scope', 'not a translation': 'not a translation', 'not a translation, original text': 'not a translation, original text', 'not a translation, please mass delete': 'not a translation, please mass delete', 'not a documentation': 'not a translation, please mass delete', 'Spam or promotional purposes': 'Spam or promotional purposes', 'Cross-wiki vandalism': 'Cross-wiki vandalism', 'Vandalism': 'Vandalism', 'Spam or nonsense': 'Spam or nonsense',

};

Twinkle.speedy.callbacks = { getTemplateCodeAndParams: function(params) { var code, parameters, i;		if (params.normalizeds.length > 1) { code = ''; params.utparams = {}; $.each(params.normalizeds, function(index, norm) {				code += ',' + norm.toLowerCase;				parameters = params.templateParams[index]';		} else {			parameters = params.templateParams[0] || [];			code =  + params.values[0];			for (i in parameters) {				if (typeof parameters[i] === 'string') {					code += ',' + i + '=' + parameters[i];				}			}			if (params.usertalk) {				code += ;			}			code += '';			params.utparams = Twinkle.speedy.getUserTalkParameters(params.normalizeds[0], parameters);		}

return [code, params.utparams]; },

parseWikitext: function(wikitext, callback) { var query = { action: 'parse', prop: 'text', pst: 'true', text: wikitext, contentmodel: 'wikitext', title: mw.config.get('wgPageName') };

var statusIndicator = new Morebits.status('Building deletion summary'); var api = new Morebits.wiki.api('Parsing deletion template', query, function(apiObj) {			var reason = decodeURIComponent($(apiObj.getXML.querySelector('text').childNodes[0].nodeValue).find('#delete-reason').text).replace(/\+/g, ' ');			if (!reason) {				statusIndicator.warn('Unable to generate summary from deletion template');			} else {				statusIndicator.info('complete');			}			callback(reason);		}, statusIndicator); api.post; },

noteToCreator: function(pageobj) { var params = pageobj.getCallbackParameters; var initialContrib = pageobj.getCreator;

// disallow notifying yourself if (initialContrib === mw.config.get('wgUserName')) { Morebits.status.warn('You (' + initialContrib + ') created this page; skipping user notification'); initialContrib = null;

// don't notify users when their user talk page is nominated/deleted } else if (initialContrib === mw.config.get('wgTitle') && mw.config.get('wgNamespaceNumber') === 3) { Morebits.status.warn('Notifying initial contributor: this user created their own user talk page; skipping notification'); initialContrib = null;

// quick hack to prevent excessive unwanted notifications, per request. Should actually be configurable on recipient page... } else if ((initialContrib === 'Cyberbot I' || initialContrib === 'SoxBot') && params.normalizeds[0] === 'f2') { Morebits.status.warn('Notifying initial contributor: page created procedurally by bot; skipping notification'); initialContrib = null;

// Check for already existing tags } else if ($('#delete-reason').length && params.warnUser && !confirm('The page is has a deletion-related tag, and thus the creator has likely been notified. Do you want to notify them for this deletion as well?')) { Morebits.status.info('Notifying initial contributor', 'canceled by user; skipping notification.'); initialContrib = null; } else { var usertalkpage = new Morebits.wiki.page('User talk:' + initialContrib, 'Notifying initial contributor (' + initialContrib + ')'), notifytext, i, editsummary;

// special cases: "db" and "db-multiple" if (params.normalizeds.length > 1) { notifytext = '\n{{subst:db-' + (params.warnUser ? 'deleted' : 'notice') + '-multiple|1=' + Morebits.pageNameNorm; var count = 2; $.each(params.normalizeds, function(index, norm) {					notifytext += '|' + count++ + '=' + norm.toLowerCase;				}); } else if (params.normalizeds[0] === 'db') { notifytext = '\n{{subst:db-reason-' + (params.warnUser ? 'deleted' : 'notice') + '|1=' + Morebits.pageNameNorm; } else { notifytext = '\n{{subst:db-csd-' + (params.warnUser ? 'deleted' : 'notice') + '-custom|1=' + Morebits.pageNameNorm + '|2=' + params.values[0]; }

for (i in params.utparams) { if (typeof params.utparams[i] === 'string') { notifytext += '|' + i + '=' + params.utparams[i]; }			}			notifytext += (params.welcomeuser ? '' : '|nowelcome=yes') + '}} ~';

editsummary = 'Notification: speedy deletion' + (params.warnUser ? '' : ' nomination'); if (params.normalizeds.indexOf('g10') === -1) { // no article name in summary for G10 taggings editsummary += ' of ' + Morebits.pageNameNorm + '.'; } else { editsummary += ' of an attack page.'; }

usertalkpage.setAppendText(notifytext); usertalkpage.setEditSummary(editsummary + Twinkle.getPref('summaryAd')); usertalkpage.setCreateOption('recreate'); usertalkpage.setFollowRedirect(true); usertalkpage.append(function onNotifySuccess {				// add this nomination to the user's userspace log, if the user has enabled it				if (params.lognomination) {					Twinkle.speedy.callbacks.user.addToLog(params, initialContrib);				}			}, function onNotifyError {				// if user could not be notified, log nomination without mentioning that notification was sent				if (params.lognomination) {					Twinkle.speedy.callbacks.user.addToLog(params, null);				}			}); }	},

sysop: { main: function(params) { var reason; if (!params.normalizeds.length && params.normalizeds[0] === 'db') { reason = prompt('Enter the deletion summary to use, which will be entered into the deletion log:', ''); Twinkle.speedy.callbacks.sysop.deletePage(reason, params); } else { var code = Twinkle.speedy.callbacks.getTemplateCodeAndParams(params)[0]; Twinkle.speedy.callbacks.parseWikitext(code, function(reason) {					if (params.promptForSummary) {						reason = prompt('Enter the deletion summary to use, or press OK to accept the automatically generated one.', reason);					}					Twinkle.speedy.callbacks.sysop.deletePage(reason, params);				}); }		},		deletePage: function(reason, params) { var thispage = new Morebits.wiki.page(mw.config.get('wgPageName'), 'Deleting page');

if (reason === null) { return Morebits.status.error('Asking for reason', 'User cancelled'); } else if (!reason || !reason.replace(/^\s*/, ).replace(/\s*$/, )) { return Morebits.status.error('Asking for reason', "you didn't give one. I don't know... what with admins and their apathetic antics... I give up..."); }			thispage.setEditSummary(reason + Twinkle.getPref('deletionSummaryAd')); thispage.deletePage(function {				thispage.getStatusElement.info('done');				Twinkle.speedy.callbacks.sysop.deleteTalk(params);			});

// look up initial contributor. If prompting user for deletion reason, just display a link. // Otherwise open the talk page directly if (params.warnUser) { thispage.setCallbackParameters(params); thispage.lookupCreation(Twinkle.speedy.callbacks.noteToCreator); }		},		deleteTalk: function(params) { // delete talk page if (params.deleteTalkPage &&					params.normalized !== 'f8' &&					document.getElementById('ca-talk').className !== 'new') { var talkpage = new Morebits.wiki.page(mw.config.get('wgFormattedNamespaces')[mw.config.get('wgNamespaceNumber') + 1] + ':' + mw.config.get('wgTitle'), 'Deleting talk page'); talkpage.setEditSummary('Talk page of deleted page "' + Morebits.pageNameNorm + '"' + Twinkle.getPref('deletionSummaryAd')); talkpage.deletePage; // this is ugly, but because of the architecture of wiki.api, it is needed // (otherwise success/failure messages for the previous action would be suppressed) window.setTimeout(function {					Twinkle.speedy.callbacks.sysop.deleteRedirects(params);				}, 1800); } else { Twinkle.speedy.callbacks.sysop.deleteRedirects(params); }		},		deleteRedirects: function(params) { // delete redirects if (params.deleteRedirects) { var query = { 'action': 'query', 'titles': mw.config.get('wgPageName'), 'prop': 'redirects', 'rdlimit': 5000 // 500 is max for normal users, 5000 for bots and sysops };				var wikipedia_api = new Morebits.wiki.api('getting list of redirects...', query, Twinkle.speedy.callbacks.sysop.deleteRedirectsMain,					new Morebits.status('Deleting redirects')); wikipedia_api.params = params; wikipedia_api.post; }

// promote Unlink tool var $link, $bigtext; if (mw.config.get('wgNamespaceNumber') === 6 && params.normalized !== 'f8') { $link = $('', {					'href': '#',					'text': 'click here to go to the Unlink tool',					'css': { 'fontSize': '130%', 'fontWeight': 'bold' },					'click': function {						Morebits.wiki.actionCompleted.redirect = null;						Twinkle.speedy.dialog.close;						Twinkle.unlink.callback('Removing usages of and/or links to deleted file ' + Morebits.pageNameNorm);					}				}); $bigtext = $(' ', {					'text': 'To orphan backlinks and remove instances of file usage',					'css': { 'fontSize': '130%', 'fontWeight': 'bold' }				}); Morebits.status.info($bigtext[0], $link[0]); } else if (params.normalized !== 'f8') { $link = $('', {					'href': '#',					'text': 'click here to go to the Unlink tool',					'css': { 'fontSize': '130%', 'fontWeight': 'bold' },					'click': function {						Morebits.wiki.actionCompleted.redirect = null;						Twinkle.speedy.dialog.close;						Twinkle.unlink.callback('Removing links to deleted page ' + Morebits.pageNameNorm);					}				}); $bigtext = $(' ', {					'text': 'To orphan backlinks',					'css': { 'fontSize': '130%', 'fontWeight': 'bold' }				}); Morebits.status.info($bigtext[0], $link[0]); }		},		deleteRedirectsMain: function(apiobj) { var xmlDoc = apiobj.getXML; var $snapshot = $(xmlDoc).find('redirects rd'); var total = $snapshot.length; var statusIndicator = apiobj.statelem;

if (!total) { statusIndicator.status('no redirects found'); return; }

statusIndicator.status('0%');

var current = 0; var onsuccess = function(apiobjInner) { var now = parseInt(100 * ++current / total, 10) + '%'; statusIndicator.update(now); apiobjInner.statelem.unlink; if (current >= total) { statusIndicator.info(now + ' (completed)'); Morebits.wiki.removeCheckpoint; }			};

Morebits.wiki.addCheckpoint;

$snapshot.each(function(key, value) {				var title = $(value).attr('title');				var page = new Morebits.wiki.page(title, 'Deleting redirect "' + title + '"');				page.setEditSummary('G8: Redirect to deleted page "' + Morebits.pageNameNorm + '"' + Twinkle.getPref('deletionSummaryAd'));				page.deletePage(onsuccess);			}); }	},

user: { main: function(pageobj) { var statelem = pageobj.getStatusElement;

// defaults to /doc for lua modules, which may not exist if (!pageobj.exists && mw.config.get('wgPageContentModel') !== 'Scribunto') { statelem.error("It seems that the page doesn't exist; perhaps it has already been deleted"); return; }

var text = pageobj.getPageText; var params = pageobj.getCallbackParameters;

statelem.status('Checking for tags on the page...');

// check for existing deletion tags var tag = /(?:\{\{\s*(db|delete|db-.*?|speedy deletion-.*?)(?:\s*\||\s*\}\}))/.exec(text); // This won't make use of the db-multiple template but it probably should if (tag && !confirm('The page already has the CSD-related template on it. Do you want to add another CSD template?')) { return; }

var xfd = /\{\{((?:article for deletion|proposed deletion|prod blp|template for discussion)\/dated|[cfm]fd\b)/i.exec(text) || /#invoke:(RfD)/.exec(text); if (xfd && !confirm('The deletion-related template was found on the page. Do you still want to add a CSD template?')) { return; }

// given the params, builds the template and also adds the user talk page parameters to the params that were passed in			// returns => [ wikitext, utparams] var buildData = Twinkle.speedy.callbacks.getTemplateCodeAndParams(params), code = buildData[0]; params.utparams = buildData[1];

var thispage = new Morebits.wiki.page(mw.config.get('wgPageName')); // patrol the page, if reached from Special:NewPages if (Twinkle.getPref('markSpeedyPagesAsPatrolled')) { thispage.patrol; }

// Wrap SD template in noinclude tags if we are in template space. // Won't work with userboxes in userspace, or any other transcluded page outside template space if (mw.config.get('wgNamespaceNumber') === 10) { // Template: code = ' ' + code + ' '; }

// Remove tags that become superfluous with this action text = text.replace(/\{\{\s*([Uu]serspace draft)\s*(\|(?:\{\{[^{}]*\}\}|[^{}])*)?\}\}\s*/g, ''); if (mw.config.get('wgNamespaceNumber') === 6) { // remove "move to Commons" tag - deletion-tagged files cannot be moved to Commons text = text.replace(/\{\{(mtc|(copy |move )?to ?commons|move to wikimedia commons|copy to wikimedia commons)[^}]*\}\}/gi, ''); }

// Generate edit summary for edit var editsummary; if (params.normalizeds.length > 1) { editsummary = 'Requesting speedy deletion (';				$.each(params.normalizeds, function(index, norm) { editsummary += norm.toLowerCase; });				editsummary = editsummary.substr(0, editsummary.length - 2); // remove trailing comma				editsummary += ').'; } else if (params.normalizeds[0] === 'db') { editsummary = 'Requesting speedy deletion: ' + params.templateParams[0]['1'] + ' '; } else { editsummary = 'Requesting speedy deletion: ' + params.normalizeds[0].toLowerCase; }

pageobj.setPageText(code + (params.normalizeds.indexOf('g10') !== -1 ? '' : '\n' + text)); // cause attack pages to be blanked pageobj.setEditSummary(editsummary + Twinkle.getPref('summaryAd')); pageobj.setWatchlist(params.watch); pageobj.setCreateOption('recreate'); // Module /doc might not exist pageobj.save(Twinkle.speedy.callbacks.user.tagComplete); },

tagComplete: function(pageobj) { var params = pageobj.getCallbackParameters;

// Notification to first contributor, will also log nomination to the user's userspace log if (params.usertalk) { var thispage = new Morebits.wiki.page(Morebits.pageNameNorm); thispage.setCallbackParameters(params); thispage.lookupCreation(Twinkle.speedy.callbacks.noteToCreator); // or, if not notifying, add this nomination to the user's userspace log without the initial contributor's name } else if (params.lognomination) { Twinkle.speedy.callbacks.user.addToLog(params, null); }		},

// note: this code is also invoked from twinkleimage // the params used are: //  for CSD: params.values, params.normalizeds  (note: normalizeds is an array) //  for DI: params.fromDI = true, params.templatename, params.normalized  (note: normalized is a string) addToLog: function(params, initialContrib) { var wikipedia_page = new Morebits.wiki.page('User:' + mw.config.get('wgUserName') + '/' + Twinkle.getPref('speedyLogPageName'), 'Adding entry to userspace log'); params.logInitialContrib = initialContrib; wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.speedy.callbacks.user.saveLog); },

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

var appendText = '';

// add blurb if log page doesn't exist if (!pageobj.exists) { appendText += "This is a log of all speedy deletion nominations made by this user using Twinkle's CSD module.\n\n" + 'If you no longer wish to keep this log, you can turn it off using the preferences panel, and ' + 'nominate this page for speedy deletion under CSD U1.'; if (Morebits.userIsInGroup('sysop')) { appendText += '\n\nThis log does not track outright speedy deletions made using Twinkle.'; }			}

// create monthly header var date = new Date; var headerRe = new RegExp('^==+\\s*' + date.getUTCMonthName + '\\s+' + date.getUTCFullYear + '\\s*==+', 'm'); if (!headerRe.exec(text)) { appendText += '\n\n=== ' + date.getUTCMonthName + ' ' + date.getUTCFullYear + ' ==='; }

var formatParamLog = function(normalize, csdparam, input) { if ((normalize === 'G4' && csdparam === 'xfd') || (normalize === 'G6' && csdparam === 'page') || (normalize === 'G6' && csdparam === 'fullvotepage') || (normalize === 'G6' && csdparam === 'sourcepage')					|| (normalize === 'A2' && csdparam === 'source') || (normalize === 'A10' && csdparam === 'article') || (normalize === 'F5' && csdparam === 'replacement')) { input =  + input + ; } else if (normalize === 'G5' && csdparam === 'user') { input = 'User:' + input + ''; } else if (normalize === 'G12' && csdparam.lastIndexOf('url', 0) === 0 && input.lastIndexOf('http', 0) === 0) { input = '[' + input + ' ' + input + ']'; } else if (normalize === 'F1' && csdparam === 'filename') { input = 'File:' + input + ''; } else if (normalize === 'T3' && csdparam === 'template') { input = 'Template:' + input + ''; } else if (normalize === 'F8' && csdparam === 'filename') { input =  + input + ; } else if (normalize === 'P1' && csdparam === 'criterion') { input = 'WP:CSD'; }				return ' {' + normalize + ' ' + csdparam + ': ' + input + '}'; };

var extraInfo = '';

// If a logged file is deleted but exists on commons, the wikilink will be blue, so provide a link to the log var fileLogLink = mw.config.get('wgNamespaceNumber') === 6 ? ' ([ log])' : '';

var editsummary = 'Logging speedy deletion nomination'; appendText += '\n# [[:' + Morebits.pageNameNorm;

if (params.fromDI) { appendText += ']]' + fileLogLink + params.normalized.toLowerCase; // The params data structure when coming from DI is quite different, // so this hardcodes the only interesting items worth logging ['reason', 'replacement', 'source'].forEach(function(item) {					if (params[item]) {						extraInfo += formatParamLog(params.normalized.toLowerCase, item, params[item]);						return false;					}				}); editsummary += ' of ' + Morebits.pageNameNorm + '.'; } else { if (params.normalizeds.indexOf('g10') === -1) { // no article name in log for G10 taggings appendText += ']]' + fileLogLink + ': '; editsummary += ' of ' + Morebits.pageNameNorm + '.'; } else { appendText += '|This]] attack page' + fileLogLink + ': '; editsummary += ' of an attack page.'; }				if (params.normalizeds.length > 1) { appendText += 'multiple criteria (';					$.each(params.normalizeds, function(index, norm) { appendText += norm.toLowerCase; });					appendText = appendText.substr(0, appendText.length - 2); // remove trailing comma					appendText += ')'; } else if (params.normalizeds[0] === 'db') { appendText += 'db-reason'; } else { appendText += params.normalizeds[0].toLowerCase; }

// If params is "empty" it will still be full of empty arrays, but ask anyway if (params.templateParams) { // Treat custom rationale individually if (params.normalizeds[0] && params.normalizeds[0] === 'db') { extraInfo += formatParamLog('Custom', 'rationale', params.templateParams[0]['1']); } else { params.templateParams.forEach(function(item, index) {							var keys = Object.keys(item);							if (keys[0] !== undefined && keys[0].length > 0) {								// Second loop required since some items (G12, F9) may have multiple keys								keys.forEach(function(key, keyIndex) { if (keys[keyIndex] === 'blanked' || keys[keyIndex] === 'ts') { return true; // Not worth logging }									extraInfo += formatParamLog(params.normalizeds[index].toLowerCase, keys[keyIndex], item[key]); });							}						});					}				}			}

if (extraInfo) { appendText += '; additional information:' + extraInfo; }			if (params.logInitialContrib) { appendText += '; notified '; }			appendText += ' \n';

pageobj.setAppendText(appendText); pageobj.setEditSummary(editsummary + Twinkle.getPref('summaryAd')); pageobj.setCreateOption('recreate'); pageobj.append; }	} };

// validate subgroups in the form passed into the speedy deletion tag Twinkle.speedy.getParameters = function twinklespeedyGetParameters(form, values) { var parameters = [];

$.each(values, function(index, value) {		var currentParams = [];		switch (value) {			case 'reason':				if (form['csd.reason_1']) {					var dbrationale = form['csd.reason_1'].value;					if (!dbrationale || !dbrationale.trim) {						alert('Custom rationale: Please specify a rationale.');						parameters = null;						return false;					}					currentParams['1'] = dbrationale;				}				break;

case 'userreq': // U1				if (form['csd.userreq_rationale']) { var u1rationale = form['csd.userreq_rationale'].value; if (mw.config.get('wgNamespaceNumber') === 3 && !(/\//).test(mw.config.get('wgTitle')) &&							(!u1rationale || !u1rationale.trim)) { alert('CSD U1: Please specify a rationale when nominating user talk pages.'); parameters = null; return false; }					currentParams.rationale = u1rationale; }				break;

default: break; }		parameters.push(currentParams); });	return parameters; };

// function for processing talk page notification template parameters Twinkle.speedy.getUserTalkParameters = function twinklespeedyGetUserTalkParameters(normalized, parameters) { var utparams = []; switch (normalized) { case 'db': utparams['2'] = parameters['1']; break; default: break; }	return utparams; };

Twinkle.speedy.resolveCsdValues = function twinklespeedyResolveCsdValues(e) { var values = (e.target.form ? e.target.form : e.target).getChecked('csd'); if (values.length === 0) { alert('Please select a criterion!'); return null; }	return values; };

Twinkle.speedy.callback.evaluateSysop = function twinklespeedyCallbackEvaluateSysop(e) { var form = e.target.form ? e.target.form : e.target;

if (e.target.type === 'checkbox' || e.target.type === 'text' ||			e.target.type === 'select') { return; }

var tag_only = form.tag_only; if (tag_only && tag_only.checked) { Twinkle.speedy.callback.evaluateUser(e); return; }

var values = Twinkle.speedy.resolveCsdValues(e); if (!values) { return; }

var normalizeds = values.map(function(value) {		return Twinkle.speedy.normalizeHash[value];	});

// analyse each criterion to determine whether to watch the page, prompt for summary, or notify the creator var watchPage, promptForSummary; normalizeds.forEach(function(norm) {		if (Twinkle.getPref('watchSpeedyPages').indexOf(norm) !== -1) {			watchPage = true;		}		if (Twinkle.getPref('promptForSpeedyDeletionSummary').indexOf(norm) !== -1) {			promptForSummary = true;		}	});

var warnusertalk = false; if (form.warnusertalk.checked) { $.each(normalizeds, function(index, norm) {			if (Twinkle.getPref('warnUserOnSpeedyDelete').indexOf(norm) !== -1) {				if (norm === 'g6' && values[index] !== 'copypaste') {					return true;				}				warnusertalk = true;				return false; // break			}		}); }

var welcomeuser = false; if (warnusertalk) { $.each(normalizeds, function(index, norm) {			if (Twinkle.getPref('welcomeUserOnSpeedyDeletionNotification').indexOf(norm) !== -1) {				welcomeuser = true;				return false; // break			}		}); }

var params = { values: values, normalizeds: normalizeds, watch: watchPage, deleteTalkPage: form.talkpage && form.talkpage.checked, deleteRedirects: form.redirects.checked, warnUser: warnusertalk, welcomeuser: welcomeuser, promptForSummary: promptForSummary, templateParams: Twinkle.speedy.getParameters(form, values) };	if (!params.templateParams) { return; }

Morebits.simpleWindow.setButtonsEnabled(false); Morebits.status.init(form);

Twinkle.speedy.callbacks.sysop.main(params); };

Twinkle.speedy.callback.evaluateUser = function twinklespeedyCallbackEvaluateUser(e) { var form = e.target.form ? e.target.form : e.target;

if (e.target.type === 'checkbox' || e.target.type === 'text' ||			e.target.type === 'select') { return; }

var values = Twinkle.speedy.resolveCsdValues(e); if (!values) { return; }	// var multiple = form.multiple.checked; var normalizeds = []; $.each(values, function(index, value) {		var norm = Twinkle.speedy.normalizeHash[value];

normalizeds.push(norm); });

// analyse each criterion to determine whether to watch the page/notify the creator var watchPage = false; $.each(normalizeds, function(index, norm) {		if (Twinkle.getPref('watchSpeedyPages').indexOf(norm) !== -1) {			watchPage = true;			return false; // break		}	});

var notifyuser = false; if (form.notify.checked) { $.each(normalizeds, function(index, norm) {			if (Twinkle.getPref('notifyUserOnSpeedyDeletionNomination').indexOf(norm) !== -1) {				if (norm === 'g6' && values[index] !== 'copypaste') {					return true;				}				notifyuser = true;				return false; // break			}		}); }

var welcomeuser = false; if (notifyuser) { $.each(normalizeds, function(index, norm) {			if (Twinkle.getPref('welcomeUserOnSpeedyDeletionNotification').indexOf(norm) !== -1) {				welcomeuser = true;				return false; // break			}		}); }

var csdlog = false; if (Twinkle.getPref('logSpeedyNominations')) { $.each(normalizeds, function(index, norm) {			if (Twinkle.getPref('noLogOnSpeedyNomination').indexOf(norm) === -1) {				csdlog = true;				return false; // break			}		}); }

var params = { values: values, normalizeds: normalizeds, watch: watchPage, usertalk: notifyuser, welcomeuser: welcomeuser, lognomination: csdlog, templateParams: Twinkle.speedy.getParameters(form, values) };	if (!params.templateParams) { return; }

Morebits.simpleWindow.setButtonsEnabled(false); Morebits.status.init(form);

Morebits.wiki.actionCompleted.redirect = mw.config.get('wgPageName'); Morebits.wiki.actionCompleted.notice = 'Tagging complete';

// Modules can't be tagged, follow standard at TfD and place on /doc subpage var isScribunto = mw.config.get('wgPageContentModel') === 'Scribunto'; var wikipedia_page = isScribunto ? new Morebits.wiki.page(mw.config.get('wgPageName') + '/doc', 'Tagging module documentation page') : new Morebits.wiki.page(mw.config.get('wgPageName'), 'Tagging page'); wikipedia_page.setCallbackParameters(params); wikipedia_page.load(Twinkle.speedy.callbacks.user.main); }; })(jQuery);

//