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/redirect-helper.ts
"use strict";(()=>{var A=Object.defineProperty;var O=(g,t)=>()=>(g&&(t=g(g=0)),t);var D=(g,t)=>{for(var i in t)A(g,i,{get:t[i],enumerable:!0})};var w,v=O(()=>{"use strict";w=class extends OO.ui.TextInputWidget{api=new mw.Api;constructor(t){super(t),OO.ui.mixin.LookupElement.call(this,t)}getLookupRequest=()=>{let t=this.getValue(),i=$.Deferred();t||i.resolve([]);let e=mw.Title.newFromText(t);return this.api.get({action:"query",formatversion:"2",gaplimit:20,gapnamespace:14,gapprefix:e?.getMainText()??t,generator:"allpages",prop:"categories"}).catch(()=>null).then(a=>{if(a?.query?.pages){let s=a.query.pages.filter(r=>!r.categories?.some(o=>o.title==="Category:Wikipedia soft redirected categories")).map(r=>{let o=r.title.split(":")[1];return{data:o,label:o}});this.emit("showing-values",s),i.resolve(s)}else i.resolve([])}),i.promise({abort(){}})};getLookupCacheDataFromResponse=t=>t??[];getLookupMenuOptionsFromData=t=>t.map(({data:i,label:e})=>new OO.ui.MenuOptionWidget({data:i,label:e}))};Object.assign(w.prototype,OO.ui.mixin.LookupElement.prototype)});var b,S=O(()=>{"use strict";b=class g extends OO.ui.ProcessDialog{api=new mw.Api;constructor(t){super(t),g.static.name="ShowChangesDialog",g.static.title="Changes to be made",g.static.actions=[{action:"cancel",label:"Close",flags:["safe","close"]}]}getSetupProcess=()=>g.super.prototype.getSetupProcess.call(this).next(()=>{let[t,i]=this.getData();return this.api.post({action:"compare",formatversion:"2",prop:["diff"],fromslots:"main","fromtext-main":t,"fromcontentmodel-main":"wikitext",toslots:"main","totext-main":i,"tocontentmodel-main":"wikitext"}).then(e=>{let a=e.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(a?`
<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>
${a}
</tbody>
</table>`:s.$element[0]),this.$body.append(r.$element)})});getActionProcess=t=>t?new OO.ui.Process(()=>{this.close()}):g.super.prototype.getActionProcess.call(this,t);getTeardownProcess=()=>g.super.prototype.getTeardownProcess.call(this).next(()=>{this.$body.empty()})};Object.assign(b.prototype,OO.ui.ProcessDialog.prototype)});var T,k=O(()=>{"use strict";T=class g extends OO.ui.ProcessDialog{api=new mw.Api;pageTitleParsed;constructor(t,i){super(t),this.pageTitleParsed=i,g.static.name="OutputPreviewDialog",g.static.title="Redirect categorization templates preview",g.static.actions=[{action:"cancel",label:"Close",flags:["safe","close"]}]}getSetupProcess=()=>g.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=>{let i=t.parse.text,e=t.parse.categorieshtml,a=new OO.ui.PanelLayout({padded:!0,expanded:!1});a.$element.append(i,e),this.$body.append(a.$element)}));getActionProcess=t=>t?new OO.ui.Process(()=>{this.close()}):g.super.prototype.getActionProcess.call(this,t);getTeardownProcess=()=>g.super.prototype.getTeardownProcess.call(this).next(()=>{this.$body.empty()})};Object.assign(T.prototype,OO.ui.ProcessDialog.prototype)});var x,R=O(()=>{"use strict";x=class extends OO.ui.TextInputWidget{api=new mw.Api;pageTitleParsed;constructor(t,i){super(t),OO.ui.mixin.LookupElement.call(this,t),this.pageTitleParsed=i}getLookupRequest=()=>{let t=this.getValue(),i=$.Deferred();if(!t)i.resolve([]);else if(t.includes("#")){let e=t.split("#")[0];this.api.get({action:"parse",page:e,prop:"sections",redirects:!0}).catch(()=>null).then(a=>{if(a){let s=a.parse.sections.filter(r=>r.line.toLowerCase().replaceAll(/<\/?i>/g,"").startsWith(t.split("#")[1].toLowerCase()));i.resolve(s.map(r=>({data:`${a.parse.title}#${r.line.replaceAll(/<\/?i>/g,"")}`,label:`${a.parse.title}#${r.line.replaceAll(/<\/?i>/g,"")}`})))}else i.resolve([])})}else{let e=mw.Title.newFromText(t);this.api.get({action:"query",formatversion:"2",gaplimit:20,gapnamespace:e?.getNamespaceId()??0,gapprefix:e?.getMainText()??t,generator:"allpages",prop:["info","pageprops"]}).catch(()=>null).then(a=>{a?i.resolve(a.query?.pages?a.query.pages.filter(s=>s.title!==this.pageTitleParsed.getPrefixedText()).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>":""}`)})):[]):i.resolve([])})}return i.promise({abort(){}})};getLookupCacheDataFromResponse=t=>t??[];getLookupMenuOptionsFromData=t=>t.map(({data:i,label:e})=>new OO.ui.MenuOptionWidget({data:i,label:e}))};Object.assign(x.prototype,OO.ui.mixin.LookupElement.prototype)});var C={};D(C,{default:()=>P});var P,I=O(()=>{"use strict";v();S();k();R();P=class{api=new mw.Api;redirectRegex=/^#redirect:?\s*\[\[\s*:?([^[\]{|}]+?)\s*(?:\|[^[\]{|}]+?)?]]\s*/i;scriptAdvert=" (via [[User:Eejit43/scripts/redirect-helper|redirect-helper]])";redirectTemplates;contentText;pageTitle;pageTitleParsed;exists;needsCheck=!0;editorBox;syncWithMainButton;redirectInput;redirectInputLayout;tagSelect;tagSelectLayout;templateParametersEditor;templateEditorsInfo=[];categorySelect;categorySelectInput;categorySelectLayout;defaultSortInput;defaultSortSuggestButton;defaultSortInputLayout;summaryInput;summaryInputLayout;submitButton;showChangesButton;showPreviewButton;syncTalkCheckbox;syncTalkCheckboxLayout;patrolCheckbox;patrolCheckboxLayout;submitLayout;talkData;pageContent="";oldRedirectTarget;oldRedirectTags;oldRedirectTagData;oldDefaultSort;oldCategories;oldStrayText;parsedDestination;constructor({redirectTemplates:t,contentText:i,pageTitle:e,pageTitleParsed:a},s){this.redirectTemplates=t,this.contentText=i,this.pageTitle=e,this.pageTitleParsed=a,this.exists=s}async load(){mw.util.addCSS(`
#create-redirect-button {
margin-bottom: 20px;
}
#redirect-helper-box {
margin-right: auto;
margin-bottom: 25px !important;
margin-left: auto;
background-color: whitesmoke;
width: 700px;
max-width: calc(100% - 50px);
color: #202122;
}
.redirect-input-layout label {
font-weight: bold;
}
.redirect-helper-redirect-possibilities::after {
content: " (redirect with possibilities)";
font-style: italic;
}
.redirect-helper-template-parameters-container,
.redirect-helper-template-parameters-container details {
margin-block: 10px;
border-radius: 5px;
background-color: #e2e2e2;
padding: 5px;
}
.redirect-helper-template-parameters-container summary {
cursor: pointer;
font-weight: bold;
}
.redirect-helper-template-parameters-container details {
margin-block: 5px;
background-color: #d1cece;
}
#redirect-helper-no-templates-message {
padding: 5px;
}
#redirect-helper-summary-layout {
margin-top: 15px;
border-top: 1px solid gray;
padding-top: 10px;
}
#redirect-helper-submit-layout {
margin-top: 10px;
}
#redirect-helper-submit-layout > * {
margin-bottom: 0;
}
.redirect-helper-warning {
margin-top: 8px;
}
.redirect-helper-autofix-button {
margin-left: 5px;
font-size: 12px;
}
.redirect-helper-autofix-button a {
padding: 3px 4px !important;
min-height: unset !important;
}`),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(){let t=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",()=>{let i=this.redirectRegex.exec(t)?.[1];if(!i)return mw.notify("Failed to parse main page content!",{type:"error"});this.redirectInput.setValue(mw.Title.newFromText(i)?.getTalkPage()?.getPrefixedText()??""),["R from move",...this.redirectTemplates["R from move"].aliases].some(a=>new RegExp(`{{\\s*[${a[0].toLowerCase()}${a[0]}]${a.slice(1)}\\s*(\\||}})`).test(t))&&this.tagSelect.setValue(["R from move"])})}loadInputElements(){this.redirectInput=new x({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.entries(this.redirectTemplates).map(([e,{redirect:a}])=>{if(!a)return{data:e,label:e};let s=new OO.ui.HtmlSnippet(`<span class="redirect-helper-redirect-possibilities">${e}</span>`);return{data:e,label:s}})}),this.tagSelect.getMenu().filterMode="substring",this.tagSelect.on("change",()=>{let e=this.tagSelect.getValue().sort((s,r)=>s.toLowerCase().localeCompare(r.toLowerCase()));this.tagSelect.getValue().join(";")!==e.join(";")&&this.tagSelect.setValue(e),this.updateSummary(),this.submitButton.setLabel("Submit"),this.needsCheck=!0;for(let s of this.templateEditorsInfo)s.details.style.display="none";let a=0;for(let s of this.tagSelect.getValue()){let r=this.templateEditorsInfo.find(o=>o.name===s);r&&(r.details.style.display="block",a++)}t.textContent=`Template parameters (${a>0?`for ${a} template${a>1?"s":""}`:"none to show"})`,i.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");let t=document.createElement("summary");t.textContent="Template parameters (none to show)",this.templateParametersEditor.append(t);for(let[e,a]of Object.entries(this.redirectTemplates)){let s=Object.entries(a.parameters);if(s.length===0)continue;let r=document.createElement("details");r.style.display="none";let o=document.createElement("summary");o.textContent=e,r.append(o);let n={name:e,details:r,parameters:[]};for(let[u,l]of s){let m=new OO.ui.TextInputWidget({placeholder:l.default?.toString(),required:l.required});m.on("change",()=>{this.updateSummary(),this.submitButton.setLabel("Submit"),this.needsCheck=!0});let p=new OO.ui.FieldLayout(m,{label:new OO.ui.HtmlSnippet(`${u}${!l.label||u.toLowerCase()===l.label?.toLowerCase()?"":` (${l.label})`}${l.description?` (${l.description})`:""} (type: ${l.type}) ${l.suggested?" (suggested)":""}${l.example?` (example: "${l.example}")`:""}`),align:"inline"});r.append(p.$element[0]),n.parameters.push({name:u,aliases:l.aliases,editor:m})}this.templateParametersEditor.append(r),this.templateEditorsInfo.push(n)}let i=document.createElement("div");i.id="redirect-helper-no-templates-message",i.textContent="No templates with parameters to display!",this.templateParametersEditor.append(i),this.defaultSortInput=new OO.ui.TextInputWidget,this.defaultSortInput.on("change",()=>{let 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 birth name","R from fictional character","R from band member","R from member","R from person","R from personal name","R from relative","R from spouse"].some(a=>this.tagSelect.getValue().includes(a))){if(!e.includes(" "))return mw.notify("redirect-helper wasn't able to determine a sort key different from the current page title!",{type:"warn"});let a="";if(/ (?:[JS]r.?|[IVX]+)$/.test(e)&&(a=e.slice(e.lastIndexOf(" ")),e=e.slice(0,e.lastIndexOf(" ")),!e.includes(" ")))return e+a;let s=e.slice(e.lastIndexOf(" ")+1).replace(/,$/,"").replace(/O'/,"O"),r=e.slice(0,e.lastIndexOf(" "));this.defaultSortInput.setValue(s+", "+r+a)}else{let a=e.replaceAll("Mr.","Mister").replaceAll("&","And");for(let s of["An","A","The"])if(a.startsWith(s+" ")){a=a.slice(s.length+1)+", "+s;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 w({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(let 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",()=>{let e=this.categorySelect.getValue().sort((a,s)=>a.toLowerCase().localeCompare(s.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(){let t=new OO.ui.WindowManager;document.body.append(t.$element[0]),this.submitButton=new OO.ui.ButtonWidget({label:"Submit",disabled:!0,flags:["progressive"]}),this.submitButton.on("click",()=>this.handleSubmitButtonClick());let i=new T({size:"large"},this.pageTitleParsed);t.addWindows([i]),this.showPreviewButton=new OO.ui.ButtonWidget({label:"Show preview",disabled:!0}),this.showPreviewButton.on("click",()=>{i.setData(this.createOutput(this.redirectInput.getValue(),this.tagSelect.getValue(),this.oldStrayText,this.defaultSortInput.getValue(),this.categorySelect.getValue())),i.open()});let e=new b({size:"large"});t.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(){let t=document.querySelector("#mwe-pt-mark .mwe-pt-tool-icon");if(t?.click(),t?.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 mw.user.getRights()).includes("patrol"))return!1;let 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(){let t=this.redirectInput.getValue().trim();if(!t)this.summaryInput.$tabIndexed[0].placeholder="";else if(this.exists){let i=this.oldRedirectTarget?.replaceAll("_"," ");i&&(i=i[0].toUpperCase()+i.slice(1));let e=t!==i,a=this.tagSelect.getValue().some(u=>!this.oldRedirectTags.includes(u))||this.oldRedirectTags.some(u=>!this.tagSelect.getValue().includes(u)),s=!1;if(this.oldRedirectTagData){let u=Object.entries(this.redirectTemplates).filter(([,l])=>Object.entries(l.parameters).length>0);for(let[l,m]of u){if(!this.oldRedirectTags.includes(l)||!this.tagSelect.getValue().includes(l))continue;let c=this.oldRedirectTagData[l]??Object.entries(m.parameters).map(([d])=>[d,""]),h=this.templateEditorsInfo.find(d=>d.name===l);for(let d of h.parameters){let f=c.find(L=>L[0]===d.name)?.[1]??"",y=d.editor.getValue().trim();if(f!==y){s=!0;break}}if(s)break}}let r=this.defaultSortInput.getValue().trim()!==this.oldDefaultSort.replaceAll("_"," "),o=this.categorySelect.getValue().some(u=>!this.oldCategories.includes(u))||this.oldCategories.some(u=>!this.categorySelect.getValue().includes(u)),n=[];e&&n.push(`retarget to [[${t}]]`),a&&n.push(`${this.tagSelect.getValue().length>0&&this.oldRedirectTags.length>0?"change":this.tagSelect.getValue().length>0?"add":"remove"} categorization templates`),s&&n.push("change categorization template arguments"),r&&n.push(`${this.defaultSortInput.getValue().trim().length>0&&this.oldDefaultSort.replaceAll("_"," ").length>0?"change":this.defaultSortInput.getValue().trim().length>0?"add":"remove"} default sort key`),o&&n.push(`${this.categorySelect.getValue().length>0&&this.oldCategories.length>0?"change":this.categorySelect.getValue().length>0?"add":"remove"} categories`),n.length===0&&n.push("perform redirect cleanup"),n[0]=n[0][0].toUpperCase()+n[0].slice(1),n.length>1&&(n[n.length-1]=`and ${n.at(-1)}`),this.summaryInput.$tabIndexed[0].placeholder=n.join(n.length>2?", ":" ")}else this.summaryInput.$tabIndexed[0].placeholder=`Create redirect to [[${t}]]`}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(([i,e])=>[i,...e.aliases].some(a=>new RegExp(`{{\\s*[${a[0].toLowerCase()}${a[0]}]${a.slice(1)}\\s*(\\||}})`).test(this.pageContent))?i:null).filter(Boolean).sort((i,e)=>i.toLowerCase().localeCompare(e.toLowerCase()));let t=Object.entries(this.redirectTemplates).flatMap(([i,e])=>[i,...e.aliases]).map(i=>new RegExp(`{{\\s*[${i[0].toLowerCase()}${i[0]}]${i.slice(1)}\\s*(\\||}})`).test(this.pageContent)?i:null).filter(Boolean);this.oldRedirectTagData=Object.fromEntries(t.map(i=>{let e=new RegExp(`{{\\s*[${i[0].toLowerCase()}${i[0]}]${i.slice(1)}\\|?(.*?)\\s*}}`).exec(this.pageContent),a=Object.entries(this.redirectTemplates).find(([o,n])=>[o,...n.aliases].includes(i))?.[0];if(!e?.[1])return null;let r=e[1].split("|").map((o,n)=>{if(!o.includes("="))return[(n+1).toString(),o.trim()];let[u,l]=o.split("=");return[u.trim(),l.trim()]});return[a,r]}).filter(Boolean)),this.oldDefaultSort=this.pageContent.match(/{{DEFAULTSORT:.*?}}/g)?.at(-1)?.slice(14,-2)?.trim()??"",this.oldCategories=this.pageContent.match(/\[\[[Cc]ategory:.+?]]/g)?.map(i=>i.slice(11,-2))??[],this.oldStrayText=[/{{short description\|.*?}}/i.exec(this.pageContent)?.[0],/{{DISPLAYTITLE:.*?}}/.exec(this.pageContent)?.[0],/{{italic title\|?.*?}}/i.exec(this.pageContent)?.[0],/{{title language\|.*?}}/.exec(this.pageContent)?.[0],/{{authority control(\|.*?)?}}/i.exec(this.pageContent)?.[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(let[i,e]of Object.entries(this.oldRedirectTagData)){let a=this.templateEditorsInfo.find(s=>s.name===i);if(a)for(let[s,r]of e){let o=a.parameters.find(n=>[n.name,...n.aliases].includes(s));o&&o.editor.setValue(r)}}this.oldDefaultSort&&this.defaultSortInput.setValue(this.oldDefaultSort);for(let i of this.oldCategories)this.categorySelect.addAllowedValue(i);this.categorySelect.setValue(this.oldCategories.map(i=>({data:i,label:i}))),this.updateSummary()}async validateSubmission(){let t=[],i=this.redirectInput.getValue().trim(),e=this.tagSelect.getValue();/^\s*[^[\]{|}]+\s*$/.test(i)||t.push({title:i,message:"is not a valid page title!"});try{this.parsedDestination=mw.Title.newFromText(i)}catch{t.length===0&&t.push({title:i,message:"is not a valid page title!"})}!this.parsedDestination&&t.length===0&&t.push({title:i,message:"is not a valid page title!"}),this.parsedDestination?.getPrefixedText()===this.pageTitleParsed.getPrefixedText()&&t.push({message:"cannot redirect to itself!"});let a=await this.api.get({action:"query",formatversion:"2",prop:["pageprops","categories"],titles:i}).catch(p=>(p==="missingtitle"?t.push({title:i,message:"does not exist!"}):t.push({title:i,message:`was not able to be fetched from the API (${p})!`}),null)),s=await this.api.get({action:"parse",page:i,prop:"sections",redirects:!0});if(s.parse.redirects?.[0]){let p=s.parse.redirects[0].to+(s.parse.redirects[0].tofragment?`#${s.parse.redirects[0].tofragment}`:"");t.push({title:i,message:`is a redirect to <a href="${mw.util.getUrl(p)}" target="_blank">${p}</a>. Retarget to that page instead, as double redirects aren't allowed.`,autoFixes:[{type:"change-target",target:p}]})}if(i.split("#").length>1)if(s.parse.sections.find(c=>c.line.replaceAll(/<\/?i>/g,"")===i.split("#")[1]))e.includes("R to anchor")&&t.push({message:"is tagged as a redirect to an anchor, but it is actually a redirect to a section!",autoFixes:[{type:"add",tag:"R to section"},{type:"remove",tag:"R to anchor"}]}),e.includes("R to section")||t.push({message:"is a redirect to a section, but it is not tagged with <code>{{R to section}}</code>!",autoFixes:[{type:"add",tag:"R to section"}]});else{let c=(await this.api.get({action:"query",formatversion:"2",prop:"revisions",rvprop:"content",rvslots:"main",titles:this.parsedDestination.getPrefixedText()})).query.pages[0].revisions[0].slots.main.content;[...c.match(/(?<={{\s*?[Aa](?:nchors?|nchor for redirect|nker|NCHOR|nc)\s*?\|).+?(?=}})/g)?.map(d=>d.split("|").map(f=>f.trim()))?.flat()??[],...c.match(/(?<={{\s*?(?:[Vv](?:isible anchors?|isanc|Anch|anchor|isibleanchor|a)|[Aa](?:nchord|chored|nchor\+)|[Tt]ext anchor)\s*?\|).+?(?=(?<!!|=)}})/g)?.map(d=>d.split("|").map(f=>f.trim()).filter(f=>!/^text\s*?=/.test(f)))?.flat()??[],...c.match(/(?<=id=)"?.+?(?="|>|\|)/g)?.map(d=>d.trim())??[],...c.match(/EpisodeNumber += +\d+/g)?.map(d=>`ep${d.split("=")[1].trim()}`)??[]].includes(i.split("#")[1])?(e.includes("R to section")&&t.push({message:"is tagged as a redirect to a section, but it is actually a redirect to an anchor!",autoFixes:[{type:"add",tag:"R to anchor"},{type:"remove",tag:"R to section"}]}),e.includes("R to anchor")||t.push({message:"is a redirect to an anchor, but it is not tagged with <code>{{R from anchor}}</code>!",autoFixes:[{type:"add",tag:"R to anchor"}]})):t.push({message:`is a redirect to <a href="${mw.util.getUrl(i)}" target="_blank">${i}</a>, but that section or anchor does not exist!`,autoFixes:[{type:"change-target",target:i.split("#")[0]}]})}if(i.split("#").length===1)for(let p of["R to section","R to anchor"])e.includes(p)&&t.push({message:`is not a redirect to a section/anchor, but it is tagged with <code>{{${p}}}</code>!`,autoFixes:[{type:"remove",tag:p}]});let r=!!(a.query.pages[0].pageprops&&"disambiguation"in a.query.pages[0].pageprops),o=!!a.query.pages[0].categories?.some(p=>p.title==="Category:Surnames"),n=["R to disambiguation page","R from incomplete disambiguation"],u=["R from ambiguous sort name","R from ambiguous term"],l=n.some(p=>e.includes(p)),m=u.some(p=>e.includes(p));r&&!l&&!m&&t.push({message:"is a redirect to a disambiguation page, but it is not tagged with a disambiguation categorization template!"}),a.query.pages[0].pageprops&&!r&&((!o&&(l||m)||o&&l)&&t.push({message:"is not a redirect to a disambiguation page, but it is tagged with a disambiguation categorization template!",autoFixes:[...n,...u].map(p=>({type:"remove",tag:p}))}),o&&!m&&t.push({message:"is a redirect to a surname list, but it is not tagged with a correct disambiguation categorization template!"})),r&&e.includes("R to disambiguation page")&&!this.pageTitleParsed.getMainText().endsWith(" (disambiguation)")&&t.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!',autoFixes:[{type:"remove",tag:"R to disambiguation page"}]});for(let p of["R protected","R semi-protected","R extended-protected","R template-protected","R fully protected"])e.includes(p)&&t.push({message:`is tagged with unnecessarily tagged with <code>{{${p}}}</code> which will be duplicated by the redirect category shell!`,autoFixes:[{type:"remove",tag:p}]});mw.config.get("wgWikibaseItemId")&&!e.includes("R with Wikidata item")&&t.push({message:"is linked to a Wikidata item but it isn't tagged with <code>{{R with Wikidata item}}</code>!",autoFixes:[{type:"add",tag:"R with Wikidata item"}]}),e.includes("R with Wikidata item")&&!mw.config.get("wgWikibaseItemId")&&t.push({message:"is tagged with <code>{{R with Wikidata item}}</code> but it is not actually linked to a Wikidata item!",autoFixes:[{type:"remove",tag:"R with Wikidata item"}]});for(let p of e){let c=this.redirectTemplates[p];if(c)for(let[h,d]of Object.entries(c.parameters)){let f=this.templateEditorsInfo.find(y=>y.name===p)?.parameters.find(y=>[y.name,...y.aliases].includes(h));f&&d.required&&!f.editor.getValue().trim()&&t.push({message:`is tagged with <code>{{${p}}}</code> but it is missing the required parameter <code>${h}</code>!`})}}return this.syncTalkCheckbox?.isSelected()&&!this.talkData.query.pages[0].missing&&!this.talkData.query.pages[0].redirect&&t.push({title:this.pageTitleParsed.getTalkPage().getPrefixedText(),message:"exists, but is not a redirect!"}),t}async handleSubmitButtonClick(){let t=[this.redirectInput,this.tagSelect,...this.templateEditorsInfo.flatMap(r=>r.parameters.map(o=>o.editor)),this.defaultSortInput,this.defaultSortSuggestButton,this.categorySelect,this.summaryInput,this.submitButton,this.showPreviewButton,this.showChangesButton,this.syncTalkCheckbox,this.patrolCheckbox].filter(Boolean);for(let r of t)r.setDisabled(!0);this.submitButton.setLabel("Checking target validity...");let i=[];if(this.needsCheck?i=await this.validateSubmission():this.parsedDestination=mw.Title.newFromText(this.redirectInput.getValue()),i.length>0){for(let r of document.querySelectorAll(".redirect-helper-warning"))r.remove();for(let{title:r,message:o,autoFixes:n}of i){let u=new OO.ui.HtmlSnippet(`${r?`<a href="${mw.util.getUrl(r)}" target="_blank">${r}</a>`:"This page"} ${o} Click again without making changes to submit anyway.`),l=new OO.ui.MessageWidget({type:"error",classes:["redirect-helper-warning"],inline:!0,label:u});if(n){let m=new OO.ui.ButtonWidget({label:"Perform auto-fix",flags:["progressive"],classes:["redirect-helper-autofix-button"]});m.on("click",()=>{let p=this.tagSelect.getValue();for(let c of n)c.type==="add"&&!p.includes(c.tag)&&this.tagSelect.addTag(c.tag,c.tag),c.type==="remove"&&p.includes(c.tag)&&this.tagSelect.removeTagByData(c.tag),c.type==="change-target"&&this.redirectInput.setValue(c.target);l.$element[0].style.textDecoration="line-through 2px black",m.$element[0].remove()}),l.$element[0].querySelector(".oo-ui-labelElement-label").append(m.$element[0])}this.editorBox.$element[0].append(l.$element[0])}for(let r of t)r.setDisabled(!1);this.submitButton.setLabel("Submit anyway"),this.needsCheck=!1;return}this.submitButton.setLabel(`${this.exists?"Editing":"Creating"} redirect...`);let 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...");let r=this.tagSelect.getValue().includes("R from move"),o=this.createOutput(this.parsedDestination.getTalkPage().getPrefixedText(),r?["R from move"]:[],void 0,void 0,[]);if(!await this.editOrCreate(this.pageTitleParsed.getTalkPage().getPrefixedText(),o,"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...");let r=document.querySelector(".patrollink a"),o=document.querySelector("#mwe-pt-mark-as-reviewed-button");r?await this.api.postWithToken("patrol",{action:"patrol",rcid:new URL(r.href).searchParams.get("rcid")}).catch((u,l)=>(mw.notify(`Error patrolling ${this.pageTitle} via API: ${l?.error.info??"Unknown error"} (${u})`,{type:"error"}),null))&&mw.notify("Redirect patrolled successfully!",{type:"success"}):o?(o.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(t,i,e,a,s){let r=mw.Title.newFromText(t),o=r?`${r.getNamespaceId()===14?":":""}${r.getPrefixedText()}${r.getFragment()?`#${r.getFragment()}`:""}`:t.trim();this.pageTitleParsed.getMainText().toLocaleLowerCase().normalize("NFD").replaceAll(/[\u0300-\u036F]/g,"")===a?.toLowerCase().normalize("NFD").replaceAll(/[\u0300-\u036F]/g,"")&&(a=void 0);let n=i.map(u=>{let l=this.templateEditorsInfo.find(c=>c.name===u);if(!l)return`{{${u}}}`;let m=l.parameters.findLastIndex((c,h)=>c.name===(h+1).toString()&&c.editor.getValue().trim()),p=l.parameters.map((c,h)=>{let d=c.editor.getValue().trim();return!d&&h>m?null:`|${c.name===(h+1).toString()?"":`${c.name}=`}${d}`}).filter(Boolean).join("");return`{{${u}${p}}}`});return[`#REDIRECT [[${o}]]
`,i.length>0?`{{Redirect category shell|
${n.join(`
`)}
}}
`:null,e?e+`
`:null,a?`{{DEFAULTSORT:${a.trim()}}}`:null,s.length>0?s.map(u=>`[[Category:${u}]]`).join(`
`):null].filter(Boolean).join(`
`)}async getPageContent(t){return(await this.api.get({action:"query",formatversion:"2",prop:"revisions",rvprop:"content",rvslots:"main",titles:t})).query.pages[0].revisions[0].slots.main.content.trim()}async editOrCreate(t,i,e){return await this.api.edit(t,()=>({text:i,summary:e})).catch((a,s)=>a==="nocreate-missing"?this.api.create(t,{summary:e},i).catch((r,o)=>{mw.notify(`Error creating ${t}: ${o?.error.info??"Unknown error"} (${r})`,{type:"error"})}):(mw.notify(`Error editing or creating ${t}: ${s?.error.info??"Unknown error"} (${a})`,{type:"error"}),null))}}});var E=["mediawiki.util","oojs-ui-core","oojs-ui-widgets","oojs-ui-windows","oojs-ui.styles.icons-content","oojs-ui.styles.icons-editing-core"];mw.loader.using(E,async()=>{let{default:g}=await Promise.resolve().then(()=>(I(),C));class t{api=new mw.Api;redirectTemplates;contentText;pageTitle;pageTitleParsed;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(){let e=await this.api.get({action:"query",formatversion:"2",prop:"info",titles:this.pageTitle}),a={redirectTemplates:this.redirectTemplates,contentText:this.contentText,pageTitle:this.pageTitle,pageTitleParsed:this.pageTitleParsed};if(e.query.pages[0].missing){mw.util.addCSS(`
#create-redirect-button {
margin-bottom: 20px;
}`);let s=new OO.ui.ButtonWidget({id:"create-redirect-button",label:"Create redirect",icon:"articleRedirect",flags:["progressive"]});s.on("click",()=>{s.$element[0].remove(),new g(a,!1).load()}),this.contentText.prepend(s.$element[0])}else if(e.query.pages[0].redirect)new g(a,!0).load();else{let s=mw.util.addPortletLink(mw.config.get("skin")==="minerva"?"p-tb":"p-cactions","#","Redirect page","redirect-helper");s.addEventListener("click",r=>{r.preventDefault(),new g(a,!1).load(),window.scrollTo({top:0,behavior:"smooth"}),s.remove()})}}}new t().run()});})();
// </nowiki>
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../scripts/redirect-helper/category-input-widget.ts", "../../scripts/redirect-helper/changes-dialog.ts", "../../scripts/redirect-helper/output-preview-dialog.ts", "../../scripts/redirect-helper/redirect-target-input-widget.ts", "../../scripts/redirect-helper/redirect-helper-dialog.ts", "../../scripts/redirect-helper/redirect-helper.ts"],
  "sourcesContent": ["import type { ApiQueryAllPagesGeneratorParameters } from '../../global-types';\nimport type { LookupElementConfig } from './redirect-target-input-widget';\n\n/**\n * An instance of this class is a category lookup element.\n */\nexport default class CategoryInputWidget extends OO.ui.TextInputWidget {\n    // Utility variables\n    private api = new mw.Api();\n\n    constructor(config: LookupElementConfig) {\n        super(config);\n\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 ApiQueryAllPagesGeneratorParameters)\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(\n                            (page) =>\n                                !page.categories?.some((category) => category.title === 'Category:Wikipedia soft redirected categories'),\n                        )\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 }[]) =>\n        data.map(({ data, label }) => new OO.ui.MenuOptionWidget({ data, label }));\n}\n\nObject.assign(CategoryInputWidget.prototype, OO.ui.mixin.LookupElement.prototype);\n", "import type { ApiComparePagesParams } from 'types-mediawiki/api_params';\n\n/**\n * An instance of this class is a dialog used for showing changes to be made.\n */\nexport default class ChangesDialog 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        ChangesDialog.static.name = 'ShowChangesDialog';\n        ChangesDialog.static.title = 'Changes to be made';\n        ChangesDialog.static.actions = [{ action: 'cancel', label: 'Close', flags: ['safe', 'close'] }];\n    }\n\n    getSetupProcess = () => {\n        return ChangesDialog.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 & {\n                    'fromtext-main': string;\n                    'fromcontentmodel-main': string;\n                    'totext-main': string;\n                    'tocontentmodel-main': string;\n                })\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.close();\n              })\n            : ChangesDialog.super.prototype.getActionProcess.call(this, action);\n    };\n\n    getTeardownProcess = () => {\n        return ChangesDialog.super.prototype.getTeardownProcess.call(this).next(() => {\n            (this as unknown as { $body: JQuery }).$body.empty();\n        });\n    };\n}\n\nObject.assign(ChangesDialog.prototype, OO.ui.ProcessDialog.prototype);\n", "import type { ApiParseParams } from 'types-mediawiki/api_params';\n\n/**\n * An instance of this class is a dialog used for previewing template output.\n */\nexport default class OutputPreviewDialog 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        OutputPreviewDialog.static.name = 'OutputPreviewDialog';\n        OutputPreviewDialog.static.title = 'Redirect categorization templates preview';\n        OutputPreviewDialog.static.actions = [{ action: 'cancel', label: 'Close', flags: ['safe', 'close'] }];\n    }\n\n    getSetupProcess = () => {\n        return OutputPreviewDialog.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.close();\n              })\n            : OutputPreviewDialog.super.prototype.getActionProcess.call(this, action);\n    };\n\n    getTeardownProcess = () => {\n        return OutputPreviewDialog.super.prototype.getTeardownProcess.call(this).next(() => {\n            (this as unknown as { $body: JQuery }).$body.empty();\n        });\n    };\n}\n\nObject.assign(OutputPreviewDialog.prototype, OO.ui.ProcessDialog.prototype);\n", "import type { ApiParseParams } from 'types-mediawiki/api_params';\nimport type { ApiQueryAllPagesGeneratorParameters, PageParseResult } from '../../global-types';\n\nexport interface LookupElementConfig extends OO.ui.TextInputWidget.ConfigOptions, OO.ui.mixin.LookupElement.ConfigOptions {}\n\n/**\n * An instance of this class is a title lookup element.\n */\nexport default class RedirectTargetInputWidget 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\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 ApiQueryAllPagesGeneratorParameters)\n                .catch(() => null)\n                .then(\n                    (\n                        result: {\n                            query: { pages: { title: string; pageprops: { disambiguation?: string }; redirect?: string }[] };\n                        } | null,\n                    ) => {\n                        if (result)\n                            deferred.resolve(\n                                result.query?.pages\n                                    ? result.query.pages\n                                          .filter((page) => page.title !== this.pageTitleParsed.getPrefixedText())\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\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 }[]) =>\n        data.map(({ data, label }) => new OO.ui.MenuOptionWidget({ data, label }));\n}\n\nObject.assign(RedirectTargetInputWidget.prototype, OO.ui.mixin.LookupElement.prototype);\n", "import type {\n    ApiParseParams,\n    ApiQueryInfoParams,\n    ApiQueryPagePropsParams,\n    ApiQueryRevisionsParams,\n    PageTriageApiPageTriageListParams,\n} from 'types-mediawiki/api_params';\nimport type {\n    CategoriesResult,\n    MediaWikiDataError,\n    PageInfoResult,\n    PageParseResult,\n    PageRevisionsResult,\n    PageTriageListResponse,\n    PagepropsResult,\n} from '../../global-types';\nimport CategoryInputWidget from './category-input-widget';\nimport ChangesDialog from './changes-dialog';\nimport OutputPreviewDialog from './output-preview-dialog';\nimport RedirectTargetInputWidget from './redirect-target-input-widget';\n\nexport type RedirectTemplateData = Record<string, { redirect?: true; parameters: RedirectTemplateParameters; aliases: string[] }>;\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\nexport interface TemplateEditorElementInfo {\n    name: string;\n    details: HTMLDetailsElement;\n    parameters: { name: string; aliases: string[]; editor: OO.ui.TextInputWidget }[];\n}\n\n/**\n * An instance of this class handles the dialog portion of redirect-helper script.\n */\nexport default 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!: RedirectTargetInputWidget;\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        {\n            redirectTemplates,\n            contentText,\n            pageTitle,\n            pageTitleParsed,\n        }: { 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    margin-right: auto;\n    margin-bottom: 25px !important;\n    margin-left: auto;\n    background-color: whitesmoke;\n    width: 700px;\n    max-width: calc(100% - 50px);\n    color: #202122;\n}\n\n.redirect-input-layout label {\n    font-weight: bold;\n}\n\n.redirect-helper-redirect-possibilities::after {\n    content: \" (redirect with possibilities)\";\n    font-style: italic;\n}\n\n.redirect-helper-template-parameters-container,\n.redirect-helper-template-parameters-container details {\n    margin-block: 10px;\n    border-radius: 5px;\n    background-color: #e2e2e2;\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    margin-block: 5px;\n    background-color: #d1cece;\n}\n\n#redirect-helper-no-templates-message {\n    padding: 5px;\n}\n\n#redirect-helper-summary-layout {\n    margin-top: 15px;\n    border-top: 1px solid gray;\n    padding-top: 10px;\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.redirect-helper-autofix-button {\n    margin-left: 5px;\n    font-size: 12px;\n}\n\n.redirect-helper-autofix-button a {\n    padding: 3px 4px !important;\n    min-height: unset !important;\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()?.getPrefixedText() ?? '');\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(\n                    mainPageContent,\n                ),\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 RedirectTargetInputWidget({ 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, {\n            label: 'Redirect target:',\n            classes: ['redirect-input-layout'],\n            align: 'top',\n        });\n\n        /* Redirect categorization template selection */\n        this.tagSelect = new OO.ui.MenuTagMultiselectWidget({\n            allowArbitrary: false,\n            allowReordering: false,\n            options: Object.entries(this.redirectTemplates).map(([tag, { redirect }]) => {\n                if (!redirect) return { data: tag, label: tag };\n\n                const label = new OO.ui.HtmlSnippet(`<span class=\"redirect-helper-redirect-possibilities\">${tag}</span>`);\n\n                return { data: tag, label };\n            }),\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            summaryElement.textContent = `Template parameters (${shownTemplateEditors > 0 ? `for ${shownTemplateEditors} template${shownTemplateEditors > 1 ? 's' : ''}` : 'none to show'})`;\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 (none to show)';\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({\n                    placeholder: parameterData.default?.toString(),\n                    required: parameterData.required,\n                });\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\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 (\n                [\n                    'R from birth name',\n                    'R from fictional character',\n                    'R from band member',\n                    'R from member',\n                    'R from person',\n                    'R from personal name',\n                    'R from relative',\n                    'R from spouse',\n                ].some((tag) => this.tagSelect.getValue().includes(tag))\n            ) {\n                // Handling is modified from evad37's \"Rater\"\n\n                if (!name.includes(' '))\n                    return mw.notify(\"redirect-helper wasn't able to determine a sort key different from the current page title!\", {\n                        type: 'warn',\n                    });\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)\n                    mw.notify(\"redirect-helper wasn't able to determine a sort key different from the current page title!\", {\n                        type: 'warn',\n                    });\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(\n                `Default sort key (DEFAULTSORT) (see <a href=\"${mw.util.getUrl('Wikipedia:Categorization#Sort keys')}\" target=\"_blank\">guideline</a>):`,\n            ),\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({\n            allowReordering: false,\n            inputPosition: 'outline',\n            inputWidget: this.categorySelectInput,\n        });\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, {\n            label: 'Categories:',\n            classes: ['redirect-input-layout'],\n            align: 'top',\n        });\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, {\n            id: 'redirect-helper-summary-layout',\n            label: 'Summary:',\n            classes: ['redirect-input-layout'],\n            align: 'top',\n        });\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 OutputPreviewDialog({ 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 ChangesDialog({ 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({\n                content: [new OO.ui.FieldLayout(this.syncTalkCheckbox, { label: 'Sync talk page', align: 'inline' })],\n            });\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({\n                content: [new OO.ui.FieldLayout(this.patrolCheckbox, { label: 'Mark as patrolled', align: 'inline' })],\n            });\n        }\n\n        /* Setup layout */\n        this.submitLayout = new OO.ui.HorizontalLayout({\n            id: 'redirect-helper-submit-layout',\n            items: [\n                this.submitButton,\n                this.showPreviewButton,\n                this.showChangesButton,\n                this.syncTalkCheckboxLayout,\n                this.patrolCheckboxLayout,\n            ].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<HTMLImageElement>('#mwe-pt-mark .mwe-pt-tool-icon');\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 mw.user.getRights();\n            if (!userPermissions.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            let oldTarget = this.oldRedirectTarget?.replaceAll('_', ' ');\n            if (oldTarget) oldTarget = oldTarget[0].toUpperCase() + oldTarget.slice(1);\n\n            const targetChanged = redirectValue !== oldTarget;\n\n            const tagsChanged =\n                this.tagSelect.getValue().some((tag) => !this.oldRedirectTags!.includes(tag as string)) ||\n                this.oldRedirectTags!.some((tag) => !this.tagSelect.getValue().includes(tag));\n\n            let tagArgumentsChanged = false;\n            if (this.oldRedirectTagData) {\n                const tagsWithParameters = Object.entries(this.redirectTemplates).filter(\n                    ([, data]) => Object.entries(data.parameters).length > 0,\n                );\n\n                for (const [tag, data] of tagsWithParameters) {\n                    const tagWasSelected = this.oldRedirectTags!.includes(tag);\n                    if (!tagWasSelected || !this.tagSelect.getValue().includes(tag)) continue;\n\n                    const oldTagData = this.oldRedirectTagData[tag] ?? Object.entries(data.parameters).map(([name]) => [name, '']);\n\n                    const foundTagEditorData = this.templateEditorsInfo.find((template) => template.name === tag)!;\n\n                    for (const parameter of foundTagEditorData.parameters) {\n                        const oldArgument = oldTagData.find((argument) => argument[0] === parameter.name)?.[1] ?? '';\n                        const newArgument = parameter.editor.getValue().trim();\n\n                        if (oldArgument !== newArgument) {\n                            tagArgumentsChanged = true;\n                            break;\n                        }\n                    }\n\n                    if (tagArgumentsChanged) break;\n                }\n            }\n\n            const defaultSortChanged = this.defaultSortInput.getValue().trim() !== this.oldDefaultSort!.replaceAll('_', ' ');\n\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)\n                changes.push(\n                    `${this.tagSelect.getValue().length > 0 && this.oldRedirectTags!.length > 0 ? 'change' : this.tagSelect.getValue().length > 0 ? 'add' : 'remove'} categorization templates`,\n                );\n            if (tagArgumentsChanged) changes.push('change categorization template arguments');\n            if (defaultSortChanged)\n                changes.push(\n                    `${this.defaultSortInput.getValue().trim().length > 0 && this.oldDefaultSort!.replaceAll('_', ' ').length > 0 ? 'change' : this.defaultSortInput.getValue().trim().length > 0 ? 'add' : 'remove'} default sort key`,\n                );\n            if (categoriesChanged)\n                changes.push(\n                    `${this.categorySelect.getValue().length > 0 && this.oldCategories!.length > 0 ? 'change' : this.categorySelect.getValue().length > 0 ? 'add' : 'remove'} categories`,\n                );\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(\n                            `{{\\\\s*[${tagOrRedirect[0].toLowerCase()}${tagOrRedirect[0]}]${tagOrRedirect.slice(1)}\\\\s*(\\\\||}})`,\n                        ).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) =>\n                new RegExp(`{{\\\\s*[${tagOrRedirect[0].toLowerCase()}${tagOrRedirect[0]}]${tagOrRedirect.slice(1)}\\\\s*(\\\\||}})`).test(\n                    this.pageContent,\n                )\n                    ? tagOrRedirect\n                    : null,\n            )\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(\n                        this.pageContent,\n                    );\n\n                    const newTag = Object.entries(this.redirectTemplates).find(([template, tagData]) =>\n                        [template, ...tagData.aliases].includes(tag),\n                    )?.[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            /{{short description\\|.*?}}/i.exec(this.pageContent)?.[0],\n            /{{DISPLAYTITLE:.*?}}/.exec(this.pageContent)?.[0],\n            /{{italic title\\|?.*?}}/i.exec(this.pageContent)?.[0],\n            /{{title language\\|.*?}}/.exec(this.pageContent)?.[0],\n            /{{authority control(\\|.*?)?}}/i.exec(this.pageContent)?.[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) =>\n                    [parameter.name, ...parameter.aliases].includes(parameterName),\n                );\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            title?: string;\n            message: string;\n            autoFixes?: ({ type: 'add' | 'remove'; tag: string } | { type: 'change-target'; target: string })[];\n        }[] = [];\n\n        const destination = this.redirectInput.getValue().trim();\n        const tags = this.tagSelect.getValue() as string[];\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?.getPrefixedText() === this.pageTitleParsed.getPrefixedText())\n            errors.push({ message: 'cannot redirect to itself!' });\n\n        const destinationData = (await this.api\n            .get({\n                action: 'query',\n                formatversion: '2',\n                prop: ['pageprops', 'categories'],\n                titles: destination,\n            } satisfies ApiQueryPagePropsParams)\n            .catch((errorCode: string) => {\n                /* Nonexistent destination */ if (errorCode === 'missingtitle')\n                    errors.push({ title: destination, message: 'does not exist!' });\n                /* Other API error */ else\n                    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({\n            action: 'parse',\n            page: destination,\n            prop: 'sections',\n            redirects: true,\n        } satisfies ApiParseParams)) as PageParseResult;\n\n        /* Double redirects */\n        if (destinationParseResult.parse.redirects?.[0]) {\n            const destinationRedirect =\n                destinationParseResult.parse.redirects[0].to +\n                (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                autoFixes: [{ type: 'change-target', target: destinationRedirect }],\n            });\n        }\n\n        /* Nonexistent section */\n        if (destination.split('#').length > 1) {\n            const validSection = destinationParseResult.parse.sections.find(\n                (section) => section.line.replaceAll(/<\\/?i>/g, '') === destination.split('#')[1],\n            );\n            if (validSection) {\n                if (tags.includes('R to anchor'))\n                    errors.push({\n                        message: 'is tagged as a redirect to an anchor, but it is actually a redirect to a section!',\n                        autoFixes: [\n                            { type: 'add', tag: 'R to section' },\n                            { type: 'remove', tag: 'R to anchor' },\n                        ],\n                    });\n                if (!tags.includes('R to section'))\n                    errors.push({\n                        message: 'is a redirect to a section, but it is not tagged with <code>{{R to section}}</code>!',\n                        autoFixes: [{ type: 'add', tag: 'R to section' }],\n                    });\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!.getPrefixedText(),\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(\n                            /(?<={{\\s*?(?:[Vv](?:isible anchors?|isanc|Anch|anchor|isibleanchor|a)|[Aa](?:nchord|chored|nchor\\+)|[Tt]ext anchor)\\s*?\\|).+?(?=(?<!!|=)}})/g,\n                        )\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                    ...(destinationContent.match(/EpisodeNumber += +\\d+/g)?.map((anchor: string) => `ep${anchor.split('=')[1].trim()}`) ??\n                        []),\n                ];\n                if (anchors.includes(destination.split('#')[1])) {\n                    if (tags.includes('R to section'))\n                        errors.push({\n                            message: 'is tagged as a redirect to a section, but it is actually a redirect to an anchor!',\n                            autoFixes: [\n                                { type: 'add', tag: 'R to anchor' },\n                                { type: 'remove', tag: 'R to section' },\n                            ],\n                        });\n                    if (!tags.includes('R to anchor'))\n                        errors.push({\n                            message: 'is a redirect to an anchor, but it is not tagged with <code>{{R from anchor}}</code>!',\n                            autoFixes: [{ type: 'add', tag: 'R to anchor' }],\n                        });\n                } else\n                    errors.push({\n                        message: `is a redirect to <a href=\"${mw.util.getUrl(destination)}\" target=\"_blank\">${destination}</a>, but that section or anchor does not exist!`,\n                        autoFixes: [{ type: 'change-target', target: destination.split('#')[0] }],\n                    });\n            }\n        }\n\n        /* Improperly tagged as redirect to section/anchor */\n        if (destination.split('#').length === 1)\n            for (const tag of ['R to section', 'R to anchor'])\n                if (tags.includes(tag))\n                    errors.push({\n                        message: `is not a redirect to a section/anchor, but it is tagged with <code>{{${tag}}}</code>!`,\n                        autoFixes: [{ type: 'remove', tag }],\n                    });\n\n        const targetIsDisambiguationPage = !!(\n            destinationData!.query.pages[0].pageprops && 'disambiguation' in destinationData!.query.pages[0].pageprops\n        );\n        const targetIsSurnameList = !!destinationData!.query.pages[0].categories?.some(\n            (category) => category.title === 'Category:Surnames',\n        );\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({\n                message: 'is a redirect to a disambiguation page, but it is not tagged with a disambiguation categorization template!',\n            });\n\n        if (destinationData!.query.pages[0].pageprops && !targetIsDisambiguationPage) {\n            /* Improperly tagged as redirect to disambiguation page */\n            if (\n                (!targetIsSurnameList && (taggedAsRedirectToDisambiguationPage || taggedAsRedirectToSurnameList)) ||\n                (targetIsSurnameList && taggedAsRedirectToDisambiguationPage)\n            )\n                errors.push({\n                    message: 'is not a redirect to a disambiguation page, but it is tagged with a disambiguation categorization template!',\n                    autoFixes: [...toDisambiguationPageTags, ...toSurnameListTags].map((tag) => ({ type: 'remove', tag })),\n                });\n\n            /* Redirect to surname list without template */\n            if (targetIsSurnameList && !taggedAsRedirectToSurnameList)\n                errors.push({\n                    message: 'is a redirect to a surname list, but it is not tagged with a correct disambiguation categorization template!',\n                });\n        }\n\n        /* {{R to disambiguation page}} without \" (disambiguation)\" at end of title */\n        if (\n            targetIsDisambiguationPage &&\n            tags.includes('R to disambiguation page') &&\n            !this.pageTitleParsed.getMainText().endsWith(' (disambiguation)')\n        )\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                autoFixes: [{ type: 'remove', tag: 'R to disambiguation page' }],\n            });\n\n        /* Tagged with a protection template */\n        for (const template of ['R protected', 'R semi-protected', 'R extended-protected', 'R template-protected', 'R fully protected'])\n            if (tags.includes(template))\n                errors.push({\n                    message: `is tagged with unnecessarily tagged with <code>{{${template}}}</code> which will be duplicated by the redirect category shell!`,\n                    autoFixes: [{ type: 'remove', tag: template }],\n                });\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({\n                message: \"is linked to a Wikidata item but it isn't tagged with <code>{{R with Wikidata item}}</code>!\",\n                autoFixes: [{ type: 'add', tag: 'R with Wikidata item' }],\n            });\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({\n                message: 'is tagged with <code>{{R with Wikidata item}}</code> but it is not actually linked to a Wikidata item!',\n                autoFixes: [{ type: 'remove', tag: 'R with Wikidata item' }],\n            });\n\n        /* Missing tag required parameter */\n        for (const tag of tags) {\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({\n                        message: `is tagged with <code>{{${tag}}}</code> but it is missing the required parameter <code>${parameterName}</code>!`,\n                    });\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({\n                title: this.pageTitleParsed.getTalkPage()!.getPrefixedText(),\n                message: 'exists, but is not a redirect!',\n            });\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, autoFixes } 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({\n                    type: 'error',\n                    classes: ['redirect-helper-warning'],\n                    inline: true,\n                    label,\n                });\n\n                if (autoFixes) {\n                    const autoFixButton = new OO.ui.ButtonWidget({\n                        label: 'Perform auto-fix',\n                        flags: ['progressive'],\n                        classes: ['redirect-helper-autofix-button'],\n                    });\n                    autoFixButton.on('click', () => {\n                        const tags = this.tagSelect.getValue() as string[];\n\n                        for (const autoFix of autoFixes) {\n                            if (autoFix.type === 'add' && !tags.includes(autoFix.tag)) this.tagSelect.addTag(autoFix.tag, autoFix.tag);\n\n                            if (autoFix.type === 'remove' && tags.includes(autoFix.tag)) this.tagSelect.removeTagByData(autoFix.tag);\n\n                            if (autoFix.type === 'change-target') this.redirectInput.setValue(autoFix.target);\n                        }\n\n                        warningMessage.$element[0].style.textDecoration = 'line-through 2px black';\n                        autoFixButton.$element[0].remove();\n                    });\n\n                    warningMessage.$element[0].querySelector('.oo-ui-labelElement-label')!.append(autoFixButton.$element[0]);\n                }\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 =\n            (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(\n                this.parsedDestination!.getTalkPage()!.getPrefixedText(),\n                fromMove ? ['R from move'] : [],\n                undefined,\n                undefined,\n                [],\n            );\n\n            const talkResult = await this.editOrCreate(\n                this.pageTitleParsed.getTalkPage()!.getPrefixedText(),\n                output,\n                'Syncing redirect from main page' + this.scriptAdvert,\n            );\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<HTMLButtonElement>('#mwe-pt-mark-as-reviewed-button');\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(\n                            `Error patrolling ${this.pageTitle} via API: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`,\n                            { type: 'error' },\n                        );\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(\n        target: string,\n        tags: string[],\n        strayText: string | undefined,\n        defaultSort: string | undefined,\n        categories: string[],\n    ) {\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 lastNumberParameterIndex = foundArgumentEditor.parameters.findLastIndex(\n                (parameter, index) => parameter.name === (index + 1).toString() && parameter.editor.getValue().trim(),\n            );\n\n            const mappedArguments = foundArgumentEditor.parameters\n                .map((parameter, index) => {\n                    const value = parameter.editor.getValue().trim();\n                    if (!value && index > lastNumberParameterIndex) 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})`, {\n                            type: 'error',\n                        });\n                    });\n                else {\n                    mw.notify(`Error editing or creating ${title}: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, {\n                        type: 'error',\n                    });\n                    return null;\n                }\n            });\n    }\n}\n", "import type { ApiQueryInfoParams, ApiQueryRevisionsParams } from 'types-mediawiki/api_params';\nimport type { PageInfoResult, PageRevisionsResult } from '../../global-types';\nimport type { RedirectTemplateData } from './redirect-helper-dialog';\n\nconst dependencies = [\n    'mediawiki.util',\n    'oojs-ui-core',\n    'oojs-ui-widgets',\n    'oojs-ui-windows',\n    'oojs-ui.styles.icons-content',\n    'oojs-ui.styles.icons-editing-core',\n];\n\nmw.loader.using(dependencies, async () => {\n    const { default: RedirectHelperDialog } = await import('./redirect-helper-dialog'); // eslint-disable-line @typescript-eslint/naming-convention\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<HTMLDivElement>('#mw-content-text')!;\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({\n                action: 'query',\n                formatversion: '2',\n                prop: 'info',\n                titles: this.pageTitle,\n            } satisfies ApiQueryInfoParams)) as PageInfoResult;\n\n            const dialogInfo = {\n                redirectTemplates: this.redirectTemplates,\n                contentText: this.contentText,\n                pageTitle: this.pageTitle,\n                pageTitleParsed: this.pageTitleParsed,\n            };\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({\n                    id: 'create-redirect-button',\n                    label: 'Create redirect',\n                    icon: 'articleRedirect',\n                    flags: ['progressive'],\n                });\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(\n                    mw.config.get('skin') === 'minerva' ? 'p-tb' : 'p-cactions',\n                    '#',\n                    'Redirect page',\n                    'redirect-helper',\n                )!;\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    new RedirectHelper().run();\n});\n"],
  "mappings": ";;;gJAAA,IAMqBA,EANrBC,EAAAC,EAAA,kBAMqBF,EAArB,cAAiD,GAAG,GAAG,eAAgB,CAE3D,IAAM,IAAI,GAAG,IAErB,YAAYG,EAA6B,CACrC,MAAMA,CAAM,EAEZ,GAAG,GAAG,MAAM,cAAc,KAAK,KAA8CA,CAAM,CACvF,CAEA,iBAAmB,IAAM,CACrB,IAAMC,EAAQ,KAAK,SAAS,EACtBC,EAAW,EAAE,SAAS,EAEvBD,GAAOC,EAAS,QAAQ,CAAC,CAAC,EAE/B,IAAMC,EAAc,GAAG,MAAM,YAAYF,CAAK,EAE9C,YAAK,IACA,IAAI,CACD,OAAQ,QACR,cAAe,IACf,SAAU,GACV,aAAc,GACd,UAAWE,GAAa,YAAY,GAAKF,EACzC,UAAW,WACX,KAAM,YACV,CAA+C,EAC9C,MAAM,IAAM,IAAI,EAChB,KAAMG,GAA+F,CAClG,GAAIA,GAAQ,OAAO,MAAO,CACtB,IAAMC,EAAQD,EAAO,MAAM,MACtB,OACIE,GACG,CAACA,EAAK,YAAY,KAAMC,GAAaA,EAAS,QAAU,+CAA+C,CAC/G,EACC,IAAKD,GAAS,CACX,IAAME,EAAwBF,EAAK,MAAM,MAAM,GAAG,EAAE,CAAC,EAErD,MAAO,CAAE,KAAME,EAAuB,MAAOA,CAAsB,CACvE,CAAC,EAEL,KAAK,KAAK,iBAAkBH,CAAK,EAEjCH,EAAS,QAAQG,CAAK,CAC1B,MAAOH,EAAS,QAAQ,CAAC,CAAC,CAC9B,CAAC,EAEEA,EAAS,QAAQ,CAAE,OAAQ,CAAC,CAAE,CAAC,CAC1C,EAEA,+BAAqCO,GAAqCA,GAAY,CAAC,EAEvF,6BAAgCC,GAC5BA,EAAK,IAAI,CAAC,CAAE,KAAAA,EAAM,MAAAC,CAAM,IAAM,IAAI,GAAG,GAAG,iBAAiB,CAAE,KAAAD,EAAM,MAAAC,CAAM,CAAC,CAAC,CACjF,EAEA,OAAO,OAAOd,EAAoB,UAAW,GAAG,GAAG,MAAM,cAAc,SAAS,IC/DhF,IAKqBe,EALrBC,EAAAC,EAAA,kBAKqBF,EAArB,MAAqBG,UAAsB,GAAG,GAAG,aAAc,CAEnD,IAAM,IAAI,GAAG,IAErB,YAAYC,EAA2C,CACnD,MAAMA,CAAM,EAEZD,EAAc,OAAO,KAAO,oBAC5BA,EAAc,OAAO,MAAQ,qBAC7BA,EAAc,OAAO,QAAU,CAAC,CAAE,OAAQ,SAAU,MAAO,QAAS,MAAO,CAAC,OAAQ,OAAO,CAAE,CAAC,CAClG,CAEA,gBAAkB,IACPA,EAAc,MAAM,UAAU,gBAAgB,KAAK,IAAI,EAAE,KAAK,IAAM,CACvE,GAAM,CAACE,EAASC,CAAO,EAAI,KAAK,QAAQ,EAExC,OAAO,KAAK,IACP,KAAK,CACF,OAAU,UACV,cAAiB,IACjB,KAAQ,CAAC,MAAM,EACf,UAAa,OACb,gBAAiBD,EACjB,wBAAyB,WACzB,QAAW,OACX,cAAeC,EACf,sBAAuB,UAC3B,CAKC,EACA,KAAMC,GAAW,CACd,IAAMC,EAAcD,EAAyC,QAAQ,KAE/DE,EAAmB,IAAI,GAAG,GAAG,cAAc,CAAE,KAAM,UAAW,MAAO,qBAAsB,CAAC,EAE5FC,EAAc,IAAI,GAAG,GAAG,YAAY,CAAE,OAAQ,GAAM,SAAU,EAAM,CAAC,EAC3EA,EAAY,SAAS,OACjBF,EACM;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,UASpBA,CAAU;AAAA;AAAA,UAGUC,EAAiB,SAAS,CAAC,CACrC,EAEC,KAAsC,MAAM,OAAOC,EAAY,QAAQ,CAC5E,CAAC,CACT,CAAC,EAGL,iBAAoBC,GACTA,EACD,IAAI,GAAG,GAAG,QAAQ,IAAM,CACpB,KAAK,MAAM,CACf,CAAC,EACDR,EAAc,MAAM,UAAU,iBAAiB,KAAK,KAAMQ,CAAM,EAG1E,mBAAqB,IACVR,EAAc,MAAM,UAAU,mBAAmB,KAAK,IAAI,EAAE,KAAK,IAAM,CACzE,KAAsC,MAAM,MAAM,CACvD,CAAC,CAET,EAEA,OAAO,OAAOH,EAAc,UAAW,GAAG,GAAG,cAAc,SAAS,ICjFpE,IAKqBY,EALrBC,EAAAC,EAAA,kBAKqBF,EAArB,MAAqBG,UAA4B,GAAG,GAAG,aAAc,CAEzD,IAAM,IAAI,GAAG,IAGb,gBAER,YAAYC,EAA2CC,EAA2B,CAC9E,MAAMD,CAAM,EAEZ,KAAK,gBAAkBC,EAEvBF,EAAoB,OAAO,KAAO,sBAClCA,EAAoB,OAAO,MAAQ,4CACnCA,EAAoB,OAAO,QAAU,CAAC,CAAE,OAAQ,SAAU,MAAO,QAAS,MAAO,CAAC,OAAQ,OAAO,CAAE,CAAC,CACxG,CAEA,gBAAkB,IACPA,EAAoB,MAAM,UAAU,gBAAgB,KAAK,IAAI,EAAE,KAAK,IAChE,KAAK,IACP,KAAK,CACF,OAAQ,QACR,cAAe,IACf,aAAc,WACd,KAAM,CAAC,OAAQ,gBAAgB,EAC/B,MAAO,KAAK,gBAAgB,cAAc,EAC1C,KAAM,KAAK,QAAQ,CACvB,CAA0B,EACzB,KAAMG,GAAW,CACd,IAAMC,EAAeD,EAAuC,MAAM,KAC5DE,EAAqBF,EAAiD,MAAM,eAE5EG,EAAc,IAAI,GAAG,GAAG,YAAY,CAAE,OAAQ,GAAM,SAAU,EAAM,CAAC,EAC3EA,EAAY,SAAS,OAAOF,EAAaC,CAAiB,EAEzD,KAAsC,MAAM,OAAOC,EAAY,QAAQ,CAC5E,CAAC,CACR,EAGL,iBAAoBC,GACTA,EACD,IAAI,GAAG,GAAG,QAAQ,IAAM,CACpB,KAAK,MAAM,CACf,CAAC,EACDP,EAAoB,MAAM,UAAU,iBAAiB,KAAK,KAAMO,CAAM,EAGhF,mBAAqB,IACVP,EAAoB,MAAM,UAAU,mBAAmB,KAAK,IAAI,EAAE,KAAK,IAAM,CAC/E,KAAsC,MAAM,MAAM,CACvD,CAAC,CAET,EAEA,OAAO,OAAOH,EAAoB,UAAW,GAAG,GAAG,cAAc,SAAS,IC5D1E,IAQqBW,EARrBC,EAAAC,EAAA,kBAQqBF,EAArB,cAAuD,GAAG,GAAG,eAAgB,CAEjE,IAAM,IAAI,GAAG,IAGb,gBAER,YAAYG,EAA6BC,EAA2B,CAChE,MAAMD,CAAM,EAEZ,GAAG,GAAG,MAAM,cAAc,KAAK,KAA8CA,CAAM,EAEnF,KAAK,gBAAkBC,CAC3B,CAEA,iBAAmB,IAAM,CACrB,IAAMC,EAAQ,KAAK,SAAS,EACtBC,EAAW,EAAE,SAAS,EAE5B,GAAI,CAACD,EAAOC,EAAS,QAAQ,CAAC,CAAC,UACtBD,EAAM,SAAS,GAAG,EAAG,CAC1B,IAAME,EAAQF,EAAM,MAAM,GAAG,EAAE,CAAC,EAEhC,KAAK,IACA,IAAI,CAAE,OAAQ,QAAS,KAAME,EAAO,KAAM,WAAY,UAAW,EAAK,CAA0B,EAChG,MAAM,IAAM,IAAI,EAChB,KAAMC,GAAmC,CACtC,GAAIA,EAAQ,CACR,IAAMC,EAAkBD,EAAO,MAAM,SAAS,OAAQE,GAClDA,EAAQ,KACH,YAAY,EACZ,WAAW,UAAW,EAAE,EACxB,WAAWL,EAAM,MAAM,GAAG,EAAE,CAAC,EAAE,YAAY,CAAC,CACrD,EACAC,EAAS,QACLG,EAAgB,IAAKC,IAAa,CAC9B,KAAM,GAAGF,EAAO,MAAM,KAAK,IAAIE,EAAQ,KAAK,WAAW,UAAW,EAAE,CAAC,GACrE,MAAO,GAAGF,EAAO,MAAM,KAAK,IAAIE,EAAQ,KAAK,WAAW,UAAW,EAAE,CAAC,EAC1E,EAAE,CACN,CACJ,MAAOJ,EAAS,QAAQ,CAAC,CAAC,CAC9B,CAAC,CACT,KAAO,CACH,IAAMK,EAAc,GAAG,MAAM,YAAYN,CAAK,EAE9C,KAAK,IACA,IAAI,CACD,OAAQ,QACR,cAAe,IACf,SAAU,GACV,aAAcM,GAAa,eAAe,GAAK,EAC/C,UAAWA,GAAa,YAAY,GAAKN,EACzC,UAAW,WACX,KAAM,CAAC,OAAQ,WAAW,CAC9B,CAA+C,EAC9C,MAAM,IAAM,IAAI,EAChB,KAEOG,GAGC,CACGA,EACAF,EAAS,QACLE,EAAO,OAAO,MACRA,EAAO,MAAM,MACR,OAAQI,GAASA,EAAK,QAAU,KAAK,gBAAgB,gBAAgB,CAAC,EACtE,IAAKA,IAAU,CACZ,KAAMA,EAAK,MACX,MAAO,IAAI,GAAG,GAAG,YACb,GAAGA,EAAK,KAAK,GAAGA,EAAK,WAAa,mBAAoBA,EAAK,UAAY,2BAA6B,EAAE,GAAG,aAAcA,EAAO,qBAAuB,EAAE,EAC3J,CACJ,EAAE,EACN,CAAC,CACX,EACCN,EAAS,QAAQ,CAAC,CAAC,CAC5B,CACJ,CACR,CAEA,OAAOA,EAAS,QAAQ,CAAE,OAAQ,CAAC,CAAE,CAAC,CAC1C,EAEA,+BAAqCO,GAAqCA,GAAY,CAAC,EAEvF,6BAAgCC,GAC5BA,EAAK,IAAI,CAAC,CAAE,KAAAA,EAAM,MAAAC,CAAM,IAAM,IAAI,GAAG,GAAG,iBAAiB,CAAE,KAAAD,EAAM,MAAAC,CAAM,CAAC,CAAC,CACjF,EAEA,OAAO,OAAOf,EAA0B,UAAW,GAAG,GAAG,MAAM,cAAc,SAAS,ICjGtF,IAAAgB,EAAA,GAAAC,EAAAD,EAAA,aAAAE,IAAA,IA8CqBA,EA9CrBC,EAAAC,EAAA,kBAgBAC,IACAC,IACAC,IACAC,IA2BqBN,EAArB,KAA0C,CAE9B,IAAM,IAAI,GAAG,IACb,cAAgB,mEAChB,aAAe,kEAGf,kBACA,YACA,UACA,gBAEA,OAGA,WAAa,GAEb,UACA,mBACA,cACA,oBACA,UACA,gBACA,yBACA,oBAAmD,CAAC,EACpD,eACA,oBACA,qBACA,iBACA,yBACA,uBACA,aACA,mBACA,aACA,kBACA,kBACA,iBACA,uBACA,eACA,qBACA,aAEA,SAEA,YAAc,GAEd,kBACA,gBACA,mBACA,eACA,cACA,aAEA,kBAER,YACI,CACI,kBAAAO,EACA,YAAAC,EACA,UAAAC,EACA,gBAAAC,CACJ,EACAC,EACF,CACE,KAAK,kBAAoBJ,EACzB,KAAK,YAAcC,EACnB,KAAK,UAAYC,EACjB,KAAK,gBAAkBC,EAEvB,KAAK,OAASC,CAClB,CAKA,MAAM,MAAO,CACT,GAAG,KAAK,OAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAwErB,EAEM,GAAG,OAAO,WAAW,gFAAgF,EAGrG,KAAK,UAAY,IAAI,GAAG,GAAG,YAAY,CAAE,GAAI,sBAAuB,OAAQ,GAAM,SAAU,GAAO,OAAQ,EAAK,CAAC,EAE7G,KAAK,gBAAgB,WAAW,IACV,MAAM,KAAK,IAAI,IAAI,CACrC,OAAQ,QACR,cAAe,IACf,KAAM,OACN,OAAQ,KAAK,gBAAgB,eAAe,EAAG,gBAAgB,CACnE,CAA8B,GAEb,MAAM,MAAM,CAAC,EAAE,UAAU,MAAM,KAAK,uBAAuB,EAGhF,KAAK,kBAAkB,EACvB,MAAM,KAAK,mBAAmB,EAG9B,KAAK,UAAU,SAAS,CAAC,EAAE,OACvB,GAAI,CACA,KAAK,oBAAoB,WAAW,CAAC,EACrC,KAAK,oBAAoB,SAAS,CAAC,EACnC,KAAK,gBAAgB,SAAS,CAAC,EAC/B,KAAK,yBACL,KAAK,uBAAuB,SAAS,CAAC,EACtC,KAAK,qBAAqB,SAAS,CAAC,EACpC,KAAK,mBAAmB,SAAS,CAAC,EAClC,KAAK,aAAa,SAAS,CAAC,CAChC,EAAE,OAAO,OAAO,CACpB,EAEA,KAAK,YAAY,QAAQ,KAAK,UAAU,SAAS,CAAC,CAAC,EAE/C,KAAK,QAAQ,KAAK,iBAAiB,CAC3C,CAKA,MAAc,wBAAyB,CACnC,IAAMC,EAAkB,MAAM,KAAK,eAAe,KAAK,gBAAgB,eAAe,EAAG,gBAAgB,CAAC,EAE1G,KAAK,mBAAqB,IAAI,GAAG,GAAG,aAAa,CAAE,MAAO,sBAAuB,KAAM,OAAQ,MAAO,CAAC,aAAa,CAAE,CAAC,EACvH,KAAK,mBAAmB,GAAG,QAAS,IAAM,CACtC,IAAMC,EAAS,KAAK,cAAc,KAAKD,CAAe,IAAI,CAAC,EAC3D,GAAI,CAACC,EAAQ,OAAO,GAAG,OAAO,qCAAsC,CAAE,KAAM,OAAQ,CAAC,EAErF,KAAK,cAAc,SAAS,GAAG,MAAM,YAAYA,CAAM,GAAG,YAAY,GAAG,gBAAgB,GAAK,EAAE,EAC/E,CAAC,cAAe,GAAG,KAAK,kBAAkB,aAAa,EAAE,OAAO,EAAE,KAAMC,GACrF,IAAI,OAAO,UAAUA,EAAc,CAAC,EAAE,YAAY,CAAC,GAAGA,EAAc,CAAC,CAAC,IAAIA,EAAc,MAAM,CAAC,CAAC,cAAc,EAAE,KAC5GF,CACJ,CACJ,GACc,KAAK,UAAU,SAAS,CAAC,aAAa,CAAC,CACzD,CAAC,CACL,CAKQ,mBAAoB,CAExB,KAAK,cAAgB,IAAIG,EAA0B,CAAE,YAAa,mBAAoB,SAAU,EAAK,EAAG,KAAK,eAAe,EAC5H,KAAK,cAAc,GAAG,SAAU,IAAM,CAClC,IAAIC,EAAQ,KAAK,cAAc,SAAS,EACxCA,EAAQA,EAAM,QAAQ,IAAI,OAAO,mBAAmB,GAAG,OAAO,IAAI,UAAU,EAAE,QAAQ,SAAU,EAAE,CAAC,QAAQ,EAAG,EAAE,EAChHA,EAAQA,EAAM,QAAQ,KAAM,EAAE,EAE1BA,EAAM,OAAS,GACf,KAAK,cAAc,SAASA,EAAM,CAAC,EAAE,YAAY,EAAIA,EAAM,MAAM,CAAC,EAAE,WAAW,IAAK,GAAG,CAAC,EACxF,KAAK,yBAAyB,YAAY,EAAK,EAC/C,KAAK,aAAa,YAAY,EAAK,EACnC,KAAK,kBAAkB,YAAY,EAAK,EACxC,KAAK,kBAAkB,YAAY,EAAK,IAExC,KAAK,yBAAyB,YAAY,EAAI,EAC9C,KAAK,aAAa,YAAY,EAAI,EAClC,KAAK,kBAAkB,YAAY,EAAI,EACvC,KAAK,kBAAkB,YAAY,EAAI,GAG3C,KAAK,cAAc,EACnB,KAAK,aAAa,SAAS,QAAQ,EACnC,KAAK,WAAa,EACtB,CAAC,EAED,KAAK,oBAAsB,IAAI,GAAG,GAAG,YAAY,KAAK,cAAe,CACjE,MAAO,mBACP,QAAS,CAAC,uBAAuB,EACjC,MAAO,KACX,CAAC,EAGD,KAAK,UAAY,IAAI,GAAG,GAAG,yBAAyB,CAChD,eAAgB,GAChB,gBAAiB,GACjB,QAAS,OAAO,QAAQ,KAAK,iBAAiB,EAAE,IAAI,CAAC,CAACC,EAAK,CAAE,SAAAC,CAAS,CAAC,IAAM,CACzE,GAAI,CAACA,EAAU,MAAO,CAAE,KAAMD,EAAK,MAAOA,CAAI,EAE9C,IAAME,EAAQ,IAAI,GAAG,GAAG,YAAY,wDAAwDF,CAAG,SAAS,EAExG,MAAO,CAAE,KAAMA,EAAK,MAAAE,CAAM,CAC9B,CAAC,CACL,CAAC,EACA,KAAK,UAAU,QAAQ,EAA2C,WAAa,YAChF,KAAK,UAAU,GAAG,SAAU,IAAM,CAC9B,IAAMC,EAAc,KAAK,UAAU,SAAS,EAAe,KAAK,CAACC,EAAGC,IAAMD,EAAE,YAAY,EAAE,cAAcC,EAAE,YAAY,CAAC,CAAC,EAEnH,KAAK,UAAU,SAAS,EAAe,KAAK,GAAG,IAAMF,EAAW,KAAK,GAAG,GAAG,KAAK,UAAU,SAASA,CAAU,EAElH,KAAK,cAAc,EACnB,KAAK,aAAa,SAAS,QAAQ,EACnC,KAAK,WAAa,GAElB,QAAWG,KAAc,KAAK,oBAAqBA,EAAW,QAAQ,MAAM,QAAU,OAEtF,IAAIC,EAAuB,EAC3B,QAAWP,KAAO,KAAK,UAAU,SAAS,EAAe,CACrD,IAAMM,EAAa,KAAK,oBAAoB,KAAMA,GAAeA,EAAW,OAASN,CAAG,EAEpFM,IACAA,EAAW,QAAQ,MAAM,QAAU,QACnCC,IAER,CAEAC,EAAe,YAAc,wBAAwBD,EAAuB,EAAI,OAAOA,CAAoB,YAAYA,EAAuB,EAAI,IAAM,EAAE,GAAK,cAAc,IAE7KE,EAAmB,MAAM,QAAUF,EAAuB,EAAI,OAAS,OAC3E,CAAC,EAED,KAAK,gBAAkB,IAAI,GAAG,GAAG,YAAY,KAAK,UAAW,CACzD,MAAO,qCACP,QAAS,CAAC,uBAAuB,EACjC,MAAO,KACX,CAAC,EAGD,KAAK,yBAA2B,SAAS,cAAc,SAAS,EAChE,KAAK,yBAAyB,UAAU,IAAI,+CAA+C,EAE3F,IAAMC,EAAiB,SAAS,cAAc,SAAS,EACvDA,EAAe,YAAc,qCAC7B,KAAK,yBAAyB,OAAOA,CAAc,EAEnD,OAAW,CAACE,EAAcC,CAAY,IAAK,OAAO,QAAQ,KAAK,iBAAiB,EAAG,CAC/E,IAAMC,EAAa,OAAO,QAAQD,EAAa,UAAU,EACzD,GAAIC,EAAW,SAAW,EAAG,SAE7B,IAAMC,EAAU,SAAS,cAAc,SAAS,EAChDA,EAAQ,MAAM,QAAU,OAExB,IAAMC,EAAU,SAAS,cAAc,SAAS,EAChDA,EAAQ,YAAcJ,EACtBG,EAAQ,OAAOC,CAAO,EAEtB,IAAMC,EAAyC,CAAE,KAAML,EAAc,QAAAG,EAAS,WAAY,CAAC,CAAE,EAE7F,OAAW,CAACG,EAAeC,CAAa,IAAKL,EAAY,CACrD,IAAMM,EAAQ,IAAI,GAAG,GAAG,gBAAgB,CACpC,YAAaD,EAAc,SAAS,SAAS,EAC7C,SAAUA,EAAc,QAC5B,CAAC,EACDC,EAAM,GAAG,SAAU,IAAM,CACrB,KAAK,cAAc,EACnB,KAAK,aAAa,SAAS,QAAQ,EACnC,KAAK,WAAa,EACtB,CAAC,EAED,IAAMC,EAAc,IAAI,GAAG,GAAG,YAAYD,EAAO,CAC7C,MAAO,IAAI,GAAG,GAAG,YACb,GAAGF,CAAa,GAAG,CAACC,EAAc,OAASD,EAAc,YAAY,IAAMC,EAAc,OAAO,YAAY,EAAI,GAAK,KAAKA,EAAc,KAAK,GAAG,GAAGA,EAAc,YAAc,KAAKA,EAAc,WAAW,IAAM,EAAE,WAAWA,EAAc,IAAI,KAAKA,EAAc,UAAY,eAAiB,EAAE,GAAGA,EAAc,QAAU,eAAeA,EAAc,OAAO,KAAO,EAAE,EAChX,EACA,MAAO,QACX,CAAC,EACDJ,EAAQ,OAAOM,EAAY,SAAS,CAAC,CAAC,EAEtCJ,EAAY,WAAW,KAAK,CAAE,KAAMC,EAAe,QAASC,EAAc,QAAS,OAAQC,CAAM,CAAC,CACtG,CAEA,KAAK,yBAAyB,OAAOL,CAAO,EAE5C,KAAK,oBAAoB,KAAKE,CAAW,CAC7C,CAEA,IAAMN,EAAqB,SAAS,cAAc,KAAK,EACvDA,EAAmB,GAAK,uCACxBA,EAAmB,YAAc,2CAEjC,KAAK,yBAAyB,OAAOA,CAAkB,EAGvD,KAAK,iBAAmB,IAAI,GAAG,GAAG,gBAClC,KAAK,iBAAiB,GAAG,SAAU,IAAM,CACrC,IAAMV,EAAQ,KAAK,iBAAiB,SAAS,EAEzCA,EAAM,OAAS,GAAG,KAAK,iBAAiB,SAASA,EAAM,WAAW,IAAK,GAAG,CAAC,EAE/E,KAAK,cAAc,EACnB,KAAK,aAAa,SAAS,QAAQ,EACnC,KAAK,WAAa,EACtB,CAAC,EAED,KAAK,yBAA2B,IAAI,GAAG,GAAG,aAAa,CAAE,KAAM,QAAS,MAAO,UAAW,SAAU,EAAK,CAAC,EAC1G,KAAK,yBAAyB,GAAG,QAAS,IAAM,CAC5C,IAAIqB,EAAO,KAAK,gBAAgB,YAAY,EAAE,QAAQ,WAAY,EAAE,EAEpE,GACI,CACI,oBACA,6BACA,qBACA,gBACA,gBACA,uBACA,kBACA,eACJ,EAAE,KAAMpB,GAAQ,KAAK,UAAU,SAAS,EAAE,SAASA,CAAG,CAAC,EACzD,CAGE,GAAI,CAACoB,EAAK,SAAS,GAAG,EAClB,OAAO,GAAG,OAAO,6FAA8F,CAC3G,KAAM,MACV,CAAC,EAEL,IAAIC,EAAqB,GACzB,GAAI,uBAAuB,KAAKD,CAAI,IAChCC,EAAqBD,EAAK,MAAMA,EAAK,YAAY,GAAG,CAAC,EACrDA,EAAOA,EAAK,MAAM,EAAGA,EAAK,YAAY,GAAG,CAAC,EACtC,CAACA,EAAK,SAAS,GAAG,GAAG,OAAOA,EAAOC,EAG3C,IAAMC,EAAWF,EACZ,MAAMA,EAAK,YAAY,GAAG,EAAI,CAAC,EAC/B,QAAQ,KAAM,EAAE,EAChB,QAAQ,KAAM,GAAG,EAChBG,EAAaH,EAAK,MAAM,EAAGA,EAAK,YAAY,GAAG,CAAC,EAEtD,KAAK,iBAAiB,SAASE,EAAW,KAAOC,EAAaF,CAAkB,CACpF,KAAO,CACH,IAAIG,EAAUJ,EAAK,WAAW,MAAO,QAAQ,EAAE,WAAW,IAAK,KAAK,EAEpE,QAAWK,IAAkB,CAAC,KAAM,IAAK,KAAK,EAC1C,GAAID,EAAQ,WAAWC,EAAiB,GAAG,EAAG,CAC1CD,EAAUA,EAAQ,MAAMC,EAAe,OAAS,CAAC,EAAI,KAAOA,EAC5D,KACJ,CAEAD,IAAYJ,EACZ,GAAG,OAAO,6FAA8F,CACpG,KAAM,MACV,CAAC,EACA,KAAK,iBAAiB,SAASI,CAAO,CAC/C,CACJ,CAAC,EAED,KAAK,uBAAyB,IAAI,GAAG,GAAG,kBAAkB,KAAK,iBAAkB,KAAK,yBAA0B,CAC5G,MAAO,IAAI,GAAG,GAAG,YACb,gDAAgD,GAAG,KAAK,OAAO,oCAAoC,CAAC,mCACxG,EACA,QAAS,CAAC,uBAAuB,EACjC,MAAO,KACX,CAAC,EAGD,KAAK,oBAAsB,IAAIE,EAAoB,CAAE,YAAa,qBAAsB,CAAC,EACzF,KAAK,oBAAoB,GAAG,SAAU,IAAM,CACxC,IAAI3B,EAAQ,KAAK,oBAAoB,SAAS,EAC9CA,EAAQA,EAAM,QAAQ,IAAI,OAAO,mBAAmB,GAAG,OAAO,IAAI,UAAU,EAAE,QAAQ,SAAU,EAAE,CAAC,QAAQ,EAAG,EAAE,EAChHA,EAAQA,EAAM,QAAQ,aAAc,EAAE,EAElCA,EAAM,OAAS,GAAG,KAAK,oBAAoB,SAASA,EAAM,CAAC,EAAE,YAAY,EAAIA,EAAM,MAAM,CAAC,EAAE,WAAW,IAAK,GAAG,CAAC,CACxH,CAAC,EACD,KAAK,oBAAoB,GAAG,iBAAmB4B,GAA6C,CACxF,QAAWC,KAAQD,EAAO,KAAK,eAAe,gBAAgBC,EAAK,IAAI,CAC3E,CAAC,EACD,KAAK,eAAiB,IAAI,GAAG,GAAG,qBAAqB,CACjD,gBAAiB,GACjB,cAAe,UACf,YAAa,KAAK,mBACtB,CAAC,EACD,KAAK,eAAe,GAAG,SAAU,IAAM,CACnC,IAAMzB,EAAc,KAAK,eAAe,SAAS,EAAe,KAAK,CAAC,EAAGE,IAAM,EAAE,YAAY,EAAE,cAAcA,EAAE,YAAY,CAAC,CAAC,EAExH,KAAK,eAAe,SAAS,EAAe,KAAK,GAAG,IAAMF,EAAW,KAAK,GAAG,GAAG,KAAK,eAAe,SAASA,CAAU,EAE5H,KAAK,cAAc,EACnB,KAAK,aAAa,SAAS,QAAQ,EACnC,KAAK,WAAa,EACtB,CAAC,EAED,KAAK,qBAAuB,IAAI,GAAG,GAAG,YAAY,KAAK,eAAgB,CACnE,MAAO,cACP,QAAS,CAAC,uBAAuB,EACjC,MAAO,KACX,CAAC,EAGD,KAAK,aAAe,IAAI,GAAG,GAAG,oBAAoB,CAC9C,QAAS,CACL,CAAE,KAAM,yBAA0B,EAClC,CAAE,KAAM,uBAAwB,EAChC,CAAE,KAAM,wBAAyB,CACrC,CACJ,CAAC,EAED,KAAK,mBAAqB,IAAI,GAAG,GAAG,YAAY,KAAK,aAAc,CAC/D,GAAI,iCACJ,MAAO,WACP,QAAS,CAAC,uBAAuB,EACjC,MAAO,KACX,CAAC,CACL,CAKA,MAAc,oBAAqB,CAC/B,IAAM0B,EAAgB,IAAI,GAAG,GAAG,cAChC,SAAS,KAAK,OAAOA,EAAc,SAAS,CAAC,CAAC,EAG9C,KAAK,aAAe,IAAI,GAAG,GAAG,aAAa,CAAE,MAAO,SAAU,SAAU,GAAM,MAAO,CAAC,aAAa,CAAE,CAAC,EACtG,KAAK,aAAa,GAAG,QAAS,IAAM,KAAK,wBAAwB,CAAC,EAGlE,IAAMC,EAAwB,IAAIC,EAAoB,CAAE,KAAM,OAAQ,EAAG,KAAK,eAAe,EAC7FF,EAAc,WAAW,CAACC,CAAqB,CAAC,EAEhD,KAAK,kBAAoB,IAAI,GAAG,GAAG,aAAa,CAAE,MAAO,eAAgB,SAAU,EAAK,CAAC,EACzF,KAAK,kBAAkB,GAAG,QAAS,IAAM,CACrCA,EAAsB,QAClB,KAAK,aACD,KAAK,cAAc,SAAS,EAC5B,KAAK,UAAU,SAAS,EACxB,KAAK,aACL,KAAK,iBAAiB,SAAS,EAC/B,KAAK,eAAe,SAAS,CACjC,CACJ,EACAA,EAAsB,KAAK,CAC/B,CAAC,EAGD,IAAME,EAAoB,IAAIC,EAAc,CAAE,KAAM,OAAQ,CAAC,EAC7DJ,EAAc,WAAW,CAACG,CAAiB,CAAC,EAE5C,KAAK,kBAAoB,IAAI,GAAG,GAAG,aAAa,CAAE,MAAO,eAAgB,SAAU,EAAK,CAAC,EACzF,KAAK,kBAAkB,GAAG,QAAS,SAAY,CACvC,KAAK,SAAQ,KAAK,YAAc,MAAM,KAAK,eAAe,KAAK,SAAS,GAE5EA,EAAkB,QAAQ,CACtB,KAAK,YACL,KAAK,aACD,KAAK,cAAc,SAAS,EAC5B,KAAK,UAAU,SAAS,EACxB,KAAK,aACL,KAAK,iBAAiB,SAAS,EAC/B,KAAK,eAAe,SAAS,CACjC,CACJ,CAAC,EACDA,EAAkB,KAAK,CAC3B,CAAC,EAGI,KAAK,gBAAgB,WAAW,IACjC,KAAK,SAAY,MAAM,KAAK,IAAI,IAAI,CAChC,OAAQ,QACR,cAAe,IACf,KAAM,OACN,OAAQ,KAAK,gBAAgB,YAAY,EAAG,gBAAgB,CAChE,CAA8B,EAC9B,KAAK,iBAAmB,IAAI,GAAG,GAAG,oBAAoB,CAAE,SAAU,CAAC,CAAC,KAAK,SAAS,MAAM,MAAM,CAAC,EAAE,QAAS,CAAC,EAE3G,KAAK,uBAAyB,IAAI,GAAG,GAAG,OAAO,CAC3C,QAAS,CAAC,IAAI,GAAG,GAAG,YAAY,KAAK,iBAAkB,CAAE,MAAO,iBAAkB,MAAO,QAAS,CAAC,CAAC,CACxG,CAAC,GAID,MAAM,KAAK,wBAAwB,IACnC,KAAK,eAAiB,IAAI,GAAG,GAAG,oBAAoB,CAAE,SAAU,EAAK,CAAC,EAEtE,KAAK,qBAAuB,IAAI,GAAG,GAAG,OAAO,CACzC,QAAS,CAAC,IAAI,GAAG,GAAG,YAAY,KAAK,eAAgB,CAAE,MAAO,oBAAqB,MAAO,QAAS,CAAC,CAAC,CACzG,CAAC,GAIL,KAAK,aAAe,IAAI,GAAG,GAAG,iBAAiB,CAC3C,GAAI,gCACJ,MAAO,CACH,KAAK,aACL,KAAK,kBACL,KAAK,kBACL,KAAK,uBACL,KAAK,oBACT,EAAE,OAAO,OAAO,CACpB,CAAC,CACL,CAKA,MAAc,yBAA0B,CACpC,IAAME,EAAuB,SAAS,cAAgC,gCAAgC,EAItG,GAHAA,GAAsB,MAAM,EAC5BA,GAAsB,MAAM,EAExB,GAAG,OAAO,IAAI,mBAAmB,IAAM,EAAG,MAAO,GAChD,GAAI,SAAS,cAAc,aAAa,EAAG,MAAO,GAClD,GAAI,SAAS,cAAc,iCAAiC,EAAG,MAAO,GACtE,GAAI,SAAS,cAAc,mCAAmC,EAAG,MAAO,GACxE,CAGD,GAFI,CAAC,GAAG,OAAO,IAAI,aAAa,GAE5B,EADoB,MAAM,GAAG,KAAK,UAAU,GAC3B,SAAS,QAAQ,EAAG,MAAO,GAEhD,IAAMC,EAAkB,MAAM,KAAK,IAAI,IAAI,CACvC,OAAQ,iBACR,QAAS,GAAG,OAAO,IAAI,aAAa,CACxC,CAA6C,EAE7C,OAAIA,EAAe,eAAe,MAAM,CAAC,GAAG,YAAc,GAAG,OAAO,IAAI,YAAY,GAC3EA,EAAe,eAAe,SAAW,WAAaA,EAAe,eAAe,MAAM,SAAW,EADhB,GAElF,CAAC,OAAO,SAASA,EAAe,eAAe,MAAM,CAAC,GAAG,aAAa,CACtF,CACJ,CAKQ,eAAgB,CACpB,IAAMC,EAAgB,KAAK,cAAc,SAAS,EAAE,KAAK,EAEzD,GAAI,CAACA,EAAgB,KAAK,aAAa,YAAY,CAAC,EAAuB,YAAc,WAChF,KAAK,OAAQ,CAClB,IAAIC,EAAY,KAAK,mBAAmB,WAAW,IAAK,GAAG,EACvDA,IAAWA,EAAYA,EAAU,CAAC,EAAE,YAAY,EAAIA,EAAU,MAAM,CAAC,GAEzE,IAAMC,EAAgBF,IAAkBC,EAElCE,EACF,KAAK,UAAU,SAAS,EAAE,KAAMvC,GAAQ,CAAC,KAAK,gBAAiB,SAASA,CAAa,CAAC,GACtF,KAAK,gBAAiB,KAAMA,GAAQ,CAAC,KAAK,UAAU,SAAS,EAAE,SAASA,CAAG,CAAC,EAE5EwC,EAAsB,GAC1B,GAAI,KAAK,mBAAoB,CACzB,IAAMC,EAAqB,OAAO,QAAQ,KAAK,iBAAiB,EAAE,OAC9D,CAAC,CAAC,CAAEC,CAAI,IAAM,OAAO,QAAQA,EAAK,UAAU,EAAE,OAAS,CAC3D,EAEA,OAAW,CAAC1C,EAAK0C,CAAI,IAAKD,EAAoB,CAE1C,GAAI,CADmB,KAAK,gBAAiB,SAASzC,CAAG,GAClC,CAAC,KAAK,UAAU,SAAS,EAAE,SAASA,CAAG,EAAG,SAEjE,IAAM2C,EAAa,KAAK,mBAAmB3C,CAAG,GAAK,OAAO,QAAQ0C,EAAK,UAAU,EAAE,IAAI,CAAC,CAACtB,CAAI,IAAM,CAACA,EAAM,EAAE,CAAC,EAEvGwB,EAAqB,KAAK,oBAAoB,KAAMC,GAAaA,EAAS,OAAS7C,CAAG,EAE5F,QAAW8C,KAAaF,EAAmB,WAAY,CACnD,IAAMG,EAAcJ,EAAW,KAAMK,GAAaA,EAAS,CAAC,IAAMF,EAAU,IAAI,IAAI,CAAC,GAAK,GACpFG,EAAcH,EAAU,OAAO,SAAS,EAAE,KAAK,EAErD,GAAIC,IAAgBE,EAAa,CAC7BT,EAAsB,GACtB,KACJ,CACJ,CAEA,GAAIA,EAAqB,KAC7B,CACJ,CAEA,IAAMU,EAAqB,KAAK,iBAAiB,SAAS,EAAE,KAAK,IAAM,KAAK,eAAgB,WAAW,IAAK,GAAG,EAEzGC,EACF,KAAK,eAAe,SAAS,EAAE,KAAMC,GAAa,CAAC,KAAK,cAAe,SAASA,CAAkB,CAAC,GACnG,KAAK,cAAe,KAAMA,GAAa,CAAC,KAAK,eAAe,SAAS,EAAE,SAASA,CAAQ,CAAC,EAEvFC,EAAU,CAAC,EAEbf,GAAee,EAAQ,KAAK,iBAAiBjB,CAAa,IAAI,EAC9DG,GACAc,EAAQ,KACJ,GAAG,KAAK,UAAU,SAAS,EAAE,OAAS,GAAK,KAAK,gBAAiB,OAAS,EAAI,SAAW,KAAK,UAAU,SAAS,EAAE,OAAS,EAAI,MAAQ,QAAQ,2BACpJ,EACAb,GAAqBa,EAAQ,KAAK,0CAA0C,EAC5EH,GACAG,EAAQ,KACJ,GAAG,KAAK,iBAAiB,SAAS,EAAE,KAAK,EAAE,OAAS,GAAK,KAAK,eAAgB,WAAW,IAAK,GAAG,EAAE,OAAS,EAAI,SAAW,KAAK,iBAAiB,SAAS,EAAE,KAAK,EAAE,OAAS,EAAI,MAAQ,QAAQ,mBACpM,EACAF,GACAE,EAAQ,KACJ,GAAG,KAAK,eAAe,SAAS,EAAE,OAAS,GAAK,KAAK,cAAe,OAAS,EAAI,SAAW,KAAK,eAAe,SAAS,EAAE,OAAS,EAAI,MAAQ,QAAQ,aAC5J,EAEAA,EAAQ,SAAW,GAAGA,EAAQ,KAAK,0BAA0B,EAEjEA,EAAQ,CAAC,EAAIA,EAAQ,CAAC,EAAE,CAAC,EAAE,YAAY,EAAIA,EAAQ,CAAC,EAAE,MAAM,CAAC,EACzDA,EAAQ,OAAS,IAAGA,EAAQA,EAAQ,OAAS,CAAC,EAAI,OAAOA,EAAQ,GAAG,EAAE,CAAC,IAE1E,KAAK,aAAa,YAAY,CAAC,EAAuB,YAAcA,EAAQ,KAAKA,EAAQ,OAAS,EAAI,KAAO,GAAG,CACrH,MAAQ,KAAK,aAAa,YAAY,CAAC,EAAuB,YAAc,wBAAwBjB,CAAa,IACrH,CAKA,MAAc,kBAAmB,CACzB,KAAK,SAAQ,KAAK,YAAc,MAAM,KAAK,eAAe,KAAK,SAAS,GAE5E,KAAK,kBAAoB,KAAK,cAAc,KAAK,KAAK,WAAW,IAAI,CAAC,EAEtE,KAAK,gBACD,OAAO,QAAQ,KAAK,iBAAiB,EAChC,IAAI,CAAC,CAACpC,EAAKsD,CAAO,IACf,CAACtD,EAAK,GAAGsD,EAAQ,OAAO,EAAE,KAAMzD,GAC5B,IAAI,OACA,UAAUA,EAAc,CAAC,EAAE,YAAY,CAAC,GAAGA,EAAc,CAAC,CAAC,IAAIA,EAAc,MAAM,CAAC,CAAC,cACzF,EAAE,KAAK,KAAK,WAAW,CAC3B,EACMG,EACA,IACV,EACC,OAAO,OAAO,EACrB,KAAK,CAACI,EAAGC,IAAMD,EAAE,YAAY,EAAE,cAAcC,EAAE,YAAY,CAAC,CAAC,EAE/D,IAAMkD,EAAuB,OAAO,QAAQ,KAAK,iBAAiB,EAC7D,QAAQ,CAAC,CAACvD,EAAKsD,CAAO,IAAM,CAACtD,EAAK,GAAGsD,EAAQ,OAAO,CAAC,EACrD,IAAKzD,GACF,IAAI,OAAO,UAAUA,EAAc,CAAC,EAAE,YAAY,CAAC,GAAGA,EAAc,CAAC,CAAC,IAAIA,EAAc,MAAM,CAAC,CAAC,cAAc,EAAE,KAC5G,KAAK,WACT,EACMA,EACA,IACV,EACC,OAAO,OAAO,EAEnB,KAAK,mBAAqB,OAAO,YAC7B0D,EACK,IAAKvD,GAAQ,CACV,IAAMwD,EAAQ,IAAI,OAAO,UAAUxD,EAAI,CAAC,EAAE,YAAY,CAAC,GAAGA,EAAI,CAAC,CAAC,IAAIA,EAAI,MAAM,CAAC,CAAC,iBAAiB,EAAE,KAC/F,KAAK,WACT,EAEMyD,EAAS,OAAO,QAAQ,KAAK,iBAAiB,EAAE,KAAK,CAAC,CAACZ,EAAUS,CAAO,IAC1E,CAACT,EAAU,GAAGS,EAAQ,OAAO,EAAE,SAAStD,CAAG,CAC/C,IAAI,CAAC,EAGL,GAAI,CADsBwD,IAAQ,CAAC,EACX,OAAO,KAE/B,IAAME,EAAqBF,EAAM,CAAC,EAAE,MAAM,GAAG,EAAE,IAAI,CAACR,EAAUW,IAAU,CACpE,GAAI,CAACX,EAAS,SAAS,GAAG,EAAG,MAAO,EAAEW,EAAQ,GAAG,SAAS,EAAGX,EAAS,KAAK,CAAC,EAE5E,GAAM,CAAC5B,EAAMrB,CAAK,EAAIiD,EAAS,MAAM,GAAG,EAExC,MAAO,CAAC5B,EAAK,KAAK,EAAGrB,EAAM,KAAK,CAAC,CACrC,CAAC,EAED,MAAO,CAAC0D,EAAQC,CAAkB,CACtC,CAAC,EACA,OAAO,OAAO,CACvB,EAEA,KAAK,eACD,KAAK,YACA,MAAM,sBAAsB,GAC3B,GAAG,EAAE,GACL,MAAM,GAAI,EAAE,GACZ,KAAK,GAAK,GAEpB,KAAK,cAAgB,KAAK,YAAY,MAAM,wBAAwB,GAAG,IAAKN,GAAaA,EAAS,MAAM,GAAI,EAAE,CAAC,GAAK,CAAC,EAErH,KAAK,aAAe,CAChB,8BAA8B,KAAK,KAAK,WAAW,IAAI,CAAC,EACxD,uBAAuB,KAAK,KAAK,WAAW,IAAI,CAAC,EACjD,0BAA0B,KAAK,KAAK,WAAW,IAAI,CAAC,EACpD,0BAA0B,KAAK,KAAK,WAAW,IAAI,CAAC,EACpD,iCAAiC,KAAK,KAAK,WAAW,IAAI,CAAC,CAC/D,EACK,OAAO,OAAO,EACd,KAAK;AAAA,CAAI,EAEV,KAAK,kBAAmB,KAAK,cAAc,SAAS,KAAK,kBAAkB,WAAW,IAAK,GAAG,CAAC,EAC9F,GAAG,OAAO,kCAAmC,CAAE,KAAM,OAAQ,CAAC,EAEnE,KAAK,UAAU,SAAS,KAAK,eAAe,EAE5C,OAAW,CAAC1C,EAAcgC,CAAI,IAAK,OAAO,QAAQ,KAAK,kBAAkB,EAAG,CACxE,IAAMkB,EAAsB,KAAK,oBAAoB,KAAMtD,GAAeA,EAAW,OAASI,CAAY,EAC1G,GAAKkD,EAEL,OAAW,CAAC5C,EAAegC,CAAQ,IAAKN,EAAM,CAC1C,IAAMmB,EAAuBD,EAAoB,WAAW,KAAMd,GAC9D,CAACA,EAAU,KAAM,GAAGA,EAAU,OAAO,EAAE,SAAS9B,CAAa,CACjE,EAEI6C,GAAsBA,EAAqB,OAAO,SAASb,CAAQ,CAC3E,CACJ,CAEI,KAAK,gBAAgB,KAAK,iBAAiB,SAAS,KAAK,cAAc,EAE3E,QAAWI,KAAY,KAAK,cAAe,KAAK,eAAe,gBAAgBA,CAAQ,EACvF,KAAK,eAAe,SAAS,KAAK,cAAc,IAAKA,IAAc,CAAE,KAAMA,EAAU,MAAOA,CAAS,EAAE,CAAC,EAExG,KAAK,cAAc,CACvB,CAKA,MAAc,oBAAqB,CAC/B,IAAMU,EAIA,CAAC,EAEDC,EAAc,KAAK,cAAc,SAAS,EAAE,KAAK,EACjDC,EAAO,KAAK,UAAU,SAAS,EAGhC,qBAAqB,KAAKD,CAAW,GAAGD,EAAO,KAAK,CAAE,MAAOC,EAAa,QAAS,4BAA6B,CAAC,EAGtH,GAAI,CACA,KAAK,kBAAoB,GAAG,MAAM,YAAYA,CAAW,CAC7D,MAAQ,CACAD,EAAO,SAAW,GAAGA,EAAO,KAAK,CAAE,MAAOC,EAAa,QAAS,4BAA6B,CAAC,CACtG,CACI,CAAC,KAAK,mBAAqBD,EAAO,SAAW,GAAGA,EAAO,KAAK,CAAE,MAAOC,EAAa,QAAS,4BAA6B,CAAC,EAGzH,KAAK,mBAAmB,gBAAgB,IAAM,KAAK,gBAAgB,gBAAgB,GACnFD,EAAO,KAAK,CAAE,QAAS,4BAA6B,CAAC,EAEzD,IAAMG,EAAmB,MAAM,KAAK,IAC/B,IAAI,CACD,OAAQ,QACR,cAAe,IACf,KAAM,CAAC,YAAa,YAAY,EAChC,OAAQF,CACZ,CAAmC,EAClC,MAAOG,IAC8BA,IAAc,eAC5CJ,EAAO,KAAK,CAAE,MAAOC,EAAa,QAAS,iBAAkB,CAAC,EAE9DD,EAAO,KAAK,CAAE,MAAOC,EAAa,QAAS,4CAA4CG,CAAS,IAAK,CAAC,EACnG,KACV,EACCC,EAA0B,MAAM,KAAK,IAAI,IAAI,CAC/C,OAAQ,QACR,KAAMJ,EACN,KAAM,WACN,UAAW,EACf,CAA0B,EAG1B,GAAII,EAAuB,MAAM,YAAY,CAAC,EAAG,CAC7C,IAAMC,EACFD,EAAuB,MAAM,UAAU,CAAC,EAAE,IACzCA,EAAuB,MAAM,UAAU,CAAC,EAAE,WAAa,IAAIA,EAAuB,MAAM,UAAU,CAAC,EAAE,UAAU,GAAK,IACzHL,EAAO,KAAK,CACR,MAAOC,EACP,QAAS,6BAA6B,GAAG,KAAK,OAC1CK,CACJ,CAAC,qBAAqBA,CAAmB,2EACzC,UAAW,CAAC,CAAE,KAAM,gBAAiB,OAAQA,CAAoB,CAAC,CACtE,CAAC,CACL,CAGA,GAAIL,EAAY,MAAM,GAAG,EAAE,OAAS,EAIhC,GAHqBI,EAAuB,MAAM,SAAS,KACtDE,GAAYA,EAAQ,KAAK,WAAW,UAAW,EAAE,IAAMN,EAAY,MAAM,GAAG,EAAE,CAAC,CACpF,EAEQC,EAAK,SAAS,aAAa,GAC3BF,EAAO,KAAK,CACR,QAAS,oFACT,UAAW,CACP,CAAE,KAAM,MAAO,IAAK,cAAe,EACnC,CAAE,KAAM,SAAU,IAAK,aAAc,CACzC,CACJ,CAAC,EACAE,EAAK,SAAS,cAAc,GAC7BF,EAAO,KAAK,CACR,QAAS,uFACT,UAAW,CAAC,CAAE,KAAM,MAAO,IAAK,cAAe,CAAC,CACpD,CAAC,MACF,CACH,IAAMQ,GACD,MAAM,KAAK,IAAI,IAAI,CAChB,OAAQ,QACR,cAAe,IACf,KAAM,YACN,OAAQ,UACR,QAAS,OACT,OAAQ,KAAK,kBAAmB,gBAAgB,CACpD,CAAmC,GACrC,MAAM,MAAM,CAAC,EAAE,UAAU,CAAC,EAAE,MAAM,KAAK,QAEzB,CACZ,GAAIA,EACC,MAAM,6EAA6E,GAClF,IAAKC,GAAmBA,EAAO,MAAM,GAAG,EAAE,IAAKC,GAASA,EAAK,KAAK,CAAC,CAAC,GACpE,KAAK,GAAK,CAAC,EACjB,GAAIF,EACC,MACG,8IACJ,GACE,IAAKC,GACHA,EACK,MAAM,GAAG,EACT,IAAKC,GAASA,EAAK,KAAK,CAAC,EACzB,OAAQA,GAAS,CAAC,aAAa,KAAKA,CAAI,CAAC,CAClD,GACE,KAAK,GAAK,CAAC,EACjB,GAAIF,EAAmB,MAAM,0BAA0B,GAAG,IAAKC,GAAmBA,EAAO,KAAK,CAAC,GAAK,CAAC,EACrG,GAAID,EAAmB,MAAM,wBAAwB,GAAG,IAAKC,GAAmB,KAAKA,EAAO,MAAM,GAAG,EAAE,CAAC,EAAE,KAAK,CAAC,EAAE,GAC9G,CAAC,CACT,EACY,SAASR,EAAY,MAAM,GAAG,EAAE,CAAC,CAAC,GACtCC,EAAK,SAAS,cAAc,GAC5BF,EAAO,KAAK,CACR,QAAS,oFACT,UAAW,CACP,CAAE,KAAM,MAAO,IAAK,aAAc,EAClC,CAAE,KAAM,SAAU,IAAK,cAAe,CAC1C,CACJ,CAAC,EACAE,EAAK,SAAS,aAAa,GAC5BF,EAAO,KAAK,CACR,QAAS,wFACT,UAAW,CAAC,CAAE,KAAM,MAAO,IAAK,aAAc,CAAC,CACnD,CAAC,GAELA,EAAO,KAAK,CACR,QAAS,6BAA6B,GAAG,KAAK,OAAOC,CAAW,CAAC,qBAAqBA,CAAW,mDACjG,UAAW,CAAC,CAAE,KAAM,gBAAiB,OAAQA,EAAY,MAAM,GAAG,EAAE,CAAC,CAAE,CAAC,CAC5E,CAAC,CACT,CAIJ,GAAIA,EAAY,MAAM,GAAG,EAAE,SAAW,EAClC,QAAW/D,IAAO,CAAC,eAAgB,aAAa,EACxCgE,EAAK,SAAShE,CAAG,GACjB8D,EAAO,KAAK,CACR,QAAS,wEAAwE9D,CAAG,aACpF,UAAW,CAAC,CAAE,KAAM,SAAU,IAAAA,CAAI,CAAC,CACvC,CAAC,EAEb,IAAMyE,EAA6B,CAAC,EAChCR,EAAiB,MAAM,MAAM,CAAC,EAAE,WAAa,mBAAoBA,EAAiB,MAAM,MAAM,CAAC,EAAE,WAE/FS,EAAsB,CAAC,CAACT,EAAiB,MAAM,MAAM,CAAC,EAAE,YAAY,KACrEb,GAAaA,EAAS,QAAU,mBACrC,EAEMuB,EAA2B,CAAC,2BAA4B,kCAAkC,EAC1FC,EAAoB,CAAC,6BAA8B,uBAAuB,EAE1EC,EAAuCF,EAAyB,KAAM9B,GAAamB,EAAK,SAASnB,CAAQ,CAAC,EAC1GiC,EAAgCF,EAAkB,KAAM/B,GAAamB,EAAK,SAASnB,CAAQ,CAAC,EAG9F4B,GAA8B,CAACI,GAAwC,CAACC,GACxEhB,EAAO,KAAK,CACR,QAAS,6GACb,CAAC,EAEDG,EAAiB,MAAM,MAAM,CAAC,EAAE,WAAa,CAACQ,KAGzC,CAACC,IAAwBG,GAAwCC,IACjEJ,GAAuBG,IAExBf,EAAO,KAAK,CACR,QAAS,8GACT,UAAW,CAAC,GAAGa,EAA0B,GAAGC,CAAiB,EAAE,IAAK5E,IAAS,CAAE,KAAM,SAAU,IAAAA,CAAI,EAAE,CACzG,CAAC,EAGD0E,GAAuB,CAACI,GACxBhB,EAAO,KAAK,CACR,QAAS,8GACb,CAAC,GAKLW,GACAT,EAAK,SAAS,0BAA0B,GACxC,CAAC,KAAK,gBAAgB,YAAY,EAAE,SAAS,mBAAmB,GAEhEF,EAAO,KAAK,CACR,QACI,2MACJ,UAAW,CAAC,CAAE,KAAM,SAAU,IAAK,0BAA2B,CAAC,CACnE,CAAC,EAGL,QAAWjB,IAAY,CAAC,cAAe,mBAAoB,uBAAwB,uBAAwB,mBAAmB,EACtHmB,EAAK,SAASnB,CAAQ,GACtBiB,EAAO,KAAK,CACR,QAAS,oDAAoDjB,CAAQ,qEACrE,UAAW,CAAC,CAAE,KAAM,SAAU,IAAKA,CAAS,CAAC,CACjD,CAAC,EAGL,GAAG,OAAO,IAAI,kBAAkB,GAAK,CAACmB,EAAK,SAAS,sBAAsB,GAC1EF,EAAO,KAAK,CACR,QAAS,+FACT,UAAW,CAAC,CAAE,KAAM,MAAO,IAAK,sBAAuB,CAAC,CAC5D,CAAC,EAGDE,EAAK,SAAS,sBAAsB,GAAK,CAAC,GAAG,OAAO,IAAI,kBAAkB,GAC1EF,EAAO,KAAK,CACR,QAAS,yGACT,UAAW,CAAC,CAAE,KAAM,SAAU,IAAK,sBAAuB,CAAC,CAC/D,CAAC,EAGL,QAAW9D,KAAOgE,EAAM,CACpB,IAAMV,EAAU,KAAK,kBAAkBtD,CAAG,EAC1C,GAAKsD,EAEL,OAAW,CAACtC,EAAeC,CAAa,IAAK,OAAO,QAAQqC,EAAQ,UAAU,EAAG,CAC7E,IAAMyB,EAAiB,KAAK,oBACvB,KAAMzE,GAAeA,EAAW,OAASN,CAAG,GAC3C,WAAW,KAAM8C,GAAc,CAACA,EAAU,KAAM,GAAGA,EAAU,OAAO,EAAE,SAAS9B,CAAa,CAAC,EAE9F+D,GAED9D,EAAc,UAAY,CAAC8D,EAAe,OAAO,SAAS,EAAE,KAAK,GACjEjB,EAAO,KAAK,CACR,QAAS,0BAA0B9D,CAAG,4DAA4DgB,CAAa,UACnH,CAAC,CACT,CACJ,CAGA,OAAI,KAAK,kBAAkB,WAAW,GAAK,CAAC,KAAK,SAAU,MAAM,MAAM,CAAC,EAAE,SAAW,CAAC,KAAK,SAAU,MAAM,MAAM,CAAC,EAAE,UAChH8C,EAAO,KAAK,CACR,MAAO,KAAK,gBAAgB,YAAY,EAAG,gBAAgB,EAC3D,QAAS,gCACb,CAAC,EAEEA,CACX,CAKA,MAAc,yBAA0B,CACpC,IAAMkB,EAAoB,CACtB,KAAK,cACL,KAAK,UACL,GAAG,KAAK,oBAAoB,QAASnC,GAAaA,EAAS,WAAW,IAAKC,GAAcA,EAAU,MAAM,CAAC,EAC1G,KAAK,iBACL,KAAK,yBACL,KAAK,eACL,KAAK,aACL,KAAK,aACL,KAAK,kBACL,KAAK,kBACL,KAAK,iBACL,KAAK,cACT,EAAE,OAAO,OAAO,EAEhB,QAAWmC,KAAWD,EAAoBC,EAAyB,YAAY,EAAI,EAEnF,KAAK,aAAa,SAAS,6BAA6B,EAExD,IAAInB,EAA8D,CAAC,EAInE,GAHI,KAAK,WAAYA,EAAS,MAAM,KAAK,mBAAmB,EACvD,KAAK,kBAAoB,GAAG,MAAM,YAAY,KAAK,cAAc,SAAS,CAAC,EAE5EA,EAAO,OAAS,EAAG,CACnB,QAAWmB,KAAW,SAAS,iBAAiB,0BAA0B,EAAGA,EAAQ,OAAO,EAC5F,OAAW,CAAE,MAAAC,EAAO,QAAAC,EAAS,UAAAC,CAAU,IAAKtB,EAAQ,CAChD,IAAM5D,EAAQ,IAAI,GAAG,GAAG,YACpB,GAAGgF,EAAQ,YAAY,GAAG,KAAK,OAAOA,CAAK,CAAC,qBAAqBA,CAAK,OAAS,WAAW,IAAIC,CAAO,uDACzG,EACME,EAAiB,IAAI,GAAG,GAAG,cAAc,CAC3C,KAAM,QACN,QAAS,CAAC,yBAAyB,EACnC,OAAQ,GACR,MAAAnF,CACJ,CAAC,EAED,GAAIkF,EAAW,CACX,IAAME,EAAgB,IAAI,GAAG,GAAG,aAAa,CACzC,MAAO,mBACP,MAAO,CAAC,aAAa,EACrB,QAAS,CAAC,gCAAgC,CAC9C,CAAC,EACDA,EAAc,GAAG,QAAS,IAAM,CAC5B,IAAMtB,EAAO,KAAK,UAAU,SAAS,EAErC,QAAWuB,KAAWH,EACdG,EAAQ,OAAS,OAAS,CAACvB,EAAK,SAASuB,EAAQ,GAAG,GAAG,KAAK,UAAU,OAAOA,EAAQ,IAAKA,EAAQ,GAAG,EAErGA,EAAQ,OAAS,UAAYvB,EAAK,SAASuB,EAAQ,GAAG,GAAG,KAAK,UAAU,gBAAgBA,EAAQ,GAAG,EAEnGA,EAAQ,OAAS,iBAAiB,KAAK,cAAc,SAASA,EAAQ,MAAM,EAGpFF,EAAe,SAAS,CAAC,EAAE,MAAM,eAAiB,yBAClDC,EAAc,SAAS,CAAC,EAAE,OAAO,CACrC,CAAC,EAEDD,EAAe,SAAS,CAAC,EAAE,cAAc,2BAA2B,EAAG,OAAOC,EAAc,SAAS,CAAC,CAAC,CAC3G,CAEA,KAAK,UAAU,SAAS,CAAC,EAAE,OAAOD,EAAe,SAAS,CAAC,CAAC,CAChE,CAEA,QAAWJ,KAAWD,EAAoBC,EAAyB,YAAY,EAAK,EAEpF,KAAK,aAAa,SAAS,eAAe,EAC1C,KAAK,WAAa,GAElB,MACJ,CAGA,KAAK,aAAa,SAAS,GAAG,KAAK,OAAS,UAAY,UAAU,cAAc,EAEhF,IAAMO,EAAS,KAAK,aAChB,KAAK,cAAc,SAAS,EAC5B,KAAK,UAAU,SAAS,EACxB,KAAK,aACL,KAAK,iBAAiB,SAAS,EAC/B,KAAK,eAAe,SAAS,CACjC,EAEM1E,GACD,KAAK,aAAa,SAAS,GAAM,KAAK,aAAa,YAAY,CAAC,EAAuB,aAAe,KAAK,aAGhH,GADe,MAAM,KAAK,aAAa,KAAK,UAAW0E,EAAQ1E,CAAO,EAMtE,IAHA,GAAG,OAAO,YAAY,KAAK,OAAS,SAAW,SAAS,iBAAkB,CAAE,KAAM,SAAU,CAAC,EAGzF,KAAK,kBAAkB,WAAW,EAAG,CACrC,KAAK,aAAa,SAAS,sBAAsB,EAEjD,IAAM2E,EAAW,KAAK,UAAU,SAAS,EAAE,SAAS,aAAa,EAE3DD,EAAS,KAAK,aAChB,KAAK,kBAAmB,YAAY,EAAG,gBAAgB,EACvDC,EAAW,CAAC,aAAa,EAAI,CAAC,EAC9B,OACA,OACA,CAAC,CACL,EAOA,GAAI,CALe,MAAM,KAAK,aAC1B,KAAK,gBAAgB,YAAY,EAAG,gBAAgB,EACpDD,EACA,kCAAoC,KAAK,YAC7C,EACiB,OAEjB,GAAG,OAAO,iCAAkC,CAAE,KAAM,SAAU,CAAC,CACnE,CAGA,GAAI,KAAK,gBAAgB,WAAW,EAAG,CACnC,KAAK,aAAa,SAAS,wBAAwB,EAEnD,IAAME,EAAuC,SAAS,cAAc,eAAe,EAC7EC,EAAqB,SAAS,cAAiC,iCAAiC,EAElGD,EACqB,MAAM,KAAK,IAC3B,cAAc,SAAU,CAAE,OAAQ,SAAU,KAAM,IAAI,IAAIA,EAAW,IAAI,EAAE,aAAa,IAAI,MAAM,CAAG,CAAC,EACtG,MAAM,CAACxB,EAAmB0B,KACvB,GAAG,OACC,oBAAoB,KAAK,SAAS,aAAaA,GAAW,MAAM,MAAQ,eAAe,KAAK1B,CAAS,IACrG,CAAE,KAAM,OAAQ,CACpB,EACO,KACV,GACa,GAAG,OAAO,mCAAoC,CAAE,KAAM,SAAU,CAAC,EAC5EyB,GACPA,EAAmB,MAAM,EACzB,GAAG,OAAO,mCAAoC,CAAE,KAAM,SAAU,CAAC,GAC9D,GAAG,OAAO,iEAAkE,CAAE,KAAM,OAAQ,CAAC,CACxG,CAEA,KAAK,aAAa,SAAS,wBAAwB,EAEnD,OAAO,SAAS,KAAO,GAAG,KAAK,OAAO,KAAK,UAAW,CAAE,SAAU,IAAK,CAAC,EAC5E,CAKQ,aACJ/F,EACAoE,EACA6B,EACAC,EACAC,EACF,CACE,IAAMC,EAAe,GAAG,MAAM,YAAYpG,CAAM,EAE1CqG,EAAiBD,EACjB,GAAGA,EAAa,eAAe,IAAM,GAAK,IAAM,EAAE,GAAGA,EAAa,gBAAgB,CAAC,GAAGA,EAAa,YAAY,EAAI,IAAIA,EAAa,YAAY,CAAC,GAAK,EAAE,GACxJpG,EAAO,KAAK,EAGd,KAAK,gBACA,YAAY,EACZ,kBAAkB,EAClB,UAAU,KAAK,EACf,WAAW,mBAAoB,EAAE,IACtCkG,GACM,YAAY,EACb,UAAU,KAAK,EACf,WAAW,mBAAoB,EAAE,IAEtCA,EAAc,QAElB,IAAMI,EAAoBlC,EAAK,IAAKhE,GAAQ,CACxC,IAAMmG,EAAsB,KAAK,oBAAoB,KAAM7F,GAAeA,EAAW,OAASN,CAAG,EACjG,GAAI,CAACmG,EAAqB,MAAO,KAAKnG,CAAG,KAEzC,IAAMoG,EAA2BD,EAAoB,WAAW,cAC5D,CAACrD,EAAWa,IAAUb,EAAU,QAAUa,EAAQ,GAAG,SAAS,GAAKb,EAAU,OAAO,SAAS,EAAE,KAAK,CACxG,EAEMuD,EAAkBF,EAAoB,WACvC,IAAI,CAACrD,EAAWa,IAAU,CACvB,IAAM5D,EAAQ+C,EAAU,OAAO,SAAS,EAAE,KAAK,EAC/C,MAAI,CAAC/C,GAAS4D,EAAQyC,EAAiC,KAEhD,IAAItD,EAAU,QAAUa,EAAQ,GAAG,SAAS,EAAI,GAAK,GAAGb,EAAU,IAAI,GAAG,GAAG/C,CAAK,EAC5F,CAAC,EACA,OAAO,OAAO,EACd,KAAK,EAAE,EAEZ,MAAO,KAAKC,CAAG,GAAGqG,CAAe,IACrC,CAAC,EAED,MAAO,CACH,eAAeJ,CAAc;AAAA,EAC7BjC,EAAK,OAAS,EAAI;AAAA,EAA+BkC,EAAkB,KAAK;AAAA,CAAI,CAAC;AAAA;AAAA,EAAW,KACxFL,EAAYA,EAAY;AAAA,EAAO,KAC/BC,EAAc,iBAAiBA,EAAY,KAAK,CAAC,KAAO,KACxDC,EAAW,OAAS,EAAIA,EAAW,IAAK3C,GAAa,cAAcA,CAAQ,IAAI,EAAE,KAAK;AAAA,CAAI,EAAI,IAClG,EACK,OAAO,OAAO,EACd,KAAK;AAAA,CAAI,CAClB,CAMA,MAAc,eAAe8B,EAAe,CACxC,OACK,MAAM,KAAK,IAAI,IAAI,CAChB,OAAQ,QACR,cAAe,IACf,KAAM,YACN,OAAQ,UACR,QAAS,OACT,OAAQA,CACZ,CAAmC,GACrC,MAAM,MAAM,CAAC,EAAE,UAAU,CAAC,EAAE,MAAM,KAAK,QAAQ,KAAK,CAC1D,CAQA,MAAc,aAAaA,EAAeoB,EAAcxF,EAAiB,CACrE,OAAO,MAAM,KAAK,IACb,KAAKoE,EAAO,KAAO,CAAE,KAAAoB,EAAM,QAAAxF,CAAQ,EAAE,EACrC,MAAM,CAACoD,EAAmB0B,IACnB1B,IAAc,mBACP,KAAK,IAAI,OAAOgB,EAAO,CAAE,QAAApE,CAAQ,EAAGwF,CAAI,EAAE,MAAM,CAACpC,EAAmB0B,IAAkC,CACzG,GAAG,OAAO,kBAAkBV,CAAK,KAAKU,GAAW,MAAM,MAAQ,eAAe,KAAK1B,CAAS,IAAK,CAC7F,KAAM,OACV,CAAC,CACL,CAAC,GAED,GAAG,OAAO,6BAA6BgB,CAAK,KAAKU,GAAW,MAAM,MAAQ,eAAe,KAAK1B,CAAS,IAAK,CACxG,KAAM,OACV,CAAC,EACM,KAEd,CACT,CACJ,ICpxCA,IAAMqC,EAAe,CACjB,iBACA,eACA,kBACA,kBACA,+BACA,mCACJ,EAEA,GAAG,OAAO,MAAMA,EAAc,SAAY,CACtC,GAAM,CAAE,QAASC,CAAqB,EAAI,KAAM,qCAKhD,MAAMC,CAAe,CAET,IAAM,IAAI,GAAG,IAGb,kBACA,YACA,UACA,gBAKR,MAAM,KAAM,CACR,GAAK,KAAK,gBAAgB,EAK1B,IAHA,KAAK,kBAAoB,MAAM,KAAK,uBAAuB,EAE3D,KAAK,YAAc,SAAS,cAA8B,kBAAkB,EACxE,CAAC,KAAK,YAAa,OAAO,GAAG,OAAO,wDAAyD,CAAE,KAAM,OAAQ,CAAC,EAKlH,GAHA,KAAK,UAAY,GAAG,OAAO,IAAI,YAAY,EAE3C,KAAK,gBAAkB,GAAG,MAAM,YAAY,KAAK,SAAS,EACtD,CAAC,KAAK,gBAAiB,OAAO,GAAG,OAAO,+CAAgD,CAAE,KAAM,OAAQ,CAAC,EAE7G,MAAM,KAAK,iBAAiB,EAChC,CAKQ,iBAAkB,CAUtB,MATmB,CACf,GAAG,OAAO,IAAI,mBAAmB,GAAK,EACtC,GAAG,OAAO,IAAI,sBAAsB,EACpC,GAAG,OAAO,IAAI,aAAa,EAC3B,GAAG,OAAO,IAAI,UAAU,IAAM,OAC9B,GAAG,OAAO,IAAI,cAAc,IAAM,GAAG,OAAO,IAAI,iBAAiB,EACjE,CAAC,GAAG,OAAO,IAAI,aAAa,CAChC,EAEkB,MAAM,OAAO,CACnC,CAKA,MAAc,wBAAyB,CACnC,OAAO,KAAK,OAEH,MAAM,KAAK,IAAI,IAAI,CAChB,OAAQ,QACR,cAAe,IACf,KAAM,YACN,OAAQ,UACR,QAAS,OACT,OAAQ,2CACZ,CAAmC,GACrC,MAAM,QAAQ,CAAC,GAAG,YAAY,CAAC,GAAG,OAAO,MAAM,SAAW,IAChE,CACJ,CAKA,MAAc,kBAAmB,CAC7B,IAAMC,EAAY,MAAM,KAAK,IAAI,IAAI,CACjC,OAAQ,QACR,cAAe,IACf,KAAM,OACN,OAAQ,KAAK,SACjB,CAA8B,EAExBC,EAAa,CACf,kBAAmB,KAAK,kBACxB,YAAa,KAAK,YAClB,UAAW,KAAK,UAChB,gBAAiB,KAAK,eAC1B,EAEA,GAAID,EAAS,MAAM,MAAM,CAAC,EAAE,QAAS,CACjC,GAAG,KAAK,OAAO;AAAA;AAAA;AAAA,EAG7B,EAEc,IAAME,EAAS,IAAI,GAAG,GAAG,aAAa,CAClC,GAAI,yBACJ,MAAO,kBACP,KAAM,kBACN,MAAO,CAAC,aAAa,CACzB,CAAC,EACDA,EAAO,GAAG,QAAS,IAAM,CACrBA,EAAO,SAAS,CAAC,EAAE,OAAO,EAC1B,IAAIJ,EAAqBG,EAAY,EAAK,EAAE,KAAK,CACrD,CAAC,EAED,KAAK,YAAY,QAAQC,EAAO,SAAS,CAAC,CAAC,CAC/C,SAAWF,EAAS,MAAM,MAAM,CAAC,EAAE,SAAU,IAAIF,EAAqBG,EAAY,EAAI,EAAE,KAAK,MACxF,CACD,IAAME,EAAc,GAAG,KAAK,eACxB,GAAG,OAAO,IAAI,MAAM,IAAM,UAAY,OAAS,aAC/C,IACA,gBACA,iBACJ,EACAA,EAAY,iBAAiB,QAAUC,GAAU,CAC7CA,EAAM,eAAe,EAErB,IAAIN,EAAqBG,EAAY,EAAK,EAAE,KAAK,EAEjD,OAAO,SAAS,CAAE,IAAK,EAAG,SAAU,QAAS,CAAC,EAE9CE,EAAY,OAAO,CACvB,CAAC,CACL,CACJ,CACJ,CAEA,IAAIJ,EAAe,EAAE,IAAI,CAC7B,CAAC",
  "names": ["CategoryInputWidget", "init_category_input_widget", "__esmMin", "config", "value", "deferred", "parsedTitle", "result", "pages", "page", "category", "titleWithoutNamespace", "response", "data", "label", "ChangesDialog", "init_changes_dialog", "__esmMin", "_ChangesDialog", "config", "oldText", "newText", "result", "comparison", "noChangesElement", "panelLayout", "action", "OutputPreviewDialog", "init_output_preview_dialog", "__esmMin", "_OutputPreviewDialog", "config", "pageTitleParsed", "result", "tagsContent", "categoriesContent", "panelLayout", "action", "RedirectTargetInputWidget", "init_redirect_target_input_widget", "__esmMin", "config", "pageTitleParsed", "value", "deferred", "title", "result", "matchedSections", "section", "parsedTitle", "page", "response", "data", "label", "redirect_helper_dialog_exports", "__export", "RedirectHelperDialog", "init_redirect_helper_dialog", "__esmMin", "init_category_input_widget", "init_changes_dialog", "init_output_preview_dialog", "init_redirect_target_input_widget", "redirectTemplates", "contentText", "pageTitle", "pageTitleParsed", "exists", "mainPageContent", "target", "tagOrRedirect", "RedirectTargetInputWidget", "value", "tag", "redirect", "label", "sortedTags", "a", "b", "editorInfo", "shownTemplateEditors", "summaryElement", "noTemplatesMessage", "templateName", "templateData", "parameters", "details", "summary", "elementData", "parameterName", "parameterData", "input", "inputLayout", "name", "generationalSuffix", "lastName", "otherNames", "newName", "leadingArticle", "CategoryInputWidget", "pages", "page", "windowManager", "templatePreviewDialog", "OutputPreviewDialog", "showChangesDialog", "ChangesDialog", "pageTriageMarkButton", "patrolResponse", "redirectValue", "oldTarget", "targetChanged", "tagsChanged", "tagArgumentsChanged", "tagsWithParameters", "data", "oldTagData", "foundTagEditorData", "template", "parameter", "oldArgument", "argument", "newArgument", "defaultSortChanged", "categoriesChanged", "category", "changes", "tagData", "originalRedirectTags", "match", "newTag", "formattedArguments", "index", "foundTemplateEditor", "foundParameterEditor", "errors", "destination", "tags", "destinationData", "errorCode", "destinationParseResult", "destinationRedirect", "section", "destinationContent", "anchor", "part", "targetIsDisambiguationPage", "targetIsSurnameList", "toDisambiguationPageTags", "toSurnameListTags", "taggedAsRedirectToDisambiguationPage", "taggedAsRedirectToSurnameList", "foundParameter", "elementsToDisable", "element", "title", "message", "autoFixes", "warningMessage", "autoFixButton", "autoFix", "output", "fromMove", "patrolLink", "markReviewedButton", "errorInfo", "strayText", "defaultSort", "categories", "parsedTarget", "formattedTitle", "tagsWithArguments", "foundArgumentEditor", "lastNumberParameterIndex", "mappedArguments", "text", "dependencies", "RedirectHelperDialog", "RedirectHelper", "pageInfo", "dialogInfo", "button", "portletLink", "event"]
}
