MediaWiki REL1_32
MonoBookTemplate.php
Go to the documentation of this file.
1<?php
31
38 public function execute() {
39 // Open html, body elements, etc
40 $html = $this->get( 'headelement' );
41 $html .= Html::openElement( 'div', [ 'id' => 'globalWrapper' ] );
42
43 $html .= Html::openElement( 'div', [ 'id' => 'column-content' ] );
44 $html .= Html::rawElement( 'div', [ 'id' => 'content', 'class' => 'mw-body', 'role' => 'main' ],
45 Html::element( 'a', [ 'id' => 'top' ] ) .
46 $this->getIfExists( 'sitenotice', [
47 'wrapper' => 'div',
48 'parameters' => [ 'id' => 'siteNotice', 'class' => 'mw-body-content' ]
49 ] ) .
50 $this->getIndicators() .
51 $this->getIfExists( 'title', [
52 'loose' => true,
53 'wrapper' => 'h1',
54 'parameters' => [
55 'id' => 'firstHeading',
56 'class' => 'firstHeading',
57 'lang' => $this->getSkin()->getTitle()->getPageViewLanguage()->getHtmlCode()
58 ]
59 ] ) .
60 Html::rawElement( 'div', [ 'id' => 'bodyContent', 'class' => 'mw-body-content' ],
61 Html::rawElement( 'div', [ 'id' => 'siteSub' ], $this->getMsg( 'tagline' )->parse() ) .
62 Html::rawElement(
63 'div',
64 [ 'id' => 'contentSub', 'lang' => $this->get( 'userlang' ), 'dir' => $this->get( 'dir' ) ],
65 $this->get( 'subtitle' )
66 ) .
67 $this->getIfExists( 'undelete', [ 'wrapper' => 'div', 'parameters' => [
68 'id' => 'contentSub2'
69 ] ] ) .
70 $this->getIfExists( 'newtalk', [ 'wrapper' => 'div', 'parameters' => [
71 'class' => 'usermessage'
72 ] ] ) .
73 Html::element( 'div', [ 'id' => 'jump-to-nav' ] ) .
74 Html::element( 'a', [ 'href' => '#column-one', 'class' => 'mw-jump-link' ],
75 $this->getMsg( 'monobook-jumptonavigation' )->text()
76 ) .
77 Html::element( 'a', [ 'href' => '#searchInput', 'class' => 'mw-jump-link' ],
78 $this->getMsg( 'monobook-jumptosearch' )->text()
79 ) .
80 '<!-- start content -->' .
81
82 $this->get( 'bodytext' ) .
83 $this->getIfExists( 'catlinks' ) .
84
85 '<!-- end content -->' .
86 $this->getIfExists( 'dataAfterContent' ) .
87 $this->getClear()
88 )
89 );
90 $html .= $this->deprecatedHookHack( 'MonoBookAfterContent' );
91 $html .= Html::closeElement( 'div' );
92
93 $html .= Html::rawElement( 'div',
94 [
95 'id' => 'column-one',
96 'lang' => $this->get( 'userlang' ),
97 'dir' => $this->get( 'dir' )
98 ],
99 Html::element( 'h2', [], $this->getMsg( 'navigation-heading' )->text() ) .
100 $this->getCactions() .
101 $this->getBox( 'personal', $this->getPersonalTools(), 'personaltools' ) .
102 Html::rawElement( 'div', [ 'class' => 'portlet', 'id' => 'p-logo', 'role' => 'banner' ],
103 Html::element( 'a',
104 [
105 'href' => $this->data['nav_urls']['mainpage']['href'],
106 'class' => 'mw-wiki-logo',
107 ]
109 )
110 ) .
111 Html::rawElement( 'div', [ 'id' => 'sidebar' ], $this->getRenderedSidebar() ) .
113 'sidebar',
114 $this->getMsg( 'jumptonavigation' )->text()
115 ) .
117 'p-personal',
118 $this->getMsg( 'monobook-jumptopersonal' )->text()
119 ) .
121 'globalWrapper',
122 $this->getMsg( 'monobook-jumptotop' )->text()
123 )
124 );
125 $html .= '<!-- end of the left (by default at least) column -->';
126
127 $html .= $this->getClear();
128 $html .= $this->getSimpleFooter();
129 $html .= Html::closeElement( 'div' );
130
131 $html .= $this->getTrail();
132
133 $html .= Html::closeElement( 'body' );
134 $html .= Html::closeElement( 'html' );
135
136 // The unholy echo
137 echo $html;
138 }
139
150 protected function getMobileNavigationIcon( $target, $title ) {
151 return Html::element( 'a', [
152 'href' => "#$target",
153 'title' => $title,
154 'class' => 'menu-toggle',
155 'id' => "$target-toggle"
156 ] );
157 }
158
164 protected function getCactions() {
165 $html = '';
166 $allTabs = $this->data['content_actions'];
167 $tabCount = count( $allTabs );
168
169 // Normal cactions
170 if ( $tabCount > 2 ) {
171 $html .= $this->getBox( 'cactions', $allTabs, 'monobook-cactions-label' );
172 } else {
173 // Is redundant with spoof, hide normal cactions entirely in mobile
174 $html .= $this->getBox( 'cactions', $allTabs, 'monobook-cactions-label',
175 [ 'extra-classes' => 'nomobile' ]
176 );
177 }
178
179 // Mobile cactions tabs
180 $tabs = $this->data['content_navigation']['namespaces'];
181 foreach ( $tabs as $tab => $attribs ) {
182 $tabs[$tab]['id'] = $attribs['id'] . '-mobile';
183 $tabs[$tab]['title'] = $attribs['text'];
184 }
185
186 if ( $tabCount !== 1 ) {
187 // Is not special page or stuff, append a 'more'
188 $tabs['more'] = [
189 'text' => $this->getMsg( 'monobook-more-actions' )->text(),
190 'href' => '#p-cactions',
191 'id' => 'ca-more'
192 ];
193 }
194 $tabs['toolbox'] = [
195 'text' => $this->getMsg( 'toolbox' )->text(),
196 'href' => '#p-tb',
197 'id' => 'ca-tools',
198 'title' => $this->getMsg( 'toolbox' )->text()
199 ];
200 if ( $this->data['language_urls'] !== false ) {
201 $tabs['languages'] = [
202 'text' => $this->getMsg( 'otherlanguages' )->text(),
203 'href' => '#p-lang',
204 'id' => 'ca-languages',
205 'title' => $this->getMsg( 'otherlanguages' )->text()
206 ];
207 }
208
209 $html .= $this->getBox( 'cactions-mobile', $tabs, 'monobook-cactions-label' );
210
211 return $html;
212 }
213
219 protected function getRenderedSidebar() {
220 $sidebar = $this->data['sidebar'];
221 $html = '';
222
223 if ( !isset( $sidebar['SEARCH'] ) ) {
224 $sidebar['SEARCH'] = true;
225 }
226 if ( !isset( $sidebar['TOOLBOX'] ) ) {
227 $sidebar['TOOLBOX'] = true;
228 }
229 if ( !isset( $sidebar['LANGUAGES'] ) ) {
230 $sidebar['LANGUAGES'] = true;
231 }
232
233 foreach ( $sidebar as $boxName => $content ) {
234 if ( $content === false ) {
235 continue;
236 }
237
238 // Numeric strings gets an integer when set as key, cast back - T73639
239 $boxName = (string)$boxName;
240
241 if ( $boxName == 'SEARCH' ) {
242 $html .= $this->getSearchBox();
243 } elseif ( $boxName == 'TOOLBOX' ) {
244 $html .= $this->getToolboxBox();
245 } elseif ( $boxName == 'LANGUAGES' ) {
246 $html .= $this->getLanguageBox();
247 } else {
248 $html .= $this->getBox(
249 $boxName,
250 $content,
251 null,
252 [ 'extra-classes' => 'generated-sidebar' ]
253 );
254 }
255 }
256
257 return $html;
258 }
259
265 protected function getSearchBox() {
266 $html = '';
267
268 if ( $this->config->get( 'UseTwoButtonsSearchForm' ) ) {
269 $optionButtons = '&#160; ' . $this->makeSearchButton(
270 'fulltext',
271 [ 'id' => 'mw-searchButton', 'class' => 'searchButton' ]
272 );
273 } else {
274 $optionButtons = Html::rawElement( 'div', [],
275 Html::rawElement( 'a', [ 'href' => $this->get( 'searchaction' ), 'rel' => 'search' ],
276 $this->getMsg( 'powersearch-legend' )->escaped()
277 )
278 );
279 }
280 $searchInputId = 'searchInput';
281 $searchForm = Html::rawElement( 'form', [
282 'action' => $this->get( 'wgScript' ),
283 'id' => 'searchform'
284 ],
285 Html::hidden( 'title', $this->get( 'searchtitle' ) ) .
286 $this->makeSearchInput( [ 'id' => $searchInputId ] ) .
287 $this->makeSearchButton( 'go', [ 'id' => 'searchGoButton', 'class' => 'searchButton' ] ) .
288 $optionButtons
289 );
290
291 $html .= $this->getBox( 'search', $searchForm, null, [
292 'search-input-id' => $searchInputId,
293 'role' => 'search',
294 'body-id' => 'searchBody'
295 ] );
296
297 return $html;
298 }
299
305 protected function getToolboxBox() {
306 $html = '';
307 $skin = $this;
308
309 $html .= $this->getBox( 'tb', $this->getToolbox(), 'toolbox', [ 'hooks' => [
310 // Deprecated hooks
311 'MonoBookTemplateToolboxEnd' => [ &$skin ],
312 'SkinTemplateToolboxEnd' => [ &$skin, true ]
313 ] ] );
314
315 $html .= $this->deprecatedHookHack( 'MonoBookAfterToolbox' );
316
317 return $html;
318 }
319
325 protected function getLanguageBox() {
326 $html = '';
327
328 if ( $this->data['language_urls'] !== false ) {
329 $html .= $this->getBox( 'lang', $this->data['language_urls'], 'otherlanguages' );
330 }
331
332 return $html;
333 }
334
345 protected function getBox( $name, $contents, $msg = null, $setOptions = [] ) {
346 $options = [
347 'class' => 'portlet',
348 'body-class' => 'pBody',
349 'text-wrapper' => ''
350 ];
351 foreach ( $setOptions as $key => $value ) {
352 $options[$key] = $value;
353 }
354
355 // Do some special stuff for the personal menu
356 if ( $name == 'personal' ) {
357 $prependiture = '';
358
359 // Extension:UniversalLanguageSelector order - T121793
360 if ( array_key_exists( 'uls', $contents ) ) {
361 $prependiture .= $this->makeListItem( 'uls', $contents['uls'] );
362 unset( $contents['uls'] );
363 }
364 if ( !$this->getSkin()->getUser()->isLoggedIn() &&
365 User::groupHasPermission( '*', 'edit' )
366 ) {
367 $prependiture .= Html::rawElement(
368 'li',
369 [ 'id' => 'pt-anonuserpage' ],
370 $this->getMsg( 'notloggedin' )->escaped()
371 );
372 }
373 $options['list-prepend'] = $prependiture;
374 }
375
376 return $this->getPortlet( $name, $contents, $msg, $options );
377 }
378
389 protected function getPortlet( $name, $content, $msg = null, $setOptions = [] ) {
390 // random stuff to override with any provided options
391 $options = [
392 // handle role=search a little differently
393 'role' => 'navigation',
394 'search-input-id' => 'searchInput',
395 // extra classes/ids
396 'id' => 'p-' . $name,
397 'class' => 'mw-portlet',
398 'extra-classes' => '',
399 'body-id' => null,
400 'body-class' => 'mw-portlet-body',
401 'body-extra-classes' => '',
402 // wrapper for individual list items
403 'text-wrapper' => [ 'tag' => 'span' ],
404 // old toolbox hook support (use: [ 'SkinTemplateToolboxEnd' => [ &$skin, true ] ])
405 'hooks' => '',
406 // option to stick arbitrary stuff at the beginning of the ul
407 'list-prepend' => ''
408 ];
409 // set options based on input
410 foreach ( $setOptions as $key => $value ) {
411 $options[$key] = $value;
412 }
413
414 // Handle the different $msg possibilities
415 if ( $msg === null ) {
416 $msg = $name;
417 $msgParams = [];
418 } elseif ( is_array( $msg ) ) {
419 $msgString = array_shift( $msg );
420 $msgParams = $msg;
421 $msg = $msgString;
422 } else {
423 $msgParams = [];
424 }
425 $msgObj = $this->getMsg( $msg, $msgParams );
426 if ( $msgObj->exists() ) {
427 $msgString = $msgObj->parse();
428 } else {
429 $msgString = htmlspecialchars( $msg );
430 }
431
432 $labelId = Sanitizer::escapeIdForAttribute( "p-$name-label" );
433
434 if ( is_array( $content ) ) {
435 $contentText = Html::openElement( 'ul',
436 [ 'lang' => $this->get( 'userlang' ), 'dir' => $this->get( 'dir' ) ]
437 );
438 $contentText .= $options['list-prepend'];
439 foreach ( $content as $key => $item ) {
440 if ( is_array( $options['text-wrapper'] ) ) {
441 $contentText .= $this->makeListItem(
442 $key,
443 $item,
444 [ 'text-wrapper' => $options['text-wrapper'] ]
445 );
446 } else {
447 $contentText .= $this->makeListItem(
448 $key,
449 $item
450 );
451 }
452 }
453 // Compatibility with extensions still using SkinTemplateToolboxEnd or similar
454 if ( is_array( $options['hooks'] ) ) {
455 foreach ( $options['hooks'] as $hook => $hookOptions ) {
456 $contentText .= $this->deprecatedHookHack( $hook, $hookOptions );
457 }
458 }
459
460 $contentText .= Html::closeElement( 'ul' );
461 } else {
462 $contentText = $content;
463 }
464
465 // Special handling for role=search
466 $divOptions = [
467 'role' => $options['role'],
468 'class' => $this->mergeClasses( $options['class'], $options['extra-classes'] ),
469 'id' => Sanitizer::escapeIdForAttribute( $options['id'] ),
470 'title' => Linker::titleAttrib( $options['id'] )
471 ];
472 if ( $options['role'] !== 'search' ) {
473 $divOptions['aria-labelledby'] = $labelId;
474 }
475 $labelOptions = [
476 'id' => $labelId,
477 'lang' => $this->get( 'userlang' ),
478 'dir' => $this->get( 'dir' )
479 ];
480 if ( $options['role'] == 'search' ) {
481 $msgString = Html::rawElement( 'label', [ 'for' => $options['search-input-id'] ], $msgString );
482 }
483
484 $bodyDivOptions = [
485 'class' => $this->mergeClasses( $options['body-class'], $options['body-extra-classes'] )
486 ];
487 if ( is_string( $options['body-id'] ) ) {
488 $bodyDivOptions['id'] = $options['body-id'];
489 }
490
491 $html = Html::rawElement( 'div', $divOptions,
492 Html::rawElement( 'h3', $labelOptions, $msgString ) .
493 Html::rawElement( 'div', $bodyDivOptions,
494 $contentText .
495 $this->getAfterPortlet( $name )
496 )
497 );
498
499 return $html;
500 }
501
513 protected function mergeClasses( $class, $extraClasses ) {
514 if ( !is_array( $class ) ) {
515 $class = [ $class ];
516 }
517 if ( !is_array( $extraClasses ) ) {
518 $extraClasses = [ $extraClasses ];
519 }
520
521 return array_merge( $class, $extraClasses );
522 }
523
533 protected function deprecatedHookHack( $hook, $hookOptions = [] ) {
534 $hookContents = '';
535 ob_start();
536 Hooks::run( $hook, $hookOptions );
537 $hookContents = ob_get_contents();
538 ob_end_clean();
539 if ( !trim( $hookContents ) ) {
540 $hookContents = '';
541 }
542
543 return $hookContents;
544 }
545
554 protected function getIfExists( $object, $setOptions = [] ) {
555 $options = [
556 'loose' => false,
557 'wrapper' => 'none',
558 'parameters' => []
559 ];
560 foreach ( $setOptions as $key => $value ) {
561 $options[$key] = $value;
562 }
563
564 $html = '';
565
566 if ( ( $options['loose'] && $this->data[$object] != '' ) ||
567 ( !$options['loose'] && $this->data[$object] ) ) {
568 if ( $options['wrapper'] == 'none' ) {
569 $html .= $this->get( $object );
570 } else {
571 $html .= Html::rawElement(
572 $options['wrapper'],
573 $options['parameters'],
574 $this->get( $object )
575 );
576 }
577 }
578
579 return $html;
580 }
581
587 protected function getSimpleFooter() {
588 $validFooterIcons = $this->getFooterIcons( 'icononly' );
589 $validFooterLinks = $this->getFooterLinks( 'flat' );
590
591 $html = '';
592
593 $html .= Html::openElement( 'div', [
594 'id' => 'footer',
595 'role' => 'contentinfo',
596 'lang' => $this->get( 'userlang' ),
597 'dir' => $this->get( 'dir' )
598 ] );
599
600 foreach ( $validFooterIcons as $blockName => $footerIcons ) {
601 $html .= Html::openElement( 'div', [
602 'id' => Sanitizer::escapeIdForAttribute( "f-{$blockName}ico" ),
603 'class' => 'footer-icons'
604 ] );
605 foreach ( $footerIcons as $icon ) {
606 $html .= $this->getSkin()->makeFooterIcon( $icon );
607 }
608 $html .= Html::closeElement( 'div' );
609 }
610 if ( count( $validFooterLinks ) > 0 ) {
611 $html .= Html::openElement( 'ul', [ 'id' => 'f-list' ] );
612 foreach ( $validFooterLinks as $aLink ) {
613 $html .= Html::rawElement(
614 'li',
615 [ 'id' => Sanitizer::escapeIdForAttribute( $aLink ) ],
616 $this->get( $aLink )
617 );
618 }
619 $html .= Html::closeElement( 'ul' );
620 }
621 $html .= Html::closeElement( 'div' );
622
623 return $html;
624 }
625}
This list may contain false positives That usually means there is additional text with links below the first Each row contains links to the first and second as well as the first line of the second redirect text
New base template for a skin's template extended from QuickTemplate this class features helper method...
getFooterLinks( $option=null)
Returns an array of footerlinks trimmed down to only those footer links that are valid.
makeSearchButton( $mode, $attrs=[])
getToolbox()
Create an array of common toolbox items from the data in the quicktemplate stored by SkinTemplate.
getTrail()
Get the basic end-page trail including bottomscripts, reporttime, and debug stuff.
getPersonalTools()
Create an array of personal tools items from the data in the quicktemplate stored by SkinTemplate.
getFooterIcons( $option=null)
Returns an array of footer icons filtered down by options relevant to how the skin wishes to display ...
makeListItem( $key, $item, $options=[])
Generates a list item for a navigation, portlet, portal, sidebar... list.
getMsg( $name)
Get a Message object with its context set.
getIndicators()
Get the suggested HTML for page status indicators: icons (or short text snippets) usually displayed i...
makeSearchInput( $attrs=[])
getClear()
Get a div with the core visualClear class, for clearing floats.
getAfterPortlet( $name)
Allows extensions to hook into known portlets and add stuff to them.
static titleAttrib( $name, $options=null, array $msgParams=[])
Given the id of an interface element, constructs the appropriate title attribute from the system mess...
Definition Linker.php:1967
static tooltipAndAccesskeyAttribs( $name, array $msgParams=[], $options=null)
Returns the attributes for the tooltip and access key.
Definition Linker.php:2133
execute()
Template filter callback for MonoBook skin.
getBox( $name, $contents, $msg=null, $setOptions=[])
Generate a sidebar box using getPortlet(); prefill some common stuff.
getSimpleFooter()
Renderer for getFooterIcons and getFooterLinks as a generic footer block.
getMobileNavigationIcon( $target, $title)
Create a wrapped link to create a mobile toggle/jump icon Needs to be an on-page link (as opposed to ...
getSearchBox()
Generate the search, using config options for buttons (?)
getPortlet( $name, $content, $msg=null, $setOptions=[])
Generates a block of navigation links with a header.
getRenderedSidebar()
Generate the full sidebar.
getLanguageBox()
Generate the languages box.
getIfExists( $object, $setOptions=[])
Simple wrapper for random if-statement-wrapped $this->data things.
mergeClasses( $class, $extraClasses)
Helper function for getPortlet.
getCactions()
Generate the cactions (content actions) tabs, as well as a second set of spoof tabs for mobile.
getToolboxBox()
Generate the toolbox, complete with all three old hooks.
deprecatedHookHack( $hook, $hookOptions=[])
Wrapper to catch output of old hooks expecting to write directly to page We no longer do things that ...
getSkin()
Get the Skin object related to this object.
static groupHasPermission( $group, $role)
Check, if the given group has the given permission.
Definition User.php:5013
This code would result in ircNotify being run twice when an article is and once for brion Hooks can return three possible true was required This is the default since MediaWiki *some string
Definition hooks.txt:181
null means default in associative array with keys and values unescaped Should be merged with default with a value of false meaning to suppress the attribute in associative array with keys and values unescaped & $options
Definition hooks.txt:2050
null means default in associative array with keys and values unescaped Should be merged with default with a value of false meaning to suppress the attribute in associative array with keys and values unescaped noclasses just before the function returns a value If you return an< a > element with HTML attributes $attribs and contents $html will be returned If you return $ret will be returned and may include noclasses & $html
Definition hooks.txt:2062
Allows to change the fields on the form that will be generated $name
Definition hooks.txt:302
null means default in associative array with keys and values unescaped Should be merged with default with a value of false meaning to suppress the attribute in associative array with keys and values unescaped noclasses just before the function returns a value If you return an< a > element with HTML attributes $attribs and contents $html will be returned If you return $ret will be returned and may include noclasses after processing & $attribs
Definition hooks.txt:2063
$content