TbExtendedGridView

The most powerful Yii Grid

TbExtendedGridView booster.widgets.TbExtendedGridView

Have you ever thought to display a more elaborated summary of the data displayed on your grid? Or wanted to have a fixed header as you go down on thousand records? What about if you wish to display a chart of the data but not willing to leave the grid? And also, you wish to maintain that awesomeness even throughout AJAX calls? And what about if we tell you that we have done that?

With TbExtendedGridView you will be able to solve all those issues, plus we have also included some other columns that will help make the grid even more powerful:

  • TbJEditableColumn - for inline editing
  • TbTotalSumColumn - to sum up the total of a column and display it at its footer

New by setting the responsiveTable property to true (this is for all extended versions of TbGridView), tables will be automatically converted to suit mobile size. Try it by resizing the browser window and check this first example.

Fixed Header

Thanks to the great jquery plugin StickyTableHeaders we have managed to get this feature and make it as easy as to setup a simple boolean fixedHeader property to true.

Note It is important for the sake of the following examples that you reduce the height of your browser to see the effect.

#First nameLast nameLanguageHours workedEdit
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget(
  2. 'booster.widgets.TbExtendedGridView',
  3. array(
  4. 'fixedHeader' => true,
  5. 'headerOffset' => 40,
  6. // 40px is the height of the main navigation at bootstrap
  7. 'type' => 'striped',
  8. 'dataProvider' => $gridDataProvider,
  9. 'responsiveTable' => true,
  10. 'template' => "{items}",
  11. 'columns' => $gridColumns,
  12. )
  13. );

Fixed header with Filter

#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget(
  2. 'booster.widgets.TbExtendedGridView',
  3. array(
  4. 'filter' => $person,
  5. 'fixedHeader' => true,
  6. 'type' => 'striped bordered',
  7. 'headerOffset' => 40,
  8. // 40px is the height of the main navigation at bootstrap
  9. 'dataProvider' => $gridDataProvider,
  10. 'template' => "{items}",
  11. 'columns' => $gridColumns,
  12. )
  13. );

Sortable Rows

Sometimes, we may need to display a set of records that are ordered by a `position` on the database, and that order is important on the list. Most of the times, they are just a small amount of records that have a preference in a first-in first-out scenario. The extended grid view provides a mechanism to order the rows and report back through a javascript callback the new position of the data, so you could easily make an AJAX request to the server and update the list.

Note: It also will update the keys position, so you can safely use regular $.fn.yiiGridView functions.

#First nameLast nameLanguageHours worked
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'sortableRows'=>true,
  3. 'afterSortableUpdate' => 'js:function(id, position){ console.log("id: "+id+", position:"+position);}',
  4. 'type'=>'striped bordered',
  5. 'dataProvider' => $gridDataProvider,
  6. 'template' => "{items}",
  7. 'columns' => $gridColumns,
  8. ));

Selectable Cells

The TbExtendedGridView allows you also to provide a selectable mechanism to its cells. This is very useful when you wish to display some feedback information related to the selected cells.

The following example demonstrates the above. Select the cells of the Hours worked column to see its total on the span tag with id: 'hours'.

Sum Of Selected Hours: 0
#First nameLast nameLanguageHours worked
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. <span class="label label-info">Sum Of Selected Hours</span>: <span id="hours">0</span>
  2. <?php
  3. $this->widget('booster.widgets.TbExtendedGridView', array(
  4. 'selectableCells'=>true,
  5. 'selectableCellsFilter'=>'td.tobeselected',
  6. 'afterSelectableCells' => 'js:function(selected){
  7. var sum = 0;
  8. $.each(selected, function(){
  9. sum += parseInt($(this).text());
  10. });
  11. $("#hours").html(sum);
  12. }',
  13. 'type' => 'striped bordered',
  14. 'dataProvider' => $gridDataProvider,
  15. 'template' => "{items}",
  16. 'columns' => array_slice($gridColumns, 0, count($gridColumns)-1),
  17. ));
  18. ?>

Bulk Actions

The extended grid allows you to display bulk action buttons on its footer. The configuration options of the buttons are the same as regular TbButton widgets, the only difference is that you can set its 'click' handler, which will be triggered having the selected checkbox elements as a parameter to the function.

Note: The buttons won't be functional (active) until at least one checkbox element is on its checked state.

Note: The the bulk action is working only on the current visible checked items, and does not maintain state between pages, or after sort/filtering

#First nameLast nameLanguageHours workedEdit
    
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'type' => 'striped bordered',
  3. 'dataProvider' => $gridDataProvider,
  4. 'template' => "{items}",
  5. 'selectableRows' => 2,
  6. 'bulkActions' => array(
  7. 'actionButtons' => array(
  8. array(
  9. 'buttonType' => 'button',
  10. 'context' => 'primary',
  11. 'size' => 'small',
  12. 'label' => 'Testing Primary Bulk Actions',
  13. 'click' => 'js:function(values){console.log(values);}'
  14. )
  15. ),
  16. // if grid doesn't have a checkbox column type, it will attach
  17. // one and this configuration will be part of it
  18. 'checkBoxColumnConfig' => array(
  19. 'name' => 'id'
  20. ),
  21. ),
  22. 'columns' => $gridColumns,
  23. ));

Extended Summary

There are so many times that a client wishes to see a summary of the records displayed on your grid. When that occurs, we found ourselves, poor programmers building a sub-grid that will display the summary and forcing us to loop through the data provider once more, slowing down our app.

We have included an extended summary version into this great grid component in order to save us time and headaches. The grid makes use of the following components for different displays:

Name Description
TbSumOperation Will display the sum of an specific column
TbCountOfTypeOperation Displays the number of times a type of value appears in the specified column cell (ie. total of zeros, total of ones).
TbPercentOfTypeOperation Displays the percent number of times a type of value appears in the specified column cell (ie. percent of zeros, percent of ones)
TbPercentOfTypeEasyPieOperation Displays the percent number of times a type of value appears in the specified column cell on pie charts. The handling of its display is taken care by the nice jquery plugin easy-pie-chart.
TbPercentOfTypeGooglePieOperation Well, you guessed right. It is the same operation as TbPercentOfTypeEasyPieOperation but with the difference that this pie is more powerful as it can render a single pie to display the summary results.
Important! when using charts you may need to check browser compatibility. As you know, most of these plugins do use canvas for their rendering.

Examples

TbSumOperation
#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15

Total Employee Hours

Total Hours: 60
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}\n{extendedSummary}",
  6. 'columns' => $gridColumns,
  7. 'extendedSummary' => array(
  8. 'title' => 'Total Employee Hours',
  9. 'columns' => array(
  10. 'hours' => array('label'=>'Total Hours', 'class'=>'TbSumOperation')
  11. )
  12. ),
  13. 'extendedSummaryOptions' => array(
  14. 'class' => 'well pull-right',
  15. 'style' => 'width:300px'
  16. ),
  17. ));
TbCountOfTypeOperation
#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15

Expertise

Total Expertise: Css(1) Js(1) html(2)
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}\n{extendedSummary}",
  6. 'columns' => $gridColumns,
  7. 'extendedSummary' => array(
  8. 'title' => 'Expertise',
  9. 'columns' => array(
  10. 'language' => array(
  11. 'label'=>'Total Expertise',
  12. 'types' => array(
  13. 'CSS'=>array('label'=>'Css'),
  14. 'JavaScript'=>array('label'=>'Js'),
  15. 'HTML'=>array('label'=>'html')
  16. ),
  17. 'class'=>'TbCountOfTypeOperation'
  18. )
  19. )
  20. ),
  21. 'extendedSummaryOptions' => array(
  22. 'class' => 'well pull-right',
  23. 'style' => 'width:300px'
  24. ),
  25. ));
TbPercentOfTypeOperation
#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15

Expertise

Total Expertise: Css(25.0%) Js(25.0%) html(50.0%)
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}\n{extendedSummary}",
  6. 'columns' => $gridColumns,
  7. 'extendedSummary' => array(
  8. 'title' => 'Expertise',
  9. 'columns' => array(
  10. 'language' => array(
  11. 'label'=>'Total Expertise',
  12. 'types' => array(
  13. 'CSS'=>array('label'=>'Css'),
  14. 'JavaScript'=>array('label'=>'Js'),
  15. 'HTML'=>array('label'=>'html')
  16. ),
  17. 'class'=>'TbPercentOfTypeOperation'
  18. )
  19. )
  20. ),
  21. 'extendedSummaryOptions' => array(
  22. 'class' => 'well pull-right',
  23. 'style' => 'width:400px'
  24. ),
  25. ));
TbPercentOfTypeEasyPieOperation
#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}\n{extendedSummary}",
  6. 'columns' => $gridColumns,
  7. 'extendedSummary' => array(
  8. 'title' => 'Expertise',
  9. 'columns' => array(
  10. 'language' => array(
  11. 'label'=>'Total Expertise',
  12. 'types' => array(
  13. 'CSS'=>array('label'=>'Css'),
  14. 'JavaScript'=>array('label'=>'Js'),
  15. 'HTML'=>array('label'=>'html')
  16. ),
  17. 'class'=>'TbPercentOfTypeEasyPieOperation',
  18. // you can also configure how the chart looks like
  19. 'chartOptions' => array(
  20. 'barColor' => '#333',
  21. 'trackColor' => '#999',
  22. 'lineWidth' => 8 ,
  23. 'lineCap' => 'square'
  24. )
  25. )
  26. )
  27. ),
  28. 'extendedSummaryOptions' => array(
  29. 'class' => 'well pull-right',
  30. 'style' => 'width:350px'
  31. ),
  32. ))
TbPercentOfTypeGooglePieOperation
#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}\n{extendedSummary}",
  6. 'columns' => $gridColumns,
  7. 'extendedSummary' => array(
  8. 'title' => 'Expertise',
  9. 'columns' => array(
  10. 'language' => array(
  11. 'label'=>'Total Expertise',
  12. 'types' => array(
  13. 'CSS'=>array('label'=>'Css'),
  14. 'JavaScript'=>array('label'=>'Js'),
  15. 'HTML'=>array('label'=>'html')
  16. ),
  17. 'class'=>'TbPercentOfTypeGooglePieOperation',
  18. )
  19. )
  20. ),
  21. 'extendedSummaryOptions' => array(
  22. 'class' => 'well pull-right',
  23. 'style' => 'width:300px'
  24. ),
  25. ))

The Grid/Chart switcher

There will be times, where you wish to see a bit more details on a graphic than just those values on a pie chart. TbExtendedGridView aims to provide an easy to configure chart display, where the data is automatically extracted from its dataProvider.

The way it works, is that you set a couple of configuration options of the chartOptions property and thats it. The grid will automatically display a chart by making use of the great Higcharts JS library. We highly recommend you to get familiar with this library if you are going to make use of this feature.

The chartOptions property is a configurable chart array with three elements:

  • defaultView, which is a flag used to set the chart as the default view of the widget
  • data, which will contain the series attribute of Highcharts JS
  • config, will hold the rest of the chart configuration options
  • htmlOptions, the HTML tag attributes of the chart container.
Important! the style and data-config attributes of the container will be overrided, as they are required for the correct functionality of the chart.

Important: Highcharts JS is not open source for commercial products. We have include this chart library as we thought is one of the best around. We will update the library to support more charts in future releases.

Example

#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}\n{extendedSummary}",
  6. 'columns' => $gridColumns,
  7. 'chartOptions' => array(
  8. 'defaultView' => true,
  9. 'data' => array(
  10. 'series' => array(
  11. array(
  12. 'name' => 'Hours worked',
  13. 'attribute' => 'hours'
  14. )
  15. ),
  16. 'xAxis' => array(
  17. 'categories' => 'firstName',
  18. ),
  19. ),
  20. 'config' => array(
  21. 'chart' => array(
  22. // 'width' => '800' // default reflow
  23. )
  24. ),
  25. ),
  26. ));

The TbGroupGridView widget

We have worked a bit and modified slightly the great work of Vitaliy Potapov, BootGroupView, to provide our library with a grid that features row and cell merging and/or grouping.

Examples

Merge in one column
#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyMan15
  1. $this->widget('booster.widgets.TbGroupGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}",
  6. 'extraRowColumns'=> array('firstLetter'),
  7. 'extraRowExpression' => '"<b style=\"font-size: 3em; color: #333;\">".substr($data->firstName, 0, 1)."</b>"',
  8. 'extraRowHtmlOptions' => array('style'=>'padding:10px'),
  9. 'columns' => $gridColumns,
  10. 'mergeColumns' => array('language')
  11. ));
Merge in two columns
#First nameLast nameLanguageHours workedEdit
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyMan
  1. $this->widget('booster.widgets.TbGroupGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}",
  6. 'extraRowColumns'=> array('firstLetter'),
  7. 'extraRowExpression' => '"<b style=\"font-size: 3em; color: #333;\">".substr($data->firstName, 0, 1)."</b>"',
  8. 'extraRowHtmlOptions' => array('style'=>'padding:10px'),
  9. 'columns' => $gridColumns,
  10. 'mergeColumns' => array('language', 'hours')
  11. ));
Extra Row
Note That TbGroupGridView now has its own js plugin $.fn.yiiGroupGridView() , it is just o copy of the yiiGridView one with some modifications -same API interface- , for example we use the following code to get the currently selected row id:
  1. $('#grid-id').yiiGroupGridView('getSelection');
#First nameLast nameLanguageHours workedEdit
 
 
M
1MarkOttoCSS10 M
J
2JacobThorntonJavaScript20 J
S
3StuDentHTML15 S
4SunnyManHTML15 S
  1. $groupGridColumns = $gridColumns;
  2. $groupGridColumns[] = array(
  3. 'name' => 'firstLetter',
  4. 'value' => 'substr($data->firstName, 0, 1)',
  5. 'headerHtmlOptions' => array('style'=>'display:none'),
  6. 'htmlOptions' =>array('style'=>'display:none')
  7. );
  8.  
  9. $this->widget('booster.widgets.TbGroupGridView', array(
  10. 'filter'=>$person,
  11. 'type'=>'striped bordered',
  12. 'dataProvider' => $gridDataProvider,
  13. 'template' => "{items}",
  14. 'extraRowColumns'=> array('firstLetter'),
  15. 'extraRowExpression' => '"<b style=\"font-size: 3em; color: #333;\">".substr($data->firstName, 0, 1)."</b>"',
  16. 'extraRowHtmlOptions' => array('style'=>'padding:10px'),
  17. 'columns' => $groupGridColumns
  18. ));
Extra Row + Merge
#First nameLast nameLanguageHours workedEdit
 
 
M
1MarkOttoCSS10 M
J
2JacobThorntonJavaScript20 J
S
3StuDentHTML15 S
4SunnyManHTML S
  1. $groupGridColumns = $gridColumns;
  2. $groupGridColumns[] = array(
  3. 'name' => 'firstLetter',
  4. 'value' => 'substr($data->firstName, 0, 1)',
  5. 'headerHtmlOptions' => array('style'=>'display:none'),
  6. 'htmlOptions' =>array('style'=>'display:none')
  7. );
  8.  
  9. $this->widget('booster.widgets.TbGroupGridView', array(
  10. 'filter'=>$person,
  11. 'type'=>'striped bordered',
  12. 'dataProvider' => $groupDataProvider,
  13. 'template' => "{items}",
  14. 'extraRowColumns'=> array('firstLetter'),
  15. 'extraRowExpression' => '"<b style=\"font-size: 3em; color: #333;\">".substr($data->firstName, 0, 1)."</b>"',
  16. 'extraRowHtmlOptions' => array('style'=>'padding:10px'),
  17. 'columns' => $groupGridColumns,
  18. 'mergeColumns' => array('hours')
  19. ));

The TbExtendedFilter widget

There are times, where you wish to save your filters and we are forced to use different types of storage systems to keep track of the filters (ie: Session variables with serialized objects, Cookies, Db, etc).

YiiBooster proposed a different approach and even though the first version is working with JSON storage (at the server side) we want to provide filters to be saved at the database instead.

It use is a bit tricky, you can use it with any CGridView instance of Booster or even the CGridView itself, but the grid has to be extended and then override renderTableHeader() to display it.

Example

Overriding TbGridView

As previously said, we first need to override an CGridView, for the example we are going to make use of TbGridView

  1. // import required classes to my widget
  2. Yii::import('booster.widgets.TbGridView');
  3. Yii::import('booster.widgets.TbExtendedFilter');
  4.  
  5. class WMyGridView extends TbGridView {
  6.  
  7. /**
  8. * We need this attribute in order to fire the saved filter.
  9. * In fact, you could remove its requirement from TbExtendedFilter but
  10. * we thought is better to provide 'less' magic.
  11. */
  12. public $redirectRoute;
  13.  
  14. public function renderTableHeader()
  15. {
  16. if(!$this->hideHeader)
  17. {
  18. echo ">thead<\n";
  19. // Heads up! We are going to display our filter here
  20. $this->renderExtendedFilter();
  21. if($this->filterPosition===self::FILTER_POS_HEADER)
  22. $this->renderFilter();
  23.  
  24. echo "<tr>\n";
  25. foreach($this->columns as $column)
  26. $column->renderHeaderCell();
  27. echo "</tr>\n";
  28.  
  29. if($this->filterPosition===self::FILTER_POS_BODY)
  30. $this->renderFilter();
  31.  
  32. echo "</thead>\n";
  33. }
  34. elseif($this->filter!==null && ($this->filterPosition===self::FILTER_POS_HEADER || $this->filterPosition===self::FILTER_POS_BODY))
  35. {
  36. echo "<thead>\n";
  37. // Heads up! We are going to display our filter here
  38. $this->renderExtendedFilter();
  39. $this->renderFilter();
  40. echo "</thead>\n";
  41. }
  42. }
  43.  
  44. protected function renderExtendedFilter()
  45. {
  46. // at the moment it only works with instances of CActiveRecord
  47. if(!$this->filter instanceof CActiveRecord)
  48. {
  49. return false;
  50. }
  51. $extendedFilter = Yii::createComponent(array(
  52. 'class' => 'TbExtendedFilter',
  53. 'model' => $this->filter,
  54. 'grid' => $this,
  55. 'redirectRoute' => $this->redirectRoute //ie: array('/report/index', 'ajax'=>$this->id)
  56. ));
  57.  
  58. $extendedFilter->init();
  59. $extendedFilter->run();
  60. }
  61. }
Using our widget
Can't see it? Filter the next grid in order to see how the extended grid appears!
Note We save filters on the server, please make good use of this.
Country Code Region Name
CAAlberta
CABritish Columbia
CAManitoba
CANew Brunswick
  1. $this->widget('WMyGridView', array(
  2. 'type' => 'striped bordered',
  3. 'redirectRoute' => CHtml::normalizeUrl(''),
  4. 'dataProvider' => $region->search(),
  5. 'template' => "{items}",
  6. 'filter' => $region,
  7. 'columns' => array(
  8. array('name' => 'country_code', 'header' => 'Country Code'),
  9. array('name' => 'name', 'header' => 'Region Name'),
  10. ))
  11. );?>
Extra Row
#First nameLast nameLanguageHours workedEdit
 
 
M
1MarkOttoCSS10 M
J
2JacobThorntonJavaScript20 J
S
3StuDentHTML15 S
4SunnyManHTML15 S
  1. $groupGridColumns = $gridColumns;
  2. $groupGridColumns[] = array(
  3. 'name' => 'firstLetter',
  4. 'value' => 'substr($data->firstName, 0, 1)',
  5. 'headerHtmlOptions' => array('style'=>'display:none'),
  6. 'htmlOptions' =>array('style'=>'display:none')
  7. );
  8.  
  9. $this->widget('booster.widgets.TbGroupGridView', array(
  10. 'filter'=>$person,
  11. 'type'=>'striped bordered',
  12. 'dataProvider' => $gridDataProvider,
  13. 'template' => "{items}",
  14. 'extraRowColumns'=> array('firstLetter'),
  15. 'extraRowExpression' => '"<b style=\"font-size: 3em; color: #333;\">".substr($data->firstName, 0, 1)."</b>"',
  16. 'extraRowHtmlOptions' => array('style'=>'padding:10px'),
  17. 'columns' => $groupGridColumns
  18. ));
Extra Row + Merge
#First nameLast nameLanguageHours workedEdit
 
 
M
1MarkOttoCSS10 M
J
2JacobThorntonJavaScript20 J
S
3StuDentHTML15 S
4SunnyManHTML S
  1. $groupGridColumns = $gridColumns;
  2. $groupGridColumns[] = array(
  3. 'name' => 'firstLetter',
  4. 'value' => 'substr($data->firstName, 0, 1)',
  5. 'headerHtmlOptions' => array('style'=>'display:none'),
  6. 'htmlOptions' =>array('style'=>'display:none')
  7. );
  8.  
  9. $this->widget('booster.widgets.TbGroupGridView', array(
  10. 'filter'=>$person,
  11. 'type'=>'striped bordered',
  12. 'dataProvider' => $groupDataProvider,
  13. 'template' => "{items}",
  14. 'extraRowColumns'=> array('firstLetter'),
  15. 'extraRowExpression' => '"<b style=\"font-size: 3em; color: #333;\">".substr($data->firstName, 0, 1)."</b>"',
  16. 'extraRowHtmlOptions' => array('style'=>'padding:10px'),
  17. 'columns' => $groupGridColumns,
  18. 'mergeColumns' => array('hours')
  19. ));

Additional Columns

If you thought the above wasn't enough, we have created a couple of extra components (that we expect to grow) that increase the beauty of this grid.

Note: You can also use this column types with regular TbGridView widget.

TbImageColumn

Display an image on your grid. In order to display an image related to the data provider of the grid, you set its imagePathExpression. The expression string will have the following variables passed to it: $row (the row number), $data (the data module of the row) and $this (the column object).

 #First nameLast nameLanguageHours workedEdit
 
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}",
  6. 'columns' => array_merge(array(array('class'=>'booster.widgets.TbImageColumn')),$gridColumns),
  7. ));

TbRelationalColumn

Display relational data or extra info from an ajax call into a dynamically created sub-row. This column is thought in order to display sub-grids or a more extended information about a model. The TbPopoverColumn has a similar behavior but the amount of information that we can display is a less that with TbRelationalColumn.

Important: Do not use this type of column to display a link. It is recommended to work with css classes instead to change the style of its contents.

Important: Be sure to include the required js files if your ajax returns a partial view that requires specific js files.

firstLetter#First nameLast nameLanguageHours workedEdit
 
test-subgrid1MarkOttoCSS10
test-subgrid2JacobThorntonJavaScript20
test-subgrid3StuDentHTML15
test-subgrid4SunnyManHTML15
  1. // on your view
  2. $this->widget('booster.widgets.TbExtendedGridView', array(
  3. 'filter'=>$person,
  4. 'type'=>'striped bordered',
  5. 'dataProvider' => $gridDataProvider,
  6. 'template' => "{items}",
  7. 'columns' => array_merge(array(
  8. array(
  9. 'class'=>'booster.widgets.TbRelationalColumn',
  10. 'name' => 'firstLetter',
  11. 'url' => $this->createUrl('site/relational'),
  12. 'value'=> '"test-subgrid"',
  13. 'afterAjaxUpdate' => 'js:function(tr,rowid,data){
  14. bootbox.alert("I have afterAjax events too!
    This will only happen once for row with id: "+rowid);
  15. }'
  16. )
  17. ),$gridColumns),
  18. ));
  19.  
  20. // on your controller
  21. // example code for action rendering the relational data
  22. public function actionRelational()
  23. {
  24. // partially rendering "_relational" view
  25. $this->renderPartial('_relational', array(
  26. 'id' => Yii::app()->getRequest()->getParam('id'),
  27. 'gridDataProvider' => $this->getGridDataProvider(),
  28. 'gridColumns' => $this->getGridColumns()
  29. ));
  30. }
  31.  
  32. // example code for the view "_relational" that returns the HTML content
  33. echo CHtml::tag('h3',array(),'RELATIONAL DATA EXAMPLE ROW : "'.$id.'"');
  34. $this->widget('booster.widgets.TbExtendedGridView', array(
  35. 'type'=>'striped bordered',
  36. 'dataProvider' => $gridDataProvider,
  37. 'template' => "{items}",
  38. 'columns' => array_merge(array(array('class'=>'booster.widgets.TbImageColumn')),$gridColumns),
  39. ));

TbTotalSumColumn

Sometimes we just need to sum up and there is no need to display it on a summary.

#First nameLast nameLanguageHours worked 
 
   Total Hours60 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget('booster.widgets.TbExtendedGridView', array(
  2. 'filter'=>$person,
  3. 'type'=>'striped bordered',
  4. 'dataProvider' => $gridDataProvider,
  5. 'template' => "{items}",
  6. 'columns' =>array(
  7. array('name'=>'id', 'header'=>'#', 'htmlOptions'=>array('style'=>'width: 60px')),
  8. array('name'=>'firstName', 'header'=>'First name'),
  9. array('name'=>'lastName', 'header'=>'Last name'),
  10. array('name'=>'language', 'header'=>'Language', 'footer'=>'Total Hours'),
  11. array(
  12. 'name'=>'hours',
  13. 'header'=>'Hours worked',
  14. 'class'=>'booster.widgets.TbTotalSumColumn'
  15. ),
  16. array(
  17. 'htmlOptions' => array('nowrap'=>'nowrap'),
  18. 'class'=>'booster.widgets.TbButtonColumn',
  19. 'viewButtonUrl'=>null,
  20. 'updateButtonUrl'=>null,
  21. 'deleteButtonUrl'=>null,
  22. )
  23. ),
  24. ));

TbToggleColumn

Allows to modify the value of column on the fly. This type of column is good when you wish to modify the value of a displayed model that has two states: yes-no, true-false, 1-0, yin-yan, etc...

The widget works in conjunction with an action that will receive the id and the attribute parameters. YiiBooster comes with an action ready to use for that purpose: TbToggleAction.

See the next example on how to use:

Country Code Region Name Toggle
CAAlberta
CABritish Columbia
CAManitoba
CANew Brunswick
  1. $this->widget(
  2. 'booster.widgets.TbGridView',
  3. array(
  4. 'type' => 'striped bordered',
  5. 'dataProvider' => new CActiveDataProvider('Region', array('criteria' => array('condition' => 'id<5'))),
  6. 'template' => "{items}",
  7. 'columns' => array(
  8. array('name' => 'country_code', 'header' => 'Country Code'),
  9. array('name' => 'name', 'header' => 'Region Name'),
  10. array(
  11. 'class' => 'booster.widgets.TbToggleColumn',
  12. 'toggleAction' => 'example/toggle',
  13. 'name' => 'status',
  14. 'header' => 'Toggle'
  15. ),
  16. )
  17. )
  18. );

At the toggleAction endpoint you should have something like this:

  1. public function actions()
  2. {
  3. return array(
  4. 'toggle' => array(
  5. 'class'=>'booster.actions.TbToggleAction',
  6. 'modelName' => 'Region',
  7. )
  8. );
  9. }

Here you see that we install TbToggleAction from YiiBooster inside our controller actions.

TbPopoverColumn

Display whatever content you wish with this cool plugin. It makes use of a bootstrap tooltip plugin that we have developed, so you can configure your picker with the regular bootstrap tooltip options plus an extra one: width.

#First nameLast nameLanguageHours worked 
 
1MarkOttoCSS10
2JacobThorntonJavaScript20
3StuDentHTML15
4SunnyManHTML15
  1. $this->widget(
  2. 'booster.widgets.TbExtendedGridView',
  3. array(
  4. 'filter' => $person,
  5. 'type' => 'striped bordered',
  6. 'dataProvider' => $gridDataProvider,
  7. 'template' => "{items}",
  8. 'columns' => array(
  9. array(
  10. 'name' => 'id',
  11. 'header' => '#',
  12. 'htmlOptions' => array('style' => 'width: 60px')
  13. ),
  14. array('name' => 'firstName', 'header' => 'First name'),
  15. array('name' => 'lastName', 'header' => 'Last name'),
  16. array('name' => 'language', 'header' => 'Language'),
  17. array(
  18. 'name' => 'hours',
  19. 'header' => 'Hours worked',
  20. 'class' => 'booster.widgets.TbPopoverColumn',
  21. 'options' => array(
  22. 'title' => 'Where did I spend my hours',
  23. 'content' => 'js:function(){ return "Why do you want to know?"; }',
  24. )
  25. ),
  26. array(
  27. 'htmlOptions' => array('nowrap' => 'nowrap'),
  28. 'class' => 'booster.widgets.TbButtonColumn',
  29. 'viewButtonUrl' => null,
  30. 'updateButtonUrl' => null,
  31. 'deleteButtonUrl' => null,
  32. )
  33. )
  34. )
  35. );

TbEditableColumn

You can provide of inline editing capabilities with this column to your grid. The following example show how it works. Please, click on any of the region names to edit.

# Country Code Region Name 
2332MLKoulikoro
1152GBSalford
1432GRKerkira
3077RUTyumen'
2894PLMazowieckie
547BZToledo
3505THNong Bua Lamphu
2427MWThyolo
1055GBBirmingham
186ARMisiones
  1. $this->widget(
  2. 'booster.widgets.TbExtendedGridView',
  3. array(
  4. 'type' => 'striped bordered',
  5. 'dataProvider' => new CActiveDataProvider('Region', array(
  6. 'criteria' => array(
  7. 'limit' => 10,
  8. 'order' => 'RAND()'
  9. )
  10. )),
  11. 'template' => "{items}",
  12. 'columns' => array(
  13. array(
  14. 'name' => 'id',
  15. 'header' => '#',
  16. 'htmlOptions' => array('style' => 'width: 60px')
  17. ),
  18. array('name' => 'country_code', 'header' => 'Country Code'),
  19. array(
  20. 'name' => 'name',
  21. 'header' => 'Region Name',
  22. 'class' => 'booster.widgets.TbEditableColumn',
  23. 'headerHtmlOptions' => array('style' => 'width:200px'),
  24. 'editable' => array(
  25. 'type' => 'text',
  26. 'url' => '/example/editable'
  27. )
  28. ),
  29. array(
  30. 'htmlOptions' => array('nowrap' => 'nowrap'),
  31. 'class' => 'booster.widgets.TbButtonColumn',
  32. 'viewButtonUrl' => null,
  33. 'updateButtonUrl' => null,
  34. 'deleteButtonUrl' => null,
  35. )
  36. )
  37. )
  38. );

For the sake of the example, we have just echo the value to the client, but you get the idea on how it should work on the server

  1. $r = Yii::app()->getRequest();
  2. // we can check whether is comming from a specific grid id too
  3. // avoided for the sake of the example
  4. if($r->getParam('editable'))
  5. {
  6. //echo $r->getParam('attribute');
  7. echo $r->getParam('value');
  8. Yii::app()->end();
  9. }