2011-07-04 19:51:35 +00:00
|
|
|
<?php
|
|
|
|
|
/** tests for includes/Html.php */
|
|
|
|
|
|
|
|
|
|
class HtmlTest extends MediaWikiTestCase {
|
|
|
|
|
private static $oldLang;
|
2011-08-05 15:10:08 +00:00
|
|
|
private static $oldContLang;
|
2012-01-25 03:45:01 +00:00
|
|
|
private static $oldLanguageCode;
|
2012-01-25 03:25:54 +00:00
|
|
|
private static $oldNamespaces;
|
2011-07-04 19:51:35 +00:00
|
|
|
|
|
|
|
|
public function setUp() {
|
2011-08-05 15:10:08 +00:00
|
|
|
global $wgLang, $wgContLang, $wgLanguageCode;
|
2011-07-04 19:51:35 +00:00
|
|
|
|
|
|
|
|
self::$oldLang = $wgLang;
|
2011-08-05 15:10:08 +00:00
|
|
|
self::$oldContLang = $wgContLang;
|
2012-01-27 13:00:26 +00:00
|
|
|
self::$oldNamespaces = $wgContLang->getNamespaces();
|
2012-01-25 03:45:01 +00:00
|
|
|
self::$oldLanguageCode = $wgLanguageCode;
|
2011-08-05 15:10:08 +00:00
|
|
|
|
2011-07-04 19:51:35 +00:00
|
|
|
$wgLanguageCode = 'en';
|
2011-08-05 15:10:08 +00:00
|
|
|
$wgContLang = $wgLang = Language::factory( $wgLanguageCode );
|
2012-01-25 03:25:54 +00:00
|
|
|
|
|
|
|
|
// Hardcode namespaces during test runs,
|
|
|
|
|
// so that html output based on existing namespaces
|
|
|
|
|
// can be properly evaluated.
|
2012-01-27 13:00:26 +00:00
|
|
|
$wgContLang->setNamespaces( array(
|
2012-01-25 03:25:54 +00:00
|
|
|
-2 => 'Media',
|
|
|
|
|
-1 => 'Special',
|
|
|
|
|
0 => '',
|
|
|
|
|
1 => 'Talk',
|
|
|
|
|
2 => 'User',
|
|
|
|
|
3 => 'User_talk',
|
|
|
|
|
4 => 'MyWiki',
|
|
|
|
|
5 => 'MyWiki_Talk',
|
|
|
|
|
6 => 'File',
|
|
|
|
|
7 => 'File_talk',
|
|
|
|
|
8 => 'MediaWiki',
|
|
|
|
|
9 => 'MediaWiki_talk',
|
|
|
|
|
10 => 'Template',
|
|
|
|
|
11 => 'Template_talk',
|
2012-02-13 14:53:40 +00:00
|
|
|
14 => 'Category',
|
|
|
|
|
15 => 'Category_talk',
|
2012-01-25 03:25:54 +00:00
|
|
|
100 => 'Custom',
|
|
|
|
|
101 => 'Custom_talk',
|
2012-01-27 13:00:26 +00:00
|
|
|
) );
|
2011-07-04 19:51:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function tearDown() {
|
2011-08-05 15:10:08 +00:00
|
|
|
global $wgLang, $wgContLang, $wgLanguageCode;
|
2012-01-25 03:45:01 +00:00
|
|
|
|
2012-01-27 13:00:26 +00:00
|
|
|
$wgContLang->setNamespaces( self::$oldNamespaces );
|
2011-07-04 19:51:35 +00:00
|
|
|
$wgLang = self::$oldLang;
|
2011-08-05 15:10:08 +00:00
|
|
|
$wgContLang = self::$oldContLang;
|
2012-01-25 03:45:01 +00:00
|
|
|
$wgLanguageCode = self::$oldLanguageCode;
|
2011-07-04 19:51:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testExpandAttributesSkipsNullAndFalse() {
|
|
|
|
|
|
|
|
|
|
### EMPTY ########
|
|
|
|
|
$this->AssertEmpty(
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'foo' => null ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'skip keys with null value'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEmpty(
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'foo' => false ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'skip keys with false value'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertNotEmpty(
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'foo' => '' ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'keep keys with an empty string'
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testExpandAttributesForBooleans() {
|
2011-07-19 22:10:05 +00:00
|
|
|
global $wgHtml5;
|
2011-07-04 19:51:35 +00:00
|
|
|
$this->AssertEquals(
|
|
|
|
|
'',
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'selected' => false ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'Boolean attributes do not generates output when value is false'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
'',
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'selected' => null ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'Boolean attributes do not generates output when value is null'
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$this->AssertEquals(
|
2011-07-19 22:10:05 +00:00
|
|
|
$wgHtml5 ? ' selected=""' : ' selected="selected"',
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'selected' => true ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'Boolean attributes skip value output'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
2011-07-19 22:10:05 +00:00
|
|
|
$wgHtml5 ? ' selected=""' : ' selected="selected"',
|
2011-07-04 19:51:35 +00:00
|
|
|
Html::expandAttributes( array( 'selected' ) ),
|
|
|
|
|
'Boolean attributes (ex: selected) do not need a value'
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Test for Html::expandAttributes()
|
|
|
|
|
* Please note it output a string prefixed with a space!
|
|
|
|
|
*/
|
|
|
|
|
public function testExpandAttributesVariousExpansions() {
|
|
|
|
|
### NOT EMPTY ####
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' empty_string=""',
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'empty_string' => '' ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'Value with an empty string'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' key="value"',
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'key' => 'value' ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'Value is a string'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' one="1"',
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'one' => 1 ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'Value is a numeric one'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' zero="0"',
|
2011-09-03 01:42:43 +00:00
|
|
|
Html::expandAttributes( array( 'zero' => 0 ) ),
|
2011-07-04 19:51:35 +00:00
|
|
|
'Value is a numeric zero'
|
|
|
|
|
);
|
|
|
|
|
}
|
2011-09-03 03:55:23 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Html::expandAttributes has special features for HTML
|
|
|
|
|
* attributes that use space separated lists and also
|
|
|
|
|
* allows arrays to be used as values.
|
|
|
|
|
*/
|
|
|
|
|
public function testExpandAttributesListValueAttributes() {
|
|
|
|
|
### STRING VALUES
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' class="redundant spaces here"',
|
|
|
|
|
Html::expandAttributes( array( 'class' => ' redundant spaces here ' ) ),
|
|
|
|
|
'Normalization should strip redundant spaces'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' class="foo bar"',
|
|
|
|
|
Html::expandAttributes( array( 'class' => 'foo bar foo bar bar' ) ),
|
|
|
|
|
'Normalization should remove duplicates in string-lists'
|
|
|
|
|
);
|
|
|
|
|
### "EMPTY" ARRAY VALUES
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' class=""',
|
|
|
|
|
Html::expandAttributes( array( 'class' => array() ) ),
|
|
|
|
|
'Value with an empty array'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' class=""',
|
|
|
|
|
Html::expandAttributes( array( 'class' => array( null, '', ' ', ' ' ) ) ),
|
|
|
|
|
'Array with null, empty string and spaces'
|
|
|
|
|
);
|
|
|
|
|
### NON-EMPTY ARRAY VALUES
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' class="foo bar"',
|
|
|
|
|
Html::expandAttributes( array( 'class' => array(
|
|
|
|
|
'foo',
|
|
|
|
|
'bar',
|
|
|
|
|
'foo',
|
|
|
|
|
'bar',
|
|
|
|
|
'bar',
|
|
|
|
|
) ) ),
|
|
|
|
|
'Normalization should remove duplicates in the array'
|
|
|
|
|
);
|
|
|
|
|
$this->AssertEquals(
|
|
|
|
|
' class="foo bar"',
|
|
|
|
|
Html::expandAttributes( array( 'class' => array(
|
|
|
|
|
'foo bar',
|
|
|
|
|
'bar foo',
|
|
|
|
|
'foo',
|
|
|
|
|
'bar bar',
|
|
|
|
|
) ) ),
|
|
|
|
|
'Normalization should remove duplicates in string-lists in the array'
|
|
|
|
|
);
|
|
|
|
|
}
|
2011-10-24 17:45:45 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Test feature added by r96188, let pass attributes values as
|
|
|
|
|
* a PHP array. Restricted to class,rel, accesskey.
|
|
|
|
|
*/
|
|
|
|
|
function testExpandAttributesSpaceSeparatedAttributesWithBoolean() {
|
|
|
|
|
$this->assertEquals(
|
|
|
|
|
' class="booltrue one"',
|
|
|
|
|
Html::expandAttributes( array( 'class' => array(
|
|
|
|
|
'booltrue' => true,
|
|
|
|
|
'one' => 1,
|
|
|
|
|
|
|
|
|
|
# Method use isset() internally, make sure we do discard
|
|
|
|
|
# attributes values which have been assigned well known values
|
|
|
|
|
'emptystring' => '',
|
|
|
|
|
'boolfalse' => false,
|
|
|
|
|
'zero' => 0,
|
|
|
|
|
'null' => null,
|
|
|
|
|
)))
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* How do we handle duplicate keys in HTML attributes expansion?
|
|
|
|
|
* We could pass a "class" the values: 'GREEN' and array( 'GREEN' => false )
|
|
|
|
|
* The later will take precedence.
|
|
|
|
|
*
|
|
|
|
|
* Feature added by r96188
|
|
|
|
|
*/
|
|
|
|
|
function testValueIsAuthoritativeInSpaceSeparatedAttributesArrays() {
|
|
|
|
|
$this->assertEquals(
|
|
|
|
|
' class=""',
|
2011-10-27 15:54:49 +00:00
|
|
|
Html::expandAttributes( array( 'class' => array(
|
2011-10-24 17:45:45 +00:00
|
|
|
'GREEN',
|
|
|
|
|
'GREEN' => false,
|
|
|
|
|
'GREEN',
|
|
|
|
|
)))
|
|
|
|
|
);
|
|
|
|
|
}
|
2012-01-25 03:25:54 +00:00
|
|
|
|
|
|
|
|
function testNamespaceSelector() {
|
2012-02-13 15:08:26 +00:00
|
|
|
global $wgContLang;
|
|
|
|
|
|
2012-01-25 03:25:54 +00:00
|
|
|
$this->assertEquals(
|
2012-02-02 18:42:02 +00:00
|
|
|
'<select id="namespace" name="namespace">' . "\n" .
|
|
|
|
|
'<option value="0">(Main)</option>' . "\n" .
|
|
|
|
|
'<option value="1">Talk</option>' . "\n" .
|
|
|
|
|
'<option value="2">User</option>' . "\n" .
|
|
|
|
|
'<option value="3">User talk</option>' . "\n" .
|
|
|
|
|
'<option value="4">MyWiki</option>' . "\n" .
|
|
|
|
|
'<option value="5">MyWiki Talk</option>' . "\n" .
|
|
|
|
|
'<option value="6">File</option>' . "\n" .
|
|
|
|
|
'<option value="7">File talk</option>' . "\n" .
|
|
|
|
|
'<option value="8">MediaWiki</option>' . "\n" .
|
|
|
|
|
'<option value="9">MediaWiki talk</option>' . "\n" .
|
|
|
|
|
'<option value="10">Template</option>' . "\n" .
|
|
|
|
|
'<option value="11">Template talk</option>' . "\n" .
|
2012-02-13 14:53:40 +00:00
|
|
|
'<option value="14">Category</option>' . "\n" .
|
|
|
|
|
'<option value="15">Category talk</option>' . "\n" .
|
2012-02-02 18:42:02 +00:00
|
|
|
'<option value="100">Custom</option>' . "\n" .
|
|
|
|
|
'<option value="101">Custom talk</option>' . "\n" .
|
|
|
|
|
'</select>',
|
2012-01-25 03:25:54 +00:00
|
|
|
Html::namespaceSelector(),
|
|
|
|
|
'Basic namespace selector without custom options'
|
|
|
|
|
);
|
|
|
|
|
$this->assertEquals(
|
2012-02-02 18:42:02 +00:00
|
|
|
'<label for="mw-test-namespace">Select a namespace:</label> ' .
|
|
|
|
|
'<select id="mw-test-namespace" name="wpNamespace">' . "\n" .
|
|
|
|
|
'<option value="all">all</option>' . "\n" .
|
|
|
|
|
'<option value="0">(Main)</option>' . "\n" .
|
|
|
|
|
'<option value="1">Talk</option>' . "\n" .
|
|
|
|
|
'<option value="2" selected="">User</option>' . "\n" .
|
|
|
|
|
'<option value="3">User talk</option>' . "\n" .
|
|
|
|
|
'<option value="4">MyWiki</option>' . "\n" .
|
|
|
|
|
'<option value="5">MyWiki Talk</option>' . "\n" .
|
|
|
|
|
'<option value="6">File</option>' . "\n" .
|
|
|
|
|
'<option value="7">File talk</option>' . "\n" .
|
|
|
|
|
'<option value="8">MediaWiki</option>' . "\n" .
|
|
|
|
|
'<option value="9">MediaWiki talk</option>' . "\n" .
|
|
|
|
|
'<option value="10">Template</option>' . "\n" .
|
|
|
|
|
'<option value="11">Template talk</option>' . "\n" .
|
2012-02-13 14:53:40 +00:00
|
|
|
'<option value="14">Category</option>' . "\n" .
|
|
|
|
|
'<option value="15">Category talk</option>' . "\n" .
|
2012-02-02 18:42:02 +00:00
|
|
|
'<option value="100">Custom</option>' . "\n" .
|
|
|
|
|
'<option value="101">Custom talk</option>' . "\n" .
|
|
|
|
|
'</select>',
|
2012-01-25 03:25:54 +00:00
|
|
|
Html::namespaceSelector(
|
|
|
|
|
array( 'selected' => '2', 'all' => 'all', 'label' => 'Select a namespace:' ),
|
|
|
|
|
array( 'name' => 'wpNamespace', 'id' => 'mw-test-namespace' )
|
|
|
|
|
),
|
|
|
|
|
'Basic namespace selector with custom values'
|
|
|
|
|
);
|
2012-02-13 15:08:26 +00:00
|
|
|
$immovable = array();
|
|
|
|
|
$namespaces = $wgContLang->getNamespaces();
|
|
|
|
|
foreach ( $namespaces as $nsId => $nsName ) {
|
|
|
|
|
if ( !MWNamespace::isMovable( intval( $nsId ) ) ) {
|
|
|
|
|
$immovable[] = $nsId;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
$this->assertEquals(
|
|
|
|
|
'<select id="namespace" name="namespace">' . "\n" .
|
|
|
|
|
'<option value="0">(Main)</option>' . "\n" .
|
|
|
|
|
'<option value="1">Talk</option>' . "\n" .
|
|
|
|
|
'<option value="2">User</option>' . "\n" .
|
|
|
|
|
'<option value="3">User talk</option>' . "\n" .
|
|
|
|
|
'<option value="4">MyWiki</option>' . "\n" .
|
|
|
|
|
'<option value="5">MyWiki Talk</option>' . "\n" .
|
|
|
|
|
'<option value="6">File</option>' . "\n" .
|
|
|
|
|
'<option value="7">File talk</option>' . "\n" .
|
|
|
|
|
'<option value="8">MediaWiki</option>' . "\n" .
|
|
|
|
|
'<option value="9">MediaWiki talk</option>' . "\n" .
|
|
|
|
|
'<option value="10">Template</option>' . "\n" .
|
|
|
|
|
'<option value="11">Template talk</option>' . "\n" .
|
|
|
|
|
'<option disabled="" value="14">Category</option>' . "\n" .
|
|
|
|
|
'<option value="15">Category talk</option>' . "\n" .
|
|
|
|
|
'</select>',
|
|
|
|
|
Html::namespaceSelector(
|
|
|
|
|
array( 'exclude' => array( 100, 101 ), 'disable' => $immovable )
|
|
|
|
|
),
|
|
|
|
|
'Namespace selector without the custom namespace and immovable namespaces disabled.'
|
|
|
|
|
);
|
2012-01-25 03:25:54 +00:00
|
|
|
}
|
2012-01-30 10:39:51 +00:00
|
|
|
|
|
|
|
|
function testNamespaceSelectorIdAndNameDefaultsAttributes() {
|
|
|
|
|
|
|
|
|
|
$this->assertNsSelectorIdAndName(
|
|
|
|
|
'namespace', 'namespace',
|
|
|
|
|
Html::namespaceSelector( array(), array(
|
|
|
|
|
# neither 'id' nor 'name' key given
|
|
|
|
|
)),
|
|
|
|
|
"Neither 'id' nor 'name' key given"
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$this->assertNsSelectorIdAndName(
|
|
|
|
|
'namespace', 'select_name',
|
|
|
|
|
Html::namespaceSelector( array(), array(
|
|
|
|
|
'name' => 'select_name',
|
|
|
|
|
# no 'id' key given
|
|
|
|
|
)),
|
|
|
|
|
"No 'id' key given, 'name' given"
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$this->assertNsSelectorIdAndName(
|
|
|
|
|
'select_id', 'namespace',
|
|
|
|
|
Html::namespaceSelector( array(), array(
|
|
|
|
|
'id' => 'select_id',
|
|
|
|
|
# no 'name' key given
|
|
|
|
|
)),
|
|
|
|
|
"'id' given, no 'name' key given"
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$this->assertNsSelectorIdAndName(
|
|
|
|
|
'select_id', 'select_name',
|
|
|
|
|
Html::namespaceSelector( array(), array(
|
|
|
|
|
'id' => 'select_id',
|
|
|
|
|
'name' => 'select_name',
|
|
|
|
|
)),
|
|
|
|
|
"Both 'id' and 'name' given"
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Helper to verify <select> attributes generated by Html::namespaceSelector()
|
|
|
|
|
* This helper expect the Html method to use 'namespace' as a default value for
|
|
|
|
|
* both 'id' and 'name' attributes.
|
|
|
|
|
*
|
|
|
|
|
* @param String $expectedId <select> id attribute value
|
|
|
|
|
* @param String $expectedName <select> name attribute value
|
|
|
|
|
* @param String $html Output of a call to Html::namespaceSelector()
|
|
|
|
|
* @param String $msg Optional message (default: '')
|
|
|
|
|
*/
|
|
|
|
|
function assertNsSelectorIdAndName( $expectedId, $expectedName, $html, $msg = '' ) {
|
|
|
|
|
$actualId = 'namespace';
|
|
|
|
|
if( 1 === preg_match( '/id="(.+?)"/', $html, $m ) ) {
|
|
|
|
|
$actualId = $m[1];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$actualName = 'namespace';
|
|
|
|
|
if( 1 === preg_match( '/name="(.+?)"/', $html, $m ) ) {
|
|
|
|
|
$actualName = $m[1];
|
|
|
|
|
}
|
|
|
|
|
$this->assertEquals(
|
|
|
|
|
array( #expected
|
|
|
|
|
'id' => $expectedId,
|
|
|
|
|
'name' => $expectedName,
|
|
|
|
|
),
|
|
|
|
|
array( #actual
|
|
|
|
|
'id' => $actualId,
|
|
|
|
|
'name' => $actualName,
|
|
|
|
|
),
|
|
|
|
|
'Html::namespaceSelector() got wrong id and/or name attribute(s). ' . $msg
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-04 19:51:35 +00:00
|
|
|
}
|