User:Evad37/MoveToDraft.js

/***************************************************************************************************

MoveToDraft

-------------

Version 2.5.0

-------------

A script to move unsourced articles to draft space, including cleanup and author notification.

- Moves page to draftspace

- Checks if any files used are non-free

- Checks if any redirects pointed to the page

- Comments out non-free files, turn categories into links, add afc draft template, add redirects

- Adds notification message on author talk page

- Updates talk page banners

- Logs draftification in user subpage

  • /

/* jshint laxbreak: true, undef: true, maxerr:999 */

/* globals console, window, document, $, mw, OO, extraJs */

//

$.when(

// Resource loader modules

mw.loader.using(['mediawiki.util', 'mediawiki.api', 'mediawiki.Title', 'ext.gadget.libExtraUtil']),

// Page ready

$.ready

).then(function() {

/* ========== Config ============================================================================ */

var config = {

// Script info

script: {

advert: ' (via script)', // For edit summaries

version: '2.5.1'

},

// MediaWiki configuration values

mw: mw.config.get( [

'wgArticleId',

'wgCurRevisionId',

'wgPageName',

'wgUserGroups',

'wgUserName',

'wgMonthNames',

'wgNamespaceNumber',

'wgTitle'

] )

};

/* ========== API =============================================================================== */

var API = new mw.Api( {

ajax: {

headers: {

'Api-User-Agent': 'MoveToDraft/' + config.script.version +

' ( https://en.wikipedia.org/wiki/User:Evad37/MoveToDraft )'

}

}

} );

var moveToDraft = function moveToDraft() {

/* ========== Additional config ================================================================= */

// Wikitext strings

config.wikitext = {

'rationale': window.m2d_rationale || 'Not ready for mainspace, incubate in draftspace',

'editsummary': window.m2d_editsummary || window.m2d_rationale || 'AFC draft',

'notification_heading': '$1 moved to draftspace',

'notification': window.m2d_notification || "An article you recently created, $1, is not suitable as written to remain published. It needs more citations from reliable, independent sources. (?) Information that can't be referenced should be removed (verifiability is of central importance on Wikipedia). I've moved your draft to draftspace (with a prefix of \"Draft:\" before the article title) where you can incubate the article with minimal disruption. When you feel the article meets Wikipedia's general notability guideline and thus is ready for mainspace, please click on the \"Submit your draft for review!\" button at the top of the page. ~~~~",

'logMsg': '#$1 moved to $2 at ~~~~~'

};

config.doNotLog = window.m2d_doNotLog ? true : false;

// Page data -- to be retreived later from api

config.pagedata = {};

// Helper functions

// - prettify an encoded page title (or at least replace underscores with spaces)

var getPageText = function(p) {

var t = mw.Title.newFromText( decodeURIComponent(p) );

if (t) {

return t.getPrefixedText();

} else {

return p.replace(/_/g, " ");

}

};

/* ========== Tasks ============================================================================= */

// Grab page data - initial author, current wikitext, any redirects, if Draft: page already exists

var grabPageData = function() {

var patt_isRedirect = /^\s*#redirect/i;

var checkedPageTriageStatus = false;

// Function to check if all done

var checkPageData = function() {

if (

config.pagedata.author != null &&

config.pagedata.oldwikitext != null &&

config.pagedata.redirects != null &&

checkedPageTriageStatus

) {

//all done - go to next screen

screen1();

}

};

/* ---------- Initial author ---------------------------------------------------------------- */

/* Try making an api call for just the first revision - but if that is a redirect, then get 'max'

number of revisions, and look for first non-redirect revision - use this as the initial author,

not the creator of the redirect.

*/

var processMaxRvAuthorQuery = function (result) {

var revisions = result.query.pages[config.mw.wgArticleId].revisions;

for ( var i=1; i

if ( !patt_isRedirect.test(revisions[i]['*']) ) {

config.pagedata.author = revisions[i].user;

break;

}

}

//Check that we actually found an author (i.e. not all revisions were redirects

if ( config.pagedata.author == null ) {

API.abort();

var retry = confirm("Could not retrieve page author:\n"+extraJs.makeErrorMsg(c, r)+"\n\nTry again?");

if ( retry ) {

screen0();

} else {

$("#M2D-modal").remove();

}

}

checkPageData();

};

var processAuthorQuery = function (result) {

// Check if page is currently a redirect

if ( result.query.pages[config.mw.wgArticleId].redirect ) {

API.abort();

alert("Error: " + config.mw.wgPageName + " is a redirect");

return;

}

// Check if first revision is a redirect

rvwikitext = result.query.pages[config.mw.wgArticleId].revisions[0]['*'];

if ( patt_isRedirect.test(rvwikitext) ) {

// query to look for first non-redirect revision

API.get( {

action: 'query',

pageids: config.mw.wgArticleId,

prop: 'revisions',

rvprop: ['user', 'content'],

rvlimit: 'max',

rvdir: 'newer'

} )

.done( processMaxRvAuthorQuery )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

API.abort();

var retry = confirm("Could not retrieve page author:\n"+extraJs.makeErrorMsg(c, r)+"\n\nTry again?");

if ( retry ) {

screen0();

} else {

$("#M2D-modal").remove();

}

} );

return;

}

config.pagedata.author = result.query.pages[config.mw.wgArticleId].revisions[0].user;

checkPageData();

};

//Get author

API.get( {

action: 'query',

pageids: config.mw.wgArticleId,

prop: ['revisions', 'info'],

rvprop: ['user', 'content'],

rvlimit: 1,

rvdir: 'newer'

} )

.done( processAuthorQuery )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

API.abort();

var retry = confirm("Could not retrieve page author:\n"+extraJs.makeErrorMsg(c, r)+"\n\nTry again?");

if ( retry ) {

screen0();

} else {

$("#M2D-modal").remove();

}

} );

/* ---------- Current wikitext -------------------------------------------------------------- */

API.get( {

action: 'query',

pageids: config.mw.wgArticleId,

prop: 'revisions',

rvprop: 'content'

} )

.done( function(result) {

config.pagedata.oldwikitext = result.query.pages[config.mw.wgArticleId].revisions[0]['*'];

checkPageData();

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

API.abort();

var retry = confirm("Could not retrieve page wikitext:\n"+ extraJs.makeErrorMsg(c, r)+"\n\nTry again?");

if ( retry ) {

screen0();

} else {

$("#M2D-modal").remove();

}

} );

//TODO(?): also get proposed Draft: page (to check if it is empty or not)

/* ---------- Redirects --------------------------------------------------------------------- */

var redirectTitles = [];

var processRedirectsQuery = function(result) {

if ( !result.query || !result.query.pages ) {

// No results

config.pagedata.redirects = false;

checkPageData();

return;

}

// Gather redirect titles into array

$.each(result.query.pages, function(_id, info) {

redirectTitles.push(info.title);

});

// Continue query if needed

if ( result.continue ) {

doRedirectsQuery($.extend(redirectsQuery, result.continue));

return;

}

// Check if redirects were found

if ( redirectTitles.length === 0 ) {

config.pagedata.redirects = false;

checkPageData();

return;

}

// Set redirects

config.pagedata.redirects = ( redirectTitles.length === 0 ) ? false : redirectTitles;

checkPageData();

};

var redirectsQuery = {

action: 'query',

pageids: config.mw.wgArticleId,

generator: 'redirects',

grdlimit: 500

};

var doRedirectsQuery = function(q) {

API.get( q )

.done( processRedirectsQuery )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

API.abort();

var retry = confirm("Could not retrieve redirects:\n" + extraJs.makeErrorMsg(c, r) +

"\n\nTry again? (or Cancel to skip)");

if ( retry ) {

screen0();

} else {

config.pagedata.redirects = false;

checkPageData();

}

} );

};

doRedirectsQuery(redirectsQuery);

/* ---------- Review (Page Triage) status ----------------------------------------------------------------- */

API.get( {

action: 'pagetriagelist',

page_id: config.mw.wgArticleId

} )

.done( function(result) {

if ( !result.pagetriagelist.pages.length ) {

var keepGoing = confirm('WARNING: Page has already been reviewed by a New Page Patroller. Are you sure you want to draftify this page?');

if ( !keepGoing ) {

API.abort();

$("#M2D-modal").remove();

return;

}

}

checkedPageTriageStatus = true;

checkPageData();

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

API.abort();

var retry = confirm("Could not retrieve page triage status:\n"+ extraJs.makeErrorMsg(c, r)+"\n\nTry again?");

if ( retry ) {

screen0();

} else {

$("#M2D-modal").remove();

}

} );

};

//Move page

var movePage = function() {

$("#M2D-task0").css({"color":"#00F", "font-weight":"bold"});

$("#M2D-status0").html("...");

// First check the page hasn't been draftified in the meantime

API.get({

action: "query",

pageids: config.mw.wgArticleId,

format: "json",

formatversion: "2"

}).then(function(response) {

var page = response && response.query && response.query.pages && response.query.pages[0];

if (!page) {

return $.Deferred().reject();

} else if (page.missing) {

return $.Deferred().reject("moveToDraft-pagemissing");

} else if (page.ns === 118 /* Draft NS */) {

return $.Deferred().reject("moveToDraft-alreadydraft");

} else if (page.ns !== config.mw.wgNamespaceNumber) {

return $.Deferred().reject("moveToDraft-movednamespace");

}

return API.postWithToken( 'csrf', {

action: 'move',

fromid: config.mw.wgArticleId,

to: config.inputdata.newTitle,

movetalk: 1,

noredirect: 1,

reason: config.inputdata.rationale + config.script.advert

} );

})

.done( function() {

if (

-1 === $.inArray('sysop', config.mw.wgUserGroups) &&

-1 === $.inArray('extendedmover', config.mw.wgUserGroups)

) {

// Newly created redirect to be tagged for speedy deletion

tagRedrect();

return;

}

$("#M2D-task0").css({"color":"#000", "font-weight":""});

$("#M2D-status0").html("Done!");

getImageInfo();

} )

.fail( function(c,r) {

if ( r && r.textStatus === 'abort' ) {

return;

} else if (c === "moveToDraft-pagemissing") {

alert("The page no longer appears to exists. It may have been deleted.");

$("#M2D-modal").remove();

window.location.reload();

return;

} else if (c === "moveToDraft-alreadydraft") {

alert("Aborted: The page has already been moved to draftspace.");

$("#M2D-modal").remove();

window.location.reload();

return;

} else if (c === "moveToDraft-alreadydraft") {

alert("Aborted: The page has already been moved out of mainspace.");

$("#M2D-modal").remove();

window.location.reload();

return;

}

var retry = confirm("Could not move page:\n"+ extraJs.makeErrorMsg(c, r)+"\n\nTry again?");

if ( retry ) {

movePage();

} else {

screen1(true);

}

} );

};

var tagRedrect = function() {

$("#M2D-status0").html("Done,
Tagging redirect for speedy deletion...");

API.postWithToken( 'csrf', {

action: 'edit',

title: config.mw.wgPageName,

prependtext: '{{Db-r2}}\n',

summary: 'R2 speedy deletion request (article moved to draftspace)' + config.script.advert

} )

.done( function() {

$("#M2D-task0").css({"color":"#000", "font-weight":""});

$("#M2D-status0").append(" Done!");

getImageInfo();

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not tag redirect for speedy deletion:\n"+

extraJs.makeErrorMsg(c, r) + "\n\nTry again?");

if ( retry ) {

tagRedrect();

} else {

$("#M2D-task0").css({"color":"#F00", "font-weight":""});

$("#M2D-status0").append(" Skipped");

getImageInfo();

}

} );

};

//Find which images are non-free

var getImageInfo = function() {

$("#M2D-task1").css({"color":"#00F", "font-weight":"bold"});

$("#M2D-status1").html("...");

processImageInfo = function(result) {

var nonfreefiles = [];

if ( result && result.query ) {

$.each(result.query.pages, function(id, page) {

if ( id > 0 && page.categories ) {

nonfreefiles.push(page.title);

}

});

}

editWikitext(nonfreefiles);

};

API.get( {

action: 'query',

pageids: config.mw.wgArticleId,

generator: 'images',

gimlimit: 'max',

prop: 'categories',

cllimit: 'max',

clcategories: 'Category:All non-free media',

} )

.done( function(result){

$("#M2D-task1").css({"color":"#000", "font-weight":""});

$("#M2D-status1").html("Done!");

processImageInfo(result);

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not find if there are non-free files:\n"+ extraJs.makeErrorMsg(c, r)+"\n\n[Okay] to try again, or [Cancel] to skip");

if ( retry ) {

getImageInfo();

} else {

$("#M2D-task1").css({"color":"#F00", "font-weight":""});

$("#M2D-status1").html("Skipped");

editWikitext([]);

}

} );

};

//Comment out non-free files, turn categories into links, add afc draft template, list any redirects

var editWikitext = function(nonfreefiles) {

$("#M2D-task2").css({"color":"#00F", "font-weight":"bold"});

$("#M2D-status2").html("...");

var redirectsList = ( !config.pagedata.redirects ) ? '' : '\n'+

'\n';

var wikitext = "{{subst:AFC draft|" + config.inputdata.authorName + "}}\n" +

redirectsList +

config.pagedata.oldwikitext.replace(/\[\[\s*[Cc]ategory\s*:/g, "[[:Category:") +

"\n{{subst:Drafts moved from mainspace}}";

// non-free files

// (derived from WP:XFDC - https://en.wikipedia.org/wiki/User:Evad37/XFDcloser.js )

if ( nonfreefiles.length > 0 ) {

// Start building regex strings

normal_regex_str = "(";

gallery_regex_str = "(";

free_regex_str = "(";

for ( var i=0; i

// Take off namespace prefix

filename = nonfreefiles[i].replace(/^.*?:/, "");

// For regex matching: first character can be either upper or lower case, special

// characters need to be escaped, spaces can be either spaces or underscores

filename_regex_str = "[" + mw.util.escapeRegExp(filename.slice(0, 1).toUpperCase()) +

mw.util.escapeRegExp(filename.slice(0, 1).toLowerCase()) + "]" +

mw.util.escapeRegExp(filename.slice(1)).replace(/ /g, "[ _]");

// Add to regex strings

normal_regex_str += "\\[\\[\\s*(?:[Ii]mage|[Ff]ile)\\s*:\\s*" + filename_regex_str +

"\\s*\\|?.*?(?:(?:\\[\\[.*?\\]\\]).*?)*\\]\\]";

gallery_regex_str += "^\\s*(?:[Ii]mage|[Ff]ile):\\s*" + filename_regex_str + ".*?$";

free_regex_str += "\\|\\s*(?:[\\w\\s]+\\=)?\\s*(?:(?:[Ii]mage|[Ff]ile):\\s*)?" +

filename_regex_str;

if ( i+1 === nonfreefiles.length ) {

normal_regex_str += ")(?![^<]*?-->)";

gallery_regex_str += ")(?![^<]*?-->)";

free_regex_str += ")(?![^<]*?-->)";

} else {

normal_regex_str += "|";

gallery_regex_str += "|";

free_regex_str += "|";

}

}

// Check for normal file usage, i.e. ...

var normal_regex = new RegExp( normal_regex_str, "g");

wikitext = wikitext.replace(normal_regex, "");

// Check for gallery usage, i.e. instances that must start on a new line, eventually

// preceded with some space, and must include File: or Image: prefix

var gallery_regex = new RegExp( gallery_regex_str, "mg" );

wikitext = wikitext.replace(gallery_regex, "");

// Check for free usages, for example as template argument, might have the File: or Image:

// prefix excluded, but must be preceeded by an |

var free_regex = new RegExp( free_regex_str, "mg" );

wikitext = wikitext.replace(free_regex, "");

}

API.postWithToken( 'csrf', {

action: 'edit',

pageid: config.mw.wgArticleId,

text: wikitext,

summary: config.wikitext.editsummary + config.script.advert

} )

.done( function(){

$("#M2D-task2").css({"color":"#000", "font-weight":""});

$("#M2D-status2").html("Done!");

notifyAuthor();

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not edit draft artice:\n"+ extraJs.makeErrorMsg(c, r)+"\n\n[Okay] to try again, or [Cancel] to skip");

if ( retry ) {

editWikitext(nonfreefiles);

} else {

$("#M2D-task2").css({"color":"#F00", "font-weight":""});

$("#M2D-status2").html("Skipped");

notifyAuthor();

}

} );

};

var notifyAuthor = function() {

if ( !config.inputdata.notifyEnable ) {

updateTalk();

return;

}

$("#M2D-task3").css({"color":"#00F", "font-weight":"bold"});

$("#M2D-status3").html("...");

API.postWithToken( 'csrf', {

action: 'edit',

title: 'User talk:' + config.inputdata.authorName,

section: 'new',

sectiontitle: config.inputdata.notifyMsgHead,

text: config.inputdata.notifyMsg,

} )

.done( function(){

$("#M2D-task3").css({"color":"#000", "font-weight":""});

$("#M2D-status3").html("Done!");

updateTalk();

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not edit author talk page:\n"+ extraJs.makeErrorMsg(c, r)+"\n\n[Okay] to try again, or [Cancel] to skip");

if ( retry ) {

notifyAuthor();

} else {

$("#M2D-task3").css({"color":"#F00", "font-weight":""});

$("#M2D-status3").html("Skipped");

updateTalk();

}

} );

};

var updateTalk = function() {

$("#M2D-task4").css({"color":"#00F", "font-weight":"bold"});

$("#M2D-status4").html("...");

//if page exists, do a regex search/repace for class/importances parameters

var processTalkWikitext = function(result) {

var talk_id = result.query.pageids[0];

if ( talk_id < 0 ) {

$("#M2D-task4").css({"color":"#000", "font-weight":""});

$("#M2D-status4").html("Done (talk page does not exist)");

draftifyLog();

return;

}

var old_talk_wikitext = result.query.pages[talk_id].revisions[0]['*'];

var new_talk_wikitext = old_talk_wikitext.replace(/(\|\s*(?:class|importance)\s*=\s*)[^\|}]*(?=[^}]*}})/g, "$1");

if ( new_talk_wikitext === old_talk_wikitext ) {

$("#M2D-task4").css({"color":"#000", "font-weight":""});

$("#M2D-status4").html("Done (no changes needed)");

draftifyLog();

return;

}

API.postWithToken( 'csrf', {

action: 'edit',

pageid: talk_id,

section: '0',

text: new_talk_wikitext,

summary: 'Remove class/importance from project banners' + config.script.advert

} )

.done( function(){

$("#M2D-task4").css({"color":"#000", "font-weight":""});

$("#M2D-status4").html("Done!");

draftifyLog();

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not edit draft's talk page:\n"+ extraJs.makeErrorMsg(c, r)+"\n\n[Okay] to try again, or [Cancel] to skip");

if ( retry ) {

updateTalk();

} else {

$("#M2D-task4").css({"color":"#F00", "font-weight":""});

$("#M2D-status4").html("Skipped");

draftifyLog();

}

} );

};

//get talk page wikitext (section 0)

API.get( {

action: 'query',

titles: config.inputdata.newTitle.replace("Draft:", "Draft talk:"),

prop: 'revisions',

rvprop: 'content',

rvsection: '0',

indexpageids: 1

} )

.done( processTalkWikitext )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not find draft's talk page:\n"+ extraJs.makeErrorMsg(c, r)+"\n\n[Okay] to try again, or [Cancel] to skip");

if ( retry ) {

updateTalk();

} else {

$("#M2D-task4").css({"color":"#F00", "font-weight":""});

$("#M2D-status4").html("Skipped");

draftifyLog();

}

} );

};

var draftifyLog = function() {

if (config.doNotLog) {

$("#M2D-finished, #M2D-abort").toggle();

return;

}

$("#M2D-task5").css({"color":"#00F", "font-weight":"bold"});

$("#M2D-status5").html("...");

var logpage = 'User:' + config.mw.wgUserName + '/Draftify_log';

var monthNames = config.mw.wgMonthNames.slice(1);

var now = new Date();

var heading = '== ' + monthNames[now.getUTCMonth()] + ' ' + now.getUTCFullYear() + ' ==';

var headingPatt = RegExp(heading);

var processLogWikitext = function(result) {

var logpage_wikitext = '';

var id = result.query.pageids[0];

if ( id < 0 ) {

var createlog = confirm('Log draftification (at ' + logpage + ') ?');

if ( !createlog ) {

$("#M2D-task5").css({"color":"#F00", "font-weight":""});

$("#M2D-status5").empty().append("Skipped");

$("#M2D-finished, #M2D-abort").toggle();

return;

}

logpage_wikitext = 'This is a log of pages moved to draftspace using the MoveToDraft script.';

} else {

logpage_wikitext = result.query.pages[id].revisions[0]['*'].trim();

}

if ( !headingPatt.test(logpage_wikitext) ) {

logpage_wikitext += '\n\n' + heading;

}

logpage_wikitext += '\n' + config.inputdata.logMsg;

API.postWithToken( 'csrf', {

action: 'edit',

title: logpage,

text: logpage_wikitext,

summary: 'Logging '+config.inputdata.newTitle+'' + config.script.advert

} )

.done( function(){

$("#M2D-task5").css({"color":"#000", "font-weight":""});

$("#M2D-status5").html("Done!");

$("#M2D-finished, #M2D-abort").toggle();

} )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not edit log page:\n"+ extraJs.makeErrorMsg(c, r)+"\n\n[Okay] to try again, or [Cancel] to skip");

if ( retry ) {

draftifyLog();

} else {

$("#M2D-task5").css({"color":"#F00", "font-weight":""});

$("#M2D-status5").html("Skipped");

$("#M2D-finished, #M2D-abort").toggle();

}

} );

};

//get log page wikitext

API.get( {

action: 'query',

titles: logpage,

prop: 'revisions',

rvprop: 'content',

indexpageids: 1

} )

.done( processLogWikitext )

.fail( function(c,r) {

if ( r.textStatus === 'abort' ) { return; }

var retry = confirm("Could not find log page:\n"+ extraJs.makeErrorMsg(c, r)+"\n\n[Okay] to try again, or [Cancel] to skip");

if ( retry ) {

draftifyLog();

} else {

$("#M2D-task5").css({"color":"#F00", "font-weight":""});

$("#M2D-status5").html("Skipped");

$("#M2D-finished, #M2D-abort").toggle();

}

} );

};

// --- Interface screens ---

//0) Initial screen

var screen0 = function() {

$("#M2D-interface-header, #M2D-interface-content, #M2D-interface-footer").empty();

$("#M2D-interface-header").text("Move To Draft...");

$("#M2D-interface-content").text("Loading...");

grabPageData();

};

//1) User inputs

/**

*

* @param {boolean} restoreValues Restore previously set values

*/

var screen1 = function(restoreValues) {

$("#M2D-interface-header, #M2D-interface-content, #M2D-interface-footer").empty();

$("#M2D-interface-header").text("Move To Draft: options");

$("#M2D-interface-content").append(

$('

').css('margin-bottom','0.5em').append(

$('')

.css({

display: 'block',

color: 'darkred'

}).append(

'This script is no longer being maintained.',

' Please switch to the current version: Edit your ',

extraJs.makeLink('Special:MyPage/common.js', 'common.js'),

' file by changing ', extraJs.makeLink('User:Evad37/MoveToDraft.js'),

' to ', extraJs.makeLink('User:MPGuy2824/MoveToDraft.js'), '

',

'Please ensure draftifying is appropriate per ',

extraJs.makeLink("WP:DRAFTIFY")

),

$('

'Move to ',

$('').text('Draft:')

),

$('').attr({'type':'text', 'name':'M2D-option-newtitle', 'id':'M2D-option-newtitle'})

),

$('

').css('margin-bottom','0.5em').append(

$('

.css('display','block').text('Reason for move (log summary):'),

$('