משתמש:😡/common.js: הבדלים בין גרסאות בדף

מתוך ויקי נטפרי
קפיצה לניווט קפיצה לחיפוש
 
(גרסת ביניים אחת של אותו משתמש אינה מוצגת)
שורה 1: שורה 1:
mw.loader.load('//en.wikipedia.org/w/index.php?title=User:Joeytje50/JWB.js/load.js&action=raw&ctype=text/javascript');
+
mw.loader.load( '//en.wikipedia.org/w/index.php?title=User:Joeytje50/JWB.js/load.js&action=raw&ctype=text/javascript' ); // Backlink: [[en:User:Joeytje50/JWB.js/load.js]]
// Copied from [[User:Joeytje50/JWB.js]]
 
/**<nowiki>
 
* Install this script by pasting the following in your personal JavaScript file:
 
 
 
mw.loader.load('//en.wikipedia.org/w/index.php?title=User:Joeytje50/JWB.js/load.js&action=raw&ctype=text/javascript');
 
 
 
* Or for users on en.wikipedia.org:
 
 
 
importScript( 'User:Joeytje50/JWB.js/load.js' ); // Backlink: [[User:Joeytje50/JWB.js/load.js]]
 
 
 
* Note that this script will only run on the 'Project:AutoWikiBrowser/Script' page.
 
* This script is based on the downloadable AutoWikiBrowser.
 
*
 
* @licence
 
* This program is free software; you can redistribute it and/or modify
 
* it under the terms of the GNU General Public License as published by
 
* the Free Software Foundation; either version 2 of the License, or
 
* (at your option) any later version.
 
*
 
* This program is distributed in the hope that it will be useful,
 
* but WITHOUT ANY WARRANTY; without even the implied warranty of
 
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 
* GNU General Public License for more details.
 
*
 
* You should have received a copy of the GNU General Public License along
 
* with this program; if not, write to the Free Software Foundation, Inc.,
 
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 
* http://www.gnu.org/copyleft/gpl.html
 
* @version 3.1.0
 
* @author Joeytje50
 
*/
 
 
 
//TODO: Re-enable summary box (possibly more) when not busy submitting.
 
//TODO: more advanced pagelist-generating options
 
//TODO: generate page list based on images on a page
 
//TODO: Split up i18n to separate files per language (in the same way MediaWiki does it)
 
//TODO: Add feature to perform general cleanup (<table> to {|, fullurl-links to wikilinks, removing underscores from wikilinks)
 
 
 
window.JWB = {}; //The main global object for the script.
 
 
 
/***** User verification *****/
 
 
 
(function() {
 
if (mw.config.get('wgCanonicalNamespace')+':'+mw.config.get('wgTitle') !== 'Project:AutoWikiBrowser/Script Beta' || JWB.allowed === false || mw.config.get('wgUserName') === null) {
 
JWB.allowed = false;
 
return;
 
}
 
mw.loader.load('//he.wikipedia.org/w/index.php?title=User:Uziel302/JWB.css&action=raw&ctype=text/css', 'text/css');
 
mw.loader.load('mediawiki.diff.styles');
 
 
$.getScript('//en.wikipedia.org/w/index.php?title=User:Joeytje50/JWB.js/i18n.js&action=raw&ctype=text/javascript', function() {
 
if (JWB.allowed === true) {
 
JWB.init(); //init if verification has already returned true
 
} else if (JWB.allowed === false) {
 
alert(JWB.msg('not-on-list'));
 
}
 
});
 
 
//RegEx Typo Fixing
 
$.getScript('//en.wikipedia.org/w/index.php?title=User:Joeytje50/RETF.js&action=raw&ctype=text/javascript', function() {
 
$('#refreshRETF').click(RETF.load);
 
});
 
 
 
(new mw.Api()).get({
 
action: 'query',
 
titles: 'Project:AutoWikiBrowser/CheckPage',
 
prop: 'revisions',
 
meta: 'userinfo|siteinfo',
 
rvprop: 'content',
 
rvlimit: 1,
 
uiprop: 'groups',
 
siprop: 'namespaces',
 
indexpageids: true,
 
format: 'json',
 
}).done(function(response) {
 
if (response.error) {
 
alert('API error: ' + response.error.info);
 
JWB = false; //preventing further access. No verification => no access.
 
return;
 
}
 
JWB.ns = response.query.namespaces; //saving for later
 
 
JWB.username = response.query.userinfo.name; //preventing any "hacks" that change wgUserName or mw.config.wgUserName
 
var groups = response.query.userinfo.groups;
 
var page = response.query.pages[response.query.pageids[0]];
 
var users, bots;
 
if (response.query.pageids[0] !== '-1' && /<!--\s*enabledusersbegins\s*-->/.test(page.revisions[0]['*'])) {
 
var cont = page.revisions[0]['*'];
 
users = cont.substring(
 
cont.search(/<!--\s*enabledusersbegins\s*-->/),
 
cont.search(/<!--\s*enabledusersends\s*-->/)
 
).split('\n');
 
if (/<!--\s*enabledbots\s*-->/.test(cont)) {
 
bots = cont.substring(
 
cont.search(/<!--\s*enabledbots\s*-->/),
 
cont.search(/<!--\s*enabledbotsends\s*-->/)
 
).split('\n');
 
} else bots = [];
 
var i=0;
 
while (i<users.length) {
 
    if (users[i].charAt(0) !== '*') {
 
    users.splice(i,1);
 
    } else {
 
    users[i] = $.trim(users[i].substr(1));
 
    i++;
 
    }
 
}
 
i=0;
 
while (i<bots.length) {
 
    if (bots[i].charAt(0) !== '*') {
 
    bots.splice(i,1);
 
    } else {
 
    bots[i] = $.trim(bots[i].substr(1));
 
    i++;
 
    }
 
}
 
} else {
 
users = false; //fallback when page doesn't exist
 
}
 
// Temporary global debugging variables
 
JWB.debug = [groups.indexOf('bot'), users === false, bots && bots.indexOf(JWB.username)];
 
JWB.bot = groups.indexOf('bot') !== -1 && (users === false || bots.indexOf(JWB.username) !== -1);
 
JWB.sysop = groups.indexOf('sysop') !== -1;
 
if (JWB.username === "Uziel302" && response.query.userinfo.id === 13299994) {//TEMP: Dev full access to entire interface.
 
JWB.bot = true;
 
users.push("Uziel302");
 
}
 
if (JWB.sysop || response.query.pageids[0] === '-1' || users.indexOf(JWB.username) !== -1 || users === false) {
 
JWB.allowed = true;
 
if (JWB.messages.en) JWB.init(); //init if messages have already loaded
 
} else {
 
if (JWB.messages.en) {
 
//run this after messages have loaded, so the message that shows is in the user's language
 
alert(JWB.msg('not-on-list'));
 
}
 
JWB = false; //prevent further access
 
}
 
}).fail(function(xhr, error) {
 
alert(JWB.msg('verify-error') + '\n' + error);
 
JWB = false; //preventing further access. No verification => no access.
 
});
 
})();
 
 
 
/***** Global object/variables *****/
 
 
 
var objs = ['page', 'api', 'fn', 'pl', 'messages', 'setup', 'settings', 'ns'];
 
for (var i=0;i<objs.length;i++) {
 
JWB[objs[i]] = {};
 
}
 
JWB.summarySuffix = ' (דרך [[WP:JWB]])';
 
if (document.location.hostname == 'en.wikipedia.org') JWB.summarySuffix = ' (via [[WP:JWB]])'
 
JWB.lang = mw.config.get('wgUserLanguage');
 
JWB.index_php = mw.config.get('wgScript');
 
JWB.isStopped = true;
 
JWB.tooltip = window.tooltipAccessKeyPrefix || '';
 
JWB.configext = 'js';
 
if (document.location.hostname.split('.').slice(-2).join('.') == 'wikia.com' || document.location.hostname.split('.').slice(-2).join('.') == 'fandom.com') {
 
//it makes no sense, but Wikia does not allow users to create .js subpages of their userpage.
 
//Because the settings should REALLY be protected from vandalism automatically, the backup is .css
 
//even though this has nothing to do with CSS.
 
JWB.configext = 'css';
 
}
 
JWB.settingspage = 'JWB';
 
if (window.hasOwnProperty('JWBSETTINGS')) {
 
JWB.settingspage = JWBSETTINGS;
 
delete window.JWBSETTINGS; //clean up the global variable
 
}
 
 
 
/***** Google Analytics *****/
 
 
 
JWB.loadAnalytics = function() {
 
$.getScript('https://www.googletagmanager.com/gtag/js?id=UA-149754182-1', function() {
 
window.dataLayer = window.dataLayer || [];
 
function gtag(){
 
dataLayer.push(arguments);
 
}
 
gtag('js', new Date());
 
gtag('config', 'UA-149754182-1', { 'anonymize_ip': true }); // anonymize_ip to comply with GDPR
 
});
 
}
 
 
 
/***** API functions *****/
 
 
 
//Main template for API calls
 
JWB.api.call = function(data, callback, onerror) {
 
data.format = 'json';
 
if (data.action !== 'query') data.bot = true;
 
$.ajax({
 
data: data,
 
dataType: 'json',
 
url: mw.config.get('wgScriptPath') + '/api.php',
 
type: 'POST',
 
success: function(response) {
 
if (response.error) {
 
alert('API error: ' + response.error.info);
 
JWB.stop();
 
} else {
 
callback(response);
 
}
 
},
 
error: function(xhr, error) {
 
alert('AJAX error: ' + error);
 
JWB.stop();
 
if (onerror) onerror();
 
}
 
});
 
};
 
 
 
//Get page diff, and process it for more interactivity
 
JWB.api.diff = function(callback) {
 
JWB.status('diff');
 
var editBoxInput = $('#editBoxArea').val();
 
var redirects = $('input.redirects:checked').val()==='follow'?'1':'0';
 
var data = {
 
'action': 'query',
 
'prop': 'info|revisions',
 
'indexpageids': true,
 
'titles': JWB.page.name,
 
'rvlimit': '1',
 
'rvdifftotext': editBoxInput,
 
'redirects': redirects
 
};
 
JWB.api.call(data, function(response) {
 
var pageExists = response.query.pageids[0] !== '-1';
 
var diff;
 
if (pageExists) {
 
var diffpage = response.query.pages[response.query.pageids[0]];
 
diff = diffpage.revisions[0].diff['*'];
 
if (diff === '') {
 
diff = '<h2>'+JWB.msg('no-changes-made')+'</h2>';
 
} else {
 
diff = '<table class="diff">'+
 
'<colgroup>'+
 
'<col class="diff-marker">'+
 
'<col class="diff-content">'+
 
'<col class="diff-marker">'+
 
'<col class="diff-content">'+
 
'</colgroup>'+
 
'<tbody>'+diff+'</tbody></table>';
 
}
 
} else {
 
diff = '<span style="font-weight:bold;color:red;">'+JWB.msg('page-not-exists')+'</span>';
 
}
 
$('#resultWindow').html(diff);
 
$('.diff-lineno').each(function() {
 
$(this).parent().attr('data-line',parseInt($(this).html().match(/\d+/)[0])-1).addClass('lineheader');
 
});
 
$('table.diff tr').each(function() { //add data-line attribute to every line, relative to the previous one. Used for click event.
 
if (!$(this).next().is('[data-line]') && !$(this).next().has('td.diff-deletedline + td.diff-empty')) {
 
$(this).next().attr('data-line',parseInt($(this).data('line'))+1);
 
} else if ($(this).next().has('td.diff-deletedline + td.diff-empty')) {
 
$(this).next().attr('data-line',$(this).data('line')); //copy over current data-line for deleted lines to prevent them from messing up counting.
 
}
 
});
 
JWB.status('done', true);
 
if (typeof(callback) === 'function') {
 
callback();
 
}
 
});
 
};
 
 
 
//Retrieve page contents/info, process them, and store information in JWB.page object.
 
JWB.api.get = function(pagename) {
 
JWB.pageCount();
 
if (!JWB.list[0] || JWB.isStopped) {
 
return JWB.stop();
 
}
 
if (pagename === '#PRE-PARSE-STOP') {
 
var curval = $('#articleList').val();
 
$('#articleList').val(curval.substr(curval.indexOf('\n') + 1));
 
$('#preparse').prop('checked', false);
 
JWB.stop();
 
return;
 
}
 
var redirect = $('input.redirects:checked').val();
 
var data = {
 
'action': 'query',
 
'prop': 'info|revisions',
 
'inprop': 'watched',
 
'intoken': 'edit|delete|protect|move|watch',
 
'titles': pagename,
 
'rvprop': 'content|timestamp|ids',
 
'rvlimit': '1',
 
'indexpageids': true,
 
'meta': 'userinfo',
 
'uiprop': 'hasmsg'
 
};
 
if (redirect=='follow'||redirect=='skip') data.redirects = true;
 
if (JWB.sysop) {
 
data.list = 'deletedrevs';
 
data.drprop = 'token';
 
}
 
JWB.status('load-page');
 
JWB.api.call(data, function(response) {
 
if (response.query.userinfo.hasOwnProperty('messages')) {
 
var view = mw.config.get('wgScriptPath') + '?title=Special:MyTalk';
 
var viewNew = view + '&diff=cur';
 
JWB.status(
 
'<span style="color:red;font-weight:bold;">'+
 
JWB.msg('status-newmsg',
 
'<a href="'+view+'" target="_blank">'+JWB.msg('status-talklink')+'</a>',
 
'<a href="'+viewNew+'" target="_blank">'+JWB.msg('status-difflink')+'</a>')+
 
'</span>', true);
 
alert(JWB.msg('new-message'));
 
JWB.stop();
 
return;
 
}
 
JWB.page = response.query.pages[response.query.pageids[0]];
 
JWB.page.name = JWB.list[0].split('|')[0];
 
var varOffset = JWB.list[0].indexOf('|') !== -1 ? JWB.list[0].indexOf('|') + 1 : 0;
 
JWB.page.pagevar = JWB.list[0].substr(varOffset);
 
JWB.page.content = JWB.page.revisions ? JWB.page.revisions[0]['*'] : '';
 
JWB.page.exists = !response.query.pages["-1"];
 
JWB.page.deletedrevs = response.query.deletedrevs;
 
JWB.page.watched = JWB.page.hasOwnProperty('watched');
 
if (response.query.redirects) {
 
JWB.page.name = response.query.redirects[0].to;
 
}
 
var newContent = JWB.replace(JWB.page.content);
 
if (JWB.stopped === true) return;
 
JWB.status('done', true);
 
var containRegex = $('#containRegex').prop('checked'), containFlags = $('#containFlags').val();
 
var skipContains = containRegex ? new RegExp($('#skipContains').val(), containFlags) : $('#skipContains').val();
 
var skipNotContains = containRegex ? new RegExp($('#skipNotContains').val(), containFlags) : $('#skipContains').val();
 
if (
 
($('#skipNoChange').prop('checked') && JWB.page.content === newContent) || //skip if no changes are made
 
($('#skipContains').val() && JWB.page.content.match(skipContains)) ||
 
($('#skipNotContains').val() && !JWB.page.content.match(skipNotContains)) ||
 
($('#exists-no').prop('checked') && !JWB.page.exists) ||
 
($('#exists-yes').prop('checked') && JWB.page.exists) ||
 
(redirect==='skip' && response.query.redirects) // variable  redirect  is defined outside this callback function.
 
) {
 
JWB.log('skip', JWB.page.name);
 
return JWB.next();
 
} else {
 
$('#editBoxArea').val(newContent);
 
$('#currentpage').html(JWB.msg('editbox-currentpage', JWB.page.name, encodeURIComponent(JWB.page.name)));
 
if ($('#preparse').prop('checked')) {
 
$('#articleList').val($.trim($('#articleList').val()) + '\n' + JWB.list[0]); //move current page to the bottom
 
JWB.next();
 
return;
 
} else if (JWB.bot && $('#autosave').prop('checked')) {
 
JWB.api.diff(function() {
 
//timeout will take #throttle's value * 1000, if it's a number above 0. Currently defaults to 0.
 
setTimeout(JWB.api.submit, Math.max(+$('#throttle').val() || 0, 0) * 1000, JWB.page.name);
 
});
 
} else {
 
JWB.api.diff();
 
}
 
}
 
JWB.updateButtons();
 
});
 
};
 
 
 
//Some functions with self-explanatory names:
 
JWB.api.submit = function(page) {
 
JWB.status('submit');
 
var summary = $('#summary').val();
 
if ($('#summary').parent('label').hasClass('viaJWB')) summary += JWB.summarySuffix;
 
if ((typeof page === 'text' && page !== JWB.page.name) || $('#currentpage a').html().replace(/&amp;/g, '&') !== JWB.page.name) {
 
console.log(page, JWB.page.name, $('#currentpage a').html())
 
JWB.stop();
 
alert(JWB.msg('autosave-error', JWB.msg('tab-log')));
 
$('#currentpage').html(JWB.msg('editbox-currentpage', ' ', ' '));
 
return;
 
}
 
var data = {
 
'title': JWB.page.name,
 
'summary': summary,
 
'action': 'edit',
 
'basetimestamp': JWB.page.revisions ? JWB.page.revisions[0].timestamp : '',
 
'token': JWB.page.edittoken,
 
'text': $('#editBoxArea').val(),
 
'watchlist': $('#watchPage').val(),
 
'bot':true
 
};
 
if ($('#minorEdit').prop('checked')) data.minor = true;
 
JWB.api.call(data, function(response) {
 
JWB.log('edit', response.edit.title, response.edit.newrevid);
 
JWB.status('done', true);
 
JWB.next();
 
});
 
};
 
JWB.api.preview = function() {
 
JWB.status('preview');
 
JWB.api.call({
 
'title': JWB.page.name,
 
'action': 'parse',
 
'text': $('#editBoxArea').val()
 
}, function(response) {
 
$('#resultWindow').html(response.parse.text['*']);
 
$('#resultWindow div.previewnote').remove();
 
JWB.status('done', true);
 
});
 
};
 
JWB.api.move = function() {
 
JWB.status('move');
 
var topage = $('#moveTo').val().replace(/\$x/gi, JWB.page.pagevar);
 
var summary = $('#summary').val();
 
if ($('#summary').parent('label').hasClass('viaJWB')) summary += JWB.summarySuffix;
 
var data = {
 
'action':'move',
 
'from': JWB.page.name,
 
'to': topage,
 
'token': JWB.page.movetoken,
 
'reason': summary,
 
'ignorewarnings': 'yes'
 
};
 
if ($('#moveTalk').prop('checked')) data.movetalk = true;
 
if ($('#moveSubpage').prop('checked')) data.movesubpages = true;
 
if ($('#suppressRedir').prop('checked')) data.noredirect = true;
 
JWB.api.call(data, function(response) {
 
JWB.log('move', response.move.from, reponse.move.to);
 
JWB.status('done', true);
 
if (!$('#moveTo').val().match(/\$x/i)) $('#moveTo').val('')[0].focus(); //clear entered move-to pagename if it's not based on the pagevar
 
JWB.next(topage);
 
});
 
};
 
JWB.api.delete = function() {
 
JWB.status(($('#deletePage').is('.undelete') ? 'un' : '') + 'delete');
 
var summary = $('#summary').val();
 
if ($('#summary').parent('label').hasClass('viaJWB')) summary += JWB.summarySuffix;
 
var undeltoken = JWB.page.deletedrevs && JWB.page.deletedrevs[0] ? JWB.page.deletedrevs[0].token : '';
 
JWB.api.call({
 
'action': (!JWB.page.exists ? 'un' : '') + 'delete',
 
'title': JWB.page.name,
 
'token': JWB.page.exists ? JWB.page.deletetoken : undeltoken,
 
'reason': summary
 
}, function(response) {
 
JWB.log((!JWB.page.exists ? 'un' : '') + 'delete', (response.delete||response.undelete).title);
 
JWB.status('done', true);
 
JWB.next(response.undelete && response.undelete.title);
 
});
 
};
 
JWB.api.protect = function() {
 
JWB.status('protect');
 
var summary = $('#summary').val();
 
if ($('#summary').parent('label').hasClass('viaJWB')) summary += JWB.summarySuffix;
 
var editprot = $('#editProt').val();
 
var moveprot = $('#moveProt').val();
 
JWB.api.call({
 
'action':'protect',
 
'title': JWB.page.name,
 
'token': JWB.page.protecttoken,
 
'reason': summary,
 
'expiry': $('#protectExpiry').val()!==''?$('#protectExpiry').val():'infinite',
 
'protections': (JWB.page.exists?'edit='+editprot+'|move='+moveprot:'create='+editprot)
 
}, function(response) {
 
var protactions = '';
 
var prots = response.protect.protections;
 
for (var i=0;i<prots.length;i++) {
 
if (typeof prots[i].edit == 'string') {
 
protactions += ' edit: '+(prots[i].edit?prots[i].edit:'all');
 
} else if (typeof prots[i].move == 'string') {
 
protactions += ' move: '+(prots[i].move?prots[i].move:'all');
 
} else if (typeof prots[i].create == 'string') {
 
protactions += ' create: '+(prots[i].create?prots[i].create:'all');
 
}
 
}
 
protactions += ' expires: '+prots[0].expiry;
 
JWB.log('protect', response.protect.title, protactions);
 
JWB.status('done', false);
 
JWB.next(response.protect.title);
 
});
 
};
 
 
 
JWB.api.watch = function() {
 
JWB.status('watch');
 
var data = {
 
'action':'watch',
 
'title':JWB.page.name,
 
'token':JWB.page.watchtoken
 
};
 
if (JWB.page.watched) data.unwatch = true;
 
JWB.api.call(data, function(response) {
 
JWB.status('<span style="color:green;">'+
 
JWB.msg('status-watch-'+(JWB.page.watched ? 'removed' : 'added'), "'"+JWB.page.name+"'")+
 
'</span>', true);
 
JWB.page.watched = !JWB.page.watched;
 
$('#watchNow').html( JWB.msg('watch-' + (JWB.page.watched ? 'remove' : 'add')) );
 
});
 
};
 
 
 
/***** Pagelist functions *****/
 
 
 
JWB.pl.list = [];
 
JWB.pl.iterations = 0;
 
 
 
JWB.pl.stop = function() {
 
JWB.pl.iterations = 0;
 
$('#pagelistPopup [disabled]:not(fieldset [disabled]), #pagelistPopup legend input').prop('disabled', false);
 
$('#pagelistPopup legend input').trigger('change');
 
$('#pagelistPopup button img').remove();
 
}
 
 
 
JWB.pl.getNSpaces = function() {
 
var list = $('#pagelistPopup [name="namespace"]')[0];
 
if (list.selectedOptions.length == list.options.length) {
 
return ''; //return empty string if every namespace is selected; this will make the request default to having no filter
 
} else {
 
return $('#pagelistPopup [name="namespace"]').val().join('|'); //.val() returns an array of selected options.
 
}
 
};
 
 
 
/**
 
* abbrs: array of abbreviations for each page generator enabled.
 
* ps = prefix search, pl = links on page, etc.
 
* note that the "Links to page" section may generate multiple generators (each
 
* of wikilinks, transclusions, and file usage are done separately)
 
* lists: as above, but full names. Possible elements include:
 
* prefixsearch, links, watchlistraw
 
* These correspond to possible "list" values for https://www.mediawiki.org/wiki/API:Query
 
* Er, except when they don't. "links" is not a valid list value (but this just causes a benign warning)
 
* data: object with keys for each value in lists, plus "continue". For each generator,
 
* the corresponding value will be the user input related to that generator
 
* (e.g. the category name, the title prefix). If there are multiple values for
 
* a generator, they'll be pipe-delimited within the string.
 
* More keys will be added to this obj during this function's execution.
 
* (This is ultimately used as POST data for API calls)
 
**/
 
JWB.pl.getList = function(abbrs, lists, data) {
 
$('#pagelistPopup button, #pagelistPopup input, #pagelistPopup select').prop('disabled', true);
 
JWB.pl.iterations++;
 
// cf. https://www.mediawiki.org/wiki/API:Query
 
data.action = 'query';
 
var nspaces = JWB.pl.getNSpaces();
 
for (var i=0;i<abbrs.length;i++) {
 
if (nspaces) data[abbrs[i]+'namespace'] = nspaces;
 
data[abbrs[i]+'limit'] = 'max';
 
}
 
if (lists.indexOf('links') !== -1) {
 
// If using the "Links on page" feature, then ask the API to give us the links
 
// on each returned page
 
data.prop = 'links';
 
}
 
data.list = lists.join('|');
 
JWB.api.call(data, function(response) {
 
var maxiterate = 100; //allow up to 100 consecutive requests at a time to avoid overloading the server.
 
if (!response.query) response.query = {};
 
if (response.watchlistraw) response.query.watchlistraw = response.watchlistraw; //adding some consistency
 
var plist = [];
 
if (response.query.pages) {
 
var links;
 
for (var id in response.query.pages) {
 
links = response.query.pages[id].links;
 
for (var i=0;i<links.length;i++) {
 
plist.push(links[i].title);
 
}
 
}
 
}
 
for (var l in response.query) {
 
if (l === 'pages') continue;
 
for (var i=0;i<response.query[l].length;i++) {
 
plist.push(response.query[l][i].title);
 
}
 
}
 
//add the result to the pagelist immediately, as opposed to saving it all up and adding in 1 go like AWB does
 
$('#articleList').val($.trim($('#articleList').val()) + '\n' + plist.join('\n'));
 
JWB.pageCount();
 
var cont = response.continue;
 
console.log("Continue",JWB.pl.iterations, cont);
 
if (cont && JWB.pl.iterations <= maxiterate) {
 
var lists = [];
 
if (response.query) { //compatibility with the code I wrote for the old query-continue. TODO: make this unnecessary?
 
for (var list in response.query) {
 
lists.push(list); //add to the new array of &list= values
 
}
 
}
 
var abbrs = [];
 
for (var abbr in cont) {
 
data[abbr] = cont[abbr]; //add the &xxcontinue= value to the data
 
if (abbr != 'continue') {
 
abbrs.push(abbr.replace('continue','')); //find out what xx is and add it to the list of abbrs
 
}
 
}
 
JWB.pl.getList(abbrs, lists, data); //recursive function to get every page of a list
 
} else {
 
if (JWB.pl.iterations > maxiterate) {
 
JWB.status('pl-over-lim', true);
 
} else {
 
JWB.status('done', true);
 
}
 
JWB.pl.stop();
 
}
 
}, function() { //on error, simply reset and let the user work with what he has
 
JWB.status('done', true);
 
JWB.pl.stop();
 
});
 
};
 
 
 
//JWB.pl.getList(['wr'], ['watchlistraw'], {}) for watchlists
 
JWB.pl.generate = function() {
 
var $fields = $('#pagelistPopup fieldset').not('[disabled]');
 
var spinner = '<img src="//upload.wikimedia.org/wikipedia/commons/d/de/Ajax-loader.gif" width="15" height="15" alt="'+JWB.msg('status-alt')+'"/>';
 
$('#pagelistPopup').find('button[type="submit"]').append(spinner);
 
var abbrs = [], lists = [], data = {'continue': ''};
 
$fields.each(function() {
 
var list = $(this).find('legend input').attr('name');
 
var abbr;
 
if (list === 'linksto') { //Special case since this fieldset features 3 merged lists in 1 fieldset
 
if (!$('[name="title"]').val()) return;
 
$('[name="backlinks"], [name="embeddedin"], [name="imageusage"]').filter(':checked').each(function() {
 
var val = this.value;
 
abbrs.push(val);
 
lists.push(this.name);
 
data[val+'title'] = $('[name="title"]').val();
 
data[val+'filterredir'] = $('[name="filterredir"]:checked').val();
 
if ($('[name="redirect"]').prop('checked')) data[val+'redirect'] = true;
 
});
 
} else { //default input system
 
abbr = $(this).find('legend input').val();
 
lists.push(list);
 
abbrs.push(abbr);
 
// For each input in this fieldset other than the checkbox enabling it,
 
// add a key-value pair to data, where the key is the input's name and
 
// the value is the value (or pipe-separated string of values, if multiple)
 
// (These will correspond to params of the API query)
 
$(this).find('input').not('legend input').each(function() {
 
if ((this.type === 'checkbox' || this.type === 'radio') && this.checked === false) return;
 
if ($(this).is('[name="cmtitle"]')) {
 
//making sure every page has a Category: prefix, in case the user left it out
 
var newval = $(this).val().replace(new RegExp(JWB.ns[14]['*']+':', 'gi'), '').split('|')[0];
 
$(this).val(JWB.ns[14]['*']+':'+newval);
 
}
 
var name = this.name;
 
var val = this.value;
 
if (data.hasOwnProperty(name)) {
 
data[name] += '|'+val;
 
} else {
 
data[name] = val;
 
}
 
});
 
console.log("abbrs, lists, data:", abbrs, lists, data);
 
}
 
});
 
if (abbrs.length) JWB.pl.getList(abbrs, lists, data);
 
};
 
 
 
/***** Setup functions *****/
 
 
 
JWB.setup.save = function(name) {
 
name = name || prompt(JWB.msg('setup-prompt', JWB.msg('setup-prompt-store')), $('#loadSettings').val());
 
if (name === null) return;
 
var self = JWB.settings[name] = {
 
string: {},
 
bool: {},
 
replaces: []
 
};
 
//inputs with a text value
 
$('textarea, input[type="text"], input[type="number"], select').not('.replaces input, #editBoxArea, #settings *, #google-analytics').each(function() {
 
if (typeof $(this).val() == 'string') {
 
self.string[this.id] = this.value.replace(/\n{2,}/g,'\n');
 
} else {
 
self.string[this.id] = $(this).val();
 
}
 
});
 
self.replaces = [];
 
$('.replaces').each(function() {
 
if ($(this).find('.replaceText').val() || $(this).find('.replaceWith').val()) {
 
self.replaces.push({
 
replaceText: $(this).find('.replaceText').val(),
 
replaceWith: $(this).find('.replaceWith').val(),
 
useRegex: $(this).find('.useRegex').prop('checked'),
 
regexFlags: $(this).find('.regexFlags').val(),
 
ignoreNowiki: $(this).find('.ignoreNowiki').prop('checked')
 
});
 
}
 
});
 
$('input[type="radio"], input[type="checkbox"]').not('.replaces input').each(function() {
 
self.bool[this.id] = this.checked;
 
});
 
if (!$('#loadSettings option[value="'+name+'"]').length) {
 
$('#loadSettings').append('<option value="'+name+'">'+name+'</option>');
 
}
 
$('#loadSettings').val(name);
 
console.log(self);
 
};
 
 
 
JWB.setup.apply = function(name) {
 
name = name && JWB.settings[name] ? name : 'default';
 
var self = JWB.settings[name];
 
$('#loadSettings').val(name);
 
$('.replaces + .replaces').remove(); //reset find&replace inputs
 
$('.replaces input[type="text"]').val('');
 
$('.useRegex').each(function() {this.checked = false;});
 
$('#pagelistPopup legend input').trigger('change'); //fix checked state of pagelist generating inputs
 
for (var a in self.string) {
 
$('#'+a).val(self.string[a]);
 
}
 
for (var b in self.bool) {
 
$('#'+b).prop('checked', self.bool[b]);
 
}
 
var cur;
 
for (var c=0;c<self.replaces.length;c++) {
 
if ($('.replaces').length <= c) $('#moreReplaces')[0].click();
 
cur = self.replaces[c];
 
for (var d in cur) {
 
if (cur[d] === true || cur[d] === false) {
 
$('.replaces').eq(c).find('.'+d).prop('checked', cur[d]);
 
} else {
 
$('.replaces').eq(c).find('.'+d).val(cur[d]);
 
}
 
}
 
}
 
$('.useRegex, #containRegex,'+
 
  '#pagelistPopup legend input,'+
 
  '#viaJWB').trigger('change'); //reset disabled inputs
 
};
 
 
 
JWB.setup.getObj = function() {
 
// google analytics preferences do not depend on which settings profile, so should be stored outside that.
 
JWB.settings['_google_analytics'] = $('#allow-analytics').prop('checked');
 
var settings = [];
 
for (var i in JWB.settings) {
 
if (i != '_blank') {
 
settings.push('"' + i + '": ' + JSON.stringify(JWB.settings[i]));
 
}
 
}
 
return '{\n\t' + settings.join(',\n\t') + '\n}';
 
};
 
 
 
JWB.setup.submit = function() {
 
var name = prompt(JWB.msg('setup-prompt', JWB.msg('setup-prompt-save')), $('#loadSettings').val());
 
if (name === null) return;
 
if ($.trim(name) === '') name = 'default';
 
JWB.setup.save(name);
 
JWB.status('setup-submit');
 
JWB.api.call({
 
'title': 'User:'+JWB.username+'/'+JWB.settingspage+'-settings.'+JWB.configext,
 
'summary': JWB.msg(['setup-summary', mw.config.get('wgContentLanguage')]),
 
'action': 'edit',
 
'token': JWB.setup.edittoken,
 
'text': JWB.setup.getObj(),
 
'minor': true
 
}, function(response) {
 
JWB.status('done', true);
 
});
 
};
 
 
 
//TODO: use blob uri
 
JWB.setup.download = function() {
 
var name = prompt(JWB.msg('setup-prompt', JWB.msg('setup-prompt-save')), $('#loadSettings').val());
 
if (name === null) return;
 
if ($.trim(name) === '') name = 'default';
 
JWB.setup.save(name);
 
JWB.status('setup-dload');
 
var url = 'data:application/json;base64,' + btoa(JWB.setup.getObj());
 
var elem = $('#download-anchor')[0];
 
if (elem.hasOwnProperty('download')) { //use download attribute when possible, for its ability to specify a filename
 
elem.href = url;
 
elem.click();
 
setTimeout(function() {elem.removeAttribute('href');}, 2000);
 
} else { //fallback to iframes for browsers with no support for download="" attributes
 
elem = $('#download-iframe')[0];
 
elem.src = url.replace('application/json', 'application/octet-stream');
 
setTimeout(function() {elem.removeAttribute('src');}, 2000);
 
}
 
JWB.status('done', true);
 
};
 
 
 
JWB.setup.import = function(e) {
 
e.preventDefault();
 
file = (e.dataTransfer||this).files[0];
 
if ($(this).is('#import')) { //reset input
 
this.outerHTML = this.outerHTML;
 
$('#import').change(JWB.setup.import);
 
}
 
if (!window.hasOwnProperty('FileReader')) {
 
alert(JWB.msg('old-browser'));
 
JWB.status('old-browser', '<a target="_blank" href="'+JWB.index_php+'?title=Special:MyPage/'+JWB.settingspage+'-settings.'+JWB.configext+'">/'+JWB.settingspage+'-settings.'+JWB.configext+'</a>');
 
return;
 
}
 
if (file.name.split('.').pop().toLowerCase() !== 'json') {
 
alert(JWB.msg('not-json'));
 
return;
 
}
 
JWB.status('Processing file');
 
var reader = new FileReader();
 
reader.readAsText(file);
 
reader.onload = function(e) {
 
JWB.status('done', true);
 
try {
 
//Exclusion regex based on http://stackoverflow.com/a/23589204/1256925
 
//Removes all JS comments from the file, except when they're between quotes.
 
var data = JSON.parse(reader.result.replace(/("[^"]*")|(\/\*[\w\W]*\*\/|\/\/[^\n]*)/g, function(match, g1, g2) {
 
if (g1) return g1;
 
}));
 
} catch(e) {
 
alert(JWB.msg('json-err', e.message, JWB.msg('json-err-upload')));
 
console.log(e); //also log the error for further info
 
return;
 
}
 
JWB.setup.extend(data);
 
};
 
 
JWB.status('Processing file');
 
};
 
 
 
JWB.setup.load = function() {
 
 
//addition forcing LTR orientation
 
/**
 
* Forces left-to-right layout and editing on RTL wikis.
 
* @see https://meta.wikimedia.org/wiki/Force_ltr
 
* @update-token [[File:pathoschild/forceltr.js]]
 
*/
 
mw.loader.load('//tools-static.wmflabs.org/meta/scripts/pathoschild.forceltr.js');
 
 
JWB.status('setup-load');
 
JWB.api.call({
 
'action': 'query',
 
'titles': 'User:' + (JWB.username||mw.config.get('wgUserName')) + '/'+JWB.settingspage+'-settings.'+JWB.configext,
 
'prop': 'info|revisions',
 
'intoken': 'edit',
 
'rvprop': 'content',
 
'indexpageids': true
 
}, function(response) {
 
JWB.status('done', true);
 
if (JWB === false) return; //user is not allowed to use JWB
 
var firstrun =  JWB.setup.edittoken ? false : true;
 
var page = response.query.pages[response.query.pageids[0]];
 
JWB.setup.edittoken = page.edittoken;
 
if (response.query.pageids[0] === '-1') {
 
if (JWB.allowed && firstrun) JWB.setup.save('default'); //this runs when this callback returns after the init has loaded.
 
JWB.loadAnalytics(); // load Analytics, because user did not opt-out (no settings page)
 
return;
 
}
 
var data = page.revisions[0]['*'];
 
if (!data) {
 
if (JWB.allowed && firstrun) JWB.setup.save('default'); //this runs when this callback returns after the init has loaded.
 
JWB.loadAnalytics(); // load Analytics, because user did not opt-out (empty settings page)
 
return;
 
}
 
try {
 
data = JSON.parse(data);
 
} catch(e) {
 
alert(JWB.msg('json-err', e.message, JWB.msg('json-err-page', JWB.settingspage)) || 'JSON error:\n'+e.message);
 
JWB.setup.save('default');
 
JWB.loadAnalytics(); // load Analytics, because user did not opt-out (no valid settings page)
 
return;
 
}
 
if (data["_google_analytics"] === false) {
 
$('#allow-analytics').prop('checked', false); // user opts out of analytics
 
} else {
 
JWB.loadAnalytics(); // if user's settings exist, and the google analytics preference is not false, load GA.
 
delete data["_google_analytics"]; // prevent "_google_analytics" from appearing as a profile
 
}
 
JWB.setup.extend(data);
 
});
 
};
 
 
 
JWB.setup.extend = function(obj) {
 
$.extend(JWB.settings, obj);
 
if (!JWB.settings.hasOwnProperty('default')) {
 
JWB.setup.save('default');
 
}
 
for (var i in JWB.settings) {
 
if ($('#loadSettings').find('option[value="'+i+'"]').length) continue;
 
$('#loadSettings').append('<option value="'+i+'">'+i+'</option>');
 
}
 
JWB.setup.apply($('#loadSettings').val());
 
};
 
 
 
JWB.setup.delete = function() {
 
var name = $('#loadSettings').val();
 
if (name === '_blank') return alert(JWB.msg('setup-delete-blank'));
 
var temp = {};
 
temp[name] = JWB.settings[name];
 
JWB.setup.temp = $.extend({}, temp);
 
delete JWB.settings[name];
 
$('#loadSettings').val('default');
 
if (name === 'default') {
 
JWB.setup.apply('_blank');
 
JWB.setup.save('default');
 
JWB.status(JWB.msg('status-del-default', '<a href="javascript:JWB.setup.undelete();">'+JWB.msg('status-del-undo')+'</a>'), true);
 
} else {
 
$('#loadSettings').find('[value="'+name+'"]').remove();
 
JWB.setup.apply();
 
JWB.status(JWB.msg('status-del-setup', name, '<a href="javascript:JWB.setup.undelete();">'+JWB.msg('status-del-undo')+'</a>'), true);
 
}
 
};
 
JWB.setup.undelete = function() {
 
JWB.setup.extend(JWB.setup.temp);
 
JWB.status('done', true);
 
};
 
 
 
/***** Main other functions *****/
 
 
 
//Show status message
 
JWB.status = function(action, done) {
 
$('#summary, .editbutton').prop('disabled', !done); //Disable box when not done (so busy loading). re-enable when done loading.
 
var status = JWB.msg('status-'+action);
 
if (status === false) return;
 
var spinImg = '<img src="//upload.wikimedia.org/wikipedia/commons/d/de/Ajax-loader.gif" width="15" height="15" alt="'+JWB.msg('status-alt')+'"/>';
 
if (status) {
 
if (!done) { //spinner if not done
 
status += ' ' + spinImg;
 
}
 
} else {
 
status = action;
 
}
 
$('#status').html(status);
 
JWB.pageCount();
 
return action=='done';
 
};
 
 
 
JWB.pageCount = function() {
 
if (JWB.allowed === false||!$('#articleList').length) return;
 
$('#articleList').val(($('#articleList').val()||'').replace(/(^[ \t]*$\n)*/gm, ''));
 
JWB.list = $('#articleList').val().split('\n');
 
var count = JWB.list.length;
 
if (count === 1 && JWB.list[0] === '') count = 0;
 
$('#totPages').html(count);
 
};
 
 
 
//Perform all specified find&replace actions
 
JWB.replace = function(input) {
 
JWB.pageCount();
 
var varOffset = JWB.list[0].indexOf('|') !== -1 ? JWB.list[0].indexOf('|') + 1 : 0;
 
JWB.page.pagevar = JWB.list[0].substr(varOffset);
 
$('.replaces').each(function() {
 
var $this = $(this);
 
var regexFlags = $this.find('.regexFlags').val();
 
var replace = $this.find('.replaceText').val().replace(/\$x/gi, JWB.page.pagevar).replace(/\\{2}/g, '\\').replace(/\\n/g,'\n') || '$';
 
var useRegex = replace === '$' || $this.find('.useRegex').prop('checked');
 
if (useRegex && regexFlags.indexOf('_') !== -1) {
 
replace = replace.replace(/[ _]/g, '[ _]'); //replaces any of [Space OR underscore] with a match for spaces or underscores.
 
replace = replace.replace(/(\[[^\]]*)\[ _\]/g, '$1 _'); //in case a [ _] was placed inside another [] match, remove the [].
 
regexFlags = regexFlags.replace('_', '');
 
}
 
//apply replaces where \n and \\ work in both regular text and regex mode.
 
var rWith = $this.find('.replaceWith').val().replace(/\$x/gi, JWB.page.pagevar).replace(/\\{2}/g, '\\').replace(/\\n/g,'\n');
 
try {
 
if ($this.find('.ignoreNowiki').prop('checked')) {
 
if (!useRegex) {
 
replace = replace.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&');
 
regexFlags = 'g';
 
}
 
input = JWB.replaceParsed(input, replace, regexFlags, rWith);
 
} else if (useRegex) {
 
replace = new RegExp(replace, regexFlags);
 
input = input.replace(replace, rWith);
 
} else {
 
input = input.split(replace).join(rWith); //global replacement without having to escape all special chars.
 
}
 
} catch(e) {
 
JWB.stop();
 
return JWB.status('regex-err', false);
 
}
 
});
 
if ($('#enableRETF').prop('checked')) {
 
input = RETF.replace(input);
 
}
 
return input;
 
};
 
 
 
//function to *only* replace the parsed wikitext
 
//It excludes comments (<!-- -->), and nowiki, math, source, syntaxhighlight, pre, code, gallery and timeline tags)
 
//Based on http://stackoverflow.com/a/23589204/1256925
 
JWB.replaceParsed = function(str, replace, flags, rwith) {
 
var exclude = '(<!--[\\s\\S]*?-->|<(nowiki|math|source|syntaxhighlight|pre|gallery|timeline)[^>]*?>[\\s\\S]*?<\\/\\2>)';
 
//add /i flag, to exclude the correct tags regardless of casing.
 
//This won't matter for the actual replacing, as the specified flags are used there.
 
var re = new RegExp(exclude + '|(' + replace + ')', flags.replace(/i|$/, 'i'));
 
return str.replace(re, function(match, g1, g2, g3) {
 
if (g3 !== undefined) { //continue to perform replacement if the match is the group that's supposed to be the match
 
return match.replace(new RegExp(replace, flags), rwith);
 
} else { //do nothing if the match is one of the excluded groups
 
return match;
 
}
 
});
 
};
 
 
 
//Adds a line to the logs tab.
 
JWB.log = function(action, page, info) {
 
var d = new Date();
 
var pagee = encodeURIComponent(page);
 
var extraInfo = '', actionStat = '';
 
switch (action) {
 
case 'edit':
 
if (typeof info === 'undefined') {
 
action = 'null-edit';
 
actionStat = 'nullEdits';
 
extraInfo = '';
 
} else {
 
extraInfo = ' (<a target="_blank" href="'+JWB.index_php+'?title='+pagee+'&diff='+info+'">diff</a>)';
 
actionStat = 'pagesSaved';
 
}
 
break;
 
case 'skip':
 
actionStat = 'pagesSkipped';
 
break;
 
case 'move':
 
extraInfo = ' to <a target="_blank" href="/wiki/'+encodeURIComponent(info)+'" title="'+info+'">'+info+'</a>';
 
break;
 
case 'protect':
 
extraInfo = info;
 
break;
 
}
 
actionStat = '#' + (actionStat || 'otherActions');
 
$(actionStat).html(+$(actionStat).html() + 1);
 
$('#actionlog tbody')
 
.append('<tr>'+
 
'<td>'+(JWB.fn.pad0(d.getHours())+':'+JWB.fn.pad0(d.getMinutes())+':'+JWB.fn.pad0(d.getSeconds()))+'</td>'+
 
'<th>'+action+'</th>'+
 
'<td><a target="_blank" href="/wiki/'+pagee+'" title="'+page+'">'+page+'</a>'+ extraInfo +'</td>'+
 
'</tr>')
 
.parents('.JWBtabc').scrollTop($('#actionlog tbody').parents('.JWBtabc')[0].scrollHeight);
 
};
 
 
 
//Move to the next page in the list
 
JWB.next = function(nextPage) {
 
if ($.trim(nextPage) && !$('#skipAfterAction').prop('checked')) {
 
nextPage = $.trim(nextPage) + '\n';
 
} else {
 
nextPage = '';
 
}
 
$('#articleList').val($('#articleList').val().replace(/^.*\n?/, nextPage));
 
JWB.list.splice(0,1);
 
JWB.pageCount();
 
JWB.api.get(JWB.list[0].split('|')[0]);
 
};
 
 
 
//Stop everything, reset inputs and editor
 
JWB.stop = function() {
 
$('#stopbutton,'+
 
  '.editbutton,'+
 
  '#watchNow,'+
 
  '.JWBtabc[data-tab="2"] .editbutton,'+
 
  '#watchNow'+
 
  '.JWBtabc[data-tab="4"] button').prop('disabled', true);
 
$('#startbutton, #articleList,'+
 
  '.JWBtabc[data-tab="1"] button,'+
 
  '#replacesPopup button,'+
 
  '#replacesPopup input,'+
 
  '.JWBtabc input, select').prop('disabled', false);
 
$('#resultWindow').html('');
 
$('#editBoxArea').val('');
 
$('#currentpage').html(JWB.msg('editbox-currentpage', ' ', ' '));
 
JWB.pl.stop();
 
JWB.status('done', true);
 
JWB.isStopped = true;
 
};
 
 
 
//Start AutoWikiBrowsing
 
JWB.start = function() {
 
JWB.pageCount();
 
if (JWB.list.length === 0 || (JWB.list.length === 1 && !JWB.list[0])) {
 
alert(JWB.msg('no-pages-listed'));
 
} else if ($('#skipNoChange').prop('checked') && !$('.replaceText').val() && !$('.replaceWith').val() && !$('#enableRETF').prop('checked')) {
 
alert(JWB.msg('infinite-skip-notice'));
 
} else {
 
JWB.isStopped = false;
 
if ($('#preparse').prop('checked') && !$('#articleList').val().match('#PRE-PARSE-STOP')) {
 
$('#articleList').val($.trim($('#articleList').val()) + '\n#PRE-PARSE-STOP'); //mark where to stop pre-parsing
 
} else {
 
$('#preparse-reset').click();
 
}
 
$('#stopbutton, .editbutton, #watchNow, .JWBtabc[data-tab="2"] button, .JWBtabc[data-tab="4"] button').prop('disabled', false);
 
$('#startbutton, #articleList, .JWBtabc[data-tab="1"] button, #replacesPopup button, #replacesPopup input, .JWBtabc input, select').prop('disabled', true);
 
JWB.api.get(JWB.list[0].split('|')[0]);
 
}
 
};
 
 
 
JWB.updateButtons = function() {
 
if (!JWB.page.exists && $('#deletePage').is('.delete')) {
 
$('#deletePage').removeClass('delete').addClass('undelete').html('Undelete');
 
JWB.fn.blink('#deletePage'); //Indicate the button has changed
 
} else if (JWB.page.exists && $('#deletePage').is('.undelete')) {
 
$('#deletePage').removeClass('undelete').addClass('delete').html('Delete');
 
JWB.fn.blink('#deletePage'); //Indicate the button has changed
 
}
 
if (!JWB.page.exists) {
 
$('#movePage').prop('disabled', true);
 
} else {
 
$('#movePage').prop('disabled', false);
 
}
 
$('#watchNow').html( JWB.msg('watch-' + (JWB.page.watched ? 'remove' : 'add')) );
 
};
 
 
 
/***** General functions *****/
 
 
 
//Clear all existing timers to prevent them from getting errors
 
JWB.fn.clearAllTimeouts = function() {
 
var i = setTimeout(function() {
 
return void(0);
 
}, 1000);
 
for (var n=0;n<=i;n++) {
 
clearTimeout(n);
 
clearInterval(i);
 
}
 
console.log('Cleared all running intervals up to index',i);
 
};
 
 
 
//Filter an array to only contain unique values.
 
JWB.fn.uniques = function(arr) {
 
var a = [];
 
for (var i=0, l=arr.length; i<l; i++) {
 
if (a.indexOf(arr[i]) === -1 && arr[i] !== '') {
 
a.push(arr[i]);
 
}
 
}
 
return a;
 
};
 
 
 
//Prepends zeroes until the number has the desired length of len (default 2)
 
JWB.fn.pad0 = function(n, len) {
 
n = n.toString();
 
len = len||2;
 
return n.length < len ? Array(len-n.length).join('0')+n : n;
 
};
 
 
 
JWB.fn.blink = function(el,t) {
 
t=t?t:500;
 
$(el).prop('disabled', true)
 
.children().animate({opacity:'0.1'},t-100)
 
.animate({opacity:'1'},t)
 
.animate({opacity:'0.1'},t-100)
 
.animate({opacity:'1'},t);
 
setTimeout("$('"+el+"').prop('disabled', false)",t*4-400);
 
};
 
 
 
JWB.fn.setSelection = function(el, start, end, dir) {
 
    dir = dir||'none'; //Default value
 
    end = end||start; //If no end is specified, assume the caret is placed without creating text selection.
 
    if (el.setSelectionRange) {
 
        el.focus();
 
        el.setSelectionRange(start, end, dir);
 
    } else if (el.createTextRange) {
 
        var rng = el.createTextRange();
 
        rng.collapse(true);
 
        rng.moveStart('character', start);
 
        rng.moveEnd('character', end);
 
        rng.select();
 
    }
 
};
 
 
 
JWB.fn.scrollSelection = function(el, index) { //function to fix scrolling to selection - doesn't do that automatically.
 
var newEl = document.createElement('textarea'); //create a new textarea to simulate the same conditions
 
var elStyle = getComputedStyle(el);
 
newEl.style.height = elStyle.height; //copy over size-influencing styles
 
newEl.style.width = elStyle.width;
 
newEl.style.lineHeight = elStyle.lineHeight;
 
newEl.style.fontSize = elStyle.fontSize;
 
newEl.value = el.value.substr(0,index);
 
document.body.appendChild(newEl); //needs to be added to the HTML for the scrollHeight and clientHeight to work.
 
if (newEl.scrollHeight != newEl.clientHeight) {
 
el.scrollTop = newEl.scrollHeight - 2;
 
} else {
 
el.scrollTop = 0;
 
}
 
newEl.remove(); //clean up the mess I've made
 
};
 
 
 
//i18n function
 
JWB.msg = function(message) {
 
var args = arguments;
 
var lang = JWB.lang;
 
if (typeof message === 'object') {
 
lang = message[1];
 
message = message[0];
 
}
 
if (lang == 'qqx') return message;
 
if (!JWB.messages || !JWB.messages.en) return message;
 
var msg;
 
if (JWB.messages.hasOwnProperty(lang) && JWB.messages[lang].hasOwnProperty(message)) {
 
msg = JWB.messages[lang][message];
 
} else {
 
msg = (JWB.messages.en.hasOwnProperty(message)) ? JWB.messages.en[message] : '';
 
}
 
msg = msg.replace(/\$(\d+)/g, function(match, num) {
 
return args[+num] || match;
 
});
 
return msg;
 
};
 
 
 
/***** Init *****/
 
 
 
JWB.init = function() {
 
console.log(JWB.messages.en, !!JWB.messages.en);
 
JWB.setup.load();
 
JWB.fn.clearAllTimeouts();
 
if (!JWB.messages[JWB.lang] && JWB.lang != 'qqx') JWB.lang = 'en';
 
 
var findreplace = '<div class="replaces">'+
 
'<label style="display:block;">'+JWB.msg('label-replace')+' <input type="text" class="replaceText"/></label>'+
 
'<label style="display:block;">'+JWB.msg('label-rwith')+' <input type="text" class="replaceWith"/></label>'+
 
'<div class="regexswitch">'+
 
'<label><input type="checkbox" class="useRegex"> '+JWB.msg('label-useregex')+'</label>'+
 
'<a class="re101" href="http://regex101.com/#javascript" target="_blank">?</a>'+
 
'<label class="divisor" title="'+JWB.msg('tip-regex-flags')+'" style="display:none;">'+
 
JWB.msg('label-regex-flags')+' <input type="text" class="regexFlags" value="g"/>'+ //default: global replacement
 
'</label>'+
 
'<br/>'+
 
'</div>'+
 
'<label title="'+JWB.msg('tip-ignore-comment')+'">'+
 
'<input type="checkbox" class="ignoreNowiki"> '+JWB.msg('label-ignore-comment')+
 
'</label>'+
 
'</div>';
 
 
var NSList = '<select multiple name="namespace" id="namespacelist">';
 
for (var i in JWB.ns) {
 
if (parseInt(i) < 0) continue; //No Special: or Media: in the list
 
if (parseInt(i) == 0) NSList += '<option value="'+JWB.ns[i].id+'" selected>'+(JWB.ns[i]['*'] || '('+JWB.msg('namespace-main')+')')+'</option>';
 
else NSList += '<option value="'+JWB.ns[i].id+'" _blank>'+(JWB.ns[i]['*'] || '('+JWB.msg('namespace-main')+')')+'</option>';
 
}
 
NSList += '</select>';
 
 
/***** Interface *****/
 
 
document.title = 'AWB+search Script'+(document.title.split('-')[1] ? ' -'+document.title.split('-')[1] : '');
 
$('body').html(
 
'<article id="resultWindow"></article>'+
 
'<main id="inputsWindow">'+
 
'<div id="inputsBox">'+
 
'<aside id="articleBox">'+
 
'<b>'+JWB.msg('pagelist-caption')+'</b>'+
 
'<textarea id="articleList"></textarea>'+
 
'</aside>'+
 
'<section id="tabs">'+
 
'<nav class="tabholder">'+
 
'<span class="JWBtab" data-tab="1">'+JWB.msg('tab-setup')+'</span> '+
 
'<span class="JWBtab active" data-tab="2">'+JWB.msg('tab-editing')+'</span> '+
 
'<span class="JWBtab" data-tab="3">'+JWB.msg('tab-skip')+'</span> '+
 
(JWB.sysop?'<span class="JWBtab" data-tab="4">'+JWB.msg('tab-other')+'</span> ':'')+
 
' <span class="JWBtab log" data-tab="5">'+JWB.msg('tab-log')+'</span> '+
 
'</nav>'+
 
'<section class="JWBtabc" data-tab="1"></section>'+
 
'<section class="JWBtabc active" data-tab="2"></section>'+
 
'<section class="JWBtabc" data-tab="3"></section>'+
 
(JWB.sysop?'<section class="JWBtabc" data-tab="4"></section>':'')+
 
'<section class="JWBtabc log" data-tab="5"></section>'+
 
'<footer id="status">done</footer>'+
 
'</section>'+
 
'<aside id="editBox">'+
 
'<b>'+JWB.msg('editbox-caption')+' - <span id="currentpage">'+JWB.msg('editbox-currentpage', ' ', ' ')+'</span></b>'+
 
'<textarea id="editBoxArea"></textarea>'+
 
'</aside>'+
 
'</div>'+
 
'</main>'+
 
'<footer id="stats">'+
 
JWB.msg('stat-pages')+' <span id="totPages">0</span>;&emsp;'+
 
JWB.msg('stat-save')+' <span id="pagesSaved">0</span>;&emsp;'+
 
JWB.msg('stat-null')+' <span id="nullEdits">0</span>;&emsp;'+
 
JWB.msg('stat-skip')+' <span id="pagesSkipped">0</span>;&emsp;'+
 
JWB.msg('stat-other')+' <span id="otherActions">0</span>;&emsp;'+
 
'</footer>'+
 
'<div id="overlay" style="display:none;"></div>'+
 
'<section class="JWBpopup" id="replacesPopup" style="display:none;">'+
 
'<button id="moreReplaces">'+JWB.msg('button-more-fields')+'</button>'+
 
'<br>'+findreplace+
 
'</section>'+
 
'<section class="JWBpopup" id="pagelistPopup" style="display:none;">'+
 
'<form action="#" onsubmit="JWB.pl.generate();event.preventDefault();"></form>'+
 
'</section>'
 
);
 
 
$('.JWBtabc[data-tab="1"]').html(
 
'<fieldset id="pagelist">'+
 
'<legend>'+JWB.msg('label-pagelist')+'</legend>'+
 
'<button id="removeDupes">'+JWB.msg('button-remove-dupes')+'</button> '+
 
'<button id="sortArticles">'+JWB.msg('button-sort')+'</button>'+
 
'<br>'+
 
'<label title="'+JWB.msg('tip-preparse')+'">'+
 
'<input type="checkbox" id="preparse"> '+JWB.msg('preparse')+
 
'</label>'+
 
'<span class="divisor"></span>'+
 
'<button id="preparse-reset" title="'+JWB.msg('tip-preparse-reset')+'">'+JWB.msg('preparse-reset')+'</button>'+
 
'<br>'+
 
'<button id="pagelistButton">'+JWB.msg('pagelist-generate')+'</button>'+
 
'</fieldset>'+
 
'<fieldset id="settings">'+
 
'<legend>'+JWB.msg('label-settings')+'</legend>'+
 
'<button id="saveAs" title="'+JWB.msg('tip-store-setup')+'">'+JWB.msg('store-setup')+'</button>'+
 
'<br>'+
 
'<label>'+
 
JWB.msg('load-settings') + ' '+
 
'<select id="loadSettings">'+
 
'<option value="default" selected>default</option>'+
 
'<option value="_blank">'+JWB.msg('blank-setup')+'</option>'+
 
'</select>'+
 
'</label>'+
 
'<span class="divisor"></span>'+
 
'<button id="deleteSetup" title="'+JWB.msg('tip-delete-setup')+'">'+JWB.msg('delete-setup')+'</button>'+
 
'<hr>'+
 
'<button id="saveToWiki">'+JWB.msg('save-setup')+'</button>'+
 
'<span class="divisor"></span>'+
 
'<button id="download">'+JWB.msg('download-setup')+'</button>'+
 
'<hr>'+
 
'<label class="button" id="importLabel" title="'+JWB.msg('tip-import-setup')+'">'+
 
'<input type="file" id="import" accept=".json">'+
 
JWB.msg('import-setup')+
 
'</label>'+
 
'<span class="divisor"></span>'+
 
'<button id="updateSetups" title="'+JWB.msg('tip-update-setup', JWB.settingspage)+'">'+JWB.msg('update-setup')+'</button>'+
 
'<div id="downloads">'+
 
'<a download="JWB-settings.json" target="_blank" id="download-anchor"></a>'+
 
'<iframe id="download-iframe"></iframe>'+
 
'</div>'+
 
'</fieldset>'+
 
'<fieldset id="google-analytics">'+
 
'<legend>'+JWB.msg('label-analytics')+'</legend>'+
 
'<label title="'+JWB.msg('tip-analytics')+'">'+
 
'<input id="allow-analytics" type="checkbox" checked> '+
 
JWB.msg('allow-analytics')+
 
'</label>'+
 
'<span class="divisor">'+
 
'<a target="_blank" href="https://en.wikipedia.org/wiki/User:Joeytje50/JWB#Analytics" title="'+JWB.msg('tip-analytics-link')+'"><i>'+
 
JWB.msg('analytics-link')+
 
'</i></a>'+
 
'</fieldset>'
 
);
 
$('.JWBtabc[data-tab="2"]').html(
 
'<label class="minorEdit"><input type="checkbox" id="minorEdit" checked> '+JWB.msg('minor-edit')+'</label>'+
 
'<label class="editSummary viaJWB">'+JWB.msg('edit-summary')+' <input class="fullwidth" type="text" id="summary" maxlength="250"></label>'+
 
' <input type="checkbox" id="viaJWB" checked title="'+JWB.msg('tip-via-JWB')+'">'+
 
'<select id="watchPage">'+
 
'<option value="watch">'+JWB.msg('watch-watch')+'</option>'+
 
'<option value="unwatch">'+JWB.msg('watch-unwatch')+'</option>'+
 
'<option value="nochange" selected>'+JWB.msg('watch-nochange')+'</option>'+
 
'<option value="preferences">'+JWB.msg('watch-preferences')+'</option>'+
 
'</select>'+
 
'<span class="divisor"></span>'+
 
'<button id="watchNow" disabled accesskey="w" title="['+JWB.tooltip+'w]">'+
 
JWB.msg('watch-add')+
 
'</button>'+
 
'<br>'+
 
(JWB.bot?
 
'<label><input type="checkbox" id="autosave"> '+JWB.msg('auto-save')+'</label>'+
 
'<label title="'+JWB.msg('tip-save-interval')+'" class="divisor">'+
 
JWB.msg('save-interval', '<input type="number" min="0" value="0" style="width:50px" id="throttle" disabled>')+
 
'</label>'+
 
'<br>'
 
:'')+
 
'<span id="startstop">'+
 
'<button id="startbutton" accesskey="a" title="['+JWB.tooltip+'a]">'+JWB.msg('editbutton-start')+'</button>'+
 
'<br>'+
 
'<button id="stopbutton" disabled accesskey="q" title="['+JWB.tooltip+'q]">'+JWB.msg('editbutton-stop')+'</button> '+
 
'</span>'+
 
'<button class="editbutton" id="skipButton" disabled accesskey="n" title="['+JWB.tooltip+'n]">'+JWB.msg('editbutton-skip')+'</button>'+
 
'<button class="editbutton" id="submitButton" disabled accesskey="s" title="['+JWB.tooltip+'s]">'+JWB.msg('editbutton-save')+'</button>'+
 
'<br>'+
 
'<button class="editbutton" id="previewButton" disabled accesskey="p" title="['+JWB.tooltip+'p]">'+JWB.msg('editbutton-preview')+'</button>'+
 
'<button class="editbutton" id="diffButton" disabled accesskey="d" title="['+JWB.tooltip+'d]">'+JWB.msg('editbutton-diff')+'</button>'+
 
'<button id="replacesButton">'+JWB.msg('button-open-popup')+'</button>'+
 
findreplace+
 
'<hr>'+
 
'<label><input type="checkbox" id="enableRETF">'+
 
JWB.msg('label-enable-RETF',
 
'<a href="/wiki/Project:AutoWikiBrowser/Typos" target="_blank">'+
 
JWB.msg('label-RETF')+
 
'</a>')+
 
'</label>'+
 
' <img src="https://upload.wikimedia.org/wikipedia/commons/thumb/2/2a/Gnome-view-refresh.svg/20px-Gnome-view-refresh.svg.png"'+
 
'id="refreshRETF" title="'+JWB.msg('tip-refresh-RETF')+'">'
 
);
 
$('.JWBtabc[data-tab="3"]').html(
 
'<fieldset>'+
 
'<legend>'+JWB.msg('label-redirects')+'</legend>'+
 
'<label title="'+JWB.msg('tip-redirects-follow')+'">'+
 
'<input type="radio" class="redirects" value="follow" name="redir" id="redir-follow"> '+JWB.msg('redirects-follow')+' '+
 
'</label>'+
 
'<label title="'+JWB.msg('tip-redirects-skip')+'">'+
 
'<input type="radio" class="redirects" value="skip" name="redir" id="redir-skip"> '+JWB.msg('redirects-skip')+' '+
 
'</label>'+
 
'<label title="'+JWB.msg('tip-redirects-edit')+'">'+
 
'<input type="radio" class="redirects" value="edit" name="redir" id="redir-edit" checked> '+JWB.msg('redirects-edit')+''+
 
'</label>'+
 
'</fieldset>'+
 
'<fieldset>'+
 
'<legend>'+JWB.msg('label-skip-when')+'</legend>'+
 
'<label><input type="checkbox" id="skipNoChange"> '+JWB.msg('skip-no-change')+'</label>'+
 
'<br>'+
 
'<label><input type="radio" id="exists-yes" name="exists" value="yes"> '+JWB.msg('skip-exists-yes')+'</label>'+
 
'<label><input type="radio" id="exists-no" name="exists" value="no" checked> '+JWB.msg('skip-exists-no')+'</label>'+
 
'<label><input type="radio" id="exists-neither" name="exists" value="neither">'+JWB.msg('skip-exists-neither')+'</label>'+
 
'<br>'+
 
(JWB.sysop?'<label><input type="checkbox" id="skipAfterAction" checked> '+JWB.msg('skip-after-action')+'</label>':'')+
 
'</fieldset>'+
 
'<label>'+JWB.msg('skip-contains')+' <input class="fullwidth" type="text" id="skipContains"></label>'+
 
'<label>'+JWB.msg('skip-not-contains')+' <input class="fullwidth" type="text" id="skipNotContains"></label>'+
 
'<div class="regexswitch">'+
 
'<label><input type="checkbox" id="containRegex"> '+JWB.msg('label-useregex')+'</label>'+
 
'<a class="re101" href="http://regex101.com/#javascript" target="_blank">?</a>'+
 
'<label class="divisor" title="'+JWB.msg('tip-regex-flags')+'" style="display:none;">'+
 
JWB.msg('label-regex-flags')+' <input type="text" id="containFlags"/>'+
 
'</label>'+
 
'</div>'
 
);
 
if (JWB.sysop) $('.JWBtabc[data-tab="4"]').html(
 
'<fieldset>'+
 
'<legend>'+JWB.msg('move-header')+'</legend>'+
 
'<label><input type="checkbox" id="suppressRedir"> '+JWB.msg('move-redir-suppress')+'</label>'+
 
'<br>'+
 
JWB.msg('move-also')+' '+
 
'<label><input type="checkbox" id="movetalk"> '+JWB.msg('move-talk-page')+'</label> '+
 
'<label><input type="checkbox" id="movesubpage"> '+JWB.msg('move-subpage')+'</label>'+
 
'<br>'+
 
'<label>'+JWB.msg('move-new-name')+' <input type="text" id="moveTo"></label>'+
 
'</fieldset>'+
 
'<fieldset>'+
 
'<legend>'+JWB.msg('protect-header')+'</legend>'+
 
JWB.msg('protect-edit')+
 
'<select id="editProt">'+
 
'<option value="all" selected>'+JWB.msg('protect-none')+'</option>'+
 
'<option value="autoconfirmed">'+JWB.msg('protect-autoconf')+'</option>'+
 
'<option value="sysop">'+JWB.msg('protect-sysop')+'</option>'+
 
'</select> '+
 
'<br>'+
 
JWB.msg('protect-move')+
 
'<select id="moveProt">'+
 
'<option value="all" selected>'+JWB.msg('protect-none')+'</option>'+
 
'<option value="autoconfirmed">'+JWB.msg('protect-autoconf')+'</option>'+
 
'<option value="sysop">'+JWB.msg('protect-sysop')+'</option>'+
 
'</select> '+
 
'<br>'+
 
'<label>'+JWB.msg('protect-expiry')+' <input type="text" id="protectExpiry"/></label>'+
 
'</fieldset>'+
 
'<button id="movePage" disabled accesskey="m" title="['+JWB.tooltip+'m]">'+JWB.msg('editbutton-move')+'</button> '+
 
'<button id="deletePage" disabled accesskey="x" title="['+JWB.tooltip+'x]">'+JWB.msg('editbutton-delete')+'</button> '+
 
'<button id="protectPage" disabled accesskey="z" title="['+JWB.tooltip+'z]">'+JWB.msg('editbutton-protect')+'</button> '+
 
'<button id="skipPage" disabled title="['+JWB.tooltip+'n]">'+JWB.msg('editbutton-skip')+'</button>'
 
);
 
$('.JWBtabc[data-tab="5"]').html('<table id="actionlog"><tbody></tbody></table>');
 
$('#pagelistPopup form').html(
 
'<div id="ns-filter" title="'+JWB.msg('tip-ns-select')+'">' + JWB.msg('label-ns-select') + NSList + '</div>'+
 
'<fieldset>'+
 
'<legend><label><input type="checkbox" id="categorymembers" name="categorymembers" value="cm"> '+JWB.msg('legend-cm')+'</label></legend>'+
 
'<label title="Namespace prefix not required.">'+JWB.msg('label-cm')+' <input type="text" name="cmtitle" id="cmtitle"></label>'+
 
'<div>'+JWB.msg('cm-include')+' '+
 
'<label><input type="checkbox" id="cmtype-page" name="cmtype" value="page" checked> '+JWB.msg('cm-include-pages')+'</label>'+
 
'<label><input type="checkbox" id="cmtype-subcg" name="cmtype" value="subcat" checked> '+JWB.msg('cm-include-subcgs')+'</label>'+
 
'<label><input type="checkbox" id="cmtype-file" name="cmtype" value="file" checked> '+JWB.msg('cm-include-files')+'</label>'+
 
'</div>'+
 
'</fieldset>'+
 
'<fieldset>'+
 
'<legend><label><input type="checkbox" name="linksto" id="linksto"> '+JWB.msg('legend-linksto')+'</label></legend>'+
 
'<label>'+JWB.msg('label-linksto')+' <input type="text" name="title" id="linksto-title"></label>'+
 
'<div>'+JWB.msg('links-include')+' '+
 
'<label><input type="checkbox" id="backlinks" name="backlinks" value="bl" checked> '+JWB.msg('links-include-links')+'</label>'+
 
'<label><input type="checkbox" id="embeddedin" name="embeddedin" value="ei"> '+JWB.msg('links-include-templ')+'</label>'+
 
'<label><input type="checkbox" id="imageusage" name="imageusage" value="iu"> '+JWB.msg('links-include-files')+'</label>'+
 
'</div>'+
 
'<div>'+JWB.msg('links-redir')+' '+
 
'<label><input type="radio" id="rfilter-redir" name="filterredir" value="redirects"> '+JWB.msg('links-redir-redirs')+'</label>'+
 
'<label><input type="radio" id="rfilter-nonredir" name="filterredir" value="nonredirects"> '+JWB.msg('links-redir-noredirs')+'</label>'+
 
'<label><input type="radio" id="rfilter-all" name="filterredir" value="all" checked> '+JWB.msg('links-redir-all')+'</label>'+
 
'</div>'+
 
'<label title="'+JWB.msg('tip-link-redir')+'">'+
 
'<input type="checkbox" name="redirect" value="true" checked id="linksto-redir"> '+JWB.msg('label-link-redir')+
 
'</label>'+
 
'</fieldset>'+
 
'<fieldset>'+
 
'<legend><label><input type="checkbox" id="prefixsearch" name="prefixsearch" value="ps"> '+JWB.msg('legend-ps')+'</label></legend>'+
 
'<label>'+JWB.msg('label-ps')+' <input type="text" name="pssearch" id="pssearch"></label>'+
 
'</fieldset>'+
 
'<fieldset>'+
 
'<legend><label><input type="checkbox" id="watchlistraw" name="watchlistraw" value="wr"> '+JWB.msg('legend-wr')+'</label></legend>'+
 
JWB.msg('label-wr')+
 
'</fieldset>'+
 
'<fieldset>'+
 
'<legend><label><input type="checkbox" id="proplinks" name="links" value="pl"> '+JWB.msg('legend-pl')+'</label></legend>'+
 
'<label title="'+JWB.msg('tip-pl')+'">'+JWB.msg('label-pl')+' <input type="text" id="pltitles" name="titles"></label>'+
 
'</fieldset>'+
 
'<fieldset>'+
 
'<legend><label><input type="checkbox" name="search" value="sr">Search</label></legend>'+
 
'<label title="Query">Query<input type="text" name="srsearch"></label>'+
 
'</fieldset>'+
 
'<button type="submit">'+JWB.msg('pagelist-generate')+'</button>'
 
);
 
$('body').addClass('AutoWikiBrowser'); //allow easier custom styling of JWB.
 
 
/***** Setup *****/
 
JWB.setup.save('_blank'); //default setup
 
if (JWB.settings.hasOwnProperty('default')) {
 
JWB.setup.apply();
 
} else if (JWB.setup.hasOwnProperty('edittoken')) {
 
JWB.setup.save('default');
 
}
 
JWB.setup.extend({});
 
 
 
/***** Event handlers *****/
 
 
//Alert user when leaving the tab, to prevent accidental closing.
 
onbeforeunload = function() {
 
return "Closing this tab will cause you to lose all progress.";
 
};
 
ondragover = function(e) {
 
e.preventDefault();
 
};
 
 
$('.JWBtab').click(function() {
 
$('.active').removeClass('active');
 
$(this).addClass('active');
 
$('.JWBtabc[data-tab="'+$(this).attr('data-tab')+'"]').addClass('active');
 
});
 
 
function showRegexFlags() {
 
// >>this<< is the element that's triggered
 
$(this).parent().nextAll('label').toggle(this.checked);
 
}
 
$('body').on('change', '#useRegex, #containRegex, .useRegex', showRegexFlags);
 
 
$('#preparse-reset').click(function() {
 
$('#articleList').val($('#articleList').val().replace(/#PRE-PARSE-STOP/g,'').replace(/\n\n/g, '\n'));
 
});
 
$('#saveAs').click(function() {
 
JWB.setup.save();
 
});
 
$('#loadSettings').change(function() {
 
JWB.setup.apply(this.value);
 
});
 
$('#download').click(JWB.setup.download);
 
$('#saveToWiki').click(JWB.setup.submit);
 
$('#import').change(JWB.setup.import);
 
ondrop = JWB.setup.import;
 
$('#updateSetups').click(JWB.setup.load);
 
$('#deleteSetup').click(JWB.setup.delete);
 
 
if (window.RETF) $('#refreshRETF').click(RETF.load);
 
 
 
$('#replacesButton, #pagelistButton').click(function() {
 
var popup = this.id.slice(0, -6); //omits the 'Button' in the id by cutting off the last 6 characters
 
$('#'+popup+'Popup, #overlay').show();
 
});
 
$('#overlay').click(function() {
 
$('#replacesPopup, #pagelistPopup, #overlay').hide();
 
});
 
$('#moreReplaces').click(function() {
 
$('#replacesPopup').append(findreplace);
 
});
 
$('#replacesPopup').on('keydown', '.replaces:last', function(e) {
 
if (e.which === 9) $('#moreReplaces')[0].click();
 
});
 
 
$('#pagelistPopup legend input').change(function() {
 
//remove disabled attr when checked, add when not.
 
$(this).parents('fieldset').find('input').not('legend input').prop('disabled', !this.checked);
 
$(this).parents('fieldset').prop('disabled', !this.checked);
 
}).trigger('change');
 
 
$('#resultWindow').on('click', 'tr[data-line]:not(.lineheader) *', function(e) {
 
var line = +$(e.target).closest('tr[data-line]').data('line');
 
var index = $('#editBoxArea').val().split('\n').slice(0, line-1).join('\n').length;
 
$('#editBoxArea')[0].focus();
 
JWB.fn.setSelection($('#editBoxArea')[0], index+1);
 
JWB.fn.scrollSelection($('#editBoxArea')[0], index);
 
});
 
 
$('#removeDupes').click(function() {
 
$('#articleList').val(JWB.fn.uniques($('#articleList').val().split('\n')).join('\n'));
 
JWB.pageCount();
 
});
 
$('#sortArticles').click(function() {
 
$('#articleList').val($('#articleList').val().split('\n').sort().join('\n'));
 
JWB.pageCount();
 
});
 
 
$('#watchNow').click(JWB.api.watch);
 
$('#autosave').change(function() {
 
$('#throttle').prop('disabled', !this.checked);
 
});
 
 
$('#viaJWB').change(function() {
 
$('#summary').parent('label')
 
.toggleClass('viaJWB', this.checked)
 
.attr('maxlength', 250 - this.checked*JWB.summarySuffix.length); // Change the max size of the allowed summary according to having a suffix or not.
 
});
 
$('#startbutton').click(JWB.start);
 
$('#stopbutton').click(JWB.stop);
 
$('#submitButton').click(JWB.api.submit);
 
$('#previewButton').click(JWB.api.preview);
 
$('#diffButton').click(JWB.api.diff);
 
 
$('#skipButton, #skipPage').click(function() {
 
JWB.log('skip', JWB.list[0].split('|')[0]);
 
JWB.next();
 
});
 
 
if (JWB.sysop) {
 
$('#movePage').click(function() {
 
if ($('#moveTo').val().length === 0) {
 
return alert(JWB.msg('alert-no-move'));
 
}
 
JWB.api.move();
 
});
 
$('#protectPage').click(JWB.api.protect);
 
$('#deletePage').click(JWB.api.delete);
 
}
 
};
 
 
 
//Disable JWB altogether when it's loaded on a page other than Project:AutoWikiBrowser/Script. This script shouldn't be loaded on any other page in the first place.
 
if (JWB.allowed === false) JWB = false;
 

גרסה אחרונה מ־23:12, 9 במרץ 2026

mw.loader.load( '//en.wikipedia.org/w/index.php?title=User:Joeytje50/JWB.js/load.js&action=raw&ctype=text/javascript' ); // Backlink: [[en:User:Joeytje50/JWB.js/load.js]]