User:Aaron Liu/Watchlyst Greybar Unsin.js
Appearance
Code that you insert on this page could contain malicious content capable of compromising your account. If you import a script from another page with "importScript", "mw.loader.load", "iusc", or "lusc", take note that this causes you to dynamically load a remote script, which could be changed by others. Editors are responsible for all edits and actions they perform, including by scripts. User scripts are not centrally supported and may malfunction or become inoperable due to software changes. A guide to help you find broken scripts is available. If you are unsure whether code you are adding to this page is safe, you can ask at the appropriate village pump. This code will be executed when previewing this page. |
This user script seems to have a documentation page at User:Aaron Liu/Watchlyst Greybar Unsin. |
/* eslint-disable es-x/no-string-prototype-replaceall */
/* Watchlyst Greybar Unsin ([[User:Aaron Liu/Watchlyst Greybar Unsin.js]]) <section begin=version />3.4<section end=version /> */
// <syntaxhighlight lang="js">
( function () {
// utility function that adds CSS so that user can easily override it
function addCss( style ) {
mw.loader.addStyleTag( style, document.head.children[ 0 ] );
}
// non-talk namespaces have an even number
if ( mw.config.get( 'wgNamespaceNumber' ) % 2 === 0 && mw.config.get( 'action' ) !== 'history' ) {
// non-talk, non-history and non-watchlist pages don't have this style for some reason
addCss( '.autocomment,.autocomment a,.autocomment a:visited{color:#72777d}' );
}
const watchlistLink = '<a class="mw-changeslist-watchlist" href="' + mw.util.getUrl( 'Special:Watchlist' ) + '">watchlist</a>';
function getWatchlyst( data ) {
// Don't display if no new watchlist items or if you're on the watchlist page
if ( data.query.watchlist.length !== 0 && mw.config.get( 'wgPageName' ) !== 'Special:Watchlist' ) {
data = data.query.watchlist[ 0 ];
const $dismiss = $( '<button>' ).attr( 'id', 'watchlystDismiss' ).addClass( 'dismissButton' ).text( 'dismiss' )
.on( 'click', () => refreshWatchlyst( data.title, data.timestamp ) );
let summary = data.parsedcomment
.replaceAll( '<', '<' ).replaceAll( '>', '>' ).replaceAll( '"', '"' ); // might edit under categorization
const user = '<a href="' + mw.util.getUrl( data.anon ? 'Special:Contributions/' : 'User:' + data.user ) + '">' + data.user + '</a>';
const page = data.title;
switch ( data.type ) {
case 'new':
case 'edit': {
const changed = data.type === 'edit' ? 'edited' : ( data.type === 'new' ? 'created' : 'changed' );
return $( '<span>' ).html(
'"<a class="mw-changeslist-title" href="' + mw.util.getUrl( page ) + '">' + page + '</a>' +
'" ' + changed + ' by ' + user + ( summary.length === 0 ? '' : ': "' + summary + '"' ) +
'. (<a class="mw-changeslist-diff" href="' + mw.util.getUrl( page, { diff: data.revid } ) + '">diff</a>' +
', <a class="mw-changeslist-hist" href="' + mw.util.getUrl( page, { action: 'history' } ) + '">hist</a>' +
', ' + watchlistLink + ') ' )
.append( $dismiss );
}
case 'categorize':
summary = summary.replace( ' category', ' <a class="mw-changelist-title" href="' + mw.util.getUrl( page ) + '">' + page + ' by ' + user );
// so that we can reuse the same code for quite similar formats...
/* fall through */
case 'log':
return $( '<span>' ).html(
data.logdisplay.replaceAll( '<', '<' ).replaceAll( '>', '>' ).replaceAll( '"', '"' ) +
( summary.length === 0 ? '' : ': "' + summary + '"' ) +
'. (' + watchlistLink + ') ' )
.append( $dismiss );
case 'external': // assuming only WIkidata for now
summary = summary.replaceAll( ' (page does not exist)', '' )
.replaceAll( /<a href="\/w\/index\.php\?title=(.*?)&action=edit&redlink=1" class="new/g, '<a href="https://wikidata.org/wiki/$1" class="' );
return $( '<span>' ).html(
'"<a class="mw-changeslist-title" href="' + mw.util.getUrl( page ) + '">' + page + '</a>' +
'"\'s Wikidata item changed by ' + user + ( summary.length === 0 ? '' : ': "' + summary + '"' ) +
'. (' + watchlistLink + ') ' )
.append( $dismiss );
default:
mw.notify( $( '<p>' ).text( 'New, unsupported watchlist item type found! Please report it to ' )
.append( $( '<a>' ).prop( 'href', 'https://en.wikipedia.org/w/index.php?title=User_talk:Aaron_Liu/Watchlyst_Greybar_Unsin&action=edit§ion=new' )
.text( 'the script\'s talk page.' ) )
, { type: 'error', title: 'Watchlyst' } );
}
}
return '';
}
const $watchlyst = $( '<aside>' ).attr( 'id', 'watchlyst' ); // the parent div
const api = new mw.Api( {
ajax: {
headers: { 'Api-User-Agent': 'Watchlyst/3.4' } },
parameters: { format: 'json', formatversion: '2', errorformat: 'html' }
} );
function refreshWatchlyst( title = null, timestamp = null ) {
if ( typeof ( title ) === 'string' ) {
// strings are immutable, so lets convert which would bump levels correctly
timestamp = new Date( timestamp );
timestamp.setSeconds( timestamp.getSeconds() + 1 );
timestamp = timestamp.toISOString().slice( 0, 19 ) + 'Z'; // API doesn't accept microseconds
api.postWithToken( 'csrf', { action: 'setnotificationtimestamp', titles: [ title ], timestamp: timestamp } ).done( refreshWatchlyst );
return;
}
$watchlyst.addClass( 'loading' );
/* Find the top unread item in the watchlist.
We only need one item, so set the limit to 1 to ease the load on the server. */
api.get( {
action: 'query', list: 'watchlist', wllimit: 1, wldir: 'older', wlshow: 'unread', wltype: [ 'edit', 'new', 'log', 'categorize', 'external' ],
wlexcludeuser: mw.config.get( 'wgUserName' ), wlprop: [ 'parsedcomment', 'ids', 'title', 'user', 'loginfo', 'timestamp' ]
} ).done( ( data ) => {
data = getWatchlyst( data );
if ( data === '' ) {
$watchlyst.remove();
} else {
$watchlyst.html( data ).removeClass( 'loading' );
}
} ).fail( ( data ) => {
$watchlyst.removeClass( 'loading' );
for ( const err in data.errors ) {
$watchlyst.html( err.module + ': ' + err.html + ' (' + err.code + ') ' );
}
} );
}
$( () => {
if ( mw.config.get( 'wgCanonicalSpecialPageName' ) !== 'Watchlist' ) {
addCss( ".dismissButton::before { content: '['; color: #202122; } .dismissButton::after { content: ']'; color: #202122; }" );
addCss( `.dismissButton {
background: transparent;
border: 0; padding: 0;
cursor: pointer;
}` );
addCss( '@media (prefers-reduced-motion: no-preference) { #watchlyst { transition: opacity 0.5s; } }' );
try { // determine color of link
addCss( '.dismissButton { color: ' + getComputedStyle( document.querySelector( '.mw-body-content a:link:not([class])' ) ).getPropertyValue( 'color' ) + '; }' );
} catch ( _ ) {
addCss( '.dismissButton { color: #36c; }' );
}
refreshWatchlyst();
addCss( '.loading { opacity: 0; }' );
$( '#mw-content-subtitle' ).prepend( $watchlyst );
}
} );
}() );
// </syntaxhighlight>