Integrates the Simple Datatables library (https://github.com/fiduswriter/simple-datatables/) as a Symfony UX Stimulus controller, and provides Twig components for common rendering.
composer req survos/simple-datatables-bundleTo turn any HTML <table> into a datatable, simply add the stimulus controller to the tag:
<table class="table" {{ stimulus_controller('@survos/simple-datatables-bundle/table', { perPage: 5, sortable: true }) }}>This component renders a <table> and wires up the datatable stimulus controller.
{% set columns = [
{ name: 'id' },
{ name: 'title', title: 'name' },
'brand',
'price',
] %}
<twig:simple_datatables
perPage="20"
:columns="columns"
:data="products"
>
<twig:block name="price">
${{ row.price|number_format(2) }}
</twig:block>
</twig:simple_datatables>Notes:
columnscan be strings ('price') or arrays passed toSurvos\SimpleDatatables\Model\Column(e.g.{ name: 'price', title: 'Price' }).- Inside a column block you typically get
row,column,idx.
simple_item_grid renders a single item/record as a definition list (<dl>), which is handy for a “show” page.
Component: Survos\SimpleDatatables\Components\ItemGridComponent.
Template: templates/components/item.html.twig.
simple_item_grid renders the fields you list in columns for a single associative array or object passed as data:
<twig:simple_item_grid
:data="product"
:columns="['title', 'brand', 'price']"
/>Columns can be strings or Column-style arrays. For the item grid template, name and title are the relevant fields.
{% set columns = [
{ name: 'title', title: 'Name' },
{ name: 'price', title: 'Price' },
'brand',
] %}
<twig:simple_item_grid :data="product" :columns="columns" />The exclude option is used when the component auto-generates columns. Auto-generation currently only kicks in when data is a list of rows and columns is omitted.
<twig:simple_item_grid :data="items" exclude="internalNotes,ssn" />For custom rendering, define a block named after the column’s name. The block receives data.
<twig:simple_item_grid :data="product" :columns="['title','price']">
<twig:block name="price">
${{ data.price|number_format(2) }}
</twig:block>
</twig:simple_item_grid>The wrapper <div> can be wired to a stimulus controller via stimulusController.
By default it uses @survos/grid-bundle/item_grid.
Cut and paste to create a new Symfony project with a dynamic, searchable datatable, without writing a single line of JavaScript. No webpack or build step either.
symfony new simple-datatables-demo --webapp && cd simple-datatables-demo
rm .git -rf
composer config extra.symfony.allow-contrib true
composer req survos/simple-datatables-bundle
bin/console make:controller Simple -i
cat > templates/simple.html.twig <<END
{% extends 'base.html.twig' %}
{% block body %}
<table class="table" {{ stimulus_controller('@survos/simple-datatables-bundle/table', {perPage: 5, sortable: true}) }}>
<thead>
<tr>
<th>abbr</th>
<th>name</th>
<th>number</th>
</tr>
</thead>
<tbody>
{% for j in 1..12 %}
<tr>
<td>{{ j |date('2023-' ~ j ~ '-01') |date('M') }}</td>
<td>{{ j |date('2023-' ~ j ~ '-01') |date('F') }}</td>
<td>{{ j }}</td>
</tr>
{% endfor %}
</tbody>
</table>
{% endblock %}
END
symfony server:start -d
symfony open:local --path=/simpleOr even easier, use the Twig component. To simplify the example we're going to add the fetch and json_decode functions to Twig; normally that would be done in the controller, but the copy/paste is faster this way.
composer require zenstruck/twig-service-bundle
cat > config/packages/zenstruck_twig_service.yaml <<END
zenstruck_twig_service:
functions:
fetch: file_get_contents # available as "fn_fetch()" in twig
json_decode: json_decode
END
cat > templates/simple.html.twig <<'END'
{% extends 'base.html.twig' %}
{% block body %}
{% set columns = [
{name: 'id'},
{name: 'title', title: 'name'},
'brand',
'price'
] %}
<twig:simple_datatables
perPage="20"
:caller="_self"
:columns="columns"
:data="fn_json_decode(fn_fetch('https://dummyjson.com/products')).products"
>
<twig:block name="price">
${{ row.price|number_format(2) }}
</twig:block>
</twig:simple_datatables>
{% endblock %}
END
symfony server:start -d
symfony open:local --path=/simple