mirror of
https://github.com/Universal-Omega/PortableInfobox.git
synced 2024-11-15 11:59:56 +00:00
merge with dev
This commit is contained in:
parent
8e1834c810
commit
01ccae3e79
|
@ -104,7 +104,7 @@ class PortableInfoboxParserTagController extends WikiaController {
|
|||
$value = isset( $params[ 'theme-source' ] ) ? $frame->getArgument( $params[ 'theme-source' ] ) : false;
|
||||
$themeName = $this->getThemeName( $params, $value );
|
||||
//make sure no whitespaces, prevents side effects
|
||||
return self::INFOBOX_THEME_PREFIX . preg_replace( '|\s+|s', '-', $themeName );
|
||||
return Sanitizer::escapeClass( self::INFOBOX_THEME_PREFIX . preg_replace( '|\s+|s', '-', $themeName ) );
|
||||
}
|
||||
|
||||
private function getThemeName( $params, $value ) {
|
||||
|
|
|
@ -12,6 +12,7 @@ class Node {
|
|||
const LABEL_TAG_NAME = 'label';
|
||||
|
||||
protected $xmlNode;
|
||||
protected $parent = null;
|
||||
|
||||
/* @var $externalParser ExternalParser */
|
||||
protected $externalParser;
|
||||
|
@ -21,6 +22,24 @@ class Node {
|
|||
$this->infoboxData = $infoboxData;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return mixed
|
||||
*/
|
||||
public function getParent() {
|
||||
return $this->parent;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $parent
|
||||
*/
|
||||
public function setParent( Node $parent ) {
|
||||
$this->parent = $parent;
|
||||
}
|
||||
|
||||
public function ignoreNodeWhenEmpty() {
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return ExternalParser
|
||||
*/
|
||||
|
@ -39,7 +58,13 @@ class Node {
|
|||
}
|
||||
|
||||
public function getType() {
|
||||
return $this->xmlNode->getName();
|
||||
/*
|
||||
* Node type generation is based on XML tag name.
|
||||
* It's worth to remember that SimpleXMLElement::getName method is
|
||||
* case - sensitive ( "<Data>" != "<data>" ), so we need to sanitize Node Type
|
||||
* by using strtolower function
|
||||
*/
|
||||
return strtolower( $this->xmlNode->getName() );
|
||||
}
|
||||
|
||||
public function getData() {
|
||||
|
|
|
@ -10,7 +10,7 @@ class NodeComparison extends Node {
|
|||
if ( $this->externalParser ) {
|
||||
$nodeFactory->setExternalParser( $this->externalParser );
|
||||
}
|
||||
return [ 'value' => $nodeFactory->getDataFromNodes( $this->xmlNode ) ];
|
||||
return [ 'value' => $nodeFactory->getDataFromNodes( $this->xmlNode, $this ) ];
|
||||
}
|
||||
|
||||
public function isEmpty( $data ) {
|
||||
|
|
|
@ -3,6 +3,17 @@ namespace Wikia\PortableInfobox\Parser\Nodes;
|
|||
|
||||
class NodeData extends Node {
|
||||
|
||||
public function ignoreNodeWhenEmpty() {
|
||||
$parent = $this->getParent();
|
||||
if ( $parent instanceof NodeSet ) {
|
||||
if ( $parent->getParent() instanceof NodeComparison ) {
|
||||
// data tag inside comparison tag can not be ignored
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public function getData() {
|
||||
return [
|
||||
'label' => $this->getExternalParser()->parseRecursive(
|
||||
|
|
|
@ -10,7 +10,7 @@ class NodeGroup extends Node {
|
|||
if ( $this->externalParser ) {
|
||||
$nodeFactory->setExternalParser( $this->externalParser );
|
||||
}
|
||||
return [ 'value' => $nodeFactory->getDataFromNodes( $this->xmlNode ) ];
|
||||
return [ 'value' => $nodeFactory->getDataFromNodes( $this->xmlNode, $this ) ];
|
||||
}
|
||||
|
||||
public function isEmpty( $data ) {
|
||||
|
|
|
@ -5,18 +5,22 @@ use Wikia\PortableInfobox\Helpers\ImageFilenameSanitizer;
|
|||
|
||||
class NodeImage extends Node {
|
||||
const ALT_TAG_NAME = 'alt';
|
||||
const CAPTION_TAG_NAME = 'caption';
|
||||
|
||||
public function getData() {
|
||||
$title = $this->getImageAsTitleObject( $this->getRawValueWithDefault( $this->xmlNode ) );
|
||||
$ref = null;
|
||||
$alt = $this->getValueWithDefault( $this->xmlNode->{self::ALT_TAG_NAME} );
|
||||
$caption = $this->getValueWithDefault( $this->xmlNode->{self::CAPTION_TAG_NAME} );
|
||||
|
||||
wfRunHooks( 'PortableInfoboxNodeImage::getData', [ $title, &$ref, $alt ] );
|
||||
|
||||
return [
|
||||
'url' => $this->resolveImageUrl( $title ),
|
||||
'name' => ( $title ) ? $title->getDBkey() : '',
|
||||
'name' => ( $title ) ? $title->getText() : '',
|
||||
'key' => ( $title ) ? $title->getDBKey() : '',
|
||||
'alt' => $alt,
|
||||
'caption' => $caption,
|
||||
'ref' => $ref
|
||||
];
|
||||
}
|
||||
|
|
|
@ -10,7 +10,7 @@ class NodeSet extends Node {
|
|||
if ( $this->externalParser ) {
|
||||
$nodeFactory->setExternalParser( $this->externalParser );
|
||||
}
|
||||
return [ 'value' => $nodeFactory->getDataFromNodes( $this->xmlNode ) ];
|
||||
return [ 'value' => $nodeFactory->getDataFromNodes( $this->xmlNode, $this ) ];
|
||||
}
|
||||
|
||||
public function isEmpty( $data ) {
|
||||
|
|
|
@ -28,16 +28,18 @@ class XmlParser {
|
|||
* @param \SimpleXMLElement $xmlIterable
|
||||
* @return array
|
||||
*/
|
||||
public function getDataFromNodes( \SimpleXMLElement $xmlIterable ) {
|
||||
public function getDataFromNodes( \SimpleXMLElement $xmlIterable, $parentNode = null ) {
|
||||
wfProfileIn(__METHOD__);
|
||||
$data = [ ];
|
||||
foreach ( $xmlIterable as $node ) {
|
||||
$nodeHandler = $this->getNode( $node );
|
||||
$nodeHandler = $this->getNode( $node, $parentNode );
|
||||
$nodeData = $nodeHandler->getData();
|
||||
if ( !$nodeHandler->isEmpty( $nodeData ) ) {
|
||||
// add data if node is not empty or - when node can not be ignored when empty
|
||||
if ( !$nodeHandler->isEmpty( $nodeData ) || !$nodeHandler->ignoreNodeWhenEmpty() ) {
|
||||
$data[ ] = [
|
||||
'type' => $nodeHandler->getType(),
|
||||
'data' => $nodeData
|
||||
'data' => $nodeData,
|
||||
'isEmpty' => $nodeHandler->isEmpty( $nodeData )
|
||||
];
|
||||
}
|
||||
}
|
||||
|
@ -81,9 +83,10 @@ class XmlParser {
|
|||
|
||||
/**
|
||||
* @param \SimpleXMLElement $xmlNode
|
||||
* @param Node $parent
|
||||
* @return \Wikia\PortableInfobox\Parser\Nodes\Node
|
||||
*/
|
||||
public function getNode( \SimpleXMLElement $xmlNode ) {
|
||||
public function getNode( \SimpleXMLElement $xmlNode, $parent = null ) {
|
||||
wfProfileIn(__METHOD__);
|
||||
$tagType = $xmlNode->getName();
|
||||
$className = 'Wikia\\PortableInfobox\\Parser\\Nodes\\' . 'Node' . ucfirst( strtolower( $tagType ) );
|
||||
|
@ -93,6 +96,9 @@ class XmlParser {
|
|||
if ( !empty( $this->externalParser ) ) {
|
||||
$instance->setExternalParser( $this->externalParser );
|
||||
}
|
||||
if ( $parent ) {
|
||||
$instance->setParent( $parent );
|
||||
}
|
||||
wfProfileOut( __METHOD__ );
|
||||
return $instance;
|
||||
}
|
||||
|
|
|
@ -150,8 +150,9 @@ class PortableInfoboxRenderService extends WikiaService {
|
|||
// 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 );
|
||||
$data[ 'key' ] = urlencode( $data[ 'key' ] );
|
||||
|
||||
if ( F::app()->checkSkin( 'wikiamobile' ) ) {
|
||||
if ( $this->isWikiaMobile() ) {
|
||||
$type = $type . self::MOBILE_TEMPLATE_POSTFIX;
|
||||
}
|
||||
}
|
||||
|
@ -185,7 +186,7 @@ class PortableInfoboxRenderService extends WikiaService {
|
|||
private function createVignetteThumbnail( $url ) {
|
||||
return VignetteRequest::fromUrl( $url )
|
||||
->thumbnailDown()
|
||||
->width( F::app()->checkSkin( 'wikiamobile' ) ?
|
||||
->width( $this->isWikiaMobile() ?
|
||||
self::MOBILE_THUMBNAIL_WIDTH :
|
||||
self::DESKTOP_THUMBNAIL_WIDTH
|
||||
)
|
||||
|
@ -211,6 +212,12 @@ class PortableInfoboxRenderService extends WikiaService {
|
|||
);
|
||||
}
|
||||
return '';
|
||||
/**
|
||||
* required for testing mobile template rendering
|
||||
* @return bool
|
||||
*/
|
||||
protected function isWikiaMobile() {
|
||||
return F::app()->checkSkin( 'wikiamobile' );
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -3,5 +3,6 @@
|
|||
<a href="{{url}}" class="image image-thumbnail" title="{{alt}}">
|
||||
<img src="{{thumbnail}}" class="portable-infobox-image" alt="{{alt}}" data-image-key="{{name}}" data-image-name="{{name}}"/>
|
||||
</a>
|
||||
{{#caption}}<figcaption class="portable-infobox-item-margins portable-infobox-image-caption">{{{caption}}}</figcaption>{{/caption}}
|
||||
</figure>
|
||||
</div>
|
|
@ -1 +1 @@
|
|||
<aside class="portable-infobox{{#theme}} {{{theme}}}{{/theme}}">{{{content}}}</aside>
|
||||
<aside class="portable-infobox{{#theme}} {{theme}}{{/theme}}">{{{content}}}</aside>
|
||||
|
|
|
@ -28,17 +28,27 @@ class PortableInfoboxParserNodesTest extends WikiaBaseTest {
|
|||
}
|
||||
|
||||
public function testNodeImage() {
|
||||
$string = '<image source="image2"><alt source="alt-source"><default>default-alt</default></alt></image>';
|
||||
$string = '<image source="image2">
|
||||
<alt source="alt-source"><default>default-alt</default></alt>
|
||||
<caption source="caption"><default>default caption</default></caption>
|
||||
</image>';
|
||||
$xml = simplexml_load_string( $string );
|
||||
|
||||
$nodeDefault = new Wikia\PortableInfobox\Parser\Nodes\NodeImage( $xml, [ ] );
|
||||
|
||||
$node = $this->getMockBuilder( 'Wikia\PortableInfobox\Parser\Nodes\NodeImage' )->setConstructorArgs( [ $xml, [ 'image2' => 'aaa.jpg', 'alt-source' => 'bbb' ] ] )->setMethods( [ 'resolveImageUrl' ] )->getMock();
|
||||
$node = $this->getMockBuilder( 'Wikia\PortableInfobox\Parser\Nodes\NodeImage' )
|
||||
->setConstructorArgs( [ $xml, [ 'image2' => 'aaa.jpg',
|
||||
'alt-source' => 'bbb',
|
||||
'caption' => 'capt' ] ] )
|
||||
->setMethods( [ 'resolveImageUrl' ] )
|
||||
->getMock();
|
||||
$node->expects( $this->any() )->method( 'resolveImageUrl' )->will( $this->returnValue( 'aaa.jpg' ) );
|
||||
$this->assertTrue( $node->getData()[ 'url' ] == 'aaa.jpg', 'value is not aaa.jpg' );
|
||||
$this->assertTrue( $node->getData()[ 'name' ] == 'Aaa.jpg', 'value is not aaa.jpg' );
|
||||
$this->assertTrue( $node->getData()[ 'alt' ] == 'bbb', 'alt is not bbb' );
|
||||
$this->assertTrue( $node->getData()[ 'caption' ] == 'capt', 'caption is not "capt"' );
|
||||
$this->assertTrue( $nodeDefault->getData()[ 'alt' ] == 'default-alt', 'default alt' );
|
||||
$this->assertTrue( $nodeDefault->getData()[ 'caption' ] == 'default caption', 'default caption' );
|
||||
}
|
||||
|
||||
public function testNodeHeader() {
|
||||
|
|
|
@ -13,6 +13,11 @@ class PortableInfoboxRenderServiceTest extends PHPUnit_Framework_TestCase {
|
|||
$this->infoboxRenderService = $mock;
|
||||
}
|
||||
|
||||
private function setWikiaMobileSkin($bool) {
|
||||
$this->infoboxRenderService->expects( $this->any() )->method( 'isWikiaMobile' )->will( $this->returnValue
|
||||
($bool) );
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $input
|
||||
* @param $expectedOutput
|
||||
|
@ -20,6 +25,7 @@ class PortableInfoboxRenderServiceTest extends PHPUnit_Framework_TestCase {
|
|||
* @dataProvider testRenderInfoboxDataProvider
|
||||
*/
|
||||
public function testRenderInfobox( $input, $expectedOutput, $description ) {
|
||||
$this->setWikiaMobileSkin(false);
|
||||
|
||||
$actualOutput = $this->infoboxRenderService->renderInfobox( $input );
|
||||
|
||||
|
@ -38,6 +44,31 @@ class PortableInfoboxRenderServiceTest extends PHPUnit_Framework_TestCase {
|
|||
$this->assertEquals( $expectedHtml, $actualHtml, $description );
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $input
|
||||
* @param $expectedOutput
|
||||
* @param $description
|
||||
* @dataProvider testRenderInfoboxDataProvider
|
||||
*/
|
||||
public function testRenderMobileInfobox( $input, $expectedOutput, $description ) {
|
||||
$this->setWikiaMobileSkin(true);
|
||||
$actualOutput = $this->infoboxRenderService->renderInfobox( $input );
|
||||
|
||||
$actualDOM = new DOMDocument('1.0');
|
||||
$expectedDOM = new DOMDocument('1.0');
|
||||
$actualDOM->formatOutput = true;
|
||||
$actualDOM->preserveWhiteSpace = false;
|
||||
$expectedDOM->formatOutput = true;
|
||||
$expectedDOM->preserveWhiteSpace = false;
|
||||
$actualDOM->loadXML($actualOutput);
|
||||
$expectedDOM->loadXML($expectedOutput);
|
||||
|
||||
$expectedHtml = $expectedDOM->saveXML();
|
||||
$actualHtml = $actualDOM->saveXML();
|
||||
|
||||
$this->assertEquals( $expectedHtml, $actualHtml, $description );
|
||||
}
|
||||
|
||||
public function testRenderInfoboxDataProvider() {
|
||||
return [
|
||||
[
|
||||
|
@ -61,6 +92,29 @@ class PortableInfoboxRenderServiceTest extends PHPUnit_Framework_TestCase {
|
|||
</aside>',
|
||||
'description' => 'Only title'
|
||||
],
|
||||
[
|
||||
'input' => [
|
||||
[
|
||||
'type' => 'image',
|
||||
'data' => [
|
||||
'alt' => 'image alt',
|
||||
'url' => 'http://image.jpg',
|
||||
'caption' => 'Lorem ipsum dolor'
|
||||
]
|
||||
]
|
||||
],
|
||||
'output' => '<aside class="portable-infobox">
|
||||
<div class="portable-infobox-item item-type-image no-margins">
|
||||
<figure class="portable-infobox-image-wrapper">
|
||||
<a href="http://image.jpg" class="image image-thumbnail" title="image alt">
|
||||
<img src="http://image.jpg" class="portable-infobox-image" alt="image alt" data-image-key="" data-image-name=""/>
|
||||
</a>
|
||||
<figcaption class="portable-infobox-item-margins portable-infobox-image-caption">Lorem ipsum dolor</figcaption>
|
||||
</figure>
|
||||
</div>
|
||||
</aside>',
|
||||
'description' => 'Only image'
|
||||
],
|
||||
[
|
||||
'input' => [
|
||||
[
|
||||
|
@ -75,27 +129,6 @@ class PortableInfoboxRenderServiceTest extends PHPUnit_Framework_TestCase {
|
|||
</aside>',
|
||||
'description' => 'Footer only'
|
||||
],
|
||||
[
|
||||
'input' => [
|
||||
[
|
||||
'type' => 'image',
|
||||
'data' => [
|
||||
'alt' => 'image alt',
|
||||
'url' => 'http://image.jpg'
|
||||
]
|
||||
]
|
||||
],
|
||||
'output' => '<aside class="portable-infobox">
|
||||
<div class="portable-infobox-item item-type-image no-margins">
|
||||
<figure class="portable-infobox-image-wrapper">
|
||||
<a href="http://image.jpg" class="image image-thumbnail" title="image alt">
|
||||
<img src="http://image.jpg" class="portable-infobox-image" alt="image alt" data-image-key="" data-image-name=""/>
|
||||
</a>
|
||||
</figure>
|
||||
</div>
|
||||
</aside>',
|
||||
'description' => 'Only image'
|
||||
],
|
||||
[
|
||||
'input' => [
|
||||
[
|
||||
|
@ -322,4 +355,29 @@ class PortableInfoboxRenderServiceTest extends PHPUnit_Framework_TestCase {
|
|||
]
|
||||
];
|
||||
}
|
||||
|
||||
public function testRenderMobileInfoboxDataProvider() {
|
||||
return [
|
||||
[
|
||||
'input' => [
|
||||
[
|
||||
'type' => 'image',
|
||||
'data' => [
|
||||
'alt' => 'image alt',
|
||||
'url' => 'http://image.jpg',
|
||||
'thumbnail' => 'thumbnail.jpg',
|
||||
'ref' => 1,
|
||||
'name' => 'test1'
|
||||
]
|
||||
]
|
||||
],
|
||||
'output' => '<aside class="portable-infobox">
|
||||
<div class="portable-infobox-item item-type-image no-margins">
|
||||
<img src="%3D%3D" class="portable-infobox-image lazy media article-media" alt="image alt" data-image-key="test1" data-image-name="test1" data-ref="1" data-src="thumbnail.jpg" data-params=\'[{"name":"test1", "full":"http://image.jpg"}]\' />
|
||||
</div>
|
||||
</aside>',
|
||||
'description' => 'Only image for mobile'
|
||||
]
|
||||
];
|
||||
}
|
||||
}
|
||||
|
|
|
@ -59,6 +59,34 @@ class XmlParserTest extends WikiaBaseTest {
|
|||
$this->assertTrue( $data[1]['data']['value'][0]['data']['value'][2]['data']['value'] == 'parseRecursive(LALALA)' );
|
||||
}
|
||||
|
||||
public function testNotProvidingDataSource() {
|
||||
$xmlParser = new \Wikia\PortableInfobox\Parser\XmlParser( ['a'=>'1'] );
|
||||
|
||||
$data1 = $xmlParser->getDataFromXmlString('<infobox><data source="b" /><data source="a" /></infobox>');
|
||||
$this->assertTrue( $data1[0]['data']['value'] == '1', 'Data with empty source "b" should be ignored' );
|
||||
}
|
||||
|
||||
public function testNotProvidingDataSourceInsideComparison() {
|
||||
$xmlParser = new \Wikia\PortableInfobox\Parser\XmlParser( ['a'=>'1'] );
|
||||
|
||||
$data1 = $xmlParser->getDataFromXmlString( '<infobox>
|
||||
<comparison>
|
||||
<set>
|
||||
<data source="b" />
|
||||
<data source="a" />
|
||||
</set>
|
||||
</comparison>
|
||||
</infobox>' );
|
||||
|
||||
$setElement = $data1[0]['data']['value'][0]['data']['value'];
|
||||
|
||||
//Data with empty source "b" should not be ignored inside comparison:
|
||||
$this->assertTrue( $setElement[0]['data']['value'] == '', 'Data with empty source "b" inside comparison' );
|
||||
$this->assertTrue( $setElement[0]['isEmpty'] == true );
|
||||
|
||||
$this->assertTrue( $setElement[1]['data']['value'] == '1', '' );
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider errorHandlingDataProvider
|
||||
*/
|
||||
|
|
Loading…
Reference in a new issue