User:Eejit43/scripts/to-monitor-list.js
Appearance
From Wikipedia, the free encyclopedia
< 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 code will be executed when previewing this page.
Documentation for this user script can be added at User:Eejit43/scripts/to-monitor-list. This user script seems to have an accompanying .json page at User:Eejit43/scripts/to-monitor-list.json.
// <pre> // 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";(()=>{vara=newmw.Api;asyncfunctionl(c){return(awaita.get({action:"query",formatversion:"2",prop:"revisions",rvprop:"content",rvslots:"main",titles:c})).query.pages[0].revisions?.[0].slots.main.content.trim()}varo=class{link;toCheck;totalToCheck;isRunning=!1;handledRequests=0;load(){mw.util.addCSS(` .to-monitor-list-count { cursor: pointer; }`);leti=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",asyncr=>{if(r.preventDefault(),!this.isRunning){this.isRunning=!0;for(leteofdocument.querySelectorAll(".to-monitor-list-count"))e.id&&(e.innerHTML='<span style="color: #ed8e07; font-weight: bold">?</span>');awaitthis.loadToCheckData();for(leteofthis.toCheck.categories)this.handleCheck(e,async()=>{lett=awaita.get({action:"query",list:"search",srinfo:"totalhits",srnamespace:this.getCategory(e),srsearch:`incategory:"${e.category}"`}).catch((s,n)=>(mw.notify(`An error occurred while trying to get category members: ${n?.error?.info??"Unknown error"} (${s})`,{type:"error"}),null));if(t)returnt.query.searchinfo.totalhits});for(leteofthis.toCheck.searches)this.handleCheck(e,async()=>{lett=awaita.get({action:"query",list:"search",srinfo:"totalhits",srnamespace:this.getCategory(e),srsearch:e.search}).catch((s,n)=>(mw.notify(`An error occurred while trying to get search results: ${n?.error?.info??"Unknown error"} (${s})`,{type:"error"}),null));if(t)returnt.query.searchinfo.totalhits});for(leteofthis.toCheck.whatLinksHere)this.handleCheck(e,async()=>{lett=awaita.get({action:"query",list:"backlinks",bllimit:500,blnamespace:this.getCategory(e),bltitle:e.title}).catch((s,n)=>(mw.notify(`An error occurred while trying to get backlinks: ${n?.error?.info??"Unknown error"} (${s})`,{type:"error"}),null));if(t)returnt.query.backlinks.length});for(leteofthis.toCheck.transclusions)this.handleCheck(e,async()=>{lett=awaita.get({action:"query",list:"embeddedin",eilimit:500,einamespace:this.getCategory(e),eititle:e.title}).catch((s,n)=>(mw.notify(`An error occurred while trying to get transclusions: ${n?.error?.info??"Unknown error"} (${s})`,{type:"error"}),null));if(t)returnt.query.embeddedin.length})}}),i.querySelector("a").replaceWith(this.link),document.querySelector("h2#Stuff_to_monitor + .mw-editsection").after(i);for(letrofdocument.querySelectorAll(".to-monitor-list-count"))r.addEventListener("click",()=>{r.innerHTML='<span style="color: #00733f">None</span>'})}asyncloadToCheckData(){this.toCheck=JSON.parse(awaitl("User:Eejit43/scripts/to-monitor-list.json")??"{}"),this.totalToCheck=Object.values(this.toCheck).flat().length}asynchandleCheck(i,r){lete=awaitr();if(e===void0)return;lett=document.querySelector(`#to-monitor-list-${i.id}`);if(!t)returnmw.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({namespaces:i,notNamespaces:r}){if(!i&&!r)return0;if(i){lete=i.map(t=>Object.entries(mw.config.get("wgFormattedNamespaces")).find(([,s])=>s===t)).filter(Boolean);returne.length>0?e.map(t=>Number.parseInt(t[0])):0}elsereturnObject.entries(mw.config.get("wgFormattedNamespaces")).filter(([,e])=>!r.includes(e||"Article")).map(([e])=>Number.parseInt(e))}};mw.loader.using(["mediawiki.util"],()=>{mw.config.get("wgPageName")==="User:Eejit43"&&newo().load()});})(); // </pre> //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../utility.ts", "../scripts/to-monitor-list.ts"],
  "sourcesContent": ["import type { ApiQueryRevisionsParams } from 'types-mediawiki-api';\nimport type { PageRevisionsResult } from './global-types';\n\nexport const api = new mw.Api();\n\n/**\n * Fetches the content of a given page.\n * @param title The title to fetch.\n */\nexport async function getPageContent(title: string) {\n    return (\n        (await 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", "import type { ApiQueryBacklinksParams, ApiQuerySearchParams } from 'types-mediawiki-api';\nimport type { BacklinksResult, EmbeddedinResult, MediaWikiDataError, SearchResult } from '../global-types';\nimport { api, getPageContent } from '../utility';\n\ninterface SearchData {\n    categories: { id: string; category: string; namespaces?: string[]; notNamespaces?: string[] }[];\n    searches: { id: string; search: string; namespaces?: string[]; notNamespaces?: string[] }[];\n    whatLinksHere: { id: string; title: string; namespaces?: string[]; notNamespaces?: string[] }[];\n    transclusions: { id: string; title: string; namespaces?: string[]; notNamespaces?: string[] }[];\n}\n\ntype SearchDataCheck = SearchData[keyof SearchData][0];\n\n/**\n * An instance of this class handles the entire functionality of the to-monitor-list script.\n */\nclass MonitoringListManager {\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        mw.util.addCSS(`\n.to-monitor-list-count {\n    cursor: pointer;\n}`);\n\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            for (const element of document.querySelectorAll('.to-monitor-list-count'))\n                if (element.id) element.innerHTML = '<span style=\"color: #ed8e07; font-weight: bold\">?</span>';\n\n            await this.loadToCheckData();\n\n            for (const check of this.toCheck.categories)\n                void this.handleCheck(check, async () => {\n                    const data = (await 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, errorInfo) => {\n                            mw.notify(\n                                `An error occurred while trying to get category members: ${(errorInfo as MediaWikiDataError)?.error?.info ?? 'Unknown error'} (${errorCode})`,\n                                { type: 'error' },\n                            );\n                            return null;\n                        })) as SearchResult | null;\n                    if (!data) return;\n\n                    return data.query!.searchinfo.totalhits;\n                });\n\n            for (const check of this.toCheck.searches)\n                void this.handleCheck(check, async () => {\n                    const data = (await 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, errorInfo) => {\n                            mw.notify(\n                                `An error occurred while trying to get search results: ${(errorInfo as MediaWikiDataError)?.error?.info ?? 'Unknown error'} (${errorCode})`,\n                                { type: 'error' },\n                            );\n                            return null;\n                        })) as SearchResult | null;\n                    if (!data) return;\n\n                    return data.query!.searchinfo.totalhits;\n                });\n\n            for (const check of this.toCheck.whatLinksHere)\n                void this.handleCheck(check, async () => {\n                    const data = (await 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, errorInfo) => {\n                            mw.notify(\n                                `An error occurred while trying to get backlinks: ${(errorInfo as MediaWikiDataError)?.error?.info ?? 'Unknown error'} (${errorCode})`,\n                                { type: 'error' },\n                            );\n                            return null;\n                        })) as BacklinksResult | null;\n                    if (!data) return;\n\n                    return data.query!.backlinks.length;\n                });\n\n            for (const check of this.toCheck.transclusions)\n                void this.handleCheck(check, async () => {\n                    const data = (await 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, errorInfo) => {\n                            mw.notify(\n                                `An error occurred while trying to get transclusions: ${(errorInfo as MediaWikiDataError)?.error?.info ?? 'Unknown error'} (${errorCode})`,\n                                { type: 'error' },\n                            );\n                            return null;\n                        })) as EmbeddedinResult | null;\n                    if (!data) return;\n\n                    return data.query!.embeddedin.length;\n                });\n        });\n\n        fullLinkElement.querySelector('a')!.replaceWith(this.link);\n\n        document.querySelector('h2#Stuff_to_monitor + .mw-editsection')!.after(fullLinkElement);\n\n        for (const element of document.querySelectorAll('.to-monitor-list-count'))\n            element.addEventListener('click', () => {\n                element.innerHTML = '<span style=\"color: #00733f\">None</span>';\n            });\n    }\n\n    /**\n     * Loads the data of checks to handle.\n     */\n    public async loadToCheckData() {\n        this.toCheck = JSON.parse((await getPageContent('User:Eejit43/scripts/to-monitor-list.json')) ?? '{}') 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 | undefined>) {\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 =\n            count === 0\n                ? '<span style=\"color: #00733f\">None</span>'\n                : `<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.namespaces The namespace to search in.\n     * @param check.notNamespaces The namespaces to exclude from the search.\n     * @returns The category ID or array of category IDs.\n     */\n    private getCategory({ namespaces, notNamespaces }: { namespaces?: string[]; notNamespaces?: string[] }) {\n        if (!namespaces && !notNamespaces) return 0;\n        else if (namespaces) {\n            const foundNamespaces = namespaces\n                .map((namespace) =>\n                    Object.entries(mw.config.get('wgFormattedNamespaces')).find(([, namespaceName]) => namespaceName === namespace),\n                )\n                .filter(Boolean) as [string, string][];\n\n            return foundNamespaces.length > 0 ? foundNamespaces.map((foundNamespace) => Number.parseInt(foundNamespace[0])) : 0;\n        } else\n            return Object.entries(mw.config.get('wgFormattedNamespaces'))\n                .filter(([, namespaceName]) => !notNamespaces!.includes(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": ["api", "getPageContent", "title", "MonitoringListManager", "fullLinkElement", "event", "element", "check", "data", "api", "errorCode", "errorInfo", "getPageContent", "handler", "count", "namespaces", "notNamespaces", "foundNamespaces", "namespace", "namespaceName", "foundNamespace", "namespaceId"]
}
