User:Eejit43/scripts/redirect-helper.js
Appearance
< User:Eejit43 | scripts
Code that you insert on this page could contain malicious content capable of compromising your account. If you import a script from another page with "importScript", "mw.loader.load", "iusc", or "lusc", take note that this causes you to dynamically load a remote script, which could be changed by others. Editors are responsible for all edits and actions they perform, including by scripts. User scripts are not centrally supported and may malfunction or become inoperable due to software changes. A guide to help you find broken scripts is available. If you are unsure whether code you are adding to this page is safe, you can ask at the appropriate village pump. This code will be executed when previewing this page. |
![]() | This user script seems to have a documentation page at User:Eejit43/scripts/redirect-helper. |
// <nowiki>
// Note: This script was compiled and minified from TypeScript. For a more readable version, see https://github.com/Eejit43/wikipedia-scripts/blob/main/scripts/redirect-helper.ts
"use strict";mw.loader.using(["mediawiki.util","oojs-ui-core","oojs-ui-widgets","oojs-ui-windows","oojs-ui.styles.icons-content","oojs-ui.styles.icons-editing-core"],()=>{class O extends OO.ui.TextInputWidget{constructor(t,e){super(t);this.api=new mw.Api;this.getLookupRequest=()=>{const t=this.getValue(),e=$.Deferred();if(!t)e.resolve([]);else if(t.includes("#")){const a=t.split("#")[0];this.api.get({action:"parse",page:a,prop:"sections",redirects:!0}).catch(()=>null).then(n=>{if(n){const s=n.parse.sections.filter(r=>r.line.toLowerCase().replaceAll(/<\/?i>/g,"").startsWith(t.split("#")[1].toLowerCase()));e.resolve(s.map(r=>({data:`${n.parse.title}#${r.line.replaceAll(/<\/?i>/g,"")}`,label:`${n.parse.title}#${r.line.replaceAll(/<\/?i>/g,"")}`})))}else e.resolve([])})}else{const a=mw.Title.newFromText(t);this.api.get({action:"query",formatversion:"2",gaplimit:20,gapnamespace:a?.getNamespaceId()??0,gapprefix:a?.getMainText()??t,generator:"allpages",prop:["info","pageprops"]}).catch(()=>null).then(n=>{n?e.resolve(n.query?.pages?n.query.pages.filter(s=>s.title!==this.pageTitleParsed.toString()).map(s=>({data:s.title,label:new OO.ui.HtmlSnippet(`${s.title}${s.pageprops&&"disambiguation"in s.pageprops?" <i>(disambiguation)</i>":""}${"redirect"in s?" <i>(redirect)</i>":""}`)})):[]):e.resolve([])})}return e.promise({abort(){}})};this.getLookupCacheDataFromResponse=t=>t??[];this.getLookupMenuOptionsFromData=t=>t.map(({data:e,label:a})=>new OO.ui.MenuOptionWidget({data:e,label:a}));OO.ui.mixin.LookupElement.call(this,t),this.pageTitleParsed=e}}Object.assign(O.prototype,OO.ui.mixin.LookupElement.prototype);class P extends OO.ui.TextInputWidget{constructor(t){super(t);this.api=new mw.Api;this.getLookupRequest=()=>{const t=this.getValue(),e=$.Deferred();t||e.resolve([]);const a=mw.Title.newFromText(t);return this.api.get({action:"query",formatversion:"2",gaplimit:20,gapnamespace:14,gapprefix:a?.getMainText()??t,generator:"allpages",prop:"categories"}).catch(()=>null).then(n=>{if(n?.query?.pages){const s=n.query.pages.filter(r=>!r.categories?.some(o=>o.title==="Category:Wikipedia soft redirected categories")).map(r=>{const o=r.title.split(":")[1];return{data:o,label:o}});this.emit("showing-values",s),e.resolve(s)}else e.resolve([])}),e.promise({abort(){}})};this.getLookupCacheDataFromResponse=t=>t??[];this.getLookupMenuOptionsFromData=t=>t.map(({data:e,label:a})=>new OO.ui.MenuOptionWidget({data:e,label:a}));OO.ui.mixin.LookupElement.call(this,t)}}Object.assign(P.prototype,OO.ui.mixin.LookupElement.prototype);class d extends OO.ui.ProcessDialog{constructor(t,e){super(t);this.api=new mw.Api;this.getSetupProcess=()=>d.super.prototype.getSetupProcess.call(this).next(()=>this.api.post({action:"parse",formatversion:"2",contentmodel:"wikitext",prop:["text","categorieshtml"],title:this.pageTitleParsed.getPrefixedDb(),text:this.getData()}).then(t=>{const e=t.parse.text,a=t.parse.categorieshtml,n=new OO.ui.PanelLayout({padded:!0,expanded:!1});n.$element.append(e,a),this.$body.append(n.$element)}));this.getActionProcess=t=>t?new OO.ui.Process(()=>{this.getManager().closeWindow(this)}):d.super.prototype.getActionProcess.call(this,t);this.getTeardownProcess=()=>d.super.prototype.getTeardownProcess.call(this).next(()=>{this.$body.empty()});this.pageTitleParsed=e,d.static.name="TemplatePreviewDialog",d.static.title="Redirect categorization templates preview",d.static.actions=[{action:"cancel",label:"Close",flags:["safe","close"]}]}}Object.assign(d.prototype,OO.ui.ProcessDialog.prototype);class m extends OO.ui.ProcessDialog{constructor(t){super(t);this.api=new mw.Api;this.getSetupProcess=()=>m.super.prototype.getSetupProcess.call(this).next(()=>{const[t,e]=this.getData();return this.api.post({action:"compare",formatversion:"2",prop:["diff"],fromslots:"main","fromtext-main":t,"fromcontentmodel-main":"wikitext",toslots:"main","totext-main":e,"tocontentmodel-main":"wikitext"}).then(a=>{const n=a.compare.body,s=new OO.ui.MessageWidget({type:"warning",label:"No changes to make!"}),r=new OO.ui.PanelLayout({padded:!0,expanded:!1});r.$element.append(n?`
<table class="diff diff-editfont-monospace">
<colgroup>
<col class="diff-marker">
<col class="diff-content">
<col class="diff-marker">
<col class="diff-content">
</colgroup>
<tbody>
${n}
</tbody>
</table>`:s.$element[0]),this.$body.append(r.$element)})});this.getActionProcess=t=>t?new OO.ui.Process(()=>{this.getManager().closeWindow(this)}):m.super.prototype.getActionProcess.call(this,t);this.getTeardownProcess=()=>m.super.prototype.getTeardownProcess.call(this).next(()=>{this.$body.empty()});m.static.name="ShowChangesDialog",m.static.title="Changes to be made",m.static.actions=[{action:"cancel",label:"Close",flags:["safe","close"]}]}}Object.assign(m.prototype,OO.ui.ProcessDialog.prototype);class x{constructor(){this.api=new mw.Api}async run(){if(this.passesPreChecks()){if(this.redirectTemplates=await this.fetchRedirectTemplates(),this.contentText=document.querySelector("#mw-content-text"),!this.contentText)return mw.notify("redirect-helper: Failed to find content text element!",{type:"error"});if(this.pageTitle=mw.config.get("wgPageName"),this.pageTitleParsed=mw.Title.newFromText(this.pageTitle),!this.pageTitleParsed)return mw.notify("redirect-helper: Failed to parse page title!",{type:"error"});await this.checkPageAndLoad()}}passesPreChecks(){return[mw.config.get("wgNamespaceNumber")>=0,mw.config.get("wgIsProbablyEditable"),mw.config.get("wgIsArticle"),mw.config.get("wgAction")==="view",mw.config.get("wgRevisionId")===mw.config.get("wgCurRevisionId"),!mw.config.get("wgDiffOldId")].every(Boolean)}async fetchRedirectTemplates(){return JSON.parse((await this.api.get({action:"query",formatversion:"2",prop:"revisions",rvprop:"content",rvslots:"main",titles:"User:Eejit43/scripts/redirect-helper.json"})).query.pages?.[0]?.revisions?.[0]?.slots?.main?.content||"{}")}async checkPageAndLoad(){const i=await this.api.get({action:"query",formatversion:"2",prop:"info",titles:this.pageTitle}),t={redirectTemplates:this.redirectTemplates,contentText:this.contentText,pageTitle:this.pageTitle,pageTitleParsed:this.pageTitleParsed};if(i.query.pages[0].missing){mw.util.addCSS(`
#create-redirect-button {
margin-bottom: 20px;
}`);const e=new OO.ui.ButtonWidget({id:"create-redirect-button",label:"Create redirect",icon:"articleRedirect",flags:["progressive"]});e.on("click",()=>{e.$element[0].remove(),new b(t,!1).load()}),this.contentText.prepend(e.$element[0])}else if(i.query.pages[0].redirect)new b(t,!0).load();else{const e=mw.util.addPortletLink(mw.config.get("skin")==="minerva"?"p-tb":"p-cactions","#","Redirect page","redirect-helper");e.addEventListener("click",a=>{a.preventDefault(),new b(t,!1).load(),window.scrollTo({top:0,behavior:"smooth"}),e.remove()})}}}class b{constructor({redirectTemplates:i,contentText:t,pageTitle:e,pageTitleParsed:a},n){this.api=new mw.Api;this.redirectRegex=/^#redirect:?\s*\[\[\s*:?([^[\]{|}]+?)\s*(?:\|[^[\]{|}]+?)?]]\s*/i;this.scriptAdvert=" (via [[User:Eejit43/scripts/redirect-helper|redirect-helper]])";this.needsCheck=!0;this.templateEditorsInfo=[];this.pageContent="";this.redirectTemplates=i,this.contentText=t,this.pageTitle=e,this.pageTitleParsed=a,this.exists=n}async load(){mw.util.addCSS(`
#create-redirect-button {
margin-bottom: 20px;
}
#redirect-helper-box {
background-color: whitesmoke;
width: 700px;
max-width: calc(100% - 50px);
margin-left: auto;
margin-right: auto;
margin-bottom: 25px !important;
}
.redirect-input-layout label {
font-weight: bold;
}
.redirect-helper-template-parameters-container, .redirect-helper-template-parameters-container details {
background-color: #e2e2e2;
border-radius: 5px;
margin-block: 10px;
padding: 5px;
}
.redirect-helper-template-parameters-container summary {
cursor: pointer;
font-weight: bold;
}
.redirect-helper-template-parameters-container details {
background-color: #d1cece;
margin-block: 5px;
}
.redirect-helper-template-parameters-contained #redirect-helper-no-templates-message {
padding: 5px;
}
#redirect-helper-summary-layout {
padding-top: 10px;
margin-top: 15px;
border-top: 1px solid gray;
}
#redirect-helper-submit-layout {
margin-top: 10px;
}
#redirect-helper-submit-layout > * {
margin-bottom: 0;
}
.redirect-helper-warning {
margin-top: 8px;
}`),mw.loader.addLinkTag("https://www.mediawiki.org/w/load.php?modules=mediawiki.diff.styles&only=styles"),this.editorBox=new OO.ui.PanelLayout({id:"redirect-helper-box",padded:!0,expanded:!1,framed:!0}),this.pageTitleParsed.isTalkPage()&&(await this.api.get({action:"query",formatversion:"2",prop:"info",titles:this.pageTitleParsed.getSubjectPage().getPrefixedText()})).query.pages[0].redirect&&await this.loadSyncWithMainButton(),this.loadInputElements(),await this.loadSubmitElements(),this.editorBox.$element[0].append(...[this.syncWithMainButton?.$element?.[0],this.redirectInputLayout.$element[0],this.tagSelectLayout.$element[0],this.templateParametersEditor,this.defaultSortInputLayout.$element[0],this.categorySelectLayout.$element[0],this.summaryInputLayout.$element[0],this.submitLayout.$element[0]].filter(Boolean)),this.contentText.prepend(this.editorBox.$element[0]),this.exists&&this.loadExistingData()}async loadSyncWithMainButton(){const i=await this.getPageContent(this.pageTitleParsed.getSubjectPage().getPrefixedText());this.syncWithMainButton=new OO.ui.ButtonWidget({label:"Sync with main page",icon:"link",flags:["progressive"]}),this.syncWithMainButton.on("click",()=>{const t=this.redirectRegex.exec(i)?.[1];if(!t)return mw.notify("Failed to parse main page content!",{type:"error"});this.redirectInput.setValue(mw.Title.newFromText(t)?.getTalkPage()?.toString()??""),["R from move",...this.redirectTemplates["R from move"].aliases].some(a=>new RegExp(`{{\\s*[${a[0].toLowerCase()}${a[0]}]${a.slice(1)}\\s*(\\||}})`).test(i))&&this.tagSelect.setValue(["R from move"])})}loadInputElements(){this.redirectInput=new O({placeholder:"Target page name",required:!0},this.pageTitleParsed),this.redirectInput.on("change",()=>{let e=this.redirectInput.getValue();e=e.replace(new RegExp(`^(https?:)?/{2}?${mw.config.get("wgServer").replace(/^\/{2}/,"")}/wiki/`),""),e=e.replace(/^:/,""),e.length>0?(this.redirectInput.setValue(e[0].toUpperCase()+e.slice(1).replaceAll("_"," ")),this.defaultSortSuggestButton.setDisabled(!1),this.submitButton.setDisabled(!1),this.showPreviewButton.setDisabled(!1),this.showChangesButton.setDisabled(!1)):(this.defaultSortSuggestButton.setDisabled(!0),this.submitButton.setDisabled(!0),this.showPreviewButton.setDisabled(!0),this.showChangesButton.setDisabled(!0)),this.updateSummary(),this.submitButton.setLabel("Submit"),this.needsCheck=!0}),this.redirectInputLayout=new OO.ui.FieldLayout(this.redirectInput,{label:"Redirect target:",classes:["redirect-input-layout"],align:"top"}),this.tagSelect=new OO.ui.MenuTagMultiselectWidget({allowArbitrary:!1,allowReordering:!1,options:Object.keys(this.redirectTemplates).map(e=>({data:e,label:e}))}),this.tagSelect.getMenu().filterMode="substring",this.tagSelect.on("change",()=>{const e=this.tagSelect.getValue().sort((n,s)=>n.toLowerCase().localeCompare(s.toLowerCase()));this.tagSelect.getValue().join(";")!==e.join(";")&&this.tagSelect.setValue(e),this.updateSummary(),this.submitButton.setLabel("Submit"),this.needsCheck=!0;for(const n of this.templateEditorsInfo)n.details.style.display="none";let a=0;for(const n of this.tagSelect.getValue()){const s=this.templateEditorsInfo.find(r=>r.name===n);s&&(s.details.style.display="block",a++)}t.style.display=a>0?"none":"block"}),this.tagSelectLayout=new OO.ui.FieldLayout(this.tagSelect,{label:"Redirect categorization templates:",classes:["redirect-input-layout"],align:"top"}),this.templateParametersEditor=document.createElement("details"),this.templateParametersEditor.classList.add("redirect-helper-template-parameters-container");const i=document.createElement("summary");i.textContent="Template parameters",this.templateParametersEditor.append(i);for(const[e,a]of Object.entries(this.redirectTemplates)){const n=Object.entries(a.parameters);if(n.length===0)continue;const s=document.createElement("details");s.style.display="none";const r=document.createElement("summary");r.textContent=e,s.append(r);const o={name:e,details:s,parameters:[]};for(const[u,c]of n){const p=new OO.ui.TextInputWidget({placeholder:c.default?.toString(),required:c.required});p.on("change",()=>{this.updateSummary(),this.submitButton.setLabel("Submit"),this.needsCheck=!0});const l=new OO.ui.FieldLayout(p,{label:new OO.ui.HtmlSnippet(`${u}${!c.label||u.toLowerCase()===c.label?.toLowerCase()?"":` (${c.label})`}${c.description?` (${c.description})`:""} (type: ${c.type}) ${c.suggested?" (suggested)":""}${c.example?` (example: "${c.example}")`:""}`),align:"inline"});s.append(l.$element[0]),o.parameters.push({name:u,aliases:c.aliases,editor:p})}this.templateParametersEditor.append(s),this.templateEditorsInfo.push(o)}const t=document.createElement("div");t.id="redirect-helper-no-templates-message",t.textContent="No templates with parameters to display!",t.style.display=this.exists?"none":"block",this.templateParametersEditor.append(t),this.defaultSortInput=new OO.ui.TextInputWidget,this.defaultSortInput.on("change",()=>{const e=this.defaultSortInput.getValue();e.length>0&&this.defaultSortInput.setValue(e.replaceAll("_"," ")),this.updateSummary(),this.submitButton.setLabel("Submit"),this.needsCheck=!0}),this.defaultSortSuggestButton=new OO.ui.ButtonWidget({icon:"robot",label:"Suggest",disabled:!0}),this.defaultSortSuggestButton.on("click",()=>{let e=this.pageTitleParsed.getMainText().replace(/ \(.*\)$/,"");if(["R from person","R from birth name"].some(a=>this.tagSelect.getValue().includes(a))){if(!e.includes(" "))return;let a="";if(/ (?:[JS]r.?|[IVX]+)$/.test(e)&&(a=e.slice(e.lastIndexOf(" ")),e=e.slice(0,e.lastIndexOf(" ")),!e.includes(" ")))return e+a;const n=e.slice(e.lastIndexOf(" ")+1).replace(/,$/,"").replace(/O'/,"O"),s=e.slice(0,e.lastIndexOf(" "));this.defaultSortInput.setValue(n+", "+s+a)}else{let a=e.replaceAll("Mr.","Mister").replaceAll("&","And");for(const n of["An","A","The"])if(a.startsWith(n+" ")){a=a.slice(n.length+1)+", "+n;break}a===e?mw.notify("redirect-helper wasn't able to determine a sort key different from the current page title!",{type:"warn"}):this.defaultSortInput.setValue(a)}}),this.defaultSortInputLayout=new OO.ui.ActionFieldLayout(this.defaultSortInput,this.defaultSortSuggestButton,{label:new OO.ui.HtmlSnippet(`Default sort key (DEFAULTSORT) (see <a href="${mw.util.getUrl("Wikipedia:Categorization#Sort keys")}" target="_blank">guideline</a>):`),classes:["redirect-input-layout"],align:"top"}),this.categorySelectInput=new P({placeholder:"Add categories here"}),this.categorySelectInput.on("change",()=>{let e=this.categorySelectInput.getValue();e=e.replace(new RegExp(`^(https?:)?/{2}?${mw.config.get("wgServer").replace(/^\/{2}/,"")}/wiki/`),""),e=e.replace(/^Category:/,""),e.length>0&&this.categorySelectInput.setValue(e[0].toUpperCase()+e.slice(1).replaceAll("_"," "))}),this.categorySelectInput.on("showing-values",e=>{for(const a of e)this.categorySelect.addAllowedValue(a.data)}),this.categorySelect=new OO.ui.TagMultiselectWidget({allowReordering:!1,inputPosition:"outline",inputWidget:this.categorySelectInput}),this.categorySelect.on("change",()=>{const e=this.categorySelect.getValue().sort((a,n)=>a.toLowerCase().localeCompare(n.toLowerCase()));this.categorySelect.getValue().join(";")!==e.join(";")&&this.categorySelect.setValue(e),this.updateSummary(),this.submitButton.setLabel("Submit"),this.needsCheck=!0}),this.categorySelectLayout=new OO.ui.FieldLayout(this.categorySelect,{label:"Categories:",classes:["redirect-input-layout"],align:"top"}),this.summaryInput=new OO.ui.ComboBoxInputWidget({options:[{data:"Resolve double redirect"},{data:"Resolve self redirect"},{data:"Remove incorrect rcats"}]}),this.summaryInputLayout=new OO.ui.FieldLayout(this.summaryInput,{id:"redirect-helper-summary-layout",label:"Summary:",classes:["redirect-input-layout"],align:"top"})}async loadSubmitElements(){const i=new OO.ui.WindowManager;document.body.append(i.$element[0]),this.submitButton=new OO.ui.ButtonWidget({label:"Submit",disabled:!0,flags:["progressive"]}),this.submitButton.on("click",()=>this.handleSubmitButtonClick());const t=new d({size:"large"},this.pageTitleParsed);i.addWindows([t]),this.showPreviewButton=new OO.ui.ButtonWidget({label:"Show preview",disabled:!0}),this.showPreviewButton.on("click",()=>{t.setData(this.createOutput(this.redirectInput.getValue(),this.tagSelect.getValue(),this.oldStrayText,this.defaultSortInput.getValue(),this.categorySelect.getValue())),t.open()});const e=new m({size:"large"});i.addWindows([e]),this.showChangesButton=new OO.ui.ButtonWidget({label:"Show changes",disabled:!0}),this.showChangesButton.on("click",async()=>{this.exists&&(this.pageContent=await this.getPageContent(this.pageTitle)),e.setData([this.pageContent,this.createOutput(this.redirectInput.getValue(),this.tagSelect.getValue(),this.oldStrayText,this.defaultSortInput.getValue(),this.categorySelect.getValue())]),e.open()}),this.pageTitleParsed.isTalkPage()||(this.talkData=await this.api.get({action:"query",formatversion:"2",prop:"info",titles:this.pageTitleParsed.getTalkPage().getPrefixedText()}),this.syncTalkCheckbox=new OO.ui.CheckboxInputWidget({selected:!!this.talkData.query.pages[0].redirect}),this.syncTalkCheckboxLayout=new OO.ui.Widget({content:[new OO.ui.FieldLayout(this.syncTalkCheckbox,{label:"Sync talk page",align:"inline"})]})),await this.checkShouldPromptPatrol()&&(this.patrolCheckbox=new OO.ui.CheckboxInputWidget({selected:!0}),this.patrolCheckboxLayout=new OO.ui.Widget({content:[new OO.ui.FieldLayout(this.patrolCheckbox,{label:"Mark as patrolled",align:"inline"})]})),this.submitLayout=new OO.ui.HorizontalLayout({id:"redirect-helper-submit-layout",items:[this.submitButton,this.showPreviewButton,this.showChangesButton,this.syncTalkCheckboxLayout,this.patrolCheckboxLayout].filter(Boolean)})}async checkShouldPromptPatrol(){const i=document.querySelector("#mwe-pt-mark .mwe-pt-tool-icon");if(i?.click(),i?.click(),mw.config.get("wgNamespaceNumber")!==0)return!1;if(document.querySelector(".patrollink"))return!0;if(document.querySelector("#mwe-pt-mark-as-reviewed-button"))return!0;if(document.querySelector("#mwe-pt-mark-as-unreviewed-button"))return!1;{if(!mw.config.get("wgArticleId")||!(await this.api.get({action:"query",meta:"userinfo",uiprop:"rights"})).query.userinfo.rights.includes("patrol"))return!1;const e=await this.api.get({action:"pagetriagelist",page_id:mw.config.get("wgArticleId")});return e.pagetriagelist.pages[0]?.user_name===mw.config.get("wgUserName")||e.pagetriagelist.result!=="success"||e.pagetriagelist.pages.length===0?!1:!Number.parseInt(e.pagetriagelist.pages[0]?.patrol_status)}}updateSummary(){const i=this.redirectInput.getValue().trim();if(!i)this.summaryInput.$tabIndexed[0].placeholder="";else if(this.exists){const t=i!==this.oldRedirectTarget?.replaceAll("_"," "),e=this.tagSelect.getValue().some(o=>!this.oldRedirectTags.includes(o))||this.oldRedirectTags.some(o=>!this.tagSelect.getValue().includes(o)),a=this.oldRedirectTagData?this.tagSelect.getValue().some(o=>this.templateEditorsInfo.find(u=>u.name===o)?.parameters.some(u=>{const c=this.oldRedirectTagData[o]?.find(p=>p[0]===u.name)?.[1];return c?c!==u.editor.getValue().trim():!1})):!1,n=this.defaultSortInput.getValue().trim()!==this.oldDefaultSort.replaceAll("_"," "),s=this.categorySelect.getValue().some(o=>!this.oldCategories.includes(o))||this.oldCategories.some(o=>!this.categorySelect.getValue().includes(o)),r=[];t&&r.push(`retarget to [[${i}]]`),e&&r.push("change categorization templates"),a&&r.push("change categorization template arguments"),n&&r.push("change default sort key"),s&&r.push("change categories"),r.length===0&&r.push("perform redirect cleanup"),r[0]=r[0][0].toUpperCase()+r[0].slice(1),r.length>1&&(r[r.length-1]=`and ${r.at(-1)}`),this.summaryInput.$tabIndexed[0].placeholder=r.join(r.length>2?", ":" ")}else this.summaryInput.$tabIndexed[0].placeholder=`Create redirect to [[${i}]]`}async loadExistingData(){this.exists&&(this.pageContent=await this.getPageContent(this.pageTitle)),this.oldRedirectTarget=this.redirectRegex.exec(this.pageContent)?.[1],this.oldRedirectTags=Object.entries(this.redirectTemplates).map(([t,e])=>[t,...e.aliases].some(a=>new RegExp(`{{\\s*[${a[0].toLowerCase()}${a[0]}]${a.slice(1)}\\s*(\\||}})`).test(this.pageContent))?t:null).filter(Boolean).sort((t,e)=>t.toLowerCase().localeCompare(e.toLowerCase()));const i=Object.entries(this.redirectTemplates).flatMap(([t,e])=>[t,...e.aliases]).map(t=>new RegExp(`{{\\s*[${t[0].toLowerCase()}${t[0]}]${t.slice(1)}\\s*(\\||}})`).test(this.pageContent)?t:null).filter(Boolean);this.oldRedirectTagData=Object.fromEntries(i.map(t=>{const e=new RegExp(`{{\\s*[${t[0].toLowerCase()}${t[0]}]${t.slice(1)}\\|?(.*?)\\s*}}`).exec(this.pageContent),a=Object.entries(this.redirectTemplates).find(([r,o])=>[r,...o.aliases].includes(t))?.[0];if(!e?.[1])return null;const s=e[1].split("|").map((r,o)=>{if(!r.includes("="))return[(o+1).toString(),r.trim()];const[u,c]=r.split("=");return[u.trim(),c.trim()]});return[a,s]}).filter(Boolean)),this.oldDefaultSort=this.pageContent.match(/{{DEFAULTSORT:.*?}}/g)?.at(-1)?.slice(14,-2)?.trim()??"",this.oldCategories=this.pageContent.match(/\[\[[Cc]ategory:.+?]]/g)?.map(t=>t.slice(11,-2))??[],this.oldStrayText=[this.pageContent.match(/{{short description\|.*?}}/i)?.[0],this.pageContent.match(/{{DISPLAYTITLE:.*?}}/)?.[0],this.pageContent.match(/{{italic title\|?.*?}}/i)?.[0],this.pageContent.match(/{{title language\|.*?}}/)?.[0]].filter(Boolean).join(`
`),this.oldRedirectTarget?this.redirectInput.setValue(this.oldRedirectTarget.replaceAll("_"," ")):mw.notify("Could not find redirect target!",{type:"error"}),this.tagSelect.setValue(this.oldRedirectTags);for(const[t,e]of Object.entries(this.oldRedirectTagData)){const a=this.templateEditorsInfo.find(n=>n.name===t);if(a)for(const[n,s]of e){const r=a.parameters.find(o=>[o.name,...o.aliases].includes(n));r&&r.editor.setValue(s)}}this.oldDefaultSort&&this.defaultSortInput.setValue(this.oldDefaultSort);for(const t of this.oldCategories)this.categorySelect.addAllowedValue(t);this.categorySelect.setValue(this.oldCategories.map(t=>({data:t,label:t}))),this.updateSummary()}async validateSubmission(){const i=[],t=this.redirectInput.getValue().trim(),e=this.tagSelect.getValue();/^\s*[^[\]{|}]+\s*$/.test(t)||i.push({title:t,message:"is not a valid page title!"});try{this.parsedDestination=mw.Title.newFromText(t)}catch{i.length===0&&i.push({title:t,message:"is not a valid page title!"})}!this.parsedDestination&&i.length===0&&i.push({title:t,message:"is not a valid page title!"}),this.parsedDestination?.toString()===this.pageTitleParsed.toString()&&i.push({message:"cannot redirect to itself!"});const a=await this.api.get({action:"query",formatversion:"2",prop:["pageprops","categories"],titles:t}).catch(l=>(l==="missingtitle"?i.push({title:t,message:"does not exist!"}):i.push({title:t,message:`was not able to be fetched from the API (${l})!`}),null)),n=await this.api.get({action:"parse",page:t,prop:"sections",redirects:!0});if(n.parse.redirects?.[0]){const l=n.parse.redirects[0].to+(n.parse.redirects[0].tofragment?`#${n.parse.redirects[0].tofragment}`:"");i.push({title:t,message:`is a redirect to <a href="${mw.util.getUrl(l)}" target="_blank">${l}</a>. Retarget to that page instead, as double redirects aren't allowed.`})}if(t.split("#").length>1)if(n.parse.sections.find(g=>g.line.replaceAll(/<\/?i>/g,"")===t.split("#")[1]))e.includes("R to anchor")&&i.push({message:"is tagged as a redirect to an anchor, but it is actually a redirect to a section!"}),e.includes("R to section")||i.push({message:"is a redirect to a section, but it is not tagged with <code>{{R to section}}</code>!"});else{const g=(await this.api.get({action:"query",formatversion:"2",prop:"revisions",rvprop:"content",rvslots:"main",titles:this.parsedDestination.toString()})).query.pages[0].revisions[0].slots.main.content;[...g.match(/(?<={{\s*?[Aa](?:nchors?|nchor for redirect|nker|NCHOR|nc)\s*?\|).+?(?=}})/g)?.map(f=>f.split("|").map(h=>h.trim()))?.flat()??[],...g.match(/(?<={{\s*?(?:[Vv](?:isible anchors?|isanc|Anch|anchor|isibleanchor|a)|[Aa](?:nchord|chored|nchor\+)|[Tt]ext anchor)\s*?\|).+?(?=(?<!!|=)}})/g)?.map(f=>f.split("|").map(h=>h.trim()).filter(h=>!/^text\s*?=/.test(h)))?.flat()??[],...g.match(/(?<=id=)"?.+?(?="|>|\|)/g)?.map(f=>f.trim())??[]].includes(t.split("#")[1])?(e.includes("R to section")&&i.push({message:"is tagged as a redirect to a section, but it is actually a redirect to an anchor!"}),e.includes("R to anchor")||i.push({message:"is a redirect to an anchor, but it is not tagged with <code>{{R from anchor}}</code>!"})):i.push({message:`is a redirect to <a href="${mw.util.getUrl(t)}" target="_blank">${t}</a>, but that section or anchor does not exist!`})}t.split("#").length===1&&(e.includes("R to section")||e.includes("R to anchor"))&&i.push({message:"is not a redirect to a section/anchor, but it is tagged with <code>{{R from section}}</code> or <code>{{R from anchor}}</code>!"});const s=!!(a.query.pages[0].pageprops&&"disambiguation"in a.query.pages[0].pageprops),r=!!a.query.pages[0].categories?.some(l=>l.title==="Category:Surnames"),o=["R to disambiguation page","R from incomplete disambiguation"],u=["R from ambiguous sort name","R from ambiguous term"],c=o.some(l=>e.includes(l)),p=u.some(l=>e.includes(l));s&&!c&&!p&&i.push({message:"is a redirect to a disambiguation page, but it is not tagged with a disambiguation categorization template!"}),a.query.pages[0].pageprops&&!s&&(c&&i.push({message:"is not a redirect to a disambiguation page, but it is tagged with a disambiguation categorization template!"}),r&&!p&&i.push({message:"is a redirect to a surname list, but it is not tagged with a correct disambiguation categorization template!"})),s&&e.includes("R to disambiguation page")&&!this.pageTitleParsed.getMainText().endsWith(" (disambiguation)")&&i.push({message:'is tagged with <code>{{R to disambiguation page}}</code>, but this title does not end with " (disambiguation)". Use <code>{{R from ambiguous term}}</code> or a similar categorization template instead!'});for(const l of["R semi-protected","R extended-protected","R template-protected","R fully protected"])e.includes(l)&&i.push({message:`is tagged with unnecessarily tagged with <code>{{${l}}}</code> which will be duplicated by the redirect category shell!`});mw.config.get("wgWikibaseItemId")&&!e.includes("R with Wikidata item")&&i.push({message:"is linked to a Wikidata item but it isn't tagged with <code>{{R with Wikidata item}}</code>!"}),e.includes("R with Wikidata item")&&!mw.config.get("wgWikibaseItemId")&&i.push({message:"is tagged with <code>{{R with Wikidata item}}</code> but it is not actually linked to a Wikidata item!"});for(const l of e){const g=this.redirectTemplates[l];if(g)for(const[y,f]of Object.entries(g.parameters)){const h=this.templateEditorsInfo.find(w=>w.name===l)?.parameters.find(w=>[w.name,...w.aliases].includes(y));h&&f.required&&!h.editor.getValue().trim()&&i.push({message:`is tagged with <code>{{${l}}}</code> but it is missing the required parameter <code>${y}</code>!`})}}return this.syncTalkCheckbox?.isSelected()&&!this.talkData.query.pages[0].missing&&!this.talkData.query.pages[0].redirect&&i.push({title:this.pageTitleParsed.getTalkPage().getPrefixedText(),message:"exists, but is not a redirect!"}),i}async handleSubmitButtonClick(){const i=[this.redirectInput,this.tagSelect,...this.templateEditorsInfo.flatMap(s=>s.parameters.map(r=>r.editor)),this.defaultSortInput,this.defaultSortSuggestButton,this.categorySelect,this.summaryInput,this.submitButton,this.showPreviewButton,this.showChangesButton,this.syncTalkCheckbox,this.patrolCheckbox].filter(Boolean);for(const s of i)s.setDisabled(!0);this.submitButton.setLabel("Checking target validity...");let t=[];if(this.needsCheck?t=await this.validateSubmission():this.parsedDestination=mw.Title.newFromText(this.redirectInput.getValue()),t.length>0){for(const s of document.querySelectorAll(".redirect-helper-warning"))s.remove();for(const{title:s,message:r}of t){const o=new OO.ui.HtmlSnippet(`${s?`<a href="${mw.util.getUrl(s)}" target="_blank">${s}</a>`:"This page"} ${r} Click again without making changes to submit anyway.`),u=new OO.ui.MessageWidget({type:"error",classes:["redirect-helper-warning"],inline:!0,label:o});this.editorBox.$element[0].append(u.$element[0])}for(const s of i)s.setDisabled(!1);this.submitButton.setLabel("Submit anyway"),this.needsCheck=!1;return}this.submitButton.setLabel(`${this.exists?"Editing":"Creating"} redirect...`);const e=this.createOutput(this.redirectInput.getValue(),this.tagSelect.getValue(),this.oldStrayText,this.defaultSortInput.getValue(),this.categorySelect.getValue()),a=(this.summaryInput.getValue()||this.summaryInput.$tabIndexed[0].placeholder)+this.scriptAdvert;if(await this.editOrCreate(this.pageTitle,e,a)){if(mw.notify(`Redirect ${this.exists?"edited":"created"} successfully!`,{type:"success"}),this.syncTalkCheckbox?.isSelected()){this.submitButton.setLabel("Editing talk page...");const s=this.tagSelect.getValue().includes("R from move"),r=this.createOutput(this.parsedDestination.getTalkPage().getPrefixedText(),s?["R from move"]:[],void 0,void 0,[]);if(!await this.editOrCreate(this.pageTitleParsed.getTalkPage().getPrefixedText(),r,"Syncing redirect from main page"+this.scriptAdvert))return;mw.notify("Talk page synced successfully!",{type:"success"})}if(this.patrolCheckbox?.isSelected()){this.submitButton.setLabel("Patrolling redirect...");const s=document.querySelector(".patrollink a"),r=document.querySelector("#mwe-pt-mark-as-reviewed-button");s?await this.api.postWithToken("patrol",{action:"patrol",rcid:new URL(s.href).searchParams.get("rcid")}).catch((u,c)=>(mw.notify(`Error patrolling ${this.pageTitle} via API: ${c?.error.info??"Unknown error"} (${u})`,{type:"error"}),null))&&mw.notify("Redirect patrolled successfully!",{type:"success"}):r?(r.click(),mw.notify("Redirect patrolled successfully!",{type:"success"})):mw.notify("Page curation toolbar not found, redirect cannot be patrolled!",{type:"error"})}this.submitButton.setLabel("Complete, reloading..."),window.location.href=mw.util.getUrl(this.pageTitle,{redirect:"no"})}}createOutput(i,t,e,a,n){const s=mw.Title.newFromText(i),r=s?`${s.getNamespaceId()===14?":":""}${s.getPrefixedText()}${s.getFragment()?`#${s.getFragment()}`:""}`:i.trim();this.pageTitleParsed.getMainText().toLocaleLowerCase().normalize("NFD").replaceAll(/[\u0300-\u036F]/g,"")===a?.toLowerCase().normalize("NFD").replaceAll(/[\u0300-\u036F]/g,"")&&(a=void 0);const o=t.map(u=>{const c=this.templateEditorsInfo.find(l=>l.name===u);if(!c)return`{{${u}}}`;const p=c.parameters.map((l,g)=>{const y=l.editor.getValue().trim();return y?`|${l.name===(g+1).toString()?"":`${l.name}=`}${y}`:null}).filter(Boolean).join("");return`{{${u}${p}}}`});return[`#REDIRECT [[${r}]]
`,t.length>0?`{{Redirect category shell|
${o.join(`
`)}
}}
`:null,e?e+`
`:null,a?`{{DEFAULTSORT:${a.trim()}}}`:null,n.length>0?n.map(u=>`[[Category:${u}]]`).join(`
`):null].filter(Boolean).join(`
`)}async getPageContent(i){return(await this.api.get({action:"query",formatversion:"2",prop:"revisions",rvprop:"content",rvslots:"main",titles:i})).query.pages[0].revisions[0].slots.main.content.trim()}async editOrCreate(i,t,e){return await this.api.edit(i,()=>({text:t,summary:e})).catch((a,n)=>a==="nocreate-missing"?this.api.create(i,{summary:e},t).catch((s,r)=>{mw.notify(`Error creating ${i}: ${r?.error.info??"Unknown error"} (${s})`,{type:"error"})}):(mw.notify(`Error editing or creating ${i}: ${n?.error.info??"Unknown error"} (${a})`,{type:"error"}),null))}}new x().run()});
// </nowiki>
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../scripts/redirect-helper.ts"],
  "sourcesContent": ["import {\n    ApiComparePagesParams,\n    ApiParseParams,\n    ApiQueryInfoParams,\n    ApiQueryPagePropsParams,\n    ApiQueryRevisionsParams,\n    ApiQueryUserInfoParams,\n    PageTriageApiPageTriageListParams,\n} from 'types-mediawiki/api_params';\nimport {\n    ApiQueryAllPagesGeneratorParams, // eslint-disable-line unicorn/prevent-abbreviations\n    CategoriesResult,\n    MediaWikiDataError,\n    PageInfoResult,\n    PageParseResult,\n    PageRevisionsResult,\n    PageTriageListResponse,\n    PagepropsResult,\n    UserPermissionsResponse,\n} from '../global-types';\n\ntype RedirectTemplateParameters = Record<\n    string,\n    {\n        aliases: string[];\n        label: string | null;\n        description: string | null;\n        type: string;\n        required: boolean;\n        suggested: boolean;\n        default: string | number | boolean | null;\n        example: string | number | boolean | null;\n    }\n>;\n\ntype RedirectTemplateData = Record<string, { parameters: RedirectTemplateParameters; aliases: string[] }>;\n\ninterface TemplateEditorElementInfo {\n    name: string;\n    details: HTMLDetailsElement;\n    parameters: { name: string; aliases: string[]; editor: OO.ui.TextInputWidget }[];\n}\n\ninterface LookupElementConfig extends OO.ui.TextInputWidget.ConfigOptions, OO.ui.mixin.LookupElement.ConfigOptions {}\n\nmw.loader.using(['mediawiki.util', 'oojs-ui-core', 'oojs-ui-widgets', 'oojs-ui-windows', 'oojs-ui.styles.icons-content', 'oojs-ui.styles.icons-editing-core'], () => {\n    // Setup RedirectInputWidget\n\n    /**\n     * An instance of this class is a title lookup element.\n     */\n    class RedirectInputWidget extends OO.ui.TextInputWidget {\n        // Utility variables\n        private api = new mw.Api();\n\n        // Assigned in constructor\n        private pageTitleParsed: mw.Title;\n\n        constructor(config: LookupElementConfig, pageTitleParsed: mw.Title) {\n            super(config);\n            OO.ui.mixin.LookupElement.call(this as unknown as OO.ui.mixin.LookupElement, config);\n\n            this.pageTitleParsed = pageTitleParsed;\n        }\n\n        getLookupRequest = () => {\n            const value = this.getValue();\n            const deferred = $.Deferred();\n\n            if (!value) deferred.resolve([]);\n            else if (value.includes('#')) {\n                const title = value.split('#')[0];\n\n                this.api\n                    .get({ action: 'parse', page: title, prop: 'sections', redirects: true } satisfies ApiParseParams)\n                    .catch(() => null)\n                    .then((result: PageParseResult | null) => {\n                        if (result) {\n                            const matchedSections = result.parse.sections.filter((section) =>\n                                section.line\n                                    .toLowerCase()\n                                    .replaceAll(/<\\/?i>/g, '')\n                                    .startsWith(value.split('#')[1].toLowerCase()),\n                            );\n                            deferred.resolve(\n                                matchedSections.map((section) => ({\n                                    data: `${result.parse.title}#${section.line.replaceAll(/<\\/?i>/g, '')}`,\n                                    label: `${result.parse.title}#${section.line.replaceAll(/<\\/?i>/g, '')}`,\n                                })),\n                            );\n                        } else deferred.resolve([]);\n                    });\n            } else {\n                const parsedTitle = mw.Title.newFromText(value);\n\n                this.api\n                    .get({\n                        action: 'query',\n                        formatversion: '2',\n                        gaplimit: 20,\n                        gapnamespace: parsedTitle?.getNamespaceId() ?? 0,\n                        gapprefix: parsedTitle?.getMainText() ?? value,\n                        generator: 'allpages',\n                        prop: ['info', 'pageprops'],\n                    } satisfies ApiQueryAllPagesGeneratorParams)\n                    .catch(() => null)\n                    .then((result: { query: { pages: { title: string; pageprops: { disambiguation?: string }; redirect?: string }[] } } | null) => {\n                        if (result)\n                            deferred.resolve(\n                                result.query?.pages //\n                                    ? result.query.pages //\n                                          .filter((page) => page.title !== this.pageTitleParsed.toString())\n                                          .map((page) => ({\n                                              data: page.title,\n                                              label: new OO.ui.HtmlSnippet(\n                                                  `${page.title}${page.pageprops && 'disambiguation' in page.pageprops ? ' <i>(disambiguation)</i>' : ''}${'redirect' in page ? ' <i>(redirect)</i>' : ''}`,\n                                              ),\n                                          }))\n                                    : [],\n                            );\n                        else deferred.resolve([]);\n                    });\n            }\n\n            return deferred.promise({ abort() {} }); // eslint-disable-line @typescript-eslint/no-empty-function\n        };\n\n        getLookupCacheDataFromResponse = <T>(response: T[] | null | undefined) => response ?? [];\n\n        getLookupMenuOptionsFromData = (data: { data: string; label: string }[]) => data.map(({ data, label }) => new OO.ui.MenuOptionWidget({ data, label }));\n    }\n\n    Object.assign(RedirectInputWidget.prototype, OO.ui.mixin.LookupElement.prototype);\n\n    // Setup CategoryInputWidget\n\n    /**\n     * An instance of this class is a category lookup element.\n     */\n    class CategoryInputWidget extends OO.ui.TextInputWidget {\n        // Utility variables\n        private api = new mw.Api();\n\n        constructor(config: LookupElementConfig) {\n            super(config);\n            OO.ui.mixin.LookupElement.call(this as unknown as OO.ui.mixin.LookupElement, config);\n        }\n\n        getLookupRequest = () => {\n            const value = this.getValue();\n            const deferred = $.Deferred();\n\n            if (!value) deferred.resolve([]);\n\n            const parsedTitle = mw.Title.newFromText(value);\n\n            this.api\n                .get({\n                    action: 'query',\n                    formatversion: '2',\n                    gaplimit: 20,\n                    gapnamespace: 14,\n                    gapprefix: parsedTitle?.getMainText() ?? value,\n                    generator: 'allpages',\n                    prop: 'categories',\n                } satisfies ApiQueryAllPagesGeneratorParams)\n                .catch(() => null)\n                .then((result: { query: { pages: { title: string; categories?: { title: string }[] }[] } } | null) => {\n                    if (result?.query?.pages) {\n                        const pages = result.query.pages //\n                            .filter((page) => !page.categories?.some((category) => category.title === 'Category:Wikipedia soft redirected categories'))\n                            .map((page) => {\n                                const titleWithoutNamespace = page.title.split(':')[1];\n\n                                return { data: titleWithoutNamespace, label: titleWithoutNamespace };\n                            });\n\n                        this.emit('showing-values', pages);\n\n                        deferred.resolve(pages);\n                    } else deferred.resolve([]);\n                });\n\n            return deferred.promise({ abort() {} }); // eslint-disable-line @typescript-eslint/no-empty-function\n        };\n\n        getLookupCacheDataFromResponse = <T>(response: T[] | null | undefined) => response ?? [];\n\n        getLookupMenuOptionsFromData = (data: { data: string; label: string }[]) => data.map(({ data, label }) => new OO.ui.MenuOptionWidget({ data, label }));\n    }\n\n    Object.assign(CategoryInputWidget.prototype, OO.ui.mixin.LookupElement.prototype);\n\n    // Setup TemplatePreviewDialog\n\n    /**\n     * An instance of this class is a dialog used for previewing templates.\n     */\n    class TemplatePreviewDialog extends OO.ui.ProcessDialog {\n        // Utility variables\n        private api = new mw.Api();\n\n        // Assigned in constructor\n        private pageTitleParsed: mw.Title;\n\n        constructor(config: OO.ui.ProcessDialog.ConfigOptions, pageTitleParsed: mw.Title) {\n            super(config);\n\n            this.pageTitleParsed = pageTitleParsed;\n\n            TemplatePreviewDialog.static.name = 'TemplatePreviewDialog';\n            TemplatePreviewDialog.static.title = 'Redirect categorization templates preview';\n            TemplatePreviewDialog.static.actions = [{ action: 'cancel', label: 'Close', flags: ['safe', 'close'] }];\n        }\n\n        getSetupProcess = () => {\n            return TemplatePreviewDialog.super.prototype.getSetupProcess.call(this).next(() => {\n                return this.api\n                    .post({\n                        action: 'parse',\n                        formatversion: '2',\n                        contentmodel: 'wikitext',\n                        prop: ['text', 'categorieshtml'],\n                        title: this.pageTitleParsed.getPrefixedDb(),\n                        text: this.getData() as string,\n                    } satisfies ApiParseParams)\n                    .then((result) => {\n                        const tagsContent = (result as { parse: { text: string } }).parse.text;\n                        const categoriesContent = (result as { parse: { categorieshtml: string } }).parse.categorieshtml;\n\n                        const panelLayout = new OO.ui.PanelLayout({ padded: true, expanded: false });\n                        panelLayout.$element.append(tagsContent, categoriesContent);\n\n                        (this as unknown as { $body: JQuery }).$body.append(panelLayout.$element);\n                    });\n            });\n        };\n\n        getActionProcess = (action: string) => {\n            return action\n                ? new OO.ui.Process(() => {\n                      this.getManager().closeWindow(this);\n                  })\n                : TemplatePreviewDialog.super.prototype.getActionProcess.call(this, action);\n        };\n\n        getTeardownProcess = () => {\n            return TemplatePreviewDialog.super.prototype.getTeardownProcess.call(this).next(() => {\n                (this as unknown as { $body: JQuery }).$body.empty();\n            });\n        };\n    }\n\n    Object.assign(TemplatePreviewDialog.prototype, OO.ui.ProcessDialog.prototype);\n\n    // Setup TemplatePreviewDialog\n\n    /**\n     * An instance of this class is a dialog used for showing changes to be made.\n     */\n    class ShowChangesDialog extends OO.ui.ProcessDialog {\n        // Utility variables\n        private api = new mw.Api();\n\n        constructor(config: OO.ui.ProcessDialog.ConfigOptions) {\n            super(config);\n\n            ShowChangesDialog.static.name = 'ShowChangesDialog';\n            ShowChangesDialog.static.title = 'Changes to be made';\n            ShowChangesDialog.static.actions = [{ action: 'cancel', label: 'Close', flags: ['safe', 'close'] }];\n        }\n\n        getSetupProcess = () => {\n            return ShowChangesDialog.super.prototype.getSetupProcess.call(this).next(() => {\n                const [oldText, newText] = this.getData() as string[];\n\n                return this.api\n                    .post({\n                        action: 'compare',\n                        formatversion: '2',\n                        prop: ['diff'],\n                        fromslots: 'main',\n                        'fromtext-main': oldText,\n                        'fromcontentmodel-main': 'wikitext',\n                        toslots: 'main',\n                        'totext-main': newText,\n                        'tocontentmodel-main': 'wikitext',\n                    } satisfies ApiComparePagesParams & { 'fromtext-main': string; 'fromcontentmodel-main': string; 'totext-main': string; 'tocontentmodel-main': string })\n                    .then((result) => {\n                        const comparison = (result as { compare: { body: string } }).compare.body;\n\n                        const noChangesElement = new OO.ui.MessageWidget({ type: 'warning', label: 'No changes to make!' });\n\n                        const panelLayout = new OO.ui.PanelLayout({ padded: true, expanded: false });\n                        panelLayout.$element.append(\n                            comparison\n                                ? `\n<table class=\"diff diff-editfont-monospace\">\n    <colgroup>\n        <col class=\"diff-marker\">\n        <col class=\"diff-content\">\n        <col class=\"diff-marker\">\n        <col class=\"diff-content\">\n    </colgroup>\n    <tbody>\n        ${comparison}\n    </tbody>\n</table>`\n                                : noChangesElement.$element[0],\n                        );\n\n                        (this as unknown as { $body: JQuery }).$body.append(panelLayout.$element);\n                    });\n            });\n        };\n\n        getActionProcess = (action: string) => {\n            return action\n                ? new OO.ui.Process(() => {\n                      this.getManager().closeWindow(this);\n                  })\n                : ShowChangesDialog.super.prototype.getActionProcess.call(this, action);\n        };\n\n        getTeardownProcess = () => {\n            return ShowChangesDialog.super.prototype.getTeardownProcess.call(this).next(() => {\n                (this as unknown as { $body: JQuery }).$body.empty();\n            });\n        };\n    }\n\n    Object.assign(ShowChangesDialog.prototype, OO.ui.ProcessDialog.prototype);\n\n    /**\n     * An instance of this class handles the entire functionality of the redirect-helper script.\n     */\n    class RedirectHelper {\n        // Utility variables\n        private api = new mw.Api();\n\n        // Assigned in constructor\n        private redirectTemplates!: RedirectTemplateData;\n        private contentText!: HTMLDivElement;\n        private pageTitle!: string;\n        private pageTitleParsed!: mw.Title;\n\n        /**\n         * Runs the redirect helper.\n         */\n        async run() {\n            if (!this.passesPreChecks()) return;\n\n            this.redirectTemplates = await this.fetchRedirectTemplates();\n\n            this.contentText = document.querySelector('#mw-content-text') as HTMLDivElement;\n            if (!this.contentText) return mw.notify('redirect-helper: Failed to find content text element!', { type: 'error' });\n\n            this.pageTitle = mw.config.get('wgPageName');\n\n            this.pageTitleParsed = mw.Title.newFromText(this.pageTitle)!;\n            if (!this.pageTitleParsed) return mw.notify('redirect-helper: Failed to parse page title!', { type: 'error' });\n\n            await this.checkPageAndLoad();\n        }\n\n        /**\n         * Checks if the page passes pre checks.\n         */\n        private passesPreChecks() {\n            const conditions = [\n                mw.config.get('wgNamespaceNumber') >= 0, // Is not virtual namespace\n                mw.config.get('wgIsProbablyEditable'), // Page is editable\n                mw.config.get('wgIsArticle'), // Viewing the content of a page\n                mw.config.get('wgAction') === 'view', // Viewing the page (not editing)\n                mw.config.get('wgRevisionId') === mw.config.get('wgCurRevisionId'), // Viewing the current revision\n                !mw.config.get('wgDiffOldId'), // Not viewing a diff\n            ];\n\n            return conditions.every(Boolean);\n        }\n\n        /**\n         * Fetches the redirect templates.\n         */\n        private async fetchRedirectTemplates() {\n            return JSON.parse(\n                (\n                    (await this.api.get({\n                        action: 'query',\n                        formatversion: '2',\n                        prop: 'revisions',\n                        rvprop: 'content',\n                        rvslots: 'main',\n                        titles: 'User:Eejit43/scripts/redirect-helper.json',\n                    } satisfies ApiQueryRevisionsParams)) as PageRevisionsResult\n                ).query.pages?.[0]?.revisions?.[0]?.slots?.main?.content || '{}',\n            ) as RedirectTemplateData;\n        }\n\n        /**\n         * Checks a page's status and loads the helper appropriately.\n         */\n        private async checkPageAndLoad() {\n            const pageInfo = (await this.api.get({ action: 'query', formatversion: '2', prop: 'info', titles: this.pageTitle } satisfies ApiQueryInfoParams)) as PageInfoResult;\n\n            const dialogInfo = { redirectTemplates: this.redirectTemplates, contentText: this.contentText, pageTitle: this.pageTitle, pageTitleParsed: this.pageTitleParsed };\n\n            if (pageInfo.query.pages[0].missing) {\n                mw.util.addCSS(`\n#create-redirect-button {\n    margin-bottom: 20px;\n}`);\n\n                const button = new OO.ui.ButtonWidget({ id: 'create-redirect-button', label: 'Create redirect', icon: 'articleRedirect', flags: ['progressive'] });\n                button.on('click', () => {\n                    button.$element[0].remove();\n                    new RedirectHelperDialog(dialogInfo, false).load();\n                });\n\n                this.contentText.prepend(button.$element[0]);\n            } else if (pageInfo.query.pages[0].redirect) new RedirectHelperDialog(dialogInfo, true).load();\n            else {\n                const portletLink = mw.util.addPortletLink(mw.config.get('skin') === 'minerva' ? 'p-tb' : 'p-cactions', '#', 'Redirect page', 'redirect-helper')!;\n                portletLink.addEventListener('click', (event) => {\n                    event.preventDefault();\n\n                    new RedirectHelperDialog(dialogInfo, false).load();\n\n                    window.scrollTo({ top: 0, behavior: 'smooth' });\n\n                    portletLink.remove();\n                });\n            }\n        }\n    }\n\n    /**\n     * An instance of this class handles the dialog portion of redirect-helper script.\n     */\n    class RedirectHelperDialog {\n        // Utility variables\n        private api = new mw.Api();\n        private redirectRegex = /^#redirect:?\\s*\\[\\[\\s*:?([^[\\]{|}]+?)\\s*(?:\\|[^[\\]{|}]+?)?]]\\s*/i;\n        private scriptAdvert = ' (via [[User:Eejit43/scripts/redirect-helper|redirect-helper]])';\n\n        // Assigned in constructor\n        private redirectTemplates: RedirectTemplateData;\n        private contentText: HTMLDivElement;\n        private pageTitle: string;\n        private pageTitleParsed: mw.Title;\n\n        private exists: boolean;\n\n        // Used during run()\n        private needsCheck = true;\n\n        private editorBox!: OO.ui.PanelLayout;\n        private syncWithMainButton?: OO.ui.ButtonWidget;\n        private redirectInput!: RedirectInputWidget;\n        private redirectInputLayout!: OO.ui.FieldLayout;\n        private tagSelect!: OO.ui.MenuTagMultiselectWidget;\n        private tagSelectLayout!: OO.ui.ActionFieldLayout;\n        private templateParametersEditor!: HTMLDetailsElement;\n        private templateEditorsInfo: TemplateEditorElementInfo[] = [];\n        private categorySelect!: OO.ui.TagMultiselectWidget;\n        private categorySelectInput!: CategoryInputWidget;\n        private categorySelectLayout!: OO.ui.FieldLayout;\n        private defaultSortInput!: OO.ui.TextInputWidget;\n        private defaultSortSuggestButton!: OO.ui.ButtonWidget;\n        private defaultSortInputLayout!: OO.ui.ActionFieldLayout;\n        private summaryInput!: OO.ui.ComboBoxInputWidget;\n        private summaryInputLayout!: OO.ui.FieldLayout;\n        private submitButton!: OO.ui.ButtonWidget;\n        private showChangesButton!: OO.ui.ButtonWidget;\n        private showPreviewButton!: OO.ui.ButtonWidget;\n        private syncTalkCheckbox?: OO.ui.CheckboxInputWidget;\n        private syncTalkCheckboxLayout?: OO.ui.Widget;\n        private patrolCheckbox?: OO.ui.CheckboxInputWidget;\n        private patrolCheckboxLayout?: OO.ui.Widget;\n        private submitLayout!: OO.ui.HorizontalLayout;\n\n        private talkData?: PageInfoResult;\n\n        private pageContent = '';\n\n        private oldRedirectTarget?: string;\n        private oldRedirectTags?: string[];\n        private oldRedirectTagData?: Record<string, string[][]>;\n        private oldDefaultSort?: string;\n        private oldCategories?: string[];\n        private oldStrayText?: string;\n\n        private parsedDestination!: mw.Title | null;\n\n        constructor(\n            { redirectTemplates, contentText, pageTitle, pageTitleParsed }: { redirectTemplates: RedirectTemplateData; contentText: HTMLDivElement; pageTitle: string; pageTitleParsed: mw.Title },\n            exists: boolean,\n        ) {\n            this.redirectTemplates = redirectTemplates;\n            this.contentText = contentText;\n            this.pageTitle = pageTitle;\n            this.pageTitleParsed = pageTitleParsed;\n\n            this.exists = exists;\n        }\n\n        /**\n         * Loads the redirect-helper dialog into the page.\n         */\n        async load() {\n            mw.util.addCSS(`\n#create-redirect-button {\n    margin-bottom: 20px;\n}\n\n#redirect-helper-box {\n    background-color: whitesmoke;\n    width: 700px;\n    max-width: calc(100% - 50px);\n    margin-left: auto;\n    margin-right: auto;\n    margin-bottom: 25px !important;\n}\n\n.redirect-input-layout label {\n    font-weight: bold;\n}\n\n.redirect-helper-template-parameters-container, .redirect-helper-template-parameters-container details {\n    background-color: #e2e2e2;\n    border-radius: 5px;\n    margin-block: 10px;\n    padding: 5px;\n}\n\n.redirect-helper-template-parameters-container summary {\n    cursor: pointer;\n    font-weight: bold;\n}\n\n.redirect-helper-template-parameters-container details {\n    background-color: #d1cece;\n    margin-block: 5px;\n}\n\n.redirect-helper-template-parameters-contained #redirect-helper-no-templates-message {\n    padding: 5px;\n}\n\n#redirect-helper-summary-layout {\n    padding-top: 10px;\n    margin-top: 15px;\n    border-top: 1px solid gray;\n}\n\n#redirect-helper-submit-layout {\n    margin-top: 10px;\n}\n\n#redirect-helper-submit-layout > * {\n    margin-bottom: 0;\n}\n\n.redirect-helper-warning {\n    margin-top: 8px;\n}`);\n\n            mw.loader.addLinkTag('https://www.mediawiki.org/w/load.php?modules=mediawiki.diff.styles&only=styles');\n\n            /* Load elements */\n            this.editorBox = new OO.ui.PanelLayout({ id: 'redirect-helper-box', padded: true, expanded: false, framed: true });\n\n            if (this.pageTitleParsed.isTalkPage()) {\n                const mainPageData = (await this.api.get({\n                    action: 'query',\n                    formatversion: '2',\n                    prop: 'info',\n                    titles: this.pageTitleParsed.getSubjectPage()!.getPrefixedText(),\n                } satisfies ApiQueryInfoParams)) as PageInfoResult;\n\n                if (mainPageData.query.pages[0].redirect) await this.loadSyncWithMainButton();\n            }\n\n            this.loadInputElements();\n            await this.loadSubmitElements();\n\n            /* Add elements to screen and load data (if applicable) */\n            this.editorBox.$element[0].append(\n                ...([\n                    this.syncWithMainButton?.$element?.[0],\n                    this.redirectInputLayout.$element[0],\n                    this.tagSelectLayout.$element[0],\n                    this.templateParametersEditor,\n                    this.defaultSortInputLayout.$element[0],\n                    this.categorySelectLayout.$element[0],\n                    this.summaryInputLayout.$element[0],\n                    this.submitLayout.$element[0],\n                ].filter(Boolean) as HTMLElement[]),\n            );\n\n            this.contentText.prepend(this.editorBox.$element[0]);\n\n            if (this.exists) this.loadExistingData();\n        }\n\n        /**\n         * Loads the \"Sync with main page\" button\" on talk pages.\n         */\n        private async loadSyncWithMainButton() {\n            const mainPageContent = await this.getPageContent(this.pageTitleParsed.getSubjectPage()!.getPrefixedText());\n\n            this.syncWithMainButton = new OO.ui.ButtonWidget({ label: 'Sync with main page', icon: 'link', flags: ['progressive'] });\n            this.syncWithMainButton.on('click', () => {\n                const target = this.redirectRegex.exec(mainPageContent)?.[1];\n                if (!target) return mw.notify('Failed to parse main page content!', { type: 'error' });\n\n                this.redirectInput.setValue(mw.Title.newFromText(target)?.getTalkPage()?.toString() ?? '');\n                const fromMove = ['R from move', ...this.redirectTemplates['R from move'].aliases].some((tagOrRedirect) =>\n                    new RegExp(`{{\\\\s*[${tagOrRedirect[0].toLowerCase()}${tagOrRedirect[0]}]${tagOrRedirect.slice(1)}\\\\s*(\\\\||}})`).test(mainPageContent),\n                );\n                if (fromMove) this.tagSelect.setValue(['R from move']);\n            });\n        }\n\n        /**\n         * Loads the input elements.\n         */\n        private loadInputElements() {\n            /* Redirect target input */\n            this.redirectInput = new RedirectInputWidget({ placeholder: 'Target page name', required: true }, this.pageTitleParsed);\n            this.redirectInput.on('change', () => {\n                let value = this.redirectInput.getValue();\n                value = value.replace(new RegExp(`^(https?:)?/{2}?${mw.config.get('wgServer').replace(/^\\/{2}/, '')}/wiki/`), '');\n                value = value.replace(/^:/, '');\n\n                if (value.length > 0) {\n                    this.redirectInput.setValue(value[0].toUpperCase() + value.slice(1).replaceAll('_', ' '));\n                    this.defaultSortSuggestButton.setDisabled(false);\n                    this.submitButton.setDisabled(false);\n                    this.showPreviewButton.setDisabled(false);\n                    this.showChangesButton.setDisabled(false);\n                } else {\n                    this.defaultSortSuggestButton.setDisabled(true);\n                    this.submitButton.setDisabled(true);\n                    this.showPreviewButton.setDisabled(true);\n                    this.showChangesButton.setDisabled(true);\n                }\n\n                this.updateSummary();\n                this.submitButton.setLabel('Submit');\n                this.needsCheck = true;\n            });\n\n            this.redirectInputLayout = new OO.ui.FieldLayout(this.redirectInput, { label: 'Redirect target:', classes: ['redirect-input-layout'], align: 'top' });\n\n            /* Redirect categorization template selection */\n            this.tagSelect = new OO.ui.MenuTagMultiselectWidget({\n                allowArbitrary: false,\n                allowReordering: false,\n                options: Object.keys(this.redirectTemplates).map((tag) => ({ data: tag, label: tag })),\n            });\n            (this.tagSelect.getMenu() as OO.ui.MenuSelectWidget.ConfigOptions).filterMode = 'substring';\n            this.tagSelect.on('change', () => {\n                const sortedTags = (this.tagSelect.getValue() as string[]).sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));\n\n                if ((this.tagSelect.getValue() as string[]).join(';') !== sortedTags.join(';')) this.tagSelect.setValue(sortedTags);\n\n                this.updateSummary();\n                this.submitButton.setLabel('Submit');\n                this.needsCheck = true;\n\n                for (const editorInfo of this.templateEditorsInfo) editorInfo.details.style.display = 'none';\n\n                let shownTemplateEditors = 0;\n                for (const tag of this.tagSelect.getValue() as string[]) {\n                    const editorInfo = this.templateEditorsInfo.find((editorInfo) => editorInfo.name === tag);\n\n                    if (editorInfo) {\n                        editorInfo.details.style.display = 'block';\n                        shownTemplateEditors++;\n                    }\n                }\n\n                noTemplatesMessage.style.display = shownTemplateEditors > 0 ? 'none' : 'block';\n            });\n\n            this.tagSelectLayout = new OO.ui.FieldLayout(this.tagSelect, {\n                label: 'Redirect categorization templates:',\n                classes: ['redirect-input-layout'],\n                align: 'top',\n            });\n\n            /* Redirect categorization template parameters */\n            this.templateParametersEditor = document.createElement('details');\n            this.templateParametersEditor.classList.add('redirect-helper-template-parameters-container');\n\n            const summaryElement = document.createElement('summary');\n            summaryElement.textContent = 'Template parameters';\n            this.templateParametersEditor.append(summaryElement);\n\n            for (const [templateName, templateData] of Object.entries(this.redirectTemplates)) {\n                const parameters = Object.entries(templateData.parameters);\n                if (parameters.length === 0) continue;\n\n                const details = document.createElement('details');\n                details.style.display = 'none';\n\n                const summary = document.createElement('summary');\n                summary.textContent = templateName;\n                details.append(summary);\n\n                const elementData: TemplateEditorElementInfo = { name: templateName, details, parameters: [] };\n\n                for (const [parameterName, parameterData] of parameters) {\n                    const input = new OO.ui.TextInputWidget({ placeholder: parameterData.default?.toString(), required: parameterData.required });\n                    input.on('change', () => {\n                        this.updateSummary();\n                        this.submitButton.setLabel('Submit');\n                        this.needsCheck = true;\n                    });\n\n                    const inputLayout = new OO.ui.FieldLayout(input, {\n                        label: new OO.ui.HtmlSnippet(\n                            `${parameterName}${!parameterData.label || parameterName.toLowerCase() === parameterData.label?.toLowerCase() ? '' : ` (${parameterData.label})`}${parameterData.description ? ` (${parameterData.description})` : ''} (type: ${parameterData.type}) ${parameterData.suggested ? ' (suggested)' : ''}${parameterData.example ? ` (example: \"${parameterData.example}\")` : ''}`,\n                        ),\n                        align: 'inline',\n                    });\n                    details.append(inputLayout.$element[0]);\n\n                    elementData.parameters.push({ name: parameterName, aliases: parameterData.aliases, editor: input });\n                }\n\n                this.templateParametersEditor.append(details);\n\n                this.templateEditorsInfo.push(elementData);\n            }\n\n            const noTemplatesMessage = document.createElement('div');\n            noTemplatesMessage.id = 'redirect-helper-no-templates-message';\n            noTemplatesMessage.textContent = 'No templates with parameters to display!';\n            noTemplatesMessage.style.display = this.exists ? 'none' : 'block';\n\n            this.templateParametersEditor.append(noTemplatesMessage);\n\n            /* DEFAULTSORT input */\n            this.defaultSortInput = new OO.ui.TextInputWidget();\n            this.defaultSortInput.on('change', () => {\n                const value = this.defaultSortInput.getValue();\n\n                if (value.length > 0) this.defaultSortInput.setValue(value.replaceAll('_', ' '));\n\n                this.updateSummary();\n                this.submitButton.setLabel('Submit');\n                this.needsCheck = true;\n            });\n\n            this.defaultSortSuggestButton = new OO.ui.ButtonWidget({ icon: 'robot', label: 'Suggest', disabled: true });\n            this.defaultSortSuggestButton.on('click', () => {\n                let name = this.pageTitleParsed.getMainText().replace(/ \\(.*\\)$/, ''); // Remove disambiguation\n\n                if (['R from person', 'R from birth name'].some((tag) => this.tagSelect.getValue().includes(tag))) {\n                    // Handling is modified from evad37's \"Rater\"\n\n                    if (!name.includes(' ')) return;\n\n                    let generationalSuffix = '';\n                    if (/ (?:[JS]r.?|[IVX]+)$/.test(name)) {\n                        generationalSuffix = name.slice(name.lastIndexOf(' '));\n                        name = name.slice(0, name.lastIndexOf(' '));\n                        if (!name.includes(' ')) return name + generationalSuffix;\n                    }\n\n                    const lastName = name\n                        .slice(name.lastIndexOf(' ') + 1)\n                        .replace(/,$/, '')\n                        .replace(/O'/, 'O');\n                    const otherNames = name.slice(0, name.lastIndexOf(' '));\n\n                    this.defaultSortInput.setValue(lastName + ', ' + otherNames + generationalSuffix);\n                } else {\n                    let newName = name.replaceAll('Mr.', 'Mister').replaceAll('&', 'And');\n\n                    for (const leadingArticle of ['An', 'A', 'The'])\n                        if (newName.startsWith(leadingArticle + ' ')) {\n                            newName = newName.slice(leadingArticle.length + 1) + ', ' + leadingArticle;\n                            break;\n                        }\n\n                    if (newName === name) mw.notify(\"redirect-helper wasn't able to determine a sort key different from the current page title!\", { type: 'warn' });\n                    else this.defaultSortInput.setValue(newName);\n                }\n            });\n\n            this.defaultSortInputLayout = new OO.ui.ActionFieldLayout(this.defaultSortInput, this.defaultSortSuggestButton, {\n                label: new OO.ui.HtmlSnippet(`Default sort key (DEFAULTSORT) (see <a href=\"${mw.util.getUrl('Wikipedia:Categorization#Sort keys')}\" target=\"_blank\">guideline</a>):`),\n                classes: ['redirect-input-layout'],\n                align: 'top',\n            });\n\n            /* Categories selection */\n            this.categorySelectInput = new CategoryInputWidget({ placeholder: 'Add categories here' });\n            this.categorySelectInput.on('change', () => {\n                let value = this.categorySelectInput.getValue();\n                value = value.replace(new RegExp(`^(https?:)?/{2}?${mw.config.get('wgServer').replace(/^\\/{2}/, '')}/wiki/`), '');\n                value = value.replace(/^Category:/, '');\n\n                if (value.length > 0) this.categorySelectInput.setValue(value[0].toUpperCase() + value.slice(1).replaceAll('_', ' '));\n            });\n            this.categorySelectInput.on('showing-values', (pages: { data: string; label: string }[]) => {\n                for (const page of pages) this.categorySelect.addAllowedValue(page.data);\n            });\n            this.categorySelect = new OO.ui.TagMultiselectWidget({ allowReordering: false, inputPosition: 'outline', inputWidget: this.categorySelectInput });\n            this.categorySelect.on('change', () => {\n                const sortedTags = (this.categorySelect.getValue() as string[]).sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));\n\n                if ((this.categorySelect.getValue() as string[]).join(';') !== sortedTags.join(';')) this.categorySelect.setValue(sortedTags);\n\n                this.updateSummary();\n                this.submitButton.setLabel('Submit');\n                this.needsCheck = true;\n            });\n\n            this.categorySelectLayout = new OO.ui.FieldLayout(this.categorySelect, { label: 'Categories:', classes: ['redirect-input-layout'], align: 'top' });\n\n            /* Summary input */\n            this.summaryInput = new OO.ui.ComboBoxInputWidget({\n                options: [\n                    { data: 'Resolve double redirect' }, //\n                    { data: 'Resolve self redirect' },\n                    { data: 'Remove incorrect rcats' },\n                ],\n            });\n\n            this.summaryInputLayout = new OO.ui.FieldLayout(this.summaryInput, { id: 'redirect-helper-summary-layout', label: 'Summary:', classes: ['redirect-input-layout'], align: 'top' });\n        }\n\n        /**\n         * Loads the elements in the submit button row.\n         */\n        private async loadSubmitElements() {\n            const windowManager = new OO.ui.WindowManager();\n            document.body.append(windowManager.$element[0]);\n\n            /* Setup submit button */\n            this.submitButton = new OO.ui.ButtonWidget({ label: 'Submit', disabled: true, flags: ['progressive'] });\n            this.submitButton.on('click', () => this.handleSubmitButtonClick());\n\n            /* Setup show preview button */\n            const templatePreviewDialog = new TemplatePreviewDialog({ size: 'large' }, this.pageTitleParsed);\n            windowManager.addWindows([templatePreviewDialog]);\n\n            this.showPreviewButton = new OO.ui.ButtonWidget({ label: 'Show preview', disabled: true });\n            this.showPreviewButton.on('click', () => {\n                templatePreviewDialog.setData(\n                    this.createOutput(\n                        this.redirectInput.getValue(),\n                        this.tagSelect.getValue() as string[],\n                        this.oldStrayText,\n                        this.defaultSortInput.getValue(),\n                        this.categorySelect.getValue() as string[],\n                    ),\n                );\n                templatePreviewDialog.open();\n            });\n\n            /* Setup show changes button */\n            const showChangesDialog = new ShowChangesDialog({ size: 'large' });\n            windowManager.addWindows([showChangesDialog]);\n\n            this.showChangesButton = new OO.ui.ButtonWidget({ label: 'Show changes', disabled: true });\n            this.showChangesButton.on('click', async () => {\n                if (this.exists) this.pageContent = await this.getPageContent(this.pageTitle);\n\n                showChangesDialog.setData([\n                    this.pageContent,\n                    this.createOutput(\n                        this.redirectInput.getValue(),\n                        this.tagSelect.getValue() as string[],\n                        this.oldStrayText,\n                        this.defaultSortInput.getValue(),\n                        this.categorySelect.getValue() as string[],\n                    ),\n                ]);\n                showChangesDialog.open();\n            });\n\n            /* Setup sync talk checkbox */\n            if (!this.pageTitleParsed.isTalkPage()) {\n                this.talkData = (await this.api.get({\n                    action: 'query',\n                    formatversion: '2',\n                    prop: 'info',\n                    titles: this.pageTitleParsed.getTalkPage()!.getPrefixedText(),\n                } satisfies ApiQueryInfoParams)) as PageInfoResult;\n                this.syncTalkCheckbox = new OO.ui.CheckboxInputWidget({ selected: !!this.talkData.query.pages[0].redirect });\n\n                this.syncTalkCheckboxLayout = new OO.ui.Widget({ content: [new OO.ui.FieldLayout(this.syncTalkCheckbox, { label: 'Sync talk page', align: 'inline' })] });\n            }\n\n            /* Setup patrol checkbox */\n            if (await this.checkShouldPromptPatrol()) {\n                this.patrolCheckbox = new OO.ui.CheckboxInputWidget({ selected: true });\n\n                this.patrolCheckboxLayout = new OO.ui.Widget({ content: [new OO.ui.FieldLayout(this.patrolCheckbox, { label: 'Mark as patrolled', align: 'inline' })] });\n            }\n\n            /* Setup layout */\n            this.submitLayout = new OO.ui.HorizontalLayout({\n                id: 'redirect-helper-submit-layout',\n                items: [this.submitButton, this.showPreviewButton, this.showChangesButton, this.syncTalkCheckboxLayout, this.patrolCheckboxLayout].filter(Boolean) as OO.ui.Widget[],\n            });\n        }\n\n        /**\n         * Determines if the user should be prompted to patrol the page.\n         */\n        private async checkShouldPromptPatrol() {\n            const pageTriageMarkButton = document.querySelector('#mwe-pt-mark .mwe-pt-tool-icon') as HTMLImageElement | null;\n            pageTriageMarkButton?.click();\n            pageTriageMarkButton?.click();\n\n            if (mw.config.get('wgNamespaceNumber') !== 0) return false;\n            else if (document.querySelector('.patrollink')) return true;\n            else if (document.querySelector('#mwe-pt-mark-as-reviewed-button')) return true;\n            else if (document.querySelector('#mwe-pt-mark-as-unreviewed-button')) return false;\n            else {\n                if (!mw.config.get('wgArticleId')) return false;\n                const userPermissions = (await this.api.get({ action: 'query', meta: 'userinfo', uiprop: 'rights' } satisfies ApiQueryUserInfoParams)) as UserPermissionsResponse;\n                if (!userPermissions.query.userinfo.rights.includes('patrol')) return false;\n\n                const patrolResponse = (await this.api.get({\n                    action: 'pagetriagelist',\n                    page_id: mw.config.get('wgArticleId'), // eslint-disable-line @typescript-eslint/naming-convention\n                } satisfies PageTriageApiPageTriageListParams)) as PageTriageListResponse;\n\n                if (patrolResponse.pagetriagelist.pages[0]?.user_name === mw.config.get('wgUserName')) return false;\n                else if (patrolResponse.pagetriagelist.result !== 'success' || patrolResponse.pagetriagelist.pages.length === 0) return false;\n                else return !Number.parseInt(patrolResponse.pagetriagelist.pages[0]?.patrol_status);\n            }\n        }\n\n        /**\n         * Updates the summary input placeholder.\n         */\n        private updateSummary() {\n            const redirectValue = this.redirectInput.getValue().trim();\n\n            if (!redirectValue) (this.summaryInput.$tabIndexed[0] as HTMLInputElement).placeholder = '';\n            else if (this.exists) {\n                const targetChanged = redirectValue !== this.oldRedirectTarget?.replaceAll('_', ' ');\n                const tagsChanged =\n                    this.tagSelect.getValue().some((tag) => !this.oldRedirectTags!.includes(tag as string)) || this.oldRedirectTags!.some((tag) => !this.tagSelect.getValue().includes(tag));\n                const tagArgumentsChanged = this.oldRedirectTagData\n                    ? this.tagSelect.getValue().some((tag) =>\n                          this.templateEditorsInfo\n                              .find((template) => template.name === tag)\n                              ?.parameters.some((parameter) => {\n                                  const foundOldArgument = this.oldRedirectTagData![tag as string]?.find((argument) => argument[0] === parameter.name)?.[1];\n\n                                  return foundOldArgument ? foundOldArgument !== parameter.editor.getValue().trim() : false;\n                              }),\n                      )\n                    : false;\n                const defaultSortChanged = this.defaultSortInput.getValue().trim() !== this.oldDefaultSort!.replaceAll('_', ' ');\n                const categoriesChanged =\n                    this.categorySelect.getValue().some((category) => !this.oldCategories!.includes(category as string)) ||\n                    this.oldCategories!.some((category) => !this.categorySelect.getValue().includes(category));\n\n                const changes = [];\n\n                if (targetChanged) changes.push(`retarget to [[${redirectValue}]]`);\n                if (tagsChanged) changes.push('change categorization templates');\n                if (tagArgumentsChanged) changes.push('change categorization template arguments');\n                if (defaultSortChanged) changes.push('change default sort key');\n                if (categoriesChanged) changes.push('change categories');\n\n                if (changes.length === 0) changes.push('perform redirect cleanup');\n\n                changes[0] = changes[0][0].toUpperCase() + changes[0].slice(1);\n                if (changes.length > 1) changes[changes.length - 1] = `and ${changes.at(-1)}`;\n\n                (this.summaryInput.$tabIndexed[0] as HTMLInputElement).placeholder = changes.join(changes.length > 2 ? ', ' : ' ');\n            } else (this.summaryInput.$tabIndexed[0] as HTMLInputElement).placeholder = `Create redirect to [[${redirectValue}]]`;\n        }\n\n        /**\n         * Loads existing page data.\n         */\n        private async loadExistingData() {\n            if (this.exists) this.pageContent = await this.getPageContent(this.pageTitle);\n\n            this.oldRedirectTarget = this.redirectRegex.exec(this.pageContent)?.[1];\n\n            this.oldRedirectTags = (\n                Object.entries(this.redirectTemplates)\n                    .map(([tag, tagData]) =>\n                        [tag, ...tagData.aliases].some((tagOrRedirect) =>\n                            new RegExp(`{{\\\\s*[${tagOrRedirect[0].toLowerCase()}${tagOrRedirect[0]}]${tagOrRedirect.slice(1)}\\\\s*(\\\\||}})`).test(this.pageContent),\n                        )\n                            ? tag\n                            : null,\n                    )\n                    .filter(Boolean) as string[]\n            ).sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));\n\n            const originalRedirectTags = Object.entries(this.redirectTemplates)\n                .flatMap(([tag, tagData]) => [tag, ...tagData.aliases])\n                .map((tagOrRedirect) => (new RegExp(`{{\\\\s*[${tagOrRedirect[0].toLowerCase()}${tagOrRedirect[0]}]${tagOrRedirect.slice(1)}\\\\s*(\\\\||}})`).test(this.pageContent) ? tagOrRedirect : null))\n                .filter(Boolean) as string[];\n\n            this.oldRedirectTagData = Object.fromEntries(\n                originalRedirectTags\n                    .map((tag) => {\n                        const match = new RegExp(`{{\\\\s*[${tag[0].toLowerCase()}${tag[0]}]${tag.slice(1)}\\\\|?(.*?)\\\\s*}}`).exec(this.pageContent);\n\n                        const newTag = Object.entries(this.redirectTemplates).find(([template, tagData]) => [template, ...tagData.aliases].includes(tag))?.[0];\n\n                        const originalArguments = match?.[1];\n                        if (!originalArguments) return null;\n\n                        const formattedArguments = match[1].split('|').map((argument, index) => {\n                            if (!argument.includes('=')) return [(index + 1).toString(), argument.trim()];\n\n                            const [name, value] = argument.split('=');\n\n                            return [name.trim(), value.trim()];\n                        });\n\n                        return [newTag, formattedArguments];\n                    })\n                    .filter(Boolean) as [string, string[][]][],\n            );\n\n            this.oldDefaultSort =\n                this.pageContent\n                    .match(/{{DEFAULTSORT:.*?}}/g)\n                    ?.at(-1)\n                    ?.slice(14, -2)\n                    ?.trim() ?? '';\n\n            this.oldCategories = this.pageContent.match(/\\[\\[[Cc]ategory:.+?]]/g)?.map((category) => category.slice(11, -2)) ?? [];\n\n            this.oldStrayText = [\n                this.pageContent.match(/{{short description\\|.*?}}/i)?.[0],\n                this.pageContent.match(/{{DISPLAYTITLE:.*?}}/)?.[0],\n                this.pageContent.match(/{{italic title\\|?.*?}}/i)?.[0],\n                this.pageContent.match(/{{title language\\|.*?}}/)?.[0],\n            ]\n                .filter(Boolean)\n                .join('\\n');\n\n            if (this.oldRedirectTarget) this.redirectInput.setValue(this.oldRedirectTarget.replaceAll('_', ' '));\n            else mw.notify('Could not find redirect target!', { type: 'error' });\n\n            this.tagSelect.setValue(this.oldRedirectTags);\n\n            for (const [templateName, data] of Object.entries(this.oldRedirectTagData)) {\n                const foundTemplateEditor = this.templateEditorsInfo.find((editorInfo) => editorInfo.name === templateName);\n                if (!foundTemplateEditor) continue;\n\n                for (const [parameterName, argument] of data) {\n                    const foundParameterEditor = foundTemplateEditor.parameters.find((parameter) => [parameter.name, ...parameter.aliases].includes(parameterName));\n\n                    if (foundParameterEditor) foundParameterEditor.editor.setValue(argument);\n                }\n            }\n\n            if (this.oldDefaultSort) this.defaultSortInput.setValue(this.oldDefaultSort);\n\n            for (const category of this.oldCategories) this.categorySelect.addAllowedValue(category);\n            this.categorySelect.setValue(this.oldCategories.map((category) => ({ data: category, label: category })));\n\n            this.updateSummary();\n        }\n\n        /**\n         * Runs checks on the provided data and returns the errors (if any).\n         */\n        private async validateSubmission() {\n            const errors = [];\n\n            const destination = this.redirectInput.getValue().trim();\n            const tags = this.tagSelect.getValue();\n\n            /* Invalid characters */\n            if (!/^\\s*[^[\\]{|}]+\\s*$/.test(destination)) errors.push({ title: destination, message: 'is not a valid page title!' });\n\n            /* Failed during title parsing */\n            try {\n                this.parsedDestination = mw.Title.newFromText(destination);\n            } catch {\n                if (errors.length === 0) errors.push({ title: destination, message: 'is not a valid page title!' });\n            }\n            if (!this.parsedDestination && errors.length === 0) errors.push({ title: destination, message: 'is not a valid page title!' });\n\n            /* Self redirects */\n            if (this.parsedDestination?.toString() === this.pageTitleParsed.toString()) errors.push({ message: 'cannot redirect to itself!' });\n\n            const destinationData = (await this.api\n                .get({ action: 'query', formatversion: '2', prop: ['pageprops', 'categories'], titles: destination } satisfies ApiQueryPagePropsParams)\n                .catch((errorCode: string) => {\n                    /* Nonexistent destination */ if (errorCode === 'missingtitle') errors.push({ title: destination, message: 'does not exist!' });\n                    /* Other API error */ else errors.push({ title: destination, message: `was not able to be fetched from the API (${errorCode})!` });\n                    return null;\n                })) as (PagepropsResult & CategoriesResult) | null;\n            const destinationParseResult = (await this.api.get({ action: 'parse', page: destination, prop: 'sections', redirects: true } satisfies ApiParseParams)) as PageParseResult;\n\n            /* Double redirects */\n            if (destinationParseResult.parse.redirects?.[0]) {\n                const destinationRedirect =\n                    destinationParseResult.parse.redirects[0].to + (destinationParseResult.parse.redirects[0].tofragment ? `#${destinationParseResult.parse.redirects[0].tofragment}` : '');\n                errors.push({\n                    title: destination,\n                    message: `is a redirect to <a href=\"${mw.util.getUrl(\n                        destinationRedirect,\n                    )}\" target=\"_blank\">${destinationRedirect}</a>. Retarget to that page instead, as double redirects aren't allowed.`,\n                });\n            }\n\n            /* Nonexistent section */\n            if (destination.split('#').length > 1) {\n                const validSection = destinationParseResult.parse.sections.find((section) => section.line.replaceAll(/<\\/?i>/g, '') === destination.split('#')[1]);\n                if (validSection) {\n                    if (tags.includes('R to anchor')) errors.push({ message: 'is tagged as a redirect to an anchor, but it is actually a redirect to a section!' });\n                    if (!tags.includes('R to section')) errors.push({ message: 'is a redirect to a section, but it is not tagged with <code>{{R to section}}</code>!' });\n                } else {\n                    const destinationContent = (\n                        (await this.api.get({\n                            action: 'query',\n                            formatversion: '2',\n                            prop: 'revisions',\n                            rvprop: 'content',\n                            rvslots: 'main',\n                            titles: this.parsedDestination!.toString(),\n                        } satisfies ApiQueryRevisionsParams)) as PageRevisionsResult\n                    ).query.pages[0].revisions[0].slots.main.content;\n\n                    const anchors = [\n                        ...(destinationContent\n                            .match(/(?<={{\\s*?[Aa](?:nchors?|nchor for redirect|nker|NCHOR|nc)\\s*?\\|).+?(?=}})/g)\n                            ?.map((anchor: string) => anchor.split('|').map((part) => part.trim()))\n                            ?.flat() ?? []),\n                        ...(destinationContent\n                            .match(/(?<={{\\s*?(?:[Vv](?:isible anchors?|isanc|Anch|anchor|isibleanchor|a)|[Aa](?:nchord|chored|nchor\\+)|[Tt]ext anchor)\\s*?\\|).+?(?=(?<!!|=)}})/g)\n                            ?.map((anchor: string) =>\n                                anchor\n                                    .split('|')\n                                    .map((part) => part.trim())\n                                    .filter((part) => !/^text\\s*?=/.test(part)),\n                            )\n                            ?.flat() ?? []),\n                        ...(destinationContent.match(/(?<=id=)\"?.+?(?=\"|>|\\|)/g)?.map((anchor: string) => anchor.trim()) ?? []),\n                    ];\n                    if (anchors.includes(destination.split('#')[1])) {\n                        if (tags.includes('R to section')) errors.push({ message: 'is tagged as a redirect to a section, but it is actually a redirect to an anchor!' });\n                        if (!tags.includes('R to anchor')) errors.push({ message: 'is a redirect to an anchor, but it is not tagged with <code>{{R from anchor}}</code>!' });\n                    } else errors.push({ message: `is a redirect to <a href=\"${mw.util.getUrl(destination)}\" target=\"_blank\">${destination}</a>, but that section or anchor does not exist!` });\n                }\n            }\n\n            /* Improperly tagged as redirect to section/anchor */\n            if (destination.split('#').length === 1 && (tags.includes('R to section') || tags.includes('R to anchor')))\n                errors.push({ message: 'is not a redirect to a section/anchor, but it is tagged with <code>{{R from section}}</code> or <code>{{R from anchor}}</code>!' });\n\n            const targetIsDisambiguationPage = !!(destinationData!.query.pages[0].pageprops && 'disambiguation' in destinationData!.query.pages[0].pageprops);\n            const targetIsSurnameList = !!destinationData!.query.pages[0].categories?.some((category) => category.title === 'Category:Surnames');\n\n            const toDisambiguationPageTags = ['R to disambiguation page', 'R from incomplete disambiguation'];\n            const toSurnameListTags = ['R from ambiguous sort name', 'R from ambiguous term'];\n\n            const taggedAsRedirectToDisambiguationPage = toDisambiguationPageTags.some((template) => tags.includes(template));\n            const taggedAsRedirectToSurnameList = toSurnameListTags.some((template) => tags.includes(template));\n\n            /* Redirect to disambiguation page without template */\n            if (targetIsDisambiguationPage && !taggedAsRedirectToDisambiguationPage && !taggedAsRedirectToSurnameList)\n                errors.push({ message: 'is a redirect to a disambiguation page, but it is not tagged with a disambiguation categorization template!' });\n\n            if (destinationData!.query.pages[0].pageprops && !targetIsDisambiguationPage) {\n                /* Improperly tagged as redirect to disambiguation page */\n                if (taggedAsRedirectToDisambiguationPage) errors.push({ message: 'is not a redirect to a disambiguation page, but it is tagged with a disambiguation categorization template!' });\n\n                /* Redirect to surname list without template */\n                if (targetIsSurnameList && !taggedAsRedirectToSurnameList)\n                    errors.push({ message: 'is a redirect to a surname list, but it is not tagged with a correct disambiguation categorization template!' });\n            }\n\n            /* {{R to disambiguation page}} without \" (disambiguation)\" at end of title */\n            if (targetIsDisambiguationPage && tags.includes('R to disambiguation page') && !this.pageTitleParsed.getMainText().endsWith(' (disambiguation)'))\n                errors.push({\n                    message:\n                        'is tagged with <code>{{R to disambiguation page}}</code>, but this title does not end with \" (disambiguation)\". Use <code>{{R from ambiguous term}}</code> or a similar categorization template instead!',\n                });\n\n            /* Tagged with a protection template */\n            for (const template of ['R semi-protected', 'R extended-protected', 'R template-protected', 'R fully protected'])\n                if (tags.includes(template)) errors.push({ message: `is tagged with unnecessarily tagged with <code>{{${template}}}</code> which will be duplicated by the redirect category shell!` });\n\n            /* Linked to a Wikidata item without being tagged with {{R with Wikidata item}} */\n            if (mw.config.get('wgWikibaseItemId') && !tags.includes('R with Wikidata item'))\n                errors.push({ message: \"is linked to a Wikidata item but it isn't tagged with <code>{{R with Wikidata item}}</code>!\" });\n\n            /* Tagged with {{R with Wikidata item}} without being linked to an item */\n            if (tags.includes('R with Wikidata item') && !mw.config.get('wgWikibaseItemId'))\n                errors.push({ message: 'is tagged with <code>{{R with Wikidata item}}</code> but it is not actually linked to a Wikidata item!' });\n\n            /* Missing tag required parameter */\n            for (const tag of tags as string[]) {\n                const tagData = this.redirectTemplates[tag];\n                if (!tagData) continue;\n\n                for (const [parameterName, parameterData] of Object.entries(tagData.parameters)) {\n                    const foundParameter = this.templateEditorsInfo\n                        .find((editorInfo) => editorInfo.name === tag)\n                        ?.parameters.find((parameter) => [parameter.name, ...parameter.aliases].includes(parameterName));\n\n                    if (!foundParameter) continue;\n\n                    if (parameterData.required && !foundParameter.editor.getValue().trim())\n                        errors.push({ message: `is tagged with <code>{{${tag}}}</code> but it is missing the required parameter <code>${parameterName}</code>!` });\n                }\n            }\n\n            /* Syncing talk page but talk page exists and isn't a redirect */\n            if (this.syncTalkCheckbox?.isSelected() && !this.talkData!.query.pages[0].missing && !this.talkData!.query.pages[0].redirect)\n                errors.push({ title: this.pageTitleParsed.getTalkPage()!.getPrefixedText(), message: 'exists, but is not a redirect!' });\n\n            return errors;\n        }\n\n        /**\n         * Handles the event when the user clicks the \"Submit\" button.\n         */\n        private async handleSubmitButtonClick() {\n            const elementsToDisable = [\n                this.redirectInput,\n                this.tagSelect,\n                ...this.templateEditorsInfo.flatMap((template) => template.parameters.map((parameter) => parameter.editor)),\n                this.defaultSortInput,\n                this.defaultSortSuggestButton,\n                this.categorySelect,\n                this.summaryInput,\n                this.submitButton,\n                this.showPreviewButton,\n                this.showChangesButton,\n                this.syncTalkCheckbox,\n                this.patrolCheckbox,\n            ].filter(Boolean);\n\n            for (const element of elementsToDisable) (element as OO.ui.Widget).setDisabled(true);\n\n            this.submitButton.setLabel('Checking target validity...');\n\n            let errors: Awaited<ReturnType<typeof this.validateSubmission>> = [];\n            if (this.needsCheck) errors = await this.validateSubmission();\n            else this.parsedDestination = mw.Title.newFromText(this.redirectInput.getValue());\n\n            if (errors.length > 0) {\n                for (const element of document.querySelectorAll('.redirect-helper-warning')) element.remove();\n                for (const { title, message } of errors) {\n                    const label = new OO.ui.HtmlSnippet(\n                        `${title ? `<a href=\"${mw.util.getUrl(title)}\" target=\"_blank\">${title}</a>` : 'This page'} ${message} Click again without making changes to submit anyway.`,\n                    );\n                    const warningMessage = new OO.ui.MessageWidget({ type: 'error', classes: ['redirect-helper-warning'], inline: true, label });\n\n                    this.editorBox.$element[0].append(warningMessage.$element[0]);\n                }\n\n                for (const element of elementsToDisable) (element as OO.ui.Widget).setDisabled(false);\n\n                this.submitButton.setLabel('Submit anyway');\n                this.needsCheck = false;\n\n                return;\n            }\n\n            /* Edit/create redirect */\n            this.submitButton.setLabel(`${this.exists ? 'Editing' : 'Creating'} redirect...`);\n\n            const output = this.createOutput(\n                this.redirectInput.getValue(),\n                this.tagSelect.getValue() as string[],\n                this.oldStrayText,\n                this.defaultSortInput.getValue(),\n                this.categorySelect.getValue() as string[],\n            );\n\n            const summary = (this.summaryInput.getValue() || (this.summaryInput.$tabIndexed[0] as HTMLInputElement).placeholder) + this.scriptAdvert;\n\n            const result = await this.editOrCreate(this.pageTitle, output, summary);\n            if (!result) return;\n\n            mw.notify(`Redirect ${this.exists ? 'edited' : 'created'} successfully!`, { type: 'success' });\n\n            /* Sync talk page checkbox handler */\n            if (this.syncTalkCheckbox?.isSelected()) {\n                this.submitButton.setLabel('Editing talk page...');\n\n                const fromMove = this.tagSelect.getValue().includes('R from move');\n\n                const output = this.createOutput(this.parsedDestination!.getTalkPage()!.getPrefixedText(), fromMove ? ['R from move'] : [], undefined, undefined, []);\n\n                const talkResult = await this.editOrCreate(this.pageTitleParsed.getTalkPage()!.getPrefixedText(), output, 'Syncing redirect from main page' + this.scriptAdvert);\n                if (!talkResult) return;\n\n                mw.notify('Talk page synced successfully!', { type: 'success' });\n            }\n\n            /* Patrol checkbox handler */\n            if (this.patrolCheckbox?.isSelected()) {\n                this.submitButton.setLabel('Patrolling redirect...');\n\n                const patrolLink: HTMLAnchorElement | null = document.querySelector('.patrollink a');\n                const markReviewedButton = document.querySelector('#mwe-pt-mark-as-reviewed-button') as HTMLButtonElement | null;\n\n                if (patrolLink) {\n                    const patrolResult = await this.api\n                        .postWithToken('patrol', { action: 'patrol', rcid: new URL(patrolLink.href).searchParams.get('rcid')! })\n                        .catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                            mw.notify(`Error patrolling ${this.pageTitle} via API: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                            return null;\n                        });\n                    if (patrolResult) mw.notify('Redirect patrolled successfully!', { type: 'success' });\n                } else if (markReviewedButton) {\n                    markReviewedButton.click();\n                    mw.notify('Redirect patrolled successfully!', { type: 'success' });\n                } else mw.notify('Page curation toolbar not found, redirect cannot be patrolled!', { type: 'error' });\n            }\n\n            this.submitButton.setLabel('Complete, reloading...');\n\n            window.location.href = mw.util.getUrl(this.pageTitle, { redirect: 'no' });\n        }\n\n        /*\n         * Takes provided values to create the page output.\n         */\n        private createOutput(target: string, tags: string[], strayText: string | undefined, defaultSort: string | undefined, categories: string[]) {\n            const parsedTarget = mw.Title.newFromText(target);\n\n            const formattedTitle = parsedTarget\n                ? `${parsedTarget.getNamespaceId() === 14 ? ':' : ''}${parsedTarget.getPrefixedText()}${parsedTarget.getFragment() ? `#${parsedTarget.getFragment()}` : ''}`\n                : target.trim();\n\n            if (\n                this.pageTitleParsed\n                    .getMainText()\n                    .toLocaleLowerCase()\n                    .normalize('NFD')\n                    .replaceAll(/[\\u0300-\\u036F]/g, '') ===\n                defaultSort\n                    ?.toLowerCase()\n                    .normalize('NFD')\n                    .replaceAll(/[\\u0300-\\u036F]/g, '')\n            )\n                defaultSort = undefined; // Check if titles normalize to the same text, and removes the DEFAULTSORT if so\n\n            const tagsWithArguments = tags.map((tag) => {\n                const foundArgumentEditor = this.templateEditorsInfo.find((editorInfo) => editorInfo.name === tag);\n                if (!foundArgumentEditor) return `{{${tag}}}`;\n\n                const mappedArguments = foundArgumentEditor.parameters\n                    .map((parameter, index) => {\n                        const value = parameter.editor.getValue().trim();\n                        if (!value) return null;\n\n                        return `|${parameter.name === (index + 1).toString() ? '' : `${parameter.name}=`}${value}`;\n                    })\n                    .filter(Boolean)\n                    .join('');\n\n                return `{{${tag}${mappedArguments}}}`;\n            });\n\n            return [\n                `#REDIRECT [[${formattedTitle}]]\\n`, //\n                tags.length > 0 ? `{{Redirect category shell|\\n${tagsWithArguments.join('\\n')}\\n}}\\n` : null,\n                strayText ? strayText + '\\n' : null,\n                defaultSort ? `{{DEFAULTSORT:${defaultSort.trim()}}}` : null,\n                categories.length > 0 ? categories.map((category) => `[[Category:${category}]]`).join('\\n') : null,\n            ]\n                .filter(Boolean)\n                .join('\\n');\n        }\n\n        /**\n         * Fetches the content of a given page.\n         * @param title The title to fetch.\n         */\n        private async getPageContent(title: string) {\n            return (\n                (await this.api.get({\n                    action: 'query',\n                    formatversion: '2',\n                    prop: 'revisions',\n                    rvprop: 'content',\n                    rvslots: 'main',\n                    titles: title,\n                } satisfies ApiQueryRevisionsParams)) as PageRevisionsResult\n            ).query.pages[0].revisions[0].slots.main.content.trim();\n        }\n\n        /**\n         * Edits or creates a page with given text.\n         * @param title The page title.\n         * @param text The page text.\n         * @param summary The edit summary.\n         */\n        private async editOrCreate(title: string, text: string, summary: string) {\n            return await this.api\n                .edit(title, () => ({ text, summary }))\n                .catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                    if (errorCode === 'nocreate-missing')\n                        return this.api.create(title, { summary }, text).catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                            mw.notify(`Error creating ${title}: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                        });\n                    else {\n                        mw.notify(`Error editing or creating ${title}: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                        return null;\n                    }\n                });\n        }\n    }\n\n    new RedirectHelper().run();\n});\n"],
  "mappings": "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",
  "names": ["RedirectInputWidget", "config", "pageTitleParsed", "value", "deferred", "title", "result", "matchedSections", "section", "parsedTitle", "page", "response", "data", "label", "CategoryInputWidget", "pages", "category", "titleWithoutNamespace", "TemplatePreviewDialog", "tagsContent", "categoriesContent", "panelLayout", "action", "ShowChangesDialog", "oldText", "newText", "comparison", "noChangesElement", "RedirectHelper", "pageInfo", "dialogInfo", "button", "RedirectHelperDialog", "portletLink", "event", "redirectTemplates", "contentText", "pageTitle", "exists", "mainPageContent", "target", "tagOrRedirect", "tag", "sortedTags", "a", "b", "editorInfo", "shownTemplateEditors", "noTemplatesMessage", "summaryElement", "templateName", "templateData", "parameters", "details", "summary", "elementData", "parameterName", "parameterData", "input", "inputLayout", "name", "generationalSuffix", "lastName", "otherNames", "newName", "leadingArticle", "windowManager", "templatePreviewDialog", "showChangesDialog", "pageTriageMarkButton", "patrolResponse", "redirectValue", "targetChanged", "tagsChanged", "tagArgumentsChanged", "template", "parameter", "foundOldArgument", "argument", "defaultSortChanged", "categoriesChanged", "changes", "tagData", "originalRedirectTags", "match", "newTag", "formattedArguments", "index", "foundTemplateEditor", "foundParameterEditor", "errors", "destination", "tags", "destinationData", "errorCode", "destinationParseResult", "destinationRedirect", "destinationContent", "anchor", "part", "targetIsDisambiguationPage", "targetIsSurnameList", "toDisambiguationPageTags", "toSurnameListTags", "taggedAsRedirectToDisambiguationPage", "taggedAsRedirectToSurnameList", "foundParameter", "elementsToDisable", "element", "message", "warningMessage", "output", "fromMove", "patrolLink", "markReviewedButton", "errorInfo", "strayText", "defaultSort", "categories", "parsedTarget", "formattedTitle", "tagsWithArguments", "foundArgumentEditor", "mappedArguments", "text"]
}
