<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="en">
	<id>https://eurovision.jobogamer.xyz/index.php?action=history&amp;feed=atom&amp;title=MediaWiki%3AGadget-twinklebatchundelete.js</id>
	<title>MediaWiki:Gadget-twinklebatchundelete.js - Revision history</title>
	<link rel="self" type="application/atom+xml" href="https://eurovision.jobogamer.xyz/index.php?action=history&amp;feed=atom&amp;title=MediaWiki%3AGadget-twinklebatchundelete.js"/>
	<link rel="alternate" type="text/html" href="https://eurovision.jobogamer.xyz/index.php?title=MediaWiki:Gadget-twinklebatchundelete.js&amp;action=history"/>
	<updated>2026-06-11T01:25:16Z</updated>
	<subtitle>Revision history for this page on the wiki</subtitle>
	<generator>MediaWiki 1.45.3</generator>
	<entry>
		<id>https://eurovision.jobogamer.xyz/index.php?title=MediaWiki:Gadget-twinklebatchundelete.js&amp;diff=69&amp;oldid=prev</id>
		<title>Alfie: 1 revision imported</title>
		<link rel="alternate" type="text/html" href="https://eurovision.jobogamer.xyz/index.php?title=MediaWiki:Gadget-twinklebatchundelete.js&amp;diff=69&amp;oldid=prev"/>
		<updated>2026-03-19T09:58:26Z</updated>

		<summary type="html">&lt;p&gt;1 revision imported&lt;/p&gt;
&lt;table style=&quot;background-color: #fff; color: #202122;&quot; data-mw=&quot;interface&quot;&gt;
				&lt;tr class=&quot;diff-title&quot; lang=&quot;en&quot;&gt;
				&lt;td colspan=&quot;1&quot; style=&quot;background-color: #fff; color: #202122; text-align: center;&quot;&gt;← Older revision&lt;/td&gt;
				&lt;td colspan=&quot;1&quot; style=&quot;background-color: #fff; color: #202122; text-align: center;&quot;&gt;Revision as of 05:58, 19 March 2026&lt;/td&gt;
				&lt;/tr&gt;&lt;tr&gt;&lt;td colspan=&quot;2&quot; class=&quot;diff-notice&quot; lang=&quot;en&quot;&gt;&lt;div class=&quot;mw-diff-empty&quot;&gt;(No difference)&lt;/div&gt;
&lt;/td&gt;&lt;/tr&gt;&lt;/table&gt;</summary>
		<author><name>Alfie</name></author>
	</entry>
	<entry>
		<id>https://eurovision.jobogamer.xyz/index.php?title=MediaWiki:Gadget-twinklebatchundelete.js&amp;diff=68&amp;oldid=prev</id>
		<title>en&gt;Novem Linguae: Repo at 58a0dd8: switch to camelcase, part 1 (#2245)</title>
		<link rel="alternate" type="text/html" href="https://eurovision.jobogamer.xyz/index.php?title=MediaWiki:Gadget-twinklebatchundelete.js&amp;diff=68&amp;oldid=prev"/>
		<updated>2025-11-25T20:41:31Z</updated>

		<summary type="html">&lt;p&gt;Repo at 58a0dd8: switch to camelcase, part 1 (#2245)&lt;/p&gt;
&lt;p&gt;&lt;b&gt;New page&lt;/b&gt;&lt;/p&gt;&lt;div&gt;// &amp;lt;nowiki&amp;gt;&lt;br /&gt;
&lt;br /&gt;
(function() {&lt;br /&gt;
&lt;br /&gt;
/*&lt;br /&gt;
 ****************************************&lt;br /&gt;
 *** twinklebatchundelete.js: Batch undelete module&lt;br /&gt;
 ****************************************&lt;br /&gt;
 * Mode of invocation:     Tab (&amp;quot;Und-batch&amp;quot;)&lt;br /&gt;
 * Active on:              Existing user and project pages&lt;br /&gt;
 */&lt;br /&gt;
&lt;br /&gt;
Twinkle.batchundelete = function twinklebatchundelete() {&lt;br /&gt;
	if (!Morebits.userIsSysop || !mw.config.get(&amp;#039;wgArticleId&amp;#039;) || (&lt;br /&gt;
		mw.config.get(&amp;#039;wgNamespaceNumber&amp;#039;) !== mw.config.get(&amp;#039;wgNamespaceIds&amp;#039;).user &amp;amp;&amp;amp;&lt;br /&gt;
		mw.config.get(&amp;#039;wgNamespaceNumber&amp;#039;) !== mw.config.get(&amp;#039;wgNamespaceIds&amp;#039;).project)) {&lt;br /&gt;
		return;&lt;br /&gt;
	}&lt;br /&gt;
	Twinkle.addPortletLink(Twinkle.batchundelete.callback, &amp;#039;Und-batch&amp;#039;, &amp;#039;tw-batch-undel&amp;#039;, &amp;quot;Undelete &amp;#039;em all&amp;quot;);&lt;br /&gt;
};&lt;br /&gt;
&lt;br /&gt;
Twinkle.batchundelete.callback = function twinklebatchundeleteCallback() {&lt;br /&gt;
	const Window = new Morebits.SimpleWindow(600, 400);&lt;br /&gt;
	Window.setScriptName(&amp;#039;Twinkle&amp;#039;);&lt;br /&gt;
	Window.setTitle(&amp;#039;Batch undelete&amp;#039;);&lt;br /&gt;
	Window.addFooterLink(&amp;#039;Twinkle help&amp;#039;, &amp;#039;WP:TW/DOC#batchundelete&amp;#039;);&lt;br /&gt;
	Window.addFooterLink(&amp;#039;Give feedback&amp;#039;, &amp;#039;WT:TW&amp;#039;);&lt;br /&gt;
&lt;br /&gt;
	const form = new Morebits.QuickForm(Twinkle.batchundelete.callback.evaluate);&lt;br /&gt;
	form.append({&lt;br /&gt;
		type: &amp;#039;checkbox&amp;#039;,&lt;br /&gt;
		list: [&lt;br /&gt;
			{&lt;br /&gt;
				label: &amp;#039;Restore talk pages of undeleted pages if they existed&amp;#039;,&lt;br /&gt;
				name: &amp;#039;undel_talk&amp;#039;,&lt;br /&gt;
				value: &amp;#039;undel_talk&amp;#039;,&lt;br /&gt;
				checked: true&lt;br /&gt;
			}&lt;br /&gt;
		]&lt;br /&gt;
	});&lt;br /&gt;
	form.append({&lt;br /&gt;
		type: &amp;#039;input&amp;#039;,&lt;br /&gt;
		name: &amp;#039;reason&amp;#039;,&lt;br /&gt;
		label: &amp;#039;Reason:&amp;#039;,&lt;br /&gt;
		size: 60&lt;br /&gt;
	});&lt;br /&gt;
&lt;br /&gt;
	const statusdiv = document.createElement(&amp;#039;div&amp;#039;);&lt;br /&gt;
	statusdiv.style.padding = &amp;#039;15px&amp;#039;; // just so it doesn&amp;#039;t look broken&lt;br /&gt;
	Window.setContent(statusdiv);&lt;br /&gt;
	Morebits.Status.init(statusdiv);&lt;br /&gt;
	Window.display();&lt;br /&gt;
&lt;br /&gt;
	const query = {&lt;br /&gt;
		action: &amp;#039;query&amp;#039;,&lt;br /&gt;
		generator: &amp;#039;links&amp;#039;,&lt;br /&gt;
		prop: &amp;#039;info&amp;#039;,&lt;br /&gt;
		inprop: &amp;#039;protection&amp;#039;,&lt;br /&gt;
		titles: mw.config.get(&amp;#039;wgPageName&amp;#039;),&lt;br /&gt;
		gpllimit: Twinkle.getPref(&amp;#039;batchMax&amp;#039;),&lt;br /&gt;
		format: &amp;#039;json&amp;#039;&lt;br /&gt;
	};&lt;br /&gt;
	const statelem = new Morebits.Status(&amp;#039;Grabbing list of pages&amp;#039;);&lt;br /&gt;
	const wikipediaApi = new Morebits.wiki.Api(&amp;#039;loading...&amp;#039;, query, ((apiobj) =&amp;gt; {&lt;br /&gt;
		const response = apiobj.getResponse();&lt;br /&gt;
		let pages = (response.query &amp;amp;&amp;amp; response.query.pages) || [];&lt;br /&gt;
		pages = pages.filter((page) =&amp;gt; page.missing);&lt;br /&gt;
		const list = [];&lt;br /&gt;
		pages.sort(Twinkle.sortByNamespace);&lt;br /&gt;
		pages.forEach((page) =&amp;gt; {&lt;br /&gt;
			const editProt = page.protection.filter((pr) =&amp;gt; pr.type === &amp;#039;create&amp;#039; &amp;amp;&amp;amp; pr.level === &amp;#039;sysop&amp;#039;).pop();&lt;br /&gt;
&lt;br /&gt;
			const title = page.title;&lt;br /&gt;
			list.push({&lt;br /&gt;
				label: title + (editProt ? &amp;#039; (fully create protected&amp;#039; +&lt;br /&gt;
					(editProt.expiry === &amp;#039;infinity&amp;#039; ? &amp;#039; indefinitely&amp;#039; : &amp;#039;, expires &amp;#039; + new Morebits.Date(editProt.expiry).calendar(&amp;#039;utc&amp;#039;) + &amp;#039; (UTC)&amp;#039;) + &amp;#039;)&amp;#039; : &amp;#039;&amp;#039;),&lt;br /&gt;
				value: title,&lt;br /&gt;
				checked: true,&lt;br /&gt;
				style: editProt ? &amp;#039;color:red&amp;#039; : &amp;#039;&amp;#039;&lt;br /&gt;
			});&lt;br /&gt;
		});&lt;br /&gt;
		apiobj.params.form.append({ type: &amp;#039;header&amp;#039;, label: &amp;#039;Pages to undelete&amp;#039; });&lt;br /&gt;
		apiobj.params.form.append({&lt;br /&gt;
			type: &amp;#039;button&amp;#039;,&lt;br /&gt;
			label: &amp;#039;Select All&amp;#039;,&lt;br /&gt;
			event: function(e) {&lt;br /&gt;
				$(Morebits.QuickForm.getElements(e.target.form, &amp;#039;pages&amp;#039;)).prop(&amp;#039;checked&amp;#039;, true);&lt;br /&gt;
			}&lt;br /&gt;
		});&lt;br /&gt;
		apiobj.params.form.append({&lt;br /&gt;
			type: &amp;#039;button&amp;#039;,&lt;br /&gt;
			label: &amp;#039;Deselect All&amp;#039;,&lt;br /&gt;
			event: function(e) {&lt;br /&gt;
				$(Morebits.QuickForm.getElements(e.target.form, &amp;#039;pages&amp;#039;)).prop(&amp;#039;checked&amp;#039;, false);&lt;br /&gt;
			}&lt;br /&gt;
		});&lt;br /&gt;
		apiobj.params.form.append({&lt;br /&gt;
			type: &amp;#039;checkbox&amp;#039;,&lt;br /&gt;
			name: &amp;#039;pages&amp;#039;,&lt;br /&gt;
			shiftClickSupport: true,&lt;br /&gt;
			list: list&lt;br /&gt;
		});&lt;br /&gt;
		apiobj.params.form.append({ type: &amp;#039;submit&amp;#039; });&lt;br /&gt;
&lt;br /&gt;
		const result = apiobj.params.form.render();&lt;br /&gt;
		apiobj.params.Window.setContent(result);&lt;br /&gt;
&lt;br /&gt;
		Morebits.QuickForm.getElements(result, &amp;#039;pages&amp;#039;).forEach(Twinkle.generateArrowLinks);&lt;br /&gt;
&lt;br /&gt;
	}), statelem);&lt;br /&gt;
	wikipediaApi.params = { form: form, Window: Window };&lt;br /&gt;
	wikipediaApi.post();&lt;br /&gt;
};&lt;br /&gt;
&lt;br /&gt;
Twinkle.batchundelete.callback.evaluate = function(event) {&lt;br /&gt;
	Morebits.wiki.actionCompleted.notice = &amp;#039;Batch undeletion is now complete&amp;#039;;&lt;br /&gt;
&lt;br /&gt;
	const numProtected = Morebits.QuickForm.getElements(event.target, &amp;#039;pages&amp;#039;).filter((element) =&amp;gt; element.checked &amp;amp;&amp;amp; element.nextElementSibling.style.color === &amp;#039;red&amp;#039;).length;&lt;br /&gt;
	if (numProtected &amp;gt; 0 &amp;amp;&amp;amp; !confirm(&amp;#039;You are about to undelete &amp;#039; + numProtected + &amp;#039; fully create protected page(s). Are you sure?&amp;#039;)) {&lt;br /&gt;
		return;&lt;br /&gt;
	}&lt;br /&gt;
&lt;br /&gt;
	const input = Morebits.QuickForm.getInputData(event.target);&lt;br /&gt;
&lt;br /&gt;
	if (!input.reason) {&lt;br /&gt;
		alert(&amp;#039;You need to give a reason, you cabal crony!&amp;#039;);&lt;br /&gt;
		return;&lt;br /&gt;
	}&lt;br /&gt;
	Morebits.SimpleWindow.setButtonsEnabled(false);&lt;br /&gt;
	Morebits.Status.init(event.target);&lt;br /&gt;
&lt;br /&gt;
	if (!input.pages || !input.pages.length) {&lt;br /&gt;
		Morebits.Status.error(&amp;#039;Error&amp;#039;, &amp;#039;nothing to undelete, aborting&amp;#039;);&lt;br /&gt;
		return;&lt;br /&gt;
	}&lt;br /&gt;
&lt;br /&gt;
	const pageUndeleter = new Morebits.BatchOperation(&amp;#039;Undeleting pages&amp;#039;);&lt;br /&gt;
	pageUndeleter.setOption(&amp;#039;chunkSize&amp;#039;, Twinkle.getPref(&amp;#039;batchChunks&amp;#039;));&lt;br /&gt;
	pageUndeleter.setOption(&amp;#039;preserveIndividualStatusLines&amp;#039;, true);&lt;br /&gt;
	pageUndeleter.setPageList(input.pages);&lt;br /&gt;
	pageUndeleter.run((pageName) =&amp;gt; {&lt;br /&gt;
		const params = {&lt;br /&gt;
			page: pageName,&lt;br /&gt;
			undel_talk: input.undel_talk,&lt;br /&gt;
			reason: input.reason,&lt;br /&gt;
			pageUndeleter: pageUndeleter&lt;br /&gt;
		};&lt;br /&gt;
&lt;br /&gt;
		const wikipediaPage = new Morebits.wiki.Page(pageName, &amp;#039;Undeleting page &amp;#039; + pageName);&lt;br /&gt;
		wikipediaPage.setCallbackParameters(params);&lt;br /&gt;
		wikipediaPage.setEditSummary(input.reason);&lt;br /&gt;
		wikipediaPage.setChangeTags(Twinkle.changeTags);&lt;br /&gt;
		wikipediaPage.suppressProtectWarning();&lt;br /&gt;
		wikipediaPage.setMaxRetries(3); // temporary increase from 2 to make batchundelete more likely to succeed [[phab:T222402]] #613&lt;br /&gt;
		wikipediaPage.undeletePage(Twinkle.batchundelete.callbacks.doExtras, pageUndeleter.workerFailure);&lt;br /&gt;
	});&lt;br /&gt;
};&lt;br /&gt;
&lt;br /&gt;
Twinkle.batchundelete.callbacks = {&lt;br /&gt;
	// this stupid parameter name is a temporary thing until I implement an overhaul&lt;br /&gt;
	// of Morebits.wiki.* callback parameters&lt;br /&gt;
	doExtras: function(thingWithParameters) {&lt;br /&gt;
		const params = thingWithParameters.parent ? thingWithParameters.parent.getCallbackParameters() :&lt;br /&gt;
			thingWithParameters.getCallbackParameters();&lt;br /&gt;
		// the initial batch operation&amp;#039;s job is to delete the page, and that has&lt;br /&gt;
		// succeeded by now&lt;br /&gt;
		params.pageUndeleter.workerSuccess(thingWithParameters);&lt;br /&gt;
&lt;br /&gt;
		let query, wikipediaApi;&lt;br /&gt;
&lt;br /&gt;
		if (params.undel_talk) {&lt;br /&gt;
			const talkpagename = new mw.Title(params.page).getTalkPage().getPrefixedText();&lt;br /&gt;
			if (talkpagename !== params.page) {&lt;br /&gt;
				query = {&lt;br /&gt;
					action: &amp;#039;query&amp;#039;,&lt;br /&gt;
					prop: &amp;#039;deletedrevisions&amp;#039;,&lt;br /&gt;
					drvprop: &amp;#039;ids&amp;#039;,&lt;br /&gt;
					drvlimit: 1,&lt;br /&gt;
					titles: talkpagename,&lt;br /&gt;
					format: &amp;#039;json&amp;#039;&lt;br /&gt;
				};&lt;br /&gt;
				wikipediaApi = new Morebits.wiki.Api(&amp;#039;Checking talk page for deleted revisions&amp;#039;, query, Twinkle.batchundelete.callbacks.undeleteTalk);&lt;br /&gt;
				wikipediaApi.params = params;&lt;br /&gt;
				wikipediaApi.params.talkPage = talkpagename;&lt;br /&gt;
				wikipediaApi.post();&lt;br /&gt;
			}&lt;br /&gt;
		}&lt;br /&gt;
	},&lt;br /&gt;
	undeleteTalk: function(apiobj) {&lt;br /&gt;
		const page = apiobj.getResponse().query.pages[0];&lt;br /&gt;
		const exists = !page.missing;&lt;br /&gt;
		const delrevs = page.deletedrevisions &amp;amp;&amp;amp; page.deletedrevisions[0].revid;&lt;br /&gt;
&lt;br /&gt;
		if (exists || !delrevs) {&lt;br /&gt;
			// page exists or has no deleted revisions; forget about it&lt;br /&gt;
			return;&lt;br /&gt;
		}&lt;br /&gt;
&lt;br /&gt;
		const talkpage = new Morebits.wiki.Page(apiobj.params.talkPage, &amp;#039;Undeleting talk page of &amp;#039; + apiobj.params.page);&lt;br /&gt;
		talkpage.setEditSummary(&amp;#039;Undeleting [[Help:Talk page|talk page]] of &amp;quot;&amp;#039; + apiobj.params.page + &amp;#039;&amp;quot;&amp;#039;);&lt;br /&gt;
		talkpage.setChangeTags(Twinkle.changeTags);&lt;br /&gt;
		talkpage.undeletePage();&lt;br /&gt;
	}&lt;br /&gt;
};&lt;br /&gt;
&lt;br /&gt;
Twinkle.addInitCallback(Twinkle.batchundelete, &amp;#039;batchundelete&amp;#039;);&lt;br /&gt;
}());&lt;br /&gt;
&lt;br /&gt;
// &amp;lt;/nowiki&amp;gt;&lt;/div&gt;</summary>
		<author><name>en&gt;Novem Linguae</name></author>
	</entry>
</feed>