base_list.html.twig 21 KB

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