MediaWiki:Gadget-watchlist.js: различия между версиями

Материал из Википедии — свободной энциклопедии
Перейти к навигации Перейти к поиску
Содержимое удалено Содержимое добавлено
м «[href*=&days=7]» без кавычек вызывает Sizzle «syntax error» в MediaWiki 1.18 (already deployed in enwiki)
Используем контекст из аргумента хука "wikipage.content" для поиска списка изменений
 
(не показаны 64 промежуточные версии 4 участников)
Строка 1: Строка 1:
function WLScript(){
(function () {


var firstRun = true;
var whenPageLoaded = +(new Date()) - 20000 //add 20 sec just in case
var namespace, $content, mainTab, hideInterfaceCSS
var alreadySorted, alreadyAddedUnwatch, isEnhanced, mm
var inProgress = null, timeoutID = null


if (mw.config.get('wgCanonicalSpecialPageName') === 'Watchlist') {
mm = {
// Polyfill
sortTip:'Сортировать страницы по пространствам',
if (!String.prototype.includes) {
sortDone:'Изменения уже отсортированы',
String.prototype.includes = function (search, start) {
unwatchTip:'Добавить (x) ссылки для вычёркивания страниц из списка наблюдения',
'use strict';
unwatchDone:'Для удаления страниц из списка наблюдения используйте появившиеся ссылки (x)',
if (typeof start !== 'number') {
onlynew:'Только новые',
start = 0;
onlynewTip:'Показать изменения с момента загрузки этой страницы',
}
expandAll:'Показать/спрятать все свёрнутые правки',
fullPage:'Спрятать/показать элементы интерфейса'
if (start + search.length > this.length) {
}
return false;
} else {
return this.indexOf(search, start) !== -1;
}
};
}
var SECONDS_IN_A_DAY = 1000 * 60 * 60 * 24;
var whenPageLoadedOrUpdated;
var hideInterfaceCSS, rvOffCSS;
var sortingDone, allStarsShown, hoverOnTitlesDone;
// ruwiki strings
var strings = {
watch: 'Следить',
unwatch: 'Не следить',
sortTip: 'Сортировать изменения по пространствам имён',
sortDone: 'Изменения уже отсортированы',
unwatchTip: 'Добавить/убрать звёздочки для вычёркивания страниц из списка наблюдения',
newOnly: 'Только новые',
newOnlyTip: 'Изменения с момента загрузки этой страницы',
watchlistTabNewOnlyTip: 'По клику откроются изменения с момента загрузки этой страницы',
all: 'Все',
allTip: 'Все изменения',
expandAll: 'Показать/спрятать все свёрнутые правки',
switchRevert: 'Спрятать/показать ссылки «откатить»',
fullPage: 'Спрятать/показать элементы интерфейса',
error: 'Ошибка',
unrecognizedReponse: 'ответ не распознан'
};
mw.hook('wikipage.content').add(function ($content) {
function main() {
/* FUNCTIONS */
function addLink(contents, tip, classes, href) {
return $('<a>')
.attr('href', href || 'javascript:')
.attr('title', tip)
.addClass(classes)
.append(contents)
.appendTo($linksIn);
}
function generateNewEntriesOnlyUrl() {
var uri = new mw.Uri();
uri.query.from = whenPageLoadedOrUpdated;
delete uri.query.days;
return uri.toString();
}
function generateAllEntriesUrl() {
var uri = new mw.Uri();
delete uri.query.from;
return uri.toString();
}
function newEntriesOnly(e) {
e.target.href = generateNewEntriesOnlyUrl();
}
function allEntries(e) {
e.target.href = generateAllEntriesUrl();
}
function showAllStars(e) {
if (e) {
e.preventDefault();
}
// cookie set = stars are on
if (!allStarsShown) {
$changeslist.find('.mw-title')
.each(function (i, link) {
updateStar(getRow($(this)));
});
if (e) {
var cookieDate = new Date($.now() + SECONDS_IN_A_DAY * 90).toGMTString();
document.cookie = 'wlunw=1; expires=' + cookieDate + '; path=/';
}
allStarsShown = true;
} else { // otherwise remove
$changeslist.find('.gadgetWatchlist-unwatchLink').remove();
document.cookie = 'wlunw=0; expires=' + (new Date()).toGMTString() + '; path=/';
if (!hoverOnTitlesDone) {
bindHoverOnTitles();
}
allStarsShown = false;
}
}
function sortWatchlist(e) {
e.preventDefault();
if (sortingDone) {
alert(strings.sortDone);
return;
}
$changeslist.find('h4').each(function () { // sort all days separately
var $container = $(this).next('div, ul');
var $rows = $container.children('li, table');
// create sorting keys
var key;
$rows.each(function (i) {
// use built-in class: either li.watchlist-5-<title> or
// table.mw-changeslist-ns100-<title> in enhanced recent changes
key = /(\d+)-(\S+)/.exec(this.className) || ['', 0, ' ']; // logs might not have this class
if (key[1] % 2) {
key[1]--; // sort talk page as if it was a base page
}
if (window.watchlistSortNamespaceOnly) {
key[2] = zzz(i); // keep timestamp order within each NS block
}
this.skey = zzz(key[1]) + ':' + key[2];
});
// sort array and then HTML
$rows.sort(function (a, b) {
return a.skey > b.skey ? 1 : (a.skey < b.skey ? -1 : 0);
});
for (i = 0; i < $rows.length; i++) {
$container.append($rows.eq(i));
}
});
$('.mw-rcfilters-ui-changesListWrapperWidget-previousChangesIndicator').remove();
sortingDone = true;
}
function expandMultipleEdits(e) {
e.preventDefault();
var $collapsibles = $('.mw-changeslist .mw-collapsible:not(.mw-changeslist-legend)');
// If at lease one branch is collapsed, we expand everything.
$collapsibles
[$collapsibles.hasClass('mw-collapsed') ? 'filter' : 'not']('.mw-collapsed')
.find('.mw-enhancedchanges-arrow')
.click();
}
function switchRevert(e) {
if (e) {
e.preventDefault();
}
if (!rvOffCSS) {
rvOffCSS = mw.util.addCSS('\
.mw-rollback-link {\
display: none;\
}\
');
} else {
rvOffCSS.disabled = !rvOffCSS.disabled;
}
if (rvOffCSS.disabled) {
document.cookie = 'wlrvoff=0; expires=' + (new Date()).toGMTString() + '; path=/';
} else if (e) {
var cookieDate = (new Date($.now() + SECONDS_IN_A_DAY * 90)).toGMTString();
document.cookie = 'wlrvoff=1; expires=' + cookieDate + '; path=/';
}
}
function bindHoverOnTitles() { // find all title links and assign hover event
if (hoverOnTitlesDone) return;
$changeslist.find('.mw-title')
.each(function () {
getRow($(this))
.find('.mw-title')
.children('a')
.hover(hoverOnTitle);
});
hoverOnTitlesDone = true;
}
function hoverOnTitle(e) { // on hover: add "unwatch" star after 1 second
var $link = $(this);
if (e.type === 'mouseenter') {
$link.data('unwatchTimeout', setTimeout(function () {
showStarOnHover($link);
}, 1000));
} else {
clearTimeout($link.data('unwatchTimeout'));
}
}
function showStarOnHover($link) {
var $row = getRow($link);
updateStar($row);
// attach mouseleave to remove the star
if ($row.data('leaveAssigned')) return;
$row.data('leaveAssigned', true);
$row.mouseleave(function (e) {
var $unwatchLink = $(this).find('.gadgetWatchlist-unwatchLink');
if ($unwatchLink.length &&
!($unwatchLink.hasClass('gadgetWatchlist-failure') ||
$unwatchLink.hasClass('gadgetWatchlist-waiting')
) &&
!allStarsShown
) {
$unwatchLink.remove();
}
});
}
function updateStar($row) {
var action = $row.hasClass('gadgetWatchlist-unwatchedRow') ? 'watch' : 'unwatch';
var $star = $row.find('.gadgetWatchlist-unwatchLink, .gadgetWatchlist-watchLink');
if (!$star.length) { // create
var $link = $row.find('.mw-title').children('a');
if (!$link.length) return;
$star = $('<a>')
.attr('href', $link.attr('href').replace(/\/wiki\//, '/ruwiki/w/index.php?title=') + '&action=' +
action)
.addClass('gadgetWatchlist-' + action + 'Link gadgetWatchlist-icon')
.click(changeWatchState)
.insertBefore($link);
} else { // update
$star
.attr('href', $star
.attr('href')
.replace(/&action=\w+/, '&action=' + action))
.removeClass('gadgetWatchlist-' + (action === 'unwatch' ? 'watch' : 'unwatch') + 'Link')
.addClass('gadgetWatchlist-' + action + 'Link');
}
$star.attr('title', mw.messages.get(action) || strings[action]);
}
function getRow($el) {
return $el.closest(isEnhanced ? 'tr' : 'li');
}
function changeWatchState(e) {
var $star = $(this), errorMsg = '';
var action;
var unwatch;
if ($star.attr('href').includes('&action=unwatch')) {
unwatch = '';
action = 'unwatch';
} else {
action = 'watch';
}
var titles = getLinkTitle($star);
$star
.removeClass('gadgetWatchlist-failure')
.addClass('gadgetWatchlist-waiting');
new mw.Api().postWithToken('watch', {
action: 'watch',
titles: titles,
unwatch: unwatch,
formatversion: 2,
}).then(
function (resp) {
if (!resp.watch || !resp.watch[0]) {
errorMsg = 'empty response';
} else if (typeof resp.watch[0].unwatched === 'boolean') {
changeWatchStateSuccess(titles, true);
} else if (typeof resp.watch[0].watched === 'boolean') {
changeWatchStateSuccess(titles, false);
} else {
errorMsg = strings.unrecognizedReponse;
}
},
function (code, data) {
var errorMsg = (
// HTTP error
data.textStatus ||


// API error
namespace = document.getElementById('namespace')
(data.error && data.error.info)
if (!namespace) return
);

if (errorMsg) {
$content = $('#bodyContent')
errorMsg += ' (' + code + ')';
if (!$content.length){
} else {
$content = $('#mw_content, #article').eq(0)
errorMsg = code;
if (!$content.length) $content = $(document.body) // covered all the skin but who knows...
}
}
).always(function () { // update unwatch link
$star.removeClass('gadgetWatchlist-waiting');
if (errorMsg) {
$star
.attr('title', strings.error + ': ' + errorMsg)
.addClass('gadgetWatchlist-failure gadgetWatchlist-' + action + 'Link');
}
});
e.preventDefault();
}
function changeWatchStateSuccess(name, isUnwatched) {
// find full name of associated talk page (or vice versa)
var ns = getTitleNamespace(name);
var name2 = name;
if (ns > 0) {
name2 = name2.replace(/^.+?:/, ''); // remove old prefix
}
ns = ns % 2 ? ns - 1 : ns + 1; // switch to associated namespace
if (ns > 0) {
name2 = mw.config.get('wgFormattedNamespaces')[ns] + ':' + name2; // add new prefix
}
// mark all rows that are either name or name2
$changeslist.find('.mw-changeslist-title').each(function () {
var title = getLinkTitle($(this));
if (title !== name && title !== name2) return;
var $row = getRow($(this));
$row.toggleClass('gadgetWatchlist-unwatchedRow', isUnwatched);
if (!isUnwatched && !allStarsShown) {
$row.find('.gadgetWatchlist-watchLink').remove();
} else {
updateStar($row);
}
});
}
function hideInterface(e) {
if (e) {
e.preventDefault();
}
var hideInterfaceCSSCode = '\
#firstHeading,\
div#siteNotice,\
#contentSub,\
.mw-wlheader-showupdated,\
fieldset#mw-watchlist-options,\
div.mw-rc-label-legend,\
#mw-fr-watchlist-pending-notice,\
.mw-indicators,\
.mw-rcfilters-ui-filterTagMultiselectWidget,\
.mw-rcfilters-ui-watchlistTopSectionWidget-savedLinksTable,\
.mw-rcfilters-ui-watchlistTopSectionWidget-editWatchlistButton,\
.mw-rcfilters-ui-watchlistTopSectionWidget-separator {\
display: none;\
}\
\
.client-js .mw-special-Watchlist .rcfilters-head.rcfilters-head {\
min-height: auto;\
}\
';
if (!hideInterfaceCSS) {
// If the new filters are on, wait until the interface is initialized.
if (!mw.user.options.get('wlenhancedfilters-disable')) {
hideInterfaceCSS = mw.util.addCSS(hideInterfaceCSSCode);
$('.mw-rcfilters-ui-markSeenButtonWidget')
.wrap('<div>')
.parent()
.addClass('mw-rcfilters-ui-markSeenButtonWidget-container')
.appendTo('.mw-rcfilters-ui-watchlistTopSectionWidget-watchlistDetails');
} else {
hideInterfaceCSS = mw.util.addCSS(hideInterfaceCSSCode);
}
} else {
hideInterfaceCSS.disabled = !hideInterfaceCSS.disabled;
if (!mw.user.options.get('wlenhancedfilters-disable')) {
$('.mw-rcfilters-ui-markSeenButtonWidget-container')
.appendTo(hideInterfaceCSS.disabled ?
'.mw-rcfilters-ui-watchlistTopSectionWidget-savedLinksTable .mw-rcfilters-ui-cell:first-child' :
'.mw-rcfilters-ui-watchlistTopSectionWidget-watchlistDetails');
}
}
if (e) {
if (!hideInterfaceCSS.disabled) {
var cookieDate = new Date($.now() + SECONDS_IN_A_DAY * 90).toGMTString();
document.cookie = 'wlmax=1; expires=' + cookieDate + '; path=/';
} else {
document.cookie = 'wlmax=0; expires=' + (new Date()).toGMTString() + '; path=/';
}
}
}
function getTitleNamespace(title) { // returns namespace number
var prefix = /^(.+?):/.exec(title);
if (!prefix) {
return 0; // no prefix means article
}
return mw.config.get('wgNamespaceIds')[ prefix[1].toLowerCase().replace(/ /g, '_') ] || 0;
}
function getLinkTitle($link) { // gets title for unwatch/watch links & common page links
// Titles can be absent for .mw-changeslist-title elements because of Popups extension.
var title = $link.filter('.mw-changeslist-title').attr('title');
if (!title) {
title = mw.util.getParamValue('title', $link.attr('href'));
title = title && title.replace(/_/g, ' ');
}
if (!title) {
title = $link.filter('.mw-changeslist-title').text();
}
return title;
}
function zzz(s) { // 5 -> 005
s = s.toString();
if (s.length === 1) {
return '00' + s;
} else if (s.length === 2) {
return '0' + s;
} else {
return s;
}
}
function zeroPad(n, p) {
return ('0000' + n).slice(-p);
}
function generateTimestamp(date) {
return (
zeroPad(date.getUTCFullYear(), 4) +
zeroPad(date.getUTCMonth() + 1, 2) +
zeroPad(date.getUTCDate(), 2) +
zeroPad(date.getUTCHours(), 2) +
zeroPad(date.getUTCMinutes(), 2) +
zeroPad(date.getUTCSeconds(), 2)
);
}
/* MAIN BLOCK */
// add 5–20 seconds just in case (for example, the scripting could take too much)
whenPageLoadedOrUpdated = generateTimestamp(new Date(new Date().getTime() - (firstRun ? 20000 : 5000)));
sortingDone = false;
allStarsShown = false;
hoverOnTitlesDone = false;
// UNWATCH LINKS
// on every line
if (document.cookie.includes('wlunw=1')) {
showAllStars();
} else {
bindHoverOnTitles(); // mouseover on title
}
if (firstRun) {
if (document.cookie.includes('wlrvoff=1')) {
switchRevert();
}
// find insertion point for links
$linksIn = $('<div>')
.addClass('mw-rcfilters-ui-cell gadgetWatchlist-linksContainer')
.insertBefore($(
!mw.user.options.get('wlenhancedfilters-disable') ?
'.mw-rcfilters-ui-watchlistTopSectionWidget-savedLinks' :
'.wlinfo'
));
// "show all stars" link
addLink('', strings.unwatchTip, 'gadgetWatchlist-icon gadgetWatchlist-icon-unwatched')
.click(showAllStars);
// FUNCTION LINKS
// "sort" link
addLink('↑↓', strings.sortTip).click(sortWatchlist);
// "expand all" link
// Auto-update could be used, so even if there is no $('.mw-enhancedchanges-arrow') elements, we keep
// the button.
if (isEnhanced) {
addLink('±', strings.expandAll).click(expandMultipleEdits);
}
// "switch revert" link
if (mw.config.get('wgUserGroups').indexOf('rollbacker') !== -1 ||
mw.config.get('wgUserGroups').indexOf('sysop') !== -1
) {
addLink('⎌', strings.switchRevert).click(switchRevert);
}
// "new only" link
addLink(strings.newOnly, strings.newOnlyTip, undefined, generateNewEntriesOnlyUrl())
.mousedown(newEntriesOnly);
if (mw.util.getParamValue('from') !== null) {
// "all" link
addLink(strings.all, strings.allTip, undefined, generateAllEntriesUrl()).mousedown(allEntries);
}
// TABS
if (!window.wlNoTabs) {
var $mainTab = $('#ca-nstab-special').first(); // "Special" tab
// change main tab into "watchlist Δ"
var watchlistTitle = $.trim($('#firstHeading').text());
$mainTab
.find('a') // replace "Special" with "Watchlist" as tab text
.text(watchlistTitle + ' △') // Δ is good but monobook makes is lowercase
.attr('title', strings.watchlistTabNewOnlyTip)
.attr('href', generateNewEntriesOnlyUrl())
.on('mousedown keydown', newEntriesOnly);
// add "hideInterface" tab
$mainTab
.clone(true)
.attr('id', '')
.attr('href', '')
.removeClass('selected')
.click(hideInterface)
.appendTo($mainTab.parent())
.find('a')
.text('↸')
.attr('title', strings.fullPage)
.attr('accesskey', '');
}
// OTHER TASKS
if (document.cookie.includes('wlmax=1')) {
hideInterface();
}
mw.util.addCSS('\
.mw-special-Watchlist .mw-changeslist .mw-rollback-link.mw-rollback-link {\
visibility: visible;\
}\
');
firstRun = false;
}
return;
}
// Occurs in watchlist when mediawiki.rcfilters.filters.ui module for some reason fires
// wikipage.content for the second time with an element that is not in the DOM,
// fieldset#mw-watchlist-options (in mw.rcfilters.ui.FormWrapperWidget.prototype.onChangesModelUpdate
// function).
if (!$content.parent().length) return;
var $changeslist = $content.find('.mw-changeslist');
// Recent changes type in preferences. Don't confuse enhanced recent changes with enhanced filters.
// mw.user.options.get('usenewrc') shouldn't be used here: RC mode could be set from URL.
var isEnhanced = !$changeslist.find('ul.special').length;
if (mw.util.getParamValue('from') !== null) {
$('#mw-watchlist-form input[name="from"]').remove();
}
if (!mw.user.options.get('wlenhancedfilters-disable')) {
mw.hook('structuredChangeFilters.ui.initialized').add(main);
} else {
if (!navigator.userAgent.toLowerCase().includes('firefox')) {
main();
} else {
// Cure for Firefox
setTimeout(main, 0);
}
}
});
}
}


})();
//enhanced RC: make (x) appear by clicking on timestamp
isEnhanced = $content.find('ul.special').length == 0
if (isEnhanced) $content.click(toggleXLink)

//find insertion points for links: after "days all"
var linksAt = $('#mw-watchlist-options').find('a[href*="&days=7"]').next().next()
if (!linksAt.length) linksAt = $('#mw-watchlist-options').find('hr').last()
//find "Special" tab
mainTab = $('#ca-special, #ca-nstab-special').eq(0)

//"only new" link and tab
addLnk(mm.onlynew, mm.onlynewTip).mousedown(onlyNewEntries).attr('id', 'listSince')
addTab(mm.onlynew, mm.onlynewTip).mousedown(onlyNewEntries)

//"unwatch" link(s)
if (window.unwatchLinksOnLoad) addXLinks()
else addLnk('x' , mm.unwatchTip).click(addXLinks)

//"sort" link
addLnk('↑↓', mm.sortTip).click(sortWatchlist)

//"expand all" link
if ($('#mw-rc-openarrow-0').length) addLnk('±', mm.expandAll).click(expandMultipleEdits)

//"hideInterface" tab
addTab('↸',mm.fullPage).click(hideInterface).attr('href','#')
if (document.cookie.indexOf('wlmax=1') != -1) hideInterface()

function addLnk(txt, tip){
linksAt.before(' | ')
return $('<a href=# title="'+tip+'" style="font-style:italic">'+txt+'</a>').insertBefore(linksAt)
}
function addTab(txt, tip){
var tab = mainTab.clone(true).removeClass('selected').attr('id','')
tab.find('a').text(txt).attr('title', tip).attr('accesskey','')
return tab.appendTo(mainTab.parent())
}

return



function onlyNewEntries(e) {
var url = window.location.href.split('#')[0]
var days = ( +(new Date()) - whenPageLoaded)/(1000 * 3600 * 24)
e.target.href = /[?&]days=/.test(url)
? url.replace(/([?&]days=)[^&]*/, '$1'+days)
: url + (url.indexOf('?') < 0 ? '?':'&') + 'days=' + days
return true
}



function sortWatchlist(e){
e.preventDefault()
if (alreadySorted) return alert(mm.sortDone)
$content.find('h4').each(function(i, H4){ //sort all days separately
sortDay($(H4).next('div, ul'))
})
alreadySorted = true
}

function sortDay(dayDiv){
var i, pgname, rowElem, hiddenDiv, rows = dayDiv.find('a[href*="&action=history"]')
for (i = 0; i < rows.length; i++){
pgname = getTitleFromURL(rows[i].href)
ns = getNSFromTitle(pgname)
if (ns>0) pgname = pgname.substring(getNamespace(ns).length+1) //title w/o prefix
if (ns%2) ns-- //sort talk page as if it was a base page
rows[i].sortkey = zzz(ns) + ':' + pgname //assign custom tag attribute: namespace+title
}
//sort rows array
rows.sort(function(a,b){
if (a.sortkey > b.sortkey) return 1
else if (a.sortkey < b.sortkey) return -1
else return 0
})
//sort rows in HTML, by moving all to the bottom
if (isEnhanced)
for (i=0; i<rows.length; i++){
rowElem = rows.eq(i).closest('table')
hiddenDiv = rowElem.next('div')
dayDiv.append(rowElem, hiddenDiv)
}
else
for (i=0; i<rows.length; i++){
rowElem = rows.eq(i).closest('li')
dayDiv.append(rowElem)
}
}



function expandMultipleEdits(e){
e.preventDefault()
var i = 0, sp, state = $('#mw-rc-openarrow-0')[0].style.display
while (sp=document.getElementById('mw-rc-openarrow-'+(i++).toString()))
if (sp.style.display == state) $(sp.firstChild).click()
}




function addXLinks(e){
if (e) e.preventDefault()
if (alreadyAddedUnwatch) return alert(mm.unwatchDone)
$content.find('a[href*="&action=history"]').each(function(i, lnk){ addXLink(lnk) })
alreadyAddedUnwatch = true
}

function addXLink(lnk){
var x = $('<a class=aj-unwatch style="font-size:smaller" title="'+mw.msg('unwatch')+'" />')
.attr('href', lnk.href.replace(/&curid=\d+/,'').replace(/action=history/,'action=unwatch'))
.click(ajaxUnwatch)
lnk = $(lnk)
if (isEnhanced) lnk.parent().prepend( x.append('(x)'), ' ' )
else lnk.after( ' | ', x.append('x') )
}

function toggleXLink(e){ //add (x) when clicking on timestamp
if ( ! $(e.target).filter('td.mw-enhanced-rc').length ) return
var tbl = $(e.target).parents('table.mw-enhanced-rc')
var x = tbl.find('a.aj-unwatch')
if (x.length) x.remove()
else addXLink(tbl.find('a[href*="&action=history"]')[0])
}


function ajaxUnwatch(e) {
if (inProgress) return false
inProgress = getTitleFromURL(this.href)
timeoutID = window.setTimeout( function() {inProgress = null}, 10000 )
//call server
var req = { action:'watch', format:'json', title:inProgress }
if (/&action=unwatch/.test(this.href)) req.unwatch = ''
$.getJSON( '/ruwiki/w/api.php', req, showUnwatch)
return false
}

function showUnwatch (response) {
if (timeoutID) window.clearTimeout(timeoutID)
response = response.watch
var name = inProgress, state, pg
inProgress = null
if (response.watched !== undefined) state = false
else if (response.unwatched !== undefined) state = true
else return //unrecognized response
//find the full name of "other page"
var ns = getNSFromTitle(name)
var name2 = name
if (ns > 0) name2 = name2.substring(getNamespace(ns).length+1) //remove old prefix
if (ns % 2) ns--; else ns++ //switch to "other" namespace
if (ns > 0) name2 = getNamespace(ns) + ':' + name2 //add new prefix
//now mark all rows that are either name or name2
$content.find('a.aj-unwatch').each(function(i, lnk){
pg = getTitleFromURL(lnk.href)
if (pg != name && pg != name2) return
lnk.title = mw.msg(state?'watch':'unwatch')
$(lnk).parent().find('a').css('text-decoration', state ? 'line-through' : '')
lnk.href = lnk.href.replace(/&action=\w+/, '&action='+ (state?'watch':'unwatch'))
})
}




function hideInterface(e){

if (e) e.preventDefault()

if (!hideInterfaceCSS) hideInterfaceCSS = mw.util.addCSS('\
div#siteNotice, h1#firstHeading, #siteSub, #contentSub, fieldset#mw-watchlist-options,\
div.mw-rc-label-legend, #mw-fr-watchlist-pending-notice {display:none}')
else hideInterfaceCSS.disabled = !hideInterfaceCSS.disabled

document.cookie = 'wlmax=' + (!hideInterfaceCSS.disabled ? '1' : '0;expires=' + (new Date()).toGMTString() + ';;')

var a = mainTab.find('a') //replace "Special" tab text with "Watchlist"
if (hideInterfaceCSS.disabled){ //restore
a.text( a.attr('oldtext') )
}else{//set to "watchlist"
a.attr('oldtext', a.text())
a.text($('h1#firstHeading').text())
}

}




//common functions; need 'namespace' select element

function getNSFromTitle(title){ //returns namespace number
var i = title.indexOf(':')
if (i == -1) return 0
var prefix = title.substring(0,i+1) //including ':'
for (i=2; i < namespace.options.length; i++)
if (namespace.options[i].text+':' == prefix)
return i-1
return 0 // ':' was just a part of the title
}

function getNamespace(ns){ //returns namespace name
if (ns==0) return ''
else return namespace.options[ns+1].text
}

function getTitleFromURL (url){ //gets 'title=' part from a link
var ma = url.match(/(&|\?)title=([^&]+)/)
if (ma) return decodeURIComponent(ma[2]).replace(/_/g,' ')
else return ''
}

function zzz(s){ // 5 -> 005
s = s.toString()
if (s.length==1) return '00'+s
else if (s.length==2) return '0'+s
else return s
}


}


if (wgCanonicalSpecialPageName == 'Watchlist' && wgAction == 'view') WLScript()

Текущая версия от 04:06, 15 августа 2021

(function () {

var firstRun = true;

if (mw.config.get('wgCanonicalSpecialPageName') === 'Watchlist') {
	// Polyfill
	if (!String.prototype.includes) {
		String.prototype.includes = function (search, start) {
			'use strict';
			if (typeof start !== 'number') {
				start = 0;
			}
			
			if (start + search.length > this.length) {
				return false;
			} else {
				return this.indexOf(search, start) !== -1;
			}
		};
	}
	
	var SECONDS_IN_A_DAY = 1000 * 60 * 60 * 24;
	
	var whenPageLoadedOrUpdated;
	var hideInterfaceCSS, rvOffCSS;
	var sortingDone, allStarsShown, hoverOnTitlesDone;
	
	// ruwiki strings
	var strings = {
		watch: 'Следить',
		unwatch: 'Не следить',
		sortTip: 'Сортировать изменения по пространствам имён',
		sortDone: 'Изменения уже отсортированы',
		unwatchTip: 'Добавить/убрать звёздочки для вычёркивания страниц из списка наблюдения',
		newOnly: 'Только новые',
		newOnlyTip: 'Изменения с момента загрузки этой страницы',
		watchlistTabNewOnlyTip: 'По клику откроются изменения с момента загрузки этой страницы',
		all: 'Все',
		allTip: 'Все изменения',
		expandAll: 'Показать/спрятать все свёрнутые правки',
		switchRevert: 'Спрятать/показать ссылки «откатить»',
		fullPage: 'Спрятать/показать элементы интерфейса',
		error: 'Ошибка',
		unrecognizedReponse: 'ответ не распознан'
	};
	
	mw.hook('wikipage.content').add(function ($content) {
		function main() {
			/* FUNCTIONS */
			
			function addLink(contents, tip, classes, href) {
				return $('<a>')
					.attr('href', href || 'javascript:')
					.attr('title', tip)
					.addClass(classes)
					.append(contents)
					.appendTo($linksIn);
			}
			
			function generateNewEntriesOnlyUrl() {
				var uri = new mw.Uri();
				uri.query.from = whenPageLoadedOrUpdated;
				delete uri.query.days;
				return uri.toString();
			}
			
			function generateAllEntriesUrl() {
				var uri = new mw.Uri();
				delete uri.query.from;
				return uri.toString();
			}
			
			function newEntriesOnly(e) {
				e.target.href = generateNewEntriesOnlyUrl();
			}
			
			function allEntries(e) {
				e.target.href = generateAllEntriesUrl();
			}
			
			function showAllStars(e) {
				if (e) {
					e.preventDefault();
				}
				// cookie set = stars are on
				if (!allStarsShown) {
					$changeslist.find('.mw-title')
						.each(function (i, link) {
							updateStar(getRow($(this)));
						});
					if (e) {
						var cookieDate = new Date($.now() + SECONDS_IN_A_DAY * 90).toGMTString();
						document.cookie = 'wlunw=1; expires=' + cookieDate + '; path=/';
					}
					allStarsShown = true;
				} else {  // otherwise remove
					$changeslist.find('.gadgetWatchlist-unwatchLink').remove();
					document.cookie = 'wlunw=0; expires=' + (new Date()).toGMTString() + '; path=/';
					if (!hoverOnTitlesDone) {
						bindHoverOnTitles();
					}
					allStarsShown = false;
				}
			}
			
			function sortWatchlist(e) {
				e.preventDefault();
				if (sortingDone) {
					alert(strings.sortDone);
					return;
				}
				$changeslist.find('h4').each(function () {  // sort all days separately
					var $container = $(this).next('div, ul');
					var $rows = $container.children('li, table');
					// create sorting keys
					var key;
					$rows.each(function (i) {
						// use built-in class: either li.watchlist-5-<title> or
						// table.mw-changeslist-ns100-<title> in enhanced recent changes
						key = /(\d+)-(\S+)/.exec(this.className) || ['', 0, ' '];  // logs might not have this class
						if (key[1] % 2) {
							key[1]--;  // sort talk page as if it was a base page
						}
						if (window.watchlistSortNamespaceOnly) {
							key[2] = zzz(i);  // keep timestamp order within each NS block
						}
						this.skey = zzz(key[1]) + ':' +  key[2];
					});
					// sort array and then HTML
					$rows.sort(function (a, b) {
						return a.skey > b.skey ? 1 : (a.skey < b.skey ? -1 : 0);
					});
					for (i = 0; i < $rows.length; i++) {
						$container.append($rows.eq(i));
					}
				});
				$('.mw-rcfilters-ui-changesListWrapperWidget-previousChangesIndicator').remove();
				sortingDone = true;
			}
			
			function expandMultipleEdits(e) {
				e.preventDefault();
				var $collapsibles = $('.mw-changeslist .mw-collapsible:not(.mw-changeslist-legend)');
				// If at lease one branch is collapsed, we expand everything.
				$collapsibles
					[$collapsibles.hasClass('mw-collapsed') ? 'filter' : 'not']('.mw-collapsed')
					.find('.mw-enhancedchanges-arrow')
					.click();
			}
			
			function switchRevert(e) {
				if (e) {
					e.preventDefault();
				}
				if (!rvOffCSS) {
					rvOffCSS = mw.util.addCSS('\
						.mw-rollback-link {\
							display: none;\
						}\
					');
				} else {
					rvOffCSS.disabled = !rvOffCSS.disabled;
				}
				if (rvOffCSS.disabled) {
					document.cookie = 'wlrvoff=0; expires=' + (new Date()).toGMTString() + '; path=/';
				} else if (e) {
					var cookieDate = (new Date($.now() + SECONDS_IN_A_DAY * 90)).toGMTString();
					document.cookie = 'wlrvoff=1; expires=' + cookieDate + '; path=/';
				}
			}
			
			function bindHoverOnTitles() {  // find all title links and assign hover event
				if (hoverOnTitlesDone) return;
				$changeslist.find('.mw-title')
					.each(function () {
						getRow($(this))
							.find('.mw-title')
							.children('a')
							.hover(hoverOnTitle);
					});
				hoverOnTitlesDone = true;
			}
			
			function hoverOnTitle(e) {  // on hover: add "unwatch" star after 1 second
				var $link = $(this);
				if (e.type === 'mouseenter') {
					$link.data('unwatchTimeout', setTimeout(function () {
						showStarOnHover($link);
					}, 1000));
				} else {
					clearTimeout($link.data('unwatchTimeout'));
				}
			}
			
			function showStarOnHover($link) {
				var $row = getRow($link);
				updateStar($row);
				// attach mouseleave to remove the star
				if ($row.data('leaveAssigned')) return;
				$row.data('leaveAssigned', true);
				$row.mouseleave(function (e) {
					var $unwatchLink = $(this).find('.gadgetWatchlist-unwatchLink');
					if ($unwatchLink.length &&
						!($unwatchLink.hasClass('gadgetWatchlist-failure') ||
							$unwatchLink.hasClass('gadgetWatchlist-waiting')
						) &&
						!allStarsShown
					) {
						$unwatchLink.remove();
					}
				});
			}
			
			function updateStar($row) {
				var action = $row.hasClass('gadgetWatchlist-unwatchedRow') ? 'watch' : 'unwatch';
				var $star = $row.find('.gadgetWatchlist-unwatchLink, .gadgetWatchlist-watchLink');
				if (!$star.length) {  // create
					var $link = $row.find('.mw-title').children('a');
					if (!$link.length) return;
					$star = $('<a>')
						.attr('href', $link.attr('href').replace(/\/wiki\//, '/ruwiki/w/index.php?title=') + '&action=' +
							action)
						.addClass('gadgetWatchlist-' + action + 'Link gadgetWatchlist-icon')
						.click(changeWatchState)
						.insertBefore($link);
				} else {  // update
					$star
						.attr('href', $star
							.attr('href')
							.replace(/&action=\w+/, '&action=' + action))
						.removeClass('gadgetWatchlist-' + (action === 'unwatch' ? 'watch' : 'unwatch') + 'Link')
						.addClass('gadgetWatchlist-' + action + 'Link');
				}
				$star.attr('title', mw.messages.get(action) || strings[action]);
			}
			
			function getRow($el) {
				return $el.closest(isEnhanced ? 'tr' : 'li');
			}
			
			function changeWatchState(e) {
				var $star = $(this), errorMsg = '';
				var action;
				var unwatch;
				if ($star.attr('href').includes('&action=unwatch')) {
					unwatch = '';
					action = 'unwatch';
				} else {
					action = 'watch';
				}
				var titles = getLinkTitle($star);
				$star
					.removeClass('gadgetWatchlist-failure')
					.addClass('gadgetWatchlist-waiting');
				new mw.Api().postWithToken('watch', {
					action: 'watch',
					titles: titles,
					unwatch: unwatch,
					formatversion: 2,
				}).then(
					function (resp) {
						if (!resp.watch || !resp.watch[0]) {
							errorMsg = 'empty response';
						} else if (typeof resp.watch[0].unwatched === 'boolean') {
							changeWatchStateSuccess(titles, true);
						} else if (typeof resp.watch[0].watched === 'boolean') {
							changeWatchStateSuccess(titles, false);
						} else {
							errorMsg = strings.unrecognizedReponse;
						}
					},
					function (code, data) {
						var errorMsg = (
							// HTTP error
							data.textStatus ||

							// API error
							(data.error && data.error.info)
						);
						if (errorMsg) {
							errorMsg += ' (' + code + ')';
						} else {
							errorMsg = code;
						}
					}
				).always(function () {  // update unwatch link
					$star.removeClass('gadgetWatchlist-waiting');
					if (errorMsg) {
						$star
							.attr('title', strings.error + ': ' + errorMsg)
							.addClass('gadgetWatchlist-failure gadgetWatchlist-' + action + 'Link');
					}
				});
				e.preventDefault();
			}
			
			function changeWatchStateSuccess(name, isUnwatched) {
				// find full name of associated talk page (or vice versa)
				var ns = getTitleNamespace(name);
				var name2 = name;
				if (ns > 0) {
					name2 = name2.replace(/^.+?:/, '');  // remove old prefix
				}
				ns = ns % 2 ? ns - 1 : ns + 1;  // switch to associated namespace
				if (ns > 0) {
					name2 = mw.config.get('wgFormattedNamespaces')[ns] + ':' + name2;  // add new prefix
				}
				// mark all rows that are either name or name2
				$changeslist.find('.mw-changeslist-title').each(function () {
					var title = getLinkTitle($(this));
					if (title !== name && title !== name2) return;
					var $row = getRow($(this));
					$row.toggleClass('gadgetWatchlist-unwatchedRow', isUnwatched);
					if (!isUnwatched && !allStarsShown) {
						$row.find('.gadgetWatchlist-watchLink').remove();
					} else {
						updateStar($row);
					}
				});
			}
			
			function hideInterface(e) {
				if (e) {
					e.preventDefault();
				}
				
				var hideInterfaceCSSCode = '\
					#firstHeading,\
					div#siteNotice,\
					#contentSub,\
					.mw-wlheader-showupdated,\
					fieldset#mw-watchlist-options,\
					div.mw-rc-label-legend,\
					#mw-fr-watchlist-pending-notice,\
					.mw-indicators,\
					.mw-rcfilters-ui-filterTagMultiselectWidget,\
					.mw-rcfilters-ui-watchlistTopSectionWidget-savedLinksTable,\
					.mw-rcfilters-ui-watchlistTopSectionWidget-editWatchlistButton,\
					.mw-rcfilters-ui-watchlistTopSectionWidget-separator {\
						display: none;\
					}\
					\
					.client-js .mw-special-Watchlist .rcfilters-head.rcfilters-head {\
						min-height: auto;\
					}\
				';
				
				if (!hideInterfaceCSS) {
					// If the new filters are on, wait until the interface is initialized.
					if (!mw.user.options.get('wlenhancedfilters-disable')) {
						hideInterfaceCSS = mw.util.addCSS(hideInterfaceCSSCode);
						$('.mw-rcfilters-ui-markSeenButtonWidget')
							.wrap('<div>')
							.parent()
							.addClass('mw-rcfilters-ui-markSeenButtonWidget-container')
							.appendTo('.mw-rcfilters-ui-watchlistTopSectionWidget-watchlistDetails');
					} else {
						hideInterfaceCSS = mw.util.addCSS(hideInterfaceCSSCode);
					}
				} else {
					hideInterfaceCSS.disabled = !hideInterfaceCSS.disabled;
					if (!mw.user.options.get('wlenhancedfilters-disable')) {
						$('.mw-rcfilters-ui-markSeenButtonWidget-container')
							.appendTo(hideInterfaceCSS.disabled ?
								'.mw-rcfilters-ui-watchlistTopSectionWidget-savedLinksTable .mw-rcfilters-ui-cell:first-child' :
								'.mw-rcfilters-ui-watchlistTopSectionWidget-watchlistDetails');
					}
				}
				
				if (e) {
					if (!hideInterfaceCSS.disabled) {
						var cookieDate = new Date($.now() + SECONDS_IN_A_DAY * 90).toGMTString();
						document.cookie = 'wlmax=1; expires=' + cookieDate + '; path=/';
					} else {
						document.cookie = 'wlmax=0; expires=' + (new Date()).toGMTString() + '; path=/';
					}
				}
			}
			
			function getTitleNamespace(title) {  // returns namespace number
				var prefix = /^(.+?):/.exec(title);
				if (!prefix) {
					return 0;  // no prefix means article
				}
				return mw.config.get('wgNamespaceIds')[ prefix[1].toLowerCase().replace(/ /g, '_') ] || 0;
			}
			
			function getLinkTitle($link) {  // gets title for unwatch/watch links & common page links
				// Titles can be absent for .mw-changeslist-title elements because of Popups extension.
				var title = $link.filter('.mw-changeslist-title').attr('title');
				
				if (!title) {
					title = mw.util.getParamValue('title', $link.attr('href'));
					title = title && title.replace(/_/g, ' ');
				}
				
				if (!title) {
					title = $link.filter('.mw-changeslist-title').text();
				}
				
				return title;
			}
			
			function zzz(s) {  // 5 -> 005
				s = s.toString();
				if (s.length === 1) {
					return '00' + s;
				} else if (s.length === 2) {
					return '0' + s;
				} else {
					return s;
				}
			}
			
			function zeroPad(n, p) {
				return ('0000' + n).slice(-p);
			}
			
			function generateTimestamp(date) {
				return (
					zeroPad(date.getUTCFullYear(), 4) +
					zeroPad(date.getUTCMonth() + 1, 2) +
					zeroPad(date.getUTCDate(), 2) +
					zeroPad(date.getUTCHours(), 2) +
					zeroPad(date.getUTCMinutes(), 2) +
					zeroPad(date.getUTCSeconds(), 2)
				);
			}
			
			
			/* MAIN BLOCK */
			
			// add 5–20 seconds just in case (for example, the scripting could take too much)
			whenPageLoadedOrUpdated = generateTimestamp(new Date(new Date().getTime() - (firstRun ? 20000 : 5000)));
			
			sortingDone = false;
			allStarsShown = false;
			hoverOnTitlesDone = false;
			
			// UNWATCH LINKS
			// on every line
			if (document.cookie.includes('wlunw=1')) {
				showAllStars();
			} else {
				bindHoverOnTitles();  // mouseover on title
			}
			
			if (firstRun) {
				if (document.cookie.includes('wlrvoff=1')) {
					switchRevert();
				}
				
				// find insertion point for links
				$linksIn = $('<div>')
					.addClass('mw-rcfilters-ui-cell gadgetWatchlist-linksContainer')
					.insertBefore($(
						!mw.user.options.get('wlenhancedfilters-disable') ?
							'.mw-rcfilters-ui-watchlistTopSectionWidget-savedLinks' :
							'.wlinfo'
					));
				
				// "show all stars" link
				addLink('', strings.unwatchTip, 'gadgetWatchlist-icon gadgetWatchlist-icon-unwatched')
					.click(showAllStars);
				
				// FUNCTION LINKS
				// "sort" link
				addLink('↑↓', strings.sortTip).click(sortWatchlist);
				
				// "expand all" link
				// Auto-update could be used, so even if there is no $('.mw-enhancedchanges-arrow') elements, we keep
				// the button.
				if (isEnhanced) {
					addLink('±', strings.expandAll).click(expandMultipleEdits);
				}
				
				// "switch revert" link
				if (mw.config.get('wgUserGroups').indexOf('rollbacker') !== -1 ||
					mw.config.get('wgUserGroups').indexOf('sysop') !== -1
				) {
					addLink('⎌', strings.switchRevert).click(switchRevert);
				}
				
				// "new only" link
				addLink(strings.newOnly, strings.newOnlyTip, undefined, generateNewEntriesOnlyUrl())
					.mousedown(newEntriesOnly);
				
				if (mw.util.getParamValue('from') !== null) {
					// "all" link
					addLink(strings.all, strings.allTip, undefined, generateAllEntriesUrl()).mousedown(allEntries);
				}
				
				// TABS
				if (!window.wlNoTabs) {
					var $mainTab = $('#ca-nstab-special').first();  // "Special" tab
					
					// change main tab into "watchlist Δ"
					var watchlistTitle = $.trim($('#firstHeading').text());
					$mainTab
						.find('a')  // replace "Special" with "Watchlist" as tab text
						.text(watchlistTitle + ' △')  // Δ is good but monobook makes is lowercase
						.attr('title', strings.watchlistTabNewOnlyTip)
						.attr('href', generateNewEntriesOnlyUrl())
						.on('mousedown keydown', newEntriesOnly);
					
					// add "hideInterface" tab
					$mainTab
						.clone(true)
							.attr('id', '')
							.attr('href', '')
							.removeClass('selected')
							.click(hideInterface)
							.appendTo($mainTab.parent())
						.find('a')
							.text('↸')
							.attr('title', strings.fullPage)
							.attr('accesskey', '');
				}
				
				
				// OTHER TASKS
				if (document.cookie.includes('wlmax=1')) {
					hideInterface();
				}
				
				mw.util.addCSS('\
					.mw-special-Watchlist .mw-changeslist .mw-rollback-link.mw-rollback-link {\
						visibility: visible;\
					}\
				');
				
				firstRun = false;
			}
			
			return;
		}
		
		// Occurs in watchlist when mediawiki.rcfilters.filters.ui module for some reason fires
		// wikipage.content for the second time with an element that is not in the DOM,
		// fieldset#mw-watchlist-options (in mw.rcfilters.ui.FormWrapperWidget.prototype.onChangesModelUpdate
		// function).
		if (!$content.parent().length) return;
		
		var $changeslist = $content.find('.mw-changeslist');
		
		// Recent changes type in preferences. Don't confuse enhanced recent changes with enhanced filters.
		// mw.user.options.get('usenewrc') shouldn't be used here: RC mode could be set from URL.
		var isEnhanced = !$changeslist.find('ul.special').length;
		
		if (mw.util.getParamValue('from') !== null) {
			$('#mw-watchlist-form input[name="from"]').remove();
		}
		
		if (!mw.user.options.get('wlenhancedfilters-disable')) {
			mw.hook('structuredChangeFilters.ui.initialized').add(main);
		} else {
			if (!navigator.userAgent.toLowerCase().includes('firefox')) {
				main();
			} else {
				// Cure for Firefox
				setTimeout(main, 0);
			}
		}
	});
}

})();