wikimedia/mediawiki-core

View on GitHub
resources/Resources.php

Summary

Maintainability
F
1 wk
Test Coverage
<?php
/**
 * Definition of core ResourceLoader modules.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 * http://www.gnu.org/copyleft/gpl.html
 *
 * @file
 */

use MediaWiki\Config\Config;
use MediaWiki\HookContainer\HookRunner;
use MediaWiki\Languages\LanguageNameUtils;
use MediaWiki\MainConfigNames;
use MediaWiki\MediaWikiServices;
use MediaWiki\Parser\Sanitizer;
use MediaWiki\ResourceLoader as RL;
use MediaWiki\ResourceLoader\CodexModule;
use MediaWiki\ResourceLoader\Context;
use MediaWiki\ResourceLoader\FilePath;
use MediaWiki\ResourceLoader\ForeignApiModule;
use MediaWiki\ResourceLoader\LessVarFileModule;
use MediaWiki\ResourceLoader\Module;
use MediaWiki\ResourceLoader\MwUrlModule;
use MediaWiki\ResourceLoader\OOUIFileModule;
use MediaWiki\ResourceLoader\OOUIIconPackModule;
use MediaWiki\ResourceLoader\OOUIImageModule;
use MediaWiki\ResourceLoader\ResourceLoader;
use MediaWiki\ResourceLoader\SiteModule;
use MediaWiki\ResourceLoader\SiteStylesModule;
use MediaWiki\ResourceLoader\SkinModule;
use MediaWiki\ResourceLoader\UserModule;
use MediaWiki\ResourceLoader\UserOptionsModule;
use MediaWiki\ResourceLoader\UserStylesModule;
use MediaWiki\ResourceLoader\WikiModule;
use MediaWiki\SpecialPage\ChangesListSpecialPage;
use MediaWiki\Title\Title;
use Wikimedia\ParamValidator\TypeDef\ExpiryDef;

if ( !defined( 'MEDIAWIKI' ) ) {
    die( 'Not an entry point.' );
}

global $wgResourceBasePath;

return [
    // Scripts managed by the local wiki (stored in the MediaWiki namespace)
    'site' => [ 'class' => SiteModule::class ],
    'site.styles' => [ 'class' => SiteStylesModule::class ],
    'noscript' => [
        'class' => WikiModule::class,
        'styles' => [ 'MediaWiki:Noscript.css' ],
        'group' => Module::GROUP_NOSCRIPT,
    ],
    'filepage' => [
        'class' => WikiModule::class,
        'styles' => [ 'MediaWiki:Filepage.css' ],
    ],

    // Scripts managed by the current user (stored in their user space)
    'user' => [ 'class' => UserModule::class ],
    'user.styles' => [ 'class' => UserStylesModule::class ],

    'user.options' => [ 'class' => UserOptionsModule::class ],

    'mediawiki.skinning.interface' => [
        'class' => SkinModule::class,
        'features' => [
            'elements' => true,
            'content-media' => true,
            'interface' => true,
            'interface-message-box' => true,
            'logo' => true,
        ],
    ],
    'jquery.makeCollapsible.styles' => [
        'class' => LessVarFileModule::class,
        'lessMessages' => [
            'collapsible-collapse',
            'collapsible-expand',
        ],
        'styles' => [
            'resources/src/jquery/jquery.makeCollapsible.styles.less',
        ],
    ],

    'mediawiki.skinning.content.parsoid' => [
        // Style Parsoid HTML+RDFa output consistent with wikitext from PHP parser
        // with the interface.css styles; skinStyles should be used if your
        // skin over-rides common content styling.
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.skinning/content.parsoid.less',
                'resources/src/mediawiki.skinning/content.media-common.less',
                'resources/src/mediawiki.skinning/content.media-screen.less',
                'resources/src/mediawiki.page.gallery.styles/content.media.less',
            ],
        ],
    ],

    /* Base modules */
    // These modules' dependencies MUST also be included in StartUpModule::getBaseModules().
    // These modules' dependencies MUST be dependency-free (having dependencies would cause recursion).

    'jquery' => [
        'scripts' => [
            'resources/lib/jquery/jquery.js'
        ],
    ],
    'web2017-polyfills' => [
        'scripts' => [
            'resources/lib/intersection-observer/intersection-observer.js',
            'resources/lib/fetch-polyfill/fetch.umd.js',
            // The URL polyfill depends on the following in addition to the ES6 baseline
            // https://github.com/Financial-Times/polyfill-library/blob/v3.110.1/polyfills/URL/config.toml#L10
            // - ES6 Symbol.iterator (no fill needed, used conditionally)
            'resources/lib/url/URL.js',
            'resources/lib/url/URL-toJSON.js',
        ],
        'skipFunction' => 'resources/src/skip-web2017-polyfills.js',
        'dependencies' => []
    ],
    'mediawiki.base' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.base',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.base",
        'packageFiles' => [
            // This MUST be kept in sync with maintenance/jsduck/eg-iframe.html
            'mediawiki.base.js',
            'log.js',
            'errorLogger.js',

            // (not this though)
            [ 'name' => 'config.json', 'callback' => [ ResourceLoader::class, 'getSiteConfigSettings' ] ],
            [
                'name' => 'user.json',
                'callback' => static function ( Context $context ) {
                    $services = MediaWikiServices::getInstance();

                    return ResourceLoader::getUserDefaults(
                        $context,
                        $services->getHookContainer(),
                        $services->getUserOptionsLookup()
                    );
                }
            ],
        ],
        'dependencies' => 'jquery',
    ],

    /* jQuery Plugins */

    'jquery.chosen' => [
        'scripts' => 'resources/lib/jquery.chosen/chosen.jquery.js',
        'styles' => 'resources/lib/jquery.chosen/chosen.css',
    ],
    'jquery.client' => [
        'scripts' => 'resources/lib/jquery.client/jquery.client.js',
    ],
    'jquery.confirmable' => [
        'scripts' => [
            'resources/src/jquery/jquery.confirmable.js',
            'resources/src/jquery/jquery.confirmable.mediawiki.js',
        ],
        'messages' => [
            'confirmable-confirm',
            'confirmable-yes',
            'confirmable-no',
            'word-separator',
        ],
        'styles' => 'resources/src/jquery/jquery.confirmable.css',
        'dependencies' => 'mediawiki.jqueryMsg',
    ],
    'jquery.highlightText' => [
        'scripts' => 'resources/src/jquery/jquery.highlightText.js',
        'dependencies' => [
            'mediawiki.util',
        ],
    ],
    'jquery.i18n' => [
        'scripts' => [
            'resources/lib/jquery.i18n/src/jquery.i18n.js',
            'resources/lib/jquery.i18n/src/jquery.i18n.messagestore.js',
            'resources/lib/jquery.i18n/src/jquery.i18n.parser.js',
            'resources/lib/jquery.i18n/src/jquery.i18n.emitter.js',
            'resources/lib/jquery.i18n/src/jquery.i18n.emitter.bidi.js',
            'resources/lib/jquery.i18n/src/jquery.i18n.language.js',
            'resources/lib/jquery.i18n/src/jquery.i18n.fallbacks.js',
        ],
        'dependencies' => 'mediawiki.libs.pluralruleparser',
        'languageScripts' => [
            'bs' => 'resources/lib/jquery.i18n/src/languages/bs.js',
            'dsb' => 'resources/lib/jquery.i18n/src/languages/dsb.js',
            'fi' => 'resources/lib/jquery.i18n/src/languages/fi.js',
            'ga' => 'resources/lib/jquery.i18n/src/languages/ga.js',
            'he' => 'resources/lib/jquery.i18n/src/languages/he.js',
            'hsb' => 'resources/lib/jquery.i18n/src/languages/hsb.js',
            'hu' => 'resources/lib/jquery.i18n/src/languages/hu.js',
            'hy' => 'resources/lib/jquery.i18n/src/languages/hy.js',
            'la' => 'resources/lib/jquery.i18n/src/languages/la.js',
            'ml' => 'resources/lib/jquery.i18n/src/languages/ml.js',
            'os' => 'resources/lib/jquery.i18n/src/languages/os.js',
            'ru' => 'resources/lib/jquery.i18n/src/languages/ru.js',
            'sl' => 'resources/lib/jquery.i18n/src/languages/sl.js',
            'uk' => 'resources/lib/jquery.i18n/src/languages/uk.js',
        ],
    ],
    'jquery.lengthLimit' => [
        'scripts' => 'resources/src/jquery.lengthLimit.js',
        'dependencies' => 'mediawiki.String',
    ],
    'jquery.makeCollapsible' => [
        'dependencies' => [
            'jquery.makeCollapsible.styles',
            'mediawiki.util',
        ],
        'scripts' => 'resources/src/jquery/jquery.makeCollapsible.js',
        'styles' => 'resources/src/jquery/jquery.makeCollapsible.less',
        'messages' => [ 'collapsible-expand', 'collapsible-collapse' ],
    ],
    'jquery.spinner' => [
        'scripts' => 'resources/src/jquery.spinner/spinner.js',
        'dependencies' => [ 'jquery.spinner.styles' ],
    ],
    'jquery.spinner.styles' => [
        'styles' => 'resources/src/jquery.spinner/spinner.less',
    ],
    'jquery.suggestions' => [
        'scripts' => 'resources/src/jquery/jquery.suggestions.js',
        'styles' => 'resources/src/jquery/jquery.suggestions.css',
        'dependencies' => 'jquery.highlightText',
    ],
    'jquery.tablesorter' => [
        'scripts' => 'resources/src/jquery.tablesorter/jquery.tablesorter.js',
        'messages' => [ 'sort-descending', 'sort-ascending', 'sort-initial', 'sort-rowspan-error' ],
        'dependencies' => [
            'jquery.tablesorter.styles',
            'mediawiki.util',
            'mediawiki.language.months',
        ],
    ],
    'jquery.tablesorter.styles' => [
        'styles' => 'resources/src/jquery.tablesorter.styles/jquery.tablesorter.styles.less',
    ],
    'jquery.textSelection' => [
        'scripts' => 'resources/src/jquery/jquery.textSelection.js',
        'dependencies' => [
            'jquery.client',
        ]
    ],

    /* jQuery UI */

    'jquery.ui' => [
        'deprecated' => 'Please use Codex instead.',
        'scripts' => [
            'resources/lib/jquery.ui/jquery.ui.core.js',
            'resources/lib/jquery.ui/jquery.ui.widget.js',
            'resources/lib/jquery.ui/jquery.ui.mouse.js',
            'resources/lib/jquery.ui/jquery.ui.draggable.js',
            'resources/lib/jquery.ui/jquery.ui.droppable.js',
            'resources/lib/jquery.ui/jquery.ui.resizable.js',
            'resources/lib/jquery.ui/jquery.ui.selectable.js',
            'resources/lib/jquery.ui/jquery.ui.sortable.js',
            'resources/lib/jquery.ui/jquery.ui.effect.js',
            'resources/lib/jquery.ui/jquery.ui.accordion.js',
            'resources/lib/jquery.ui/jquery.ui.autocomplete.js',
            'resources/lib/jquery.ui/jquery.ui.button.js',
            'resources/lib/jquery.ui/jquery.ui.datepicker.js',
            'resources/lib/jquery.ui/jquery.ui.dialog.js',
            'resources/lib/jquery.ui/jquery.ui.effect-blind.js',
            'resources/lib/jquery.ui/jquery.ui.effect-clip.js',
            'resources/lib/jquery.ui/jquery.ui.effect-drop.js',
            'resources/lib/jquery.ui/jquery.ui.effect-highlight.js',
            'resources/lib/jquery.ui/jquery.ui.effect-scale.js',
            'resources/lib/jquery.ui/jquery.ui.effect-shake.js',
            'resources/lib/jquery.ui/jquery.ui.menu.js',
            'resources/lib/jquery.ui/jquery.ui.position.js',
            'resources/lib/jquery.ui/jquery.ui.progressbar.js',
            'resources/lib/jquery.ui/jquery.ui.slider.js',
            'resources/lib/jquery.ui/jquery.ui.tabs.js',
            'resources/lib/jquery.ui/jquery.ui.tooltip.js',
        ],
        'languageScripts' => [
            'af' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-af.js',
            'ar' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ar.js',
            'ar-dz' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ar-DZ.js',
            'az' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-az.js',
            'bg' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-bg.js',
            'bs' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-bs.js',
            'ca' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ca.js',
            'cs' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-cs.js',
            'da' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-da.js',
            'de-at' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-de-AT.js',
            'de-ch' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-de-CH.js',
            'de' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-de.js',
            'el' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-el.js',
            'en-au' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-en-AU.js',
            'en-gb' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-en-GB.js',
            'en-nz' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-en-NZ.js',
            'eo' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-eo.js',
            'es' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-es.js',
            'et' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-et.js',
            'eu' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-eu.js',
            'fa' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-fa.js',
            'fi' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-fi.js',
            'fo' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-fo.js',
            'fr' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-fr.js',
            'fr-ch' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-fr-CH.js',
            'gl' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-gl.js',
            'he' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-he.js',
            'hi' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-hi.js',
            'hr' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-hr.js',
            'hu' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-hu.js',
            'hy' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-hy.js',
            'id' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-id.js',
            'is' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-is.js',
            'it' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-it.js',
            'ja' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ja.js',
            'ka' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ka.js',
            'kk' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-kk.js',
            'km' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-km.js',
            'ko' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ko.js',
            'lb' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-lb.js',
            'lt' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-lt.js',
            'lv' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-lv.js',
            'mk' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-mk.js',
            'ml' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ml.js',
            'ms' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ms.js',
            'nl' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-nl.js',
            'nl-be' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-nl-BE.js',
            'no' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-no.js',
            'pl' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-pl.js',
            'pt' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-pt.js',
            'pt-br' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-pt-BR.js',
            'rm' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-rm.js',
            'ro' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ro.js',
            'ru' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ru.js',
            'sk' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-sk.js',
            'sl' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-sl.js',
            'sq' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-sq.js',
            'sr-ec' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-sr.js',
            'sr-el' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-sr-SR.js',
            'sv' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-sv.js',
            'ta' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-ta.js',
            'th' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-th.js',
            'tj' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-tj.js',
            'tr' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-tr.js',
            'uk' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-uk.js',
            'vi' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-vi.js',
            'zh-cn' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-zh-CN.js',
            'zh-hk' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-zh-HK.js',
            'zh-tw' => 'resources/lib/jquery.ui/i18n/jquery.ui.datepicker-zh-TW.js',
        ],
        'skinStyles' => [
            'default' => [
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.core.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.accordion.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.autocomplete.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.button.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.datepicker.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.dialog.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.menu.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.progressbar.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.resizable.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.selectable.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.slider.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.tabs.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.tooltip.css',
                'resources/lib/jquery.ui/themes/smoothness/jquery.ui.theme.css',
            ],
        ],
    ],

    /* Moment.js */

    'moment' => [
        'scripts' => [
            'resources/lib/moment/moment.js',
            'resources/src/moment/moment-module.js',
        ],
        'languageScripts' => [
            'aeb-arab' => 'resources/lib/moment/locale/ar-tn.js',
            'af' => 'resources/lib/moment/locale/af.js',
            'ar' => 'resources/lib/moment/locale/ar.js',
            'ar-ma' => 'resources/lib/moment/locale/ar-ma.js',
            'ar-sa' => 'resources/lib/moment/locale/ar-sa.js',
            'az' => 'resources/lib/moment/locale/az.js',
            'be' => 'resources/lib/moment/locale/be.js',
            'bg' => 'resources/lib/moment/locale/bg.js',
            'bm' => 'resources/lib/moment/locale/bm.js',
            'bn' => 'resources/lib/moment/locale/bn.js',
            'bo' => 'resources/lib/moment/locale/bo.js',
            'br' => 'resources/lib/moment/locale/br.js',
            'bs' => 'resources/lib/moment/locale/bs.js',
            'ca' => 'resources/lib/moment/locale/ca.js',
            'cs' => 'resources/lib/moment/locale/cs.js',
            'cv' => 'resources/lib/moment/locale/cv.js',
            'cy' => 'resources/lib/moment/locale/cy.js',
            'da' => 'resources/lib/moment/locale/da.js',
            'de' => 'resources/lib/moment/locale/de.js',
            'de-at' => 'resources/lib/moment/locale/de-at.js',
            'de-ch' => 'resources/lib/moment/locale/de-ch.js',
            'dv' => 'resources/lib/moment/locale/dv.js',
            'el' => 'resources/lib/moment/locale/el.js',
            'en' => 'resources/src/moment/moment-dmy.js',
            'en-au' => 'resources/lib/moment/locale/en-au.js',
            'en-ca' => 'resources/lib/moment/locale/en-ca.js',
            'en-gb' => 'resources/lib/moment/locale/en-gb.js',
            'eo' => 'resources/lib/moment/locale/eo.js',
            'es' => 'resources/lib/moment/locale/es.js',
            'et' => 'resources/lib/moment/locale/et.js',
            'eu' => 'resources/lib/moment/locale/eu.js',
            'fa' => 'resources/lib/moment/locale/fa.js',
            'fi' => 'resources/lib/moment/locale/fi.js',
            'fo' => 'resources/lib/moment/locale/fo.js',
            'fr' => 'resources/lib/moment/locale/fr.js',
            'fr-ca' => 'resources/lib/moment/locale/fr-ca.js',
            'fy' => 'resources/lib/moment/locale/fy.js',
            'gd' => 'resources/lib/moment/locale/gd.js',
            'gl' => 'resources/lib/moment/locale/gl.js',
            'gom' => 'resources/lib/moment/locale/gom-latn.js',
            'gom-deva' => 'resources/lib/moment/locale/gom-deva.js',
            'gom-latn' => 'resources/lib/moment/locale/gom-latn.js',
            'gu' => 'resources/lib/moment/locale/gu.js',
            'he' => 'resources/lib/moment/locale/he.js',
            'hi' => 'resources/lib/moment/locale/hi.js',
            'hr' => 'resources/lib/moment/locale/hr.js',
            'hu' => 'resources/lib/moment/locale/hu.js',
            'hy-am' => 'resources/lib/moment/locale/hy-am.js',
            'id' => 'resources/lib/moment/locale/id.js',
            'is' => 'resources/lib/moment/locale/is.js',
            'it' => 'resources/lib/moment/locale/it.js',
            'ja' => 'resources/lib/moment/locale/ja.js',
            'jv' => 'resources/lib/moment/locale/jv.js',
            'ka' => 'resources/lib/moment/locale/ka.js',
            'kk-cyrl' => 'resources/lib/moment/locale/kk.js',
            'kn' => 'resources/lib/moment/locale/kn.js',
            'ko' => 'resources/lib/moment/locale/ko.js',
            'ky' => 'resources/lib/moment/locale/ky.js',
            'lo' => 'resources/lib/moment/locale/lo.js',
            'lt' => 'resources/lib/moment/locale/lt.js',
            'lv' => 'resources/lib/moment/locale/lv.js',
            'mi' => 'resources/lib/moment/locale/mi.js',
            'mk' => 'resources/lib/moment/locale/mk.js',
            'ml' => 'resources/lib/moment/locale/ml.js',
            'mr' => 'resources/lib/moment/locale/mr.js',
            'ms-my' => 'resources/lib/moment/locale/ms-my.js',
            'ms' => 'resources/lib/moment/locale/ms.js',
            'my' => 'resources/lib/moment/locale/my.js',
            'nb' => 'resources/lib/moment/locale/nb.js',
            'ne' => 'resources/lib/moment/locale/ne.js',
            'nl' => 'resources/lib/moment/locale/nl.js',
            'nn' => 'resources/lib/moment/locale/nn.js',
            'pa' => 'resources/lib/moment/locale/pa-in.js',
            'pl' => 'resources/lib/moment/locale/pl.js',
            'pt' => 'resources/lib/moment/locale/pt.js',
            'pt-br' => 'resources/lib/moment/locale/pt-br.js',
            'ro' => 'resources/lib/moment/locale/ro.js',
            'ru' => 'resources/lib/moment/locale/ru.js',
            'sd' => 'resources/lib/moment/locale/sd.js',
            'se' => 'resources/lib/moment/locale/se.js',
            'si' => 'resources/lib/moment/locale/si.js',
            'sk' => 'resources/lib/moment/locale/sk.js',
            'sl' => 'resources/lib/moment/locale/sl.js',
            'sq' => 'resources/lib/moment/locale/sq.js',
            'sr-ec' => 'resources/lib/moment/locale/sr-cyrl.js',
            'sr-el' => 'resources/lib/moment/locale/sr.js',
            'ss' => 'resources/lib/moment/locale/ss.js',
            'sv' => 'resources/lib/moment/locale/sv.js',
            'sw' => 'resources/lib/moment/locale/sw.js',
            'ta' => 'resources/lib/moment/locale/ta.js',
            'te' => 'resources/lib/moment/locale/te.js',
            'tet' => 'resources/lib/moment/locale/tet.js',
            'th' => 'resources/lib/moment/locale/th.js',
            'tl' => 'resources/lib/moment/locale/tl-ph.js',
            'tr' => 'resources/lib/moment/locale/tr.js',
            'tzm' => 'resources/lib/moment/locale/tzm.js',
            'tzm-latn' => 'resources/lib/moment/locale/tzm-latn.js',
            'uk' => 'resources/lib/moment/locale/uk.js',
            'ur' => 'resources/lib/moment/locale/ur.js',
            'uz' => 'resources/lib/moment/locale/uz-latn.js', # https://phabricator.wikimedia.org/T308123
            'uz-latn' => 'resources/lib/moment/locale/uz-latn.js',
            'vi' => 'resources/lib/moment/locale/vi.js',
            'yo' => 'resources/lib/moment/locale/yo.js',
            'zh-hans' => 'resources/lib/moment/locale/zh-cn.js',
            'zh-hant' => 'resources/lib/moment/locale/zh-tw.js',
            'zh-cn' => 'resources/lib/moment/locale/zh-cn.js',
            'zh-hk' => 'resources/lib/moment/locale/zh-hk.js',
            'zh-mo' => 'resources/lib/moment/locale/zh-mo.js',
            'zh-tw' => 'resources/lib/moment/locale/zh-tw.js',
        ],
        // HACK: skinScripts come after languageScripts, and we need locale overrides to come
        // after locale definitions
        'skinScripts' => [
            'default' => [
                'resources/src/moment/moment-locale-overrides.js',
            ],
        ],
        'dependencies' => [
            'mediawiki.language',
            'mediawiki.util',
        ],
    ],

    /* Vue */

    'vue' => [
        'packageFiles' => [
            'resources/src/vue/index.js',
            'resources/src/vue/errorLogger.js',
            'resources/src/vue/i18n.js',
            [
                'name' => 'resources/lib/vue/vue.js',
                'callback' => static function ( Context $context, Config $config ) {
                    // Use the development version if development mode is enabled, or if we're in debug mode
                    $file = $config->get( MainConfigNames::VueDevelopmentMode ) || $context->getDebug() ?
                        'resources/lib/vue/vue.global.js' :
                        'resources/lib/vue/vue.global.prod.js';
                    // The file shipped by Vue does var Vue = ...;, but doesn't export it
                    // Add module.exports = Vue; programmatically
                    return file_get_contents( MW_INSTALL_PATH . "/$file" ) .
                        ';module.exports=Vue;';
                },
                'versionCallback' => static function ( Context $context, Config $config ) {
                    $file = $config->get( MainConfigNames::VueDevelopmentMode ) || $context->getDebug() ?
                        'resources/lib/vue/vue.global.js' :
                        'resources/lib/vue/vue.global.prod.js';
                    return new FilePath( $file );
                }
            ],

        ],
        'dependencies' => [
            'mediawiki.page.ready'
        ]
    ],

    // Alias for 'vue', for backwards compatibility
    '@vue/composition-api' => [
        'packageFiles' => [
            'resources/src/vue/composition-api.js'
        ],
        'dependencies' => [
            'vue'
        ],
        'deprecated' => 'No longer needed, use the "vue" module directly'
    ],

    'vuex' => [
        'packageFiles' => [
            [
                'name' => 'resources/lib/vuex/vuex.js',
                'callback' => static function ( Context $context, Config $config ) {
                    // Use the development version if development mode is enabled, or if we're in debug mode
                    $file = $config->get( MainConfigNames::VueDevelopmentMode ) || $context->getDebug() ?
                        'resources/lib/vuex/vuex.global.js' :
                        'resources/lib/vuex/vuex.global.prod.js';
                    // The file shipped by Vuex does var Vuex = ...;, but doesn't export it
                    // Add module.exports = Vuex; programmatically, and import Vue
                    return "var Vue=require('vue');" .
                        file_get_contents( MW_INSTALL_PATH . "/$file" ) .
                        ';module.exports=Vuex;';
                },
                'versionCallback' => static function ( Context $context, Config $config ) {
                    $file = $config->get( MainConfigNames::VueDevelopmentMode ) || $context->getDebug() ?
                        'resources/lib/vuex/vuex.global.js' :
                        'resources/lib/vuex/vuex.global.prod.js';
                    return new FilePath( $file );
                }
            ],
        ],
        'dependencies' => [
            'vue',
        ],
        'deprecated' => '[1.42] Use Pinia instead. See migration guidelines: https://w.wiki/7pLU'
    ],

    'pinia' => [
        'packageFiles' => [
            [
                'name' => 'resources/lib/pinia/pinia.js',
                'callback' => static function ( Context $context, Config $config ) {
                    // Use the development version if development mode is enabled, or if we're in debug mode
                    $file = $config->get( MainConfigNames::VueDevelopmentMode ) || $context->getDebug() ?
                        'resources/lib/pinia/pinia.iife.js' :
                        'resources/lib/pinia/pinia.iife.prod.js';
                    // The file shipped by Pinia does var Pinia = ...;, but doesn't export it
                    // Add module.exports = Pinia; programmatically. We also import VueDemi
                    // by actually loading the vue instance.
                    return "var VueDemi=require('vue');" .
                        file_get_contents( MW_INSTALL_PATH . "/$file" ) .
                        ';module.exports=Pinia;';
                },
                'versionCallback' => static function ( Context $context, Config $config ) {
                    $file = $config->get( MainConfigNames::VueDevelopmentMode ) || $context->getDebug() ?
                        'resources/lib/pinia/pinia.iife.js' :
                        'resources/lib/pinia/pinia.iife.prod.js';
                    return new FilePath( $file );
                }
            ]
        ],
        'dependencies' => [
            'vue'
        ],
    ],

    '@wikimedia/codex' => [
        'class' => CodexModule::class,
        'codexFullLibrary' => true,
        'codexScriptOnly' => true,
        'dependencies' => [
            'vue',
            'codex-styles',
        ],
    ],

    'codex-styles' => [
        'class' => CodexModule::class,
        'codexFullLibrary' => true,
        'codexStyleOnly' => true,
        // Do not flip styles in RTL contexts, because we're already providing RTL-specific styles
        'noflip' => true,
    ],

    '@wikimedia/codex-search' => [
        'class' => CodexModule::class,
        'codexComponents' => [ 'CdxTypeaheadSearch' ],
        'codexScriptOnly' => true,
        'dependencies' => [
            'codex-search-styles',
        ],
    ],

    'codex-search-styles' => [
        'class' => CodexModule::class,
        'codexComponents' => [ 'CdxTypeaheadSearch' ],
        'codexStyleOnly' => true,
    ],

    /* MediaWiki */
    'mediawiki.template' => [
        'scripts' => 'resources/src/mediawiki.template.js',
    ],
    'mediawiki.template.mustache' => [
        'scripts' => [
            'resources/lib/mustache/mustache.js',
            'resources/src/mediawiki.template.mustache.js',
        ],
        'dependencies' => 'mediawiki.template',
    ],
    'mediawiki.apipretty' => [
        'styles' => [
            'resources/src/mediawiki.apipretty/apipretty.css',
            'resources/src/mediawiki.apipretty/apihelp.css',
        ],
    ],
    'mediawiki.api' => [
        'scripts' => [
            'resources/src/mediawiki.api/index.js',
            'resources/src/mediawiki.api/rest.js',
            'resources/src/mediawiki.api/category.js',
            'resources/src/mediawiki.api/edit.js',
            'resources/src/mediawiki.api/login.js',
            'resources/src/mediawiki.api/messages.js',
            'resources/src/mediawiki.api/options.js',
            'resources/src/mediawiki.api/parse.js',
            'resources/src/mediawiki.api/rollback.js',
            'resources/src/mediawiki.api/upload.js',
            'resources/src/mediawiki.api/user.js',
            'resources/src/mediawiki.api/watch.js',
        ],
        'dependencies' => [
            'mediawiki.Title',
            'mediawiki.util',
            'mediawiki.jqueryMsg',
            'user.options',
        ],
        'messages' => [
            'api-clientside-error-noconnect',
            'api-clientside-error-http',
            'api-clientside-error-timeout',
            'api-clientside-error-aborted',
            'api-clientside-error-invalidresponse',
        ],
    ],
    'mediawiki.content.json' => [
        'styles' => 'resources/src/mediawiki.content.json.less',
    ],
    'mediawiki.confirmCloseWindow' => [
        'scripts' => [
            'resources/src/mediawiki.confirmCloseWindow.js',
        ],
        'messages' => [
            'confirmleave-warning',
        ],
    ],
    'mediawiki.debug' => [
        'scripts' => [
            'resources/src/mediawiki.debug/jquery.footHovzer.js',
            'resources/src/mediawiki.debug/debug.js',
        ],
        'styles' => [
            'resources/src/mediawiki.debug/jquery.footHovzer.css',
            'resources/src/mediawiki.debug/debug.less',
        ],
        'dependencies' => [
            'oojs-ui-core',
        ],
    ],
    'mediawiki.diff' => [
        'packageFiles' => [
            'resources/src/mediawiki.diff/diff.js',
            'resources/src/mediawiki.diff/inlineFormatToggle.js',
        ],
        'styles' => [
            'resources/src/mediawiki.diff/styles.less'
        ],
        'dependencies' => [
            'mediawiki.api',
        ],
        'messages' => [
            'diff-inline-tooltip-ins',
            'diff-inline-tooltip-del',
            'diff-inline-format-label',
            'diff-inline-switch-desc'
        ]
    ],
    'mediawiki.diff.styles' => [
        'class' => LessVarFileModule::class,
        'lessMessages' => [
            'diff-line-deleted',
            'diff-newline'
        ],
        'styles' => [
            'resources/src/mediawiki.diff.styles/diff.less',
            'resources/src/mediawiki.diff.styles/print.css' => [
                'media' => 'print'
            ],
        ],
    ],
    'mediawiki.feedback' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.feedback',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.feedback",
        'packageFiles' => [
            'feedback.js',
            'FeedbackDialog.js',
        ],
        'styles' => 'feedback.less',
        'dependencies' => [
            'mediawiki.jqueryMsg',
            'mediawiki.messagePoster',
            'mediawiki.Title',
            'oojs-ui-core',
            'oojs-ui-windows',
        ],
        'messages' => [
            'feedback-dialog-intro',
            'feedback-external-bug-report-button',
            'feedback-cancel',
            'feedback-close',
            'feedback-dialog-title',
            'feedback-error1',
            'feedback-error4',
            'feedback-message',
            'feedback-subject',
            'feedback-submit',
            'feedback-terms',
            'feedback-termsofuse',
            'feedback-thanks',
            'feedback-thanks-title',
            'feedback-useragent'
        ],
    ],
    'mediawiki.feedlink' => [
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.feedlink/feedlink.css',
            ],
            'vector-2022' => [],
        ],
    ],
    'mediawiki.filewarning' => [
        'scripts' => 'resources/src/mediawiki.filewarning/filewarning.js',
        'styles' => 'resources/src/mediawiki.filewarning/filewarning.less',
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui.styles.icons-alerts',
        ],
    ],
    'mediawiki.ForeignApi' => [
        'class' => ForeignApiModule::class,
        // Additional dependencies generated dynamically
        'dependencies' => 'mediawiki.ForeignApi.core',
    ],
    'mediawiki.ForeignApi.core' => [
        'packageFiles' => [
            'resources/src/mediawiki.ForeignApi/index.js',
            'resources/src/mediawiki.ForeignApi/mediawiki.ForeignApi.core.js',
            'resources/src/mediawiki.ForeignApi/mediawiki.ForeignRest.core.js'
        ],
        'dependencies' => [
            'mediawiki.api',
            'oojs',
            'mediawiki.Uri',
        ],
    ],
    'mediawiki.helplink' => [
        'styles' => [
            'resources/src/mediawiki.helplink/helplink.less',
        ],
    ],
    'mediawiki.hlist' => [
        'class' => LessVarFileModule::class,
        'lessMessages' => [
            'colon-separator',
            'parentheses-start',
            'parentheses-end',
        ],
        'styles' => [
            'resources/src/mediawiki.hlist/hlist.less',
        ],
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.hlist/default.less',
        ],
    ],
    'mediawiki.htmlform' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.htmlform',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.htmlform",
        'packageFiles' => [
            'htmlform.js',
            'autocomplete.js',
            'autoinfuse.js',
            'cloner.js',
            'cond-state.js',
            'multiselect.js',
            'selectandother.js',
            'selectorother.js',
            'timezone.js',
            [
                'name' => 'contentMessages.json',
                'callback' => static function ( Context $context ) {
                    return [
                        'colonSeparator' => $context->msg( 'colon-separator' )->inContentLanguage()->text(),
                    ];
                }
            ],
        ],
        'dependencies' => [
            'mediawiki.util',
            'mediawiki.widgets.visibleLengthLimit',
        ],
        'messages' => [
            'htmlform-chosen-placeholder',
        ],
    ],
    'mediawiki.htmlform.ooui' => [
        'scripts' => [
            'resources/src/mediawiki.htmlform.ooui/Element.js',
        ],
        'dependencies' => [
            'oojs-ui-core',
        ],
    ],
    'mediawiki.htmlform.styles' => [
        'styles' => 'resources/src/mediawiki.htmlform.styles/styles.less',
    ],
    'mediawiki.htmlform.codex.styles' => [
        'styles' => [
            'resources/src/mediawiki.htmlform.codex.styles.less'
        ],
    ],
    'mediawiki.htmlform.ooui.styles' => [
        'styles' => [
            'resources/src/mediawiki.collapsiblefieldsetlayout.styles.less',
            'resources/src/mediawiki.htmlform.ooui.styles.less'
        ],
    ],
    'mediawiki.icon' => [
        'deprecated' => '[1.42] Use @cdx-icon-down-triangle and cdx-mixin-css-icon instead.',
        'styles' => 'resources/src/mediawiki.icon/icon.less',
    ],
    'mediawiki.inspect' => [
        'scripts' => 'resources/src/mediawiki.inspect.js',
        'dependencies' => [
            'mediawiki.String',
            'mediawiki.util',
        ],
    ],
    'mediawiki.notification' => [
        'styles' => [
            'resources/src/mediawiki.notification/common.css',
            'resources/src/mediawiki.notification/print.css'
                => [ 'media' => 'print' ],
        ],
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.notification/default.css',
        ],
        'scripts' => 'resources/src/mediawiki.notification/notification.js',
        'dependencies' => [
            'mediawiki.util',
            'mediawiki.visibleTimeout',
        ],
    ],
    'mediawiki.notification.convertmessagebox' => [
        'scripts' => 'resources/src/mediawiki.notification.convertmessagebox.js',
        'dependencies' => [
            'mediawiki.notification',
        ],
    ],
    'mediawiki.notification.convertmessagebox.styles' => [
        'styles' => [
            'resources/src/mediawiki.notification.convertmessagebox.styles.less',
        ],
    ],
    'mediawiki.String' => [
        'scripts' => 'resources/src/mediawiki.String.js',
    ],
    'mediawiki.pager.styles' => [
        'styles' => [
            'resources/src/mediawiki.pager.styles/TablePager.less',
            'resources/src/mediawiki.pager.styles/DataTable.less',
        ],
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.pager.styles/IndexPager.less',
            ]
        ],
    ],
    'mediawiki.pager.tablePager' => [
        'deprecated' => 'Deprecated in 1.38. Use `mediawiki.pager.styles`',
        'styles' => [
            'resources/src/mediawiki.pager.styles/TablePager.less',
            'resources/src/mediawiki.pager.styles/DataTable.less',
        ],
    ],
    'mediawiki.pulsatingdot' => [
        'styles' => [
            'resources/src/mediawiki.pulsatingdot/mediawiki.pulsatingdot.less',
        ],
    ],
    'mediawiki.searchSuggest' => [
        'scripts' => 'resources/src/mediawiki.searchSuggest/searchSuggest.js',
        'styles' => 'resources/src/mediawiki.searchSuggest/searchSuggest.css',
        'messages' => [
            'searchsuggest-search',
            'searchsuggest-containing',
        ],
        'dependencies' => [
            'jquery.suggestions',
            'mediawiki.api',
            'user.options',
        ],
    ],
    'mediawiki.storage' => [
        'packageFiles' => [
            'resources/src/mediawiki.storage/index.js',
            'resources/src/mediawiki.storage/SafeStorage.js',
        ],
        'dependencies' => [
            'mediawiki.util',
        ],
    ],
    'mediawiki.Title' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.Title',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.Title",
        'packageFiles' => [
            'Title.js',
            'phpCharToUpper.json'
        ],
        'dependencies' => [
            'mediawiki.String',
            'mediawiki.util',
        ],
    ],
    'mediawiki.Upload' => [
        'scripts' => 'resources/src/mediawiki.Upload.js',
        'dependencies' => [
            'mediawiki.api',
        ],
    ],
    'mediawiki.ForeignUpload' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src',
        'remoteBasePath' => "$wgResourceBasePath/resources/src",
        'packageFiles' => [
            'mediawiki.ForeignUpload.js',
            [
                'name' => 'config.json',
                'config' => [ MainConfigNames::ForeignUploadTargets, MainConfigNames::EnableUploads ],
            ],
        ],
        'dependencies' => [
            'mediawiki.ForeignApi',
            'mediawiki.Upload',
            'oojs',
        ],
        'messages' => [
            'uploaddisabledtext',
            'upload-dialog-disabled',
            'upload-foreign-cant-upload',
        ]
    ],
    'mediawiki.Upload.Dialog' => [
        'scripts' => [
            'resources/src/mediawiki.Upload.Dialog.js',
        ],
        'dependencies' => [
            'mediawiki.Upload.BookletLayout',
        ],
        'messages' => [
            'upload-dialog-title',
            'upload-dialog-button-cancel',
            'upload-dialog-button-back',
            'upload-dialog-button-done',
            'upload-dialog-button-save',
            'upload-dialog-button-upload',
        ],
    ],
    'mediawiki.Upload.BookletLayout' => [
        'scripts' => [
            'resources/src/mediawiki.Upload.BookletLayout/mw.widgets.StashedFileWidget.js',
            'resources/src/mediawiki.Upload.BookletLayout/BookletLayout.js',
        ],
        'styles' => [
            'resources/src/mediawiki.Upload.BookletLayout/BookletLayout.less',
        ],
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.Upload.BookletLayout/mw.widgets.StashedFileWidget.less',
            ],
        ],
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui-widgets',
            'oojs-ui-windows',
            'oojs-ui.styles.icons-content',
            'oojs-ui.styles.icons-editing-advanced',
            'moment',
            'mediawiki.Title',
            'mediawiki.api',
            'mediawiki.user',
            'mediawiki.Upload',
            'mediawiki.jqueryMsg',
        ],
        'messages' => [
            'upload-form-label-infoform-title',
            'upload-form-label-infoform-name',
            'upload-form-label-infoform-name-tooltip',
            'upload-form-label-infoform-description',
            'upload-form-label-infoform-description-tooltip',
            'upload-form-label-usage-title',
            'upload-form-label-usage-filename',
            'action-upload',
            'apierror-mustbeloggedin',
            'apierror-permissiondenied',
            'badaccess-groups',
            'api-error-unknown-warning',
            'fileexists',
            'filepageexists',
            'file-exists-duplicate',
            'file-deleted-duplicate',
            'filename-bad-prefix',
            'filename-thumb-name',
            'filewasdeleted',
            'badfilename',
            'protectedpagetext',
        ],
    ],
    'mediawiki.ForeignStructuredUpload.BookletLayout' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.ForeignStructuredUpload.BookletLayout',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.ForeignStructuredUpload.BookletLayout",
        'packageFiles' => [
            "index.js",
            'ForeignStructuredUpload.js',
            [ 'name' => 'config.json', 'config' => [ MainConfigNames::UploadDialog ] ],
            "BookletLayout.js",
        ],
        'dependencies' => [
            'mediawiki.ForeignUpload',
            'mediawiki.Upload.BookletLayout',
            'mediawiki.widgets.CategoryMultiselectWidget',
            'mediawiki.widgets.DateInputWidget',
            'mediawiki.jqueryMsg',
            'mediawiki.api',
            'moment',
            'mediawiki.libs.jpegmeta',
        ],
        'messages' => [
            'upload-foreign-cant-load-config',
            'upload-form-label-own-work',
            'upload-form-label-infoform-categories',
            'upload-form-label-infoform-date',
            'upload-form-label-own-work-message-generic-local',
            'upload-form-label-not-own-work-message-generic-local',
            'upload-form-label-not-own-work-local-generic-local',
            'upload-form-label-own-work-message-generic-foreign',
            'upload-form-label-not-own-work-message-generic-foreign',
            'upload-form-label-not-own-work-local-generic-foreign',
        ],
    ],
    'mediawiki.toc' => [
        'scripts' => 'resources/src/mediawiki.toc/toc.js',
        'styles' => [
            'resources/src/mediawiki.toc/toc.css'
                => [ 'media' => 'screen' ],
        ],
        'dependencies' => [
            'mediawiki.cookie',
        ],
    ],
    'mediawiki.Uri' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.Uri',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.Uri",
        'packageFiles' => [
            'Uri.js',
            [ 'name' => 'loose.regexp.js',
                'callback' => static function ( Context $context, Config $config ) {
                    return MwUrlModule::makeJsFromExtendedRegExp(
                        file_get_contents( MW_INSTALL_PATH . '/resources/src/mediawiki.Uri/loose.regexp' )
                    );
                },
                'versionCallback' => static function () {
                    return new FilePath( 'loose.regexp' );
                },
            ],
            [ 'name' => 'strict.regexp.js',
                'callback' => static function ( Context $context, Config $config ) {
                    return MwUrlModule::makeJsFromExtendedRegExp(
                        file_get_contents( MW_INSTALL_PATH . '/resources/src/mediawiki.Uri/strict.regexp' )
                    );
                },
                'versionCallback' => static function () {
                    return new FilePath( 'strict.regexp' );
                },
            ],
        ],
        'dependencies' => 'mediawiki.util',
    ],
    'mediawiki.user' => [
        'scripts' => 'resources/src/mediawiki.user.js',
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.cookie',
            // user.options is not directly used in mediawiki.user, but it
            // provides part of the mw.user API that consumers expect
            'user.options',
        ],
    ],
    'mediawiki.userSuggest' => [
        'scripts' => 'resources/src/mediawiki.userSuggest.js',
        'dependencies' => [
            'jquery.suggestions',
            'mediawiki.api'
        ],
    ],
    'mediawiki.util' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.util',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.util",
        'packageFiles' => [
            'util.js',
            'jquery.accessKeyLabel.js',
            [ 'name' => 'config.json', 'config' => [
                MainConfigNames::FragmentMode,
                MainConfigNames::GenerateThumbnailOnParse,
                MainConfigNames::LoadScript,
                MainConfigNames::AutoCreateTempUser,
            ] ],
            [ 'name' => 'portletLinkOptions.json', 'callback' => [ Skin::class, 'getPortletLinkOptions' ] ],
            [
                'name' => 'infinityValues.json',
                'callback' => static function () {
                    return ExpiryDef::INFINITY_VALS;
                }
            ]
        ],
        'dependencies' => [
            'jquery.client',
            'web2017-polyfills',
        ],
        'messages' => [ 'brackets', 'word-separator' ],
    ],
    'mediawiki.checkboxtoggle' => [
        'scripts' => 'resources/src/mediawiki.checkboxtoggle.js',
    ],
    'mediawiki.checkboxtoggle.styles' => [
        'styles' => 'resources/src/mediawiki.checkboxtoggle.styles.css',
    ],
    'mediawiki.cookie' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.cookie',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.cookie",
        'packageFiles' => [
            'index.js',
            'jar.js',
            'jquery.js',
            [ 'name' => 'config.json', 'config' => [
                'prefix' => MainConfigNames::CookiePrefix,
                'domain' => MainConfigNames::CookieDomain,
                'path' => MainConfigNames::CookiePath,
                'expires' => MainConfigNames::CookieExpiration,
            ] ],
        ],
    ],
    'mediawiki.experiments' => [
        'scripts' => 'resources/src/mediawiki.experiments.js',
    ],
    'mediawiki.editfont.styles' => [
        'styles' => 'resources/src/mediawiki.editfont.less',
    ],
    'mediawiki.visibleTimeout' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.visibleTimeout',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.visibleTimeout",
        'packageFiles' => [
            'visibleTimeout.js'
        ],
    ],

    /* MediaWiki Action */

    'mediawiki.action.edit' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.action.edit',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.action.edit",
        'packageFiles' => [
            'edit.js',
            'stash.js',
            'watchlistExpiry.js',
        ],
        'styles' => 'edit.css',
        'dependencies' => [
            'mediawiki.action.edit.styles',
            'mediawiki.editfont.styles',
            'jquery.textSelection',
            'oojs-ui-core',
            'mediawiki.widgets.visibleLengthLimit',
            'mediawiki.api',
            'mediawiki.util',
        ],
    ],
    'mediawiki.action.edit.styles' => [
        'styles' => [
            'resources/src/mediawiki.action/mediawiki.action.edit.styles.less',
            'resources/src/mediawiki.action/mediawiki.action.edit.checkboxes.less',
        ]
    ],
    'mediawiki.action.edit.collapsibleFooter' => [
        'scripts' => 'resources/src/mediawiki.action/mediawiki.action.edit.collapsibleFooter.js',
        'styles' => 'resources/src/mediawiki.action/mediawiki.action.edit.collapsibleFooter.css',
        'dependencies' => [
            'jquery.makeCollapsible',
            'mediawiki.storage',
            'mediawiki.icon',
        ],
    ],
    'mediawiki.action.edit.preview' => [
        'packageFiles' => [
            'resources/src/mediawiki.action/mediawiki.action.edit.preview.js',
            [
                'name' => 'resources/src/mediawiki.action/mediawiki.action.edit.preview.parsedMessages.json',
                'callback' => static function ( MessageLocalizer $messageLocalizer ) {
                    return [
                        'previewnote' => $messageLocalizer->msg( 'previewnote' )->parse(),
                    ];
                },
                // Use versionCallback to avoid calling the parser from version invalidation code.
                'versionCallback' => static function ( MessageLocalizer $messageLocalizer ) {
                    return [
                        'previewnote' => [
                            // Include the text of the message, in case the canonical translation changes
                            $messageLocalizer->msg( 'previewnote' )->plain(),
                            // Include the page touched time, in case the on-wiki override is invalidated
                            Title::makeTitle( NS_MEDIAWIKI, 'Previewnote' )->getTouched(),
                        ],
                    ];
                },
            ]
        ],
        'dependencies' => [
            'jquery.spinner',
            'mediawiki.api',
            'mediawiki.diff',
            'mediawiki.diff.styles',
            'mediawiki.user',
            'mediawiki.page.preview',
        ],
        'messages' => [
            'continue-editing',
            'currentrev',
            'otherlanguages',
            'preview',
            'previewerrortext',
            'yourtext',
        ],
    ],
    'mediawiki.action.history' => [
        'dependencies' => [ 'jquery.makeCollapsible' ],
        'scripts' => 'resources/src/mediawiki.action/mediawiki.action.history.js',
        'styles' => 'resources/src/mediawiki.action/mediawiki.action.history.css',
    ],
    'mediawiki.action.history.styles' => [
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.action/mediawiki.action.history.styles.less',
        ],
    ],
    'mediawiki.action.protect' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.action',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.action",
        'packageFiles' => [
            'mediawiki.action.protect.js',
            [ 'name' => 'config.json', 'config' => [ MainConfigNames::CascadingRestrictionLevels ] ],
        ],
        'dependencies' => [
            'oojs-ui-core',
            'mediawiki.widgets.visibleLengthLimit',
        ],
        'messages' => [
            'protect-unchain-permissions',
        ],
    ],
    'mediawiki.action.view.metadata' => [
        'styles' => 'resources/src/mediawiki.action/mediawiki.action.view.metadata.css',
        'scripts' => 'resources/src/mediawiki.action/mediawiki.action.view.metadata.js',
        'messages' => [
            'metadata-expand',
            'metadata-collapse',
        ],
        'dependencies' => 'mediawiki.action.view.filepage',
    ],

    'mediawiki.editRecovery.postEdit' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.editRecovery',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.editRecovery",
        'packageFiles' => [
            'postEdit.js',
            'storage.js',
            [
                'name' => 'config.json',
                'config' => [ 'EditRecoveryExpiry' ],
            ],
        ],
    ],
    'mediawiki.editRecovery.edit' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.editRecovery',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.editRecovery",
        'packageFiles' => [
            'edit.js',
            'storage.js',
            'LoadNotification.js',
            [
                'name' => 'config.json',
                'config' => [ 'EditRecoveryExpiry' ],
            ],
        ],
        'styles' => [
            'styles.less',
        ],
        'dependencies' => [
            'mediawiki.widgets.AbandonEditDialog',
            'mediawiki.notification',
            'oojs-ui.styles.icons-editing-core',
        ],
        'messages' => [
            'edit-recovery-loaded-title',
            'edit-recovery-loaded-message',
            'edit-recovery-loaded-message-different-rev',
            'edit-recovery-loaded-message-different-rev-publish',
            'edit-recovery-loaded-message-different-rev-save',
            'edit-recovery-loaded-show',
            'edit-recovery-loaded-discard',
            'word-separator',
        ],
    ],

    'mediawiki.action.view.postEdit' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.action',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.action",
        'packageFiles' => [
            'mediawiki.action.view.postEdit.js',
            [ 'name' => 'config.json', 'config' => [ MainConfigNames::EditSubmitButtonLabelPublish ] ],
        ],
        'dependencies' => [
            'mediawiki.tempUserCreated',
            'mediawiki.jqueryMsg',
            'mediawiki.notification',
            'mediawiki.storage',
            'oojs-ui-core',
            'oojs-ui.styles.icons-interactions',
        ],
        'messages' => [
            'postedit-confirmation-created',
            'postedit-confirmation-restored',
            'postedit-confirmation-saved',
            'postedit-confirmation-published',
            'postedit-temp-created-label',
            'postedit-temp-created',
        ],
    ],
    'mediawiki.action.view.redirect' => [
        'scripts' => 'resources/src/mediawiki.action/mediawiki.action.view.redirect.js',
    ],
    'mediawiki.action.view.redirectPage' => [
        'styles' => 'resources/src/mediawiki.action/mediawiki.action.view.redirectPage.css',
    ],
    'mediawiki.action.edit.editWarning' => [
        'scripts' => 'resources/src/mediawiki.action/mediawiki.action.edit.editWarning.js',
        'dependencies' => [
            'jquery.textSelection',
            'mediawiki.jqueryMsg',
            'mediawiki.confirmCloseWindow',
            'user.options',
        ],
    ],
    'mediawiki.action.view.filepage' => [
        'styles' => [
            'resources/src/mediawiki.action/mediawiki.action.view.filepage.print.less' =>
                [ 'media' => 'print' ],
            'resources/src/mediawiki.action/mediawiki.action.view.filepage.less',
        ],
    ],

    // This bundles small stylesheets (<2KB) that:
    // - .. are not loaded when viewing or editing content pages.
    // - .. style the rendering of other wikipage actions and/or other namespaces.
    'mediawiki.action.styles' => [
        'styles' => [
            'resources/src/mediawiki.action.styles/styles.less',
            'resources/src/mediawiki.action.styles/categoryPage.less',
        ],
    ],

    /* MediaWiki Language */

    'mediawiki.language' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.language',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.language",
        'scripts' => [
            'mediawiki.language.init.js',
            'mediawiki.language.js',
            'mediawiki.language.numbers.js',
            'mediawiki.language.fallback.js',
            [
                'name' => 'mediawiki.language.config.js',
                'callback' => static function ( Context $context, Config $config ) {
                    $langCode = $context->getLanguage();
                    $language = MediaWikiServices::getInstance()->getLanguageFactory()
                        ->getLanguage( $langCode );
                    return 'mw.language.setData('
                    . $context->encodeJson( $langCode ) . ','
                    . $context->encodeJson( $language->getJsData() )
                    . ');';
                }
            ],
        ],
        'languageScripts' => [
            'bs' => 'languages/bs.js',
            'dsb' => 'languages/dsb.js',
            'fi' => 'languages/fi.js',
            'ga' => 'languages/ga.js',
            'hsb' => 'languages/hsb.js',
            'hu' => 'languages/hu.js',
            'hy' => 'languages/hy.js',
            'la' => 'languages/la.js',
            'os' => 'languages/os.js',
            'sl' => 'languages/sl.js',
        ],
        'dependencies' => [
            'mediawiki.cldr',
        ],
        'messages' => [
            'and',
            'comma-separator',
            'word-separator'
        ],
    ],

    'mediawiki.cldr' => [
        'scripts' => 'resources/src/mediawiki.cldr/index.js',
        'dependencies' => [
            'mediawiki.libs.pluralruleparser',
        ],
    ],

    'mediawiki.libs.pluralruleparser' => [
        'scripts' => [
            'resources/lib/CLDRPluralRuleParser/CLDRPluralRuleParser.js',
            'resources/src/mediawiki.libs.pluralruleparser/export.js',
        ],
    ],

    'mediawiki.jqueryMsg' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.jqueryMsg',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.jqueryMsg",
        'packageFiles' => [
            'mediawiki.jqueryMsg.js',
            [ 'name' => 'parserDefaults.json', 'callback' => static function (
                Context $context, Config $config
            ) {
                $tagData = Sanitizer::getRecognizedTagData();
                $allowedHtmlElements = array_merge(
                    array_keys( $tagData['htmlpairs'] ),
                    array_diff(
                        array_keys( $tagData['htmlsingle'] ),
                        array_keys( $tagData['htmlsingleonly'] )
                    )
                );

                $magicWords = [
                    'SITENAME' => $config->get( MainConfigNames::Sitename ),
                ];
                ( new RL\HookRunner( MediaWikiServices::getInstance()->getHookContainer() ) )
                    ->onResourceLoaderJqueryMsgModuleMagicWords( $context, $magicWords );

                // if changing this structure, please update the static JSON example file
                // at resources/src/mediawiki.jqueryMsg
                return [
                    'allowedHtmlElements' => $allowedHtmlElements,
                    'magic' => $magicWords,
                ];
            } ],
        ],
        'dependencies' => [
            'mediawiki.util',
            'mediawiki.language',
            'mediawiki.String',
            'user.options',
        ],
    ],

    'mediawiki.language.months' => [
        'scripts' => 'resources/src/mediawiki.language.months/months.js',
        'dependencies' => 'mediawiki.language',
        'messages' => array_merge(
            Language::MONTH_MESSAGES,
            Language::MONTH_GENITIVE_MESSAGES,
            Language::MONTH_ABBREVIATED_MESSAGES
        )
    ],

    'mediawiki.language.names' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.language.names',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.language.names",
        'packageFiles' => [
            'names.js',
            [ 'name' => 'names.json', 'callback' => static function ( Context $context ) {
                return MediaWikiServices::getInstance()
                    ->getLanguageNameUtils()
                    ->getLanguageNames( $context->getLanguage(), LanguageNameUtils::ALL );
            } ],
        ],
        'dependencies' => 'mediawiki.language',
    ],

    'mediawiki.language.specialCharacters' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.language.specialCharacters',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.language.specialCharacters",
        'packageFiles' => [
            'specialCharacters.js',
            'specialcharacters.json'
        ],
        'dependencies' => 'mediawiki.language',
        'messages' => [
            'special-characters-group-latin',
            'special-characters-group-latinextended',
            'special-characters-group-ipa',
            'special-characters-group-symbols',
            'special-characters-group-greek',
            'special-characters-group-greekextended',
            'special-characters-group-cyrillic',
            'special-characters-group-arabic',
            'special-characters-group-arabicextended',
            'special-characters-group-persian',
            'special-characters-group-hebrew',
            'special-characters-group-bangla',
            'special-characters-group-tamil',
            'special-characters-group-telugu',
            'special-characters-group-sinhala',
            'special-characters-group-devanagari',
            'special-characters-group-gujarati',
            'special-characters-group-thai',
            'special-characters-group-lao',
            'special-characters-group-khmer',
            'special-characters-group-canadianaboriginal',
            'special-characters-group-runes',
            'special-characters-title-endash',
            'special-characters-title-emdash',
            'special-characters-title-minus'
        ]
    ],

    /* MediaWiki Libs */

    'mediawiki.libs.jpegmeta' => [
        'scripts' => [
            'resources/src/mediawiki.libs.jpegmeta/jpegmeta.js',
            'resources/src/mediawiki.libs.jpegmeta/export.js',
        ],
    ],

    /* MediaWiki Page */

    'mediawiki.page.gallery' => [
        'scripts' => 'resources/src/mediawiki.page.gallery.js',
        'dependencies' => [
            'mediawiki.page.gallery.styles',
            'mediawiki.util'
        ],
    ],
    'mediawiki.page.gallery.styles' => [
        'styles' => [
            'resources/src/mediawiki.page.gallery.styles/gallery.less',
            'resources/src/mediawiki.page.gallery.styles/print.less' => [ 'media' => 'print' ],
        ] + ( !$GLOBALS['wgParserEnableLegacyMediaDOM'] || $GLOBALS['wgUseContentMediaStyles'] ?
            [ 'resources/src/mediawiki.page.gallery.styles/content.media.less' => [ 'media' => 'all' ] ] : [] ),
    ],
    'mediawiki.page.gallery.slideshow' => [
        'scripts' => 'resources/src/mediawiki.page.gallery.slideshow.js',
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.Title',
            'oojs',
            'oojs-ui-core',
            'oojs-ui-widgets',
            'oojs-ui.styles.icons-media',
            'oojs-ui.styles.icons-movement'
        ],
        'messages' => [
            'gallery-slideshow-toggle'
        ]
    ],
    'mediawiki.page.ready' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.page.ready',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.page.ready",
        'packageFiles' => [
            'ready.js',
            'checkboxShift.js',
            'checkboxHack.js',
            'teleportTarget.js',
            'toggleAllCollapsibles.js',
            [ 'name' => 'config.json', 'callback' => static function (
                Context $context,
                Config $config
            ) {
                $readyConfig = [
                    'search' => true,
                    'collapsible' => true,
                    'sortable' => true,
                    'selectorLogoutLink' => '#pt-logout a[data-mw="interface"]'
                ];

                ( new HookRunner( MediaWikiServices::getInstance()->getHookContainer() ) )
                    ->onSkinPageReadyConfig( $context, $readyConfig );
                return $readyConfig;
            } ],
        ],
        'dependencies' => [
            'mediawiki.util',
            'mediawiki.api'
        ],
        'messages' => [
            'collapsible-collapse-all-text',
            'collapsible-collapse-all-tooltip',
            'collapsible-expand-all-text',
            'collapsible-expand-all-tooltip',
            'logging-out-notify'
        ],
        'skinStyles' => [
            'default' => 'teleportTarget.less'
        ]
    ],
    'mediawiki.page.watch.ajax' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.page.watch.ajax',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.page.watch.ajax",
        'packageFiles' => [
            'watch-ajax.js',
            [ 'name' => 'config.json', 'config' => [ MainConfigNames::WatchlistExpiry ] ],
        ],
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.user',
            'mediawiki.util',
            'mediawiki.Title',
            'mediawiki.jqueryMsg',
        ],
        'messages' => [
            'watch',
            'unwatch',
            'watching',
            'unwatching',
            'tooltip-ca-watch',
            'tooltip-ca-unwatch',
            'tooltip-ca-unwatch-expiring',
            'tooltip-ca-unwatch-expiring-hours',
            'addedwatchtext',
            'addedwatchtext-talk',
            'removedwatchtext',
            'removedwatchtext-talk',
        ],
    ],
    'mediawiki.page.preview' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src',
        'remoteBasePath' => "$wgResourceBasePath/resources/src",
        'packageFiles' => [
            'mediawiki.page.preview.js',
        ],
        'styles' => 'mediawiki.page.preview.css',
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.diff',
            'mediawiki.diff.styles',
            'mediawiki.jqueryMsg',
            'mediawiki.language',
            'mediawiki.util',
            'mediawiki.user',
            'jquery.makeCollapsible',
            'jquery.textSelection',
            'oojs-ui-core',
        ],
        'messages' => [
            'summary-preview',
            'parentheses',
            'word-separator',
            'comma-separator',
            'templatesusedpreview',
            'editlink',
            'viewsourcelink',
            'template-semiprotected',
            'template-protected',
            'restriction-level-sysop',
            'restriction-level-autoconfirmed',
            'diff-empty',
        ]
    ],
    'mediawiki.page.image.pagination' => [
        'scripts' => 'resources/src/mediawiki.page.image.pagination.js',
        'dependencies' => [
            'mediawiki.util',
            'jquery.spinner',
        ],
    ],
    'mediawiki.page.media' => [
        'scripts' => 'resources/src/mediawiki.page.media.js',
    ],

    /* MediaWiki Special pages */

    'mediawiki.rcfilters.filters.base.styles' => [
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.rcfilters/styles/mw.rcfilters.less',
        ],
    ],
    'mediawiki.rcfilters.highlightCircles.seenunseen.styles' => [
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.rcfilters/' .
                'styles/mw.rcfilters.ui.ChangesListWrapperWidget.highlightCircles.seenunseen.less',
            ],
        ],
    ],
    // TODO consider renaming to mediawiki.rcfilters.filters following merge of
    // mediawiki.rcfilters.filters.dm into mediawiki.rcfilters.filters.ui, see T256836
    'mediawiki.rcfilters.filters.ui' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.rcfilters',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.rcfilters",
        'packageFiles' => [
            'mw.rcfilters.js',
            'utils.js',
            'Controller.js',
            'UriProcessor.js',
            'dm/ChangesListViewModel.js',
            'dm/FilterGroup.js',
            'dm/FilterItem.js',
            'dm/FiltersViewModel.js',
            'dm/ItemModel.js',
            'dm/SavedQueriesModel.js',
            'dm/SavedQueryItemModel.js',
            // TODO consider merging this with the config.json for the ui code
            [ 'name' => 'dmConfig.json', 'config' =>
                [ MainConfigNames::StructuredChangeFiltersLiveUpdatePollingRate ] ],
            'HighlightColors.js',
            'ui/CheckboxInputWidget.js',
            'ui/FilterTagMultiselectWidget.js',
            'ui/ItemMenuOptionWidget.js',
            'ui/FilterMenuOptionWidget.js',
            'ui/FilterMenuSectionOptionWidget.js',
            'ui/TagItemWidget.js',
            'ui/FilterTagItemWidget.js',
            'ui/FilterMenuHeaderWidget.js',
            'ui/MenuSelectWidget.js',
            'ui/MainWrapperWidget.js',
            'ui/ViewSwitchWidget.js',
            'ui/ValuePickerWidget.js',
            'ui/ChangesLimitPopupWidget.js',
            'ui/ChangesLimitAndDateButtonWidget.js',
            'ui/DatePopupWidget.js',
            'ui/FilterWrapperWidget.js',
            'ui/ChangesListWrapperWidget.js',
            'ui/SavedLinksListWidget.js',
            'ui/SavedLinksListItemWidget.js',
            'ui/SaveFiltersPopupButtonWidget.js',
            'ui/FormWrapperWidget.js',
            'ui/FilterItemHighlightButton.js',
            'ui/HighlightPopupWidget.js',
            'ui/HighlightColorPickerWidget.js',
            'ui/LiveUpdateButtonWidget.js',
            'ui/MarkSeenButtonWidget.js',
            'ui/RcTopSectionWidget.js',
            'ui/RclTopSectionWidget.js',
            'ui/RclTargetPageWidget.js',
            'ui/RclToOrFromWidget.js',
            'ui/WatchlistTopSectionWidget.js',
            [ 'name' => 'config.json',
                'versionCallback' => [ ChangesListSpecialPage::class, 'getRcFiltersConfigSummary' ],
                'callback' => [ ChangesListSpecialPage::class, 'getRcFiltersConfigVars' ],
            ],
        ],
        'styles' => [
            'styles/mw.rcfilters.mixins.less',
            'styles/mw.rcfilters.variables.less',
            'styles/mw.rcfilters.ui.less',
            'styles/mw.rcfilters.ui.Overlay.less',
            'styles/mw.rcfilters.ui.FilterTagMultiselectWidget.less',
            'styles/mw.rcfilters.ui.ItemMenuOptionWidget.less',
            'styles/mw.rcfilters.ui.FilterMenuOptionWidget.less',
            'styles/mw.rcfilters.ui.FilterMenuSectionOptionWidget.less',
            'styles/mw.rcfilters.ui.TagItemWidget.less',
            'styles/mw.rcfilters.ui.FilterMenuHeaderWidget.less',
            'styles/mw.rcfilters.ui.MenuSelectWidget.less',
            'styles/mw.rcfilters.ui.ViewSwitchWidget.less',
            'styles/mw.rcfilters.ui.ValuePickerWidget.less',
            'styles/mw.rcfilters.ui.ChangesLimitPopupWidget.less',
            'styles/mw.rcfilters.ui.DatePopupWidget.less',
            'styles/mw.rcfilters.ui.FilterWrapperWidget.less',
            'styles/mw.rcfilters.ui.ChangesListWrapperWidget.less',
            'styles/mw.rcfilters.ui.HighlightColorPickerWidget.less',
            'styles/mw.rcfilters.ui.FilterItemHighlightButton.less',
            'styles/mw.rcfilters.ui.SavedLinksListWidget.less',
            'styles/mw.rcfilters.ui.SavedLinksListItemWidget.less',
            'styles/mw.rcfilters.ui.SaveFiltersPopupButtonWidget.less',
            'styles/mw.rcfilters.ui.LiveUpdateButtonWidget.less',
            'styles/mw.rcfilters.ui.RcTopSectionWidget.less',
            'styles/mw.rcfilters.ui.RclToOrFromWidget.less',
            'styles/mw.rcfilters.ui.RclTargetPageWidget.less',
            'styles/mw.rcfilters.ui.WatchlistTopSectionWidget.less',
            'styles/mw.rcfilters.ui.FilterTagMultiselectWidgetMobile.less'
        ],
        'skinStyles' => [
            'monobook' => [
                'styles/mw.rcfilters.ui.CapsuleItemWidget.monobook.less',
                'styles/mw.rcfilters.ui.FilterMenuOptionWidget.monobook.less',
            ],
        ],
        'messages' => [
            'rcfilters-tag-remove',
            'rcfilters-activefilters',
            'rcfilters-activefilters-hide',
            'rcfilters-activefilters-show',
            'rcfilters-activefilters-hide-tooltip',
            'rcfilters-activefilters-show-tooltip',
            'rcfilters-advancedfilters',
            'rcfilters-group-results-by-page',
            'rcfilters-limit-title',
            'rcfilters-limit-and-date-label',
            'rcfilters-limit-and-date-popup-dialog-aria-label',
            'rcfilters-date-popup-title',
            'rcfilters-days-title',
            'rcfilters-hours-title',
            'rcfilters-days-show-days',
            'rcfilters-days-show-hours',
            'rcfilters-highlighted-filters-list',
            'rcfilters-quickfilters',
            'rcfilters-quickfilters-placeholder-title',
            'rcfilters-quickfilters-placeholder-description',
            'rcfilters-savedqueries-defaultlabel',
            'rcfilters-savedqueries-rename',
            'rcfilters-savedqueries-setdefault',
            'rcfilters-savedqueries-unsetdefault',
            'rcfilters-savedqueries-remove',
            'rcfilters-savedqueries-new-name-label',
            'rcfilters-savedqueries-new-name-placeholder',
            'rcfilters-savedqueries-add-new-title',
            'rcfilters-savedqueries-already-saved',
            'rcfilters-savedqueries-apply-label',
            'rcfilters-savedqueries-cancel-label',
            'rcfilters-restore-default-filters',
            'rcfilters-clear-all-filters',
            'rcfilters-show-new-changes',
            'rcfilters-search-placeholder',
            'rcfilters-search-placeholder-mobile',
            'rcfilters-invalid-filter',
            'rcfilters-empty-filter',
            'rcfilters-filterlist-title',
            'rcfilters-filterlist-noresults',
            'rcfilters-filterlist-whatsthis',
            'rcfilters-highlightbutton-title',
            'rcfilters-highlightmenu-title',
            'rcfilters-highlightmenu-help',
            'rcfilters-noresults-conflict',
            'rcfilters-state-message-subset',
            'rcfilters-state-message-fullcoverage',
            'rcfilters-filter-excluded',
            'rcfilters-tag-prefix-namespace',
            'rcfilters-tag-prefix-namespace-inverted',
            'rcfilters-tag-prefix-tags',
            'rcfilters-tag-prefix-tags-inverted',
            'rcfilters-exclude-button-off',
            'rcfilters-exclude-button-on',
            'rcfilters-view-tags',
            'rcfilters-view-namespaces-tooltip',
            'rcfilters-view-tags-tooltip',
            'rcfilters-view-return-to-default-tooltip',
            'rcfilters-view-tags-help-icon-tooltip',
            'rcfilters-liveupdates-button',
            'rcfilters-liveupdates-button-title-on',
            'rcfilters-liveupdates-button-title-off',
            'rcfilters-watchlist-markseen-button',
            'rcfilters-watchlist-edit-watchlist-button',
            'rcfilters-other-review-tools',
            'rcfilters-filter-showlinkedfrom-label',
            'rcfilters-filter-showlinkedfrom-option-label',
            'rcfilters-filter-showlinkedto-label',
            'rcfilters-filter-showlinkedto-option-label',
            'rcfilters-target-page-placeholder',
            'rcfilters-allcontents-label',
            'rcfilters-alldiscussions-label',
            'blanknamespace',
            'namespaces',
            'tags-title',
            'invert',
            'recentchanges-noresult',
            'recentchanges-timeout',
            'recentchanges-network',
            'recentchanges-notargetpage',
            'allpagesbadtitle',
            'quotation-marks',
        ],
        'dependencies' => [
            'jquery.makeCollapsible',
            'mediawiki.String',
            'mediawiki.api',
            'mediawiki.jqueryMsg',
            'mediawiki.language',
            'mediawiki.Uri',
            'mediawiki.user',
            'mediawiki.util',
            'mediawiki.widgets',
            'oojs',
            'oojs-ui-widgets',
            'oojs-ui.styles.icons-content',
            'oojs-ui.styles.icons-moderation',
            'oojs-ui.styles.icons-editing-core',
            'oojs-ui.styles.icons-editing-styling',
            'oojs-ui.styles.icons-interactions',
            'oojs-ui.styles.icons-layout',
            'oojs-ui.styles.icons-media',
            'oojs-ui-windows.icons',
            'user.options',
        ],
    ],
    'mediawiki.interface.helpers.styles' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.interface.helpers.styles',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.interface.helpers.styles",
        'class' => LessVarFileModule::class,
        'lessMessages' => [
            'comma-separator',
            'parentheses-start',
            'parentheses-end',
            'semicolon-separator',
            'brackets-start',
            'brackets-end',
            'pipe-separator'
        ],
        'styles' => [
            'linker.styles.less',
        ],
        'skinStyles' => [
            'default' => 'skinStyles.less',
        ],
    ],
    'mediawiki.special' => [
        'styles' => [
            'resources/src/mediawiki.special/special.less',
            'resources/src/mediawiki.special/apisandbox.css',
            'resources/src/mediawiki.special/comparepages.less',
            'resources/src/mediawiki.special/contributions.less',
            'resources/src/mediawiki.special/edittags.css',
            'resources/src/mediawiki.special/movePage.css',
            'resources/src/mediawiki.special/newpages.less',
            'resources/src/mediawiki.special/pagesWithProp.css',
            'resources/src/mediawiki.special/upload.css',
            'resources/src/mediawiki.special/userrights.css',
            'resources/src/mediawiki.special/watchlist.css',
            'resources/src/mediawiki.special/whatlinkshere.less',
            'resources/src/mediawiki.special/block.less',
            'resources/src/mediawiki.special/listFiles.less',
            'resources/src/mediawiki.special/blocklist.less',
            'resources/src/mediawiki.special/version.css',
            'resources/src/mediawiki.special/contribute.less',
        ],
    ],
    'mediawiki.special.apisandbox' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.special.apisandbox',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.special.apisandbox",
        'styles' => 'apisandbox.less',
        'packageFiles' => [
            'apisandbox.js',
            'OptionalParamWidget.js',
            'ParamLabelWidget.js',
            'BooleanToggleSwitchParamWidget.js',
            'DateTimeParamWidget.js',
            'LimitParamWidget.js',
            'PasswordParamWidget.js',
            'UploadSelectFileParamWidget.js',
            'TextParamMixin.js',
            'UtilMixin.js',
        ],
        'dependencies' => [
            'mediawiki.Uri',
            'mediawiki.api',
            'mediawiki.jqueryMsg',
            'mediawiki.util',
            'oojs-ui',
            'oojs-ui.styles.icons-content',
            'oojs-ui.styles.icons-editing-advanced',
            'oojs-ui.styles.icons-interactions',
            'oojs-ui.styles.icons-moderation',
            'mediawiki.widgets',
            'mediawiki.widgets.datetime',
            'mediawiki.widgets.TitlesMultiselectWidget',
            'jquery.makeCollapsible',
        ],
        'messages' => [
            'apisandbox-intro',
            'apisandbox-submit',
            'apisandbox-reset',
            'apisandbox-retry',
            'apisandbox-loading',
            'apisandbox-load-error',
            'apisandbox-fetch-token',
            'apisandbox-add-multi',
            'apisandbox-helpurls',
            'apisandbox-examples',
            'apisandbox-dynamic-parameters',
            'apisandbox-dynamic-parameters-add-label',
            'apisandbox-dynamic-parameters-add-placeholder',
            'apisandbox-dynamic-error-exists',
            'apisandbox-templated-parameter-reason',
            'apisandbox-deprecated-parameters',
            'apisandbox-no-parameters',
            'paramvalidator-help-type-number-min',
            'paramvalidator-help-type-number-max',
            'paramvalidator-help-type-number-minmax',
            'api-help-param-deprecated',
            'api-help-param-deprecated-label',
            'api-help-param-internal',
            'api-help-param-internal-label',
            'api-help-param-multi-separate',
            'paramvalidator-help-multi-max',
            'paramvalidator-help-type-string-maxbytes',
            'paramvalidator-help-type-string-maxchars',
            'apisandbox-submit-invalid-fields-title',
            'apisandbox-submit-invalid-fields-message',
            'apisandbox-results',
            'apisandbox-sending-request',
            'apisandbox-loading-results',
            'apisandbox-results-error',
            'apisandbox-results-login-suppressed',
            'apisandbox-request-selectformat-label',
            'apisandbox-request-format-url-label',
            'apisandbox-request-url-label',
            'apisandbox-request-format-json-label',
            'apisandbox-request-json-label',
            'apisandbox-request-format-php-label',
            'apisandbox-request-php-label',
            'apisandbox-request-time',
            'apisandbox-request-post',
            'apisandbox-request-formdata',
            'apisandbox-results-fixtoken',
            'apisandbox-results-fixtoken-fail',
            'apisandbox-alert-page',
            'apisandbox-alert-field',
            'apisandbox-continue',
            'apisandbox-continue-clear',
            'apisandbox-continue-help',
            'apisandbox-param-limit',
            'apisandbox-multivalue-all-namespaces',
            'apisandbox-multivalue-all-values',
            'api-format-prettyprint-status',
            'blanknamespace',
            'comma-separator',
            'word-separator',
            'and'
        ],
    ],
    'mediawiki.special.block' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src',
        'remoteBasePath' => "$wgResourceBasePath/resources/src",
        'packageFiles' => [
            'mediawiki.special.block.js',
        ],
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui.styles.icons-editing-core',
            'oojs-ui.styles.icons-editing-advanced',
            'mediawiki.widgets.DateInputWidget',
            'mediawiki.widgets.SelectWithInputWidget',
            'mediawiki.widgets.NamespacesMultiselectWidget',
            'mediawiki.widgets.TitlesMultiselectWidget',
            'mediawiki.widgets.UserInputWidget',
            'mediawiki.util',
            'mediawiki.htmlform',
            'moment',
        ],
    ],
    // This bundles various small (under 5 KB?) JavaScript files that:
    // - .. are never loaded when viewing or editing wiki pages.
    // - .. are only used by logged-in users.
    // - .. depend on oojs-ui-core.
    // - .. contain UI initialisation code (e.g. no public module exports, because
    //      requiring or depending on this bundle is awkward)
    'mediawiki.misc-authed-ooui' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.misc-authed-ooui',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.misc-authed-ooui",
        'scripts' => [
            'action.delete.js',
            'special.changecredentials.js',
            'special.import.js',
            'special.movePage.js',
            'special.mute.js',
            'special.pageLanguage.js',
            'special.revisionDelete.js',
            'special.undelete.js',
            'special.undelete.loadMoreRevisions.js',
        ],
        'dependencies' => [
            'jquery.spinner',
            'mediawiki.util',
            'mediawiki.api',
            'mediawiki.htmlform.ooui',
            'mediawiki.widgets.visibleLengthLimit',
            'mediawiki.widgets',
            'oojs-ui-core',
        ],
    ],
    // This bundles various small (under 2 KB?) JavaScript files that:
    // - .. are only used by logged-in users when a non-default preference was enabled.
    // - .. may be loaded in the critical path for those users on page views.
    // - .. do NOT depend on OOUI.
    // - .. contain only UI initialisation code (e.g. no public exports)
    'mediawiki.misc-authed-pref' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.misc-authed-pref',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.misc-authed-pref",
        'scripts' => [
            'rightClickEdit.js',
            'dblClickEdit.js',
        ],
        'dependencies' => [
            'user.options',
        ],
    ],
    // This bundles various small scripts that relate to moderation or curation
    // in some way, and:
    // - .. are only loaded for a privileged subset of logged-in users.
    // - .. may be loaded in the critical path on page views.
    // - .. do NOT depend on OOUI or other "large" modules.
    // - .. contain only UI initialisation code (e.g. no public exports)
    'mediawiki.misc-authed-curate' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.misc-authed-curate',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.misc-authed-curate",
        'scripts' => [
            'patrol.js',
            'rollback.js',
            'edittags.js',
        ],
        'dependencies' => [
            'jquery.chosen',
            'jquery.lengthLimit',
            'jquery.spinner',
            'mediawiki.api',
            'mediawiki.util',
            'mediawiki.Title',
            'user.options',
            'jquery.confirmable',
        ],
        'messages' => [
            'markedaspatrollednotify',
            'rollback-confirmation-confirm',
            'rollback-confirmation-yes',
            'rollback-confirmation-no',
            'tags-edit-chosen-placeholder',
            'tags-edit-chosen-no-results',
        ],
    ],
    'mediawiki.special.block.codex' => [
        'packageFiles' => [
            'resources/src/mediawiki.special.block/init.js',
            'resources/src/mediawiki.special.block/components/BlockDetailsOptions.vue',
            'resources/src/mediawiki.special.block/components/BlockTypeField.vue',
            'resources/src/mediawiki.special.block/components/ExpirationField.vue',
            'resources/src/mediawiki.special.block/components/ReasonField.vue',
            'resources/src/mediawiki.special.block/components/TargetActiveBlocks.vue',
            'resources/src/mediawiki.special.block/components/TargetBlockLog.vue',
            'resources/src/mediawiki.special.block/components/UserLookup.vue',
            'resources/src/mediawiki.special.block/SpecialBlock.vue',
            [
                'name' => 'resources/src/mediawiki.special.block/icons.json',
                'callback' => 'MediaWiki\\ResourceLoader\\CodexModule::getIcons',
                'callbackParam' => [
                    'cdxIconSearch'
                ]
            ]
        ],
        'dependencies' => [
            'vue',
            '@wikimedia/codex',
        ],
        'messages' => [
            'block-details',
            'block-details-description',
            'block-user-label',
            'block-user-description',
            'block-user-placeholder',
            'block-save',
            'htmlform-optional-flag',
            'ipbcreateaccount',
            'ipbemailban',
            'ipb-disableusertalk'
        ],
    ],
    'mediawiki.special.changeslist' => [
        'styles' => [
            'resources/src/mediawiki.special.changeslist/changeslist.less'
        ],
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.special.changeslist/default.less',
        ],
    ],
    'mediawiki.special.changeslist.watchlistexpiry' => [
        'scripts' => 'resources/src/mediawiki.special.changeslist.watchlistexpiry/watchlistexpiry.js',
        'styles' => 'resources/src/mediawiki.special.changeslist.watchlistexpiry/watchlistexpiry.less',
        'messages' => [
            'parentheses',
            'watchlist-expiry-days-left',
            'watchlist-expiry-hours-left',
        ],
        'dependencies' => [
            'mediawiki.special',
            'oojs-ui.styles.icons-interactions'
        ],
    ],
    'mediawiki.special.changeslist.enhanced' => [
        'styles' => 'resources/src/mediawiki.special.changeslist.enhanced.less',
    ],
    'mediawiki.special.changeslist.legend' => [
        'styles' => 'resources/src/mediawiki.special.changeslist.legend.less',
    ],
    'mediawiki.special.changeslist.legend.js' => [
        'scripts' => 'resources/src/mediawiki.special.changeslist.legend.js',
        'dependencies' => [
            'mediawiki.cookie',
        ],
    ],
    'mediawiki.special.contributions' => [
        'scripts' => 'resources/src/mediawiki.special.contributions.js',
        'dependencies' => [
            'jquery.makeCollapsible',
            'oojs-ui',
            'mediawiki.widgets.DateInputWidget',
            'mediawiki.jqueryMsg',
        ],
    ],
    'mediawiki.special.import.styles.ooui' => [
        'styles' => 'resources/src/mediawiki.special.import.styles.ooui.less',
    ],
    'mediawiki.special.changecredentials' => [
        'scripts' => [ 'resources/src/mediawiki.special.changecredentails.js' ],
    ],
    'mediawiki.special.changeemail' => [
        'scripts' => [ 'resources/src/mediawiki.special.changeemail.js' ],
    ],

    'mediawiki.special.preferences.ooui' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.special.preferences.ooui',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.special.preferences.ooui",
        'packageFiles' => [
            'init.js',
            'confirmClose.js',
            'convertmessagebox.js',
            'editfont.js',
            'nav.js',
            'skinPrefs.js',
            'signature.js',
            'timezone.js',
            [
                'name' => 'layout.js',
                'callback' => static function ( Context $context ) {
                    $skinName = $context->getSkin();
                    ( new HookRunner( MediaWikiServices::getInstance()->getHookContainer() ) )
                        ->onPreferencesGetLayout( $useMobileLayout, $skinName );
                    $file = $useMobileLayout ? 'mobile.js' : 'tabs.js';
                    return new FilePath( $file );
                },
            ],
        ],
        'messages' => [
            'prefs-tabs-navigation-hint',
            'prefs-sections-navigation-hint',
            'prefs-signature-highlight-error',
            'prefs-back-title',
            'searchprefs',
            'searchprefs-noresults',
            'searchprefs-results',
        ],
        'dependencies' => [
            'mediawiki.language',
            'mediawiki.confirmCloseWindow',
            'mediawiki.notification.convertmessagebox',
            'mediawiki.storage',
            'oojs-ui-core',
            'oojs-ui-widgets',
            'oojs-ui-windows',
            'mediawiki.widgets.SelectWithInputWidget',
            'mediawiki.editfont.styles',
            'mediawiki.widgets.visibleLengthLimit',
        ],
    ],
    'mediawiki.special.preferences.styles.ooui' => [
        'styles' => 'resources/src/mediawiki.special.preferences.styles.ooui.less',
    ],
    'mediawiki.special.editrecovery.styles' => [
        'styles' => 'resources/src/mediawiki.special.editrecovery/styles.less',
    ],
    'mediawiki.special.editrecovery' => [
        'packageFiles' => [
            'resources/src/mediawiki.special.editrecovery/init.js',
            'resources/src/mediawiki.special.editrecovery/SpecialEditRecovery.vue',
            'resources/src/mediawiki.editRecovery/storage.js',
            [
                'name' => 'resources/src/mediawiki.editRecovery/config.json',
                'config' => [ 'EditRecoveryExpiry' ],
            ],
        ],
        'dependencies' => [
            'vue',
            'moment',
        ],
        'messages' => [
            'editlink',
            'parentheses-start',
            'parentheses-end',
            'pipe-separator',
            'edit-recovery-special-intro',
            'edit-recovery-special-intro-empty',
            'edit-recovery-special-view',
            'edit-recovery-special-edit',
            'edit-recovery-special-recovered-on',
            'edit-recovery-special-recovered-on-tooltip',
        ],
    ],
    'mediawiki.special.search' => [
        'scripts' => 'resources/src/mediawiki.special.search/search.js',
        'dependencies' => 'mediawiki.widgets.SearchInputWidget',
    ],
    'mediawiki.special.search.commonsInterwikiWidget' => [
        'scripts' => 'resources/src/mediawiki.special.search.commonsInterwikiWidget.js',
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.Title',
            'mediawiki.Uri'
        ],
        'messages' => [
            'search-interwiki-more-results',
            'searchprofile-images'
        ],
    ],
    'mediawiki.special.search.interwikiwidget.styles' => [
        'styles' => 'resources/src/mediawiki.special.search.interwikiwidget.styles.less',
    ],
    'mediawiki.special.search.styles' => [
        'styles' => 'resources/src/mediawiki.special.search.styles.less',
    ],
    'mediawiki.special.unwatchedPages' => [
        'scripts' => 'resources/src/mediawiki.special.unwatchedPages/unwatchedPages.js',
        'styles' => 'resources/src/mediawiki.special.unwatchedPages/unwatchedPages.css',
        'messages' => [
            'addedwatchtext-short',
            'removedwatchtext-short',
            'unwatch',
            'unwatching',
            'watch',
            'watching',
        ],
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.Title',
            'mediawiki.util',
        ],
    ],
    'mediawiki.special.upload' => [
        'templates' => [
            'thumbnail.html' => 'resources/src/mediawiki.special.upload/templates/thumbnail.html',
        ],
        'scripts' => 'resources/src/mediawiki.special.upload/upload.js',
        'messages' => [
            'widthheight',
            'size-bytes',
            'size-kilobytes',
            'size-megabytes',
            'size-gigabytes',
            'largefileserver',
        ],
        'dependencies' => [
            'mediawiki.special',
            'jquery.spinner',
            'mediawiki.jqueryMsg',
            'mediawiki.api',
            'mediawiki.libs.jpegmeta',
            'mediawiki.Title',
            'mediawiki.util',
            'mediawiki.confirmCloseWindow',
            'user.options',
        ],
    ],
    'mediawiki.special.userlogin.common.styles' => [
        'styles' => [
            'resources/src/mediawiki.special.userlogin.common.styles/userlogin.less',
        ],
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.special.userlogin.common.styles/skinStyles.less',
        ],
    ],
    'mediawiki.special.userlogin.login.styles' => [
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.special.userlogin.login.styles/login.less',
        ],
    ],
    'mediawiki.special.createaccount' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.special.createaccount',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.special.createaccount",
        'packageFiles' => [
            'signup.js',
            'HtmlformChecker.js'
        ],
        'messages' => [
            'createacct-emailrequired',
            'noname',
            'userexists',
            'createacct-normalization',
        ],
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.jqueryMsg',
            'mediawiki.util',
        ],
    ],
    'mediawiki.special.userlogin.signup.styles' => [
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.special.userlogin.signup.styles/signup.less',
        ],
    ],
    'mediawiki.special.userrights' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src',
        'remoteBasePath' => "$wgResourceBasePath/resources/src",
        'packageFiles' => [
            'mediawiki.special.userrights.js',
            [
                'name' => 'config.json',
                'config' => [
                    MainConfigNames::UserrightsInterwikiDelimiter
                ],
            ],
        ],
        'dependencies' => [
            'mediawiki.notification.convertmessagebox',
            'jquery.lengthLimit',
        ],
    ],
    'mediawiki.special.watchlist' => [
        'scripts' => [
            'resources/src/mediawiki.special.watchlist/watchlist.js',
            'resources/src/mediawiki.special.watchlist/visitedstatus.js',
            'resources/src/mediawiki.special.watchlist/editwatchlist.js'
        ],
        'messages' => [
            'addedwatchtext',
            'addedwatchtext-talk',
            'removedwatchtext',
            'removedwatchtext-talk',
            'tooltip-ca-watch',
            'tooltip-ca-unwatch',
            'tooltip-ca-unwatch-expiring',
            'tooltip-ca-unwatch-expiring-hours',
            'watchlist-unwatch',
            'watchlist-unwatch-undo',
            'watchlistedit-normal-check-all'
        ],
        'dependencies' => [
            'mediawiki.api',
            'mediawiki.jqueryMsg',
            'mediawiki.Title',
            'mediawiki.util',
            'oojs-ui-core',
            'oojs-ui.styles.icons-interactions',
            'user.options',
        ],
    ],
    'mediawiki.tempUserBanner.styles' => [
        'styles' => [
            'resources/src/mediawiki.tempUserBanner/tempUserBanner.less',
        ]
    ],
    'mediawiki.tempUserBanner' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.tempUserBanner',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.tempUserBanner",
        'packageFiles' => [
            'tempUserBanner.js',
            [ 'name' => 'config.json', 'config' => [ MainConfigNames::AutoCreateTempUser ] ],
        ],
        'dependencies' => [
            'mediawiki.jqueryMsg',
        ],
        'messages' => [
            'temp-user-banner-tooltip-title',
            'temp-user-banner-tooltip-description-expiration-soon',
            'temp-user-banner-tooltip-description-expiration-soon-day',
            'temp-user-banner-tooltip-description-learn-more',
            'temp-user-banner-tooltip-description-login'
        ]
    ],
    'mediawiki.tempUserCreated' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.tempUserCreated',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.tempUserCreated",
        'packageFiles' => [
            'mediawiki.tempUserCreated.js',
            [ 'name' => 'contLangMessages.json', 'callback' => static function ( MessageLocalizer $messageLocalizer ) {
                return [
                    'tempuser-helppage' => $messageLocalizer->msg( 'tempuser-helppage' )->inContentLanguage()->text(),
                ];
            } ],
        ],
        'dependencies' => [
            'mediawiki.util',
        ],
    ],
    /* MediaWiki Installer */

    // Used in the web installer. Test it after modifying this definition!

    /* MediaWiki Legacy */

    // Used in the web installer. Test it after modifying this definition!

    /* MediaWiki UI */

    'mediawiki.ui' => [
        'deprecated' => '[1.41] Please use Codex. See migration guidelines: ' .
            'https://www.mediawiki.org/wiki/Codex/Migrating_from_MediaWiki_UI',
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.ui/default.less',
            ],
        ],
    ],
    'mediawiki.ui.checkbox' => [
        'deprecated' => '[1.41] Please use Codex. See migration guidelines: ' .
            'https://www.mediawiki.org/wiki/Codex/Migrating_from_MediaWiki_UI',
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.ui.checkbox/checkbox.less',
            ],
        ],
    ],
    'mediawiki.ui.radio' => [
        'deprecated' => '[1.41] Please use Codex. See migration guidelines: ' .
            'https://www.mediawiki.org/wiki/Codex/Migrating_from_MediaWiki_UI',
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.ui.radio/radio.less',
            ],
        ],
    ],
    // Lightweight module for button styles
    'mediawiki.ui.button' => [
        'deprecated' => '[1.41] Please use Codex. See migration guidelines: ' .
            'https://www.mediawiki.org/wiki/Codex/Migrating_from_MediaWiki_UI',
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.ui.button/button.less',
            ],
        ],
    ],
    'mediawiki.ui.input' => [
        'deprecated' => '[1.41] Please use Codex. See migration guidelines: ' .
            'https://www.mediawiki.org/wiki/Codex/Migrating_from_MediaWiki_UI',
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.ui.input/input.less',
            ],
        ],
    ],
    'mediawiki.ui.icon' => [
        'deprecated' => '[1.41] Please use Codex. See migration guidelines: ' .
            'https://www.mediawiki.org/wiki/Codex/Migrating_from_MediaWiki_UI',
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.ui.icon/icons-2.less',
        ],
    ],

    'mediawiki.widgets' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.widgets',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.widgets",
        'packageFiles' => [
            'index.js',
            [ 'name' => 'data.json', 'callback' => static function ( Context $context ) {
                // $context only has a language code, we need to look up the language object
                $langCode = $context->getLanguage();
                $userLang = MediaWikiServices::getInstance()->getLanguageFactory()
                    ->getLanguage( $langCode );
                return [
                    'formattedNamespaces' => $userLang->getFormattedNamespaces(),
                ];
            } ],
            'mw.widgets.NamespaceInputWidget.js',
            'mw.widgets.ComplexNamespaceInputWidget.js',
            'mw.widgets.CopyTextLayout.js',
            'mw.widgets.TitleWidget.js',
            'mw.widgets.TitleInputWidget.js',
            'mw.widgets.TitleSearchWidget.js',
            'mw.widgets.ComplexTitleInputWidget.js',
            'mw.widgets.TitleOptionWidget.js',
        ],
        'styles' => [],
        'skinStyles' => [
            'default' => [
                'mw.widgets.TitleWidget.less',
            ],
        ],
        'dependencies' => [
            'oojs-ui-widgets',
            'mediawiki.widgets.styles',
            // TitleInputWidget
            'oojs-ui.styles.icons-content',
            // CopyTextLayout uses 'copy'
            'oojs-ui.styles.icons-editing-advanced',
            'mediawiki.Title',
            'mediawiki.api',
            'mediawiki.String',
            'mediawiki.language',
        ],
        'messages' => [
            // NamespaceInputWidget
            'blanknamespace',
            'namespacesall',
            // CopyTextLayout
            'mw-widgets-copytextlayout-copy-fail',
            'mw-widgets-copytextlayout-copy-success',
            // TitleInputWidget
            'mw-widgets-titleinput-description-new-page',
            'mw-widgets-titleinput-description-redirect',
        ],
    ],
    'mediawiki.widgets.styles' => [
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.widgets/mw.widgets.ComplexNamespaceInputWidget.base.css',
                'resources/src/mediawiki.widgets/mw.widgets.ComplexTitleInputWidget.base.css',
            ],
        ],
    ],
    'mediawiki.widgets.AbandonEditDialog' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.AbandonEditDialog.js',
        ],
        'messages' => [
            'mw-widgets-abandonedit',
            'mw-widgets-abandonedit-discard',
            'mw-widgets-abandonedit-keep',
            'mw-widgets-abandonedit-title',
        ],
        'dependencies' => [
            'oojs-ui-windows',
        ],
    ],
    'mediawiki.widgets.DateInputWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.CalendarWidget.js',
            'resources/src/mediawiki.widgets/mw.widgets.DateInputWidget.js',
        ],
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.widgets/mw.widgets.CalendarWidget.less',
                'resources/src/mediawiki.widgets/mw.widgets.DateInputWidget.less',
            ],
        ],
        'messages' => [
            'mw-widgets-dateinput-no-date',
            'mw-widgets-dateinput-placeholder-day',
            'mw-widgets-dateinput-placeholder-month',
        ],
        'dependencies' => [
            'oojs-ui-widgets',
            'oojs-ui.styles.icons-movement',
            'moment',
            'mediawiki.widgets.DateInputWidget.styles',
        ],
    ],
    'mediawiki.widgets.DateInputWidget.styles' => [
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.widgets/mw.widgets.DateInputWidget.styles.less',
            ],
        ],
    ],
    'mediawiki.widgets.visibleLengthLimit' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.widgets.visibleLengthLimit',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.widgets.visibleLengthLimit",
        'packageFiles' => [
            'mediawiki.widgets.visibleLengthLimit.js',
            [
                'name' => 'contentMessages.json',
                'type' => 'data',
                'callback' => static function ( Context $context ) {
                    return [
                        'colonSeparator' => $context->msg( 'colon-separator' )->inContentLanguage()->text(),
                    ];
                }
            ],
        ],
        'dependencies' => [
            'oojs-ui-core',
            'jquery.lengthLimit',
            'mediawiki.language',
            'mediawiki.String',
        ],
    ],
    'mediawiki.widgets.datetime' => [
        'scripts' => [
            'resources/src/mediawiki.widgets.datetime/mediawiki.widgets.datetime.js',
            'resources/src/mediawiki.widgets.datetime/CalendarWidget.js',
            'resources/src/mediawiki.widgets.datetime/DateTimeFormatter.js',
            'resources/src/mediawiki.widgets.datetime/DateTimeInputWidget.js',
            'resources/src/mediawiki.widgets.datetime/ProlepticGregorianDateTimeFormatter.js',
        ],
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.widgets.datetime/CalendarWidget.less',
                'resources/src/mediawiki.widgets.datetime/DateTimeInputWidget.less',
            ],
        ],
        'messages' => [
            'timezone-utc',
            'timezone-local',
            'january',
            'february',
            'march',
            'april',
            'may_long',
            'june',
            'july',
            'august',
            'september',
            'october',
            'november',
            'december',
            'jan',
            'feb',
            'mar',
            'apr',
            'may',
            'jun',
            'jul',
            'aug',
            'sep',
            'oct',
            'nov',
            'dec',
            'sunday',
            'monday',
            'tuesday',
            'wednesday',
            'thursday',
            'friday',
            'saturday',
            'sun',
            'mon',
            'tue',
            'wed',
            'thu',
            'fri',
            'sat',
            'period-am',
            'period-pm',
        ],
        'dependencies' => [
            'mediawiki.util',
            'oojs-ui-core',
            'oojs-ui.styles.icons-moderation',
            'oojs-ui.styles.icons-movement',
            'oojs-ui.styles.icons-interactions',
        ],
    ],
    'mediawiki.widgets.expiry' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.ExpiryInputWidget.js',
        ],
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui-widgets',
            'moment',
            'mediawiki.widgets.datetime'
        ],
        'skinStyles' => [
            'default' => 'resources/src/mediawiki.widgets/mw.widgets.ExpiryInputWidget.less',
        ],
    ],
    'mediawiki.widgets.CheckMatrixWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.CheckMatrixWidget.js',
        ],
        'dependencies' => [
            'oojs-ui-core',
        ],
    ],
    'mediawiki.widgets.CategoryMultiselectWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.CategoryTagItemWidget.js',
            'resources/src/mediawiki.widgets/mw.widgets.CategoryMultiselectWidget.js',
        ],
        'dependencies' => [
            'oojs-ui-widgets',
            'mediawiki.api',
            'mediawiki.ForeignApi',
            'mediawiki.Title',
        ],
        'messages' => [
            'red-link-title',
            'mw-widgets-categoryselector-add-category-placeholder',
        ],
    ],
    'mediawiki.widgets.SelectWithInputWidget' => [
        'scripts' => 'resources/src/mediawiki.widgets/mw.widgets.SelectWithInputWidget.js',
        'dependencies' => [
            'mediawiki.widgets.SelectWithInputWidget.styles',
            'oojs-ui-widgets',
        ],
    ],
    'mediawiki.widgets.SelectWithInputWidget.styles' => [
        'styles' => 'resources/src/mediawiki.widgets/mw.widgets.SelectWithInputWidget.base.css',
    ],
    'mediawiki.widgets.SizeFilterWidget' => [
        'scripts' => 'resources/src/mediawiki.widgets/mw.widgets.SizeFilterWidget.js',
        'dependencies' => [
            'mediawiki.widgets.SizeFilterWidget.styles',
            'oojs-ui-widgets',
        ],
        'messages' => [
            'minimum-size',
            'maximum-size',
            'pagesize',
        ],
    ],
    'mediawiki.widgets.SizeFilterWidget.styles' => [
        'styles' => 'resources/src/mediawiki.widgets/mw.widgets.SizeFilterWidget.base.css',
    ],
    'mediawiki.widgets.MediaSearch' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.APIResultsProvider.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.APIResultsQueue.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaResourceProvider.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaSearchProvider.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaUserUploadsProvider.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaResourceQueue.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaSearchQueue.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaUserUploadsQueue.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaSearchWidget.js',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaResultWidget.js',
        ],
        'styles' => [
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaSearchWidget.css',
            'resources/src/mediawiki.widgets/MediaSearch/mw.widgets.MediaResultWidget.less',
        ],
        'dependencies' => [
            'oojs-ui-widgets',
            'mediawiki.ForeignApi',
            'mediawiki.Title',
            'mediawiki.user',
            'mediawiki.util',
        ],
        'messages' => [
            'mw-widgets-mediasearch-noresults',
            'mw-widgets-mediasearch-input-placeholder',
            'mw-widgets-mediasearch-results-aria-label',
            'mw-widgets-mediasearch-recent-uploads',
        ],
    ],
    'mediawiki.widgets.Table' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/Table/mw.widgets.RowWidget.js',
            'resources/src/mediawiki.widgets/Table/mw.widgets.RowWidgetModel.js',
            'resources/src/mediawiki.widgets/Table/mw.widgets.TableWidget.js',
            'resources/src/mediawiki.widgets/Table/mw.widgets.TableWidgetModel.js'
        ],
        'styles' => [
            'resources/src/mediawiki.widgets/Table/mw.widgets.RowWidget.css',
            'resources/src/mediawiki.widgets/Table/mw.widgets.TableWidget.css',
        ],
        'dependencies' => [
            'oojs-ui-widgets'
        ],
        'messages' => [
            'mw-widgets-table-row-delete',
        ],
    ],
    'mediawiki.widgets.TagMultiselectWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.TagMultiselectWidget.js',
        ],
        'dependencies' => [
            'oojs-ui-widgets',
        ],
    ],
    'mediawiki.widgets.UserInputWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.UserInputWidget.js',
        ],
        'dependencies' => [
            'mediawiki.api',
            'oojs-ui-widgets',
        ],
    ],
    'mediawiki.widgets.UsersMultiselectWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.UsersMultiselectWidget.js',
        ],
        'dependencies' => [
            'mediawiki.api',
            'oojs-ui-widgets',
        ],
    ],
    'mediawiki.widgets.NamespacesMultiselectWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.NamespacesMultiselectWidget.js',
            'resources/src/mediawiki.widgets/mw.widgets.NamespacesMenuOptionWidget.js',
        ],
        'dependencies' => [
            'oojs-ui-widgets',
            // FIXME: Only needs NamespaceInputWidget
            'mediawiki.widgets',
        ],
    ],
    'mediawiki.widgets.TitlesMultiselectWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.TitlesMultiselectWidget.js',
        ],
        'dependencies' => [
            'mediawiki.api',
            'oojs-ui-widgets',
            // FIXME: Needs TitleInputWidget only
            'mediawiki.widgets',
        ],
    ],
    'mediawiki.widgets.TagMultiselectWidget.styles' => [
        'styles' => 'resources/src/mediawiki.widgets/mw.widgets.TagMultiselectWidget.base.css',
    ],
    'mediawiki.widgets.SearchInputWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.SearchInputWidget.js',
        ],
        'dependencies' => [
            'mediawiki.searchSuggest',
            'oojs-ui.styles.icons-interactions',
            // FIXME: Needs TitleInputWidget only
            'mediawiki.widgets',
        ],
    ],
    'mediawiki.widgets.SearchInputWidget.styles' => [
        'skinStyles' => [
            'default' => [
                'resources/src/mediawiki.widgets/mw.widgets.SearchInputWidget.css',
            ],
        ],
    ],
    'mediawiki.widgets.ToggleSwitchWidget' => [
        'scripts' => [
            'resources/src/mediawiki.widgets/mw.widgets.ToggleSwitchWidget.js',
        ],
        'dependencies' => [
            'oojs-ui-widgets',
        ],
    ],
    'mediawiki.watchstar.widgets' => [
        'localBasePath' => MW_INSTALL_PATH . '/resources/src/mediawiki.watchstar.widgets',
        'remoteBasePath' => "$wgResourceBasePath/resources/src/mediawiki.watchstar.widgets",
        'packageFiles' => [
            'WatchlistExpiryWidget.js',
            [ 'name' => 'data.json', 'callback' => static function ( MessageLocalizer $messageLocalizer ) {
                return WatchAction::getExpiryOptions( $messageLocalizer, false );
            } ]
        ],
        'styles' => 'WatchlistExpiryWidget.css',
        'dependencies' => [
            'oojs-ui'
        ],
        'messages' => [
            'accesskey-ca-watch',
            'addedwatchexpiry-options-label',
            'addedwatchexpirytext',
            'addedwatchexpirytext-talk',
            'addedwatchindefinitelytext',
            'addedwatchindefinitelytext-talk'
        ],
    ],

    'mediawiki.deflate' => [
        'packageFiles' => [
            'resources/src/mediawiki.deflate/mw.deflate.js',
            [
                'name' => 'resources/lib/pako/pako_deflate.js',
                'callback' => static function ( Context $context, Config $config ) {
                    return new FilePath( $context->getDebug() ?
                        'resources/lib/pako/pako_deflate.js' :
                        'resources/lib/pako/pako_deflate.min.js' );
                }
            ],
        ],
    ],

    /* OOjs */
    'oojs' => [
        'scripts' => [
            'resources/lib/oojs/oojs.js',
            'resources/src/oojs-global.js',
        ],
    ],

    'mediawiki.router' => [
        'scripts' => [
            'resources/lib/oojs-router/oojs-router.js',
            'resources/src/mediawiki.router/index.js',
        ],
        'dependencies' => [
            'oojs',
        ],
    ],

    /* OOjs UI */

    // Omnibus module.
    'oojs-ui' => [
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui-widgets',
            'oojs-ui-toolbars',
            'oojs-ui-windows',
        ],
    ],

    // The core JavaScript library.
    'oojs-ui-core' => [
        'class' => OOUIFileModule::class,
        'scripts' => [
            'resources/lib/ooui/oojs-ui-core.js',
            'resources/src/ooui-local.js',
        ],
        'themeScripts' => 'core',
        'dependencies' => [
            'oojs',
            'oojs-ui-core.styles',
            'oojs-ui-core.icons',
            'oojs-ui.styles.indicators',
            'mediawiki.language',
            'mediawiki.page.ready',
        ],
        'messages' => [
            'ooui-field-help',
            'ooui-combobox-button-label',
            'ooui-popup-widget-close-button-aria-label',
            'ooui-selectfile-button-select',
            'ooui-selectfile-button-select-multiple',
            'ooui-selectfile-dragdrop-placeholder',
            'ooui-selectfile-dragdrop-placeholder-multiple',
            'ooui-selectfile-placeholder',
        ],
    ],
    // This contains only the styles required by core widgets.
    'oojs-ui-core.styles' => [
        'class' => OOUIFileModule::class,
        'styles' => [
            'resources/lib/ooui/wikimedia-ui-base.less', // Providing Wikimedia UI LESS variables to all
        ],
        'themeStyles' => 'core',
    ],
    'oojs-ui-core.icons' => [
        'class' => OOUIIconPackModule::class,
        'icons' => [
            'add', 'alert', 'infoFilled', 'error', 'check', 'close', 'info', 'search', 'subtract', 'success'
        ],
    ],
    // Additional widgets and layouts module.
    'oojs-ui-widgets' => [
        'class' => OOUIFileModule::class,
        'scripts' => 'resources/lib/ooui/oojs-ui-widgets.js',
        'themeStyles' => 'widgets',
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui-widgets.icons',
        ],
        'messages' => [
            'ooui-copytextlayout-copy',
            'ooui-item-remove',
            'ooui-outline-control-move-down',
            'ooui-outline-control-move-up',
            'ooui-outline-control-remove',
        ],
    ],
    // You should never directly load this module. The CSS classes it defines are not a public API,
    // they depend on the internal structure of OOUI widgets, which can change at any time. If you
    // find that you need to load this module, you're probably doing something wrong or very hacky.
    'oojs-ui-widgets.styles' => [
        'class' => OOUIFileModule::class,
        'themeStyles' => 'widgets',
    ],
    'oojs-ui-widgets.icons' => [
        'class' => OOUIIconPackModule::class,
        // Do not repeat icons already used in 'oojs-ui-core.icons'
        'icons' => [ 'attachment', 'collapse', 'expand', 'trash', 'upload' ],
    ],
    // Toolbar and tools module.
    'oojs-ui-toolbars' => [
        'class' => OOUIFileModule::class,
        'scripts' => 'resources/lib/ooui/oojs-ui-toolbars.js',
        'themeStyles' => 'toolbars',
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui-toolbars.icons',
        ],
        'messages' => [
            'ooui-toolbar-more',
            'ooui-toolgroup-collapse',
            'ooui-toolgroup-expand',
        ],
    ],
    'oojs-ui-toolbars.icons' => [
        'class' => OOUIIconPackModule::class,
        // Do not repeat icons already used in 'oojs-ui-core.icons': 'check'
        'icons' => [ 'collapse', 'expand' ],
    ],
    // Windows and dialogs module.
    'oojs-ui-windows' => [
        'class' => OOUIFileModule::class,
        'scripts' => 'resources/lib/ooui/oojs-ui-windows.js',
        'themeStyles' => 'windows',
        'dependencies' => [
            'oojs-ui-core',
            'oojs-ui-windows.icons',
        ],
        'messages' => [
            'ooui-dialog-message-accept',
            'ooui-dialog-message-reject',
            'ooui-dialog-process-continue',
            'ooui-dialog-process-dismiss',
            'ooui-dialog-process-error',
            'ooui-dialog-process-retry',
        ],
    ],
    'oojs-ui-windows.icons' => [
        'class' => OOUIIconPackModule::class,
        // Do not repeat icons already used in 'oojs-ui-core.icons': 'close'
        'icons' => [ 'previous' ],
    ],

    'oojs-ui.styles.indicators' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'indicators',
    ],
    'oojs-ui.styles.icons-accessibility' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-accessibility',
    ],
    'oojs-ui.styles.icons-alerts' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-alerts',
    ],
    'oojs-ui.styles.icons-content' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-content',
    ],
    'oojs-ui.styles.icons-editing-advanced' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-editing-advanced',
    ],
    'oojs-ui.styles.icons-editing-citation' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-editing-citation',
    ],
    'oojs-ui.styles.icons-editing-core' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-editing-core',
    ],
    'oojs-ui.styles.icons-editing-list' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-editing-list',
    ],
    'oojs-ui.styles.icons-editing-styling' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-editing-styling',
    ],
    'oojs-ui.styles.icons-interactions' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-interactions',
    ],
    'oojs-ui.styles.icons-layout' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-layout',
    ],
    'oojs-ui.styles.icons-location' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-location',
    ],
    'oojs-ui.styles.icons-media' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-media',
    ],
    'oojs-ui.styles.icons-moderation' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-moderation',
    ],
    'oojs-ui.styles.icons-movement' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-movement',
    ],
    'oojs-ui.styles.icons-user' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-user',
    ],
    'oojs-ui.styles.icons-wikimedia' => [
        'class' => OOUIImageModule::class,
        'themeImages' => 'icons-wikimedia',
    ],
];