5 * @author Matthew Flaschen
9 * @todo factor tests in this class into providers and test methods
12 class OutputPageTest
extends MediaWikiTestCase
{
13 const SCREEN_MEDIA_QUERY
= 'screen and (min-width: 982px)';
14 const SCREEN_ONLY_MEDIA_QUERY
= 'only screen and (min-width: 982px)';
17 * Tests a particular case of transformCssMedia, using the given input, globals,
18 * expected return, and message
20 * Asserts that $expectedReturn is returned.
22 * options['printableQuery'] - value of query string for printable, or omitted for none
23 * options['handheldQuery'] - value of query string for handheld, or omitted for none
24 * options['media'] - passed into the method under the same name
25 * options['expectedReturn'] - expected return value
26 * options['message'] - PHPUnit message for assertion
28 * @param array $args Key-value array of arguments as shown above
30 protected function assertTransformCssMediaCase( $args ) {
32 if ( isset( $args['printableQuery'] ) ) {
33 $queryData['printable'] = $args['printableQuery'];
36 if ( isset( $args['handheldQuery'] ) ) {
37 $queryData['handheld'] = $args['handheldQuery'];
40 $fauxRequest = new FauxRequest( $queryData, false );
41 $this->setMwGlobals( array(
42 'wgRequest' => $fauxRequest,
45 $actualReturn = OutputPage
::transformCssMedia( $args['media'] );
46 $this->assertSame( $args['expectedReturn'], $actualReturn, $args['message'] );
50 * Tests print requests
51 * @covers OutputPage::transformCssMedia
53 public function testPrintRequests() {
54 $this->assertTransformCssMediaCase( array(
55 'printableQuery' => '1',
57 'expectedReturn' => null,
58 'message' => 'On printable request, screen returns null'
61 $this->assertTransformCssMediaCase( array(
62 'printableQuery' => '1',
63 'media' => self
::SCREEN_MEDIA_QUERY
,
64 'expectedReturn' => null,
65 'message' => 'On printable request, screen media query returns null'
68 $this->assertTransformCssMediaCase( array(
69 'printableQuery' => '1',
70 'media' => self
::SCREEN_ONLY_MEDIA_QUERY
,
71 'expectedReturn' => null,
72 'message' => 'On printable request, screen media query with only returns null'
75 $this->assertTransformCssMediaCase( array(
76 'printableQuery' => '1',
78 'expectedReturn' => '',
79 'message' => 'On printable request, media print returns empty string'
84 * Tests screen requests, without either query parameter set
85 * @covers OutputPage::transformCssMedia
87 public function testScreenRequests() {
88 $this->assertTransformCssMediaCase( array(
90 'expectedReturn' => 'screen',
91 'message' => 'On screen request, screen media type is preserved'
94 $this->assertTransformCssMediaCase( array(
95 'media' => 'handheld',
96 'expectedReturn' => 'handheld',
97 'message' => 'On screen request, handheld media type is preserved'
100 $this->assertTransformCssMediaCase( array(
101 'media' => self
::SCREEN_MEDIA_QUERY
,
102 'expectedReturn' => self
::SCREEN_MEDIA_QUERY
,
103 'message' => 'On screen request, screen media query is preserved.'
106 $this->assertTransformCssMediaCase( array(
107 'media' => self
::SCREEN_ONLY_MEDIA_QUERY
,
108 'expectedReturn' => self
::SCREEN_ONLY_MEDIA_QUERY
,
109 'message' => 'On screen request, screen media query with only is preserved.'
112 $this->assertTransformCssMediaCase( array(
114 'expectedReturn' => 'print',
115 'message' => 'On screen request, print media type is preserved'
120 * Tests handheld behavior
121 * @covers OutputPage::transformCssMedia
123 public function testHandheld() {
124 $this->assertTransformCssMediaCase( array(
125 'handheldQuery' => '1',
126 'media' => 'handheld',
127 'expectedReturn' => '',
128 'message' => 'On request with handheld querystring and media is handheld, returns empty string'
131 $this->assertTransformCssMediaCase( array(
132 'handheldQuery' => '1',
134 'expectedReturn' => null,
135 'message' => 'On request with handheld querystring and media is screen, returns null'
139 public static function provideMakeResourceLoaderLink() {
141 // Load module script only
143 array( 'test.foo', ResourceLoaderModule
::TYPE_SCRIPTS
),
144 '<script>var _mwq = _mwq || []; _mwq.push( function ( mw ) {' .
145 ' document.write("\u003Cscript src=\"http://127.0.0.1:8080/w/load.php?debug=' .
146 'false\u0026amp;lang=en\u0026amp;modules=test.foo\u0026amp;only=scripts\u0026' .
147 'amp;skin=fallback\u0026amp;*\"\u003E\u003C/script\u003E"); } );</script>
151 // Don't condition wrap raw modules (like the startup module)
152 array( 'test.raw', ResourceLoaderModule
::TYPE_SCRIPTS
),
153 '<script src="http://127.0.0.1:8080/w/load.php?debug=false&lang=en&' .
154 'modules=test.raw&only=scripts&skin=fallback&*"></script>
157 // Load module styles only
158 // This also tests the order the modules are put into the url
160 array( array( 'test.baz', 'test.foo', 'test.bar' ), ResourceLoaderModule
::TYPE_STYLES
),
161 '<link rel=stylesheet href="http://127.0.0.1:8080/w/load.php?debug=false&' .
162 'lang=en&modules=test.bar%2Cbaz%2Cfoo&only=styles&skin=fallback&*">
165 // Load private module (only=scripts)
167 array( 'test.quux', ResourceLoaderModule
::TYPE_SCRIPTS
),
168 '<script>var _mwq = _mwq || []; _mwq.push( function ( mw ) {' .
169 ' mw.test.baz({token:123});mw.loader.state({"test.quux":"ready"});
173 // Load private module (combined)
175 array( 'test.quux', ResourceLoaderModule
::TYPE_COMBINED
),
176 '<script>var _mwq = _mwq || []; _mwq.push( function ( mw ) {' .
177 ' mw.loader.implement("test.quux",function($,jQuery){mw.test.baz({token:123});}' .
178 ',{"css":[".mw-icon{transition:none}\n"]},{},{});
182 // Load module script with ESI
184 array( 'test.foo', ResourceLoaderModule
::TYPE_SCRIPTS
, true ),
185 '<script><esi:include src="http://127.0.0.1:8080/w/load.php?debug=false&' .
186 'lang=en&modules=test.foo&only=scripts&skin=fallback&*" /></script>
189 // Load module styles with ESI
191 array( 'test.foo', ResourceLoaderModule
::TYPE_STYLES
, true ),
192 '<style><esi:include src="http://127.0.0.1:8080/w/load.php?debug=false&' .
193 'lang=en&modules=test.foo&only=styles&skin=fallback&*" /></style>
198 array( array(), ResourceLoaderModule
::TYPE_COMBINED
),
203 array( 'test.noscript', ResourceLoaderModule
::TYPE_STYLES
),
204 '<noscript><link rel=stylesheet href="http://127.0.0.1:8080/w/load.php?debug=' .
205 'false&lang=en&modules=test.noscript&only=styles&skin=fallback' .
209 // Load two modules in separate groups
211 array( array( 'test.group.foo', 'test.group.bar' ), ResourceLoaderModule
::TYPE_COMBINED
),
212 '<script>var _mwq = _mwq || []; _mwq.push( function ( mw ) { ' .
213 'document.write("\u003Cscript src=\"http://127.0.0.1:8080/w/load.php?debug=false' .
214 '\u0026amp;lang=en\u0026amp;modules=test.group.bar\u0026amp;skin=fallback\u0026' .
215 'amp;*\"\u003E\u003C/script\u003E"); } );</script>
216 <script>var _mwq = _mwq || []; _mwq.push( function ( mw ) { document.write(' .
217 '"\u003Cscript src=\"http://127.0.0.1:8080/w/load.php?debug=false\u0026amp;' .
218 'lang=en\u0026amp;modules=test.group.foo\u0026amp;skin=fallback\u0026amp;*\\' .
219 '"\u003E\u003C/script\u003E"); } );</script>
226 * @dataProvider provideMakeResourceLoaderLink
227 * @covers OutputPage::makeResourceLoaderLink
229 public function testMakeResourceLoaderLink( $args, $expectedHtml ) {
230 $this->setMwGlobals( array(
231 'wgResourceLoaderDebug' => false,
232 'wgResourceLoaderUseESI' => true,
233 'wgLoadScript' => 'http://127.0.0.1:8080/w/load.php',
234 // Affects whether CDATA is inserted
235 'wgWellFormedXml' => false,
237 $class = new ReflectionClass( 'OutputPage' );
238 $method = $class->getMethod( 'makeResourceLoaderLink' );
239 $method->setAccessible( true );
240 $ctx = new RequestContext();
241 $ctx->setSkin( SkinFactory
::getDefaultInstance()->makeSkin( 'fallback' ) );
242 $ctx->setLanguage( 'en' );
243 $out = new OutputPage( $ctx );
244 $rl = $out->getResourceLoader();
245 $rl->register( array(
246 'test.foo' => new ResourceLoaderTestModule( array(
247 'script' => 'mw.test.foo( { a: true } );',
248 'styles' => '.mw-test-foo { content: "style"; }',
250 'test.bar' => new ResourceLoaderTestModule( array(
251 'script' => 'mw.test.bar( { a: true } );',
252 'styles' => '.mw-test-bar { content: "style"; }',
254 'test.baz' => new ResourceLoaderTestModule( array(
255 'script' => 'mw.test.baz( { a: true } );',
256 'styles' => '.mw-test-baz { content: "style"; }',
258 'test.quux' => new ResourceLoaderTestModule( array(
259 'script' => 'mw.test.baz( { token: 123 } );',
260 'styles' => '/* pref-animate=off */ .mw-icon { transition: none; }',
261 'group' => 'private',
263 'test.raw' => new ResourceLoaderTestModule( array(
264 'script' => 'mw.test.baz( { token: 123 } );',
267 'test.noscript' => new ResourceLoaderTestModule( array(
268 'styles' => '.mw-test-noscript { content: "style"; }',
269 'group' => 'noscript',
271 'test.group.bar' => new ResourceLoaderTestModule( array(
272 'styles' => '.mw-group-bar { content: "style"; }',
275 'test.group.foo' => new ResourceLoaderTestModule( array(
276 'styles' => '.mw-group-foo { content: "style"; }',
280 $links = $method->invokeArgs( $out, $args );
281 // Strip comments to avoid variation due to wgDBname in WikiID and cache key
282 $actualHtml = preg_replace( '#/\*[^*]+\*/#', '', $links['html'] );
283 $this->assertEquals( $expectedHtml, $actualHtml );