Easily make paginate and orderable list in Symfony2
- Install
Add this in your composer :
"appventus/avlist-bundle": "dev-master"
- How to use it ?
This is a classic action to list an entity :
use Symfony\Component\HttpFoundation\Request;
class FooController extends Controller
{
/**
* Lists all Foo entities.
*
*/
public function indexAction(Request $request)
{
$em = $this->getDoctrine()->getEntityManager();
$qb = $em->getRepository(MyBundle:Foo)->findAll();
// Create an AvList
$list = $this->get('av_list')->getList($qb, null, array(
'max_per_page' => 20,
));
$list->addColumn('title', null, 'myList.column.title', true, 'entity.title');
$list->addColumn('startDate', array(array("name" => "date", "params" => array('d/m/Y'))), 'myList.column.startDate', true, 'entity.startDate');
return array(
'list' => $list,
);
if ($this->get('request')->isXmlHttpRequest()) {
return $this->render($list->getTemplate(), array(
'list' => $list,
));
} else {
return array(
'list' => $list,
);
}
}
}
The addColunmn allow you to add/define the columns to display in the list. In order :
-
- You can define the field to display
-
- Is there some special Twig filters to use on the variable to display correctly ?
-
- The label (th)
-
- Do we may sort on this column ?
-
- If so, on which field
If you want to override default views, you'll have to write an index with the layout, and a partial witch just contain the list:
index.html.twig :
{% extends "MyBundle::layout.html.twig" %}
{% block content %}
<h1>Foo list</h1>
{% include 'MyBundle:Foo:indexPartial.html.twig' with {'list':list}%}
{% endblock %}
Yeah, it's pretty minimalist, but it's very important.
And now our Partial :
{% if list.pager.nbResults > 0 %}
<div id="{{ list.option('id') }}">
<table class="table">
<thead>
<tr>
<th class="sortable" data-target="f.name">Name</th>
<th class="sortable" data-target="f.price">Price</th>
<th>Description</th>
</tr>
</thead>
<tbody>
{% for foo in list.pager %}
<tr>
<td>{{ foo.name }}</td>
<td>{{ foo.price }}</td>
<td>{{ foo.name }}</td>
</tr>
{% endfor %}
</tbody>
</table>
{% include 'AvListBundle:AvList:control.html.twig' with {'list':list}%}
</div>
{% else %}
<p>No Foos :(</p>
{% endif %}
And voila !
- How it works ?
In the partial view, you set on the th tag a class sortable and an id with the value of the field you want to search on. In our exemple, the entity Foo has the alias "f" in the query builder, so to sort by Foo's name, you set the following th id: "f.name".
- Options
This bundle use the PagerFanta bundle to build paginator, more specificly the TwitterBootstrapView. Options are available for this view, and you can pass them as second argument when you instanciate AvList:
$list = $this->get('av_list');
$list->addOption("proximity", 3);
$list->addOption("previous_message", "Précédent");
Please refer to PagerFanta doc for more information
- Multiple list in a page
If you need to have several lists components in a single page, you will have to define the route option (to specify a different url to call that the request one). Optionnally and if your route requires params, you obviously may pass an array through the route_parameters option. Also, you won't be able to use the av_list service because of a singleton pattern issue (if you use the service, you always will use the same object and so on, you will overwrite the values instead of create a new list.
/**
* Lists all Help\FaqItem entities.
*
* @Route("/", name="admin_myentity_list")
* @Route("/ajax/cat/{id}", name="ajax_admin_myentity_list_by_category")
* @Method("GET")
* @Template()
*/
public function indexAction($id = null)
{
$categories = Your_function();
foreach ($categories as $key => $category) {
$list = new AvList($request, $this->get('templating'));
$list->setQueryBuilder($em->getRepository('MyBundle:MyEntity')->createQueryBuilder('e')->where('e.category = :category')->setParameter('category', $category));
$list->setTemplate($partialTemplate);
$list->setOptions(array(
'route' => 'ajax_admin_myentity_list_by_category',
'route_parameters' => array('id' => $category->getId()),
'theme' => 'range',
'maxPerPage' => 5,
'container_class' => $category->getId().'-myentity-list',
'container_id' => $category->getId().'-myentity-list'
));
$params["lists"][$category->getId()] = $list;
}
}
- Custom control theme
By default, AvList provides one theme : range. This theme adds the basics actions to get something like this : <- [10,20] on 100 ->
If you want to, you can create your own theme by setting your theme name as "theme" option and to put it in the app/Resources/AvListBundle/views/AvList/yourCustomThemeFileName.html.twig