* @since 2.0 */ class GridView extends ListViewBase { const FILTER_POS_HEADER = 'header'; const FILTER_POS_FOOTER = 'footer'; const FILTER_POS_BODY = 'body'; public $dataColumnClass; public $caption; public $captionOptions = array(); public $tableOptions = array('class' => 'table table-striped table-bordered'); public $headerRowOptions = array(); public $footerRowOptions = array(); public $beforeRow; public $afterRow; public $showHeader = true; public $showFooter = false; /** * @var array|Closure */ public $rowOptions = array(); /** * @var array|Formatter the formatter used to format model attribute values into displayable texts. * This can be either an instance of [[Formatter]] or an configuration array for creating the [[Formatter]] * instance. If this property is not set, the "formatter" application component will be used. */ public $formatter; /** * @var array grid column configuration. Each array element represents the configuration * for one particular grid column which can be either a string or an array. * * When a column is specified as a string, it should be in the format of "name:type:header", * where "type" and "header" are optional. A {@link CDataColumn} instance will be created in this case, * whose {@link CDataColumn::name}, {@link CDataColumn::type} and {@link CDataColumn::header} * properties will be initialized accordingly. * * When a column is specified as an array, it will be used to create a grid column instance, where * the 'class' element specifies the column class name (defaults to {@link CDataColumn} if absent). * Currently, these official column classes are provided: {@link CDataColumn}, * {@link CLinkColumn}, {@link CButtonColumn} and {@link CCheckBoxColumn}. */ public $columns = array(); /** * @var string the layout that determines how different sections of the list view should be organized. * The following tokens will be replaced with the corresponding section contents: * * - `{summary}`: the summary section. See [[renderSummary()]]. * - `{items}`: the list items. See [[renderItems()]]. * - `{sorter}`: the sorter. See [[renderSorter()]]. * - `{pager}`: the pager. See [[renderPager()]]. */ public $layout = "{items}\n{summary}\n{pager}"; public $emptyCell = ' '; /** * @var \yii\base\Model the model instance that keeps the user-entered filter data. When this property is set, * the grid view will enable column-based filtering. Each data column by default will display a text field * at the top that users can fill in to filter the data. * Note that in order to show an input field for filtering, a column must have its {@link CDataColumn::name} * property set or have {@link CDataColumn::filter} as the HTML code for the input field. * When this property is not set (null) the filtering is disabled. */ public $filterModel; /** * @var string whether the filters should be displayed in the grid view. Valid values include: * */ public $filterPosition = 'body'; public $filterOptions = array('class' => 'filters'); /** * Initializes the grid view. * This method will initialize required property values and instantiate {@link columns} objects. */ public function init() { parent::init(); if ($this->formatter == null) { $this->formatter = Yii::$app->getFormatter(); } elseif (is_array($this->formatter)) { $this->formatter = Yii::createObject($this->formatter); } if (!$this->formatter instanceof Formatter) { throw new InvalidConfigException('The "formatter" property must be either a Format object or a configuration array.'); } if (!isset($this->options['id'])) { $this->options['id'] = $this->getId(); } $this->initColumns(); } /** * Runs the widget. */ public function run() { $id = $this->options['id']; $view = $this->getView(); GridViewAsset::register($view); $view->registerJs("jQuery('#$id').yiiGridView();"); parent::run(); } /** * Renders the data models for the grid view. */ public function renderItems() { $content = array_filter(array( $this->renderCaption(), $this->renderColumnGroup(), $this->showHeader ? $this->renderTableHeader() : false, $this->showFooter ? $this->renderTableFooter() : false, $this->renderTableBody(), )); return Html::tag('table', implode("\n", $content), $this->tableOptions); } public function renderCaption() { if (!empty($this->caption)) { return Html::tag('caption', $this->caption, $this->captionOptions); } else { return false; } } public function renderColumnGroup() { $requireColumnGroup = false; foreach ($this->columns as $column) { /** @var \yii\widgets\grid\Column $column */ if (!empty($column->options)) { $requireColumnGroup = true; break; } } if ($requireColumnGroup) { $cols = array(); foreach ($this->columns as $column) { $cols[] = Html::tag('col', '', $column->options); } return Html::tag('colgroup', implode("\n", $cols)); } else { return false; } } /** * Renders the table header. * @return string the rendering result */ public function renderTableHeader() { $cells = array(); foreach ($this->columns as $column) { /** @var \yii\widgets\grid\Column $column */ $cells[] = $column->renderHeaderCell(); } $content = implode('', $cells); if ($this->filterPosition == self::FILTER_POS_HEADER) { $content = $this->renderFilters() . $content; } elseif ($this->filterPosition == self::FILTER_POS_BODY) { $content .= $this->renderFilters(); } return "\n" . Html::tag('tr', $content, $this->headerRowOptions) . "\n"; } /** * Renders the table footer. * @return string the rendering result */ public function renderTableFooter() { $cells = array(); foreach ($this->columns as $column) { /** @var \yii\widgets\grid\Column $column */ $cells[] = $column->renderFooterCell(); } $content = implode('', $cells); if ($this->filterPosition == self::FILTER_POS_FOOTER) { $content .= $this->renderFilters(); } return "\n" . Html::tag('tr', $content, $this->footerRowOptions) . "\n"; } /** * Renders the filter. */ public function renderFilters() { if ($this->filterModel !== null) { $cells = array(); foreach ($this->columns as $column) { /** @var \yii\widgets\grid\Column $column */ $cells[] = $column->renderFilterCell(); } return Html::tag('tr', implode('', $cells), $this->filterOptions); } else { return ''; } } /** * Renders the table body. * @return string the rendering result */ public function renderTableBody() { $models = array_values($this->dataProvider->getModels()); $keys = $this->dataProvider->getKeys(); $rows = array(); foreach ($models as $index => $model) { $key = $keys[$index]; if ($this->beforeRow !== null) { $row = call_user_func($this->beforeRow, $model, $key, $index); if (!empty($row)) { $rows[] = $row; } } $rows[] = $this->renderTableRow($model, $key, $index); if ($this->afterRow !== null) { $row = call_user_func($this->afterRow, $model, $key, $index); if (!empty($row)) { $rows[] = $row; } } } return "\n" . implode("\n", $rows) . "\n"; } /** * Renders a table row with the given data model and key. * @param mixed $model the data model to be rendered * @param mixed $key the key associated with the data model * @param integer $index the zero-based index of the data model among the model array returned by [[dataProvider]]. * @return string the rendering result */ public function renderTableRow($model, $key, $index) { $cells = array(); /** @var \yii\widgets\grid\Column $column */ foreach ($this->columns as $column) { $cells[] = $column->renderDataCell($model, $index); } if ($this->rowOptions instanceof Closure) { $options = call_user_func($this->rowOptions, $model, $key, $index); } else { $options = $this->rowOptions; } $options['data-key'] = $key; return Html::tag('tr', implode('', $cells), $options); } /** * Creates column objects and initializes them. */ protected function initColumns() { if (empty($this->columns)) { $this->guessColumns(); } foreach ($this->columns as $i => $column) { if (is_string($column)) { $column = $this->createDataColumn($column); } else { $column = Yii::createObject(array_merge(array( 'class' => $this->dataColumnClass ?: DataColumn::className(), 'grid' => $this, ), $column)); } if (!$column->visible) { unset($this->columns[$i]); continue; } $this->columns[$i] = $column; } } /** * Creates a {@link CDataColumn} based on a shortcut column specification string. * @param string $text the column specification string * @return DataColumn the column instance * @throws InvalidConfigException if the column specification is invalid */ protected function createDataColumn($text) { if (!preg_match('/^([\w\.]+)(:(\w*))?(:(.*))?$/', $text, $matches)) { throw new InvalidConfigException('The column must be specified in the format of "Attribute", "Attribute:Format" or "Attribute:Format:Header'); } return Yii::createObject(array( 'class' => $this->dataColumnClass ?: DataColumn::className(), 'grid' => $this, 'attribute' => $matches[1], 'format' => isset($matches[3]) ? $matches[3] : 'text', 'header' => isset($matches[5]) ? $matches[5] : null, )); } protected function guessColumns() { $models = $this->dataProvider->getModels(); $model = reset($models); if (is_array($model) || is_object($model)) { foreach ($model as $name => $value) { $this->columns[] = $name; } } else { throw new InvalidConfigException('Unable to generate columns from data.'); } } }