2008-07-21 11:59:24 +00:00
< ? php
class SSViewerTest extends SapphireTest {
2011-12-17 17:35:26 +13:00
function setUp () {
parent :: setUp ();
SSViewer :: set_source_file_comments ( false );
}
2010-04-13 02:13:12 +00:00
/**
* Tests for { @ link SSViewer :: current_theme ()} for different behaviour
* of user defined themes via { @ link SiteConfig } and default theme
* when no user themes are defined .
*/
function testCurrentTheme () {
2011-03-18 15:04:43 +13:00
//TODO: SiteConfig moved to CMS
2010-04-13 02:13:12 +00:00
SSViewer :: set_theme ( 'mytheme' );
$this -> assertEquals ( 'mytheme' , SSViewer :: current_theme (), 'Current theme is the default - user has not defined one' );
}
2008-07-21 11:59:24 +00:00
/**
* Test that a template without a < head > tag still renders .
*/
function testTemplateWithoutHeadRenders () {
$data = new ArrayData ( array (
'Var' => 'var value'
));
$result = $data -> renderWith ( " SSViewerTestPartialTemplate " );
2008-11-24 19:28:46 +00:00
$this -> assertEquals ( 'Test partial template: var value' , trim ( preg_replace ( " /<!--.*-->/U " , '' , $result )));
2008-07-21 11:59:24 +00:00
}
2008-11-10 01:01:55 +00:00
2009-11-27 00:24:57 +00:00
/**
* Small helper to render templates from strings
*/
function render ( $templateString , $data = null ) {
$t = SSViewer :: fromString ( $templateString );
if ( ! $data ) $data = new SSViewerTestFixture ();
return $t -> process ( $data );
}
2008-11-10 01:01:55 +00:00
function testRequirements () {
$requirements = $this -> getMock ( " Requirements_Backend " , array ( " javascript " , " css " ));
$jsFile = 'sapphire/tests/forms/a.js' ;
$cssFile = 'sapphire/tests/forms/a.js' ;
$requirements -> expects ( $this -> once ()) -> method ( 'javascript' ) -> with ( $jsFile );
$requirements -> expects ( $this -> once ()) -> method ( 'css' ) -> with ( $cssFile );
2008-11-10 01:13:42 +00:00
Requirements :: set_backend ( $requirements );
2008-11-10 01:01:55 +00:00
2009-11-27 00:24:57 +00:00
$template = $this -> render ( " <% require javascript( $jsFile ) %>
<% require css ( $cssFile ) %> " );
2008-11-10 01:01:55 +00:00
$this -> assertFalse (( bool ) trim ( $template ), " Should be no content in this return. " );
}
2008-11-14 03:43:26 +00:00
function testComments () {
2009-11-27 00:24:57 +00:00
$output = $this -> render ( <<< SS
2008-11-14 03:43:26 +00:00
This is my template <%-- this is a comment --%> This is some content <%-- this is another comment --%> This is the final content
SS
);
2008-11-24 19:28:46 +00:00
$this -> assertEquals ( " This is my templateThis is some contentThis is the final content " , preg_replace ( " / \n ?<!--.*--> \n ?/U " , '' , $output ));
2008-11-14 03:43:26 +00:00
}
2009-01-04 22:20:36 +00:00
2011-03-17 13:08:13 +13:00
function testBasicText () {
$this -> assertEquals ( '"' , $this -> render ( '"' ), 'Double-quotes are left alone' );
$this -> assertEquals ( " ' " , $this -> render ( " ' " ), 'Single-quotes are left alone' );
$this -> assertEquals ( 'A' , $this -> render ( '\\A' ), 'Escaped characters are unescaped' );
$this -> assertEquals ( '\\A' , $this -> render ( '\\\\A' ), 'Escaped back-slashed are correctly unescaped' );
}
function testBasicInjection () {
$this -> assertEquals ( '[out:Test]' , $this -> render ( '$Test' ), 'Basic stand-alone injection' );
$this -> assertEquals ( '[out:Test]' , $this -> render ( '{$Test}' ), 'Basic stand-alone wrapped injection' );
$this -> assertEquals ( 'A[out:Test]!' , $this -> render ( 'A$Test!' ), 'Basic surrounded injection' );
$this -> assertEquals ( 'A[out:Test]B' , $this -> render ( 'A{$Test}B' ), 'Basic surrounded wrapped injection' );
$this -> assertEquals ( 'A$B' , $this -> render ( 'A\\$B' ), 'No injection as $ escaped' );
$this -> assertEquals ( 'A$ B' , $this -> render ( 'A$ B' ), 'No injection as $ not followed by word character' );
$this -> assertEquals ( 'A{$ B' , $this -> render ( 'A{$ B' ), 'No injection as {$ not followed by word character' );
$this -> assertEquals ( '{$Test}' , $this -> render ( '{\\$Test}' ), 'Escapes can be used to avoid injection' );
$this -> assertEquals ( '{\\[out:Test]}' , $this -> render ( '{\\\\$Test}' ), 'Escapes before injections are correctly unescaped' );
}
2012-02-11 15:08:39 +13:00
function testGlobalVariableCalls () {
$this -> assertEquals ( Director :: absoluteBaseURL (), $this -> render ( '{$absoluteBaseURL}' ), 'Director::absoluteBaseURL can be called from within template' );
$this -> assertEquals ( Director :: absoluteBaseURL (), $this -> render ( '{$AbsoluteBaseURL}' ), 'Upper-case %AbsoluteBaseURL can be called from within template' );
$this -> assertEquals ( Director :: is_ajax (), $this -> render ( '{$isAjax}' ), 'All variations of is_ajax result in the correct call' );
$this -> assertEquals ( Director :: is_ajax (), $this -> render ( '{$IsAjax}' ), 'All variations of is_ajax result in the correct call' );
$this -> assertEquals ( Director :: is_ajax (), $this -> render ( '{$is_ajax}' ), 'All variations of is_ajax result in the correct call' );
$this -> assertEquals ( Director :: is_ajax (), $this -> render ( '{$Is_ajax}' ), 'All variations of is_ajax result in the correct call' );
$this -> assertEquals ( i18n :: get_locale (), $this -> render ( '{$i18nLocale}' ), 'i18n template functions result correct result' );
$this -> assertEquals ( i18n :: get_locale (), $this -> render ( '{$get_locale}' ), 'i18n template functions result correct result' );
$this -> assertEquals (( string ) Controller :: curr (), $this -> render ( '{$CurrentPage}' ), 'i18n template functions result correct result' );
$this -> assertEquals (( string ) Controller :: curr (), $this -> render ( '{$currentPage}' ), 'i18n template functions result correct result' );
$this -> assertEquals ( Member :: currentUser (), $this -> render ( '{$CurrentMember}' ), 'Member template functions result correct result' );
$this -> assertEquals ( Member :: currentUser (), $this -> render ( '{$CurrentUser}' ), 'Member template functions result correct result' );
$this -> assertEquals ( Member :: currentUser (), $this -> render ( '{$currentMember}' ), 'Member template functions result correct result' );
$this -> assertEquals ( Member :: currentUser (), $this -> render ( '{$currentUser}' ), 'Member template functions result correct result' );
$this -> assertEquals ( SecurityToken :: getSecurityID (), $this -> render ( '{$getSecurityID}' ), 'SecurityToken template functions result correct result' );
$this -> assertEquals ( SecurityToken :: getSecurityID (), $this -> render ( '{$SecurityID}' ), 'SecurityToken template functions result correct result' );
}
function testGlobalVariableCallsWithArguments () {
$this -> assertEquals ( Permission :: check ( " ADMIN " ), $this -> render ( '{$HasPerm(\'ADMIN\')}' ), 'Permissions template functions result correct result' );
$this -> assertEquals ( Permission :: check ( " ADMIN " ), $this -> render ( '{$hasPerm(\'ADMIN\')}' ), 'Permissions template functions result correct result' );
}
/* //TODO: enable this test
function testLocalFunctionsTakePriorityOverGlobals () {
$data = new ArrayData ( array (
'Page' => new SSViewerTest_Page ()
));
$result = $this -> render ( '<% control Page %>$absoluteBaseURL<% end_control %>' , $data );
$this -> assertEquals ( " testPageCalled " , $result , " Local Object's function called. Did not return the actual baseURL of the current site " );
} */
2011-03-17 13:08:13 +13:00
2009-01-04 22:20:36 +00:00
function testObjectDotArguments () {
$this -> assertEquals (
2009-11-27 00:24:57 +00:00
' [ out : TestObject . methodWithOneArgument ( one )]
[ out : TestObject . methodWithTwoArguments ( one , two )]
[ out : TestMethod ( Arg1 , Arg2 ) . Bar . Val ]
[ out : TestMethod ( Arg1 , Arg2 ) . Bar ]
[ out : TestMethod ( Arg1 , Arg2 )]
[ out : TestMethod ( Arg1 ) . Bar . Val ]
[ out : TestMethod ( Arg1 ) . Bar ]
[ out : TestMethod ( Arg1 )] ' ,
$this -> render ( ' $TestObject . methodWithOneArgument ( one )
$TestObject . methodWithTwoArguments ( one , two )
$TestMethod ( Arg1 , Arg2 ) . Bar . Val
$TestMethod ( Arg1 , Arg2 ) . Bar
$TestMethod ( Arg1 , Arg2 )
$TestMethod ( Arg1 ) . Bar . Val
$TestMethod ( Arg1 ) . Bar
$TestMethod ( Arg1 ) ' )
2009-01-04 22:20:36 +00:00
);
2009-11-27 00:24:57 +00:00
}
function testEscapedArguments () {
2009-01-04 22:20:36 +00:00
$this -> assertEquals (
2009-11-27 00:24:57 +00:00
' [ out : Foo ( Arg1 , Arg2 ) . Bar . Val ] . Suffix
[ out : Foo ( Arg1 , Arg2 ) . Val ] _Suffix
[ out : Foo ( Arg1 , Arg2 )] / Suffix
[ out : Foo ( Arg1 ) . Bar . Val ] textSuffix
[ out : Foo ( Arg1 ) . Bar ] . Suffix
[ out : Foo ( Arg1 )] . Suffix
[ out : Foo . Bar . Val ] . Suffix
[ out : Foo . Bar ] . Suffix
[ out : Foo ] . Suffix ' ,
$this -> render ( ' { $Foo ( Arg1 , Arg2 ) . Bar . Val } . Suffix
{ $Foo ( Arg1 , Arg2 ) . Val } _Suffix
{ $Foo ( Arg1 , Arg2 )} / Suffix
{ $Foo ( Arg1 ) . Bar . Val } textSuffix
{ $Foo ( Arg1 ) . Bar } . Suffix
{ $Foo ( Arg1 )} . Suffix
{ $Foo . Bar . Val } . Suffix
{ $Foo . Bar } . Suffix
{ $Foo } . Suffix ' )
2009-01-04 22:20:36 +00:00
);
}
2009-10-31 00:16:54 +00:00
2009-11-27 00:24:58 +00:00
function testControlWhitespace () {
$this -> assertEquals (
' before [ out : SingleItem . Test ] after
beforeTestafter ' ,
$this -> render ( ' before <% control SingleItem %> $Test <% end_control %> after
before <% control SingleItem %> Test <% end_control %> after ' )
);
// The control tags are removed from the output, but no whitespace
// This is a quirk that could be changed, but included in the test to make the current
// behaviour explicit
$this -> assertEquals (
' before
[ out : SingleItem . ItemOnItsOwnLine ]
after ' ,
$this -> render ( ' before
<% control SingleItem %>
$ItemOnItsOwnLine
<% end_control %>
after ' )
);
// The whitespace within the control tags is preserve in a loop
// This is a quirk that could be changed, but included in the test to make the current
// behaviour explicit
$this -> assertEquals (
' before
[ out : Loop3 . ItemOnItsOwnLine ]
[ out : Loop3 . ItemOnItsOwnLine ]
[ out : Loop3 . ItemOnItsOwnLine ]
after ' ,
$this -> render ( ' before
<% control Loop3 %>
$ItemOnItsOwnLine
<% end_control %>
after ' )
);
}
function testControls () {
// Single item controls
$this -> assertEquals (
' a [ out : Foo . Bar . Item ] b
[ out : Foo . Bar ( Arg1 ) . Item ]
[ out : Foo ( Arg1 ) . Item ]
[ out : Foo ( Arg1 , Arg2 ) . Item ]
[ out : Foo ( Arg1 , Arg2 , Arg3 ) . Item ] ' ,
$this -> render ( ' <% control Foo . Bar %> a { $Item } b <% end_control %>
<% control Foo . Bar ( Arg1 ) %> $Item <% end_control %>
<% control Foo ( Arg1 ) %> $Item <% end_control %>
<% control Foo ( Arg1 , Arg2 ) %> $Item <% end_control %>
<% control Foo ( Arg1 , Arg2 , Arg3 ) %> $Item <% end_control %> ' )
);
// Loop controls
$this -> assertEquals ( 'a[out:Foo.Loop2.Item]ba[out:Foo.Loop2.Item]b' ,
$this -> render ( '<% control Foo.Loop2 %>a{$Item}b<% end_control %>' ));
$this -> assertEquals ( '[out:Foo.Loop2(Arg1).Item][out:Foo.Loop2(Arg1).Item]' ,
$this -> render ( '<% control Foo.Loop2(Arg1) %>$Item<% end_control %>' ));
$this -> assertEquals ( '[out:Loop2(Arg1).Item][out:Loop2(Arg1).Item]' ,
$this -> render ( '<% control Loop2(Arg1) %>$Item<% end_control %>' ));
$this -> assertEquals ( '[out:Loop2(Arg1,Arg2).Item][out:Loop2(Arg1,Arg2).Item]' ,
$this -> render ( '<% control Loop2(Arg1, Arg2) %>$Item<% end_control %>' ));
$this -> assertEquals ( '[out:Loop2(Arg1,Arg2,Arg3).Item][out:Loop2(Arg1,Arg2,Arg3).Item]' ,
$this -> render ( '<% control Loop2(Arg1, Arg2, Arg3) %>$Item<% end_control %>' ));
}
2009-11-27 00:24:58 +00:00
function testIfBlocks () {
// Basic test
$this -> assertEquals ( 'AC' ,
$this -> render ( 'A<% if NotSet %>B$NotSet<% end_if %>C' ));
2011-12-21 17:15:18 +01:00
// Nested test
$this -> assertEquals ( 'AB1C' ,
$this -> render ( 'A<% if IsSet %>B$NotSet<% if IsSet %>1<% else %>2<% end_if %><% end_if %>C' ));
2009-11-27 00:24:58 +00:00
// else_if
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if NotSet %>B<% else_if IsSet %>C<% end_if %>D' ));
$this -> assertEquals ( 'AD' ,
$this -> render ( 'A<% if NotSet %>B<% else_if AlsoNotset %>C<% end_if %>D' ));
$this -> assertEquals ( 'ADE' ,
$this -> render ( 'A<% if NotSet %>B<% else_if AlsoNotset %>C<% else_if IsSet %>D<% end_if %>E' ));
$this -> assertEquals ( 'ADE' ,
$this -> render ( 'A<% if NotSet %>B<% else_if AlsoNotset %>C<% else_if IsSet %>D<% end_if %>E' ));
// Dot syntax
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if Foo.NotSet %>B<% else_if Foo.IsSet %>C<% end_if %>D' ));
2011-02-14 14:10:17 +13:00
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if Foo.Bar.NotSet %>B<% else_if Foo.Bar.IsSet %>C<% end_if %>D' ));
2009-11-27 00:24:58 +00:00
// Params
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if NotSet(Param) %>B<% else %>C<% end_if %>D' ));
$this -> assertEquals ( 'ABD' ,
$this -> render ( 'A<% if IsSet(Param) %>B<% else %>C<% end_if %>D' ));
2011-02-21 15:24:14 +13:00
// Negation
$this -> assertEquals ( 'AC' ,
$this -> render ( 'A<% if not IsSet %>B<% end_if %>C' ));
$this -> assertEquals ( 'ABC' ,
$this -> render ( 'A<% if not NotSet %>B<% end_if %>C' ));
2009-11-27 00:24:58 +00:00
// Or
$this -> assertEquals ( 'ABD' ,
$this -> render ( 'A<% if IsSet || NotSet %>B<% else_if A %>C<% end_if %>D' ));
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if NotSet || AlsoNotSet %>B<% else_if IsSet %>C<% end_if %>D' ));
$this -> assertEquals ( 'AD' ,
$this -> render ( 'A<% if NotSet || AlsoNotSet %>B<% else_if NotSet3 %>C<% end_if %>D' ));
2011-02-14 14:10:17 +13:00
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if NotSet || AlsoNotSet %>B<% else_if IsSet || NotSet %>C<% end_if %>D' ));
$this -> assertEquals ( 'AD' ,
$this -> render ( 'A<% if NotSet || AlsoNotSet %>B<% else_if NotSet2 || NotSet3 %>C<% end_if %>D' ));
2011-02-21 15:24:14 +13:00
// Negated Or
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if not IsSet || AlsoNotSet %>B<% else_if A %>C<% end_if %>D' ));
$this -> assertEquals ( 'ABD' ,
$this -> render ( 'A<% if not NotSet || AlsoNotSet %>B<% else_if A %>C<% end_if %>D' ));
$this -> assertEquals ( 'ABD' ,
$this -> render ( 'A<% if NotSet || not AlsoNotSet %>B<% else_if A %>C<% end_if %>D' ));
2009-11-27 00:24:58 +00:00
// And
$this -> assertEquals ( 'ABD' ,
$this -> render ( 'A<% if IsSet && AlsoSet %>B<% else_if A %>C<% end_if %>D' ));
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if IsSet && NotSet %>B<% else_if IsSet %>C<% end_if %>D' ));
$this -> assertEquals ( 'AD' ,
$this -> render ( 'A<% if NotSet && NotSet2 %>B<% else_if NotSet3 %>C<% end_if %>D' ));
2011-02-14 14:10:17 +13:00
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if IsSet && NotSet %>B<% else_if IsSet && AlsoSet %>C<% end_if %>D' ));
$this -> assertEquals ( 'AD' ,
$this -> render ( 'A<% if NotSet && NotSet2 %>B<% else_if IsSet && NotSet3 %>C<% end_if %>D' ));
2009-11-27 00:24:58 +00:00
// Equality
$this -> assertEquals ( 'ABC' ,
$this -> render ( 'A<% if RawVal == RawVal %>B<% end_if %>C' ));
$this -> assertEquals ( 'ACD' ,
$this -> render ( 'A<% if Right == Wrong %>B<% else_if RawVal == RawVal %>C<% end_if %>D' ));
$this -> assertEquals ( 'ABC' ,
$this -> render ( 'A<% if Right != Wrong %>B<% end_if %>C' ));
$this -> assertEquals ( 'AD' ,
$this -> render ( 'A<% if Right == Wrong %>B<% else_if RawVal != RawVal %>C<% end_if %>D' ));
// Else
$this -> assertEquals ( 'ADE' ,
$this -> render ( 'A<% if Right == Wrong %>B<% else_if RawVal != RawVal %>C<% else %>D<% end_if %>E' ));
2011-02-21 16:19:10 +13:00
// Empty if with else
$this -> assertEquals ( 'ABC' ,
$this -> render ( 'A<% if NotSet %><% else %>B<% end_if %>C' ));
2009-11-27 00:24:58 +00:00
}
2009-10-31 00:16:54 +00:00
function testBaseTagGeneration () {
// XHTML wil have a closed base tag
2009-11-27 00:24:57 +00:00
$tmpl1 = ' < ? xml version = " 1.0 " encoding = " UTF-8 " ?>
2009-10-31 00:16:54 +00:00
<! DOCTYPE html PUBLIC " -//W3C//DTD XHTML 1.0 Transitional//EN " " http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd " >
< html >
< head ><% base_tag %></ head >
< body >< p > test </ p >< body >
2009-11-27 00:24:57 +00:00
</ html > ' ;
2010-10-19 01:20:07 +00:00
$this -> assertRegExp ( '/<head><base href=".*" \/><\/head>/' , $this -> render ( $tmpl1 ));
2009-10-31 00:16:54 +00:00
// HTML4 and 5 will only have it for IE
2009-11-27 00:24:57 +00:00
$tmpl2 = ' <! DOCTYPE html >
2009-10-31 00:16:54 +00:00
< html >
< head ><% base_tag %></ head >
< body >< p > test </ p >< body >
2009-11-27 00:24:57 +00:00
</ html > ' ;
$this -> assertRegExp ( '/<head><base href=".*"><!--\[if lte IE 6\]><\/base><!\[endif\]--><\/head>/' , $this -> render ( $tmpl2 ));
2009-10-31 00:16:54 +00:00
2009-11-27 00:24:57 +00:00
$tmpl3 = ' <! DOCTYPE HTML PUBLIC " -//W3C//DTD HTML 4.01//EN " " http://www.w3.org/TR/html4/strict.dtd " >
2009-10-31 00:16:54 +00:00
< html >
< head ><% base_tag %></ head >
< body >< p > test </ p >< body >
2009-11-27 00:24:57 +00:00
</ html > ' ;
$this -> assertRegExp ( '/<head><base href=".*"><!--\[if lte IE 6\]><\/base><!\[endif\]--><\/head>/' , $this -> render ( $tmpl3 ));
2009-10-31 00:16:54 +00:00
// Check that the content negotiator converts to the equally legal formats
$negotiator = new ContentNegotiator ();
2009-11-27 00:24:57 +00:00
$response = new SS_HTTPResponse ( $this -> render ( $tmpl1 ));
2009-10-31 00:16:54 +00:00
$negotiator -> html ( $response );
$this -> assertRegExp ( '/<head><base href=".*"><!--\[if lte IE 6\]><\/base><!\[endif\]--><\/head>/' , $response -> getBody ());
2009-11-27 00:24:57 +00:00
$response = new SS_HTTPResponse ( $this -> render ( $tmpl1 ));
2009-10-31 00:16:54 +00:00
$negotiator -> xhtml ( $response );
2010-10-19 01:20:07 +00:00
$this -> assertRegExp ( '/<head><base href=".*" \/><\/head>/' , $response -> getBody ());
2009-10-31 00:16:54 +00:00
}
2010-03-12 03:08:59 +00:00
function testRecursiveInclude () {
$view = new SSViewer ( array ( 'SSViewerTestRecursiveInclude' ));
$data = new ArrayData ( array (
'Title' => 'A' ,
2011-05-05 20:40:24 +10:00
'Children' => new ArrayList ( array (
2010-03-12 03:08:59 +00:00
new ArrayData ( array (
'Title' => 'A1' ,
2011-05-05 20:40:24 +10:00
'Children' => new ArrayList ( array (
2010-03-12 03:08:59 +00:00
new ArrayData ( array ( 'Title' => 'A1 i' , )),
new ArrayData ( array ( 'Title' => 'A1 ii' , )),
)),
)),
new ArrayData ( array ( 'Title' => 'A2' , )),
new ArrayData ( array ( 'Title' => 'A3' , )),
)),
));
$result = $view -> process ( $data );
// We don't care about whitespace
$rationalisedResult = trim ( preg_replace ( '/\s+/' , ' ' , $result ));
$this -> assertEquals ( 'A A1 A1 i A1 ii A2 A3' , $rationalisedResult );
}
2011-02-18 17:06:11 +13:00
function assertEqualIgnoringWhitespace ( $a , $b ) {
$this -> assertEquals ( preg_replace ( '/\s+/' , '' , $a ), preg_replace ( '/\s+/' , '' , $b ));
}
2012-02-29 20:35:48 +01:00
/**
* See { @ link ViewableDataTest } for more extensive casting tests ,
* this test just ensures that basic casting is correctly applied during template parsing .
*/
function testCastingHelpers () {
$vd = new SSViewerTest_ViewableData ();
$vd -> TextValue = '<b>html</b>' ;
$vd -> HTMLValue = '<b>html</b>' ;
$vd -> UncastedValue = '<b>html</b>' ;
// Value casted as "Text"
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$TextValue' ) -> process ( $vd )
);
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$TextValue.RAW' ) -> process ( $vd )
);
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$TextValue.XML' ) -> process ( $vd )
);
// Value casted as "HTMLText"
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$HTMLValue' ) -> process ( $vd )
);
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$HTMLValue.RAW' ) -> process ( $vd )
);
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$HTMLValue.XML' ) -> process ( $vd )
);
// Uncasted value (falls back to ViewableData::$default_cast="HTMLText")
$vd = new SSViewerTest_ViewableData (); // TODO Fix caching
$vd -> UncastedValue = '<b>html</b>' ;
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$UncastedValue' ) -> process ( $vd )
);
$vd = new SSViewerTest_ViewableData (); // TODO Fix caching
$vd -> UncastedValue = '<b>html</b>' ;
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$UncastedValue.RAW' ) -> process ( $vd )
);
$vd = new SSViewerTest_ViewableData (); // TODO Fix caching
$vd -> UncastedValue = '<b>html</b>' ;
$this -> assertEquals (
'<b>html</b>' ,
$t = SSViewer :: fromString ( '$UncastedValue.XML' ) -> process ( $vd )
);
}
2011-02-18 17:06:11 +13:00
/**
* Test $Up works when the scope $Up refers to was entered with a " with " block
*/
function testUpInWith () {
// Data to run the loop tests on - three levels deep
$data = new ArrayData ( array (
'Name' => 'Top' ,
'Foo' => new ArrayData ( array (
'Name' => 'Foo' ,
'Bar' => new ArrayData ( array (
'Name' => 'Bar' ,
'Baz' => new ArrayData ( array (
'Name' => 'Baz'
)),
'Qux' => new ArrayData ( array (
'Name' => 'Qux'
))
))
))
));
// Basic functionality
$this -> assertEquals ( 'BarFoo' ,
$this -> render ( '<% with Foo %><% with Bar %>{$Name}{$Up.Name}<% end_with %><% end_with %>' , $data ));
// Two level with block, up refers to internally referenced Bar
$this -> assertEquals ( 'BarFoo' ,
$this -> render ( '<% with Foo.Bar %>{$Name}{$Up.Name}<% end_with %>' , $data ));
// Stepping up & back down the scope tree
$this -> assertEquals ( 'BazBarQux' ,
$this -> render ( '<% with Foo.Bar.Baz %>{$Name}{$Up.Name}{$Up.Qux.Name}<% end_with %>' , $data ));
// Using $Up in a with block
$this -> assertEquals ( 'BazBarQux' ,
$this -> render ( '<% with Foo.Bar.Baz %>{$Name}<% with $Up %>{$Name}{$Qux.Name}<% end_with %><% end_with %>' , $data ));
// Stepping up & back down the scope tree with with blocks
$this -> assertEquals ( 'BazBarQuxBarBaz' ,
$this -> render ( '<% with Foo.Bar.Baz %>{$Name}<% with $Up %>{$Name}<% with Qux %>{$Name}<% end_with %>{$Name}<% end_with %>{$Name}<% end_with %>' , $data ));
// Using $Up.Up, where first $Up points to a previous scope entered using $Up, thereby skipping up to Foo
$this -> assertEquals ( 'Foo' ,
$this -> render ( '<% with Foo.Bar.Baz %><% with Up %><% with Qux %>{$Up.Up.Name}<% end_with %><% end_with %><% end_with %>' , $data ));
// Using $Up.Up, where first $Up points to an Up used in a local scope lookup, should still skip to Foo
$this -> assertEquals ( 'Foo' ,
$this -> render ( '<% with Foo.Bar.Baz.Up.Qux %>{$Up.Up.Name}<% end_with %>' , $data ));
}
/**
* Test $Up works when the scope $Up refers to was entered with a " loop " block
*/
function testUpInLoop (){
// Data to run the loop tests on - one sequence of three items, each with a subitem
$data = new ArrayData ( array (
'Name' => 'Top' ,
2011-05-05 20:40:24 +10:00
'Foo' => new ArrayList ( array (
2011-02-18 17:06:11 +13:00
new ArrayData ( array (
'Name' => '1' ,
'Sub' => new ArrayData ( array (
'Name' => 'Bar'
))
)),
new ArrayData ( array (
'Name' => '2' ,
'Sub' => new ArrayData ( array (
'Name' => 'Baz'
))
)),
new ArrayData ( array (
'Name' => '3' ,
'Sub' => new ArrayData ( array (
'Name' => 'Qux'
))
))
))
));
// Make sure inside a loop, $Up refers to the current item of the loop
$this -> assertEqualIgnoringWhitespace (
'111 222 333' ,
$this -> render (
'<% loop $Foo %>$Name<% with $Sub %>$Up.Name<% end_with %>$Name<% end_loop %>' ,
$data
)
);
// Make sure inside a loop, looping over $Up uses a separate iterator,
// and doesn't interfere with the original iterator
$this -> assertEqualIgnoringWhitespace (
'1Bar123Bar1 2Baz123Baz2 3Qux123Qux3' ,
$this -> render (
' <% loop $Foo %>
$Name
<% with $Sub %>
2011-02-21 15:24:14 +13:00
$Name
<% loop $Up %> $Name <% end_loop %>
2011-02-18 17:06:11 +13:00
$Name
<% end_with %>
$Name
<% end_loop %> ' ,
$data
)
);
// Make sure inside a loop, looping over $Up uses a separate iterator,
// and doesn't interfere with the original iterator or local lookups
$this -> assertEqualIgnoringWhitespace (
'1 Bar1 123 1Bar 1 2 Baz2 123 2Baz 2 3 Qux3 123 3Qux 3' ,
$this -> render (
' <% loop $Foo %>
$Name
<% with $Sub %>
{ $Name }{ $Up . Name }
<% loop $Up %> $Name <% end_loop %>
{ $Up . Name }{ $Name }
<% end_with %>
$Name
<% end_loop %> ' ,
$data
)
);
}
2011-02-26 17:55:04 +11:00
/**
* @ covers SSViewer :: get_themes ()
*/
function testThemeRetrieval () {
$ds = DIRECTORY_SEPARATOR ;
$testThemeBaseDir = TEMP_FOLDER . $ds . 'test-themes' ;
if ( file_exists ( $testThemeBaseDir )) Filesystem :: removeFolder ( $testThemeBaseDir );
mkdir ( $testThemeBaseDir );
mkdir ( $testThemeBaseDir . $ds . 'blackcandy' );
mkdir ( $testThemeBaseDir . $ds . 'blackcandy_blog' );
mkdir ( $testThemeBaseDir . $ds . 'darkshades' );
mkdir ( $testThemeBaseDir . $ds . 'darkshades_blog' );
$this -> assertEquals ( array (
'blackcandy' => 'blackcandy' ,
'darkshades' => 'darkshades'
), SSViewer :: get_themes ( $testThemeBaseDir ), 'Our test theme directory contains 2 themes' );
$this -> assertEquals ( array (
'blackcandy' => 'blackcandy' ,
'blackcandy_blog' => 'blackcandy_blog' ,
'darkshades' => 'darkshades' ,
'darkshades_blog' => 'darkshades_blog'
), SSViewer :: get_themes ( $testThemeBaseDir , true ), 'Our test theme directory contains 2 themes and 2 sub-themes' );
// Remove all the test themes we created
Filesystem :: removeFolder ( $testThemeBaseDir );
}
2011-10-18 11:42:55 +02:00
function testRewriteHashlinks () {
$oldRewriteHashLinks = SSViewer :: getOption ( 'rewriteHashlinks' );
SSViewer :: setOption ( 'rewriteHashlinks' , true );
// Emulate SSViewer::process()
$base = Convert :: raw2att ( $_SERVER [ 'REQUEST_URI' ]);
$tmplFile = TEMP_FOLDER . '/SSViewerTest_testRewriteHashlinks_' . sha1 ( rand ()) . '.ss' ;
// Note: SSViewer_FromString doesn't rewrite hash links.
file_put_contents ( $tmplFile , ' <! DOCTYPE html >
< html >
< head ><% base_tag %></ head >
< body >
< a class = " inline " href = " #anchor " > InlineLink </ a >
$InsertedLink
< body >
</ html > ' );
$tmpl = new SSViewer ( $tmplFile );
$obj = new ViewableData ();
$obj -> InsertedLink = '<a class="inserted" href="#anchor">InsertedLink</a>' ;
$result = $tmpl -> process ( $obj );
$this -> assertContains (
'<a class="inserted" href="' . $base . '#anchor">InsertedLink</a>' ,
$result
);
$this -> assertContains (
'<a class="inline" href="' . $base . '#anchor">InlineLink</a>' ,
$result
);
unlink ( $tmplFile );
SSViewer :: setOption ( 'rewriteHashlinks' , $oldRewriteHashLinks );
}
function testRewriteHashlinksInPhpMode () {
$oldRewriteHashLinks = SSViewer :: getOption ( 'rewriteHashlinks' );
SSViewer :: setOption ( 'rewriteHashlinks' , 'php' );
$tmplFile = TEMP_FOLDER . '/SSViewerTest_testRewriteHashlinksInPhpMode_' . sha1 ( rand ()) . '.ss' ;
// Note: SSViewer_FromString doesn't rewrite hash links.
file_put_contents ( $tmplFile , ' <! DOCTYPE html >
< html >
< head ><% base_tag %></ head >
< body >
< a class = " inline " href = " #anchor " > InlineLink </ a >
$InsertedLink
< body >
</ html > ' );
$tmpl = new SSViewer ( $tmplFile );
$obj = new ViewableData ();
$obj -> InsertedLink = '<a class="inserted" href="#anchor">InsertedLink</a>' ;
$result = $tmpl -> process ( $obj );
$this -> assertContains (
'<a class="inserted" href="<?php echo strip_tags(' ,
$result
);
// TODO Fix inline links in PHP mode
// $this->assertContains(
// '<a class="inline" href="<?php echo str_replace(',
// $result
// );
unlink ( $tmplFile );
SSViewer :: setOption ( 'rewriteHashlinks' , $oldRewriteHashLinks );
}
2011-12-17 17:35:26 +13:00
function testRenderWithSourceFileComments () {
SSViewer :: set_source_file_comments ( true );
$view = new SSViewer ( array ( 'SSViewerTestCommentsFullSource' ));
$data = new ArrayData ( array ());
$result = $view -> process ( $data );
$expected = ' <! doctype html >
< html ><!-- template ' . BASE_PATH . ' / sapphire / tests / templates / SSViewerTestCommentsFullSource . ss -->
< head ></ head >
< body ></ body >
<!-- end template ' . BASE_PATH . ' / sapphire / tests / templates / SSViewerTestCommentsFullSource . ss --></ html >
' ;
$this -> assertEquals ( $result , $expected );
$view = new SSViewer ( array ( 'SSViewerTestCommentsPartialSource' ));
$data = new ArrayData ( array ());
$result = $view -> process ( $data );
$expected = '<!-- template ' . BASE_PATH . '/sapphire/tests/templates/SSViewerTestCommentsPartialSource.ss --><div class=\'typography\'></div><!-- end template ' . BASE_PATH . '/sapphire/tests/templates/SSViewerTestCommentsPartialSource.ss -->' ;
$this -> assertEquals ( $result , $expected );
$view = new SSViewer ( array ( 'SSViewerTestCommentsWithInclude' ));
$data = new ArrayData ( array ());
$result = $view -> process ( $data );
$expected = '<!-- template ' . BASE_PATH . '/sapphire/tests/templates/SSViewerTestCommentsWithInclude.ss --><div class=\'typography\'><!-- include \'SSViewerTestCommentsInclude\' --><!-- template ' . BASE_PATH . '/sapphire/tests/templates/SSViewerTestCommentsInclude.ss -->Included<!-- end template ' . BASE_PATH . '/sapphire/tests/templates/SSViewerTestCommentsInclude.ss --><!-- end include \'SSViewerTestCommentsInclude\' --></div><!-- end template ' . BASE_PATH . '/sapphire/tests/templates/SSViewerTestCommentsWithInclude.ss -->' ;
$this -> assertEquals ( $result , $expected );
SSViewer :: set_source_file_comments ( false );
}
2011-02-26 17:55:04 +11:00
2009-01-04 22:20:36 +00:00
}
2009-11-27 00:24:57 +00:00
/**
* A test fixture that will echo back the template item
*/
class SSViewerTestFixture extends ViewableData {
protected $name ;
function __construct ( $name = null ) {
$this -> name = $name ;
2011-02-14 14:10:17 +13:00
parent :: __construct ();
2009-11-27 00:24:57 +00:00
}
private function argedName ( $fieldName , $arguments ) {
$childName = $this -> name ? " $this->name . $fieldName " : $fieldName ;
if ( $arguments ) return $childName . '(' . implode ( ',' , $arguments ) . ')' ;
else return $childName ;
}
function obj ( $fieldName , $arguments = null , $forceReturnedObject = true , $cache = false , $cacheName = null ) {
$childName = $this -> argedName ( $fieldName , $arguments );
// Special field name Loop### to create a list
if ( preg_match ( '/^Loop([0-9]+)$/' , $fieldName , $matches )) {
2011-05-05 20:40:24 +10:00
$output = new ArrayList ();
2009-11-27 00:24:57 +00:00
for ( $i = 0 ; $i < $matches [ 1 ]; $i ++ ) $output -> push ( new SSViewerTestFixture ( $childName ));
return $output ;
} else if ( preg_match ( '/NotSet/i' , $fieldName )) {
return new ViewableData ();
} else {
return new SSViewerTestFixture ( $childName );
}
}
function XML_val ( $fieldName , $arguments = null ) {
if ( preg_match ( '/NotSet/i' , $fieldName )) {
return '' ;
} else if ( preg_match ( '/Raw/i' , $fieldName )) {
return $fieldName ;
} else {
return '[out:' . $this -> argedName ( $fieldName , $arguments ) . ']' ;
}
}
function hasValue ( $fieldName , $arguments = null ) {
return ( bool ) $this -> XML_val ( $fieldName , $arguments );
}
}
2010-10-19 00:58:23 +00:00
2009-01-04 22:20:36 +00:00
class SSViewerTest_ViewableData extends ViewableData implements TestOnly {
2012-02-29 20:35:48 +01:00
public static $casting = array (
'TextValue' => 'Text' ,
'HTMLValue' => 'HTMLText'
);
2009-01-04 22:20:36 +00:00
function methodWithOneArgument ( $arg1 ) {
return " arg1: { $arg1 } " ;
}
function methodWithTwoArguments ( $arg1 , $arg2 ) {
return " arg1: { $arg1 } ,arg2: { $arg2 } " ;
}
2009-11-27 00:24:57 +00:00
}
2011-03-18 15:04:43 +13:00
class SSViewerTest_Controller extends Controller {
}
2012-02-11 15:08:39 +13:00
class SSViewerTest_Page extends SiteTree {
function absoluteBaseURL () {
return " testPageCalled " ;
}
}