* @link http://www.pradosoft.com/
* @copyright Copyright © 2005-2014 PradoSoft
* @license http://www.pradosoft.com/license/
* @package Prado\Web\UI\WebControls
*/
namespace Prado\Web\UI\WebControls;
/**
* Includes TBaseDataList class
*/
Prado::using('System.Web.UI.WebControls.TBaseDataList');
/**
* Includes TRepeatInfo class
*/
Prado::using('System.Web.UI.WebControls.TRepeatInfo');
/**
* TDataList class
*
* TDataList represents a data bound and updatable list control.
*
* Like {@link TRepeater}, TDataList displays its content repeatedly based on
* the data fetched from {@link setDataSource DataSource}.
* The repeated contents in TDataList are called items, which are controls and
* can be accessed through {@link getItems Items}. When {@link dataBind()} is
* invoked, TDataList creates an item for each row of data and binds the data
* row to the item. Optionally, a TDataList can have a header, a footer and/or
* separators between items.
*
* TDataList differs from {@link TRepeater} in that it supports tiling the items
* in different manners and it maintains status of items to handle data update.
*
* The layout of the repeated contents are specified by inline templates.
* TDataList items, header, footer, etc. are being instantiated with the corresponding
* templates when data is being bound to the repeater.
*
* Since v3.1.0, the layout can also be by renderers. A renderer is a control class
* that can be instantiated as datalist items, header, etc. A renderer can thus be viewed
* as an external template (in fact, it can also be non-templated controls).
*
* A renderer can be any control class.
* - If the class implements {@link IDataRenderer}, the Data
* property will be set as the data row during databinding. Many PRADO controls
* implement this interface, such as {@link TLabel}, {@link TTextBox}, etc.
* - If the class implements {@link IItemDataRenderer}, the ItemIndex property will be set
* as the zero-based index of the item in the datalist item collection, and
* the ItemType property as the item's type (such as TListItemType::Item).
* {@link TDataListItemRenderer} may be used as the convenient base class which
* already implements {@link IDataItemRenderer}.
*
* The following properties are used to specify different types of template and renderer
* for a datalist:
* - {@link setItemTemplate ItemTemplate}, {@link setItemRenderer ItemRenderer}:
* for each repeated row of data
* - {@link setAlternatingItemTemplate AlternatingItemTemplate}, {@link setAlternatingItemRenderer AlternatingItemRenderer}:
* for each alternating row of data. If not set, {@link setItemTemplate ItemTemplate} or {@link setItemRenderer ItemRenderer}
* will be used instead.
* - {@link setHeaderTemplate HeaderTemplate}, {@link setHeaderRenderer HeaderRenderer}:
* for the datalist header.
* - {@link setFooterTemplate FooterTemplate}, {@link setFooterRenderer FooterRenderer}:
* for the datalist footer.
* - {@link setSeparatorTemplate SeparatorTemplate}, {@link setSeparatorRenderer SeparatorRenderer}:
* for content to be displayed between items.
* - {@link setEmptyTemplate EmptyTemplate}, {@link setEmptyRenderer EmptyRenderer}:
* used when data bound to the datalist is empty.
* - {@link setEditItemTemplate EditItemTemplate}, {@link setEditItemRenderer EditItemRenderer}:
* for the row being editted.
* - {@link setSelectedItemTemplate SelectedItemTemplate}, {@link setSelectedItemRenderer SelectedItemRenderer}:
* for the row being selected.
*
* If a content type is defined with both a template and a renderer, the latter takes precedence.
*
* When {@link dataBind()} is being called, TDataList undergoes the following lifecycles for each row of data:
* - create item based on templates or renderers
* - set the row of data to the item
* - raise {@link onItemCreated OnItemCreated}:
* - add the item as a child control
* - call dataBind() of the item
* - raise {@link onItemDataBound OnItemDataBound}:
*
* TDataList raises an {@link onItemCommand OnItemCommand} whenever a button control
* within some datalist item raises a OnCommand event. Therefore,
* you can handle all sorts of OnCommand event in a central place by
* writing an event handler for {@link onItemCommand OnItemCommand}.
*
* An additional event is raised if the OnCommand event has one of the following
* command names:
* - edit: user wants to edit an item. OnEditCommand event will be raised.
* - update: user wants to save the change to an item. OnUpdateCommand event will be raised.
* - select: user selects an item. OnSelectedIndexChanged event will be raised.
* - delete: user deletes an item. OnDeleteCommand event will be raised.
* - cancel: user cancels previously editting action. OnCancelCommand event will be raised.
*
* TDataList provides a few properties to support tiling the items.
* The number of columns used to display the data items is specified via
* {@link setRepeatColumns RepeatColumns} property, while the {@link setRepeatDirection RepeatDirection}
* governs the order of the items being rendered.
* The layout of the data items in the list is specified via {@link setRepeatLayout RepeatLayout},
* which can take one of the following values:
* - Table (default): items are organized using HTML table and cells.
* When using this layout, one can set {@link setCellPadding CellPadding} and
* {@link setCellSpacing CellSpacing} to adjust the cellpadding and cellpadding
* of the table, and {@link setCaption Caption} and {@link setCaptionAlign CaptionAlign}
* to add a table caption with the specified alignment.
* - Flow: items are organized using HTML spans and breaks.
* - Raw: TDataList does not generate any HTML tags to do the tiling.
*
* Items in TDataList can be in one of the three status: normal browsing,
* being editted and being selected. To change the status of a particular
* item, set {@link setSelectedItemIndex SelectedItemIndex} or
* {@link setEditItemIndex EditItemIndex}. The former will change
* the indicated item to selected mode, which will cause the item to
* use {@link setSelectedItemTemplate SelectedItemTemplate} or
* {@link setSelectedItemRenderer SelectedItemRenderer} for presentation.
* The latter will change the indicated item to edit mode and to use corresponding
* template or renderer.
* Note, if an item is in edit mode, then selecting this item will have no effect.
*
* Different styles may be applied to items in different status. The style
* application is performed in a hierarchical way: Style in higher hierarchy
* will inherit from styles in lower hierarchy.
* Starting from the lowest hierarchy, the item styles include
* - item's own style
* - {@link getItemStyle ItemStyle}
* - {@link getAlternatingItemStyle AlternatingItemStyle}
* - {@link getSelectedItemStyle SelectedItemStyle}
* - {@link getEditItemStyle EditItemStyle}.
* Therefore, if background color is set as red in {@link getItemStyle ItemStyle},
* {@link getEditItemStyle EditItemStyle} will also have red background color
* unless it is set to a different value explicitly.
*
* When a page containing a datalist is post back, the datalist will restore automatically
* all its contents, including items, header, footer and separators.
* However, the data row associated with each item will not be recovered and become null.
* To access the data, use one of the following ways:
* - Use {@link getDataKeys DataKeys} to obtain the data key associated with
* the specified datalist item and use the key to fetch the corresponding data
* from some persistent storage such as DB.
* - Save the whole dataset in viewstate, which will restore the dataset automatically upon postback.
* Be aware though, if the size of your dataset is big, your page size will become big. Some
* complex data may also have serializing problem if saved in viewstate.
*
* @author Qiang Xue
* @package Prado\Web\UI\WebControls
* @since 3.0
*/
class TDataList extends TBaseDataList implements INamingContainer, IRepeatInfoUser
{
/**
* Command name that TDataList understands. They are case-insensitive.
*/
const CMD_SELECT='Select';
const CMD_EDIT='Edit';
const CMD_UPDATE='Update';
const CMD_DELETE='Delete';
const CMD_CANCEL='Cancel';
/**
* @var TDataListItemCollection item list
*/
private $_items=null;
/**
* @var Itemplate various item templates
*/
private $_itemTemplate=null;
private $_emptyTemplate=null;
private $_alternatingItemTemplate=null;
private $_selectedItemTemplate=null;
private $_editItemTemplate=null;
private $_headerTemplate=null;
private $_footerTemplate=null;
private $_separatorTemplate=null;
/**
* @var TControl header item
*/
private $_header=null;
/**
* @var TControl footer item
*/
private $_footer=null;
/**
* @return TDataListItemCollection item list
*/
public function getItems()
{
if(!$this->_items)
$this->_items=new TDataListItemCollection;
return $this->_items;
}
/**
* @return integer number of items
*/
public function getItemCount()
{
return $this->_items?$this->_items->getCount():0;
}
/**
* @return string the class name for datalist items. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getItemRenderer()
{
return $this->getViewState('ItemRenderer','');
}
/**
* Sets the item renderer class.
*
* If not empty, the class will be used to instantiate as datalist items.
* This property takes precedence over {@link getItemTemplate ItemTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setItemTemplate
* @since 3.1.0
*/
public function setItemRenderer($value)
{
$this->setViewState('ItemRenderer',$value,'');
}
/**
* @return string the class name for alternative datalist items. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getAlternatingItemRenderer()
{
return $this->getViewState('AlternatingItemRenderer','');
}
/**
* Sets the alternative item renderer class.
*
* If not empty, the class will be used to instantiate as alternative datalist items.
* This property takes precedence over {@link getAlternatingItemTemplate AlternatingItemTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setAlternatingItemTemplate
* @since 3.1.0
*/
public function setAlternatingItemRenderer($value)
{
$this->setViewState('AlternatingItemRenderer',$value,'');
}
/**
* @return string the class name for the datalist item being editted. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getEditItemRenderer()
{
return $this->getViewState('EditItemRenderer','');
}
/**
* Sets the renderer class for the datalist item being editted.
*
* If not empty, the class will be used to instantiate as the datalist item.
* This property takes precedence over {@link getEditItemTemplate EditItemTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setEditItemTemplate
* @since 3.1.0
*/
public function setEditItemRenderer($value)
{
$this->setViewState('EditItemRenderer',$value,'');
}
/**
* @return string the class name for the datalist item being selected. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getSelectedItemRenderer()
{
return $this->getViewState('SelectedItemRenderer','');
}
/**
* Sets the renderer class for the datalist item being selected.
*
* If not empty, the class will be used to instantiate as the datalist item.
* This property takes precedence over {@link getSelectedItemTemplate SelectedItemTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setSelectedItemTemplate
* @since 3.1.0
*/
public function setSelectedItemRenderer($value)
{
$this->setViewState('SelectedItemRenderer',$value,'');
}
/**
* @return string the class name for datalist item separators. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getSeparatorRenderer()
{
return $this->getViewState('SeparatorRenderer','');
}
/**
* Sets the datalist item separator renderer class.
*
* If not empty, the class will be used to instantiate as datalist item separators.
* This property takes precedence over {@link getSeparatorTemplate SeparatorTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setSeparatorTemplate
* @since 3.1.0
*/
public function setSeparatorRenderer($value)
{
$this->setViewState('SeparatorRenderer',$value,'');
}
/**
* @return string the class name for datalist header item. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getHeaderRenderer()
{
return $this->getViewState('HeaderRenderer','');
}
/**
* Sets the datalist header renderer class.
*
* If not empty, the class will be used to instantiate as datalist header item.
* This property takes precedence over {@link getHeaderTemplate HeaderTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setHeaderTemplate
* @since 3.1.0
*/
public function setHeaderRenderer($value)
{
$this->setViewState('HeaderRenderer',$value,'');
}
/**
* @return string the class name for datalist footer item. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getFooterRenderer()
{
return $this->getViewState('FooterRenderer','');
}
/**
* Sets the datalist footer renderer class.
*
* If not empty, the class will be used to instantiate as datalist footer item.
* This property takes precedence over {@link getFooterTemplate FooterTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setFooterTemplate
* @since 3.1.0
*/
public function setFooterRenderer($value)
{
$this->setViewState('FooterRenderer',$value,'');
}
/**
* @return string the class name for empty datalist item. Defaults to empty, meaning not set.
* @since 3.1.0
*/
public function getEmptyRenderer()
{
return $this->getViewState('EmptyRenderer','');
}
/**
* Sets the datalist empty renderer class.
*
* The empty renderer is created as the child of the datalist
* if data bound to the datalist is empty.
* This property takes precedence over {@link getEmptyTemplate EmptyTemplate}.
*
* @param string the renderer class name in namespace format.
* @see setEmptyTemplate
* @since 3.1.0
*/
public function setEmptyRenderer($value)
{
$this->setViewState('EmptyRenderer',$value,'');
}
/**
* @return ITemplate the template for item
*/
public function getItemTemplate()
{
return $this->_itemTemplate;
}
/**
* @param ITemplate the template for item
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setItemTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_itemTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','ItemTemplate');
}
/**
* @return TTableItemStyle the style for item
*/
public function getItemStyle()
{
if(($style=$this->getViewState('ItemStyle',null))===null)
{
$style=new TTableItemStyle;
$this->setViewState('ItemStyle',$style,null);
}
return $style;
}
/**
* @return ITemplate the template for each alternating item
*/
public function getAlternatingItemTemplate()
{
return $this->_alternatingItemTemplate;
}
/**
* @param ITemplate the template for each alternating item
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setAlternatingItemTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_alternatingItemTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','AlternatingItemType');
}
/**
* @return TTableItemStyle the style for each alternating item
*/
public function getAlternatingItemStyle()
{
if(($style=$this->getViewState('AlternatingItemStyle',null))===null)
{
$style=new TTableItemStyle;
$this->setViewState('AlternatingItemStyle',$style,null);
}
return $style;
}
/**
* @return ITemplate the selected item template
*/
public function getSelectedItemTemplate()
{
return $this->_selectedItemTemplate;
}
/**
* @param ITemplate the selected item template
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setSelectedItemTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_selectedItemTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','SelectedItemTemplate');
}
/**
* @return TTableItemStyle the style for selected item
*/
public function getSelectedItemStyle()
{
if(($style=$this->getViewState('SelectedItemStyle',null))===null)
{
$style=new TTableItemStyle;
$this->setViewState('SelectedItemStyle',$style,null);
}
return $style;
}
/**
* @return ITemplate the edit item template
*/
public function getEditItemTemplate()
{
return $this->_editItemTemplate;
}
/**
* @param ITemplate the edit item template
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setEditItemTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_editItemTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','EditItemTemplate');
}
/**
* @return TTableItemStyle the style for edit item
*/
public function getEditItemStyle()
{
if(($style=$this->getViewState('EditItemStyle',null))===null)
{
$style=new TTableItemStyle;
$this->setViewState('EditItemStyle',$style,null);
}
return $style;
}
/**
* @return ITemplate the header template
*/
public function getHeaderTemplate()
{
return $this->_headerTemplate;
}
/**
* @param ITemplate the header template
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setHeaderTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_headerTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','HeaderTemplate');
}
/**
* @return TTableItemStyle the style for header
*/
public function getHeaderStyle()
{
if(($style=$this->getViewState('HeaderStyle',null))===null)
{
$style=new TTableItemStyle;
$this->setViewState('HeaderStyle',$style,null);
}
return $style;
}
/**
* @return TControl the header item
*/
public function getHeader()
{
return $this->_header;
}
/**
* @return ITemplate the footer template
*/
public function getFooterTemplate()
{
return $this->_footerTemplate;
}
/**
* @param ITemplate the footer template
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setFooterTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_footerTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','FooterTemplate');
}
/**
* @return TTableItemStyle the style for footer
*/
public function getFooterStyle()
{
if(($style=$this->getViewState('FooterStyle',null))===null)
{
$style=new TTableItemStyle;
$this->setViewState('FooterStyle',$style,null);
}
return $style;
}
/**
* @return TControl the footer item
*/
public function getFooter()
{
return $this->_footer;
}
/**
* @return ITemplate the template applied when no data is bound to the datalist
*/
public function getEmptyTemplate()
{
return $this->_emptyTemplate;
}
/**
* @param ITemplate the template applied when no data is bound to the datalist
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setEmptyTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_emptyTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','EmptyTemplate');
}
/**
* @return ITemplate the separator template
*/
public function getSeparatorTemplate()
{
return $this->_separatorTemplate;
}
/**
* @param ITemplate the separator template
* @throws TInvalidDataTypeException if the input is not an {@link ITemplate} or not null.
*/
public function setSeparatorTemplate($value)
{
if($value instanceof ITemplate || $value===null)
$this->_separatorTemplate=$value;
else
throw new TInvalidDataTypeException('datalist_template_required','SeparatorTemplate');
}
/**
* @return TTableItemStyle the style for separator
*/
public function getSeparatorStyle()
{
if(($style=$this->getViewState('SeparatorStyle',null))===null)
{
$style=new TTableItemStyle;
$this->setViewState('SeparatorStyle',$style,null);
}
return $style;
}
/**
* @return integer the zero-based index of the selected item in {@link getItems Items}.
* A value -1 means no item selected.
*/
public function getSelectedItemIndex()
{
return $this->getViewState('SelectedItemIndex',-1);
}
/**
* Selects an item by its index in {@link getItems Items}.
* Previously selected item will be un-selected.
* If the item to be selected is already in edit mode, it will remain in edit mode.
* If the index is less than 0, any existing selection will be cleared up.
* @param integer the selected item index
*/
public function setSelectedItemIndex($value)
{
if(($value=TPropertyValue::ensureInteger($value))<0)
$value=-1;
if(($current=$this->getSelectedItemIndex())!==$value)
{
$this->setViewState('SelectedItemIndex',$value,-1);
$items=$this->getItems();
$itemCount=$items->getCount();
if($current>=0 && $current<$itemCount)
{
$item=$items->itemAt($current);
if(($item instanceof IItemDataRenderer) && $item->getItemType()!==TListItemType::EditItem)
$item->setItemType($current%2?TListItemType::AlternatingItem : TListItemType::Item);
}
if($value>=0 && $value<$itemCount)
{
$item=$items->itemAt($value);
if(($item instanceof IItemDataRenderer) && $item->getItemType()!==TListItemType::EditItem)
$item->setItemType(TListItemType::SelectedItem);
}
}
}
/**
* @return TControl the selected item, null if no item is selected.
*/
public function getSelectedItem()
{
$index=$this->getSelectedItemIndex();
$items=$this->getItems();
if($index>=0 && $index<$items->getCount())
return $items->itemAt($index);
else
return null;
}
/**
* @return mixed the key value of the currently selected item
* @throws TInvalidOperationException if {@link getDataKeyField DataKeyField} is empty.
*/
public function getSelectedDataKey()
{
if($this->getDataKeyField()==='')
throw new TInvalidOperationException('datalist_datakeyfield_required');
$index=$this->getSelectedItemIndex();
$dataKeys=$this->getDataKeys();
if($index>=0 && $index<$dataKeys->getCount())
return $dataKeys->itemAt($index);
else
return null;
}
/**
* @return integer the zero-based index of the edit item in {@link getItems Items}.
* A value -1 means no item is in edit mode.
*/
public function getEditItemIndex()
{
return $this->getViewState('EditItemIndex',-1);
}
/**
* Edits an item by its index in {@link getItems Items}.
* Previously editting item will change to normal item state.
* If the index is less than 0, any existing edit item will be cleared up.
* @param integer the edit item index
*/
public function setEditItemIndex($value)
{
if(($value=TPropertyValue::ensureInteger($value))<0)
$value=-1;
if(($current=$this->getEditItemIndex())!==$value)
{
$this->setViewState('EditItemIndex',$value,-1);
$items=$this->getItems();
$itemCount=$items->getCount();
if($current>=0 && $current<$itemCount)
$items->itemAt($current)->setItemType($current%2?TListItemType::AlternatingItem : TListItemType::Item);
if($value>=0 && $value<$itemCount)
$items->itemAt($value)->setItemType(TListItemType::EditItem);
}
}
/**
* @return TControl the edit item
*/
public function getEditItem()
{
$index=$this->getEditItemIndex();
$items=$this->getItems();
if($index>=0 && $index<$items->getCount())
return $items->itemAt($index);
else
return null;
}
/**
* @return boolean whether the header should be shown. Defaults to true.
*/
public function getShowHeader()
{
return $this->getViewState('ShowHeader',true);
}
/**
* @param boolean whether to show header
*/
public function setShowHeader($value)
{
$this->setViewState('ShowHeader',TPropertyValue::ensureBoolean($value),true);
}
/**
* @return boolean whether the footer should be shown. Defaults to true.
*/
public function getShowFooter()
{
return $this->getViewState('ShowFooter',true);
}
/**
* @param boolean whether to show footer
*/
public function setShowFooter($value)
{
$this->setViewState('ShowFooter',TPropertyValue::ensureBoolean($value),true);
}
/**
* @return TRepeatInfo repeat information (primarily used by control developers)
*/
protected function getRepeatInfo()
{
if(($repeatInfo=$this->getViewState('RepeatInfo',null))===null)
{
$repeatInfo=new TRepeatInfo;
$this->setViewState('RepeatInfo',$repeatInfo,null);
}
return $repeatInfo;
}
/**
* @return string caption of the table layout
*/
public function getCaption()
{
return $this->getRepeatInfo()->getCaption();
}
/**
* @param string caption of the table layout
*/
public function setCaption($value)
{
$this->getRepeatInfo()->setCaption($value);
}
/**
* @return TTableCaptionAlign alignment of the caption of the table layout. Defaults to TTableCaptionAlign::NotSet.
*/
public function getCaptionAlign()
{
return $this->getRepeatInfo()->getCaptionAlign();
}
/**
* @return TTableCaptionAlign alignment of the caption of the table layout.
*/
public function setCaptionAlign($value)
{
$this->getRepeatInfo()->setCaptionAlign($value);
}
/**
* @return integer the number of columns that the list should be displayed with. Defaults to 0 meaning not set.
*/
public function getRepeatColumns()
{
return $this->getRepeatInfo()->getRepeatColumns();
}
/**
* @param integer the number of columns that the list should be displayed with.
*/
public function setRepeatColumns($value)
{
$this->getRepeatInfo()->setRepeatColumns($value);
}
/**
* @return TRepeatDirection the direction of traversing the list, defaults to TRepeatDirection::Vertical
*/
public function getRepeatDirection()
{
return $this->getRepeatInfo()->getRepeatDirection();
}
/**
* @param TRepeatDirection the direction of traversing the list
*/
public function setRepeatDirection($value)
{
$this->getRepeatInfo()->setRepeatDirection($value);
}
/**
* @return TRepeatLayout how the list should be displayed, using table or using line breaks. Defaults to TRepeatLayout::Table.
*/
public function getRepeatLayout()
{
return $this->getRepeatInfo()->getRepeatLayout();
}
/**
* @param TRepeatLayout how the list should be displayed, using table or using line breaks
*/
public function setRepeatLayout($value)
{
$this->getRepeatInfo()->setRepeatLayout($value);
}
/**
* This method overrides parent's implementation to handle
* {@link onItemCommand OnItemCommand} event which is bubbled from
* datalist items and their child controls.
* If the event parameter is {@link TDataListCommandEventParameter} and
* the command name is a recognized one, which includes 'select', 'edit',
* 'delete', 'update', and 'cancel' (case-insensitive), then a
* corresponding command event is also raised (such as {@link onEditCommand OnEditCommand}).
* This method should only be used by control developers.
* @param TControl the sender of the event
* @param TEventParameter event parameter
* @return boolean whether the event bubbling should stop here.
*/
public function bubbleEvent($sender,$param)
{
if($param instanceof TDataListCommandEventParameter)
{
$this->onItemCommand($param);
$command=$param->getCommandName();
if(strcasecmp($command,self::CMD_SELECT)===0)
{
if(($item=$param->getItem()) instanceof IItemDataRenderer)
$this->setSelectedItemIndex($item->getItemIndex());
$this->onSelectedIndexChanged($param);
return true;
}
else if(strcasecmp($command,self::CMD_EDIT)===0)
{
$this->onEditCommand($param);
return true;
}
else if(strcasecmp($command,self::CMD_DELETE)===0)
{
$this->onDeleteCommand($param);
return true;
}
else if(strcasecmp($command,self::CMD_UPDATE)===0)
{
$this->onUpdateCommand($param);
return true;
}
else if(strcasecmp($command,self::CMD_CANCEL)===0)
{
$this->onCancelCommand($param);
return true;
}
}
return false;
}
/**
* Raises OnItemCreated event.
* This method is invoked after a data list item is created and instantiated with
* template, but before added to the page hierarchy.
* The datalist item control responsible for the event
* can be determined from the event parameter.
* If you override this method, be sure to call parent's implementation
* so that event handlers have chance to respond to the event.
* @param TDataListItemEventParameter event parameter
*/
public function onItemCreated($param)
{
$this->raiseEvent('OnItemCreated',$this,$param);
}
/**
* Raises OnItemDataBound event.
* This method is invoked right after an item is data bound.
* The datalist item control responsible for the event
* can be determined from the event parameter.
* If you override this method, be sure to call parent's implementation
* so that event handlers have chance to respond to the event.
* @param TDataListItemEventParameter event parameter
*/
public function onItemDataBound($param)
{
$this->raiseEvent('OnItemDataBound',$this,$param);
}
/**
* Raises OnItemCommand event.
* This method is invoked when a child control of the data list
* raises an OnCommand event.
* @param TDataListCommandEventParameter event parameter
*/
public function onItemCommand($param)
{
$this->raiseEvent('OnItemCommand',$this,$param);
}
/**
* Raises OnEditCommand event.
* This method is invoked when a child control of the data list
* raises an OnCommand event and the command name is 'edit' (case-insensitive).
* @param TDataListCommandEventParameter event parameter
*/
public function onEditCommand($param)
{
$this->raiseEvent('OnEditCommand',$this,$param);
}
/**
* Raises OnDeleteCommand event.
* This method is invoked when a child control of the data list
* raises an OnCommand event and the command name is 'delete' (case-insensitive).
* @param TDataListCommandEventParameter event parameter
*/
public function onDeleteCommand($param)
{
$this->raiseEvent('OnDeleteCommand',$this,$param);
}
/**
* Raises OnUpdateCommand event.
* This method is invoked when a child control of the data list
* raises an OnCommand event and the command name is 'update' (case-insensitive).
* @param TDataListCommandEventParameter event parameter
*/
public function onUpdateCommand($param)
{
$this->raiseEvent('OnUpdateCommand',$this,$param);
}
/**
* Raises OnCancelCommand event.
* This method is invoked when a child control of the data list
* raises an OnCommand event and the command name is 'cancel' (case-insensitive).
* @param TDataListCommandEventParameter event parameter
*/
public function onCancelCommand($param)
{
$this->raiseEvent('OnCancelCommand',$this,$param);
}
/**
* Returns a value indicating whether this control contains header item.
* This method is required by {@link IRepeatInfoUser} interface.
* @return boolean whether the datalist has header
*/
public function getHasHeader()
{
return ($this->getShowHeader() && ($this->_headerTemplate!==null || $this->getHeaderRenderer()!==''));
}
/**
* Returns a value indicating whether this control contains footer item.
* This method is required by {@link IRepeatInfoUser} interface.
* @return boolean whether the datalist has footer
*/
public function getHasFooter()
{
return ($this->getShowFooter() && ($this->_footerTemplate!==null || $this->getFooterRenderer()!==''));
}
/**
* Returns a value indicating whether this control contains separator items.
* This method is required by {@link IRepeatInfoUser} interface.
* @return boolean always false.
*/
public function getHasSeparators()
{
return $this->_separatorTemplate!==null || $this->getSeparatorRenderer()!=='';
}
/**
* Returns a style used for rendering items.
* This method is required by {@link IRepeatInfoUser} interface.
* @param string item type (Header,Footer,Item,AlternatingItem,SelectedItem,EditItem,Separator,Pager)
* @param integer index of the item being rendered
* @return TStyle item style
*/
public function generateItemStyle($itemType,$index)
{
if(($item=$this->getItem($itemType,$index))!==null && ($item instanceof IStyleable) && $item->getHasStyle())
{
$style=$item->getStyle();
$item->clearStyle();
return $style;
}
else
return null;
}
/**
* Renders an item in the list.
* This method is required by {@link IRepeatInfoUser} interface.
* @param THtmlWriter writer for rendering purpose
* @param TRepeatInfo repeat information
* @param string item type (Header,Footer,Item,AlternatingItem,SelectedItem,EditItem,Separator,Pager)
* @param integer zero-based index of the item in the item list
*/
public function renderItem($writer,$repeatInfo,$itemType,$index)
{
$item=$this->getItem($itemType,$index);
if($repeatInfo->getRepeatLayout()===TRepeatLayout::Raw && get_class($item)==='TDataListItem')
$item->setTagName('div');
$item->renderControl($writer);
}
/**
* @param TListItemType item type
* @param integer item index
* @return TControl data list item with the specified item type and index
*/
private function getItem($itemType,$index)
{
switch($itemType)
{
case TListItemType::Item:
case TListItemType::AlternatingItem:
case TListItemType::SelectedItem:
case TListItemType::EditItem:
return $this->getItems()->itemAt($index);
case TListItemType::Header:
return $this->getControls()->itemAt(0);
case TListItemType::Footer:
return $this->getControls()->itemAt($this->getControls()->getCount()-1);
case TListItemType::Separator:
$i=$index+$index+1;
if($this->_headerTemplate!==null || $this->getHeaderRenderer()!=='')
$i++;
return $this->getControls()->itemAt($i);
}
return null;
}
/**
* Creates a datalist item.
* This method invokes {@link createItem} to create a new datalist item.
* @param integer zero-based item index.
* @param TListItemType item type
* @return TControl the created item, null if item is not created
*/
private function createItemInternal($itemIndex,$itemType)
{
if(($item=$this->createItem($itemIndex,$itemType))!==null)
{
$param=new TDataListItemEventParameter($item);
$this->onItemCreated($param);
$this->getControls()->add($item);
return $item;
}
else
return null;
}
/**
* Creates a datalist item and performs databinding.
* This method invokes {@link createItem} to create a new datalist item.
* @param integer zero-based item index.
* @param TListItemType item type
* @param mixed data to be associated with the item
* @return TControl the created item, null if item is not created
*/
private function createItemWithDataInternal($itemIndex,$itemType,$dataItem)
{
if(($item=$this->createItem($itemIndex,$itemType))!==null)
{
$param=new TDataListItemEventParameter($item);
if($item instanceof IDataRenderer)
$item->setData($dataItem);
$this->onItemCreated($param);
$this->getControls()->add($item);
$item->dataBind();
$this->onItemDataBound($param);
return $item;
}
else
return null;
}
private function getAlternatingItemDisplay()
{
if(($classPath=$this->getAlternatingItemRenderer())==='' && $this->_alternatingItemTemplate===null)
return array($this->getItemRenderer(),$this->_itemTemplate);
else
return array($classPath,$this->_alternatingItemTemplate);
}
private function getSelectedItemDisplay($itemIndex)
{
if(($classPath=$this->getSelectedItemRenderer())==='' && $this->_selectedItemTemplate===null)
{
if($itemIndex%2===0)
return array($this->getItemRenderer(),$this->_itemTemplate);
else
return $this->getAlternatingItemDisplay();
}
else
return array($classPath,$this->_selectedItemTemplate);
}
private function getEditItemDisplay($itemIndex)
{
if(($classPath=$this->getEditItemRenderer())==='' && $this->_editItemTemplate===null)
return $this->getSelectedItemDisplay($itemIndex);
else
return array($classPath,$this->_editItemTemplate);
}
/**
* Creates a datalist item instance based on the item type and index.
* @param integer zero-based item index
* @param TListItemType item type
* @return TControl created datalist item
*/
protected function createItem($itemIndex,$itemType)
{
$template=null;
$classPath=null;
switch($itemType)
{
case TListItemType::Item :
$classPath=$this->getItemRenderer();
$template=$this->_itemTemplate;
break;
case TListItemType::AlternatingItem :
list($classPath,$template)=$this->getAlternatingItemDisplay();
break;
case TListItemType::SelectedItem:
list($classPath,$template)=$this->getSelectedItemDisplay($itemIndex);
break;
case TListItemType::EditItem:
list($classPath,$template)=$this->getEditItemDisplay($itemIndex);
break;
case TListItemType::Header :
$classPath=$this->getHeaderRenderer();
$template=$this->_headerTemplate;
break;
case TListItemType::Footer :
$classPath=$this->getFooterRenderer();
$template=$this->_footerTemplate;
break;
case TListItemType::Separator :
$classPath=$this->getSeparatorRenderer();
$template=$this->_separatorTemplate;
break;
default:
throw new TInvalidDataValueException('datalist_itemtype_unknown',$itemType);
}
if($classPath!=='')
{
$item=Prado::createComponent($classPath);
if($item instanceof IItemDataRenderer)
{
$item->setItemIndex($itemIndex);
$item->setItemType($itemType);
}
}
else if($template!==null)
{
$item=new TDataListItem;
$item->setItemIndex($itemIndex);
$item->setItemType($itemType);
$template->instantiateIn($item);
}
else
$item=null;
return $item;
}
/**
* Creates empty datalist content.
*/
protected function createEmptyContent()
{
if(($classPath=$this->getEmptyRenderer())!=='')
$this->getControls()->add(Prado::createComponent($classPath));
else if($this->_emptyTemplate!==null)
$this->_emptyTemplate->instantiateIn($this);
}
/**
* Applies styles to items, header, footer and separators.
* Item styles are applied in a hierarchical way. Style in higher hierarchy
* will inherit from styles in lower hierarchy.
* Starting from the lowest hierarchy, the item styles include
* item's own style, {@link getItemStyle ItemStyle}, {@link getAlternatingItemStyle AlternatingItemStyle},
* {@link getSelectedItemStyle SelectedItemStyle}, and {@link getEditItemStyle EditItemStyle}.
* Therefore, if background color is set as red in {@link getItemStyle ItemStyle},
* {@link getEditItemStyle EditItemStyle} will also have red background color
* unless it is set to a different value explicitly.
*/
protected function applyItemStyles()
{
$itemStyle=$this->getViewState('ItemStyle',null);
$alternatingItemStyle=$this->getViewState('AlternatingItemStyle',null);
if($itemStyle!==null)
{
if($alternatingItemStyle===null)
$alternatingItemStyle=$itemStyle;
else
$alternatingItemStyle->mergeWith($itemStyle);
}
$selectedItemStyle=$this->getViewState('SelectedItemStyle',null);
$editItemStyle=$this->getViewState('EditItemStyle',null);
if($selectedItemStyle!==null)
{
if($editItemStyle===null)
$editItemStyle=$selectedItemStyle;
else
$editItemStyle->mergeWith($selectedItemStyle);
}
// apply header style if any
if($this->_header!==null && $this->_header instanceof IStyleable)
{
if($headerStyle=$this->getViewState('HeaderStyle',null))
$this->_header->getStyle()->mergeWith($headerStyle);
}
// apply footer style if any
if($this->_footer!==null && $this->_footer instanceof IStyleable)
{
if($footerStyle=$this->getViewState('FooterStyle',null))
$this->_footer->getStyle()->mergeWith($footerStyle);
}
$selectedIndex=$this->getSelectedItemIndex();
$editIndex=$this->getEditItemIndex();
// apply item styles if any
foreach($this->getItems() as $index=>$item)
{
if($index===$editIndex)
$style=$editItemStyle;
else if($index===$selectedIndex)
$style=$selectedItemStyle;
else if($index%2===0)
$style=$itemStyle;
else
$style=$alternatingItemStyle;
if($style && $item instanceof IStyleable)
$item->getStyle()->mergeWith($style);
}
// apply separator style if any
if(($separatorStyle=$this->getViewState('SeparatorStyle',null))!==null && $this->getHasSeparators())
{
$controls=$this->getControls();
$count=$controls->getCount();
for($i=$this->_header?2:1;$i<$count;$i+=2)
{
if(($separator=$controls->itemAt($i)) instanceof IStyleable)
$separator->getStyle()->mergeWith($separatorStyle);
}
}
}
/**
* Saves item count in viewstate.
* This method is invoked right before control state is to be saved.
*/
public function saveState()
{
parent::saveState();
if($this->_items)
$this->setViewState('ItemCount',$this->_items->getCount(),0);
else
$this->clearViewState('ItemCount');
}
/**
* Loads item count information from viewstate.
* This method is invoked right after control state is loaded.
*/
public function loadState()
{
parent::loadState();
if(!$this->getIsDataBound())
$this->restoreItemsFromViewState();
$this->clearViewState('ItemCount');
}
/**
* Clears up all items in the data list.
*/
public function reset()
{
$this->getControls()->clear();
$this->getItems()->clear();
$this->_header=null;
$this->_footer=null;
}
/**
* Creates data list items based on viewstate information.
*/
protected function restoreItemsFromViewState()
{
$this->reset();
if(($itemCount=$this->getViewState('ItemCount',0))>0)
{
$items=$this->getItems();
$selectedIndex=$this->getSelectedItemIndex();
$editIndex=$this->getEditItemIndex();
$hasSeparator=$this->_separatorTemplate!==null || $this->getSeparatorRenderer()!=='';
$this->_header=$this->createItemInternal(-1,TListItemType::Header);
for($i=0;$i<$itemCount;++$i)
{
if($hasSeparator && $i>0)
$this->createItemInternal($i-1,TListItemType::Separator);
if($i===$editIndex)
$itemType=TListItemType::EditItem;
else if($i===$selectedIndex)
$itemType=TListItemType::SelectedItem;
else
$itemType=$i%2?TListItemType::AlternatingItem : TListItemType::Item;
$items->add($this->createItemInternal($i,$itemType));
}
$this->_footer=$this->createItemInternal(-1,TListItemType::Footer);
}
else
$this->createEmptyContent();
$this->clearChildState();
}
/**
* Performs databinding to populate data list items from data source.
* This method is invoked by dataBind().
* You may override this function to provide your own way of data population.
* @param Traversable the data
*/
protected function performDataBinding($data)
{
$this->reset();
$keys=$this->getDataKeys();
$keys->clear();
$keyField=$this->getDataKeyField();
$itemIndex=0;
$items=$this->getItems();
$hasSeparator=$this->_separatorTemplate!==null || $this->getSeparatorRenderer()!=='';
$selectedIndex=$this->getSelectedItemIndex();
$editIndex=$this->getEditItemIndex();
foreach($data as $key=>$dataItem)
{
if($keyField!=='')
$keys->add($this->getDataFieldValue($dataItem,$keyField));
else
$keys->add($key);
if($itemIndex===0)
$this->_header=$this->createItemWithDataInternal(-1,TListItemType::Header,null);
if($hasSeparator && $itemIndex>0)
$this->createItemWithDataInternal($itemIndex-1,TListItemType::Separator,null);
if($itemIndex===$editIndex)
$itemType=TListItemType::EditItem;
else if($itemIndex===$selectedIndex)
$itemType=TListItemType::SelectedItem;
else
$itemType=$itemIndex%2?TListItemType::AlternatingItem : TListItemType::Item;
$items->add($this->createItemWithDataInternal($itemIndex,$itemType,$dataItem));
$itemIndex++;
}
if($itemIndex>0)
$this->_footer=$this->createItemWithDataInternal(-1,TListItemType::Footer,null);
else
{
$this->createEmptyContent();
$this->dataBindChildren();
}
$this->setViewState('ItemCount',$itemIndex,0);
}
/**
* Renders the data list control.
* This method overrides the parent implementation.
* @param THtmlWriter writer for rendering purpose.
*/
public function render($writer)
{
if($this->getHasControls())
{
if($this->getItemCount()>0)
{
$this->applyItemStyles();
$repeatInfo=$this->getRepeatInfo();
$repeatInfo->renderRepeater($writer,$this);
}
else if($this->_emptyTemplate!==null || $this->getEmptyRenderer()!=='')
parent::render($writer);
}
}
}