Merge "Special:Upload should not crash on failing previews"
[mediawiki.git] / includes / installer / WebInstallerOutput.php
blob62fe7852b74a2c80e11ea30585b5b1e208d75049
1 <?php
2 /**
3 * Output handler for the web installer.
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * http://www.gnu.org/copyleft/gpl.html
20 * @file
21 * @ingroup Deployment
24 /**
25 * Output class modelled on OutputPage.
27 * I've opted to use a distinct class rather than derive from OutputPage here in
28 * the interests of separation of concerns: if we used a subclass, there would be
29 * quite a lot of things you could do in OutputPage that would break the installer,
30 * that wouldn't be immediately obvious.
32 * @ingroup Deployment
33 * @since 1.17
35 class WebInstallerOutput {
37 /**
38 * The WebInstaller object this WebInstallerOutput is used by.
40 * @var WebInstaller
42 public $parent;
44 /**
45 * Buffered contents that haven't been output yet
46 * @var string
48 private $contents = '';
50 /**
51 * Has the header (or short header) been output?
52 * @var bool
54 private $headerDone = false;
56 /**
57 * @var string
59 public $redirectTarget;
61 /**
62 * Does the current page need to allow being used as a frame?
63 * If not, X-Frame-Options will be output to forbid it.
65 * @var bool
67 public $allowFrames = false;
69 /**
70 * Whether to use the limited header (used during CC license callbacks)
71 * @var bool
73 private $useShortHeader = false;
75 /**
76 * @param WebInstaller $parent
78 public function __construct( WebInstaller $parent ) {
79 $this->parent = $parent;
82 /**
83 * @param string $html
85 public function addHTML( $html ) {
86 $this->contents .= $html;
87 $this->flush();
90 /**
91 * @param string $text
93 public function addWikiText( $text ) {
94 $this->addHTML( $this->parent->parse( $text ) );
97 /**
98 * @param string $html
100 public function addHTMLNoFlush( $html ) {
101 $this->contents .= $html;
105 * @param string $url
107 * @throws MWException
109 public function redirect( $url ) {
110 if ( $this->headerDone ) {
111 throw new MWException( __METHOD__ . ' called after sending headers' );
113 $this->redirectTarget = $url;
116 public function output() {
117 $this->flush();
119 if ( !$this->redirectTarget ) {
120 $this->outputFooter();
125 * Get the stylesheet of the MediaWiki skin.
127 * @return string
129 public function getCSS() {
130 global $wgStyleDirectory;
132 $moduleNames = [
133 // See SkinTemplate::setupSkinUserCss
134 'mediawiki.legacy.shared',
135 // See Vector::setupSkinUserCss
136 'mediawiki.skinning.interface',
139 $resourceLoader = new ResourceLoader();
141 if ( file_exists( "$wgStyleDirectory/Vector/skin.json" ) ) {
142 // Force loading Vector skin if available as a fallback skin
143 // for whatever ResourceLoader wants to have as the default.
144 $registry = new ExtensionRegistry();
145 $data = $registry->readFromQueue( [
146 "$wgStyleDirectory/Vector/skin.json" => 1,
147 ] );
148 if ( isset( $data['globals']['wgResourceModules'] ) ) {
149 $resourceLoader->register( $data['globals']['wgResourceModules'] );
152 $moduleNames[] = 'skins.vector.styles';
155 $moduleNames[] = 'mediawiki.legacy.config';
157 $rlContext = new ResourceLoaderContext( $resourceLoader, new FauxRequest( [
158 'debug' => 'true',
159 'lang' => $this->getLanguageCode(),
160 'only' => 'styles',
161 ] ) );
163 $styles = [];
164 foreach ( $moduleNames as $moduleName ) {
165 /** @var ResourceLoaderFileModule $module */
166 $module = $resourceLoader->getModule( $moduleName );
167 if ( !$module ) {
168 // T98043: Don't fatal, but it won't look as pretty.
169 continue;
172 // Based on: ResourceLoaderFileModule::getStyles (without the DB query)
173 $styles = array_merge( $styles, ResourceLoader::makeCombinedStyles(
174 $module->readStyleFiles(
175 $module->getStyleFiles( $rlContext ),
176 $module->getFlip( $rlContext ),
177 $rlContext
178 ) ) );
181 return implode( "\n", $styles );
185 * "<link>" to index.php?css=1 for the "<head>"
187 * @return string
189 private function getCssUrl() {
190 return Html::linkedStyle( $_SERVER['PHP_SELF'] . '?css=1' );
193 public function useShortHeader( $use = true ) {
194 $this->useShortHeader = $use;
197 public function allowFrames( $allow = true ) {
198 $this->allowFrames = $allow;
201 public function flush() {
202 if ( !$this->headerDone ) {
203 $this->outputHeader();
205 if ( !$this->redirectTarget && strlen( $this->contents ) ) {
206 echo $this->contents;
207 flush();
208 $this->contents = '';
213 * @return string
215 public function getDir() {
216 global $wgLang;
218 return is_object( $wgLang ) ? $wgLang->getDir() : 'ltr';
222 * @return string
224 public function getLanguageCode() {
225 global $wgLang;
227 return is_object( $wgLang ) ? $wgLang->getCode() : 'en';
231 * @return string[]
233 public function getHeadAttribs() {
234 return [
235 'dir' => $this->getDir(),
236 'lang' => wfBCP47( $this->getLanguageCode() ),
241 * Get whether the header has been output
243 * @return bool
245 public function headerDone() {
246 return $this->headerDone;
249 public function outputHeader() {
250 $this->headerDone = true;
251 $this->parent->request->response()->header( 'Content-Type: text/html; charset=utf-8' );
253 if ( !$this->allowFrames ) {
254 $this->parent->request->response()->header( 'X-Frame-Options: DENY' );
257 if ( $this->redirectTarget ) {
258 $this->parent->request->response()->header( 'Location: ' . $this->redirectTarget );
260 return;
263 if ( $this->useShortHeader ) {
264 $this->outputShortHeader();
266 return;
269 <?php echo Html::htmlHeader( $this->getHeadAttribs() ); ?>
271 <head>
272 <meta name="robots" content="noindex, nofollow" />
273 <meta http-equiv="Content-type" content="text/html; charset=utf-8" />
274 <title><?php $this->outputTitle(); ?></title>
275 <?php echo $this->getCssUrl() . "\n"; ?>
276 <?php echo $this->getJQuery() . "\n"; ?>
277 <?php echo Html::linkedScript( 'config.js' ) . "\n"; ?>
278 </head>
280 <?php echo Html::openElement( 'body', [ 'class' => $this->getDir() ] ) . "\n"; ?>
281 <div id="mw-page-base"></div>
282 <div id="mw-head-base"></div>
283 <div id="content" class="mw-body">
284 <div id="bodyContent" class="mw-body-content">
286 <h1><?php $this->outputTitle(); ?></h1>
287 <?php
290 public function outputFooter() {
291 if ( $this->useShortHeader ) {
292 echo Html::closeElement( 'body' ) . Html::closeElement( 'html' );
294 return;
298 </div></div>
300 <div id="mw-panel">
301 <div class="portal" id="p-logo">
302 <a style="background-image: url(images/installer-logo.png);"
303 href="https://www.mediawiki.org/"
304 title="Main Page"></a>
305 </div>
306 <?php
307 $message = wfMessage( 'config-sidebar' )->plain();
308 foreach ( explode( '----', $message ) as $section ) {
309 echo '<div class="portal"><div class="body">';
310 echo $this->parent->parse( $section, true );
311 echo '</div></div>';
314 </div>
316 <?php
317 echo Html::closeElement( 'body' ) . Html::closeElement( 'html' );
320 public function outputShortHeader() {
322 <?php echo Html::htmlHeader( $this->getHeadAttribs() ); ?>
323 <head>
324 <meta http-equiv="Content-type" content="text/html; charset=utf-8" />
325 <meta name="robots" content="noindex, nofollow" />
326 <title><?php $this->outputTitle(); ?></title>
327 <?php echo $this->getCssUrl() . "\n"; ?>
328 <?php echo $this->getJQuery(); ?>
329 <?php echo Html::linkedScript( 'config.js' ); ?>
330 </head>
332 <body style="background-image: none">
333 <?php
336 public function outputTitle() {
337 global $wgVersion;
338 echo wfMessage( 'config-title', $wgVersion )->escaped();
342 * @return string
344 public function getJQuery() {
345 return Html::linkedScript( "../resources/lib/jquery/jquery.js" );