adsazad/datatables-bundle
Symfony bundle integrating jQuery DataTables for realtime Ajax tables (Symfony 4.4+/5+). Data source logic is decoupled via adapters; includes Doctrine ORM, MongoDB and Elastica, with support for custom adapters. Includes a console generator command.
title: Symfony DataTables Bundle
toc_footers:
search: true
This bundle provides convenient integration of the popular DataTables 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.
Recommended way of installing this library is through Composer.
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.
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.
<!-- in the <head> section -->
<link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/v/dt/jq-3.2.1/dt-1.10.16/datatables.min.css"/>
<!-- before the closing <body> tag -->
<script type="text/javascript" src="https://cdn.datatables.net/v/dt/jq-3.2.1/dt-1.10.16/datatables.min.js"></script>
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/.
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.
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.
<!-- Insert this where you want the table to appear -->
<div id="presidents">Loading...</div>
<!-- Insert this at the end of your body element, but before the closing tag -->
<script src="{{ asset('bundles/datatables/js/datatables.js') }}"></script>
<script>
$(function() {
$('#presidents').initDataTables({{ datatable_settings(datatable) }});
});
</script>
In your Twig template, list.html.twig in the example, we need to ensure the HTML has a container element
ready to contain the table. During load its contents will be erased, so you can put a loading indicator in
there like we did here.
Then you include the Javascript deployed to your public folder, and run a single command on a jQuery
selection of the container element. The datatable_settings Twig function will render a compact JSON
string with the configured settings required for initialization.
And that's it, the library will take it from here and your table will be shown on your webpage!
datatables:
# Load i18n data from DataTables CDN or locally
language_from_cdn: true
# Default HTTP method to be used for callbacks
method: POST # One of "GET"; "POST"
# Default options to load into DataTables
options:
option: value
# Where to persist the current table state automatically
persist_state: fragment # One of "none"; "query"; "fragment"; "local"; "session"
# Default service used to render templates, built-in TwigRenderer uses global Twig environment
renderer: Omines\DataTablesBundle\Twig\TwigRenderer
# Default template to be used for DataTables HTML
template: '[@DataTables](https://github.com/DataTables)/datatable_html.html.twig'
# Default parameters to be passed to the template
template_parameters:
# Default class attribute to apply to the root table elements
className: 'table table-bordered'
# If and where to enable the DataTables Filter module
columnFilter: null # One of "thead"; "tfoot"; "both"; null
# Default translation domain to be used
translation_domain: messages
Global configuration of the bundle is done in your Symfony config file. The default configuration is shown here, and should be fine in most cases. Most settings can be overridden per table, but for most applications you will want to make changes at the global level so they are applied everywhere, providing a uniform look and feel.
The following settings exist at the configuration level:
| Option | Type | Description |
|---|---|---|
| language_from_cdn | bool | Load i18n files from DataTables CDN or from Symfony Translations. |
| options | object | Default options that will be passed to DataTables clientside initialization. |
| method | string | Either GET or POST to indicate which HTTP method to use for callbacks. |
| renderer | string | Service used to render the table HTML, which must implement the 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 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.
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 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.
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.
$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.
$table->createAdapter(ORMAdapter::class, [
'entity' => Employee::class,
'query' => [
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.
$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.
use Omines\DataTablesBundle\Adapter\Elasticsearch\ElasticaAdapter;
$table = $this->createDataTable()
->setName('log')
->add('timestamp', DateTimeColumn::class, ['field' => '[@timestamp](https://github.com/timestamp)', 'format' => 'Y-m-d H:i:s', 'orderable' => true])
->add('level', MapColumn::class, [
'default' => '<span class="label label-default">Unknown</span>',
'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.
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.
TBD.
TBD.
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.
# Some example columns
$table
->add('firstName', TextColumn::class, ['label' => 'customer.name', 'className' => 'bold'])
->add('lastName', TextColumn::class, ['render' => '<strong>%s</strong>', 'raw' => true])
->add('email', TextColumn::class, ['render' => function($value, $context) {
return sprintf('<a href="%s">%s</a>', $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, or a callable function providing rendering conversion, or default null. |
$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. |
$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. |
$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.
| Option | Type | Description |
|---|---|---|
| createFromFormat | string | Custom format for creating DateTime objects from values. A format string accepted by DateTime::createFromFormat() function. |
| format | string | A date format string as accepted by the date() function. Default 'c'. |
| nullValue | string | Raw string to display for null values. Defaults to the empty string. |
$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. |
$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. |
$table->add('link', TwigStringColumn::class, [
'template' => '<a href="{{ url(\'employee.edit\', {id: row.id}) }}">{{ row.firstName }} {{ row.lastName }}</a>',
])
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.
services:
Twig\Extension\StringLoaderExtension:
tags: [twig.extension]
| Option | Type | Description |
|---|---|---|
| template | string | Template content resolvable by the Symfony templ... |
How can I help you explore Laravel packages today?