base_list.html.twig 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. {#
  2. This file is part of the Sonata package.
  3. (c) Thomas Rabaix <thomas.rabaix@sonata-project.org>
  4. For the full copyright and license information, please view the LICENSE
  5. file that was distributed with this source code.
  6. #}
  7. {% extends base_template %}
  8. {%- block actions -%}
  9. {% include 'SonataAdminBundle:CRUD:action_buttons.html.twig' %}
  10. {%- endblock -%}
  11. {% block tab_menu %}{{ knp_menu_render(admin.sidemenu(action), {'currentClass' : 'active', 'template': sonata_admin.adminPool.getTemplate('tab_menu_template')}, 'twig') }}{% endblock %}
  12. {% block list_table %}
  13. <div class="col-xs-12 col-md-12">
  14. {% set batchactions = admin.batchactions %}
  15. {% if admin.hasRoute('batch') and batchactions|length %}
  16. <form action="{{ admin.generateUrl('batch', {'filter': admin.filterParameters}) }}" method="POST" >
  17. <input type="hidden" name="_sonata_csrf_token" value="{{ csrf_token }}">
  18. {% endif %}
  19. {# Add a margin if no pager to prevent dropdown cropping on window #}
  20. <div class="box box-primary" {% if admin.datagrid.pager.lastPage == 1 %}style="margin-bottom: 100px;"{% endif %}>
  21. <div class="box-body {% if admin.datagrid.results|length > 0 %}table-responsive no-padding{% endif %}">
  22. {{ sonata_block_render_event('sonata.admin.list.table.top', { 'admin': admin }) }}
  23. {% block list_header %}{% endblock %}
  24. {% if admin.datagrid.results|length > 0 %}
  25. <table class="table table-bordered table-striped sonata-ba-list">
  26. {% block table_header %}
  27. <thead>
  28. <tr class="sonata-ba-list-field-header">
  29. {% for field_description in admin.list.elements %}
  30. {% if admin.hasRoute('batch') and field_description.getOption('code') == '_batch' and batchactions|length > 0 %}
  31. <th class="sonata-ba-list-field-header sonata-ba-list-field-header-batch">
  32. <input type="checkbox" id="list_batch_checkbox">
  33. </th>
  34. {% elseif field_description.getOption('code') == '_select' %}
  35. <th class="sonata-ba-list-field-header sonata-ba-list-field-header-select"></th>
  36. {% elseif field_description.name == '_action' and app.request.isXmlHttpRequest %}
  37. {# Action buttons disabled in ajax view! #}
  38. {% elseif field_description.getOption('ajax_hidden') == true and app.request.isXmlHttpRequest %}
  39. {# Disable fields with 'ajax_hidden' option set to true #}
  40. {% else %}
  41. {% set sortable = false %}
  42. {% if field_description.options.sortable is defined and field_description.options.sortable %}
  43. {% set sortable = true %}
  44. {% set sort_parameters = admin.modelmanager.sortparameters(field_description, admin.datagrid) %}
  45. {% set current = admin.datagrid.values._sort_by == field_description or admin.datagrid.values._sort_by.fieldName == sort_parameters.filter._sort_by %}
  46. {% set sort_active_class = current ? 'sonata-ba-list-field-order-active' : '' %}
  47. {% set sort_by = current ? admin.datagrid.values._sort_order : field_description.options._sort_order %}
  48. {% endif %}
  49. {% spaceless %}
  50. <th class="sonata-ba-list-field-header-{{ field_description.type}} {% if sortable %} sonata-ba-list-field-header-order-{{ sort_by|lower }} {{ sort_active_class }}{% endif %}{% if field_description.options.header_class is defined %} {{ field_description.options.header_class }}{% endif %}"{% if field_description.options.header_style is defined %} style="{{ field_description.options.header_style }}"{% endif %}>
  51. {% if sortable %}<a href="{{ admin.generateUrl('list', sort_parameters) }}">{% endif %}
  52. {{ admin.trans(field_description.label, {}, field_description.translationDomain) }}
  53. {% if sortable %}</a>{% endif %}
  54. </th>
  55. {% endspaceless %}
  56. {% endif %}
  57. {% endfor %}
  58. </tr>
  59. </thead>
  60. {% endblock %}
  61. {% block table_body %}
  62. <tbody>
  63. {% include admin.getTemplate('outer_list_rows_' ~ admin.getListMode()) %}
  64. </tbody>
  65. {% endblock %}
  66. {% block table_footer %}
  67. {% endblock %}
  68. </table>
  69. {% else %}
  70. {% block no_result_content %}
  71. <div class="info-box">
  72. <span class="info-box-icon bg-aqua"><i class="fa fa-arrow-circle-right"></i></span>
  73. <div class="info-box-content">
  74. <span class="info-box-text">{{ 'no_result'|trans({}, 'SonataAdminBundle') }}</span>
  75. <div class="progress">
  76. <div class="progress-bar" style="width: 0%"></div>
  77. </div>
  78. <span class="progress-description">
  79. {% if not app.request.xmlHttpRequest %}
  80. {% include 'SonataAdminBundle:Button:create_button.html.twig' %}
  81. {% endif %}
  82. </span>
  83. </div><!-- /.info-box-content -->
  84. </div>
  85. {% endblock %}
  86. {% endif %}
  87. {{ sonata_block_render_event('sonata.admin.list.table.bottom', { 'admin': admin }) }}
  88. </div>
  89. {% block list_footer %}
  90. {% if admin.datagrid.results|length > 0 %}
  91. <div class="box-footer">
  92. <div class="form-inline clearfix">
  93. {% if not app.request.isXmlHttpRequest %}
  94. <div class="pull-left">
  95. {% if admin.hasRoute('batch') and batchactions|length > 0 %}
  96. {% block batch %}
  97. <script>
  98. {% block batch_javascript %}
  99. jQuery(document).ready(function ($) {
  100. // Toggle individual checkboxes when the batch checkbox is changed
  101. $('#list_batch_checkbox').on('ifChanged change', function () {
  102. var checkboxes = $(this)
  103. .closest('table')
  104. .find('td.sonata-ba-list-field-batch input[type="checkbox"], div.sonata-ba-list-field-batch input[type="checkbox"]')
  105. ;
  106. if (window.SONATA_CONFIG.USE_ICHECK) {
  107. checkboxes.iCheck($(this).is(':checked') ? 'check' : 'uncheck');
  108. } else {
  109. checkboxes.prop('checked', this.checked);
  110. }
  111. });
  112. // Add a CSS class to rows when they are selected
  113. $('td.sonata-ba-list-field-batch input[type="checkbox"], div.sonata-ba-list-field-batch input[type="checkbox"]')
  114. .on('ifChanged change', function () {
  115. $(this)
  116. .closest('tr, div.sonata-ba-list-field-batch')
  117. .toggleClass('sonata-ba-list-row-selected', $(this).is(':checked'))
  118. ;
  119. })
  120. .trigger('ifChanged')
  121. ;
  122. });
  123. {% endblock %}
  124. </script>
  125. {% block batch_actions %}
  126. <label class="checkbox" for="{{ admin.uniqid }}_all_elements">
  127. <input type="checkbox" name="all_elements" id="{{ admin.uniqid }}_all_elements">
  128. {{ 'all_elements'|trans({}, 'SonataAdminBundle') }}
  129. ({{ admin.datagrid.pager.nbresults }})
  130. </label>
  131. <select name="action" style="width: auto; height: auto" class="form-control">
  132. {% for action, options in batchactions %}
  133. <option value="{{ action }}">{{ options.label|trans({}, options.translation_domain|default(admin.translationDomain)) }}</option>
  134. {% endfor %}
  135. </select>
  136. {% endblock %}
  137. <input type="submit" class="btn btn-small btn-primary" value="{{ 'btn_batch'|trans({}, 'SonataAdminBundle') }}">
  138. {% endblock %}
  139. {% endif %}
  140. </div>
  141. <div class="pull-right">
  142. {% if admin.hasRoute('export') and admin.isGranted('EXPORT') and admin.getExportFormats()|length %}
  143. <div class="btn-group">
  144. <button type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown">
  145. <i class="fa fa-share-square-o"></i>
  146. {{ "label_export_download"|trans({}, "SonataAdminBundle") }}
  147. <span class="caret"></span>
  148. </button>
  149. <ul class="dropdown-menu">
  150. {% for format in admin.getExportFormats() %}
  151. <li>
  152. <a href="{{ admin.generateUrl('export', admin.modelmanager.paginationparameters(admin.datagrid, 0) + {'format' : format}) }}">
  153. <i class="fa fa-arrow-circle-o-down"></i>
  154. {{ ("export_format_" ~ format)|trans({}, 'SonataAdminBundle') }}
  155. </a>
  156. <li>
  157. {% endfor %}
  158. </ul>
  159. </div>
  160. &nbsp;-&nbsp;
  161. {% endif %}
  162. {% block pager_results %}
  163. {% include admin.getTemplate('pager_results') %}
  164. {% endblock %}
  165. </div>
  166. {% endif %}
  167. </div>
  168. {% block pager_links %}
  169. {% if admin.datagrid.pager.haveToPaginate() %}
  170. <hr/>
  171. {% include admin.getTemplate('pager_links') %}
  172. {% endif %}
  173. {% endblock %}
  174. </div>
  175. {% endif %}
  176. {% endblock %}
  177. </div>
  178. {% if admin.hasRoute('batch') and batchactions|length %}
  179. </form>
  180. {% endif %}
  181. </div>
  182. {% endblock %}
  183. {% block list_filters_actions %}
  184. {%- if admin.datagrid.hasDisplayableFilters %}
  185. <ul class="nav navbar-nav navbar-right">
  186. <li class="dropdown sonata-actions">
  187. <a href="#" class="dropdown-toggle sonata-ba-action" data-toggle="dropdown">
  188. <i class="fa fa-filter"></i>
  189. {{ 'link_filters'|trans({}, 'SonataAdminBundle') }} <b class="caret"></b>
  190. </a>
  191. <ul class="dropdown-menu" role="menu">
  192. {% for filter in admin.datagrid.filters if (filter.options['show_filter'] is same as(true) or filter.options['show_filter'] is null) %}
  193. <li>
  194. <a href="#" class="sonata-toggle-filter sonata-ba-action" filter-target="filter-{{ admin.uniqid }}-{{ filter.name }}" filter-container="filter-container-{{ admin.uniqid() }}">
  195. <i class="fa {{ (filter.isActive() or filter.options['show_filter']) ? 'fa-check-square-o' : 'fa-square-o' }}"></i>{{ admin.trans(filter.label, {}, filter.translationDomain) }}
  196. </a>
  197. </li>
  198. {% endfor %}
  199. </ul>
  200. </li>
  201. </ul>
  202. {% endif -%}
  203. {% endblock %}
  204. {% block list_filters %}
  205. {% if admin.datagrid.filters %}
  206. {% form_theme form admin.getTemplate('filter') %}
  207. <div class="col-xs-12 col-md-12 sonata-filters-box" style="display: {{ admin.datagrid.hasDisplayableFilters ? 'block' : 'none' }}" id="filter-container-{{ admin.uniqid() }}">
  208. <div class="box box-primary" >
  209. <div class="box-body">
  210. <form class="sonata-filter-form form-horizontal {{ admin.isChild and 1 == admin.datagrid.filters|length ? 'hide' : '' }}" action="{{ admin.generateUrl('list') }}" method="GET" role="form">
  211. {{ form_errors(form) }}
  212. <div class="row">
  213. <div class="col-sm-9">
  214. {% set withAdvancedFilter = false %}
  215. {% for filter in admin.datagrid.filters %}
  216. <div class="form-group {% block sonata_list_filter_group_class %}{% endblock %}" id="filter-{{ admin.uniqid }}-{{ filter.name }}" sonata-filter="{{ (filter.options['show_filter'] is same as(true) or filter.options['show_filter'] is null) ? 'true' : 'false' }}" style="display: {% if (filter.isActive() and filter.options['show_filter'] is null) or (filter.options['show_filter'] is same as(true)) %}block{% else %}none{% endif %}">
  217. {% if filter.label is not same as(false) %}
  218. <label for="{{ form.children[filter.formName].children['value'].vars.id }}" class="col-sm-3 control-label">{{ admin.trans(filter.label, {}, filter.translationDomain) }}</label>
  219. {% endif %}
  220. {% set attr = form.children[filter.formName].children['type'].vars.attr|default({}) %}
  221. <div class="col-sm-4 advanced-filter">
  222. {{ form_widget(form.children[filter.formName].children['type'], {'attr': attr}) }}
  223. </div>
  224. <div class="col-sm-4">
  225. {{ form_widget(form.children[filter.formName].children['value']) }}
  226. </div>
  227. <div class="col-sm-1">
  228. <label class="control-label">
  229. <a href="#" class="sonata-toggle-filter sonata-ba-action" filter-target="filter-{{ admin.uniqid }}-{{ filter.name }}" filter-container="filter-container-{{ admin.uniqid() }}">
  230. <i class="fa fa-minus-circle"></i>
  231. </a>
  232. </label>
  233. </div>
  234. </div>
  235. {% if filter.options['advanced_filter'] %}
  236. {% set withAdvancedFilter = true %}
  237. {% endif %}
  238. {% endfor %}
  239. </div>
  240. <div class="col-sm-3 text-center">
  241. <input type="hidden" name="filter[_page]" id="filter__page" value="1">
  242. {% set foo = form.children['_page'].setRendered() %}
  243. {{ form_rest(form) }}
  244. <div class="form-group">
  245. <button type="submit" class="btn btn-primary">
  246. <i class="fa fa-filter"></i> {{ 'btn_filter'|trans({}, 'SonataAdminBundle') }}
  247. </button>
  248. <a class="btn btn-default" href="{{ admin.generateUrl('list', {filters: 'reset'}) }}">
  249. {{ 'link_reset_filter'|trans({}, 'SonataAdminBundle') }}
  250. </a>
  251. </div>
  252. {% if withAdvancedFilter %}
  253. <div class="form-group">
  254. <a href="#" data-toggle="advanced-filter">
  255. <i class="fa fa-cogs"></i>
  256. {{ 'btn_advanced_filters'|trans({}, 'SonataAdminBundle') }}
  257. </a>
  258. </div>
  259. {% endif %}
  260. </div>
  261. </div>
  262. {% for paramKey, paramValue in admin.persistentParameters %}
  263. <input type="hidden" name="{{ paramKey }}" value="{{ paramValue }}">
  264. {% endfor %}
  265. </form>
  266. </div>
  267. </div>
  268. </div>
  269. {% endif %}
  270. {% endblock %}