* @link http://www.pradosoft.com/ * @copyright Copyright © 2005 PradoSoft * @license http://www.pradosoft.com/license/ * @version $Revision: $ $Date: $ * @package System.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. * * The {@link setHeaderTemplate HeaderTemplate} property specifies the content * template that will be displayed at the beginning, while * {@link setFooterTemplate FooterTemplate} at the end. * If present, these two templates will only be rendered when the data list is * given non-empty data. In this case, for each data item the content defined * by {@link setItemTemplate ItemTemplate} will be generated and displayed once. * If {@link setAlternatingItemTemplate AlternatingItemTemplate} is not empty, * then the corresponding content will be displayed alternatively with that * in {@link setItemTemplate ItemTemplate}. The content in * {@link setSeparatorTemplate SeparatorTemplate}, if not empty, will be * displayed between items. Besides the above templates, there are two additional * templates, {@link setEditItemTemplate EditItemTemplate} and * {@link setSelectedItemTemplate SelectedItemTemplate}, which are used to display * items that are in edit and selected mode, respectively. * * All these templates are associated with styles that may be applied to * the corresponding generated items. For example, * {@link getAlternatingItemStyle AlternatingItemStyle} will be applied to * every alternating item in the data list. * * 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. * * 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} * for presentation. The latter will change the indicated item to edit mode. * Note, if an item is in edit mode, then selecting this item will have no effect. * * The layout of the data items in the list is specified via * {@link setRepeatLayout RepeatLayout}, which can be either 'Table' (default) or 'Flow'. * A table layout uses HTML table cells to organize the data items while * a flow layout uses line breaks to organize the data items. * When the layout is using 'Table', {@link setCellPadding CellPadding} and * {@link setCellSpacing CellSpacing} can be used to adjust the cellpadding and * cellpadding of the table, and {@link setCaption Caption} and {@link setCaptionAlign CaptionAlign} * can be used to add a table caption with the specified alignment. * * 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. * * You can retrive the repeated contents by the {@link getItems Items} property. * The header and footer items can be accessed by {@link getHeader Header} * and {@link getFooter Footer} properties, respectively. * * When TDataList creates an item, it will raise an {@link onItemCreated OnItemCreated} * so that you may customize the newly created item. * When databinding is performed by TDataList, for each item once it has finished * databinding, an {@link onItemDataBound OnItemDataBound} event will be raised. * * When an item is selected by an end-user, a {@link onSelectedIndexChanged OnSelectedIndexChanged} * event will be raised. Note, the selected index may not be actually changed. * The event mainly informs the server side that the end-user has made a selection. * * Each datalist item has a {@link TDataListItem::getItemType type} * which tells the position and state of the item in the datalist. An item in the header * of the repeater is of type Header. A body item may be of either * Item, AlternatingItem, SelectedItem or EditItem, depending whether the item * index is odd or even, whether it is being selected or edited. * * TDataList raises an {@link onItemCommand OnItemCommand} whenever a button control * within some TDataList item raises a OnCommand event. If the command name * is one of the followings: 'edit', 'update', 'select', 'delete', 'cancel' (case-insensitive), * another event will also be raised. For example, if the command name is 'edit', * then the new event is {@link onEditCommand OnEditCommand}. * * Note, the data bound to the datalist are reset to null after databinding. * There are several ways to access the data associated with a datalist item: * - Access the data in {@link onItemDataBound OnItemDataBound} event * - 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 data in viewstate and get it back during postbacks. * * @author Qiang Xue * @version $Revision: $ $Date: $ * @package System.Web.UI.WebControls * @since 3.0 */ class TDataList extends TBaseDataList implements INamingContainer, IRepeatInfoUser { /** * Command name that TDataList understands. */ 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 TDatListItem header item */ private $_header=null; /** * @var TDatListItem 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 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 TDataListItem 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 TDataListItem 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->getItemType()!==TListItemType::EditItem) $item->setItemType($current%2?TListItemType::AlternatingItem : TListItemType::Item); } if($value>=0 && $value<$itemCount) { $item=$items->itemAt($value); if($item->getItemType()!==TListItemType::EditItem) $item->setItemType(TListItemType::SelectedItem); } } } /** * @return TDataListItem 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 TDataListItem 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 * {@link TDataListItem} 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) { $this->setSelectedItemIndex($param->getItem()->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 {@link TDataListItem} 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 {@link TDataListItem} 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 Command 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 Command 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 Command 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 Command 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 Command 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 always false. */ public function getHasHeader() { return ($this->getShowHeader() && $this->_headerTemplate!==null); } /** * Returns a value indicating whether this control contains footer item. * This method is required by {@link IRepeatInfoUser} interface. * @return boolean always false. */ public function getHasFooter() { return ($this->getShowFooter() && $this->_footerTemplate!==null); } /** * 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; } /** * 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->getHasStyle()) return $item->getStyle(); 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); $layout=$repeatInfo->getRepeatLayout(); if($layout==='Table' || $layout==='Raw') $item->renderContents($writer); else $item->renderControl($writer); } /** * @param string item type * @param integer item index * @return TDataListItem data list item with the specified item type and index */ private function getItem($itemType,$index) { switch($itemType) { case TListItemType::Header: return $this->getControls()->itemAt(0); case TListItemType::Footer: return $this->getControls()->itemAt($this->getControls()->getCount()-1); case TListItemType::Item: case TListItemType::AlternatingItem: case TListItemType::SelectedItem: case TListItemType::EditItem: return $this->getItems()->itemAt($index); case TListItemType::Separator: $i=$index+$index+1; if($this->_headerTemplate!==null) $i++; return $this->getControls()->itemAt($i); } return null; } /** * Creates a data list item and does databinding if needed. * This method invokes {@link createItem} to create a new data list item. * @param integer zero-based item index. * @param TListItemType item type * @param boolean whether to do databinding for the item * @param mixed data to be associated with the item * @return TDataListItem the created item */ private function createItemInternal($itemIndex,$itemType,$dataBind,$dataItem) { $item=$this->createItem($itemIndex,$itemType); $this->initializeItem($item); $param=new TDataListItemEventParameter($item); if($dataBind) { $item->setDataItem($dataItem); $this->onItemCreated($param); $this->getControls()->add($item); $item->dataBind(); $this->onItemDataBound($param); $item->setDataItem(null); } else { $this->onItemCreated($param); $this->getControls()->add($item); } return $item; } /** * Creates a DataList item instance based on the item type and index. * @param integer zero-based item index * @param TListItemType item type * @return TDataListItem created data list item */ protected function createItem($itemIndex,$itemType) { return new TDataListItem($itemIndex,$itemType); } /** * 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); } $headerStyle=$this->getViewState('HeaderStyle',null); $footerStyle=$this->getViewState('FooterStyle',null); $separatorStyle=$this->getViewState('SeparatorStyle',null); foreach($this->getControls() as $index=>$item) { if(!($item instanceof TDataListItem)) continue; switch($item->getItemType()) { case TListItemType::Header: if($headerStyle) $item->getStyle()->mergeWith($headerStyle); break; case TListItemType::Footer: if($footerStyle) $item->getStyle()->mergeWith($footerStyle); break; case TListItemType::Separator: if($separatorStyle) $item->getStyle()->mergeWith($separatorStyle); break; case TListItemType::Item: if($itemStyle) $item->getStyle()->mergeWith($itemStyle); break; case TListItemType::AlternatingItem: if($alternatingItemStyle) $item->getStyle()->mergeWith($alternatingItemStyle); break; case TListItemType::SelectedItem: if($selectedItemStyle) $item->getStyle()->mergeWith($selectedItemStyle); if($index % 2==1) { if($itemStyle) $item->getStyle()->mergeWith($itemStyle); } else { if($alternatingItemStyle) $item->getStyle()->mergeWith($alternatingItemStyle); } break; case TListItemType::EditItem: if($editItemStyle) $item->getStyle()->mergeWith($editItemStyle); if($index % 2==1) { if($itemStyle) $item->getStyle()->mergeWith($itemStyle); } else { if($alternatingItemStyle) $item->getStyle()->mergeWith($alternatingItemStyle); } break; default: break; } } } /** * Initializes a data list item. * The item is added as a child of the data list and the corresponding * template is instantiated within the item. * @param TDataListItem item to be initialized */ protected function initializeItem($item) { $template=null; switch($item->getItemType()) { case TListItemType::Header: $template=$this->_headerTemplate; break; case TListItemType::Footer: $template=$this->_footerTemplate; break; case TListItemType::Item: $template=$this->_itemTemplate; break; case TListItemType::AlternatingItem: if(($template=$this->_alternatingItemTemplate)===null) $template=$this->_itemTemplate; break; case TListItemType::Separator: $template=$this->_separatorTemplate; break; case TListItemType::SelectedItem: if(($template=$this->_selectedItemTemplate)===null) { if(!($item->getItemIndex()%2) || ($template=$this->_alternatingItemTemplate)===null) $template=$this->_itemTemplate; } break; case TListItemType::EditItem: if(($template=$this->_editItemTemplate)===null) { if($item->getItemIndex()!==$this->getSelectedItemIndex() || ($template=$this->_selectedItemTemplate)===null) if(!($item->getItemIndex()%2) || ($template=$this->_alternatingItemTemplate)===null) $template=$this->_itemTemplate; } break; default: break; } if($template!==null) $template->instantiateIn($item); } /** * 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; if($this->_headerTemplate!==null) $this->_header=$this->createItemInternal(-1,TListItemType::Header,false,null); for($i=0;$i<$itemCount;++$i) { if($hasSeparator && $i>0) $this->createItemInternal($i-1,TListItemType::Separator,false,null); 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,false,null)); } if($this->_footerTemplate!==null) $this->_footer=$this->createItemInternal(-1,TListItemType::Footer,false,null); } else if($this->_emptyTemplate!==null) $this->_emptyTemplate->instantiateIn($this); $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; $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->_headerTemplate!==null) $this->_header=$this->createItemInternal(-1,TListItemType::Header,true,null); if($hasSeparator && $itemIndex>0) $this->createItemInternal($itemIndex-1,TListItemType::Separator,true,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->createItemInternal($itemIndex,$itemType,true,$dataItem)); $itemIndex++; } if($itemIndex>0 && $this->_footerTemplate!==null) $this->_footer=$this->createItemInternal(-1,TListItemType::Footer,true,null); if($itemIndex===0 && $this->_emptyTemplate!==null) { $this->_emptyTemplate->instantiateIn($this); $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) parent::render($writer); } } } /** * TDataListItemEventParameter class * * TDataListItemEventParameter encapsulates the parameter data for * {@link TDataList::onItemCreated ItemCreated} event of {@link TDataList} controls. * The {@link getItem Item} property indicates the DataList item related with the event. * * @author Qiang Xue * @version $Revision: $ $Date: $ * @package System.Web.UI.WebControls * @since 3.0 */ class TDataListItemEventParameter extends TEventParameter { /** * The TDataListItem control responsible for the event. * @var TDataListItem */ private $_item=null; /** * Constructor. * @param TDataListItem DataList item related with the corresponding event */ public function __construct(TDataListItem $item) { $this->_item=$item; } /** * @return TDataListItem DataList item related with the corresponding event */ public function getItem() { return $this->_item; } } /** * TDataListCommandEventParameter class * * TDataListCommandEventParameter encapsulates the parameter data for * {@link TDataList::onItemCommand ItemCommand} event of {@link TDataList} controls. * * The {@link getItem Item} property indicates the DataList item related with the event. * The {@link getCommandSource CommandSource} refers to the control that originally * raises the Command event. * * @author Qiang Xue * @version $Revision: $ $Date: $ * @package System.Web.UI.WebControls * @since 3.0 */ class TDataListCommandEventParameter extends TCommandEventParameter { /** * @var TDataListItem the TDataListItem control responsible for the event. */ private $_item=null; /** * @var TControl the control originally raises the Command event. */ private $_source=null; /** * Constructor. * @param TDataListItem DataList item responsible for the event * @param TControl original event sender * @param TCommandEventParameter original event parameter */ public function __construct($item,$source,TCommandEventParameter $param) { $this->_item=$item; $this->_source=$source; parent::__construct($param->getCommandName(),$param->getCommandParameter()); } /** * @return TDataListItem the TDataListItem control responsible for the event. */ public function getItem() { return $this->_item; } /** * @return TControl the control originally raises the Command event. */ public function getCommandSource() { return $this->_source; } } /** * TDataListItem class * * A TDataListItem control represents an item in the {@link TDataList} control, * such as heading section, footer section, or a data item. * The index and data value of the item can be accessed via {@link getItemIndex ItemIndex}> * and {@link getDataItem DataItem} properties, respectively. The type of the item * is given by {@link getItemType ItemType} property. * * @author Qiang Xue * @version $Revision: $ $Date: $ * @package System.Web.UI.WebControls * @since 3.0 */ class TDataListItem extends TWebControl implements INamingContainer { /** * index of the data item in the Items collection of DataList */ private $_itemIndex=''; /** * type of the TDataListItem * @var TListItemType */ private $_itemType; /** * value of the data item * @var mixed */ private $_dataItem=null; /** * Constructor. * @param integer zero-based index of the item in the item collection of DataList * @param TListItemType item type */ public function __construct($itemIndex,$itemType) { $this->_itemIndex=$itemIndex; $this->setItemType($itemType); } /** * Creates a style object for the control. * This method creates a {@link TTableItemStyle} to be used by a datalist item. * @return TStyle control style to be used */ protected function createStyle() { return new TTableItemStyle; } /** * @return TListItemType item type */ public function getItemType() { return $this->_itemType; } /** * @param TListItemType item type. */ public function setItemType($value) { $this->_itemType=TPropertyValue::ensureEnum($value,'TListItemType'); } /** * @return integer zero-based index of the item in the item collection of DataList */ public function getItemIndex() { return $this->_itemIndex; } /** * @return mixed data associated with the item */ public function getDataItem() { return $this->_dataItem; } /** * @param mixed data to be associated with the item */ public function setDataItem($value) { $this->_dataItem=$value; } /** * This method overrides parent's implementation by wrapping event parameter * for Command event with item information. * @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 TCommandEventParameter) { $this->raiseBubbleEvent($this,new TDataListCommandEventParameter($this,$sender,$param)); return true; } else return false; } } /** * TDataListItemCollection class. * * TDataListItemCollection represents a collection of data list items. * * @author Qiang Xue * @version $Revision: $ $Date: $ * @package System.Web.UI.WebControls * @since 3.0 */ class TDataListItemCollection extends TList { /** * Inserts an item at the specified position. * This overrides the parent implementation by inserting only TDataListItem. * @param integer the speicified position. * @param mixed new item * @throws TInvalidDataTypeException if the item to be inserted is not a TDataListItem. */ public function insertAt($index,$item) { if($item instanceof TDataListItem) parent::insertAt($index,$item); else throw new TInvalidDataTypeException('datalistitemcollection_datalistitem_required'); } } ?>