משתמש:שלמה/common.js
גרסה מ־01:47, 3 ביוני 2024 מאת שלמה (שיחה | תרומות) (יצירת דף עם התוכן "// Copied from User:Joeytje50/JWB.js /**<nowiki> * Install this script by pasting the following in your personal JavaScript file: mw.loader.load('//en.wiki...")
הערה: לאחר השמירה, ייתכן שיהיה צורך לנקות את זיכרון המטמון (cache) של הדפדפן כדי להבחין בשינויים.
- פיירפוקס / ספארי: להחזיק את המקש Shift בעת לחיצה על טעינה מחדש (Reload), או ללחוץ על צירוף המקשים Ctrl-F5 או Ctrl-R (במחשב מק: ⌘-R).
- גוגל כרום: ללחוץ על צירוף המקשים Ctrl-Shift-R (במחשב מק: ⌘-Shift-R).
- אינטרנט אקספלורר: להחזיק את המקש Ctrl בעת לחיצה על רענן (Refresh), או ללחוץ על צירוף המקשים Ctrl-F5.
- אופרה: לפתוח תפריט ← הגדרות (במחשב מק: Opera ← העדפות) ואז ללחוץ על פרטיות ואבטחה ← מחק היסטוריית גלישה ← Cached images and files.
// 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(/&/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>; '+ JWB.msg('stat-save')+' <span id="pagesSaved">0</span>; '+ JWB.msg('stat-null')+' <span id="nullEdits">0</span>; '+ JWB.msg('stat-skip')+' <span id="pagesSkipped">0</span>; '+ JWB.msg('stat-other')+' <span id="otherActions">0</span>; '+ '</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;