Merged from branches/2.3

git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@75590 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
Sean Harvey 2009-04-29 01:20:24 +00:00
parent 13b358a8dd
commit a5e82ddff1
23 changed files with 461 additions and 146 deletions

View File

@ -263,6 +263,7 @@ class HTTPRequest extends Object implements ArrayAccess {
$response->addHeader("Content-Type", "$mimeType; name=\"" . addslashes($fileName) . "\""); $response->addHeader("Content-Type", "$mimeType; name=\"" . addslashes($fileName) . "\"");
$response->addHeader("Content-disposition", "attachment; filename=" . addslashes($fileName)); $response->addHeader("Content-disposition", "attachment; filename=" . addslashes($fileName));
$response->addHeader("Content-Length", strlen($fileData)); $response->addHeader("Content-Length", strlen($fileData));
$response->addHeader("Pragma", ""); // Necessary because IE has issues sending files over SSL
return $response; return $response;
} }

View File

@ -133,7 +133,7 @@ class RequestHandler extends ViewableData {
// But if we have more content on the URL and we don't know what to do with it, return an error. // But if we have more content on the URL and we don't know what to do with it, return an error.
} else { } else {
return $this->httpError(400, "I can't handle sub-URLs of a $this->class object."); return $this->httpError(404, "I can't handle sub-URLs of a $this->class object.");
} }
return $this; return $this;

View File

@ -184,9 +184,7 @@ class ErrorPage extends Page {
* @package cms * @package cms
*/ */
class ErrorPage_Controller extends Page_Controller { class ErrorPage_Controller extends Page_Controller {
public function init() { public function index() {
parent::init();
Director::set_status_code($this->failover->ErrorCode ? $this->failover->ErrorCode : 404); Director::set_status_code($this->failover->ErrorCode ? $this->failover->ErrorCode : 404);
} }
} }

View File

@ -19,11 +19,11 @@ class Boolean extends DBField {
} }
function Nice() { function Nice() {
return ($this->value) ? "yes" : "no"; return ($this->value) ? _t('Boolean.YES', 'Yes') : _t('Boolean.NO', 'No');
} }
function NiceAsBoolean() { function NiceAsBoolean() {
return ($this->value) ? "true" : "false"; return ($this->value) ? 'true' : 'false';
} }
/** /**
@ -32,7 +32,7 @@ class Boolean extends DBField {
function saveInto($dataObject) { function saveInto($dataObject) {
$fieldName = $this->name; $fieldName = $this->name;
if($fieldName) { if($fieldName) {
$dataObject->$fieldName = $this->value ? 1 : 0; $dataObject->$fieldName = ($this->value) ? 1 : 0;
} else { } else {
user_error("DBField::saveInto() Called on a nameless '$this->class' object", E_USER_ERROR); user_error("DBField::saveInto() Called on a nameless '$this->class' object", E_USER_ERROR);
} }
@ -42,6 +42,16 @@ class Boolean extends DBField {
return new CheckboxField($this->name, $title); return new CheckboxField($this->name, $title);
} }
public function scaffoldSearchField($title = null) {
$anyText = _t('Boolean.ANY', 'Any');
$source = array(
1 => _t('Boolean.YES', 'Yes'),
0 => _t('Boolean.NO', 'No')
);
return new DropdownField($this->name, $title, $source, '', null, "($anyText)");
}
/** /**
* Return an encoding of the given value suitable for inclusion in a SQL statement. * Return an encoding of the given value suitable for inclusion in a SQL statement.
* If necessary, this should include quotes. * If necessary, this should include quotes.

View File

@ -91,7 +91,7 @@ class Date extends DBField {
/** /**
* Return the date formatted using the given strftime formatting string. * Return the date formatted using the given strftime formatting string.
* *
* strftime obeyes the current LC_TIME/LC_ALL when printing lexical values * strftime obeys the current LC_TIME/LC_ALL when printing lexical values
* like day- and month-names * like day- and month-names
*/ */
function FormatI18N($formattingString) { function FormatI18N($formattingString) {

View File

@ -47,27 +47,25 @@ class Enum extends DBField {
DB::requireField($this->tableName, $this->name, $values); DB::requireField($this->tableName, $this->name, $values);
} }
/**
* Return a dropdown field suitable for editing this field
*/
function formField($title = null, $name = null, $hasEmpty = false, $value = "", $form = null, $emptyString = null) {
if(!$title) $title = $this->name;
if(!$name) $name = $this->name;
$field = new DropdownField($name, $title, $this->enumValues($hasEmpty), $value, $form, $emptyString);
return $field;
}
public function scaffoldFormField($title = null, $params = null) { public function scaffoldFormField($title = null, $params = null) {
return $this->formField($title); return $this->formField($title);
} }
/**
* Return a dropdown field suitable for editing this field
*/
function formField($title = null, $name = null, $hasEmpty = false, $value = "", $form = null) {
if(!$title) $title = $this->name;
if(!$name) $name = $this->name;
$field = new DropdownField($name, $title, $this->enumValues($hasEmpty), $value, $form);
return $field;
}
function scaffoldSearchField($title = null) { function scaffoldSearchField($title = null) {
$field = $this->formField($title); $anyText = _t('Enum.ANY', 'Any');
$field->Source = array_merge(array("" => "(Any)"), $this->enumValues()); return $this->formField($title, null, false, '', null, "($anyText)");
return $field;
} }
/** /**

View File

@ -74,8 +74,8 @@ class CSVParser extends Object implements Iterator {
function __construct($filename, $delimiter = ",", $enclosure = '"') { function __construct($filename, $delimiter = ",", $enclosure = '"') {
$filename = Director::getAbsFile($filename); $filename = Director::getAbsFile($filename);
$this->filename = $filename; $this->filename = $filename;
$tis->delimiter = ","; $this->delimiter = $delimiter;
$this->enclosure = '"'; $this->enclosure = $enclosure;
} }
/** /**

View File

@ -256,7 +256,7 @@ class Debug {
echo $friendlyErrorMessage; echo $friendlyErrorMessage;
} else { } else {
$errorFilePath = ErrorPage::get_filepath_for_errorcode($statusCode, Translatable::current_locale()); $errorFilePath = ErrorPage::get_filepath_for_errorcode($statusCode, Translatable::current_locale());
if(file_exists($errorfilePath)) { if(file_exists($errorFilePath)) {
echo file_get_contents(ASSETS_PATH . "/error-$statusCode.html"); echo file_get_contents(ASSETS_PATH . "/error-$statusCode.html");
} else { } else {
$renderer = new DebugView(); $renderer = new DebugView();

View File

@ -316,7 +316,7 @@ class Folder extends File {
$this, $this,
"Files", "Files",
"File", "File",
array("Title" => _t('Folder.TITLE', "Title"), "LinkedURL" => _t('Folder.FILENAME', "Filename")), array("Title" => _t('Folder.TITLE', "Title"), "Filename" => _t('Folder.FILENAME', "Filename")),
"" ""
); );
$fileList->setFolder($this); $fileList->setFolder($this);
@ -330,9 +330,6 @@ class Folder extends File {
$deleteButton = new HiddenField('deletemarked'); $deleteButton = new HiddenField('deletemarked');
} }
$inlineFormAction = new InlineFormAction("delete_unused_thumbnails", _t('Folder.DELETEUNUSEDTHUMBNAILS', 'Delete unused thumbnails'));
$inlineFormAction->includeDefaultJS(false) ;
$fields = new FieldSet( $fields = new FieldSet(
new HiddenField("Title"), new HiddenField("Title"),
new TabSet("Root", new TabSet("Root",
@ -355,12 +352,7 @@ class Folder extends File {
) )
), ),
new Tab("UnusedFiles", _t('Folder.UNUSEDFILESTAB', "Unused files"), new Tab("UnusedFiles", _t('Folder.UNUSEDFILESTAB', "Unused files"),
new LiteralField( "UnusedAssets", "<h2>"._t('Folder.UNUSEDFILESTITLE', 'Unused files')."</h2>" ), new Folder_UnusedAssetsField($this)
$this->getAssetList(),
new FieldGroup(
new LiteralField( "UnusedThumbnails", "<h2>"._t('Folder.UNUSEDTHUMBNAILSTITLE', 'Unused thumbnails')."</h2>"),
$inlineFormAction
)
) )
), ),
new HiddenField("ID") new HiddenField("ID")
@ -371,34 +363,12 @@ class Folder extends File {
return $fields; return $fields;
} }
/**
* Creates table for displaying unused files.
*
* @returns AssetTableField
*/
protected function getAssetList() {
$where = $this->getUsedFilesList();
$assetList = new AssetTableField(
$this,
"AssetList",
"File",
array("Title" => _t('Folder.TITLE', "Title"), "LinkedURL" => _t('Folder.FILENAME', "Filename")),
"",
$where
);
$assetList->setPopupCaption(_t('Folder.VIEWASSET', "View Asset"));
$assetList->setPermissions(array("show","delete"));
$assetList->Markable = false;
return $assetList;
}
/** /**
* Looks for files used in system and create where clause which contains all ID's of files. * Looks for files used in system and create where clause which contains all ID's of files.
* *
* @returns String where clause which will work as filter. * @returns String where clause which will work as filter.
*/ */
protected function getUsedFilesList() { public function getUsedFilesList() {
$result = DB::query("SELECT DISTINCT \"FileID\" FROM \"SiteTree_ImageTracking\""); $result = DB::query("SELECT DISTINCT \"FileID\" FROM \"SiteTree_ImageTracking\"");
$usedFiles = array(); $usedFiles = array();
$where = ""; $where = "";
@ -446,4 +416,60 @@ HTML;
} }
class Folder_UnusedAssetsField extends CompositeField {
protected $folder;
public function __construct($folder) {
$this->folder = $folder;
parent::__construct(new FieldSet());
}
public function getChildren() {
if($this->children->Count() == 0) {
$inlineFormAction = new InlineFormAction("delete_unused_thumbnails", _t('Folder.DELETEUNUSEDTHUMBNAILS', 'Delete unused thumbnails'));
$inlineFormAction->includeDefaultJS(false) ;
$this->children = new FieldSet(
new LiteralField( "UnusedAssets", "<h2>"._t('Folder.UNUSEDFILESTITLE', 'Unused files')."</h2>" ),
$this->getAssetList(),
new FieldGroup(
new LiteralField( "UnusedThumbnails", "<h2>"._t('Folder.UNUSEDTHUMBNAILSTITLE', 'Unused thumbnails')."</h2>"),
$inlineFormAction
)
);
$this->children->setForm($this->form);
}
return $this->children;
}
public function FieldHolder() {
$output = "";
foreach($this->getChildren() as $child) {
$output .= $child->FieldHolder();
}
return $output;
}
/**
* Creates table for displaying unused files.
*
* @returns AssetTableField
*/
protected function getAssetList() {
$where = $this->folder->getUsedFilesList();
$assetList = new AssetTableField(
$this->folder,
"AssetList",
"File",
array("Title" => _t('Folder.TITLE', "Title"), "LinkedURL" => _t('Folder.FILENAME', "Filename")),
"",
$where
);
$assetList->setPopupCaption(_t('Folder.VIEWASSET', "View Asset"));
$assetList->setPermissions(array("show","delete"));
$assetList->Markable = false;
return $assetList;
}
}
?> ?>

View File

@ -5,20 +5,28 @@
* @subpackage fields-basic * @subpackage fields-basic
*/ */
class CheckboxField extends FormField { class CheckboxField extends FormField {
/**
* Returns a single checkbox field - used by templates.
*
* Shouldn't this have a value?
*/
protected $disabled; protected $disabled;
function setValue($value) {
$this->value = ($value) ? 1 : 0;
}
function dataValue() {
return ($this->value) ? 1 : 0;
}
function Value() {
return ($this->value) ? 1 : 0;
}
function Field() { function Field() {
$attributes = array( $attributes = array(
'type' => 'checkbox', 'type' => 'checkbox',
'class' => ($this->extraClass() ? $this->extraClass() : ''), 'class' => ($this->extraClass() ? $this->extraClass() : ''),
'id' => $this->id(), 'id' => $this->id(),
'name' => $this->Name(), 'name' => $this->Name(),
'value' => 1,
'checked' => $this->value ? 'checked' : '', 'checked' => $this->value ? 'checked' : '',
'tabindex' => $this->getTabIndex() 'tabindex' => $this->getTabIndex()
); );
@ -28,11 +36,6 @@ class CheckboxField extends FormField {
return $this->createTag('input', $attributes); return $this->createTag('input', $attributes);
} }
function dataValue() {
return $this->value ? 1 : 0;
}
/** /**
* Checkboxes use the RightLabelledFieldHolder template, to put the field on the left * Checkboxes use the RightLabelledFieldHolder template, to put the field on the left
* and the label on the right. See {@link FormField::FieldHolder} for more information about * and the label on the right. See {@link FormField::FieldHolder} for more information about
@ -76,7 +79,7 @@ HTML;
*/ */
function performReadonlyTransformation() { function performReadonlyTransformation() {
$field = new CheckboxField_Readonly($this->name, $this->title, $this->value ? 'Yes' : 'No'); $field = new CheckboxField_Readonly($this->name, $this->title, $this->value ? _t('CheckboxField.YES', 'Yes') : _t('CheckboxField.NO', 'No'));
$field->setForm($this->form); $field->setForm($this->form);
return $field; return $field;
} }
@ -99,7 +102,7 @@ class CheckboxField_Readonly extends ReadonlyField {
} }
function setValue($val) { function setValue($val) {
$this->value = ($val) ? 'Yes' : 'No'; $this->value = ($val) ? _t('CheckboxField.YES', 'Yes') : _t('CheckboxField.NO', 'No');
} }
} }

View File

@ -77,9 +77,20 @@ class DropdownField extends FormField {
} }
foreach($source as $value => $title) { foreach($source as $value => $title) {
// Blank value of field and source (e.g. "" => "(Any)")
if($value === '' && ($this->value === '' || $this->value === null)) {
$selected = 'selected';
} else {
// Normal value from the source
if($value) {
$selected = ($value == $this->value) ? 'selected' : null; $selected = ($value == $this->value) ? 'selected' : null;
if($selected && $this->value != 0) { } else {
$this->isSelected = true; // Do a type check comparison, we might have an array key of 0
$selected = ($value === $this->value) ? 'selected' : null;
}
$this->isSelected = ($selected) ? true : false;
} }
$options .= $this->createTag( $options .= $this->createTag(

View File

@ -1,13 +1,6 @@
<?php <?php
/** /**
* Routines for DNS to country resolution * Routines for IP to country resolution.
*
* - convert address (either ip or domainname) to country.
* - returns false if IP address not found / not known;
* - otherwise an array
* - set $codeOnly to true if you just want the country code
* - give a default for IP
* *
* @package sapphire * @package sapphire
* @subpackage misc * @subpackage misc
@ -272,17 +265,22 @@ class Geoip extends Object {
); );
/** /**
* Returns the country code of the person given an IP. * Find the country for an IP address.
* *
* @param address - The IP address of the user, * By default, it will return an array, keyed by
* @param codeOnly - Returns just the code of the IP instead of array( $CountryCode => $CountryName) * the country code with a value of the country
* name.
*
* To return the code only, pass in true for the
* $codeOnly parameter.
*
* @param string $address The IP address to get the country of
* @param boolean $codeOnly Returns just the country code
*/ */
static function ip2country($address, $codeOnly = false) { static function ip2country($address, $codeOnly = false) {
// Return if in CLI, or you'll get this error: "sh: geoiplookup: command not found"
if(Director::is_cli()) return false;
// Detect internal networks - this is us, so we're NZ
if(substr($address, 0, 7) == "192.168" || substr($address,0,4) == "127.") {
$code = 'NZ';
} else {
$cmd = 'geoiplookup ' . escapeshellarg($address); $cmd = 'geoiplookup ' . escapeshellarg($address);
exec($cmd, $result, $code); exec($cmd, $result, $code);
// Note: At time of writing, $result is always zero for this program // Note: At time of writing, $result is always zero for this program
@ -298,11 +296,13 @@ class Geoip extends Object {
$start = strpos($country, ':'); $start = strpos($country, ':');
if($start) $start += 2; if($start) $start += 2;
$code = substr($country, $start, 2); // skip space $code = substr($country, $start, 2); // skip space
}
if($code == 'IP' || $code == '--') { if($code == 'IP' || $code == '--') {
if(self::$default_country_code) $code = self::$default_country_code; if(self::$default_country_code) {
else return false; $code = self::$default_country_code;
} else {
return false;
}
} }
if(!$codeOnly) { if(!$codeOnly) {
@ -315,7 +315,6 @@ class Geoip extends Object {
} }
} }
/** /**
* Returns the country code, for the current visitor * Returns the country code, for the current visitor
*/ */
@ -378,5 +377,4 @@ class Geoip extends Object {
return $dropdown; return $dropdown;
} }
} }
?> ?>

View File

@ -240,6 +240,23 @@ class Member extends DataObject {
$this->extend('memberLoggedIn'); $this->extend('memberLoggedIn');
} }
/**
* Check if the member ID logged in session actually
* has a database record of the same ID. If there is
* no logged in user, FALSE is returned anyway.
*
* @return boolean TRUE record found FALSE no record found
*/
static function logged_in_session_exists() {
if($id = Member::currentUserID()) {
if($member = DataObject::get_by_id('Member', $id)) {
if($member->exists()) return true;
}
}
return false;
}
/** /**
* Log the user in if the "remember login" cookie is set * Log the user in if the "remember login" cookie is set
* *

View File

@ -43,16 +43,7 @@ class MemberLoginForm extends LoginForm {
$backURL = Session::get('BackURL'); $backURL = Session::get('BackURL');
} }
// We assume if session is storing a member ID, that member exists in the DB if($checkCurrentUser && Member::currentUserID() && Member::logged_in_session_exists()) {
$sessMemberExistsInDB = true;
if($sessionMemberID = Member::currentUserID()) {
$sessMemberInDB = DataObject::get_by_id('Member', $sessionMemberID);
if(!($sessMemberInDB && $sessMemberInDB->exists())) {
$sessMemberExistsInDB = false;
}
}
if($checkCurrentUser && Member::currentUserID() && $sessMemberExistsInDB) {
$fields = new FieldSet( $fields = new FieldSet(
new HiddenField("AuthenticationMethod", null, $this->authenticator_class, $this) new HiddenField("AuthenticationMethod", null, $this->authenticator_class, $this)
); );

View File

@ -1,31 +1,39 @@
<?php <?php
/**
class ErrorPageTest extends SapphireTest { * @package sapphire
* @subpackage tests
*/
class ErrorPageTest extends FunctionalTest {
static $fixture_file = 'sapphire/tests/ErrorPageTest.yml'; static $fixture_file = 'sapphire/tests/ErrorPageTest.yml';
function test404ErrorPage() { function test404ErrorPage() {
$errorPage = DataObject::get_one('ErrorPage', "ErrorCode = '404'"); $page = $this->objFromFixture('ErrorPage', '404');
/* We have an ErrorPage object to use */ /* The page is an instance of ErrorPage */
$this->assertTrue($errorPage instanceof ErrorPage); $this->assertTrue($page instanceof ErrorPage, 'The page is an instance of ErrorPage');
/* Test the URL of the error page out to get a response */ $response = $this->get($page->URLSegment);
$response = Director::test(Director::makeRelative($errorPage->Link()));
/* We have an HTTPResponse object for the error page */
$this->assertTrue($response instanceof HTTPResponse);
/* We have body text from the error page */ /* We have body text from the error page */
$this->assertTrue($response->getBody() != null); $this->assertNotNull($response->getBody(), 'We have body text from the error page');
/* Status code of the HTTPResponse for error page is "404" */ /* Status code of the HTTPResponse for error page is "404" */
$this->assertTrue($response->getStatusCode() == '404'); $this->assertEquals($response->getStatusCode(), '404', 'Status cod eof the HTTPResponse for error page is "404"');
/* Status message of the HTTPResponse for error page is "Not Found" */ /* Status message of the HTTPResponse for error page is "Not Found" */
$this->assertTrue($response->getStatusDescription() == 'Not Found'); $this->assertEquals($response->getStatusDescription(), 'Not Found', 'Status message of the HTTResponse for error page is "Not found"');
}
function testBehaviourOfShowInMenuAndShowInSearchFlags() {
$page = $this->objFromFixture('ErrorPage', '404');
/* Don't show the error page in the menus */
$this->assertEquals($page->ShowInMenus, 0, 'Don\'t show the error page in the menus');
/* Don't show the error page in the search */
$this->assertEquals($page->ShowInSearch, 0, 'Don\'t show the error page in search');
} }
} }
?> ?>

View File

@ -119,4 +119,3 @@ class ObjectStaticTest_Combined3 extends ObjectStaticTest_Combined2 {
public static $first = array('test_3'); public static $first = array('test_3');
public static $second = array('test_3'); public static $second = array('test_3');
} }
/**#@-*/

View File

@ -0,0 +1,123 @@
<?php
/**
* @package sapphire
* @subpackage tests
*/
class CheckboxFieldTest extends SapphireTest {
function testFieldValueTrue() {
/* Create the field, and set the value as boolean true */
$field = new CheckboxField('IsChecked', 'Checked');
$field->setValue(true);
/* dataValue() for the field is 1 */
$this->assertEquals($field->dataValue(), 1, 'dataValue() returns a 1');
/* Value() returns 1 as well */
$this->assertEquals($field->Value(), 1, 'Value() returns a 1');
}
function testFieldValueString() {
/* Create the field, and set the value as "on" (raw request field value from DOM) */
$field = new CheckboxField('IsChecked', 'Checked');
$field->setValue('on');
/* dataValue() for the field is 1 */
$this->assertEquals($field->dataValue(), 1, 'dataValue() returns a 1');
/* Value() returns 1 as well */
$this->assertEquals($field->Value(), 1, 'Value() returns a 1');
}
function testFieldValueSettingNull() {
/* Create the field, and set the value as NULL */
$field = new CheckboxField('IsChecked', 'Checked');
$field->setValue(null);
/* dataValue() for the field is 0 */
$this->assertEquals($field->dataValue(), 0, 'dataValue() returns a 0');
/* Value() returns 0 as well */
$this->assertEquals($field->Value(), 0, 'Value() returns a 0');
}
function testFieldValueSettingFalse() {
/* Create the field, and set the value as NULL */
$field = new CheckboxField('IsChecked', 'Checked');
$field->setValue(false);
/* dataValue() for the field is 0 */
$this->assertEquals($field->dataValue(), 0, 'dataValue() returns a 0');
/* Value() returns 0 as well */
$this->assertEquals($field->Value(), 0, 'Value() returns a 0');
}
function testFieldValueWithoutSettingValue() {
/* Create the field, but don't set any value on it */
$field = new CheckboxField('IsChecked', 'Checked');
/* dataValue() for the field is 0 */
$this->assertEquals($field->dataValue(), 0, 'dataValue() returns a 0');
/* Value() returns 0 as well */
$this->assertEquals($field->Value(), 0, 'Value() returns a 0');
}
function testSavingChecked() {
/* Create a new test data record */
$article = new CheckboxFieldTest_Article();
/* Create a field, with a value of 1 */
$field = new CheckboxField('IsChecked', 'Checked', 1);
/* Save the field into our Article object */
$field->saveInto($article);
/* Write the record to the test database */
$article->write();
/* Check that IsChecked column contains a 1 */
$this->assertEquals(
DB::query("SELECT IsChecked FROM CheckboxFieldTest_Article")->value(),
1,
'We have a 1 set in the database, because the field saved into as a 1'
);
/* Delete the record we tested */
$article->delete();
}
function testSavingUnchecked() {
/* Create a new test data record */
$article = new CheckboxFieldTest_Article();
/* Create a field, with no value */
$field = new CheckboxField('IsChecked', 'Checked');
/* Save the field into our Article object */
$field->saveInto($article);
/* Write the record to the test database */
$article->write();
/* Check that IsChecked column contains a 0 */
$this->assertEquals(
DB::query("SELECT IsChecked FROM CheckboxFieldTest_Article")->value(),
0,
'We have a 0 set in the database, because the field saved into as a 0'
);
/* Delete the record we tested */
$article->delete();
}
}
class CheckboxFieldTest_Article extends DataObject implements TestOnly {
public static $db = array(
'IsChecked' => 'Boolean'
);
}
?>

View File

@ -1,5 +1,8 @@
<?php <?php
/**
* @package sapphire
* @subpackage tests
*/
class CheckboxSetFieldTest extends SapphireTest { class CheckboxSetFieldTest extends SapphireTest {
static $fixture_file = 'sapphire/tests/forms/CheckboxSetFieldTest.yml'; static $fixture_file = 'sapphire/tests/forms/CheckboxSetFieldTest.yml';

View File

@ -10,7 +10,7 @@ class DropdownFieldTest extends SapphireTest {
$dropdownField = new DropdownField('FeelingOk', 'Are you feeling ok?', array(0 => 'No', 1 => 'Yes'), '', null, '(Select one)'); $dropdownField = new DropdownField('FeelingOk', 'Are you feeling ok?', array(0 => 'No', 1 => 'Yes'), '', null, '(Select one)');
$dropdownField->addExtraClass('thisIsMyExtraClassForDropdownField'); $dropdownField->addExtraClass('thisIsMyExtraClassForDropdownField');
preg_match('/thisIsMyExtraClassForDropdownField/', $dropdownField->Field(), $matches); preg_match('/thisIsMyExtraClassForDropdownField/', $dropdownField->Field(), $matches);
$this->assertTrue($matches[0] == 'thisIsMyExtraClassForDropdownField'); $this->assertEquals($matches[0], 'thisIsMyExtraClassForDropdownField');
} }
function testGetSource() { function testGetSource() {
@ -89,5 +89,127 @@ class DropdownFieldTest extends SapphireTest {
) )
); );
} }
function testNumberOfSelectOptionsAvailable() {
/* Create a field with a blank value */
$field = $this->testDropdownField('(Any)');
/* 3 options are available */
$this->assertEquals(count($this->findOptionElements($field->Field())), 3, '3 options are available');
/* There is one selected option, since a dropdown can only possibly have one! */
$selectedOptions = $this->findSelectedOptionElements($field->Field());
$this->assertEquals(count($selectedOptions), 1, 'We only have 1 selected option, since a dropdown can only possibly have one!');
/* Create a field without a blank value */
$field = $this->testDropdownField();
/* 2 options are available */
$this->assertEquals(count($this->findOptionElements($field->Field())), 2, '2 options are available');
/* There are no selected options */
$selectedOptions = $this->findSelectedOptionElements($field->Field());
$this->assertEquals(count($selectedOptions), 0, 'There are no selected options');
}
function testIntegerZeroValueSeelctedOptionBehaviour() {
$field = $this->testDropdownField('(Any)', 0);
$selectedOptions = $this->findSelectedOptionElements($field->Field());
/* The selected option is "No" */
$this->assertEquals((string) $selectedOptions[0], 'No', 'The selected option is "No"');
}
function testBlankStringValueSelectedOptionBehaviour() {
$field = $this->testDropdownField('(Any)');
$selectedOptions = $this->findSelectedOptionElements($field->Field());
/* The selected option is "(Any)" */
$this->assertEquals((string) $selectedOptions[0], '(Any)', 'The selected option is "(Any)"');
}
function testNullValueSelectedOptionBehaviour() {
$field = $this->testDropdownField('(Any)', null);
$selectedOptions = $this->findSelectedOptionElements($field->Field());
/* The selected option is "(Any)" */
$this->assertEquals((string) $selectedOptions[0], '(Any)', 'The selected option is "(Any)"');
}
function testStringValueSelectedOptionBehaviour() {
$field = $this->testDropdownField('(Any)', '1');
$selectedOptions = $this->findSelectedOptionElements($field->Field());
/* The selected option is "Yes" */
$this->assertEquals((string) $selectedOptions[0], 'Yes', 'The selected option is "Yes"');
$field->setSource(array(
'Cats' => 'Cats and Kittens',
'Dogs' => 'Dogs and Puppies'
));
$field->setValue('Cats');
$selectedOptions = $this->findSelectedOptionElements($field->Field());
/* The selected option is "Cats and Kittens" */
$this->assertEquals((string) $selectedOptions[0], 'Cats and Kittens', 'The selected option is "Cats and Kittens"');
}
/**
* Create a test dropdown field, with the option to
* set what source and blank value it should contain
* as optional parameters.
*
* @param string|null $emptyString The text to display for the empty value
* @param string|integer $value The default value of the field
* @return DropdownField object
*/
function testDropdownField($emptyString = null, $value = '') {
/* Set up source, with 0 and 1 integers as the values */
$source = array(
0 => 'No',
1 => 'Yes'
);
return new DropdownField('Field', null, $source, $value, null, $emptyString);
}
/**
* Find all the <OPTION> elements from a
* string of HTML.
*
* @param string $html HTML to scan for elements
* @return SimpleXMLElement
*/
function findOptionElements($html) {
$parser = new CSSContentParser($html);
return $parser->getBySelector('option');
}
/**
* Find all the <OPTION> elements from a
* string of HTML that have the "selected"
* attribute.
*
* @param string $html HTML to parse for elements
* @return array of SimpleXMLElement objects
*/
function findSelectedOptionElements($html) {
$options = $this->findOptionElements($html);
/* Find any elements that have the "selected" attribute and put them into a list */
$foundSelected = array();
foreach($options as $option) {
$attributes = $option->attributes();
if($attributes) foreach($attributes as $attribute => $value) {
if($attribute == 'selected') {
$foundSelected[] = $option;
}
}
}
return $foundSelected;
}
} }
?> ?>

View File

@ -27,6 +27,8 @@ class GroupTest extends FunctionalTest {
} }
function testMemberGroupRelationForm() { function testMemberGroupRelationForm() {
Session::set('loggedInAs', $this->idFromFixture('GroupTest_Member', 'admin'));
$adminGroup = $this->fixture->objFromFixture('Group', 'admingroup'); $adminGroup = $this->fixture->objFromFixture('Group', 'admingroup');
$parentGroup = $this->fixture->objFromFixture('Group', 'parentgroup'); $parentGroup = $this->fixture->objFromFixture('Group', 'parentgroup');
$childGroup = $this->fixture->objFromFixture('Group', 'childgroup'); $childGroup = $this->fixture->objFromFixture('Group', 'childgroup');
@ -42,6 +44,7 @@ class GroupTest extends FunctionalTest {
)); ));
$form->saveInto($member); $form->saveInto($member);
$updatedGroups = $member->Groups(); $updatedGroups = $member->Groups();
$controlGroups = new Member_GroupSet( $controlGroups = new Member_GroupSet(
$adminGroup, $adminGroup,
$parentGroup $parentGroup

View File

@ -23,3 +23,7 @@ GroupTest_Member:
allgroupuser: allgroupuser:
FirstName: All Group User FirstName: All Group User
Groups: =>Group.admingroup,=>Group.parentgroup,=>Group.childgroup Groups: =>Group.admingroup,=>Group.parentgroup,=>Group.childgroup
Permission:
admincode:
Code: ADMIN
Group: =>Group.admingroup