gleisonnanet / serverside-datatables-bundle
DataTables integration for Symfony
dev-master
2020-01-14 16:58 UTC
Requires
- php: >=5.6|>=7.0.0
- symfony/dependency-injection: ~2.7|~3.0|~4.0
Requires (Dev)
- doctrine/doctrine-bundle: ~1.4
- phpunit/phpunit: ^5.0
- symfony/framework-bundle: ~2.7|~3.0|~4.0
This package is auto-updated.
Last update: 2025-03-15 05:00:53 UTC
README
What it does
The DataTablesBundle let's you easily create (sortable and filterable) serverSide DataTables from Doctrine entities.
License
This bundle is under the MIT license. See the complete license in the bundle:
LICENSE
Installation
Enable the bundle by adding the following line in the app/AppKernel.php
file of your project:
# app/AppKernel.php class AppKernel extends Kernel { public function registerBundles() { $bundles = array( // ... new gleisonnanet\DataTablesBundle\gleisonnanetDataTablesBundle(), ); // ... } // ... }
Configuration
After installing the bundle, make sure you add this route to your routing:
# app/config/routing.yml
datatables:
resource: "@gleisonnanetDataTablesBundle/Resources/config/routing.xml"
Localization
# app/config/config.yml
gleisonnanet_data_tables:
localization:
locale: "%locale%"
data:
true: "Jepp"
false: "Nope"
datetime: "Y-m-d H:i:s"
Usage
Create a Table definition
# AppBundle/DataTable/CustomerTable.php <?php namespace AppBundle\DataTable; use gleisonnanet\DataTablesBundle\Table\AbstractDataTable; use gleisonnanet\DataTablesBundle\Table\TableBuilderInterface; use gleisonnanet\DataTablesBundle\Table\TableOptions; use gleisonnanet\DataTablesBundle\Table\TableSettings; use gleisonnanet\DataTablesBundle\Table\Column\Column; use gleisonnanet\DataTablesBundle\Table\Column\UnboundColumn; use gleisonnanet\DataTablesBundle\Table\Column\CallbackColumn; use gleisonnanet\DataTablesBundle\Table\Column\EntityColumn; use gleisonnanet\DataTablesBundle\Table\Column\EntitiesColumn; class CustomerTable extends AbstractDataTable { protected function configure(TableSettings $settings, TableOptions $options) { $settings->setName('customer'); $settings->setEntity('App\Entity\Customer'); $options['stateSave'] = true; } protected function build(TableBuilderInterface $builder) { $builder ->add('id') ->add('firstname') ->add('lastname', Column::class, [ 'label' => 'Lastname' ]) ->add('opening', UnboundColumn::class, [ 'callback' => function(Customer $customer) { return 'Dear ' . ('f' === $customer->getGender() ? 'Madam' : 'Sir'); } ]) ->add('status', CallbackColumn::class, [ 'callback' => function($status) { switch ($status) { case 1: return 'something'; break; case 2: return 'something else'; break; default: return 'invalid'; break; } ]) ->add('group.name')) // customer has one group ->addColumn(new EntityColumn('state', 'city.state', 'name')) // customer has one city. city has one state ->addColumn(new EntitiesColumn('orders', 'orders', 'number')) // customer has many orders ->addColumn(new EntitiesCountColumn('addresses_count', 'addresses')) // customer has many addresses ->addColumn('actions', ActionsColumn::class, [ 'actions' => [ 'edit' => [ 'title' => 'edit customer', 'label' => '<i class="fa fa-edit"></i>', 'callback' => function(Customer $customer, \Symfony\Component\Routing\RouterInterface $router) { return $router->generate('customer_edit', ['id' => $customer->getId()]); }, ], ], 'dropdown' => true, 'dropdown_label' => 'Actions', ]) ; } }
In your CustomerController
# AppBundle/Controller/CustomerController.php use AppBundle\DataTable\CustomerTable; class CustomerController extends Controller { public function indexAction() { return $this->render('AppBundle:Customer:index.html.twig', [ 'table' => new CustomerTable(), ]); } }
And in your index template
# AppBundle/Resources/views/Customer/index.html.twig
{% extends '::base.html.twig' %}
{% block body %}
{{ datatables_html(table) }}
{% endblock %}
{% block javascripts %}
<script>
{{ datatables_js(table) }}
// access the table instance
{{ table.name }}_table.on('dblclick', 'tbody tr', function () {
alert('dblclick');
});
</script>
{% endblock %}
Access DI container
use gleisonnanet\DataTablesBundle\Table\AbstractDataTable; class CustomerTable extends AbstractDataTable { protected function build() { $service = $this->container->get('service'); // or short $service = $this->get('service'); // ... ->addColumn(new Column('id', 'id', [ 'format_data_callback' => function($data, $object, Column $column) { $router = $this->container->get('router'); // or $router = $this->get('router'); }, ])) // ... } }
Table definition as a service
Define the service
# AppBundle/Resources/config/services.xml
<service id="app.table.customer" class="AppBundle\DataTable\CustomerTable">
<argument type="service" id="my.awesome.service" />
</service>
Set the service id in the table constructor
# AppBundle/DataTable/CustomerTable.php private $myAwesomeService; public function __construct($myAwesomeService) { $this->myAwesomeService = $myAwesomeService; } protected function configure(TableSettings $settings, TableOptions $options) { $settings->setName('customer'); $settings->setEntity('AppBundle\Entity\Customer'); $settings->setServiceId('app.table.customer'); }
In your controller
# AppBundle/Controller/CustomerController.php public function indexAction() { return $this->render('AppBundle:Customer:index.html.twig', [ 'table' => $this->get('app.table.customer'), ]); }
Column filter
# AppBundle/DataTable/CustomerTable.php // ... class CustomerTable extends AbstractDataTable { // ... protected function build() { $this // ... ->add('gender', Column:class, [ 'filter' => 'select', 'filter_choices' => [ 'm' => 'male', 'f' => 'female', ], ]) ->add('lastname', Column::class, [ 'filter' => 'text', ]) ; } }
Table options
$default = [ 'stateSave' => false, 'stateDuration' => 7200, ];
- 'stateDuration': -1 sessionStorage. 0 or greater localStorage. 0 infinite. > 0 duration in seconds
class CustomerTable extends AbstractTableDefinition { // ... protected function configure(TableSettings $settings, TableOptions $options) { $options['stateSave'] = true; $options['stateDuration'] = 120; } }
Column options
$default = [ 'sortable' => true, 'searchable' => true, 'filter' => false, // false|'text'|'select' 'filter_choices' => [], 'filter_empty' => false, // add a checkbox to filter empty resp. null values 'multiple' => false, 'expanded' => false, 'format_data_callback' => null, // function ($data, $object, Column $column) {} 'unbound' => false, 'order' => null, // null|'asc'|'desc' 'label' => null, // null|string|false 'abbr' => null, // null|string ];
- 'filter' != false implies 'searchable' = true
- 'multiple' has no effect if filter != 'select'
- 'expanded' has no effect if filter != 'select'