Note: After publishing, you may have to bypass your browser's cache to see the changes.

  • Firefox / Safari: Hold Shift while clicking Reload, or press either Ctrl-F5 or Ctrl-R (⌘-R on a Mac)
  • Google Chrome: Press Ctrl-Shift-R (⌘-Shift-R on a Mac)
  • Edge: Hold Ctrl while clicking Refresh, or press Ctrl-F5.
/* Uploaded from branch master, commit 5d2cd81 */ // vim: ts=4 sw=4 et //<nowiki> function loadReplyLink( $, mw ) {     var TIMESTAMP_REGEX = /\(UTC(?:(?:−|\+)\d+?(?:\.\d+)?)?\)\S*?\s*$/m;     var EDIT_REQ_REGEX = /^((Semi|Template|Extended-confirmed)-p|P)rotected edit request on \d\d? \w+ \d{4}/;     var EDIT_REQ_TPL_REGEX = /\{\{edit (template|fully|extended|semi)-protected\s*(\|.+?)*\}\}/;     var LITERAL_SIGNATURE = "~~" + "~~"; // split up because it might get processed     var i18n = {         "en": {             "rl-advert": " (using [[w:en:User:Enterprisey/reply-link|reply-link]])",             "rl-error-status": "There was an error while replying! Please leave a note at " +                 "<a href='https://en.wikipedia.org/wiki/User_talk:Enterprisey/reply-link'>the script's talk page</a>" +                 " with any errors in <a href='https://en.wikipedia.org/wiki/WP:JSERROR'>the browser console</a>, if possible.",             "rl-replying-to": "Replying to ",             "rl-reloading": "automatically reloading",             "rl-reload": "Reload",             "rl-saved": "Reply saved!",             "rl-cancel": "cancel ",             "rl-placeholder": "Reply here!",             "rl-reply": "Reply",             "rl-preview": "Preview",             "rl-cancel-button": "Cancel",             "rl-started-reply": "You've started a reply but haven't posted it",             "rl-loading": "Loading...",             "rl-reply-label": "reply",             "rl-to-label": " to ",             "rl-auto-indent": "Automatically indent?"         },         "pt": {             "rl-advert": "(usando [[w:en:User:Enterprisey/reply-link|reply-link]])",             "rl-error-status": "Ocorreu um erro ao responder! Por favor deixe um comentário na " +                 "<a href='https://en.wikipedia.org/wiki/User_talk:Enterprisey/reply-link'>página de discussão do script</a>" +                 " informando os erros que apareçam <a href='https://en.wikipedia.org/wiki/WP:JSERROR'>no console do navegador</a>, se possível.",             "rl-replying-to": "Respondendo a ",             "rl-reloading": "recarregando automaticamente",             "rl-reload": "Recarregar",             "rl-saved": "Resposta publicada!",             "rl-cancel": "cancelar ",             "rl-placeholder": "Responda aqui!",             "rl-reply": "Responder",             "rl-preview": "Prever",             "rl-cancel-button": "Cancelar",             "rl-started-reply": "Você começou a responder, mas não publicou sua resposta",             "rl-loading": "Carregando...",             "rl-reply-label": "responder",             "rl-to-label": " a ",             "rl-auto-indent": "Indentar automaticamente?"         }     };     var PARSOID_ENDPOINT = "https:" + mw.config.get( "wgServer" ) + "/api/rest_v1/page/html/";     var HEADER_SELECTOR = "h1,h2,h3,h4,h5,h6";     var MAX_UNICODE_DECIMAL = 1114111;     var HEADER_REGEX = /^\s*=(=*)\s*(.+?)\s*\1=\s*$/gm;      // T:TDYK, used at the end of loadReplyLink     var TTDYK = "Template:Did_you_know_nominations";     var RFA_PG = "Wikipedia:Requests_for_adminship/";      // Threshold for indentation when we offer to outdent     var OUTDENT_THRESH = 8;      // All of the interface message keys that we explicitly load     var INT_MSG_KEYS = [ "mycontris" ];      // Date format regexes in signatures (i.e. the "default date format")     var DATE_FMT_RGX = {         "//en.wikipedia.org": /\d\d:\d\d,\s\d{1,2}\s\w+?\s\d{4}/.source,         "//simple.wikipedia.org": /\d\d:\d\d,\s\d{1,2}\s\w+?\s\d{4}/.source,         "//en.wikisource.org": /\d\d:\d\d,\s\d{1,2}\s\w+?\s\d{4}/.source,         "//pt.wikipedia.org": /\d\dh\d\dmin\sde \d{1,2} de \w+? de \d{4}/.source     }      // Shared API object     var api;      /*      * Regex *sources* for a "userspace" link. Basically the      * localized equivalent of User( talk)?|Special:Contributions/      * Initialized in buildUserspcLinkRgx, which is called near the top      * of the closure in handleWrapperClick.      *      * Three subproperties: und for underscores instead of spaces (e.g.      * "User_talk"), spc for spaces (e.g. "User talk"), and both for      * a regex combining the two (used for matching on wikitext).      */     var userspcLinkRgx = null;      /**      * This dictionary is some global state that holds three pieces of      * information for each "(reply)" link (keyed by their unique IDs):      *      *  - the indentation string for the comment (e.g. ":*::")      *  - the header tuple for the parent section, in the form of      *    [level, text, number], where:      *      - level is 1 for a h1, 2 for a h2, etc      *      - text is the text between the equal signs      *      - number is the zero-based index of the heading from the top      *  - sigIdx, or the zero-based index of the signature from the top      *    of the section      *      * This dictionary is populated in attachLinks, and unpacked in the      * click handler for the links (defined in attachLinkAfterNode); the      * values are then passed to doReply.      */     var metadata = {};      /**      * This global string flag is:      *      *  - "AfD" if the current page is an AfD page      *  - "MfD" if the current page is an MfD page      *  - "TfD" if the current page is a TfD log page      *  - "CfD" if the current page is a CfD log page      *  - "FfD" if the current page is a FfD log page      *  - "" otherwise      *      * This flag is initialized in onReady and used in attachLinkAfterNode      */     var xfdType;      /**      * The current page name, including namespace, because we may be reading it      * a lot (especially in findUsernameInElem if we're on someone's user      * talk page)      */     var currentPageName;      /**      * A map for signatures that contain redirects, so that they can still      * pass the sanity check. This will be updated manually, because I      * don't want the overhead of a whole 'nother API call in the middle      * of the reply process. If this map grows too much, though, I'll      * consider switching to either a toolforge-hosted API or the      * Wikipedia API. Used in doReply, for the username sanity check.      */     var sigRedirectMapping = {         "Salvidrim": "Salvidrim!"     };      /**      * When the reply is saved via API, this flag is set to true to      * disable the onbeforeunload handler.      */     var replyWasSaved = false;      /**      * Cache for getWikitext. Only useful in test mode.      */     var getWikitextCache = {};      // Polyfill from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/includes     if( !String.prototype.includes ) {         String.prototype.includes = function( search, start ) {             if( search instanceof RegExp ) {                 throw TypeError('first argument must not be a RegExp');             }             if( start === undefined ) {                 start = 0;             }             return this.indexOf( search, start ) !== -1;         };     }      /**      * Get the formatted namespace name for a namespace ID.      * Quick ref: user = 2, proj = 4      */     function fmtNs( nsId ) {         return mw.config.get( "wgFormattedNamespaces" )[ nsId ];     }      /**      * Escapes a string for inclusion in a regex.      */     function escapeForRegex( s ) {         return s.replace( /[-\/\\^$*+?.()|[\]{}]/g, '\\$&' );     }      /*      * MediaWiki turns spaces before certain punctuation marks      * into non-breaking spaces, so fix those. This is done by      * the armorFrenchSpaces function in Mediawiki, in the file      * /includes/parser/Sanitizer.php      */     function deArmorFrenchSpaces( text ) {         return text.replace( /\xA0([?:;!%»›])/g, " $1" )             .replace( /([«‹])\xA0/g, "$1 " );     }      /**      * Capitalize the first letter of a string.      */     function capFirstLetter( someString ) {         return someString.charAt( 0 ).toUpperCase() + someString.slice( 1 );     }      /**      * Namespace name to ID.      * For example, nsNameToId( "Template" ) === 10.      */     function nsNameToId( nsName ) {         return mw.config.get( "wgNamespaceIds" )[ nsName.toLowerCase().replace( / /g, "_" ) ];     }      /**      * Canonical-ize a namespace.      */     function canonicalizeNs( ns ) {         return fmtNs( nsNameToId( ns ) );     }      /**      * This function converts any (index-able) iterable into a list.      */     function iterableToList( nl ) {         var len = nl.length;         var arr = new Array( len );         for( var i = 0; i < len; i++ ) arr[i] = nl[i];         return arr;     }      /**      * Process HTML character entities.      * From https://stackoverflow.com/a/46851765      */     function processCharEntities( text ) {         var el = document.createElement('div');         return text.replace( /\&[#0-9a-z]+;/gi, function ( enc ) {             el.innerHTML = enc;             return el.innerText         } );     }      /**      * Process HTML character entities, MediaWiki style      * From https://stackoverflow.com/a/46851765      */     function processCharEntitiesWikitext( text ) {         var el = document.createElement('div');         return text.replace( /\&[#0-9a-z]+;/gi, function ( enc ) {             if( /#\d+/.test( enc ) ) {                 if( parseInt( enc.slice( 1 ) ) > MAX_UNICODE_DECIMAL ) {                     return enc;                 }             }             el.innerHTML = enc;             return el.innerText         } );     }      /**      * When there's a panel being shown, this function sets the status      * in the panel to the first argument. The callback function is      * optional.      */     function setStatus ( status, callback ) {         var statusElement = $( "#reply-dialog-status" );         statusElement.fadeOut( function () {             statusElement.html( status ).fadeIn( callback );         } );     }      /**      * Sets the panel status when an error happened. Good for use in      * catch blocks.      */     function setStatusError( e ) {         console.error(e);         setStatus( mw.msg( "rl-error-status" ) );         if( e.message ) {             console.log( "Content request error: " + JSON.stringify( e.message ) );         }         console.log( "DEBUG INFORMATION: '"+currentPageName+"' @ " +                 mw.config.get( "wgCurRevisionId" ),"parsoid",PARSOID_ENDPOINT+                 encodeURIComponent(currentPageName).replace(/'/g,"%27")+"/"+mw.config.get("wgCurRevisionId") );         throw e;     }      /**      * Given some wikitext, processes it to get just the text content.      * This function should be identical to the MediaWiki function      * that gets the wikitext between the equal signs and comes up      * with the id's that anchor the headers.      */     function wikitextToTextContent( wikitext ) {         return decodeURIComponent( processCharEntities( wikitext ) )             .replace( /\[\[:?(?:[^\|\]]+?\|)?([^\]\|]+?)\]\]/g, "$1" )             .replace( /\{\{\s*tl\s*\|\s*(.+?)\s*\}\}/g, "{{$1}}" )             .replace( /\{\{\s*[Uu]\s*\|\s*(.+?)\s*\}\}/g, "$1" )             .replace( /('''?)(.+?)\1/g, "$2" )             .replace( /<s>(.+?)<\/s>/g, "$1" )             .replace( /<big>(.+?)<\/big>/g, "$1" )             .replace( /<span.*?>(.*?)<\/span>/g, "$1" );     }      function wikitextHeaderEqualsDomHeader( wikitextHeader, domHeader ) {         return wikitextToTextContent( wikitextHeader ) === deArmorFrenchSpaces( domHeader );     }      /**      * Finds and returns the div that is the immediate parent of the      * first talk page header on the page, so that we can read all the      * sections by iterating through its child nodes.      */     function findMainContentEl() {          // Which header are we looking for?         var targetHeader = "h2";         if( xfdType || currentPageName.startsWith( RFA_PG ) ) targetHeader = "h3";         if( currentPageName.startsWith( TTDYK ) ) targetHeader = "h4";          // The element itself will be the text span in the h2; its         // parent will be the h2; and the parent of the h2 is the         // content container that we want         var candidates = document.querySelectorAll( targetHeader + " > span.mw-headline" );         if( !candidates.length ) return null;         var candidate = candidates[candidates.length-1].parentElement.parentElement;          // Compatibility with User:Enterprisey/hover-edit-section         // That script puts each section in its own div, so we need to         // go out another level if it's running         if( candidate.className === "hover-edit-section" ) {             return candidate.parentElement;         } else {             return candidate;         }     }      /**      * Gets the wikitext of a page with the given title (namespace required).      * Returns an object with keys "content" and "timestamp".      */     function getWikitext( title, useCaching ) {         if( useCaching === undefined ) useCaching = false;         if( useCaching && getWikitextCache[ title ] ) {             return $.when( getWikitextCache[ title ] );         }         return $.getJSON(             mw.util.wikiScript( "api" ),             {                 format: "json",                 action: "query",                 prop: "revisions",                 rvprop: "content",                 rvslots: "main",                 rvlimit: 1,                 titles: title             }         ).then( function ( data ) {             var pageId = Object.keys( data.query.pages )[0];             if( data.query.pages[pageId].revisions ) {                 var revObj = data.query.pages[pageId].revisions[0];                 var result = { timestamp: revObj.timestamp, content: revObj.slots.main["*"] };                 getWikitextCache[ title ] = result;                 return result;             }             return {};         } );     }      /**      * Creates userspcLinkRgx. Called in handleWrapperClick and the test      * runner at the bottom.      */     function buildUserspcLinkRgx() {         var nsIdMap = mw.config.get( "wgNamespaceIds" );         var nsRgxFragments = [];         var contribsSecondFrag = ":" + escapeForRegex( mw.messages.get( "mycontris" ) ) + "\\/";         for( var nsName in nsIdMap ) {             if( !nsIdMap.hasOwnProperty( nsName ) ) continue;             switch( nsIdMap[nsName] ) {                 case 2:                 case 3:                     nsRgxFragments.push( escapeForRegex( capFirstLetter( nsName ) ) + "\\s*:" );                     break;                 case -1:                     nsRgxFragments.push( escapeForRegex( capFirstLetter( nsName ) ) + contribsSecondFrag );                     break;             }         }         userspcLinkRgx = {};         userspcLinkRgx.spc = "(?:" + nsRgxFragments.join( "|" ).replace( /_/g, " " ) + ")";         userspcLinkRgx.und = userspcLinkRgx.spc.replace( / /g, "_" );         userspcLinkRgx.both = userspcLinkRgx.spc.replace( / /g, "(?: |_)" );     }      /**      * Is there a signature (four tildes) present in the given text,      * outside of a nowiki element?      */     function hasSig( text ) {          // no literal signature?         if( !text.includes( LITERAL_SIGNATURE ) ) return false;          // if there's a literal signature and no nowiki elements,         // there must be a real signature         if( !text.includes( "<nowiki>" ) ) return true;          // Save all nowiki spans         var nowikiSpanStarts = []; // list of ignored span beginnings         var nowikiSpanLengths = []; // list of ignored span lengths         var NOWIKI_RE = /<nowiki>.*?<\/nowiki>/g;         var spanMatch;         do {             spanMatch = NOWIKI_RE.exec( text );             if( spanMatch ) {                 nowikiSpanStarts.push( spanMatch.index );                 nowikiSpanLengths.push( spanMatch[0].length );             }         } while( spanMatch );          // So that we don't check every ignore span every time         var nowikiSpanStartIdx = 0;          var LIT_SIG_RE = new RegExp( LITERAL_SIGNATURE, "g" );         var sigMatch;          matchLoop:         do {             sigMatch = LIT_SIG_RE.exec( text );             if( sigMatch ) {                  // Check that we're not inside a nowiki                 for( var nwIdx = nowikiSpanStartIdx; nwIdx <                     nowikiSpanStarts.length; nwIdx++ ) {                     if( sigMatch.index > nowikiSpanStarts[nwIdx] ) {                         if ( sigMatch.index + sigMatch[0].length <=                             nowikiSpanStarts[nwIdx] + nowikiSpanLengths[nwIdx] ) {                              // Invalid sig                             continue matchLoop;                         } else {                              // We'll never encounter this span again, since                             // headers only get later and later in the wikitext                             nowikiSpanStartIdx = nwIdx;                         }                     }                 }                  // We aren't inside a nowiki                 return true;             }         } while( sigMatch );         return false;     }      /**      * Given an Element object, attempt to recover a username from it.      * Also will check up to two elements prior to the passed element.      * Returns null if no username was found. Otherwise, returns an      * object with these properties:      *      *  - username: The username that we found.      *  - link: The DOM object for the link from which we got the      *    username.      */     function findUsernameInElem( el ) {         if( !el ) return null;         var links;         for( let i = 0; i < 3; i++ ) {             if( el === null ) break;             links = el.tagName.toLowerCase() === "a" ? [ el ]                 : el.querySelectorAll( "a" );             //console.log(i,"top of outer for in findUsernameInElem ",el, " links -> ",links);              // Compatibility with "Comments in Local Time"             if( el.className.includes( "localcomments" ) ) i--;              // If we couldn't get any links, try again with prev elem             if( !links ) continue;              var link; // his name isn't zelda             for( var j = 0; j < links.length; j++ ) {                 link = links[j];                  //console.log(link,decodeURIComponent(link.getAttribute("href")));                 if( link.className.includes( "mw-selflink" ) ) {                     return { username: currentPageName.replace( /.+:/, "" )                         .replace( /_/g, " " ), link: link };                 }                  // Also matches redlinks. Why people have redlinks in their sigs on                 // purpose, I may never know.                 //console.log( "^\\/(?:wiki\\/" + userspcLinkRgx.und + /(.+?)(?:\/.+?)?(?:#.+)?|w\/index\.php\?title=User(?:_talk)?:(.+?)&action=edit&redlink=1/.source + ")$" )                 var sigLinkRe = new RegExp( "\\/(?:wiki\\/" + userspcLinkRgx.und + /(.+?)(?:\/.+?)?(?:#.+)?|w\/index\.php\?title=/.source + userspcLinkRgx.und + /(.+?)&action=edit&redlink=1/.source + ")$" );                 var liveDecodedHref = decodeURIComponent( link.getAttribute( "href" ) );                 if( liveDecodedHref.startsWith( "/" ) ) {                     liveDecodedHref = "https:" + mw.config.get( "wgServer" ) + liveDecodedHref;                 }                 var usernameMatch = sigLinkRe.exec( liveDecodedHref );                 if( usernameMatch ) {                 //console.log("usernameMatch",usernameMatch)                     var rawUsername = usernameMatch[1] ? usernameMatch[1] : usernameMatch[2];                     return {                         username: decodeURIComponent( rawUsername ).replace( /_/g, " " ),                         link: link                     };                 }             }              // Go backwards one element and try again             el = el.previousElementSibling;         }         return null;     }      /**      * Given a reply-link-wrapper span, attempts to find who wrote      * the comment that precedes it. For information about the return      * value, see the documentation for findUsernameInElem.      */     function getCommentAuthor( wrapper ) {         var sigNode = wrapper.previousSibling;         //console.log(sigNode,sigNode.style,sigNode.style ? sigNode.style.getPropertyValue("size"):"");         var smallOrFake = sigNode.nodeType === 1 &&                 ( sigNode.tagName.toLowerCase() === "small" ||                 ( sigNode.tagName.toLowerCase() === "span" &&                     sigNode.style && ( sigNode.style.getPropertyValue( "font-size" ) === "85%" ||                                        sigNode.style.getPropertyValue( "font-size" ).indexOf( "small" ) === 0 ) ) );          var possUserLinkElem = ( smallOrFake && sigNode.children.length > 1 )             ? sigNode.children[sigNode.children.length-1]             : sigNode.previousElementSibling;         return findUsernameInElem( possUserLinkElem );     }      /**      * Given the wikitext of a section, attempt to find the first edit      * request template in it, and then mark that template as answered.      * Returns the modified section wikitext.      */     function markEditReqAnswered( sectionWikitext ) {         var editReqMatch = EDIT_REQ_TPL_REGEX.exec( sectionWikitext );         if( !editReqMatch ) {             console.error( "Couldn't find an edit request!" );             return sectionWikitext;         }          var ansParamMatch = /ans(wered)?=.*?(\||\}\})/.exec( editReqMatch[0] );         if( !ansParamMatch ) {             sectionWikitext = sectionWikitext.replace(                 editReqMatch[0],                 editReqMatch[0].replace( "}}", "answered=yes}}" )             );         } else {             var newEditReqTpl = editReqMatch[0].replace( ansParamMatch[0],                 "answered=yes" + ansParamMatch[2] );             sectionWikitext = sectionWikitext.replace(                 editReqMatch[0],                 newEditReqTpl             );         }         return sectionWikitext;     }      /**      * Ascend until dd or li, or a p directly under div.mw-parser-output.      * live is true if we're on the live DOM (and thus we have our own UI      * elements to deal with) and false if we're on the psd DOM.      */     function ascendToCommentContainer( startNode, live, recordPath ) {         var currNode = startNode;         if( recordPath === undefined ) recordPath = false;         var path = [];         var lcTag;         var headerRegex = /h\d+/i;          function hasHeaderAsAnyPreviousSibling( node ) {             do {                 if( headerRegex.test( node.tagName ) ) {                     return true;                 }                 node = node.previousElementSibling;             } while( node );         }          function isActualContainer( node, nodeLcTag ) {             if( nodeLcTag === undefined ) nodeLcTag = node.tagName.toLowerCase();             return /dd|li/.test( nodeLcTag ) ||                     ( ( nodeLcTag === "p" || nodeLcTag === "div" ) &&                         ( node.parentNode.className === "mw-parser-output" ||                             hasHeaderAsAnyPreviousSibling( node ) ||                             node.parentNode.className === "hover-edit-section" ||                             ( node.parentNode.tagName.toLowerCase() === "section" &&                                 node.parentNode.dataset.mwSectionId ) ) );         }          var smallContainerNodeLimit = live ? 3 : 1;         do {             currNode = currNode.parentNode;             lcTag = currNode.tagName.toLowerCase();             if( lcTag === "html" ) {                 console.error( "ascendToCommentContainer reached root" );                 break;             }             if( recordPath ) path.unshift( currNode );             //console.log( "checking isActualContainer for ", currNode, isActualContainer( currNode, lcTag ),             //        lcTag === "small", isActualContainer( currNode.parentNode ),             //            currNode.parentNode.childNodes,             //            currNode.parentNode.childNodes.length );         } while( !isActualContainer( currNode, lcTag ) &&             !( lcTag === "small" && isActualContainer( currNode.parentNode ) &&                 currNode.parentNode.childNodes.length <= smallContainerNodeLimit ) );         //console.log("ascendToCommentContainer from ",startNode," terminating, r.v. ",recordPath?path:currNode);         return recordPath ? path : currNode;     }      /**      * Given a Parsoid DOM and a link in the live DOM that is the link at the      * end of a signature, return the corresponding element in the Parsoid DOM      * that represents the same comment, or null if none was found.      *      * psd = Parsoid, live = in the current, live page DOM.      */     function getCorrCmt( psdDom, sigLinkElem ) {          // First, define some helper functions          // Does this node have a timestamp in it?         function hasTimestamp( node ) {             //console.log ("hasTimestamp ",node, node.nodeType === 3,node.textContent.trim(),             //            TIMESTAMP_REGEX.test( node.textContent.trim() ),             //        node.childNodes.length === 1,             //            node.childNodes.length && TIMESTAMP_REGEX.test( node.childNodes[0].textContent.trim()),             //        " => ",( node.nodeType === 3 &&             //                TIMESTAMP_REGEX.test( node.textContent.trim() ) ) ||             //           ( node.childNodes.length === 1 &&             //                TIMESTAMP_REGEX.test( node.childNodes[0].textContent.trim() ) ) );             //console.log(node,node.textContent.trim(),TIMESTAMP_REGEX.test(node.textContent.trim()));             var validTag = node.nodeType === 3 || ( node.nodeType === 1 &&                             ( node.tagName.toLowerCase() === "small" ||                                 node.tagName.toLowerCase() === "span" ) );             return ( validTag && TIMESTAMP_REGEX.test( node.textContent.trim() ) ||                    ( node.childNodes.length === 1 &&                         TIMESTAMP_REGEX.test( node.childNodes[0].textContent.trim() ) ) );         }          // Get prefix that's the actual comment         function getPrefixComment( theNodes ) {             var prefix = [];             for( var j = 0; j < theNodes.length; j++ ) {                 prefix.push( theNodes[j] );                 if( hasTimestamp( theNodes[j] ) ) break;             }             return prefix;         }          /**          * From a "container elem" (like the whole dd, li, or p that has a          * comment), get the prefix that ends in a timestamp (because other          * comments might be after the timestamp), and return the text content.          */         function surrTextContentFromElem( elem ) {             var surrListElemNodes = elem.childNodes;              // nodeType 8 is for comments             return getPrefixComment( surrListElemNodes )                     .map( function ( c ) { return ( c.nodeType !== 8 ) ? c.textContent : ""; } )                     .join( "" ).trim();         }          /** From a "container elem" (dd, li, or p), remove all but the first comment. */         function onlyFirstComment( container ) {             //console.log("onlyFirstComment top container and container.childNodes",container,container.childNodes);             if( container.childNodes.length === 1 && container.children[0].tagName.toLowerCase() === "small" ) {                 console.log( "[onlyFirstComment] container only had a small in it" );                 container = container.children[0];             }             var i, autosignedIdx, autosigned = container.querySelector( "small.autosigned" );             if( autosigned && ( autosignedIdx = iterableToList(                     container.childNodes ).includes( autosigned ) ) ) {                 i = autosignedIdx;             } else {                 var childNodes = container.childNodes;                 for( i = 0; i < childNodes.length; i++ ) {                     if( hasTimestamp( childNodes[i] ) ) {                         //console.log( "[oFC] found a timestamp in ",childNodes[i]);                         break;                     }                 }                 if( i === childNodes.length ) {                     throw new Error( "[onlyFirstComment] No timestamp found" );                 }             }             //console.log("[onlyFirstComment] killing all after ",i,container.childNodes[i]);             i++;             var elemToRemove;             while( elemToRemove = container.childNodes[i] ) {                 container.removeChild( elemToRemove );             }         }          // End helper functions, begin actual code          // We dump this object for debugging in the event of an error         var corrCmtDebug = {};          // Convert live href to psd href (aka newHref)         var newHref, liveHref = decodeURIComponent( sigLinkElem.getAttribute( "href" ) );         corrCmtDebug.liveHref = liveHref;         if( sigLinkElem.className.includes( "mw-selflink" ) ) {             newHref = "./" + currentPageName;         } else {             if( /^\/wiki/.test( liveHref ) ) {                 var hrefTokens = liveHref.split( ":" );                 if( hrefTokens.length !== 2 ) throw new Error( "Malformed href" );                 newHref = "./" + canonicalizeNs( hrefTokens[0].replace(                         /^\/wiki\//, "" ) ).replace( / /g, "_" ) + ":" +                         hrefTokens[1]                             .replace( /^Contributions%2F/, "Contributions/" )                             .replace( /%2F/g, "/" )                             .replace( /%23/g, "#" )                             .replace( /%26/g, "&" )                             .replace( /%3D/g, "=" )                             .replace( /%2C/g, "," );             } else {                 var REDLINK_HREF_RGX = /^\/w\/index\.php\?title=(.+?)&action=edit&redlink=1$/;                 var redlinkMatch = REDLINK_HREF_RGX.exec( liveHref );                 if( redlinkMatch ) {                     newHref = "./" + redlinkMatch[1];                 } else {                     newHref = liveHref.replace( /_/g, '%20' );                 }             }         }         newHref = newHref.replace( /\\/g, "\\\\" )             .replace( /'/g, "\\'" )             .replace( /\?/g, "%3F" );         var livePath = ascendToCommentContainer( sigLinkElem, /* live */ true, /* recordPath */ true );         corrCmtDebug.newHref = newHref; corrCmtDebug.livePath = livePath;          // Deal with the case where the comment has multiple links to         // sigLinkElem's href; we will store the index of the link we want.         // null means there aren't multiple links.         if( liveHref ) {             liveHref = liveHref.replace( /'/g, "\\'" );         }         var liveDupeLinks = livePath[0].querySelectorAll( "a" +                 ( liveHref ? ( "[href='" + liveHref + "']" ) : ".mw-selflink" ) );         if( !liveDupeLinks ) throw new Error( "Couldn't select live dupe link" );         var liveDupeLinkIdx = ( liveDupeLinks.length > 1 )                 ? iterableToList( liveDupeLinks ).indexOf( sigLinkElem ) : null;         //console.log("liveDupeLinkIdx",liveDupeLinkIdx);          //console.log("livePath[0]",livePath[0],livePath[0].childNodes);         var liveClone = livePath[0].cloneNode( /* deep */ true );          // Remove our own UI elements         var ourUiSelector = ".reply-link-wrapper,#reply-link-panel";         iterableToList( liveClone.querySelectorAll( ourUiSelector ) ).forEach( function ( n ) {             n.parentNode.removeChild( n );         } );          //console.log("(BEFORE) liveClone",liveClone,liveClone.childNodes);         onlyFirstComment( liveClone );         //console.log("(AFTER) liveClone",liveClone,liveClone.childNodes);          // Process it a bit to make it look a bit more like the Parsoid output         var liveAutoNumberedLinks = liveClone.querySelectorAll( "a.external.autonumber" );         for( var i = 0; i < liveAutoNumberedLinks.length; i++ ) {             liveAutoNumberedLinks[i].textContent = "";         }         var liveSelflinks = liveClone.querySelectorAll( "a.mw-selflink.selflink" );         for( var i = 0; i < liveSelflinks.length; i++ ) {             liveSelflinks[i].href = "/wiki/" + currentPageName;         }          // "Comments in Local Time" compatibility: the text content is         // gonna contain the modified time stamp, but the original time         // stamp is still there         var localCommentsSpan = liveClone.querySelector( "span.localcomments" );         if( localCommentsSpan ) {             var dateNode = document.createTextNode( localCommentsSpan.getAttribute( "title" ) );             localCommentsSpan.parentNode.replaceChild( dateNode, localCommentsSpan );         }          // User:Writ Keeper/Scripts/teahouseTalkbackLink.js compatibility:         // get rid of the |C|TB that it adds         var teahouseTalkbackLink = liveClone.querySelector( "a[id^=TBsubmit]" );         if( teahouseTalkbackLink ) {             teahouseTalkbackLink.parentNode.removeChild( teahouseTalkbackLink.nextSibling );             for( var ttlIdx = 0; ttlIdx < 3; ttlIdx++ ) {                 teahouseTalkbackLink.parentNode.removeChild( teahouseTalkbackLink.previousSibling );             }             teahouseTalkbackLink.parentNode.removeChild( teahouseTalkbackLink );         }          var adminMarksClass = liveClone.querySelectorAll( "b.adminMark" );         if ( adminMarksClass.length > 0 ) {             adminMarksClass.forEach( function ( currentValue, currentIndex, listObj ) {                 currentValue.parentNode.removeChild( currentValue );             } );         }                                // TODO: Optimization - surrTextContentFromElem does the prefixing         // operation a second time, even though we already called onlyFirstComment         // on it.         var liveTextContent = surrTextContentFromElem( liveClone );         console.log("liveTextContent >>>>>"+liveTextContent + "<<<<<");          function normalizeTextContent( tc ) {             return deArmorFrenchSpaces( tc );         }          liveTextContent = normalizeTextContent( liveTextContent );          var selector = livePath.map( function ( node ) {             return node.tagName.toLowerCase();         } ).join( " " ) + " a[href^='" + newHref + "']";          // TODO: Optimization opportunity - run querySelectorAll only on the         // section that we know contains the comment         var psdLinks = iterableToList( psdDom.querySelectorAll( selector ) );         console.log("(",liveDupeLinkIdx, ")",selector, " --> ", psdLinks);          var oldPsdLinks = psdLinks,             newHrefLen = newHref.length,             hrefSubstr;         psdLinks = [];         for( var i = 0; i < oldPsdLinks.length; i++ ) {             hrefSubstr = oldPsdLinks[i].getAttribute( "href" ).substring( newHrefLen );             if( !hrefSubstr || hrefSubstr.indexOf( "#" ) === 0 ) {                 psdLinks.push( oldPsdLinks[i] );             }         }          // Narrow down by entire textContent of list element         var psdCorrLinks = []; // the corresponding link elem(s)         if( liveDupeLinkIdx === null ) {             for( var i = 0; i < psdLinks.length; i++ ) {                 var psdContainer = ascendToCommentContainer( psdLinks[i], /* live */ false, true );                 //console.log("psdContainer",psdContainer);                 var psdTextContent = normalizeTextContent( surrTextContentFromElem( psdContainer[0] ) );                 //console.log(i,">>>"+psdTextContent+"<<<");                 if( psdTextContent === liveTextContent ) {                     psdCorrLinks.push( psdLinks[i] );                 } /* else {                     //console.log(i,"len: psd live",psdTextContent.length,liveTextContent.length);                     for(var j = 0; j < Math.min(psdTextContent.length, liveTextContent.length); j++) {                         if(psdTextContent.charAt(j)!==liveTextContent.charAt(j)) {                             //console.log(i,j,"psd live", psdTextContent.codePointAt(j), liveTextContent.codePointAt( j ) );                             break;                         }                     }                 } */             }         } else {             for( var i = 0; i < psdLinks.length; i++ ) {                 var psdContainer = ascendToCommentContainer( psdLinks[i], /* live */ false );                 if( psdContainer.dataset.replyLinkGeCorrCo ) continue;                 var psdTextContent = normalizeTextContent( surrTextContentFromElem( psdContainer ) );                 console.log(i,">>>"+psdTextContent+"<<<");                 if( psdTextContent === liveTextContent ) {                     var psdDupeLinks = psdContainer.querySelectorAll( "a[href='" + newHref + "']" );                     psdCorrLinks.push( psdDupeLinks[ liveDupeLinkIdx ] );                 }                  // Flag to ensure we don't take a link from this container again                 psdContainer.dataset.replyLinkGeCorrCo = true;             }         }          if( psdCorrLinks.length === 0 ) {             console.error( "Failed to find a matching comment in the Parsoid DOM." );             return null;         } else if( psdCorrLinks.length > 1 ) {             console.error( "Found multiple matching comments in the Parsoid DOM." );             return null;         }          return psdCorrLinks[0];     }      /**      * Given a page title, the Parsoid output (GET /page/html endpoint)      * of that page, page and a DOM object in the current page      * corresponding to a link in a signature, locate the section      * containing that comment. That section may not be in the provided      * page! Returns an object with these properties:      *      *  - page: The full title of the page directly containing the      *    comment (in its wikitext, not through transclusion).      *  - sectionName: The anticipated wikitext section name. Should      *    appear inside the equal signs at the above index.      *  - sectionDupeIdx: If there are multiple sections with the same      *    name, the 0-based index of the section with the comment among      *    those sections. Otherwise, 0.      *  - sectionLevel: The anticipated wikitext section level (e.g.      *    2 for an h2)      *  - nearbyMwId: The Parsoid ID of some element near the      *    comment (in practice, a userspace link) for jumping purposes.      *      * Parsoid is abbreviated here as "psd" in variables and comments.      */     function findSection( psdDomPageTitle, psdDomString, sigLinkElem ) {         console.log("findSection(",psdDomPageTitle,", ...)");          //console.log(psdDomString);          var domParser = new DOMParser(),             psdDom = domParser.parseFromString( psdDomString, "text/html" );          var corrLink = getCorrCmt( psdDom, sigLinkElem );         if( corrLink === null ) {             return $.when();         }         //console.log("STEP 1 SUCCESS",corrLink);          var corrCmt = ascendToCommentContainer( corrLink, /* live */ false );          // Ascend until we hit something in a transclusion         var currNode = corrLink;         var tsclnId = null;         do {             if( currNode.getAttribute( "about" ) &&                     currNode.getAttribute( "about" ).indexOf( "#mwt" ) === 0 ) {                 tsclnId = currNode.getAttribute( "about" );                 break;             }             currNode = currNode.parentNode;         } while( currNode.tagName.toLowerCase() !== "html" );         //console.log( "tsclnId", tsclnId );          // Helper function: are we in a pseudo-section? (Unused, at the moment.)         function inPseudo( headerElement ) {             var currNodeIP = headerElement;             // This requires Parsoid HTML v 2.0.0             do {                 if( currNodeIP.nodeType === 1 && currNodeIP.matches( "section" ) ) {                     return currNodeIP.dataset.mwSectionId < 0;                     break;                 }                 currNodeIP = currNodeIP.parentNode;             } while( currNodeIP );             return false;         }          // Now, get the nearest header above us         var currNode = corrCmt;         var nearestHeader = null;         var HTML_HEADER_RGX = /^h\d$/;         do {             if( HTML_HEADER_RGX.exec( currNode.tagName.toLowerCase() ) ) {                 // Commented because I don't think the !inPseudo requirement is necessary 2019-nov-01                 //if( !inPseudo( currNode ) ) {                     nearestHeader = currNode;                     break;                 //}             }             var containedHeaders = currNode.querySelectorAll( HEADER_SELECTOR );             if( containedHeaders.length ) {                 var nearestHdrIdx = containedHeaders.length - 1;                 // Commented because I don't think the !inPseudo requirement is necessary 2019-nov-01                  // TODO this is an extraordinarily silly while loop; it has been temporarily commented 2020-apr-25                 //while( nearestHdrIdx >= 0 ){//&& inPseudo( containedHeaders[ nearestHdrIdx ] ) )                  //    nearestHdrIdx--;                 //}                 if( nearestHdrIdx >= 0 ) {                     nearestHeader = containedHeaders[ nearestHdrIdx ];                     break;                 }             }             if( currNode.previousElementSibling ) {                 currNode = currNode.previousElementSibling;                 continue;             }             currNode = currNode.parentNode;         } while( currNode.tagName.toLowerCase() !== "body" );          // Get the target page (page actually containing the comment)         var targetPage;         if( tsclnId === null ) {             console.warn( "tsclnId === null" );             targetPage = psdDomPageTitle;         } else {             var tsclnInfoSel = "*[about='" + tsclnId + "'][typeof='mw:Transclusion']",                 infoJson = JSON.parse( psdDom.querySelector( tsclnInfoSel ) .dataset.mw );              // First, check the first and last wikitext segments to see if they have the header             var firstWktxtSegIdx = 0;             while( infoJson.parts[firstWktxtSegIdx].template &&                 infoJson.parts[firstWktxtSegIdx].template.target.href.startsWith( "Template:" ) &&                 firstWktxtSegIdx < infoJson.parts.length ) {                 firstWktxtSegIdx++;             }             if( firstWktxtSegIdx < infoJson.parts.length && typeof infoJson.parts[firstWktxtSegIdx] === typeof '' ) {                 var firstWktxtSeg = infoJson.parts[firstWktxtSegIdx];                 var headerMatch = null;                 do {                     headerMatch = HEADER_REGEX.exec( firstWktxtSeg );                     if( headerMatch ) {                         if( wikitextHeaderEqualsDomHeader( headerMatch[2], nearestHeader.textContent ) ) {                             targetPage = psdDomPageTitle;                             break;                         }                     }                 } while( headerMatch );             }         }          if( !targetPage ) {             var lastWktxtSegIdx = infoJson.parts.length - 1;             while( infoJson.parts[lastWktxtSegIdx].template &&                 infoJson.parts[lastWktxtSegIdx].template.target.href.startsWith( "Template:" ) &&                 lastWktxtSegIdx >= 0 ) {                 lastWktxtSegIdx--;             }             if( lastWktxtSegIdx >= 0 && typeof infoJson.parts[lastWktxtSegIdx] === typeof '' ) {                 var lastWktxtSeg = infoJson.parts[lastWktxtSegIdx];                 var headerMatch = null;                 do {                     headerMatch = HEADER_REGEX.exec( lastWktxtSeg );                     if( headerMatch ) {                         if( wikitextHeaderEqualsDomHeader( headerMatch[2], nearestHeader.textContent ) ) {                             targetPage = psdDomPageTitle;                             break;                         }                     }                 } while( headerMatch );             }         }          var recursiveCalls = $.when();         if( !targetPage ) {             // Recurse on all non-top-level Templates!              var pages = infoJson.parts.filter( function ( part ) {                 return part.template &&                     part.template.target &&                     part.template.target.href && (                         !part.template.target.href.startsWith("./Template") ||                         ( part.template.target.href.match( new RegExp( '/', 'g' ) ) || [] ).length >= 2                     );             } );             if( pages.length ) {                 var pageNames = pages.map( function ( part ) {                     return part.template.target.href.substring( 2 ); // remove the ./                 } );                 var deferreds = pageNames.map( function ( pageName ) {                     return $.get( PARSOID_ENDPOINT + encodeURIComponent( pageName ) )                         .then( function ( data ) { return data; } ); // truncate to first argument, which is the data                 } );                 recursiveCalls = $.when.apply( $, deferreds ).then( function () {                     var results = arguments; // use keyword "arguments" to access deferred results                     var deferreds2 = [];                     if( pageNames.length !== results.length ) {                         console.error(pageNames,results);                         throw new Error( "pageNames.length !== results.length: " + pageNames.length + " " + results.length );                     }                     for( var i = 0; i < pageNames.length; i++ ) {                         deferreds2.push( findSection( pageNames[i], results[i], sigLinkElem ) );                     }                     return $.when.apply( $, deferreds2 ).then( function () {                         var results2 = Array.prototype.slice.call( arguments );                         var namesAndResults2 = [];                         if( pageNames.length !== results2.length ) {                             throw new Error( "pageNames.length !== results2.length: " + pageNames.length + " " + results2.length );                         }                         for( var i = 0; i < pageNames.length; i++ ) {                             if( results2[i] ) {                                 namesAndResults2.push( [ pageNames[i], results2[i] ] );                             }                         }                         if( namesAndResults2.length === 0 ) {                             return null;                         } else if( namesAndResults2.length === 1 ) {                             return namesAndResults2[0][1];                         } else {                             var allSameName = namesAndResults2.every( function ( nameAndResult ) {                                 return nameAndResult[0] === namesAndResults2[0][0];                             } );                             if( allSameName ) {                                 return namesAndResults2[0][1];                             } else {                                 console.error( "WTF", namesAndResults2 );                             }                         }                     } );                 } );             }         }          return recursiveCalls.then( function ( data ) {             if( data ) {                 return data;             } else if( nearestHeader === null ) {                 return {                     page: targetPage,                     sectionName: "",                     sectionDupeIdx: 0,                     sectionLevel: 0,                     nearbyMwId: corrCmt.id                 };             } else {                  // We tried recursing, and it didn't work, so the                 // section must be on the current page                 targetPage = psdDomPageTitle;                  // Finally, get the index of our nearest header                 var allHeaders = iterableToList( psdDom.querySelectorAll( HEADER_SELECTOR ) );                  var sectionDupeIdx = 0;                 for( var i = 0; i < allHeaders.length; i++ ) {                     if( allHeaders[i].textContent === nearestHeader.textContent ) {                         if( allHeaders[i] === nearestHeader ) {                             break;                         } else {                             sectionDupeIdx++;                         }                     }                 }                  var result = {                     page: targetPage,                     sectionName: nearestHeader.textContent,                     sectionDupeIdx: sectionDupeIdx,                     sectionLevel: nearestHeader.tagName.substring( 1 ), // that is, cut off the "h" at the beginning                     nearbyMwId: corrCmt.id                 };                 //console.log("findSection return val: ",result);                 return result;             }         } );     }      /**      * Given some wikitext that's split into sections, return the full      * wikitext (including header and newlines until the next header) of      * the section with the given name. To get the content before the      * first header, sectionName should be "".      *      * Performs a sanity check with the given section name.      */     function getSectionWikitext( wikitext, sectionName, sectionDupeIdx ) {         console.log("In getSectionWikitext, sectionName = >" + sectionName + "< (wikitext.length = " + wikitext.length + ")");         //console.log("wikitext (first 1000 chars) is " + dirtyWikitext.substring(0, 1000));          // There are certain locations where a header may appear in the         // wikitext, but will not be present in the HTML; such as code         // blocks or comments. So we keep track of those ranges         // and ignore headings inside those.         var ignoreSpanStarts = []; // list of ignored span beginnings         var ignoreSpanLengths = []; // list of ignored span lengths         var IGNORE_RE = /(<pre>[\s\S]+?<\/pre>)|(<source.+?>[\s\S]+?<\/source>)|(<!--[\s\S]+?-->)/g;         var ignoreSpanMatch;         do {             ignoreSpanMatch = IGNORE_RE.exec( wikitext );             if( ignoreSpanMatch ) {                 //console.log("ignoreSpan ",ignoreSpanStarts.length," = ",ignoreSpanMatch);                 ignoreSpanStarts.push( ignoreSpanMatch.index );                 ignoreSpanLengths.push( ignoreSpanMatch[0].length );             }         } while( ignoreSpanMatch );          var startIdx = -1; // wikitext index of section start         var endIdx = -1; // wikitext index of section end          var headerCounter = 0;         var headerMatch;          // So that we don't check every ignore span every time         var ignoreSpanStartIdx = 0;          var dupeCount = 0;         var lookingForEnd = false;          if( sectionName === "" ) {             // Getting first section             startIdx = 0;             lookingForEnd = true;         }          // Reset regex state, if for some reason we're not running this for the first time         HEADER_REGEX.lastIndex = 0;          headerMatchLoop:         do {             headerMatch = HEADER_REGEX.exec( wikitext );             if( headerMatch ) {                  // Check that we're not inside one of the "ignore" spans                 for( var igIdx = ignoreSpanStartIdx; igIdx <                     ignoreSpanStarts.length; igIdx++ ) {                     if( headerMatch.index > ignoreSpanStarts[igIdx] ) {                         if ( headerMatch.index + headerMatch[0].length <=                             ignoreSpanStarts[igIdx] + ignoreSpanLengths[igIdx] ) {                              console.log("(IGNORED, igIdx="+igIdx+") Header " + headerCounter + " (idx " + headerMatch.index + "): >" + headerMatch[0].trim() + "<");                              // Invalid header                             continue headerMatchLoop;                         } else {                              // We'll never encounter this span again, since                             // headers only get later and later in the wikitext                             ignoreSpanStartIdx = igIdx;                         }                     }                 }                  //console.log("Header " + headerCounter + " (idx " + headerMatch.index + "): >" + headerMatch[0].trim() + "<");                 // Note that if the lookingForEnd block were second,                 // then two consecutive matching section headers might                 // result in the wrong section being matched!                 if( lookingForEnd ) {                     endIdx = headerMatch.index;                     break;                 } else if( wikitextHeaderEqualsDomHeader( /* wikitext */ headerMatch[2], /* dom */ sectionName ) ) {                     if( dupeCount === sectionDupeIdx ) {                         startIdx = headerMatch.index;                         lookingForEnd = true;                     } else {                         dupeCount++;                     }                 }             }             headerCounter++;         } while( headerMatch );          if( startIdx < 0 ) {             throw( "Could not find section named \"" + sectionName + "\" (dupe idx " + sectionDupeIdx + ")" );         }          // If we encountered no section after the target section,         // then the target was the last one and the slice will go         // until the end of wikitext         if( endIdx < 0 ) {             //console.log("[getSectionWikitext] endIdx negative, setting to " + wikitext.length);             endIdx = wikitext.length;         }          //console.log("[getSectionWikitext] Slicing from " + startIdx + " to " + endIdx);         return wikitext.slice( startIdx, endIdx );     }      /**      * Converts a signature index to a string index into the given      * section wikitext. For example, if sigIdx is 1, then this function      * will return the index in sectionWikitext pointing to right      * after the second signature appearing in sectionWikitext.      *      * Returns -1 if we couldn't find anything.      */     function sigIdxToStrIdx( sectionWikitext, sigIdx ) {         console.log( "In sigIdxToStrIdx, sigIdx = " + sigIdx );          // There are certain regions that we skip while attaching links:         //         //  - Spans with the class delsort-notice         //  - Divs with the class xfd-relist (and other divs)         //         // So, we grab the corresponding wikitext regions with regexes,         // and store each region's start index in spanStartIndices, and         // each region's length in spanLengths. Then, whenever we find a         // signature with the right index, if it's included in one of         // these regions, we skip it and move on.         var spanStartIndices = [];         var spanLengths = [];         var DELSORT_SPAN_RE_TXT = /<small class="delsort-notice">(?:<small>.+?<\/small>|.)+?<\/small>/.source;         var XFD_RELIST_RE_TXT = /<div class="xfd_relist"[\s\S]+?<\/div>(\s*|<!--.+?-->)*/.source;         var STRUCK_RE_TXT = /<s>.+?<\/s>/.source;         var SKIP_REGION_RE = new RegExp("(" + DELSORT_SPAN_RE_TXT + ")|(" +             XFD_RELIST_RE_TXT + ")|(" +             STRUCK_RE_TXT + ")", "ig");         var skipRegionMatch;         do {             skipRegionMatch = SKIP_REGION_RE.exec( sectionWikitext );             if( skipRegionMatch ) {                 spanStartIndices.push( skipRegionMatch.index );                 spanLengths.push( skipRegionMatch[0].length );             }         } while( skipRegionMatch );         //console.log(spanStartIndices,spanLengths);          var dateFmtRgx = DATE_FMT_RGX[mw.config.get( "wgServer" )];         if( !dateFmtRgx ) {             throw new Error( "Error! I don't know the native date format used by the server '" + mw.config.get( "wgServer" ) + "'!" );         }         /*          * I apologize for making you have to read this regex.          * I made a summary, though:          *          *  - a wikilink, without a ]] inside it          *  - some text, without a link to userspace or user talk space          *  - a timestamp          *  - as an alternative to all of the above, an autosigned script          *    and a timestamp          *  - some comments/whitespace or some non-whitespace          *  - finally, the end of the line          *          * It's also localized.          */         var sigRgxSrc = "(?:" + /\[\[\s*(?:m:)?:?\s*/.source + "(" + userspcLinkRgx.both +                 /([^\]]|\](?!\]))*?/.source + ")" + /\]\]\)?/.source + "(" +                 /[^\[]|\[(?!\[)|\[\[/.source + "(?!" + userspcLinkRgx.both +                 "))*?" + DATE_FMT_RGX[mw.config.get( "wgServer" )] +                 /\s+\(UTC\)|class\s*=\s*"autosigned".+?\(UTC\)<\/small>/.source +                 ")" + /(\S*([ \t\f]|<!--.*?-->)*(?:\{\{.+?\}\})?(?!\S)|\s?\S+([ \t\f]|<!--.*?-->)*)$/.source;         var sigRgx = new RegExp( sigRgxSrc, "igm" );         var matchIdx = 0;         var match;         var matchIdxEnd;         var dstSpnIdx;          sigMatchLoop:         for( ; true ; matchIdx++ ) {             match = sigRgx.exec( sectionWikitext );             if( !match ) {                 console.error("[sigIdxToStrIdx] out of matches");                 return -1;             }             //console.log( "sig match (matchIdx = " + matchIdx + ") is >" + match[0] + "< (index = " + match.index + ")" );              matchIdxEnd = match.index + match[0].length;              // Validate that we're not inside a delsort span             for( dstSpnIdx = 0; dstSpnIdx < spanStartIndices.length; dstSpnIdx++ ) {                 //console.log(spanStartIndices[dstSpnIdx],match.index,                 //    matchIdxEnd, spanStartIndices[dstSpnIdx] +                 //        spanLengths[dstSpnIdx] );                 if( match.index > spanStartIndices[dstSpnIdx] &&                     ( matchIdxEnd <= spanStartIndices[dstSpnIdx] +                         spanLengths[dstSpnIdx] ) ) {                      // That wasn't really a match (as in, this match does not                     // correspond to any sig idx in the DOM), so we can't                     // increment matchIdx                     matchIdx--;                      continue sigMatchLoop;                 }             }              if( matchIdx === sigIdx ) {                 return match.index + match[0].length;             }         }     }      /**      * Inserts fullReply on the next sensible line after strIdx in      * sectionWikitext. indentLvl is the indentation level of the      * comment we're replying to.      *      * This function essentially takes the indentation level and      * position of the current comment, and looks for the first comment      * that's indented strictly less than the current one. Then, it      * puts the reply on the line right before that comment, and returns      * the modified section wikitext.      */     function insertTextAfterIdx( sectionWikitext, strIdx, indentLvl, fullReply ) {         //console.log( "[insertTextAfterIdx] indentLvl = " + indentLvl );          // strIdx should point to the end of a line         var counter = 0;         while( ( sectionWikitext[ strIdx ] !== "\n" ) && ( counter++ <= 50 ) ) strIdx++;          var slicedSecWikitext = sectionWikitext.slice( strIdx );         //console.log("slicedSecWikitext = >>" + slicedSecWikitext.slice(0,50) + "<<");         slicedSecWikitext = slicedSecWikitext.replace( /^\n/, "" );         var candidateLines = slicedSecWikitext.split( "\n" );         //console.log( "candidateLines =", candidateLines );          // number of the line in sectionWikitext that'll be right after reply         var replyLine = 0;          var INDENT_RE = /^[:*#]+/;         if( slicedSecWikitext.trim().length > 0 ) {             var currIndentation, currIndentationLvl, i;              // Now, loop through all the comments replying to that             // one and place our reply after the last one             for( i = 0; i < candidateLines.length; i++ ) {                 if( candidateLines[i].trim() === "" ) {                     continue;                 }                  // Detect indentation level of current line                 currIndentation = INDENT_RE.exec( candidateLines[i] );                 currIndentationLvl = currIndentation ? currIndentation[0].length : 0;                 //console.log(i + ">" + candidateLines[i] + "< => " + currIndentationLvl);                  if( currIndentationLvl <= indentLvl ) {                      // If it's an XfD, we might have found a relist                     // comment instead, so check for that                     if( xfdType && /<div class="xfd_relist"/.test( candidateLines[i] ) ) {                          // Our reply might go on the line above the xfd_relist line                         var potentialReplyLine = i;                          // Walk through the relist notice, line by line                         // After this loop, i will point to the line on which                         // the notice ends                         var NEW_COMMENTS_RE = /Please add new comments below this line/;                         while( !NEW_COMMENTS_RE.test( candidateLines[i] ) ) {                             i++;                         }                          // Relists are treated as if they're indented at level 1                         if( 1 <= indentLvl ) {                             replyLine = potentialReplyLine;                             break;                         }                     } else {                         //console.log( "cIL <= iL, breaking" );                         break;                     }                 } else {                     replyLine = i + 1;                 }             }             if( i === candidateLines.length ) {                 replyLine = i;             }         } else {              // In this case, we may be replying to the last comment in a section             replyLine = candidateLines.length;         }          // Walk backwards until non-empty line         while( replyLine >= 1 && candidateLines[replyLine - 1].trim() === "" ) replyLine--;          //console.log( "replyLine = " + replyLine );          // Splice into slicedSecWikitext         slicedSecWikitext = candidateLines             .slice( 0, replyLine )             .concat( [ fullReply ], candidateLines.slice( replyLine ) )             .join( "\n" );          // Remove extra newlines         if( /\n\n\n+$/.test( slicedSecWikitext ) ) {             slicedSecWikitext = slicedSecWikitext.trim() + "\n\n";         }          // We may need an additional newline if the two slices don't have any         var optionalNewline = ( !sectionWikitext.slice( 0, strIdx ).endsWith( "\n" ) &&                     !slicedSecWikitext.startsWith( "\n" ) ) ? "\n" : "";          // Splice into sectionWikitext         sectionWikitext = sectionWikitext.slice( 0, strIdx ) +             optionalNewline + slicedSecWikitext;          return sectionWikitext;     }      /**      * Using the text in #reply-dialog-field, add a reply to the      * current page. rplyToXfdNom is true if we're replying to an XfD nom,      * in which case we should use an asterisk instead of a colon.      * cmtAuthorDom is the username of the person who wrote the comment      * we're replying to, parsed from the DOM. revObj is the object returned      * by getWikitext for the page with the comment; findSectionResult is the      * object returned by findSection for the comment.      *      * Returns a Deferred that resolves/rejects when the reply succeeds/fails.      */     function doReply( indentation, header, sigIdx, cmtAuthorDom, rplyToXfdNom, revObj, findSectionResult ) {         console.log("TOP OF doReply",header,findSectionResult);         header = [ "" + findSectionResult.sectionLevel, findSectionResult.sectionName, findSectionResult.sectionDupeIdx ];         var deferred = $.Deferred();          var wikitext = revObj.content;          try {              // Generate reply in wikitext form             var reply = document.getElementById( "reply-dialog-field" ).value.trim();              // Add a signature if one isn't already there             if( !hasSig( reply ) ) {                 reply += " " + ( window.replyLinkSigPrefix ?                     window.replyLinkSigPrefix : "" ) + LITERAL_SIGNATURE;             }              var isUsingAutoIndentation = window.replyLinkAutoIndentation === "checkbox"                 ? ( !document.getElementById( "reply-link-option-auto-indent" ) ||                     document.getElementById( "reply-link-option-auto-indent" ).checked )                 : window.replyLinkAutoIndentation === "always";             if( isUsingAutoIndentation ) {                  var replyLines = reply.split( "\n" );                  // If we're outdenting, reset indentation and add the                 // outdent template. This requires that there be at least                 // one character of indentation.                 var outdentCheckbox = document.getElementById( "reply-link-option-outdent" );                 if( outdentCheckbox && outdentCheckbox.checked ) {                     replyLines[0] = "{" + "{od|" + indentation.slice( 0, -1 ) +                         "}}" + replyLines[0];                     indentation = "";                 }                  // Compose reply by adding indentation at the beginning of                 // each line (if not replying to an XfD nom) or {{pb}}'s                 // between lines (if replying to an XfD nom)                 var fullReply;                 if( rplyToXfdNom ) {                      // If there's a list in this reply, it's a bad idea to                     // use pb's, even though the markup'll probably be broken                     if( replyLines.some( function ( l ) { return l.substr( 0, 1 ) === "*"; } ) ) {                         fullReply = replyLines.map( function ( line ) {                             return indentation + "*" + line;                         } ).join( "\n" );                     } else {                         fullReply = indentation + "* " + replyLines.join( "{{pb}}" );                     }                 } else {                     fullReply = replyLines.map( function ( line ) {                         return indentation + ":" + line;                     } ).join( "\n" );                 }             } else {                 fullReply = reply;             }              // Prepare section metadata for getSectionWikitext call             console.log( "in doReply, header =", header );             var sectionHeader, sectionIdx;             if( header === null ) {                 sectionHeader = null, sectionIdx = -1;             } else {                 sectionHeader = header[1], sectionDupeIdx = header[2];             }              // Compatibility with User:Bility/copySectionLink             if( document.querySelector( "span.mw-headline a#sectiontitlecopy0" ) ) {                  // If copySectionLink is active, the paragraph symbol at                 // the end is a fake                 sectionHeader = sectionHeader.replace( /\s*¶$/, "" );             }              // Compatibility with the "auto-number headings" preference             if( document.querySelector( "span.mw-headline-number" ) ) {                 sectionHeader = sectionHeader.replace( /^\d+ /, "" );             }              var sectionWikitext = getSectionWikitext( wikitext, sectionHeader, sectionDupeIdx );             var oldSectionWikitext = sectionWikitext; // We'll String.replace old w/ new              // Now, obtain the index of the end of the comment             var strIdx = sigIdxToStrIdx( sectionWikitext, sigIdx );              // Check for a non-negative strIdx             if( strIdx < 0 ) {                 throw( "Negative strIdx (signature not found in wikitext)" );             }              // Determine the user who wrote the comment, for             // edit-summary and sanity-check purposes             var userRgx = new RegExp( /\[\[\s*(?:m:)?:?\s*/.source + userspcLinkRgx.both + /\s*(.+?)(?:\/.+?)?(?:#.+?)?\s*(?:\|.+?)?\]\]/.source, "ig" );             var userMatches = processCharEntitiesWikitext( sectionWikitext.slice( 0, strIdx ) ).match( userRgx );             var cmtAuthorWktxt = userRgx.exec(                     userMatches[userMatches.length - 1] )[1];              if( cmtAuthorWktxt === "DoNotArchiveUntil" ) {                 userRgx.lastIndex = 0;                 cmtAuthorWktxt = userRgx.exec( userMatches[userMatches.length - 2] )[1];             }              // Normalize case, because that's what happens during             // wikitext-to-HTML processing; also underscores to spaces             function sanitizeUsername( u ) {                 u = u.charAt( 0 ).toUpperCase() + u.substr( 1 );                 return u.replace( /_/g, " " );             }             cmtAuthorWktxt = sanitizeUsername( cmtAuthorWktxt );             cmtAuthorDom = sanitizeUsername( cmtAuthorDom );              // Do a sanity check: is the sig username the same as the             // DOM one?  We attempt to check sigRedirectMapping in case             // the naive check fails             if( cmtAuthorWktxt !== cmtAuthorDom &&                     processCharEntitiesWikitext( cmtAuthorWktxt ) !== cmtAuthorDom &&                     sigRedirectMapping[ cmtAuthorWktxt ] !== cmtAuthorDom ) {                 throw new Error( "Sanity check on sig username failed! Found " +                     cmtAuthorWktxt + " but expected " + cmtAuthorDom +                     " (wikitext vs DOM)" );             }              // Actually insert our reply into the section wikitext             sectionWikitext = insertTextAfterIdx( sectionWikitext, strIdx,                     indentation.length, fullReply );              // Also, if the user wanted the edit request to be answered,             // do that             var editReqCheckbox = document.getElementById(  "reply-link-option-edit-req" );             var markedEditReq = false;             if( editReqCheckbox && editReqCheckbox.checked ) {                 sectionWikitext = markEditReqAnswered( sectionWikitext );                 markedEditReq = true;             }              // If the user preferences indicate a dry run, print what the             // wikitext would have been post-edit and bail out             var dryRunCheckbox = document.getElementById( "reply-link-option-dry-run" );             if( window.replyLinkDryRun === "always" || ( dryRunCheckbox && dryRunCheckbox.checked ) ) {                 console.log( "~~~~~~ DRY RUN CONCLUDED ~~~~~~" );                 console.log( sectionWikitext );                 setStatus( "Check the console for the dry-run results." );                 document.querySelector( "#reply-link-buttons button" ).disabled = false;                 deferred.resolve();                 return deferred;             }              var newWikitext = wikitext.replace( oldSectionWikitext,                     sectionWikitext );              // Build summary             var defaultSummmary = mw.msg( "rl-replying-to" ) +                 ( rplyToXfdNom ? xfdType + " nomination by " : "" ) +                 cmtAuthorWktxt +                 ( markedEditReq ? " and marking edit request as answered" : "" );             var customSummaryField = document.getElementById( "reply-link-summary" );             var summaryCore = defaultSummmary;             if( window.replyLinkCustomSummary && customSummaryField.value ) {                 summaryCore = customSummaryField.value.trim();             }             var summary = "/* " + sectionHeader + " */ " + summaryCore + mw.msg( "rl-advert" );              // Send another request, this time to actually edit the             // page             api.postWithToken( "csrf", {                 action: "edit",                 title: findSectionResult.page,                 summary: summary,                 text: newWikitext,                 basetimestamp: revObj.timestamp             } ).done ( function ( data ) {                  // We put this function on the window object because we                 // give the user a "reload" link, and it'll trigger the function                 window.replyLinkReload = function () {                     window.location.hash = sectionHeader.replace( / /g, "_" );                     if( findSectionResult.nearbyMwId ) {                         document.cookie = "parsoid_jump=" + findSectionResult.nearbyMwId;                     }                     window.location.reload( true );                 };                 if ( data && data.edit && data.edit.result && data.edit.result == "Success" ) {                     var needPurge = findSectionResult.page !== currentPageName;                      function finishReply( _ ) {                         var reloadHtml = window.replyLinkAutoReload ? mw.msg( "rl-reloading" )                             : "<a href='javascript:window.replyLinkReload()' class='reply-link-reload'>" + mw.msg( "rl-reload" ) + "</a>";                         setStatus( mw.msg( "rl-saved" ) + " (" + reloadHtml + ")" );                          // Required to permit reload to happen, checked in onbeforeunload                         replyWasSaved = true;                          if( window.replyLinkAutoReload ) {                             window.replyLinkReload();                         }                          deferred.resolve();                     }                      if( needPurge ) {                         setStatus( "Reply saved! Purging..." );                         api.post( { action: "purge", titles: currentPageName } ).done( finishReply );                     } else {                         finishReply();                     }                 } else {                     if( data && data.edit && data.edit.spamblacklist ) {                         setStatus( "Error! Your post contained a link on the <a href=" +                             "\"https://en.wikipedia.org/wiki/Wikipedia:Spam_blacklist\"" +                             ">spam blacklist</a>. Remove the link(s) to: " +                             data.edit.spamblacklist.split( "|" ).join( ", " ) + " to allow saving." );                         document.querySelector( "#reply-link-buttons button" ).disabled = false;                     } else {                         setStatus( "While saving, the edit query returned an error." +                             " Check the browser console for more information." );                     }                      deferred.reject();                 }                 //console.log(data);                 document.getElementById( "reply-dialog-field" ).style["background-image"] = "";             } ).fail ( function( code, result ) {                 setStatus( "While replying, the edit failed." );                 console.log(code);                 console.log(result);                 deferred.reject();             } );         } catch ( e ) {             setStatusError( e );             deferred.reject();         }          return deferred;     }       function handleWrapperClick ( linkLabel, parent, rplyToXfdNom ) {         return function ( evt ) {             $.when( mw.messages.exists( INT_MSG_KEYS[0] ) ? 1 :                     api.loadMessages( INT_MSG_KEYS ) ).then( function () {                 var newLink = this;                 var newLinkWrapper = this.parentNode;                  if( !userspcLinkRgx ) {                     buildUserspcLinkRgx();                 }                  // Remove previous panel                 var prevPanel = document.getElementById( "reply-link-panel" );                 if( prevPanel ) {                     prevPanel.remove();                 }                  // Reset previous cancel links                 var cancelLinks = iterableToList( document.querySelectorAll(                             ".reply-link-wrapper a" ) );                 cancelLinks.forEach( function ( el ) {                     if( el != newLink ) el.textContent = el.dataset.originalLabel;                 } );                  // Handle disable action                 if( newLink.textContent === linkLabel ) {                      // Disable this link                     newLink.textContent = mw.msg( "rl-cancel" ) + linkLabel;                 } else {                      // We've already cancelled the reply                     newLink.textContent = linkLabel;                     evt.preventDefault();                     return false;                 }                  // Figure out the username of the author                 // of the comment we're replying to                 var cmtAuthorAndLink = getCommentAuthor( newLinkWrapper );                  try {                     var cmtAuthor = cmtAuthorAndLink.username,                         cmtLink = cmtAuthorAndLink.link;                 } catch ( e ) {                     setStatusError( e );                 }                  // Create panel                 var panelEl = document.createElement( "div" );                 panelEl.id = "reply-link-panel";                 panelEl.innerHTML = "<textarea id='reply-dialog-field' class='mw-ui-input'" +                     " placeholder='" + mw.msg( "rl-placeholder" ) + "'></textarea>" +                     ( window.replyLinkCustomSummary ? "<label for='reply-link-summary'>Summary: </label>" +                         "<input id='reply-link-summary' class='mw-ui-input' placeholder='Edit summary' " +                         "value='Replying to " + cmtAuthor.replace( /'/g, "&#39;" ) + "'/><br />" : "" ) +                     "<table style='border-collapse:collapse'><tr><td id='reply-link-buttons' style='width: " +                     ( window.replyLinkPreloadPing === "button" ? "325" : "255" ) + "px'>" +                     "<button id='reply-dialog-button' class='mw-ui-button mw-ui-progressive'>" + mw.msg( "rl-reply" ) + "</button> " +                     "<button id='reply-link-preview-button' class='mw-ui-button'>" + mw.msg( "rl-preview" ) + "</button>" +                     ( window.replyLinkPreloadPing === "button" ?                         " <button id='reply-link-ping-button' class='mw-ui-button'>Ping</button>" : "" ) +                     "<button id='reply-link-cancel-button' class='mw-ui-button mw-ui-quiet mw-ui-destructive'>" + mw.msg( "rl-cancel-button" ) + "</button></td>" +                     "<td id='reply-dialog-status'></span><div style='clear:left'></td></tr></table>" +                     "<div id='reply-link-options' class='gone-on-empty' style='margin-top: 0.5em'></div>" +                     "<div id='reply-link-preview' class='gone-on-empty' style='border: thin dashed gray; padding: 0.5em; margin-top: 0.5em'></div>";                 parent.insertBefore( panelEl, newLinkWrapper.nextSibling );                 var replyDialogField = document.getElementById( "reply-dialog-field" );                 replyDialogField.style = "padding: 0.625em; min-height: 10em; margin-bottom: 0.75em; line-height: 1.3";                 if( window.replyLinkPreloadPing === "always" &&                         cmtAuthor &&                         cmtAuthor !== mw.config.get( "wgUserName" ) &&                         !/(\d+.){3}\d+/.test( cmtAuthor ) ) {                     replyDialogField.value = window.replyLinkPreloadPingTpl.replace( "##", cmtAuthor );                 }                  // Fill up #reply-link-options                 function newOption( id, text, defaultOn ) {                     var newCheckbox = document.createElement( "input" );                     newCheckbox.type = "checkbox";                     newCheckbox.id = id;                     if( defaultOn ) {                         newCheckbox.checked = true;                     }                     var newLabel = document.createElement( "label" );                     newLabel.htmlFor = id;                     newLabel.appendChild( document.createTextNode( text ) );                     document.getElementById( "reply-link-options" ).appendChild( newCheckbox );                     document.getElementById( "reply-link-options" ).appendChild( newLabel );                 }                  // Fetch metadata about this specific comment                 var ourMetadata = metadata[this.id];                  // If the dry-run option is "checkbox", add an option to make it                 // a dry run                 if( window.replyLinkDryRun === "checkbox" ) {                     newOption( "reply-link-option-dry-run", "Don't actually edit?", true );                 }                  // If the current section header text indicates an edit request,                 // offer to mark it as answered                 if( ourMetadata[1] && EDIT_REQ_REGEX.test( ourMetadata[1][1] ) ) {                     newOption( "reply-link-option-edit-req", "Mark edit request as answered?", false );                 }                  // If the previous comment was indented by OUTDENT_THRESH,                 // offer to outdent                 if( ourMetadata[0].length >= OUTDENT_THRESH ) {                     newOption( "reply-link-option-outdent", "Outdent?", false );                 }                  if( window.replyLinkAutoIndentation === "checkbox" ) {                     newOption( "reply-link-option-auto-indent", mw.msg( "rl-auto-indent" ), true );                 }                  /* Commented out because I could never get it to work                 // Autofill with a recommendation if we're replying to a nom                 if( rplyToXfdNom ) {                     replyDialogField.value = "'''Comment'''";                      // Highlight the "Comment" part so the user can change it                     var range = document.createRange();                     range.selectNodeContents( replyDialogField );                     //range.setStart( replyDialogField, 3 ); // start of "Comment"                     //range.setEnd( replyDialogField, 10 ); // end of "Comment"                     var sel = window.getSelection();                     sel.removeAllRanges();                     sel.addRange( range );                 }*/                  // Close handler                 window.onbeforeunload = function ( e ) {                     if( !replyWasSaved &&                             document.getElementById( "reply-dialog-field" ) &&                             document.getElementById( "reply-dialog-field" ).value ) {                         var txt = mw.msg( "rl-started-reply" );                         e.returnValue = txt;                         return txt;                     }                 };                  // Called by the "Reply" button, Ctrl-Enter in the text area, and                 // Enter/Ctrl-Enter in the summary field                 function startReply() {                      // Change UI to make it clear we're performing an operation                     document.getElementById( "reply-dialog-field" ).style["background-image"] =                         "url(" + window.replyLinkPendingImageUrl + ")";                     document.querySelector( "#reply-link-buttons button" ).disabled = true;                     setStatus( mw.msg( "rl-loading" ) );                      var parsoidUrl = PARSOID_ENDPOINT + encodeURIComponent( currentPageName ) +                             "/" + mw.config.get( "wgCurRevisionId" ),                         findSectionResultPromise = $.get( parsoidUrl )                             .then( function ( parsoidDomString ) {                                 return findSection( currentPageName, parsoidDomString, cmtLink );                         },console.error );                      var revObjPromise = findSectionResultPromise.then( function ( findSectionResult ) {                         console.log( "findSectionResult ", findSectionResult );                         return getWikitext( findSectionResult.page );                     },console.error );                      $.when( findSectionResultPromise, revObjPromise ).then( function ( findSectionResult, revObj ) {                         // ourMetadata contains data in the format:                         // [indentation, header, sigIdx]                         doReply( ourMetadata[0], ourMetadata[1], ourMetadata[2],                             cmtAuthor, rplyToXfdNom, revObj, findSectionResult );                     }, function (e) { setStatusError(new Error(e))} );                 }                  // Event listener for the "Reply" button                 document.getElementById( "reply-dialog-button" )                     .addEventListener( "click", startReply );                  // Event listener for the text area                 document.getElementById( "reply-dialog-field" )                     .addEventListener( "keydown", function ( e ) {                         if( e.ctrlKey && ( e.keyCode == 10 || e.keyCode == 13 ) ) {                             startReply();                         }                     } );                  // Event listener for the "Preview" button                 document.getElementById( "reply-link-preview-button" )                     .addEventListener( "click", function () {                         var reply = document.getElementById( "reply-dialog-field" ).value.trim();                          // Add a signature if one isn't already there                         if( !hasSig( reply ) ) {                             reply += " " + ( window.replyLinkSigPrefix ?                                 window.replyLinkSigPrefix : "" ) + LITERAL_SIGNATURE;                         }                          var sanitizedCode = encodeURIComponent( reply );                         $.post( "https:" + mw.config.get( "wgServer" ) +                             "/w/api.php?action=parse&format=json&title=" + currentPageName + "&text=" + sanitizedCode                                 + "&pst=1",                             function ( res ) {                                 if ( !res || !res.parse || !res.parse.text ) return console.log( "Preview failed" );                                 document.getElementById( "reply-link-preview" ).innerHTML = res.parse.text['*'];                                 // Add target="_blank" to links to make them open in a new tab by default                                 var links = document.querySelectorAll( "#reply-link-preview a" );                                 for( var i = 0, n = links.length; i < n; i++ ) {                                     links[i].setAttribute( "target", "_blank" );                                 }                             } );                     } );                  if( window.replyLinkPreloadPing === "button" ) {                     document.getElementById( "reply-link-ping-button" )                         .addEventListener( "click", function () {                             replyDialogField.value = window.replyLinkPreloadPingTpl                                 .replace( "##", cmtAuthor ) + replyDialogField.value;                         } );                 }                  // Event listener for the "Cancel" button                 document.getElementById( "reply-link-cancel-button" )                     .addEventListener( "click", function () {                         newLink.textContent = linkLabel;                         panelEl.remove();                     } );                  // Event listeners for the custom edit summary field                 if( window.replyLinkCustomSummary ) {                     document.getElementById( "reply-link-summary" )                         .addEventListener( "keydown", function ( e ) {                             if( e.keyCode == 10 || e.keyCode == 13 ) {                                 startReply();                             }                         } );                 }                  if( window.replyLinkTestInstantReply ) {                     startReply();                 }             }.bind( this ) );              // Cancel default event handler             evt.preventDefault();             return false;         }     }      /**      * Adds a "(reply)" link after the provided text node, giving it      * the provided element id. anyIndentation is true if there's any      * indentation (i.e. indentation string is not the empty string)      */     function attachLinkAfterNode( node, preferredId, anyIndentation ) {          // Choose a parent node - walk up tree until we're under a dd, li,         // p, or div. This walk is a bit unsafe, but this function should         // only get called in a place where the walk will succeed.         var parent = node;         do {             parent = parent.parentNode;         } while( !( /^(p|dd|li|div|td)$/.test( parent.tagName.toLowerCase() ) ) );          // Determine whether we're replying to an XfD nom         var rplyToXfdNom = false;         if( xfdType === "AfD" || xfdType === "MfD" ) {              // If the comment is non-indented, we are replying to a nom             rplyToXfdNom = !anyIndentation;         } else if( xfdType === "TfD" || xfdType === "FfD" ) {              // If the sibling before the previous sibling of this node             // is a h4, then this is a nom             rplyToXfdNom = parent.previousElementSibling &&                 parent.previousElementSibling.previousElementSibling &&                 parent.previousElementSibling.previousElementSibling.nodeType === 1 &&                 parent.previousElementSibling.previousElementSibling.tagName.toLowerCase() === "h4";         } else if( xfdType === "CfD" ) {              // If our grandparent is a dl and our grandparent's previous             // sibling is a h4, then this is a nom             rplyToXfdNom = parent.parentNode.tagName.toLowerCase() === "dl" &&                 parent.parentNode.previousElementSibling.nodeType === 1 &&                 parent.parentNode.previousElementSibling.tagName.toLowerCase() === "h4";         }          // Choose link label: if we're replying to an XfD, customize it         var linkLabel = mw.msg( "rl-reply-label" ) + ( rplyToXfdNom ? mw.msg( "rl-to-label" ) + xfdType : "" );          // Construct new link         var newLinkWrapper = document.createElement( "span" );         newLinkWrapper.className = "reply-link-wrapper";         var newLink = document.createElement( "a" );         newLink.href = "#";         newLink.id = preferredId;         newLink.dataset.originalLabel = linkLabel;         newLink.appendChild( document.createTextNode( linkLabel ) );         newLink.addEventListener( "click", handleWrapperClick( linkLabel, parent, rplyToXfdNom ) );         newLinkWrapper.appendChild( document.createTextNode( " (" ) );         newLinkWrapper.appendChild( newLink );         newLinkWrapper.appendChild( document.createTextNode( ")" ) );          // Insert new link into DOM         parent.insertBefore( newLinkWrapper, node.nextSibling );     }      /**      * Uses attachLinkAfterTextNode to add a reply link after every      * timestamp on the page.      */     function attachLinks () {         var mainContent = findMainContentEl();         if( !mainContent ) {             console.error( "No main content element found; exiting." );             return;         }          var contentEls = mainContent.children;          // Find the index of the first header in contentEls         var headerIndex = 0;         for( headerIndex = 0; headerIndex < contentEls.length; headerIndex++ ) {             if( contentEls[ headerIndex ].tagName.toLowerCase().startsWith( "h" ) ) break;         }          // If we didn't find any headers at all, that's a problem and we         // should bail         if( mainContent.querySelector( "div.hover-edit-section" ) ) {             headerIndex = 0;         } else if( headerIndex === contentEls.length ) {             console.error( "Didn't find any headers - hit end of loop!" );             return;         }          // We also should include the first header         if( headerIndex > 0 ) {             headerIndex--;         }          // Each element is a 2-element list of [level, node]         var parseStack = iterableToList( contentEls ).slice( headerIndex );         parseStack.reverse();         parseStack = parseStack.map( function ( el ) { return [ "", el ]; } );          // Main parse loop         var node;         var currIndentation; // A string of symbols, like ":*::"         var newIndentSymbol;         var stackEl; // current element from the parse stack         var idNum = 0; // used to make id's for the links         var linkId = ""; // will be the element id for this link         while( parseStack.length ) {             stackEl = parseStack.pop();             node = stackEl[1];             currIndentation = stackEl[0];              // Compatibility with "Comments in Local Time"             var isLocalCommentsSpan = node.nodeType === 1 &&                 "span" === node.tagName.toLowerCase() &&                 node.className.includes( "localcomments" );              var isSmall = node.nodeType === 1 && (                     node.tagName.toLowerCase() === "small" ||                     ( node.tagName.toLowerCase() === "span" &&                     node.style && node.style.getPropertyValue( "font-size" ) === "85%" ) );              // Small nodes are okay, unless they're delsort notices             var isOkSmallNode = isSmall &&                 !node.className.includes( "delsort-notice" );              if( ( node.nodeType === 3 ) ||                     isOkSmallNode ||                     isLocalCommentsSpan )  {                  // If the current node has a timestamp, attach a link to it                 // Also, no links after timestamps, because it's just like                 // having normal text afterwards, which is rejected (because                 // that means someone put a timestamp in the middle of a                 // paragraph)                 var hasLinkAfterwardsNotInBlockEl = node.nextElementSibling &&                     ( node.nextElementSibling.tagName.toLowerCase() === "a" ||                         ( node.nextElementSibling.tagName.match( /^(span|small)$/i ) &&                             node.nextElementSibling.querySelector( "a" ) ) );                 if( TIMESTAMP_REGEX.test( node.textContent ) &&                         ( node.previousSibling || isSmall ) &&                         !hasLinkAfterwardsNotInBlockEl ) {                     linkId = "reply-link-" + idNum;                     attachLinkAfterNode( node, linkId, !!currIndentation );                     idNum++;                      // Update global metadata dictionary                     metadata[linkId] = currIndentation;                 }             } else if( node.nodeType === 1 &&                     /^(div|p|dl|dd|ul|li|span|ol|table|tbody|tr|td)$/.test( node.tagName.toLowerCase() ) ) {                 switch( node.tagName.toLowerCase() ) {                     case "dl": newIndentSymbol = ":"; break;                     case "ul": newIndentSymbol = "*"; break;                     case "ol": newIndentSymbol = "#"; break;                     case "div":                         if( node.className.includes( "xfd_relist" ) ) {                             continue;                         }                         break;                     default: newIndentSymbol = ""; break;                 }                  var childNodes = node.childNodes;                 for( let i = 0, numNodes = childNodes.length; i < numNodes; i++ ) {                     parseStack.push( [ currIndentation + newIndentSymbol,                         childNodes[i] ] );                 }             }         }          // This loop adds two entries in the metadata dictionary:         // the header data, and the sigIdx values         var sigIdxEls = iterableToList( mainContent.querySelectorAll(                 HEADER_SELECTOR + ",span.reply-link-wrapper a" ) );         var currSigIdx = 0, j, numSigIdxEls, currHeaderEl, currHeaderData;         var headerIdx = 0; // index of the current header         var headerLvl = 0; // level of the current header         for( j = 0, numSigIdxEls = sigIdxEls.length; j < numSigIdxEls; j++ ) {             var headerTagNameMatch = /^h(\d+)$/.exec(                 sigIdxEls[j].tagName.toLowerCase() );             if( headerTagNameMatch ) {                 currHeaderEl = sigIdxEls[j];                  // Test to make sure we're not in the table of contents                 if( currHeaderEl.parentNode.className === "toctitle" ) {                     continue;                 }                  // Reset signature counter                 currSigIdx = 0;                  // Dig down one level for the header text because                 // MW buries the text in a span inside the header                 var headlineEl = null;                 if( currHeaderEl.childNodes[0].className &&                     currHeaderEl.childNodes[0].className.includes( "mw-headline" ) ) {                     headlineEl = currHeaderEl.childNodes[0];                 } else {                     for( var i = 0; i < currHeaderEl.childNodes.length; i++ ) {                         if( currHeaderEl.childNodes[i].className &&                                 currHeaderEl.childNodes[i].className.includes( "mw-headline" ) ) {                             headlineEl = currHeaderEl.childNodes[i];                             break;                         }                     }                 }                  var headerName = null;                 if( headlineEl ) {                     headerName = headlineEl.textContent;                 }                  if( headerName === null ) {                     console.error( currHeaderEl );                     throw "Couldn't parse a header element!";                 }                  headerLvl = headerTagNameMatch[1];                 currHeaderData = [ headerLvl, headerName, headerIdx ];                 headerIdx++;             } else {                  // Save all the metadata for this link                 currIndentation = metadata[ sigIdxEls[j].id ];                 metadata[ sigIdxEls[j].id ] = [ currIndentation,                     currHeaderData ? currHeaderData.slice(0) : null,                     currSigIdx ];                 currSigIdx++;             }         }         //console.log(metadata);          // Disable links inside hatnotes, archived discussions         var badRegionsSelector = [             "div.archived",             "div.resolved",             "table"             ].map( function ( s ) { return s + " .reply-link-wrapper" } ).join( "," );         var insideArchived = mainContent.querySelectorAll( badRegionsSelector );         for( var i = 0; i < insideArchived.length; i++ ) {             insideArchived[i].parentNode.removeChild( insideArchived[i] );         }     }      function runTestMode() {          // We never want to make actual edits         window.replyLinkDryRun = "always";          // Simulate having a panel open         $( "#mw-content-text" )             .append( $( "<div>" )                 .append( $( "<textarea>" ).attr( "id", "reply-dialog-field" ).val( "hi" ) )                 .append( $( "<div>" ).attr( "id", "reply-link-buttons" )                     .append( $( "<button> " ) ) ) );          mw.util.addCSS( ".reply-link-wrapper { background-color: orange; }" );          // Fetch content, Parsoid DOM, etc         var parsoidUrl = PARSOID_ENDPOINT + encodeURIComponent( currentPageName );         $.when(             $.get( parsoidUrl ),             api.loadMessages( INT_MSG_KEYS )         ).then( function ( parsoidDomString, _ ) {             buildUserspcLinkRgx();              // Statistics variables             var successes = 0, failures = 0;              // Run one test on a wrapper link             function runOneTestOn( wrapper ) {                 try {                     var cmtAuthorAndLink = getCommentAuthor( wrapper ),                         cmtAuthor = cmtAuthorAndLink.username,                         cmtLink = cmtAuthorAndLink.link;                     var ourMetadata = metadata[ wrapper.children[0].id ];                     findSection( currentPageName, parsoidDomString, cmtLink ).then( function ( findSectionResult ) {                         var revObjPromise = getWikitext( findSectionResult.page, /* useCaching */ true );                         $.when( findSectionResult, revObjPromise ).then( function ( findSectionResult, revObj ) {                                 doReply( ourMetadata[0], ourMetadata[1], ourMetadata[2],                                         cmtAuthor, false, revObj, findSectionResult ).done( function () {                                             wrapper.style.background = "green";                                             successes++;                                         } ).fail( function () {                                             wrapper.style.background = "red";                                             failures++;                                         } );                         }, function ( e ) {                             wrapper.style.background = "red";                             failures++;                         } );                     } );                 } catch ( e ) {                     console.error( e );                     wrapper.style.background = "red";                     failures++;                 }             }              var wrappers = Array.from( document.querySelectorAll( ".reply-link-wrapper" ) );             function runOneTest() {                 var wrapper = wrappers.shift();                 if( wrapper ) {                     runOneTestOn( wrapper );                     setTimeout( runOneTest, 750 );                 } else {                     var results = successes + " successes, " + failures + " failures";                     $( "#mw-content-text" ).prepend( results ).append( results );                 }             }             //console.log = function() {};             setTimeout( runOneTest, 0 );         } );     }      function onReady() {         var lang_code = mw.config.get( "wgUserLanguage" )         // Replace default English interface by translation if available         var interface_messages = $.extend( {}, i18n.en, i18n[ lang_code.split('-')[0] ], i18n[ lang_code ] );         // Define interface messages         mw.messages.set( interface_messages );          // Exit if history page or edit page         if( mw.config.get( "wgAction" ) === "history" ) return;         if( document.getElementById( "editform" ) ) return;          api = new mw.Api();          mw.util.addCSS(             "#reply-link-panel { padding: 1em; margin-left: 1.6em; "+               "max-width: 1200px; width: 66%; margin-top: 0.5em; }"+             ".gone-on-empty:empty { display: none; }"         );          // Pre-load interface messages; we will check again when a (reply)         // link is clicked         api.loadMessages( INT_MSG_KEYS );          // Initialize the xfdType global variable, which must happen         // before the call to attachLinks         currentPageName = mw.config.get( "wgPageName" );         xfdType = "";         if( mw.config.get( "wgNamespaceNumber" ) === 4) {             if( currentPageName.startsWith( "Wikipedia:Articles_for_deletion/" ) ) {                 xfdType = "AfD";             } else if( currentPageName.startsWith( "Wikipedia:Miscellany_for_deletion/" ) ) {                 xfdType = "MfD";             } else if( currentPageName.startsWith( "Wikipedia:Templates_for_discussion/Log/" ) ) {                 xfdType = "TfD";             } else if( currentPageName.startsWith( "Wikipedia:Categories_for_discussion/Log/" ) ) {                 xfdType = "CfD";             } else if( currentPageName.startsWith( "Wikipedia:Files_for_discussion/" ) ) {                 xfdType = "FfD";             }         }          // Default values for some preferences         if( window.replyLinkAutoReload === undefined ) window.replyLinkAutoReload = true;         if( window.replyLinkDryRun === undefined ) window.replyLinkDryRun = "never";         if( window.replyLinkPreloadPing === undefined ) window.replyLinkPreloadPing = "always";         if( window.replyLinkPreloadPingTpl === undefined ) window.replyLinkPreloadPingTpl = "{{u|##}}, ";         if( window.replyLinkCustomSummary === undefined ) window.replyLinkCustomSummary = false;         if( window.replyLinkTestMode === undefined ) window.replyLinkTestMode = false;         if( window.replyLinkTestInstantReply === undefined) window.replyLinkTestInstantReply = false;         if( window.replyLinkAutoIndentation === undefined ) window.replyLinkAutoIndentation = "checkbox";          // Insert "reply" links into DOM         attachLinks();          // If test mode is enabled, create a link for that         if( window.replyLinkTestMode ) {             mw.util.addPortletLink( "p-cactions", "#", "reply-link test mode", "pt-reply-link-test" )                 .addEventListener( "click", runTestMode );         }          // This large string creats the "pending" texture         window.replyLinkPendingImageUrl = "";      }      mw.loader.load( "mediawiki.ui.input", "text/css" );     mw.loader.using( [ "mediawiki.util", "mediawiki.api" ] ).then( function () {         mw.hook( "wikipage.content" ).add( onReady );     } );      $.getScript('https://en.wikipedia.org/w/index.php?title=User:Enterprisey/parsoid-jump.js&action=raw&ctype=text%2Fjavascript');      // Return functions for testing     return {         "iterableToList": iterableToList,         "sigIdxToStrIdx": sigIdxToStrIdx,         "insertTextAfterIdx": insertTextAfterIdx,         "wikitextToTextContent": wikitextToTextContent     }; }  // Export functions for testing if( typeof module === typeof {} ) {     module.exports = { "loadReplyLink": loadReplyLink }; }  // If we're in the right environment, load the script if( jQuery !== undefined && mediaWiki !== undefined ) {     var currNamespace = mw.config.get( "wgNamespaceNumber" );      // Also enable on T:TDYK and its subpages     var ttdykPage = mw.config.get( "wgPageName" ).indexOf( "Template:Did_you_know_nominations" ) === 0;      // Normal "read" view and not a diff view     var normalView = mw.config.get( "wgIsArticle" ) &&             !mw.config.get( "wgDiffOldId" );      if ( normalView && ( currNamespace % 2 === 1 || currNamespace === 4 || ttdykPage ) ) {         loadReplyLink( jQuery, mediaWiki );     } } //</nowiki>