User:Eejit43/scripts/to-monitor-list.js

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/to-monitor-list.ts

"use strict";class MonitoringListManager{api=new mw.Api;link;toCheck;totalToCheck;isRunning=!1;handledRequests=0;load(){const i=document.querySelector(".mw-editsection").cloneNode(!0);this.link=document.createElement("a"),this.link.href="#",this.link.style.fontWeight="bold",this.link.textContent="Add missing counts",this.link.addEventListener("click",async a=>{if(a.preventDefault(),!this.isRunning){this.isRunning=!0,await this.loadToCheckData();for(const e of this.toCheck.categories)this.handleCheck(e,async()=>{const t=await this.api.get({action:"query",list:"search",srinfo:"totalhits",srnamespace:this.getCategory(e),srsearch:`incategory:"${e.category}"`}).catch((r,s)=>(mw.notify(`An error occurred while trying to get category members: ${s?.error.info??"Unknown error"} (${r})`,{type:"error"}),null));if(t)return t.query.searchinfo.totalhits});for(const e of this.toCheck.searches)this.handleCheck(e,async()=>{const t=await this.api.get({action:"query",list:"search",srinfo:"totalhits",srnamespace:this.getCategory(e),srsearch:e.search}).catch((r,s)=>(mw.notify(`An error occurred while trying to get search results: ${s?.error.info??"Unknown error"} (${r})`,{type:"error"}),null));if(t)return t.query.searchinfo.totalhits});for(const e of this.toCheck.whatLinksHere)this.handleCheck(e,async()=>{const t=await this.api.get({action:"query",list:"backlinks",bllimit:500,blnamespace:this.getCategory(e),bltitle:e.title}).catch((r,s)=>(mw.notify(`An error occurred while trying to get backlinks: ${s?.error.info??"Unknown error"} (${r})`,{type:"error"}),null));if(t)return t.query.backlinks.length});for(const e of this.toCheck.transclusions)this.handleCheck(e,async()=>{const t=await this.api.get({action:"query",list:"embeddedin",eilimit:500,einamespace:this.getCategory(e),eititle:e.title}).catch((r,s)=>(mw.notify(`An error occurred while trying to get transclusions: ${s?.error.info??"Unknown error"} (${r})`,{type:"error"}),null));if(t)return t.query.embeddedin.length})}}),i.querySelector("a").replaceWith(this.link),document.querySelector("#Stuff_to_monitor.mw-headline").after(i)}async loadToCheckData(){this.toCheck=JSON.parse((await this.api.get({action:"query",formatversion:"2",prop:"revisions",rvprop:"content",rvslots:"main",titles:"User:Eejit43/scripts/to-monitor-list.json"})).query.pages[0].revisions[0].slots.main.content),this.totalToCheck=Object.values(this.toCheck).flat().length}async handleCheck(i,a){const e=await a();if(e===void 0)return;const t=document.querySelector(`#to-monitor-list-${i.id}`);if(!t)return mw.notify(`Failed to find element for ID "${i.id}"`);t.innerHTML=e===0?'<span style="color: #00733f">None</span>':`<b><span style="color: #bd2828">${e===500?"500+":e}</span></b>`,this.handledRequests++,this.link.textContent=`Add missing counts (${this.handledRequests}/${this.totalToCheck} loaded)`,this.handledRequests===this.totalToCheck&&setTimeout(()=>{this.isRunning=!1,this.handledRequests=0,this.link.textContent="Add missing counts"},1e3)}getCategory({namespace:i,notNamespace:a}){if(!i&&!a)return 0;if(i){const e=Object.entries(mw.config.get("wgFormattedNamespaces")).find(([,t])=>t===i);return e?Number.parseInt(e[0]):0}else return Object.entries(mw.config.get("wgFormattedNamespaces")).filter(([,e])=>a!==(e||"Article")).map(([e])=>Number.parseInt(e))}}mw.loader.using(["mediawiki.util"],()=>{mw.config.get("wgPageName")==="User:Eejit43"&&new MonitoringListManager().load()});

// </nowiki>
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../scripts/to-monitor-list.ts"],
  "sourcesContent": ["import { ApiQueryBacklinksParams, ApiQueryRevisionsParams, ApiQuerySearchParams } from 'types-mediawiki/api_params';\nimport { BacklinksResult, EmbeddedinResult, MediaWikiDataError, PageRevisionsResult, SearchResult } from '../global-types';\n\ninterface SearchData {\n    categories: { id: string; category: string; namespace?: string; notNamespace?: string }[];\n    searches: { id: string; search: string; namespace?: string; notNamespace?: string }[];\n    whatLinksHere: { id: string; title: string; namespace?: string; notNamespace?: string }[];\n    transclusions: { id: string; title: string; namespace?: string; notNamespace?: string }[];\n}\n\ntype SearchDataCheck = SearchData['categories'][0] | SearchData['searches'][0] | SearchData['whatLinksHere'][0] | SearchData['transclusions'][0];\n\n/**\n * An instance of this class handles the entire functionality of the to-monitor-list script.\n */\nclass MonitoringListManager {\n    private api = new mw.Api();\n\n    private link!: HTMLAnchorElement;\n\n    private toCheck!: SearchData;\n    private totalToCheck!: number;\n\n    private isRunning = false;\n\n    private handledRequests = 0;\n\n    /**\n     * Loads the \"Add missing counts\" link element.\n     */\n    public load() {\n        const fullLinkElement = document.querySelector('.mw-editsection')!.cloneNode(true) as HTMLSpanElement;\n\n        this.link = document.createElement('a');\n        this.link.href = '#';\n        this.link.style.fontWeight = 'bold';\n        this.link.textContent = 'Add missing counts';\n        this.link.addEventListener('click', async (event) => {\n            event.preventDefault();\n\n            if (this.isRunning) return;\n            this.isRunning = true;\n\n            await this.loadToCheckData();\n\n            for (const check of this.toCheck.categories)\n                this.handleCheck(check, async () => {\n                    const data: SearchResult | null = await this.api\n                        .get({\n                            action: 'query',\n                            list: 'search',\n                            srinfo: 'totalhits',\n                            srnamespace: this.getCategory(check),\n                            srsearch: `incategory:\"${check.category}\"`,\n                        } satisfies ApiQuerySearchParams)\n                        .catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                            mw.notify(`An error occurred while trying to get category members: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                            return null;\n                        });\n                    if (!data) return;\n\n                    return (data as SearchResult).query.searchinfo.totalhits;\n                });\n\n            for (const check of this.toCheck.searches)\n                this.handleCheck(check, async () => {\n                    const data: SearchResult | null = await this.api\n                        .get({\n                            action: 'query',\n                            list: 'search',\n                            srinfo: 'totalhits',\n                            srnamespace: this.getCategory(check),\n                            srsearch: check.search,\n                        } satisfies ApiQuerySearchParams)\n                        .catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                            mw.notify(`An error occurred while trying to get search results: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                            return null;\n                        });\n                    if (!data) return;\n\n                    return (data as SearchResult).query.searchinfo.totalhits;\n                });\n\n            for (const check of this.toCheck.whatLinksHere)\n                this.handleCheck(check, async () => {\n                    const data: BacklinksResult | null = await this.api\n                        .get({\n                            action: 'query',\n                            list: 'backlinks',\n                            bllimit: 500,\n                            blnamespace: this.getCategory(check),\n                            bltitle: check.title,\n                        } satisfies ApiQueryBacklinksParams)\n                        .catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                            mw.notify(`An error occurred while trying to get backlinks: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                            return null;\n                        });\n                    if (!data) return;\n\n                    return (data as BacklinksResult).query.backlinks.length;\n                });\n\n            for (const check of this.toCheck.transclusions)\n                this.handleCheck(check, async () => {\n                    const data: EmbeddedinResult | null = await this.api\n                        .get({\n                            action: 'query',\n                            list: 'embeddedin',\n                            eilimit: 500,\n                            einamespace: this.getCategory(check),\n                            eititle: check.title,\n                        } satisfies ApiQueryBacklinksParams)\n                        .catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                            mw.notify(`An error occurred while trying to get transclusions: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                            return null;\n                        });\n                    if (!data) return;\n\n                    return (data as EmbeddedinResult).query.embeddedin.length;\n                });\n        });\n\n        fullLinkElement.querySelector('a')!.replaceWith(this.link);\n\n        document.querySelector('#Stuff_to_monitor.mw-headline')!.after(fullLinkElement);\n    }\n\n    /**\n     * Loads the data of checks to handle.\n     */\n    public async loadToCheckData() {\n        this.toCheck = 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/to-monitor-list.json',\n                } satisfies ApiQueryRevisionsParams)) as PageRevisionsResult\n            ).query.pages[0].revisions[0].slots.main.content,\n        ) as SearchData;\n\n        this.totalToCheck = Object.values(this.toCheck).flat().length;\n    }\n\n    /**\n     * Handles a given check.\n     * @param check The check data to handle.\n     * @param handler The handler to find the count from a check.\n     */\n    private async handleCheck(check: SearchDataCheck, handler: () => Promise<number | void>) {\n        const count = await handler();\n        if (count === undefined) return;\n\n        const element = document.querySelector(`#to-monitor-list-${check.id}`);\n        if (!element) return mw.notify(`Failed to find element for ID \"${check.id}\"`);\n        element.innerHTML = count === 0 ? '<span style=\"color: #00733f\">None</span>' : `<b><span style=\"color: #bd2828\">${count === 500 ? '500+' : count}</span></b>`;\n\n        this.handledRequests++;\n        this.link.textContent = `Add missing counts (${this.handledRequests}/${this.totalToCheck} loaded)`;\n\n        if (this.handledRequests === this.totalToCheck)\n            setTimeout(() => {\n                this.isRunning = false;\n\n                this.handledRequests = 0;\n                this.link.textContent = 'Add missing counts';\n            }, 1000);\n    }\n\n    /**\n     * Parses the searched categories from the check object.\n     * @param check The check object.\n     * @param check.namespace The namespace to search in.\n     * @param check.notNamespace The namespace to exclude from the search.\n     * @returns The category ID or list of category IDs (separated by '|').\n     */\n    private getCategory({ namespace, notNamespace }: { namespace?: string; notNamespace?: string }) {\n        if (!namespace && !notNamespace) return 0;\n        else if (namespace) {\n            const foundNamespace = Object.entries(mw.config.get('wgFormattedNamespaces')).find(([, namespaceName]) => namespaceName === namespace);\n\n            return foundNamespace ? Number.parseInt(foundNamespace[0]) : 0;\n        } else\n            return Object.entries(mw.config.get('wgFormattedNamespaces'))\n                .filter(([, namespaceName]) => notNamespace !== (namespaceName || 'Article'))\n                .map(([namespaceId]) => Number.parseInt(namespaceId));\n    }\n}\n\nmw.loader.using(['mediawiki.util'], () => {\n    if (mw.config.get('wgPageName') === 'User:Eejit43') new MonitoringListManager().load();\n});\n"],
  "mappings": "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",
  "names": ["fullLinkElement", "event", "check", "data", "errorCode", "errorInfo", "handler", "count", "element", "namespace", "notNamespace", "foundNamespace", "namespaceName", "namespaceId"]
}
