PortableInfobox/services/PortableInfoboxRenderService.class.php
2015-05-22 16:32:02 +02:00

204 lines
5.5 KiB
PHP

<?php
class PortableInfoboxRenderService extends WikiaService {
const LOGGER_LABEL = 'portable-infobox-render-not-supported-type';
const DESKTOP_THUMBNAIL_WIDTH = 270;
const MOBILE_THUMBNAIL_WIDTH = 360;
// TODO: https://wikia-inc.atlassian.net/browse/MAIN-4601 - request for the missing vignette feature which will
// allow us to remove THUMBNAIL_HEIGHT from the code. Currently we need this value cause it it impossible to get
// vignette thumbnail without upsampling only specifying width. The height need to be big enough so each image width
// will reach our thumbnail width based on its aspect ratio
const THUMBNAIL_HEIGHT = 1000;
const MOBILE_TEMPLATE_POSTFIX = '-mobile';
private $templates = [
'wrapper' => 'PortableInfoboxWrapper.mustache',
'title' => 'PortableInfoboxItemTitle.mustache',
'header' => 'PortableInfoboxItemHeader.mustache',
'image' => 'PortableInfoboxItemImage.mustache',
'image-mobile' => 'PortableInfoboxItemImageMobile.mustache',
'data' => 'PortableInfoboxItemData.mustache',
'group' => 'PortableInfoboxItemGroup.mustache',
'comparison' => 'PortableInfoboxItemComparison.mustache',
'comparison-set' => 'PortableInfoboxItemComparisonSet.mustache',
'comparison-set-header' => 'PortableInfoboxItemComparisonSetHeader.mustache',
'comparison-set-item' => 'PortableInfoboxItemComparisonSetItem.mustache',
'footer' => 'PortableInfoboxItemFooter.mustache'
];
private $templateEngine;
function __construct() {
$this->templateEngine = ( new Wikia\Template\MustacheEngine )
->setPrefix( dirname( __FILE__ ) . '/../templates' );
}
/**
* renders infobox
*
* @param array $infoboxdata
* @return string - infobox HTML
*/
public function renderInfobox( array $infoboxdata, $theme ) {
wfProfileIn( __METHOD__ );
$infoboxHtmlContent = '';
foreach ( $infoboxdata as $item ) {
$data = $item[ 'data' ];
$type = $item[ 'type' ];
switch ( $type ) {
case 'comparison':
$infoboxHtmlContent .= $this->renderComparisonItem( $data['value'] );
break;
case 'group':
$infoboxHtmlContent .= $this->renderGroup( $data['value'] );
break;
case 'footer':
$infoboxHtmlContent .= $this->renderItem( 'footer', $data );
break;
default:
if ( $this->validateType( $type ) ) {
$infoboxHtmlContent .= $this->renderItem( $type, $data );
};
}
}
if(!empty($infoboxHtmlContent)) {
$output = $this->renderItem( 'wrapper', [ 'content' => $infoboxHtmlContent, 'theme' => $theme ] );
} else {
$output = '';
}
wfProfileOut( __METHOD__ );
return $output;
}
/**
* renders comparison infobox component
*
* @param array $comparisonData
* @return string - comparison HTML
*/
private function renderComparisonItem( $comparisonData )
{
$comparisonHTMLContent = '';
foreach ($comparisonData as $set) {
$setHTMLContent = '';
foreach ($set['data']['value'] as $item) {
$type = $item['type'];
if ($type === 'header') {
$setHTMLContent .= $this->renderItem(
'comparison-set-header',
['content' => $this->renderItem($type, $item['data'])]
);
} else {
if ($this->validateType($type)) {
$setHTMLContent .= $this->renderItem(
'comparison-set-item',
['content' => $this->renderItem($type, $item['data'])]
);
}
}
}
$comparisonHTMLContent .= $this->renderItem( 'comparison-set', [ 'content' => $setHTMLContent ] );
}
if ( !empty( $comparisonHTMLContent ) ) {
$output = $this->renderItem('comparison', [ 'content' => $comparisonHTMLContent ] );
} else {
$output = '';
}
return $output;
}
/**
* renders group infobox component
*
* @param array $groupData
* @return string - group HTML markup
*/
private function renderGroup( $groupData ) {
$groupHTMLContent = '';
foreach ( $groupData as $item ) {
$type = $item['type'];
if ( $this->validateType( $type ) ) {
$groupHTMLContent .= $this->renderItem( $type, $item['data'] );
}
}
return $this->renderItem( 'group', [ 'content' => $groupHTMLContent ] );
}
/**
* renders part of infobox
*
* @param string $type
* @param array $data
* @return string - HTML
*/
private function renderItem( $type, array $data ) {
//TODO: with validated the performance of render Service and in the next phase we want to refactor it (make
// it modular) While doing this we also need to move this logic to appropriate image render class
if ( $type === 'image' ) {
$data[ 'thumbnail' ] = $this->getThumbnailUrl( $data['url'] );
$data[ 'key' ] = urlencode( $data[ 'key' ] );
if ( $this->isWikiaMobile() ) {
$type = $type . self::MOBILE_TEMPLATE_POSTFIX;
}
}
return $this->templateEngine->clearData()
->setData( $data )
->render( $this->templates[ $type ] );
}
protected function getThumbnailUrl( $url ) {
return VignetteRequest::fromUrl( $url )
->thumbnailDown()
->width( $this->isWikiaMobile() ?
self::MOBILE_THUMBNAIL_WIDTH :
self::DESKTOP_THUMBNAIL_WIDTH
)
->height(self::THUMBNAIL_HEIGHT)
->url();
}
/**
* required for testing mobile template rendering
* @return bool
*/
protected function isWikiaMobile() {
return F::app()->checkSkin( 'wikiamobile' );
}
/**
* check if item type is supported and logs unsupported types
*
* @param string $type - template type
* @return bool
*/
private function validateType( $type ) {
$isValid = true;
if ( !isset( $this->templates[ $type ] ) ) {
Wikia\Logger\WikiaLogger::instance()->info( self::LOGGER_LABEL, [
'type' => $type
] );
$isValid = false;
}
return $isValid;
}
}