Carpenter is a PHP package to make creating HTML tables from a collection of data a breeze.
It also handles paginating, sorting, and makes tables reusable throughout your application.
- Installation
- Laravel 5 Integration
- Laravel 4 Integration
- Creating a Table Instance
- Table Markup
- Setting Table Data
- Adding Columns
- Adding Actions
- Filtering Table Data
- Rendering Tables
This package requires PHP 5.4+, and includes a Laravel 5 Service Provider and Facade.
To install through composer include the package in your composer.json
.
"michaeljennings/carpenter": "1.0.*"
Run composer install
or composer update
to download the dependencies or you can run composer require michaeljennings/carpenter
.
To use the package with Laravel 5 firstly add the carpenter service provider to the list of service providers
in app/config/app.php
.
'providers' => array(
Michaeljennings\Carpenter\CarpenterServiceProvider::class
);
Add the Carpenter
facade to your aliases array.
'aliases' => array(
'Carpenter' => Michaeljennings\Carpenter\Facades\Carpenter::class,
);
Publish the config files using php artisan vendor:publish --provider="Michaeljennings\Carpenter\CarpenterServiceProvider"
To access carpenter you can either use the Facade or the carpenter instance is bound to the IOC container and you can then dependency inject it via its contract.
Carpenter::get('foo');
public function __construct(Michaeljennings\Carpenter\Contracts\Carpenter $carpenter)
{
$this->carpenter = $carpenter;
}
To use the package with Laravel 4 firstly add the carpenter service provider to the list of service providers
in app/config/app.php
.
'providers' => array(
'Michaeljennings\Carpenter\Laravel4ServiceProvider'
);
Add the Carpenter
facade to your aliases array.
'aliases' => array(
'Carpenter' => 'Michaeljennings\Carpenter\Facades\Carpenter',
);
Publish the config files using php artisan config:publish michaeljennings/carpenter
To access carpenter you can either use the Facade or the carpenter instance is bound to the IOC container and you can then dependency inject it via its contract.
Carpenter::get('foo');
public function __construct(Michaeljennings\Carpenter\Contracts\Carpenter $carpenter)
{
$this->carpenter = $carpenter;
}
To get started creating tables firstly you want to make a table instance. There are two different ways to go about this.
Firstly Carpenter allows you to bind a table to a key and then retrieve it when you need it.
To bind an instance use the add
.
$carpenter->add('foo', function($table) {});
You can then retrieve and instance by using the get
method.
$table = $carpenter->get('foo');
Once you've retrieved a table you can run any of the table methods below to alter the table. Or if you wish you can pass a closure as a second parameter and alter the table in there.
$table = $carpenter->get('foo');
$table->column('bar');
$table = $carpenter->get('foo', function($table) {
$table->column('bar');
});
However if you wish to make a single table instance without binding the table then you can use the make
method.
$table = $carpenter->make('foo', function($table) {});
With the make
method the key passed as the first argument is not used to bind the table, but it is used to keep all
session data unique to the table.
Now that you have your table instance it's time to start defining the markup of your table. To this you may either pass
an annonymous function to either the add
or make
method or pass it the name of class to use.
$carpenter->add('foo', function($table) {
// Table logic goes here.
});
$carpenter->add('bar', Bar::class);
By default when using a class to define your table markup you need to provide a build method. This will be passed the table instance.
use Michaeljennings\Carpenter\Contracts\Table;
class Bar
{
public function build(Table $table)
{
//
}
}
If you want to use a different method name then you just need to specify it when you create your table instance by seperating the class name and method name with an @.
$carpenter->add('bar', "Bar@table");
class Bar
{
public function table(Table $table)
{
//
}
}
At present you can either use Eloquent or CodeIgniter models, or multidimensional arrays to populate the table.
To set the model to be used use the model
method.
$table->model(FooModel::class);
Or to use an array pass the array to the data
method.
$table->data($data);
To paginate the table pass the amount of results you wish to show on each page to the paginate
method.
$table->paginate(15);
To add a new column or retrieve an existing column from the table use the column
method.
$table->column('foo');
By default all columns are set to be sortable and this will be handled by carpenter. However if you wish to stop a
column from being sortable the use the unsortable
method. Or if you wish to make the column sortable use the
sortable
method.
$table->column('foo')->sortable();
$table->column('foo')->unsortable();
When sorting the package will attempt to sort by the column key, however if you are accessing a column that doesn't exist in your data store this can cause issues. An example of this would be if you were accessing a mutator in laravel.
To help get around this you can use the sort
method to set a custom closure to be used to sort the column. The
closure will be passed two parameters; the data store you are querying, and whether the sort is in descending order.
$table->column('foo')->sort(function($query, $desc) {
if ($desc) {
$query->orderBy('bar', 'desc');
} else {
$query->orderBy('bar', 'asc');
}
});
By default the column will work out the label to use for the column from the column name. If the column name has an underscore then it will replace it with a space and then it will capitalise all words.
If you want to specify a specific label you can use the setLabel
method.
$table->column('foo'); // Label: Foo
$table->column('foo_bar'); // Label: Foo Bar
$table->column('foo_bar')->setLabel('Baz') // Label: Baz
Occasionally you may wish to format each value in a column. For example if you were showing the price of an item then you may want to format to a currency. To do this you can use a presenter.
To get started use the presenter
method. This is passed an annonymous function for you to format the value.
$table->column('price')->presenter(function($value) {
return '&' . number_format($value, 2);
});
You may also wish to get data from the reset of the row. For example you may only want to show the price if the item is set to online. To do this just pass a second parameter to the closure.
$table->column('price')->presenter(function($value, $row) {
if ($row->online) {
return '&' . number_format($value, 2);
}
});
Occasionally you may need to add buttons or links to each row, or to the top of the table. To do this we use actions.
To add an action to the table, or retrieve and existing action, use the action
method. The first parameter is a key
for the action and the second is the position in the table. By default the actions go to the top of the table but to
put them at the end of the row pass the position as the second argument.
$table->action('create');
$table->action('edit', 'row');
By default actions are set as buttons, however if you set an href attribute it will become an anchor. Or you can use the
setTag
method to set a custom element. When you set an element don't pass any chevrons as this will be added when the
action is rendered.
$table->action('create')->setTag('div');
In the default templates the table is wrapped in a form which is where the actions post to. By default the form posts to
the current url you are on, however if you wish to post to a specific url you can use the setFormAction
method. Also
if you don't want the form to post you can set the method using the setFormMethod
method.
$table->setFormAction('/search');
$table->setFormMethod('GET');
To set a url for the action use the setHref
method.
$table->action('create')->setHref('/create');
If you are using a row action you may also want to access data from the row. To this pass a closure to the setHref
method.
$table->action('edit', 'row')->setHref(function($id) {
return '/edit/' . $id;
});
$table->action('edit', 'row')->setHref(function($id, $row) {
return '/edit/' . $row->slug;
});
To set the label for an action use the setLabel
method.
$table->action('edit', 'row')->setLabel('Edit');
To add a class to the action use the setClass
method.
$table->action('edit', 'row')->setClass('btn');
To set other attributes you can either use the attribute name as the method name, or use the setAttribute
method.
$table->action('edit', 'row')->id('edit-item');
$table->action('edit', 'row')->setAttribute('id', 'edit-item');
Again like with the setHref
method you can pass a closure as the value to get attributes from the row.
$table->action('edit', 'row')->setAttribute('data-id', function($id, $row) {
return $id;
});
For some actions, like deletes you may want the user to confirm that they want to run the action. To this you can use
the confirmed
method. This will leverage the default JavaScript confirm method.
The confirmed
method takes one parameter; either the text you want the confirm box to show, or if you are using a row
action then you can pass a closure to access properties from the table row.
$table->action('delete', 'table')->confirmed('Are you use you to delete that?');
$table->action('delete', 'row')->confirmed(function($id, $row) {
return "Are you sure you want to delete {$row->name}?";
});
This will then add a confirmed attribute to the action.
<button confirmed="Are you use you to delete that?"></button>
Carpenter comes with a jQuery plugin to handle the confirmed functionality or you can listen for the attribute. To use
the plugin just include the carpenter.js file and then run the carpenterJs()
method on the parent element around the
table.
<script type="text/javascript" src="/path/to/script/carpenter.js"></script>
<script type="text/javascript">
$('.table-parent').carpenterJs();
</script>
To set a title for the table use the setTitle
method. In the default templates this will be shown to the top left of
the table.
$table->setTitle('FooBar Table');
Occasionally you may find your self wanting to query the data in the tables. To do this you can use filters.
You can add a filter either when you are creating the table markup or after you have retrieved the table instance.
To use the filter pass a closure to the filter
method and the closure will be passed an instance of the data store
you are using. If you are using a model you can use any of the methods you have access to on that model.
$table->filter(function($q) {
$q->orderBy('foo');
});
$table->get('foo')->filter(function($q) {
$q->where('foo', '=', 'bar');
});
There are two ways to render tables; you can either use a template, or you can get the data from the table instance.
To render a table using a template call the render
method on the table instance.
$table->render();
By default when you call the render method it will use the template set in the config file. If you want to use a
template for a specific table then use the setTemplate
method, or you can pass through a template when you call the
render
method. You can also pass through any data to the template as an array when you call the render
method. If
you want to pass through data but you the default template then just pass through null instead of the path to the
template.
$table->setTemplate('path/to/template.php');
$table->render('path/to/template.php');
$table->render('path/to/template.php', ['foo' => 'bar']);
$table->render(null, ['foo' => 'bar']);
Currently Carpenter supports Laravel, CodeIgniter and a native PHP template renderer's. You can set which renderer to use in the config file.
Carpenter also comes with three default templates, but it's very simple to make your own.
To get the table columns use the getColumns()
method.
foreach ($table->getColumns() as $column) {
//
}
Get the column HTML attributes.
$table->getAttributes();
Get the column heading label.
$table->getLabel();
Check if the column is sortable.
$table->isSortable();
Get the column href.
$table->getHref();
To get the rows use the getRows
method.
foreach ($table->getRows() as $row) {
//
}
You can also check if the table has any rows with the hasRows
method.
if ($table->hasRows()) {
//
}
To get the cells in the row use the getCells
method.
foreach ($row->getCells() as $cell) {
echo $cell->value;
}
To check if the row has any actions use the hasActions
method.
if ($row->hasActions()) {
//
}
Then to get the actions for the row use the getActions
method. Finally to render the action use the render
method on
the action.
foreach ($row->getActions() as $action) {
echo $action->render();
}
To check if the table has any table actions use the hasActions
method.
if ($table->hasActions()) {
//
}
To get the actions to play at the top of the table use the getActions
method. Then to render the action use the
render
method on the action.
foreach ($table->getActions() as $action) {
echo $action->render();
}
To check if the table has any pagination links use the hasLinks
method.
if ($table->hasLinks()) {
//
}
Then to get the pagination links use the getLinks
method.
echo $table->getLinks();
To get the title use the getTitle
method.
$table->getTitle();
To get the form action use the getFormAction
method.
$table->getFormAction();
Also to get the form method use the getFormMethod
method.
$table->getFormMethod();