base_list.html.twig 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  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. {% include 'SonataAdminBundle:Button:create_button.html.twig' %}
  93. {% endif %}
  94. </span>
  95. </div><!-- /.info-box-content -->
  96. </div>
  97. {% endblock %}
  98. {% endif %}
  99. {{ sonata_block_render_event('sonata.admin.list.table.bottom', { 'admin': admin }) }}
  100. </div>
  101. {% block list_footer %}
  102. {% if admin.datagrid.results|length > 0 %}
  103. <div class="box-footer">
  104. <div class="form-inline clearfix">
  105. {% if not app.request.isXmlHttpRequest %}
  106. <div class="pull-left">
  107. {% if admin.hasRoute('batch') and batchactions|length > 0 %}
  108. {% block batch %}
  109. <script>
  110. {% block batch_javascript %}
  111. jQuery(document).ready(function ($) {
  112. // Toggle individual checkboxes when the batch checkbox is changed
  113. $('#list_batch_checkbox').on('ifChanged change', function () {
  114. var checkboxes = $(this)
  115. .closest('table')
  116. .find('td.sonata-ba-list-field-batch input[type="checkbox"], div.sonata-ba-list-field-batch input[type="checkbox"]')
  117. ;
  118. if (window.SONATA_CONFIG.USE_ICHECK) {
  119. checkboxes.iCheck($(this).is(':checked') ? 'check' : 'uncheck');
  120. } else {
  121. checkboxes.prop('checked', this.checked);
  122. }
  123. });
  124. // Add a CSS class to rows when they are selected
  125. $('td.sonata-ba-list-field-batch input[type="checkbox"], div.sonata-ba-list-field-batch input[type="checkbox"]')
  126. .on('ifChanged change', function () {
  127. $(this)
  128. .closest('tr, div.sonata-ba-list-field-batch')
  129. .toggleClass('sonata-ba-list-row-selected', $(this).is(':checked'))
  130. ;
  131. })
  132. .trigger('ifChanged')
  133. ;
  134. });
  135. {% endblock %}
  136. </script>
  137. {% block batch_actions %}
  138. <label class="checkbox" for="{{ admin.uniqid }}_all_elements">
  139. <input type="checkbox" name="all_elements" id="{{ admin.uniqid }}_all_elements">
  140. {{ 'all_elements'|trans({}, 'SonataAdminBundle') }}
  141. ({{ admin.datagrid.pager.nbresults }})
  142. </label>
  143. <select name="action" style="width: auto; height: auto" class="form-control">
  144. {% for action, options in batchactions %}
  145. <option value="{{ action }}">{{ options.label|trans({}, options.translation_domain|default(admin.translationDomain)) }}</option>
  146. {% endfor %}
  147. </select>
  148. {% endblock %}
  149. <input type="submit" class="btn btn-small btn-primary" value="{{ 'btn_batch'|trans({}, 'SonataAdminBundle') }}">
  150. {% endblock %}
  151. {% endif %}
  152. </div>
  153. <div class="pull-right">
  154. {% if admin.hasRoute('export') and admin.isGranted('EXPORT') and admin.getExportFormats()|length %}
  155. <div class="btn-group">
  156. <button type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown">
  157. <i class="fa fa-share-square-o" aria-hidden="true"></i>
  158. {{ "label_export_download"|trans({}, "SonataAdminBundle") }}
  159. <span class="caret"></span>
  160. </button>
  161. <ul class="dropdown-menu">
  162. {% for format in admin.getExportFormats() %}
  163. <li>
  164. <a href="{{ admin.generateUrl('export', admin.modelmanager.paginationparameters(admin.datagrid, 0) + {'format' : format}) }}">
  165. <i class="fa fa-arrow-circle-o-down" aria-hidden="true"></i>
  166. {{ ("export_format_" ~ format)|trans({}, 'SonataAdminBundle') }}
  167. </a>
  168. <li>
  169. {% endfor %}
  170. </ul>
  171. </div>
  172. &nbsp;-&nbsp;
  173. {% endif %}
  174. {% block pager_results %}
  175. {% include admin.getTemplate('pager_results') %}
  176. {% endblock %}
  177. </div>
  178. {% endif %}
  179. </div>
  180. {% block pager_links %}
  181. {% if admin.datagrid.pager.haveToPaginate() %}
  182. <hr/>
  183. {% include admin.getTemplate('pager_links') %}
  184. {% endif %}
  185. {% endblock %}
  186. </div>
  187. {% endif %}
  188. {% endblock %}
  189. </div>
  190. {% if admin.hasRoute('batch') and batchactions|length %}
  191. </form>
  192. {% endif %}
  193. </div>
  194. {% endblock %}
  195. {% block list_filters_actions %}
  196. {%- if admin.datagrid.filters|length %}
  197. <ul class="nav navbar-nav navbar-right">
  198. <li class="dropdown sonata-actions">
  199. <a href="#" class="dropdown-toggle sonata-ba-action" data-toggle="dropdown">
  200. <i class="fa fa-filter" aria-hidden="true"></i>
  201. {{ 'link_filters'|trans({}, 'SonataAdminBundle') }} <b class="caret"></b>
  202. </a>
  203. <ul class="dropdown-menu" role="menu">
  204. {% for filter in admin.datagrid.filters if (filter.options['show_filter'] is same as(true) or filter.options['show_filter'] is null) %}
  205. {% set filterActive = ((filter.isActive() or filter.options['show_filter']) and not admin.isDefaultFilter(filter.formName)) %}
  206. <li>
  207. <a href="#" class="sonata-toggle-filter sonata-ba-action" filter-target="filter-{{ admin.uniqid }}-{{ filter.name }}" filter-container="filter-container-{{ admin.uniqid() }}">
  208. <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) }}
  209. </a>
  210. </li>
  211. {% endfor %}
  212. </ul>
  213. </li>
  214. </ul>
  215. {% endif -%}
  216. {% endblock %}
  217. {% block list_filters %}
  218. {% if admin.datagrid.filters %}
  219. {% form_theme form admin.getTemplate('filter') %}
  220. <div class="col-xs-12 col-md-12 sonata-filters-box" style="display: {{ admin.datagrid.hasDisplayableFilters ? 'block' : 'none' }}" id="filter-container-{{ admin.uniqid() }}">
  221. <div class="box box-primary" >
  222. <div class="box-body">
  223. <form class="sonata-filter-form form-horizontal {{ admin.isChild and 1 == admin.datagrid.filters|length ? 'hide' : '' }}" action="{{ admin.generateUrl('list') }}" method="GET" role="form">
  224. {{ form_errors(form) }}
  225. <div class="row">
  226. <div class="col-sm-9">
  227. {% set withAdvancedFilter = false %}
  228. {% for filter in admin.datagrid.filters %}
  229. {% 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) %}
  230. {% set filterVisible = filter.options['show_filter'] is same as(true) or filter.options['show_filter'] is null %}
  231. <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 %}">
  232. {% if filter.label is not same as(false) %}
  233. <label for="{{ form.children[filter.formName].children['value'].vars.id }}" class="col-sm-3 control-label">{{ filter.label|trans({}, filter.translationDomain ?: admin.translationDomain) }}</label>
  234. {% endif %}
  235. {% set attr = form.children[filter.formName].children['type'].vars.attr|default({}) %}
  236. <div class="col-sm-4 advanced-filter">
  237. {{ form_widget(form.children[filter.formName].children['type'], {'attr': attr}) }}
  238. </div>
  239. <div class="col-sm-4">
  240. {{ form_widget(form.children[filter.formName].children['value']) }}
  241. </div>
  242. <div class="col-sm-1">
  243. <label class="control-label">
  244. <a href="#" class="sonata-toggle-filter sonata-ba-action" filter-target="filter-{{ admin.uniqid }}-{{ filter.name }}" filter-container="filter-container-{{ admin.uniqid() }}">
  245. <i class="fa fa-minus-circle" aria-hidden="true"></i>
  246. </a>
  247. </label>
  248. </div>
  249. </div>
  250. {% if filter.options['advanced_filter'] %}
  251. {% set withAdvancedFilter = true %}
  252. {% endif %}
  253. {% endfor %}
  254. </div>
  255. <div class="col-sm-3 text-center">
  256. <input type="hidden" name="filter[_page]" id="filter__page" value="1">
  257. {% set foo = form.children['_page'].setRendered() %}
  258. {{ form_rest(form) }}
  259. <div class="form-group">
  260. <button type="submit" class="btn btn-primary">
  261. <i class="fa fa-filter" aria-hidden="true"></i> {{ 'btn_filter'|trans({}, 'SonataAdminBundle') }}
  262. </button>
  263. <a class="btn btn-default" href="{{ admin.generateUrl('list', {filters: 'reset'}) }}">
  264. {{ 'link_reset_filter'|trans({}, 'SonataAdminBundle') }}
  265. </a>
  266. </div>
  267. {% if withAdvancedFilter %}
  268. <div class="form-group">
  269. <a href="#" data-toggle="advanced-filter">
  270. <i class="fa fa-cogs" aria-hidden="true"></i>
  271. {{ 'btn_advanced_filters'|trans({}, 'SonataAdminBundle') }}
  272. </a>
  273. </div>
  274. {% endif %}
  275. </div>
  276. </div>
  277. {% for paramKey, paramValue in admin.persistentParameters %}
  278. <input type="hidden" name="{{ paramKey }}" value="{{ paramValue }}">
  279. {% endfor %}
  280. </form>
  281. </div>
  282. </div>
  283. </div>
  284. {% endif %}
  285. {% endblock %}