2015-04-27 14:05:31 +00:00
|
|
|
<?php
|
|
|
|
|
2015-08-27 08:30:25 +00:00
|
|
|
use Wikia\PortableInfobox\Helpers\PortableInfoboxRenderServiceHelper;
|
2015-07-28 11:49:17 +00:00
|
|
|
|
2015-07-02 11:55:39 +00:00
|
|
|
class PortableInfoboxRenderService extends WikiaService {
|
2015-05-12 11:32:42 +00:00
|
|
|
const MOBILE_TEMPLATE_POSTFIX = '-mobile';
|
2015-09-28 15:46:40 +00:00
|
|
|
const MEDIA_CONTEXT_INFOBOX_HERO_IMAGE = 'infobox-hero-image';
|
2015-09-30 08:10:31 +00:00
|
|
|
const MEDIA_CONTEXT_INFOBOX = 'infobox';
|
2015-04-27 14:05:31 +00:00
|
|
|
|
2015-08-27 08:30:25 +00:00
|
|
|
private static $templates = [
|
2015-04-27 14:05:31 +00:00
|
|
|
'wrapper' => 'PortableInfoboxWrapper.mustache',
|
|
|
|
'title' => 'PortableInfoboxItemTitle.mustache',
|
|
|
|
'header' => 'PortableInfoboxItemHeader.mustache',
|
|
|
|
'image' => 'PortableInfoboxItemImage.mustache',
|
2015-05-12 11:32:42 +00:00
|
|
|
'image-mobile' => 'PortableInfoboxItemImageMobile.mustache',
|
2016-04-01 15:33:29 +00:00
|
|
|
'image-mobile-wikiamobile' => 'PortableInfoboxItemImageMobileWikiaMobile.mustache',
|
2015-05-04 10:48:57 +00:00
|
|
|
'data' => 'PortableInfoboxItemData.mustache',
|
2015-04-27 14:05:31 +00:00
|
|
|
'group' => 'PortableInfoboxItemGroup.mustache',
|
2015-07-28 09:46:41 +00:00
|
|
|
'horizontal-group-content' => 'PortableInfoboxHorizontalGroupContent.mustache',
|
2015-07-03 10:05:59 +00:00
|
|
|
'navigation' => 'PortableInfoboxItemNavigation.mustache',
|
2015-10-16 01:15:52 +00:00
|
|
|
'hero-mobile' => 'PortableInfoboxItemHeroMobile.mustache',
|
2016-04-01 15:33:29 +00:00
|
|
|
'hero-mobile-wikiamobile' => 'PortableInfoboxItemHeroMobileWikiaMobile.mustache',
|
2015-10-16 01:15:52 +00:00
|
|
|
'image-collection' => 'PortableInfoboxItemImageCollection.mustache',
|
2016-02-15 17:44:02 +00:00
|
|
|
'image-collection-mobile' => 'PortableInfoboxItemImageCollectionMobile.mustache',
|
2016-04-01 15:33:29 +00:00
|
|
|
'image-collection-mobile-wikiamobile' => 'PortableInfoboxItemImageCollectionMobileWikiaMobile.mustache'
|
2015-04-27 14:05:31 +00:00
|
|
|
];
|
|
|
|
private $templateEngine;
|
2016-03-17 09:37:18 +00:00
|
|
|
private $imagesWidth;
|
2015-04-27 14:05:31 +00:00
|
|
|
|
2015-07-02 11:55:39 +00:00
|
|
|
function __construct() {
|
|
|
|
$this->templateEngine = ( new Wikia\Template\MustacheEngine )
|
2015-08-27 08:30:25 +00:00
|
|
|
->setPrefix( self::getTemplatesDir() );
|
2016-03-17 09:37:18 +00:00
|
|
|
$this->imagesWidth = PortableInfoboxRenderServiceHelper::DEFAULT_DESKTOP_THUMBNAIL_WIDTH;
|
2015-08-27 08:30:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static function getTemplatesDir() {
|
|
|
|
return dirname( __FILE__ ) . '/../templates';
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function getTemplates() {
|
|
|
|
return self::$templates;
|
2015-04-27 14:05:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* renders infobox
|
|
|
|
*
|
|
|
|
* @param array $infoboxdata
|
2015-07-02 11:55:39 +00:00
|
|
|
*
|
2015-11-03 16:19:29 +00:00
|
|
|
* @param $theme
|
|
|
|
* @param $layout
|
2015-04-27 14:05:31 +00:00
|
|
|
* @return string - infobox HTML
|
|
|
|
*/
|
2016-12-27 12:50:53 +00:00
|
|
|
public function renderInfobox( array $infoboxdata, $theme, $layout, $accentColor, $accentColorText ) {
|
2015-07-02 11:55:39 +00:00
|
|
|
wfProfileIn( __METHOD__ );
|
2015-07-28 11:49:17 +00:00
|
|
|
|
2015-07-28 14:13:36 +00:00
|
|
|
$helper = new PortableInfoboxRenderServiceHelper();
|
2015-04-27 14:05:31 +00:00
|
|
|
$infoboxHtmlContent = '';
|
2015-07-02 11:55:39 +00:00
|
|
|
$heroData = [ ];
|
2015-04-27 14:05:31 +00:00
|
|
|
|
2016-03-17 09:37:18 +00:00
|
|
|
// decide on image width
|
2016-04-04 11:13:52 +00:00
|
|
|
$this->imagesWidth = $helper->isMobile() ?
|
2016-03-17 09:37:18 +00:00
|
|
|
PortableInfoboxRenderServiceHelper::MOBILE_THUMBNAIL_WIDTH :
|
|
|
|
// if europa go with bigger images! else default size
|
2016-03-29 10:31:03 +00:00
|
|
|
$helper->isEuropaTheme() ? PortableInfoboxRenderServiceHelper::EUROPA_THUMBNAIL_WIDTH :
|
2016-03-17 09:37:18 +00:00
|
|
|
PortableInfoboxRenderServiceHelper::DEFAULT_DESKTOP_THUMBNAIL_WIDTH;
|
|
|
|
|
2015-07-02 11:55:39 +00:00
|
|
|
foreach ( $infoboxdata as $item ) {
|
|
|
|
$data = $item[ 'data' ];
|
|
|
|
$type = $item[ 'type' ];
|
2015-04-27 14:05:31 +00:00
|
|
|
|
2015-07-02 11:55:39 +00:00
|
|
|
switch ( $type ) {
|
2015-04-27 14:05:31 +00:00
|
|
|
case 'group':
|
2016-12-27 12:50:53 +00:00
|
|
|
$infoboxHtmlContent .= $this->renderGroup( $data, $accentColor, $accentColorText );
|
2015-04-27 14:05:31 +00:00
|
|
|
break;
|
2015-07-01 14:19:31 +00:00
|
|
|
case 'navigation':
|
|
|
|
$infoboxHtmlContent .= $this->renderItem( 'navigation', $data );
|
2015-04-27 14:05:31 +00:00
|
|
|
break;
|
2016-12-27 11:10:11 +00:00
|
|
|
case 'title':
|
|
|
|
if ( !$helper->isMercury() ) {
|
|
|
|
$data['accentColor'] = $accentColor;
|
2016-12-27 12:50:53 +00:00
|
|
|
$data['accentColorText'] = $accentColorText;
|
2016-12-27 11:10:11 +00:00
|
|
|
}
|
|
|
|
$infoboxHtmlContent .= $this->renderItem( 'title', $data );
|
|
|
|
break;
|
2015-04-27 14:05:31 +00:00
|
|
|
default:
|
2016-04-04 11:13:52 +00:00
|
|
|
if ( $helper->isMobile() && $helper->isValidHeroDataItem( $item, $heroData ) ) {
|
2015-07-02 11:55:39 +00:00
|
|
|
$heroData[ $type ] = $data;
|
2015-07-01 13:18:20 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-08-27 08:46:38 +00:00
|
|
|
if ( $helper->isTypeSupportedInTemplates( $type, self::getTemplates() ) ) {
|
2016-03-14 09:36:08 +00:00
|
|
|
$infoboxHtmlContent .= $this->renderItem( $type, $data );
|
2015-04-27 14:05:31 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-30 22:20:39 +00:00
|
|
|
if ( !empty( $heroData ) ) {
|
2016-12-27 11:10:11 +00:00
|
|
|
$infoboxHtmlContent = $this->renderInfoboxHero( $heroData, $accentColor ) . $infoboxHtmlContent;
|
2015-07-01 13:18:20 +00:00
|
|
|
}
|
|
|
|
|
2015-07-02 11:55:39 +00:00
|
|
|
if ( !empty( $infoboxHtmlContent ) ) {
|
2016-03-29 10:31:03 +00:00
|
|
|
$output = $this->renderItem( 'wrapper', [
|
2016-03-29 10:12:11 +00:00
|
|
|
'content' => $infoboxHtmlContent,
|
|
|
|
'theme' => $theme,
|
|
|
|
'layout' => $layout,
|
2016-03-29 10:31:03 +00:00
|
|
|
'isEuropaEnabled' => $helper->isEuropaTheme()
|
|
|
|
] );
|
2015-04-27 14:05:31 +00:00
|
|
|
} else {
|
|
|
|
$output = '';
|
|
|
|
}
|
|
|
|
|
2016-05-30 22:20:39 +00:00
|
|
|
\Wikia\PortableInfobox\Helpers\PortableInfoboxDataBag::getInstance()->setFirstInfoboxAlredyRendered( true );
|
2016-05-30 16:35:11 +00:00
|
|
|
|
2015-07-02 11:55:39 +00:00
|
|
|
wfProfileOut( __METHOD__ );
|
2015-04-27 14:05:31 +00:00
|
|
|
|
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* renders group infobox component
|
|
|
|
*
|
|
|
|
* @param array $groupData
|
2015-07-02 11:55:39 +00:00
|
|
|
*
|
2015-04-27 14:05:31 +00:00
|
|
|
* @return string - group HTML markup
|
|
|
|
*/
|
2016-12-27 12:50:53 +00:00
|
|
|
private function renderGroup( $groupData, $accentColor, $accentColorText ) {
|
2016-03-17 09:37:18 +00:00
|
|
|
$cssClasses = [ ];
|
2015-12-23 10:20:39 +00:00
|
|
|
$helper = new PortableInfoboxRenderServiceHelper();
|
2015-04-27 14:05:31 +00:00
|
|
|
$groupHTMLContent = '';
|
2015-07-02 11:55:39 +00:00
|
|
|
$dataItems = $groupData[ 'value' ];
|
|
|
|
$layout = $groupData[ 'layout' ];
|
2015-10-26 21:59:34 +00:00
|
|
|
$collapse = $groupData[ 'collapse' ];
|
2015-04-27 14:05:31 +00:00
|
|
|
|
2015-09-04 15:30:54 +00:00
|
|
|
if ( $layout === 'horizontal' ) {
|
2015-07-28 09:46:41 +00:00
|
|
|
$groupHTMLContent .= $this->renderItem(
|
|
|
|
'horizontal-group-content',
|
2015-07-28 11:49:17 +00:00
|
|
|
$helper->createHorizontalGroupData( $dataItems )
|
2015-07-28 09:46:41 +00:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
foreach ( $dataItems as $item ) {
|
|
|
|
$type = $item[ 'type' ];
|
2015-04-27 14:05:31 +00:00
|
|
|
|
2016-12-27 11:10:11 +00:00
|
|
|
if ( $type === 'header' && !$helper->isMercury() ) {
|
|
|
|
$item[ 'data' ][ 'accentColor' ] = $accentColor;
|
2016-12-27 12:50:53 +00:00
|
|
|
$item[ 'data' ][ 'accentColorText' ] = $accentColorText;
|
2016-12-27 11:10:11 +00:00
|
|
|
}
|
|
|
|
|
2015-08-27 08:46:38 +00:00
|
|
|
if ( $helper->isTypeSupportedInTemplates( $type, self::getTemplates() ) ) {
|
2015-07-28 09:46:41 +00:00
|
|
|
$groupHTMLContent .= $this->renderItem( $type, $item[ 'data' ] );
|
|
|
|
}
|
2015-04-27 14:05:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-26 21:59:34 +00:00
|
|
|
if ( $collapse !== null && count( $dataItems ) > 0 && $dataItems[ 0 ][ 'type' ] === 'header' ) {
|
|
|
|
$cssClasses[] = 'pi-collapse';
|
|
|
|
$cssClasses[] = 'pi-collapse-' . $collapse;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->renderItem( 'group', [
|
|
|
|
'content' => $groupHTMLContent,
|
2016-03-17 09:37:18 +00:00
|
|
|
'cssClasses' => implode( ' ', $cssClasses )
|
2015-10-26 21:59:34 +00:00
|
|
|
] );
|
2015-04-27 14:05:31 +00:00
|
|
|
}
|
|
|
|
|
2015-07-01 13:18:20 +00:00
|
|
|
/**
|
|
|
|
* renders infobox hero component
|
|
|
|
*
|
|
|
|
* @param array $data - infobox hero component data
|
2015-07-02 11:55:39 +00:00
|
|
|
*
|
2015-07-01 13:18:20 +00:00
|
|
|
* @return string
|
|
|
|
*/
|
2016-12-27 11:10:11 +00:00
|
|
|
private function renderInfoboxHero( $data, $accentColor ) {
|
2015-07-28 14:13:36 +00:00
|
|
|
$helper = new PortableInfoboxRenderServiceHelper();
|
|
|
|
|
2016-05-30 22:20:39 +00:00
|
|
|
// In Mercury SPA content of the first infobox's hero module has been moved to the article header.
|
|
|
|
$firstInfoboxAlredyRendered = \Wikia\PortableInfobox\Helpers\PortableInfoboxDataBag::getInstance()
|
|
|
|
->isFirstInfoboxAlredyRendered();
|
|
|
|
|
2015-07-01 16:03:34 +00:00
|
|
|
if ( array_key_exists( 'image', $data ) ) {
|
2015-10-16 01:15:52 +00:00
|
|
|
$image = $data[ 'image' ][ 0 ];
|
|
|
|
$image[ 'context' ] = self::MEDIA_CONTEXT_INFOBOX_HERO_IMAGE;
|
2016-03-17 09:37:18 +00:00
|
|
|
$image = $helper->extendImageData( $image, PortableInfoboxRenderServiceHelper::MOBILE_THUMBNAIL_WIDTH );
|
|
|
|
$data[ 'image' ] = $image;
|
2016-02-05 17:02:02 +00:00
|
|
|
|
2016-04-04 11:13:52 +00:00
|
|
|
if ( !$helper->isMercury() ) {
|
2016-05-30 22:20:39 +00:00
|
|
|
return $this->renderItem( 'hero-mobile-wikiamobile', $data );
|
|
|
|
} elseif ( $firstInfoboxAlredyRendered ) {
|
|
|
|
return $this->renderItem( 'hero-mobile', $data );
|
2016-02-05 17:02:02 +00:00
|
|
|
}
|
2016-05-30 22:20:39 +00:00
|
|
|
} elseif ( !$helper->isMercury() || $firstInfoboxAlredyRendered ) {
|
|
|
|
return $this->renderItem( 'title', $data[ 'title' ] );
|
2015-07-01 16:03:34 +00:00
|
|
|
}
|
2015-07-02 11:55:39 +00:00
|
|
|
|
2016-05-30 22:20:39 +00:00
|
|
|
return '';
|
2015-07-01 16:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-07-28 11:49:17 +00:00
|
|
|
* renders part of infobox
|
|
|
|
* If image element has invalid thumbnail, doesn't render this element at all.
|
2015-07-01 16:03:34 +00:00
|
|
|
*
|
2015-07-28 11:49:17 +00:00
|
|
|
* @param string $type
|
2015-07-01 16:03:34 +00:00
|
|
|
* @param array $data
|
2015-08-27 08:30:25 +00:00
|
|
|
*
|
2015-07-28 11:49:17 +00:00
|
|
|
* @return bool|string - HTML
|
2015-07-01 16:03:34 +00:00
|
|
|
*/
|
2016-03-14 09:36:08 +00:00
|
|
|
private function renderItem( $type, array $data ) {
|
2015-07-28 14:13:36 +00:00
|
|
|
$helper = new PortableInfoboxRenderServiceHelper();
|
2015-07-16 11:21:20 +00:00
|
|
|
|
2015-07-28 11:49:17 +00:00
|
|
|
if ( $type === 'image' ) {
|
2016-04-05 15:42:21 +00:00
|
|
|
$images = [ ];
|
2015-12-23 12:20:55 +00:00
|
|
|
|
2016-03-17 09:37:18 +00:00
|
|
|
for ( $i = 0; $i < count( $data ); $i++ ) {
|
|
|
|
$data[ $i ][ 'context' ] = self::MEDIA_CONTEXT_INFOBOX;
|
|
|
|
$data[ $i ] = $helper->extendImageData( $data[ $i ], $this->imagesWidth );
|
2015-12-23 12:20:55 +00:00
|
|
|
|
2016-03-17 09:37:18 +00:00
|
|
|
if ( !!$data[ $i ] ) {
|
|
|
|
$images[] = $data[ $i ];
|
2015-10-16 01:15:52 +00:00
|
|
|
}
|
|
|
|
}
|
2015-12-23 12:20:55 +00:00
|
|
|
|
2016-03-17 09:37:18 +00:00
|
|
|
if ( count( $images ) === 0 ) {
|
2015-07-28 11:49:17 +00:00
|
|
|
return false;
|
2016-04-01 15:33:29 +00:00
|
|
|
} else if ( count( $images ) === 1 ) {
|
|
|
|
$data = $images[ 0 ];
|
|
|
|
$templateName = $type;
|
2015-10-16 01:15:52 +00:00
|
|
|
} else {
|
2016-04-05 15:42:21 +00:00
|
|
|
// More than one image means image collection
|
2016-04-04 11:13:52 +00:00
|
|
|
if ( $helper->isMobile() && !$helper->isMercury() ) {
|
2016-04-01 15:33:29 +00:00
|
|
|
// Display only the first image on WikiaMobile
|
2016-03-17 09:37:18 +00:00
|
|
|
$data = $images[ 0 ];
|
|
|
|
} else {
|
2016-04-05 15:42:21 +00:00
|
|
|
$data = $helper->extendImageCollectionData( $images );
|
2016-03-17 09:37:18 +00:00
|
|
|
}
|
2016-05-30 19:25:27 +00:00
|
|
|
|
2016-04-01 15:33:29 +00:00
|
|
|
$templateName = 'image-collection';
|
2015-07-28 11:49:17 +00:00
|
|
|
}
|
2015-12-23 12:20:55 +00:00
|
|
|
|
2016-04-04 11:13:52 +00:00
|
|
|
if ( $helper->isMobile() ) {
|
|
|
|
if ( !$helper->isMercury() ) {
|
2016-04-01 15:33:29 +00:00
|
|
|
$templateName = $templateName . self::MOBILE_TEMPLATE_POSTFIX . '-wikiamobile';
|
2016-02-05 17:02:02 +00:00
|
|
|
} else {
|
2016-02-08 14:57:57 +00:00
|
|
|
$templateName = $templateName . self::MOBILE_TEMPLATE_POSTFIX;
|
2016-02-05 17:02:02 +00:00
|
|
|
}
|
2015-07-28 11:49:17 +00:00
|
|
|
}
|
2015-10-16 01:15:52 +00:00
|
|
|
} else {
|
|
|
|
$templateName = $type;
|
2015-07-21 23:13:28 +00:00
|
|
|
}
|
2015-07-20 13:00:35 +00:00
|
|
|
|
2016-01-09 15:01:13 +00:00
|
|
|
/**
|
|
|
|
* Currently, based on business decision, sanitization happens ONLY on Mercury
|
|
|
|
*/
|
2016-04-04 11:13:52 +00:00
|
|
|
if ( $helper->isMobile() ) {
|
2015-12-28 10:59:53 +00:00
|
|
|
$data = SanitizerBuilder::createFromType( $type )->sanitize( $data );
|
2015-07-21 23:13:28 +00:00
|
|
|
}
|
2015-07-20 13:00:35 +00:00
|
|
|
|
2015-07-28 11:49:17 +00:00
|
|
|
return $this->templateEngine->clearData()
|
|
|
|
->setData( $data )
|
2015-10-16 01:15:52 +00:00
|
|
|
->render( self::getTemplates()[ $templateName ] );
|
2015-07-20 13:00:35 +00:00
|
|
|
}
|
2015-04-27 14:05:31 +00:00
|
|
|
}
|