Note: After saving, you have to bypass your browser's cache to see the changes. Google Chrome, Firefox, Microsoft Edge and Safari: Hold down the ⇧ Shift key and click the Reload toolbar button. For details and instructions about other browsers, see Wikipedia:Bypass your cache.
// <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/aligner.ts

"use strict";mw.config.get("wgNamespaceNumber")<0||mw.config.get("wgIsProbablyEditable")&&mw.loader.using(["mediawiki.util","jquery.textSelection"],()=>{mw.util.addCSS(`
#align-parameters {
    display: none;
}

#content:has(#wpTextbox1) #align-parameters {
    display: unset;
}`),mw.util.addPortletLink(mw.config.get("skin")==="minerva"?"p-navigation":"p-cactions","#","Align template parameters","align-parameters").addEventListener("click",s=>{s.preventDefault();const n=$("#wpTextbox1");if(n.length===0)return mw.notify("Edit box not found!",{type:"error",autoHideSeconds:"short"});const e=n.textSelection("getContents");if(!e)return mw.notify("Edit box value not found!",{type:"error",autoHideSeconds:"short"});let o=0;function w(t){if(t==="")return mw.notify("Infobox not found!",{type:"error",autoHideSeconds:"short"});if(a!==0)return mw.notify("Template was not properly closed!",{type:"error",autoHideSeconds:"short"});let r=0;const l=String(t),f=t.split(`
`),x=[];for(const g of f){const u=splitIntoParameters(g.trim());for(const p of u){const h=p.trim();if(!h.startsWith("|")||h.split("=").length!==2){x.push(h);continue}let[m,S]=splitParameter(h);m=m.slice(1).trim(),m.length>r&&(r=m.length),x.push("| "+m+"="+S)}}let c="";r+=2;for(let g of x){const u=splitParameter(g);if(u.length<2){c+=g+=`
`;continue}let p=u[0].trim();for(;p.length<r;)p+=" ";c+=p+" = "+u[1].trim()+`
`}c.endsWith(`
`)&&(c=c.slice(0,-1)),n.textSelection("setContents",n.textSelection("getContents").replace(l,c).replace(/\n+$/,"")),n.textSelection("setSelection",{start:0})}let d="",a=0;const y=["infobox","speciesbox","taxobox","automatic taxobox","osm location map","motorsport season"];for(let t=0;t<e.length;t++){let r=!1;for(let l of y){l="{{"+l;const f=l.length;e.length-t>f&&(e.slice(t,t+f).toLowerCase()===l||e.slice(t,t+f).toLowerCase()===l.replace(" ","_"))&&(a++,d+=e[t],r=!0)}a>=1&&!r&&(d+=e[t],e[t]==="{"?a++:e[t]==="}"&&(a--,a===0&&(o++,w(d),d="")))}mw.notify(`Successfully aligned ${o} template${o>1?"s":""}!`,{type:"success",autoHideSeconds:"short"})})});function splitParameter(i){const s=i.split("=");return s.length<=2?s:[s.shift(),s.join("=")]}function splitIntoParameters(i){if(i.startsWith("{{")&&i.endsWith("}}"))return i.includes("|")?["{{"+splitIntoParameters(i.slice(2,-2))[0],...splitIntoParameters(i.slice(2,-2)).slice(1),"}}"]:[i];const s=[];let n="",e=0;for(const o of i)n+=o,o==="{"||o==="["?e+=1:o===""||o==="]"?e--:o==="|"&&e===0&&n.trim()!=="|"&&(s.push(n.slice(0,-1).trim()),n="|");return s.push(n),s}

// </nowiki>
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../scripts/aligner.ts"],
  "sourcesContent": ["(() => {\n    if (mw.config.get('wgNamespaceNumber') < 0) return; // Don't run in virtual namespaces\n    if (!mw.config.get('wgIsProbablyEditable')) return; // Don't run if user can't edit page\n\n    mw.loader.using(['mediawiki.util', 'jquery.textSelection'], () => {\n        mw.util.addCSS(`\n#align-parameters {\n    display: none;\n}\n\n#content:has(#wpTextbox1) #align-parameters {\n    display: unset;\n}`);\n\n        const link = mw.util.addPortletLink(mw.config.get('skin') === 'minerva' ? 'p-navigation' : 'p-cactions', '#', 'Align template parameters', 'align-parameters')!;\n        link.addEventListener('click', (event) => {\n            event.preventDefault();\n\n            const editBox = $('#wpTextbox1');\n            if (editBox.length === 0) return mw.notify('Edit box not found!', { type: 'error', autoHideSeconds: 'short' });\n\n            const text = editBox.textSelection('getContents');\n            if (!text) return mw.notify('Edit box value not found!', { type: 'error', autoHideSeconds: 'short' });\n\n            let count = 0;\n\n            /**\n             * Finishes processing an infobox and updates the edit box contents.\n             * @param template The template string to process.\n             */\n            function processInfobox(template: string) {\n                if (template === '') return mw.notify('Infobox not found!', { type: 'error', autoHideSeconds: 'short' });\n\n                if (open !== 0) return mw.notify('Template was not properly closed!', { type: 'error', autoHideSeconds: 'short' });\n\n                let maxLength = 0;\n\n                const origTemplate = String(template);\n                const lines = template.split('\\n');\n                const newLines = [];\n\n                for (const line of lines) {\n                    const parametersInLine = splitIntoParameters(line.trim());\n\n                    for (const parameter of parametersInLine) {\n                        const line = parameter.trim();\n                        if (!line.startsWith('|') || line.split('=').length !== 2) {\n                            newLines.push(line);\n                            continue;\n                        }\n\n                        let [firstPart, lastPart] = splitParameter(line) as [string, string]; // eslint-disable-line prefer-const\n                        firstPart = firstPart.slice(1).trim();\n\n                        if (firstPart.length > maxLength) maxLength = firstPart.length;\n\n                        newLines.push('| ' + firstPart + '=' + lastPart);\n                    }\n                }\n\n                let output = '';\n\n                maxLength += 2; // to include '| '\n\n                for (let line of newLines) {\n                    const parts = splitParameter(line) as [string, string];\n\n                    if (parts.length < 2) {\n                        output += line += '\\n';\n                        continue;\n                    }\n\n                    let firstPart = parts[0].trim();\n\n                    while (firstPart.length < maxLength) firstPart += ' ';\n\n                    output += firstPart + ' = ' + parts[1].trim() + '\\n';\n                }\n\n                if (output.endsWith('\\n')) output = output.slice(0, -1);\n\n                editBox.textSelection('setContents', editBox.textSelection('getContents').replace(origTemplate, output).replace(/\\n+$/, ''));\n\n                editBox.textSelection('setSelection', { start: 0 });\n            }\n\n            let template = '';\n            let open = 0;\n\n            const searches = ['infobox', 'speciesbox', 'taxobox', 'automatic taxobox', 'osm location map', 'motorsport season'];\n\n            for (let index = 0; index < text.length; index++) {\n                let foo = false;\n\n                for (let search of searches) {\n                    search = '{{' + search;\n                    const searchLength = search.length;\n\n                    if (\n                        text.length - index > searchLength &&\n                        (text.slice(index, index + searchLength).toLowerCase() === search || text.slice(index, index + searchLength).toLowerCase() === search.replace(' ', '_'))\n                    ) {\n                        open++;\n                        template += text[index];\n                        foo = true;\n                    }\n                }\n\n                if (open >= 1 && !foo) {\n                    template += text[index];\n\n                    if (text[index] === '{') open++;\n                    else if (text[index] === '}') {\n                        open--;\n\n                        if (open === 0) {\n                            count++;\n                            processInfobox(template);\n                            template = '';\n                        }\n                    }\n                }\n            }\n\n            mw.notify(`Successfully aligned ${count} template${count > 1 ? 's' : ''}!`, { type: 'success', autoHideSeconds: 'short' });\n        });\n    });\n})();\n\n/**\n * Splits a string into an Array containing the key and value.\n * @param string The full string to split.\n */\nfunction splitParameter(string: string) {\n    const split = string.split('=');\n    if (split.length <= 2) return split;\n\n    const first = split.shift();\n    return [first, split.join('=')];\n}\n\n/**\n * Splits a template into an Array with all parameters.\n * @param string The template to process.\n */\nfunction splitIntoParameters(string: string): string[] {\n    if (string.startsWith('{{') && string.endsWith('}}')) {\n        if (!string.includes('|')) return [string];\n\n        const results = splitIntoParameters(string.slice(2, -2));\n        return ['{{' + results[0], ...splitIntoParameters(string.slice(2, -2)).slice(1), '}}'];\n    }\n\n    const parameters = [];\n    let temporary = '';\n    let open = 0;\n\n    for (const char of string) {\n        temporary += char;\n\n        if (char === '{' || char === '[') open += 1;\n        else if (char === '' || char === ']') open--;\n        else if (char === '|' && open === 0 && temporary.trim() !== '|') {\n            parameters.push(temporary.slice(0, -1).trim());\n            temporary = '|';\n        }\n    }\n\n    parameters.push(temporary);\n\n    return parameters;\n}\n"],
  "mappings": "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",
  "names": ["event", "editBox", "text", "count", "processInfobox", "template", "open", "maxLength", "origTemplate", "lines", "newLines", "line", "parametersInLine", "parameter", "firstPart", "lastPart", "output", "parts", "searches", "index", "foo", "search", "searchLength", "string", "split", "parameters", "temporary", "char"]
}
