* Copyright 2005-2008, Cake Software Foundation, Inc. (http://www.cakefoundation.org) * * Licensed under The Open Group Test Suite License * Redistributions of files must retain the above copyright notice. * * @filesource * @copyright Copyright 2005-2008, Cake Software Foundation, Inc. (http://www.cakefoundation.org) * @link https://trac.cakephp.org/wiki/Developement/TestSuite CakePHP(tm) Tests * @package cake * @subpackage cake.tests.cases.libs.view.helpers * @since CakePHP(tm) v 1.2.0.4206 * @version $Revision$ * @modifiedby $LastChangedBy$ * @lastmodified $Date$ * @license http://www.opensource.org/licenses/opengroup.php The Open Group Test Suite License */ App::import('Helper', array('Js', 'Html')); App::import('Core', array('View', 'ClassRegistry')); Mock::generate('JsBaseEngineHelper', 'TestJsEngineHelper', array('methodOne')); Mock::generate('View', 'JsHelperMockView'); class OptionEngineHelper extends JsBaseEngineHelper { var $_optionMap = array( 'request' => array( 'complete' => 'success', 'request' => 'beforeSend', 'type' => 'dataType' ) ); /** * test method for testing option mapping * * @return array **/ function testMap($options = array()) { return $this->_mapOptions('request', $options); } /** * test method for option parsing * * @return void **/ function testParseOptions($options, $safe = array()) { return $this->_parseOptions($options, $safe); } } /** * JsHelper TestCase. * * @package cake * @subpackage cake.tests.cases.libs.view.helpers */ class JsHelperTestCase extends CakeTestCase { /** * Regexp for CDATA start block * * @var string */ var $cDataStart = 'preg:/^\/\/[\s\r\n]*/'; /** * setUp method * * @access public * @return void */ function startTest() { $this->Js =& new JsHelper('JsBase'); $this->Js->Html =& new HtmlHelper(); $this->Js->JsBaseEngine =& new JsBaseEngineHelper(); $view =& new JsHelperMockView(); ClassRegistry::addObject('view', $view); } /** * tearDown method * * @access public * @return void */ function endTest() { ClassRegistry::removeObject('view'); unset($this->Js); } /** * Switches $this->Js to a mocked engine. * * @return void **/ function _useMock() { $this->Js =& new JsHelper(array('TestJs')); $this->Js->TestJsEngine =& new TestJsEngineHelper($this); $this->Js->Html =& new HtmlHelper(); } /** * test object construction * * @return void **/ function testConstruction() { $js =& new JsHelper(); $this->assertEqual($js->helpers, array('Html', 'JqueryEngine')); $js =& new JsHelper(array('mootools')); $this->assertEqual($js->helpers, array('Html', 'mootoolsEngine')); $js =& new JsHelper('prototype'); $this->assertEqual($js->helpers, array('Html', 'prototypeEngine')); $js =& new JsHelper('MyPlugin.Dojo'); $this->assertEqual($js->helpers, array('Html', 'MyPlugin.DojoEngine')); } /** * test that methods dispatch internally and to the engine class * * @return void **/ function testMethodDispatching() { $this->_useMock(); $this->Js->TestJsEngine->expectOnce('dispatchMethod', array('methodOne', array())); $this->Js->methodOne(); $this->Js->TestEngine =& new StdClass(); $this->expectError(); $this->Js->someMethodThatSurelyDoesntExist(); } /** * Test that method dispatching respects buffer parameters and bufferedMethods Lists. * * @return void **/ function testMethodDispatchWithBuffering() { $this->_useMock(); $this->Js->TestJsEngine->bufferedMethods = array('event', 'sortables'); $this->Js->TestJsEngine->setReturnValue('dispatchMethod', 'This is an event call', array('event', '*')); $this->Js->event('click', 'foo'); $result = $this->Js->getBuffer(); $this->assertEqual(count($result), 1); $this->assertEqual($result[0], 'This is an event call'); $result = $this->Js->event('click', 'foo', array('buffer' => false)); $buffer = $this->Js->getBuffer(); $this->assertTrue(empty($buffer)); $this->assertEqual($result, 'This is an event call'); $result = $this->Js->event('click', 'foo', false); $buffer = $this->Js->getBuffer(); $this->assertTrue(empty($buffer)); $this->assertEqual($result, 'This is an event call'); $this->Js->TestJsEngine->setReturnValue('dispatchMethod', 'I am not buffered.', array('effect', '*')); $result = $this->Js->effect('slideIn'); $buffer = $this->Js->getBuffer(); $this->assertTrue(empty($buffer)); $this->assertEqual($result, 'I am not buffered.'); $result = $this->Js->effect('slideIn', true); $buffer = $this->Js->getBuffer(); $this->assertNull($result); $this->assertEqual(count($buffer), 1); $this->assertEqual($buffer[0], 'I am not buffered.'); $result = $this->Js->effect('slideIn', array('speed' => 'slow'), true); $buffer = $this->Js->getBuffer(); $this->assertNull($result); $this->assertEqual(count($buffer), 1); $this->assertEqual($buffer[0], 'I am not buffered.'); $result = $this->Js->effect('slideIn', array('speed' => 'slow', 'buffer' => true)); $buffer = $this->Js->getBuffer(); $this->assertNull($result); $this->assertEqual(count($buffer), 1); $this->assertEqual($buffer[0], 'I am not buffered.'); } /** * test that writeScripts generates scripts inline. * * @return void **/ function testWriteScriptsNoFile() { $this->_useMock(); $this->Js->buffer('one = 1;'); $this->Js->buffer('two = 2;'); $result = $this->Js->writeBuffer(array('onDomReady' => false, 'cache' => false)); $expected = array( 'script' => array('type' => 'text/javascript'), $this->cDataStart, "one = 1;\ntwo = 2;", $this->cDataEnd, '/script', ); $this->assertTags($result, $expected, true); $this->Js->TestJsEngine->expectAtLeastOnce('domReady'); $result = $this->Js->writeBuffer(array('onDomReady' => true, 'cache' => false)); $view =& new JsHelperMockView(); $view->expectAt(0, 'addScript', array(new PatternExpectation('/one\s=\s1;\ntwo\=\2;/'))); $result = $this->Js->writeBuffer(array('onDomReady' => false, 'inline' => false, 'cache' => false)); } /** * test that writeScripts makes files, and puts the events into them. * * @return void **/ function testWriteScriptsInFile() { if ($this->skipIf(!is_writable(JS), 'webroot/js is not Writable, script caching test has been skipped')) { return; } $this->Js->JsBaseEngine = new TestJsEngineHelper(); $this->Js->buffer('one = 1;'); $this->Js->buffer('two = 2;'); $result = $this->Js->writeBuffer(array('onDomReady' => false, 'cache' => true)); $expected = array( 'script' => array('type' => 'text/javascript', 'src' => 'preg:/(.)*\.js/'), ); $this->assertTags($result, $expected); preg_match('/src="(.*\.js)"/', $result, $filename); $this->assertTrue(file_exists(WWW_ROOT . $filename[1])); $contents = file_get_contents(WWW_ROOT . $filename[1]); $this->assertPattern('/one\s=\s1;\ntwo\s=\s2;/', $contents); @unlink(WWW_ROOT . $filename[1]); } /** * test link() * * @return void **/ function testLinkWithMock() { $this->_useMock(); $options = array('update' => '#content'); $this->Js->TestJsEngine->setReturnValue('dispatchMethod', 'ajax code', array('request', '*')); $this->Js->TestJsEngine->expectAt(0, 'dispatchMethod', array('get', new AnythingExpectation())); $this->Js->TestJsEngine->expectAt(1, 'dispatchMethod', array( 'request', array('/posts/view/1', $options) )); $this->Js->TestJsEngine->expectAt(2, 'dispatchMethod', array( 'event', array('click', 'ajax code', $options) )); $result = $this->Js->link('test link', '/posts/view/1', $options); $expected = array( 'a' => array('id' => 'preg:/link-\d+/', 'href' => '/posts/view/1'), 'test link', '/a' ); $this->assertTags($result, $expected); $options = array( 'confirm' => 'Are you sure?', 'update' => '#content', 'class' => 'my-class', 'id' => 'custom-id', 'escape' => false ); $this->Js->TestJsEngine->expectAt(0, 'confirm', array($options['confirm'])); $this->Js->TestJsEngine->expectAt(1, 'request', array('/posts/view/1', '*')); $code = <<Js->TestJsEngine->expectAt(1, 'event', array('click', $code)); $result = $this->Js->link('test link »', '/posts/view/1', $options); $expected = array( 'a' => array('id' => $options['id'], 'class' => $options['class'], 'href' => '/posts/view/1'), 'test link »', '/a' ); $this->assertTags($result, $expected); } } /** * JsBaseEngine Class Test case * * @package cake.tests.view.helpers **/ class JsBaseEngineTestCase extends CakeTestCase { /** * setUp method * * @access public * @return void */ function startTest() { $this->JsEngine = new JsBaseEngineHelper(); } /** * tearDown method * * @access public * @return void */ function endTest() { ClassRegistry::removeObject('view'); unset($this->JsEngine); } /** * test escape string skills * * @return void **/ function testEscaping() { $result = $this->JsEngine->escape(''); $expected = ''; $this->assertEqual($result, $expected); $result = $this->JsEngine->escape('CakePHP' . "\n" . 'Rapid Development Framework'); $expected = 'CakePHP\\nRapid Development Framework'; $this->assertEqual($result, $expected); $result = $this->JsEngine->escape('CakePHP' . "\r\n" . 'Rapid Development Framework' . "\r" . 'For PHP'); $expected = 'CakePHP\\r\\nRapid Development Framework\\rFor PHP'; $this->assertEqual($result, $expected); $result = $this->JsEngine->escape('CakePHP: "Rapid Development Framework"'); $expected = 'CakePHP: \\"Rapid Development Framework\\"'; $this->assertEqual($result, $expected); $result = $this->JsEngine->escape("CakePHP: 'Rapid Development Framework'"); $expected = "CakePHP: 'Rapid Development Framework'"; $this->assertEqual($result, $expected); $result = $this->JsEngine->escape('my \\"string\\"'); $expected = 'my \\\\\\"string\\\\\\"'; $this->assertEqual($result, $expected); } /** * test prompt() creation * * @return void **/ function testPrompt() { $result = $this->JsEngine->prompt('Hey, hey you', 'hi!'); $expected = 'prompt("Hey, hey you", "hi!");'; $this->assertEqual($result, $expected); $result = $this->JsEngine->prompt('"Hey"', '"hi"'); $expected = 'prompt("\"Hey\"", "\"hi\"");'; $this->assertEqual($result, $expected); } /** * test alert generation * * @return void **/ function testAlert() { $result = $this->JsEngine->alert('Hey there'); $expected = 'alert("Hey there");'; $this->assertEqual($result, $expected); $result = $this->JsEngine->alert('"Hey"'); $expected = 'alert("\"Hey\"");'; $this->assertEqual($result, $expected); } /** * test confirm generation * * @return void **/ function testConfirm() { $result = $this->JsEngine->confirm('Are you sure?'); $expected = 'confirm("Are you sure?");'; $this->assertEqual($result, $expected); $result = $this->JsEngine->confirm('"Are you sure?"'); $expected = 'confirm("\"Are you sure?\"");'; $this->assertEqual($result, $expected); } /** * test Redirect * * @return void **/ function testRedirect() { $result = $this->JsEngine->redirect(array('controller' => 'posts', 'action' => 'view', 1)); $expected = 'window.location = "/posts/view/1";'; $this->assertEqual($result, $expected); } /** * testObject encoding with non-native methods. * * @return void **/ function testObject() { $this->JsEngine->useNative = false; $object = array('title' => 'New thing', 'indexes' => array(5, 6, 7, 8)); $result = $this->JsEngine->object($object); $expected = '{"title":"New thing","indexes":[5,6,7,8]}'; $this->assertEqual($result, $expected); $result = $this->JsEngine->object(array('default' => 0)); $expected = '{"default":0}'; $this->assertEqual($result, $expected); $result = $this->JsEngine->object(array( '2007' => array( 'Spring' => array( '1' => array('id' => 1, 'name' => 'Josh'), '2' => array('id' => 2, 'name' => 'Becky') ), 'Fall' => array( '1' => array('id' => 1, 'name' => 'Josh'), '2' => array('id' => 2, 'name' => 'Becky') ) ), '2006' => array( 'Spring' => array( '1' => array('id' => 1, 'name' => 'Josh'), '2' => array('id' => 2, 'name' => 'Becky') ), 'Fall' => array( '1' => array('id' => 1, 'name' => 'Josh'), '2' => array('id' => 2, 'name' => 'Becky') ) ) )); $expected = '{"2007":{"Spring":{"1":{"id":1,"name":"Josh"},"2":{"id":2,"name":"Becky"}},"Fall":{"1":{"id":1,"name":"Josh"},"2":{"id":2,"name":"Becky"}}},"2006":{"Spring":{"1":{"id":1,"name":"Josh"},"2":{"id":2,"name":"Becky"}},"Fall":{"1":{"id":1,"name":"Josh"},"2":{"id":2,"name":"Becky"}}}}'; $this->assertEqual($result, $expected); foreach (array('true' => true, 'false' => false, 'null' => null) as $expected => $data) { $result = $this->JsEngine->object($data); $this->assertEqual($result, $expected); } } /** * test compatibility of JsBaseEngineHelper::object() vs. json_encode() * * @return void **/ function testObjectAgainstJsonEncode() { $skip = $this->skipIf(!function_exists('json_encode'), 'json_encode() not found, comparison tests skipped. %s'); if ($skip) { return; } $this->JsEngine->useNative = false; $data = array(); $data['mystring'] = "simple string"; $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); $data['mystring'] = "strïng with spécial chârs"; $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); $data['mystring'] = "a two lines\nstring"; $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); $data['mystring'] = "a \t tabbed \t string"; $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); $data['mystring'] = "a \"double-quoted\" string"; $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); $data['mystring'] = 'a \\"double-quoted\\" string'; $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); unset($data['mystring']); $data[3] = array(1, 2, 3); $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); unset($data[3]); $data = array('mystring' => null, 'bool' => false, 'array' => array(1, 44, 66)); $this->assertEqual(json_encode($data), $this->JsEngine->object($data)); } /** * test that JSON made with JsBaseEngineHelper::object() against json_decode() * * @return void **/ function testObjectAgainstJsonDecode() { $skip = $this->skipIf(!function_exists('json_encode'), 'json_encode() not found, comparison tests skipped. %s'); if ($skip) { return; } $this->JsEngine->useNative = false; $data = array("simple string"); $result = $this->JsEngine->object($data); $this->assertEqual(json_decode($result), $data); $data = array('my "string"'); $result = $this->JsEngine->object($data); $this->assertEqual(json_decode($result), $data); $data = array('my \\"string\\"'); $result = $this->JsEngine->object($data); $this->assertEqual(json_decode($result), $data); } /** * test Mapping of options. * * @return void **/ function testOptionMapping() { $JsEngine = new OptionEngineHelper(); $result = $JsEngine->testMap(); $this->assertEqual($result, array()); $result = $JsEngine->testMap(array('foo' => 'bar', 'baz' => 'sho')); $this->assertEqual($result, array('foo' => 'bar', 'baz' => 'sho')); $result = $JsEngine->testMap(array('complete' => 'myFunc', 'type' => 'json', 'update' => '#element')); $this->assertEqual($result, array('success' => 'myFunc', 'dataType' => 'json', 'update' => '#element')); $result = $JsEngine->testMap(array('success' => 'myFunc', 'dataType' => 'json', 'update' => '#element')); $this->assertEqual($result, array('success' => 'myFunc', 'dataType' => 'json', 'update' => '#element')); } /** * test that option parsing escapes strings and saves what is supposed to be saved. * * @return void **/ function testOptionParsing() { $JsEngine = new OptionEngineHelper(); $result = $JsEngine->testParseOptions(array('url' => '/posts/view/1', 'key' => 1)); $expected = 'key:1, url:"\\/posts\\/view\\/1"'; $this->assertEqual($result, $expected); $result = $JsEngine->testParseOptions(array('url' => '/posts/view/1', 'success' => 'doSuccess'), array('success')); $expected = 'success:doSuccess, url:"\\/posts\\/view\\/1"'; $this->assertEqual($result, $expected); } } ?>