---
title: Symfony DataTables Bundle
toc_footers:
- Install from Packagist
- Fork me on GitHub
- Report an issue
- Omines Full Service Internetbureau
search: true
---
# Introduction
This bundle provides convenient integration of the popular [DataTables](https://datatables.net/) jQuery library
for realtime AJAX tables in your Symfony 4.1+ application.
Designed to be fully pluggable there are no limits to the data sources you can display through this library, nor
are there any bounds on how they are displayed. In full *'batteries included but replaceable'* philosophy there are
ready made adapters for common use cases like Doctrine ORM, but they are trivial to replace or extend.
# Installation
Recommended way of installing this library is through [Composer](https://getcomposer.org/).
composer require omines/datatables-bundle
Please ensure you are using Symfony 4.1 or later. If you are using Symfony Flex a recipe is included in the contrib
repository, providing automatic installation and configuration.
```php?start_inline=true
public function registerBundles()
{
// After Symfony's own bundles
new \Omines\DataTablesBundle\DataTablesBundle(),
// Before your application bundles
}
```
After installation, if not using Flex, you should register the bundle to your kernel, commonly `AppKernel.php`,
before your own bundles but after the required external bundles, such as `FrameworkBundle` and `TwigBundle`.
Run the `assets:install` command to deploy the included Javascript files to your application's public folder.
bin/console assets:install
That last step is optional, as you can also load it through Assetic or WebPack, but a good starting point.
## Clientside dependencies
```html
```
All serverside dependencies are managed by Composer. Clientside dependencies are left up the implementer to decide
how to include which DataTables dependencies. As long are you are using a fairly up to date version of DataTables you
should be fine, as the bundle does not use exotic features or depend on plugins.
The code snippets here should get you started quickly, including jQuery 3. For more extensive download options visit
[https://datatables.net/download/](https://datatables.net/download/).
# Quickstart
```php?start_inline=true
use Omines\DataTablesBundle\Adapter\ArrayAdapter;
use Omines\DataTablesBundle\Column\TextColumn;
use Omines\DataTablesBundle\DataTableFactory;
class MyController extends Controller
{
public function showAction(Request $request, DataTableFactory $dataTableFactory)
{
$table = $dataTableFactory->create()
->add('firstName', TextColumn::class)
->add('lastName', TextColumn::class)
->createAdapter(ArrayAdapter::class, [
['firstName' => 'Donald', 'lastName' => 'Trump'],
['firstName' => 'Barack', 'lastName' => 'Obama'],
])
->handleRequest($request);
if ($table->isCallback()) {
return $table->getResponse();
}
return $this->render('list.html.twig', ['datatable' => $table]);
}
}
```
This trivial bit of code in your controller prepares a fully functional DataTables instance for use.
The DataTableFactory
service is injected to expose convenience methods in your controller
for easy instantiation. The `create` function is used in this example. On the DataTable instance we
add 2 columns of type `TextColumn`, and we bind it to an adapter providing a static array as the
source of the data.
The `handleRequest` function will take care of handling any callbacks, similar to how Symfony's Form component
works. If it turns out the request originated from a callback we let the table provide the controller response,
otherwise we render a template with the table provided as a parameter.
## Controller setup
Previous versions of this bundle offered a DataTablesTrait
which assumed that the
DataTableFactory
class was available in the controller's $container
. As this
is deprecated in current versions of Symfony you should use dependency injection instead.
## Frontend code
```html
DataTableRendererInterface
.
template | string | Default template to be used for rendering the basic HTML table in your templates.
template_parameters | object | Default parameters to be passed to the template during rendering.
translation_domains | string | Default Symfony Translation Domain used where translations are used.
All settings can be overridden on individual tables by calling the corresponding setter function,
ie. `setLanguageFromCDN(bool)`.
The `options` are passed (almost) verbatim to the DataTables clientside constructor. Refer to the
[external documentation](https://datatables.net/reference/option/) below for details on individual
options. Only options which are meaningful to be defined serverside can be set at this level, so
setting callbacks and events is not possible. These are however easily set on the Javascript end.
## Table configuration
Configuring a specific table is done mainly via the methods on `DataTable`. The most common call
is `add` to add an extra column to the table as seen in all the examples. More utility methods
exist and can be chained.
`->addOrderBy($column, string $direction = DataTable::SORT_ASCENDING)`
Will set the default sort of the table to the specified column and direction. Repeatable to sort
by multiple columns.
# Adapters
Adapters are the core elements bridging DataTables functionality to their underlying data source.
Popular implementations for common data sources are provided, and more are welcomed.
An adapter is called by the bundle when a request for data has been formulated, including search
and sorting criteria, and returns a result set with metadata on record counts.
Ready-made adapters are supplied for easy integration with various data sources.
## Doctrine ORM
```php?start_inline=1
use Omines\DataTablesBundle\Adapter\Doctrine\ORMAdapter;
$table = $this->createDataTable()
->add('firstName', TextColumn::class)
->add('lastName', TextColumn::class)
->add('company', TextColumn::class, ['field' => 'company.name'])
->createAdapter(ORMAdapter::class, [
'entity' => Employee::class,
]);
```
If you have installed `doctrine/orm` and `doctrine/doctrine-bundle` you can use the provided `ORMAdapter`.
Assume a simple `Employee` table with some basic fields and a ManyToOne relationship to `Company` for
these examples.
The `ORMAdapter` has a single mandatory property `entity`, which should be set to the full FQCN of the main
entity the table is showing.
Underneath a lot of "magic" is happening in this most simple of examples. The first 2 columns automatically
have their `field` option defaulted to the "root entity" of the adapter, with the field identical to their
name. The adapter itself did not get a query, and as such injected the `AutomaticQueryBuilder` supplied by
this bundle, which scans the metadata and automatically joins and selects the right data based on the fields.
Secondly, since no criteria processors were supplied a default `SearchCriteriaProvider` was injected to
apply global search to all mapped fields.
Of course, all of this is just convenient default. For more complex scenarios you can supply your own query
builders and criteria providers, and even chain them together to easily implement multiple slightly different
tables in your site.
### Customizing queries
```php?start_inline=1
$table->createAdapter(ORMAdapter::class, [
'entity' => Employee::class,
'query' => function (QueryBuilder $builder) {
$builder
->select('e')
->addSelect('c')
->from(Employee::class, 'e')
->leftJoin('e.company', 'c')
;
},
]);
```
If you do not specify the `query` option the stock `AutomaticQueryBuilder` is used, which automatically
joins the main entity to its relationships recursively to select *only* the fields defined by the columns.
This works fine in many cases, in others you may need to customize the query.
The `query` property can be set to a single instance or an array of processors, which can either be
callables taking a single `QueryBuilder` as a parameter, or a (anonymous) class implementing the
`QueryBuilderProcessorInterface`. In case of an array the processors are called in the defined order.
In general it is recommended to implement your own query processor completely when you need custom
behavior. Chaining on the default `AutomaticQueryBuilder` is possible, but may cause unexpected
interaction based on internal changes in this bundle and/or Doctrine ORM.
### Customizing criteria
```php?start_inline=1
$table->createAdapter(ORMAdapter::class, [
'entity' => Employee::class,
'criteria' => [
function (QueryBuilder $builder) {
$builder->andWhere($builder->expr()->like('c.name', ':test'))->setParameter('test', '%ny 2%');
},
new SearchCriteriaProvider(),
],
]);
```
Analogous to queries you can separately define the criteria processors applied to table queries. The
`criteria` property also takes a single instance or an array, with the separate processors either
implementing `QueryBuilderProcessorInterface`, or being a callback returning a `Criteria` object as
in this example.
Note that implementing your own criteria overrides the default, meaning searching and sorting will no
longer work automatically. Add the `SearchCriteriaProvider` manually to combine the default behavior
with your own implementation.
### Events
```php?start_inline=1
$table->createAdapter(ORMAdapter::class, [
'entity' => Employee::class,
'query' => function (QueryBuilder $builder) {
$builder
->select('e')
->addSelect('c')
->from(Employee::class, 'e')
->leftJoin('e.company', 'c')
;
},
]);
$table->addEventListener(ORMAdapterEvents::PRE_QUERY, function(ORMAdapterQueryEvent $event) {
$event->getQuery()->useResultCache(true)->useQueryCache(true);
});
```
The `PRE_QUERY` event is dispatched after the QueryBuilder built the Query
and before the iteration starts. It can be useful to configure the cache.
## Elastica
```php?start_inline=1
use Omines\DataTablesBundle\Adapter\Elasticsearch\ElasticaAdapter;
$table = $this->createDataTable()
->setName('log')
->add('timestamp', DateTimeColumn::class, ['field' => '@timestamp', 'format' => 'Y-m-d H:i:s', 'orderable' => true])
->add('level', MapColumn::class, [
'default' => 'Unknown',
'map' => ['Emergency', 'Alert', 'Critical', 'Error', 'Warning', 'Notice', 'Info', 'Debug'],
])
->add('message', TextColumn::class, ['globalSearchable' => true])
->createAdapter(ElasticaAdapter::class, [
'client' => ['host' => 'elasticsearch'],
'index' => 'logstash-*',
]);
```
If you have installed `ruflin/elastica` you can use the provided `ElasticaAdapter` to use ElasticSearch
indexes as the data source.
## MongoDB
```php?start_inline=1
use Omines\DataTablesBundle\Adapter\MongoDB\MongoDBAdapter;
$table = $this->createDataTable()
->add('name', TextColumn::class)
->add('company', TextColumn::class)
->createAdapter(MongoDBAdapter::class, [
'collection' => 'myCollection',
]);
```
If you have installed `mongodb/mongodb` you can use the provided `MongoDBAdapter` to use MongoDB
collections as the data source.
## Arrays
TBD.
## Implementing custom adapters
TBD.
# Columns
Column classes derive from `AbstractColumn`, and implement the transformations required to convert
raw data into output ready for rendering in a DataTable.
A number of standard columns are provided for common use cases, but you can easily add your own column
types for application specific purposes.
### Common options
```php?start_inline=1
# Some example columns
$table
->add('firstName', TextColumn::class, ['label' => 'customer.name', 'className' => 'bold'])
->add('lastName', TextColumn::class, ['render' => '%s', 'raw' => true])
->add('email', TextColumn::class, ['render' => function($value, $context) {
return sprintf('%s', $value, $value);
}])
;
```
All column types have the following options:
Option | Type | Description
------ | ---- | -----------
label | string | Basic translation label shown in the header of the table. Defaults to the name of the column.
data | string/callable/`null` | The default value if a `null` value is encountered, or a callable function to transform data.
field | string/`null` | A field mapping to be used by adapters to fill data.
propertyPath | string/`null` | A property path to be applied to the raw adapter row data.
visible | bool | Whether the column will be visible. Default true.
orderable | bool/`null` | Whether the column can be sorted upon. Defaults to the presence of the `orderField`.
orderField | string/`null` | The field to order by when the column is sorted. Defaults to the value of `field`.
searchable | bool/`null` | Whether the column can be searched upon. Defaults to the presence of `field`.
globalSearchable | bool/`null` | Whether the column participates in global searches. Defaults to the presence of `field`.
className | string/`null` | A CSS class to be applied to all cells in this column.
render | string/callable/`null` | Either a [`sprintf` compatible format string](http://php.net/manual/en/function.sprintf.php), or a callable function providing rendering conversion, or default `null`.
## TextColumn
```php?start_inline=1
$table->add('customerName', TextColumn::class, ['field' => 'customer.name']);
```
Text columns are the most frequently used column type, as they can be used to display any kind of data
that is eventually rendered as plain text.
The `TextColumn` type exposes a single option on top of its ancestor `AbstractColumn`:
Option | Type | Description
------ | ---- | -----------
raw | bool | Do not escape cell content to be safe for use in HTML. Default `false`.
## BoolColumn
```php?start_inline=1
$table->add('wantsNewsletter', BoolColumn::class, [
'trueValue' => 'yes',
'falseValue' => 'no',
'nullValue' => 'unknown',
]);
```
Bool columns render a boolean value, which is allowed to be indeterminate (`null`). Three properties define
how values are rendered:
Option | Type | Description
------ | ---- | -----------
trueValue | string | Raw string to use for true-ish values. Default `true`.
falseValue | string | Raw string to use for false-ish values. Default `false`.
nullValue | string | Raw string to use for null values. Default `null`.
## DateTimeColumn
```php?start_inline=1
$table->add('registrationDate', DateTimeColumn::class, ['format' => 'd-m-Y']);
```
DateTime columns render a `\DateTimeInterface` implementing class, such as `\DateTime`, to a string
result. If data of other types is encountered automatic conversion is attempted following [common PHP formats](http://php.net/manual/en/datetime.formats.php).
Option | Type | Description
------ | ---- | -----------
createFromFormat | string | Custom format for creating DateTime objects from values. A format string accepted by [`DateTime::createFromFormat()`](https://www.php.net/manual/en/datetime.createfromformat.php) function.
format | string | A date format string as accepted by the [`date()`](http://php.net/manual/en/function.date.php) function. Default `'c'`.
nullValue | string | Raw string to display for null values. Defaults to the empty string.
## MapColumn
```php?start_inline=1
$table->add('gender', MapColumn::class, [
'default' => 'not provided',
'map' => [
'f' => 'Female',
'm' => 'Male',
],
]);
```
Map columns used to transform a discrete collection of values into proper display counterparts. This
can be useful to convert enumerated fields such as log severity levels, or genders as in the example.
Fields which are not present in the column's map return the default value, if this is `null` the
source value itself is shown unmodified.
The `MapColumn` type extends `TextColumn`, as such inheriting the `raw` property, while adding its own:
Option | Type | Description
------ | ---- | -----------
default | string/`null` | Value to be shown for source data not found in the map. Default `null`.
map | array | Associative array containing available mappings. Mandatory without default.
## TwigColumn
```php?start_inline=1
$table->add('buttons', TwigColumn::class, [
'className' => 'buttons',
'template' => 'tables/buttonbar.html.twig',
])
```
This column type allows you to specify a Twig template used to render the column's cells. The
template is rendered using the main application context by injecting the main Twig service.
Additionally the `value` and `row` parameters are being filled by the cell value and the row
level context respectively.
Option | Type | Description
------ | ---- | -----------
template | string | Template path resolvable by the Symfony templating component. Required without default.
## TwigStringColumn
```php?start_inline=1
$table->add('link', TwigStringColumn::class, [
'template' => '{{ row.firstName }} {{ row.lastName }}',
])
```
This column type allows you to inline a Twig template as a string used to render the column's cells. The
template is rendered using the main application context by injecting the main Twig service.
Additionally, the `value` and `row` parameters are being filled by the cell value and the row
level context respectively.
This column type requires `StringLoaderExtension` to be [enabled in your Twig environment](https://symfony.com/doc/4.4/reference/dic_tags.html#twig-extension).
```yaml
services:
Twig\Extension\StringLoaderExtension:
tags: [twig.extension]
```
Option | Type | Description
------ | ---- | -----------
template | string | Template content resolvable by the Symfony templating component. Required without default.
## Implementing custom columns
TBD.
# DataTable Types
```php?start_inline=1
$table = $this->createDataTableFromType(PresidentsTableType::class)
->handleRequest($request);
```
Having the table configuration in your controller is convenient, but not practical for reusable or
extensible tables, or highly customized tables. In the example above we could also create a class
`DataTable\Type\PresidentsTableType` in our app bundle, and make it implement
`Omines\DataTablesBundle\DataTableTypeInterface`. We can then use the code illustrated here to
instantiate the reusable class in the controller.
This ensures your controllers stay lean and short, and only delegate tasks. The first parameter
takes either a Fully Qualified Class Name (FQCN) to instantiate the class dynamically, or a
registered service with a `datatables.type` tag. Use a service if you need to inject dependencies
dynamically. When using Symfony's autoconfiguration the tag will be applied automatically.
Of course you can modify the base type to fit the controller's specific needs before calling
`handleRequest`. Secondly, the `createDataTableFromType` function accepts an array as a second
argument which is passed to the type class for parametrized instantiation.
# Javascript
```javascript
$('#table1').initDataTables({{ datatable_settings(datatable1) }}, {
searching: true,
dom:'<"html5buttons"B>lTfgitp',
buttons: [
'copy',
{ extend: 'pdf', title: 'domains'},
{ extend: 'print' }
]
}).then(function(dt) {
// dt contains the initialized instance of DataTables
dt.on('draw', function() {
alert('Redrawing table');
})
});
```
During the quickstart we introduced the `initDataTables` Javascript function, taking the serverside
settings as its argument. The function takes an optional second argument, which is merged into the
serverside settings to override any template-specific changes, but as this is executed in the browser
it also means this is where you can add Javascript events according to DataTables documentation.
The function returns a [`Promise`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise)
which is fulfilled with the `DataTables` instance once initialization is completed. This allows you
all the flexibility you could need to [invoke API functions](https://datatables.net/reference/api/).
# Legal
This software was developed for internal use at [Omines Full Service Internetbureau](https://www.omines.nl/)
in Eindhoven, the Netherlands. It is shared with the general public under the permissive MIT license, without
any guarantee of fitness for any particular purpose. Refer to the included `LICENSE` file for more details.