2007-07-19 10:40:28 +00:00
< ? php
/**
* Base class for all fields that contain other fields .
* Implements sequentialisation - so that when we ' re saving / loading data , we can populate
* a tabbed form properly . All of the children are stored in $this -> children
2008-01-09 04:18:36 +00:00
* @ package forms
* @ subpackage fields - structural
2007-07-19 10:40:28 +00:00
*/
class CompositeField extends FormField {
2008-10-03 15:56:14 +00:00
/**
2011-05-11 17:51:54 +10:00
* @ var FieldList
2008-10-03 15:56:14 +00:00
*/
2007-07-19 10:40:28 +00:00
protected $children ;
2008-10-03 15:56:14 +00:00
2007-07-19 10:40:28 +00:00
/**
* Set to true when this field is a readonly field
*/
protected $readonly ;
/**
* @ var $columnCount int Toggle different css - rendering for multiple columns
* ( " onecolumn " , " twocolumns " , " threecolumns " ) . The content is determined
* by the $children - array , so wrap all items you want to have grouped in a
* column inside a CompositeField .
* Caution : Please make sure that this variable actually matches the
* count of your $children .
*/
protected $columnCount = null ;
public function __construct ( $children = null ) {
2011-05-11 17:51:54 +10:00
if ( $children instanceof FieldList ) {
2007-07-19 10:40:28 +00:00
$this -> children = $children ;
} elseif ( is_array ( $children )) {
2011-05-11 17:51:54 +10:00
$this -> children = new FieldList ( $children );
2007-07-19 10:40:28 +00:00
} else {
$children = is_array ( func_get_args ()) ? func_get_args () : array ();
2011-05-11 17:51:54 +10:00
$this -> children = new FieldList ( $children );
2007-07-19 10:40:28 +00:00
}
2008-09-11 00:02:49 +00:00
$this -> children -> setContainerField ( $this );
2008-08-20 04:41:55 +00:00
2009-07-06 21:48:12 +00:00
// Skipping FormField::__construct(), but we have to make sure this
// doesn't count as a broken constructor
$this -> brokenOnConstruct = false ;
2007-07-19 10:40:28 +00:00
Object :: __construct ();
}
/**
2011-05-11 17:51:54 +10:00
* Returns all the sub - fields , suitable for <% control FieldList %>
2007-07-19 10:40:28 +00:00
*/
2011-05-11 17:51:54 +10:00
public function FieldList () {
2007-07-19 10:40:28 +00:00
return $this -> children ;
}
2011-05-11 17:51:54 +10:00
/**
* @ deprecated 3.0 Please use { @ link FieldList ()} .
*/
public function FieldSet () {
2011-10-29 12:02:11 +13:00
Deprecation :: notice ( '3.0' , 'Use FieldList() instead.' );
2011-05-11 17:51:54 +10:00
return $this -> FieldList ();
}
2007-07-19 10:40:28 +00:00
public function setID ( $id ) {
$this -> id = $id ;
}
2007-10-18 01:05:52 +00:00
public function Field () {
return $this -> FieldHolder ();
}
2008-08-20 04:41:55 +00:00
/**
* Accessor method for $this -> children
2011-05-11 17:51:54 +10:00
* @ return FieldList
2008-08-20 04:41:55 +00:00
*/
public function getChildren () {
return $this -> children ;
}
2008-12-04 22:38:32 +00:00
/**
2011-05-11 17:51:54 +10:00
* @ param FieldList $children
2008-12-04 22:38:32 +00:00
*/
public function setChildren ( $children ) {
$this -> children = $children ;
}
2007-07-19 10:40:28 +00:00
2011-12-22 13:10:57 +01:00
function extraClasses () {
$classes = array ( 'field' , 'CompositeField' , parent :: extraClasses ());
if ( $this -> columnCount ) $classes [] = 'multicolumn' ;
return implode ( ' ' , $classes );
}
function getAttributes () {
return array_merge (
parent :: getAttributes (),
array ( 'tabindex' => null , 'type' => null , 'value' => null , 'type' => null )
);
}
2007-07-19 10:40:28 +00:00
/**
* Returns the fields nested inside another DIV
*/
function FieldHolder () {
2011-12-22 13:10:57 +01:00
$content = '' ;
2011-05-11 17:51:54 +10:00
$fs = $this -> FieldList ();
2007-07-19 10:40:28 +00:00
foreach ( $fs as $subfield ) {
if ( $this -> columnCount ) {
$className = " column { $this -> columnCount } " ;
if ( ! next ( $fs )) $className .= " lastcolumn " ;
2007-10-18 01:05:52 +00:00
$content .= " \n <div class= \" { $className } \" > \n " . $subfield -> FieldHolder () . " \n </div> \n " ;
2007-07-19 10:40:28 +00:00
} else if ( $subfield ){
2007-10-18 01:05:52 +00:00
$content .= " \n " . $subfield -> FieldHolder () . " \n " ;
2007-07-19 10:40:28 +00:00
}
}
2012-01-02 21:32:13 +01:00
return $this -> createTag ( 'div' , $this -> getAttributes (), $content );
2007-09-16 16:05:05 +00:00
}
/**
* Returns the fields in the restricted field holder inside a DIV .
*/
function SmallFieldHolder () { //return $this->FieldHolder();
2011-05-11 17:51:54 +10:00
$fs = $this -> FieldList ();
2007-09-16 16:05:05 +00:00
$idAtt = isset ( $this -> id ) ? " id= \" { $this -> id } \" " : '' ;
$className = ( $this -> columnCount ) ? " field CompositeField { $this -> extraClass () } multicolumn " : " field CompositeField { $this -> extraClass () } " ;
$content = " <div class= \" $className\ " $idAtt > " ;
2011-10-29 13:07:40 +02:00
foreach ( $fs as $subfield ) { //echo ' subf'.$subfield->getName();
2007-09-16 16:05:05 +00:00
if ( $this -> columnCount ) {
$className = " column { $this -> columnCount } " ;
if ( ! next ( $fs )) $className .= " lastcolumn " ;
$content .= " <div class= \" { $className } \" > " . $subfield -> FieldHolder () . " </div> " ;
} else if ( $subfield ){
$content .= $subfield -> SmallFieldHolder () . " " ;
}
}
$content .= " </div> " ;
2007-07-19 10:40:28 +00:00
2007-09-16 16:05:05 +00:00
return $content ;
}
2007-07-19 10:40:28 +00:00
/**
* Add all of the non - composite fields contained within this field to the list .
* Sequentialisation is used when connecting the form to its data source
*/
2008-08-12 02:58:48 +00:00
public function collateDataFields ( & $list , $saveableOnly = false ) {
2007-07-19 10:40:28 +00:00
foreach ( $this -> children as $field ) {
if ( is_object ( $field )) {
2008-08-12 02:58:48 +00:00
if ( $field -> isComposite ()) $field -> collateDataFields ( $list , $saveableOnly );
if ( $saveableOnly ) {
$isIncluded = ( $field -> hasData () && ! $field -> isReadonly () && ! $field -> isDisabled ());
} else {
$isIncluded = ( $field -> hasData ());
}
if ( $isIncluded ) {
2011-10-29 17:01:52 +13:00
$name = $field -> getName ();
2007-07-19 10:40:28 +00:00
if ( $name ) {
$formName = ( isset ( $this -> form )) ? $this -> form -> FormName () : '(unknown form)' ;
if ( isset ( $list [ $name ])) user_error ( " collateDataFields() I noticed that a field called ' $name ' appears twice in your form: ' { $formName } '. One is a ' { $field -> class } ' and the other is a ' { $list [ $name ] -> class } ' " , E_USER_ERROR );
$list [ $name ] = $field ;
}
}
}
}
}
function setForm ( $form ) {
foreach ( $this -> children as $f ) if ( is_object ( $f )) $f -> setForm ( $form );
parent :: setForm ( $form );
}
function setColumnCount ( $columnCount ) {
$this -> columnCount = $columnCount ;
}
function isComposite () { return true ; }
function hasData () { return false ; }
2008-01-06 21:55:27 +00:00
public function fieldByName ( $name ) {
return $this -> children -> fieldByName ( $name );
}
2007-07-19 10:40:28 +00:00
/**
* Add a new child field to the end of the set .
*/
public function push ( FormField $field ) {
$this -> children -> push ( $field );
}
2008-11-11 02:35:54 +00:00
/**
2011-05-11 17:51:54 +10:00
* @ uses FieldList -> insertBefore ()
2008-11-11 02:35:54 +00:00
*/
2007-07-19 10:40:28 +00:00
public function insertBefore ( $field , $insertBefore ) {
2008-11-11 02:35:54 +00:00
$ret = $this -> children -> insertBefore ( $field , $insertBefore );
$this -> sequentialSet = null ;
return $ret ;
2007-09-16 16:05:05 +00:00
}
2008-11-11 02:35:54 +00:00
public function insertAfter ( $field , $insertAfter ) {
$ret = $this -> children -> insertAfter ( $field , $insertAfter );
$this -> sequentialSet = null ;
return $ret ;
}
2008-09-12 04:42:24 +00:00
/**
* Remove a field from this CompositeField by Name .
* The field could also be inside a CompositeField .
*
* @ param string $fieldName The name of the field
* @ param boolean $dataFieldOnly If this is true , then a field will only
* be removed if it ' s a data field . Dataless fields , such as tabs , will
* be left as - is .
*/
public function removeByName ( $fieldName , $dataFieldOnly = false ) {
$this -> children -> removeByName ( $fieldName , $dataFieldOnly );
2007-07-19 10:40:28 +00:00
}
public function replaceField ( $fieldName , $newField ) {
return $this -> children -> replaceField ( $fieldName , $newField );
}
2008-09-11 06:24:40 +00:00
function rootFieldSet () {
if ( is_object ( $this -> containerFieldSet )) return $this -> containerFieldSet -> rootFieldSet ();
else return $this -> children ;
}
2007-07-19 10:40:28 +00:00
/**
* Return a readonly version of this field . Keeps the composition but returns readonly
* versions of all the children
*/
public function performReadonlyTransformation () {
2011-05-11 17:51:54 +10:00
$newChildren = new FieldList ();
2008-12-04 22:38:32 +00:00
$clone = clone $this ;
foreach ( $clone -> getChildren () as $idx => $child ) {
2007-07-19 10:40:28 +00:00
if ( is_object ( $child )) $child = $child -> transform ( new ReadonlyTransformation ());
$newChildren -> push ( $child , $idx );
}
2008-12-04 22:38:32 +00:00
$clone -> children = $newChildren ;
$clone -> readonly = true ;
return $clone ;
2007-07-19 10:40:28 +00:00
}
/**
* Return a readonly version of this field . Keeps the composition but returns readonly
* versions of all the children
*/
public function performDisabledTransformation ( $trans ) {
2011-05-11 17:51:54 +10:00
$newChildren = new FieldList ();
2008-12-04 22:38:32 +00:00
$clone = clone $this ;
if ( $clone -> getChildren ()) foreach ( $clone -> getChildren () as $idx => $child ) {
2007-07-19 10:40:28 +00:00
if ( is_object ( $child )) {
$child = $child -> transform ( $trans );
}
$newChildren -> push ( $child , $idx );
}
2008-12-04 22:38:32 +00:00
$clone -> children = $newChildren ;
$clone -> readonly = true ;
2007-07-19 10:40:28 +00:00
2008-12-04 22:38:32 +00:00
return $clone ;
2007-07-19 10:40:28 +00:00
}
2007-11-23 01:10:19 +00:00
2007-07-19 10:40:28 +00:00
function IsReadonly () {
return $this -> readonly ;
}
2008-11-11 02:35:54 +00:00
/**
* Find the numerical position of a field within
* the children collection . Doesn ' t work recursively .
*
* @ param string | FormField
* @ return Position in children collection ( first position starts with 0 ) . Returns FALSE if the field can ' t be found .
*/
function fieldPosition ( $field ) {
if ( is_string ( $field )) $field = $this -> fieldByName ( $field );
if ( ! $field ) return false ;
$i = 0 ;
foreach ( $this -> children as $child ) {
2011-10-29 17:01:52 +13:00
if ( $child -> getName () == $field -> getName ()) return $i ;
2008-11-11 02:35:54 +00:00
$i ++ ;
}
return false ;
}
2008-12-04 22:38:32 +00:00
/**
* Transform the named field into a readonly feld .
*
* @ param string | FormField
*/
function makeFieldReadonly ( $field ) {
2011-10-29 17:01:52 +13:00
$fieldName = ( $field instanceof FormField ) ? $field -> getName () : $field ;
2008-12-04 22:38:32 +00:00
// Iterate on items, looking for the applicable field
foreach ( $this -> children as $i => $item ) {
if ( $item -> isComposite ()) {
$item -> makeFieldReadonly ( $fieldName );
} else {
// Once it's found, use FormField::transform to turn the field into a readonly version of itself.
2011-10-29 17:01:52 +13:00
if ( $item -> getName () == $fieldName ) {
2008-12-04 22:38:32 +00:00
$this -> children -> replaceField ( $fieldName , $item -> transform ( new ReadonlyTransformation ()));
// Clear an internal cache
$this -> sequentialSet = null ;
// A true results indicates that the field was foudn
return true ;
}
}
}
return false ;
}
2007-07-19 10:40:28 +00:00
function debug () {
2007-09-14 18:05:21 +00:00
$result = " $this->class ( $this->name ) <ul> " ;
2007-07-19 10:40:28 +00:00
foreach ( $this -> children as $child ) {
$result .= " <li> " . Debug :: text ( $child ) . " </li> " ;
}
$result .= " </ul> " ;
return $result ;
}
2007-08-23 05:47:54 +00:00
function validate ( $validator ){
$valid = true ;
foreach ( $this -> children as $idx => $child ){
2008-10-08 02:00:12 +00:00
$valid = ( $child && $child -> validate ( $validator ) && $valid );
2007-08-23 05:47:54 +00:00
}
return $valid ;
}
2007-07-19 10:40:28 +00:00
}
?>