summaryrefslogtreecommitdiff
path: root/www/wiki/includes/resourceloader/ResourceLoaderOOUIImageModule.php
blob: 5c9e1d94ae0a5867a76113a1ee50dcc0a8a00ae5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
<?php
/**
 * 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
 */

/**
 * Secret special sauce.
 *
 * @since 1.26
 */
class ResourceLoaderOOUIImageModule extends ResourceLoaderImageModule {
	use ResourceLoaderOOUIModule;

	protected function loadFromDefinition() {
		if ( $this->definition === null ) {
			// Do nothing if definition was already processed
			return;
		}

		$themes = self::getSkinThemeMap();

		// For backwards-compatibility, allow missing 'themeImages'
		$module = isset( $this->definition['themeImages'] ) ? $this->definition['themeImages'] : '';

		$definition = [];
		foreach ( $themes as $skin => $theme ) {
			// Find the path to the JSON file which contains the actual image definitions for this theme
			if ( $module ) {
				$dataPath = $this->getThemeImagesPath( $theme, $module );
			} else {
				// Backwards-compatibility for things that probably shouldn't have used this class...
				$dataPath =
					$this->definition['rootPath'] . '/' .
					strtolower( $theme ) . '/' .
					$this->definition['name'] . '.json';
			}
			$localDataPath = $this->localBasePath . '/' . $dataPath;

			// If there's no file for this module of this theme, that's okay, it will just use the defaults
			if ( !file_exists( $localDataPath ) ) {
				continue;
			}
			$data = json_decode( file_get_contents( $localDataPath ), true );

			// Expand the paths to images (since they are relative to the JSON file that defines them, not
			// our base directory)
			$fixPath = function ( &$path ) use ( $dataPath ) {
				$path = dirname( $dataPath ) . '/' . $path;
			};
			array_walk( $data['images'], function ( &$value ) use ( $fixPath ) {
				if ( is_string( $value['file'] ) ) {
					$fixPath( $value['file'] );
				} elseif ( is_array( $value['file'] ) ) {
					array_walk_recursive( $value['file'], $fixPath );
				}
			} );

			// Convert into a definition compatible with the parent vanilla ResourceLoaderImageModule
			foreach ( $data as $key => $value ) {
				switch ( $key ) {
					// Images and color variants are defined per-theme, here converted to per-skin
					case 'images':
					case 'variants':
						$definition[$key][$skin] = $data[$key];
						break;

					// Other options must be identical for each theme (or only defined in the default one)
					default:
						if ( !isset( $definition[$key] ) ) {
							$definition[$key] = $data[$key];
						} elseif ( $definition[$key] !== $data[$key] ) {
							throw new Exception(
								"Mismatched OOUI theme images definition: " .
									"key '$key' of theme '$theme' for module '$module' " .
									"does not match other themes"
							);
						}
						break;
				}
			}
		}

		// Extra selectors to allow using the same icons for old-style MediaWiki UI code
		if ( substr( $module, 0, 5 ) === 'icons' ) {
			$definition['selectorWithoutVariant'] = '.oo-ui-icon-{name}, .mw-ui-icon-{name}:before';
			$definition['selectorWithVariant'] = '.oo-ui-image-{variant}.oo-ui-icon-{name}, ' .
				'.mw-ui-icon-{name}-{variant}:before';
		}

		// Fields from module definition silently override keys from JSON files
		$this->definition += $definition;

		parent::loadFromDefinition();
	}
}