Admin.js 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641
  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. var Admin = {
  8. collectionCounters: [],
  9. /**
  10. * This function must be called when an ajax call is done, to ensure
  11. * the retrieved html is properly setup
  12. *
  13. * @param subject
  14. */
  15. shared_setup: function(subject) {
  16. Admin.log("[core|shared_setup] Register services on", subject);
  17. Admin.set_object_field_value(subject);
  18. Admin.add_filters(subject);
  19. Admin.setup_select2(subject);
  20. Admin.setup_icheck(subject);
  21. Admin.setup_xeditable(subject);
  22. Admin.setup_form_tabs_for_errors(subject);
  23. Admin.setup_inline_form_errors(subject);
  24. Admin.setup_tree_view(subject);
  25. Admin.setup_collection_counter(subject);
  26. Admin.setup_sticky_elements(subject);
  27. Admin.setup_readmore_elements(subject);
  28. // Admin.setup_list_modal(subject);
  29. },
  30. setup_list_modal: function(modal) {
  31. Admin.log('[core|setup_list_modal] configure modal on', modal);
  32. // this will force relation modal to open list of entity in a wider modal
  33. // to improve readability
  34. jQuery('div.modal-dialog', modal).css({
  35. width: '90%', //choose your width
  36. height: '85%',
  37. padding: 0
  38. });
  39. jQuery('div.modal-content', modal).css({
  40. 'border-radius':'0',
  41. height: '100%',
  42. padding: 0
  43. });
  44. jQuery('.modal-body', modal).css({
  45. width: 'auto',
  46. height: '90%',
  47. padding: 15,
  48. overflow: 'auto'
  49. });
  50. jQuery(modal).trigger('sonata-admin-setup-list-modal');
  51. },
  52. setup_select2: function(subject) {
  53. if (window.SONATA_CONFIG && window.SONATA_CONFIG.USE_SELECT2) {
  54. Admin.log('[core|setup_select2] configure Select2 on', subject);
  55. jQuery('select:not([data-sonata-select2="false"])', subject).each(function() {
  56. var select = jQuery(this);
  57. var allowClearEnabled = false;
  58. var popover = select.data('popover');
  59. select.removeClass('form-control');
  60. if (select.find('option[value=""]').length || select.attr('data-sonata-select2-allow-clear')==='true') {
  61. allowClearEnabled = true;
  62. } else if (select.attr('data-sonata-select2-allow-clear')==='false') {
  63. allowClearEnabled = false;
  64. }
  65. select.select2({
  66. width: function(){
  67. // Select2 v3 and v4 BC. If window.Select2 is defined, then the v3 is installed.
  68. // NEXT_MAJOR: Remove Select2 v3 support.
  69. return Admin.get_select2_width(window.Select2 ? this.element : jQuery(this));
  70. },
  71. dropdownAutoWidth: true,
  72. minimumResultsForSearch: 10,
  73. allowClear: allowClearEnabled
  74. });
  75. if (undefined !== popover) {
  76. select
  77. .select2('container')
  78. .popover(popover.options)
  79. ;
  80. }
  81. });
  82. }
  83. },
  84. setup_icheck: function(subject) {
  85. if (window.SONATA_CONFIG && window.SONATA_CONFIG.USE_ICHECK) {
  86. Admin.log('[core|setup_icheck] configure iCheck on', subject);
  87. jQuery("input[type='checkbox']:not('label.btn>input'), input[type='radio']:not('label.btn>input')", subject).iCheck({
  88. checkboxClass: 'icheckbox_square-blue',
  89. radioClass: 'iradio_square-blue'
  90. });
  91. }
  92. },
  93. setup_xeditable: function(subject) {
  94. Admin.log('[core|setup_xeditable] configure xeditable on', subject);
  95. jQuery('.x-editable', subject).editable({
  96. emptyclass: 'editable-empty btn btn-sm btn-default',
  97. emptytext: '<i class="fa fa-pencil"></i>',
  98. container: 'body',
  99. placement: 'auto',
  100. success: function(response) {
  101. if('KO' === response.status) {
  102. return response.message;
  103. }
  104. var html = jQuery(response.content);
  105. Admin.setup_xeditable(html);
  106. jQuery(this)
  107. .closest('td')
  108. .replaceWith(html)
  109. ;
  110. }
  111. });
  112. },
  113. /**
  114. * render log message
  115. * @param mixed
  116. */
  117. log: function() {
  118. var msg = '[Sonata.Admin] ' + Array.prototype.join.call(arguments,', ');
  119. if (window.console && window.console.log) {
  120. window.console.log(msg);
  121. } else if (window.opera && window.opera.postError) {
  122. window.opera.postError(msg);
  123. }
  124. },
  125. /**
  126. * NEXT_MAJOR: remove this function.
  127. *
  128. * @deprecated in version 3.0
  129. */
  130. add_pretty_errors: function() {
  131. console.warn('Admin.add_pretty_errors() was deprecated in version 3.0');
  132. },
  133. stopEvent: function(event) {
  134. // https://github.com/sonata-project/SonataAdminBundle/issues/151
  135. //if it is a standard browser use preventDefault otherwise it is IE then return false
  136. if(event.preventDefault) {
  137. event.preventDefault();
  138. } else {
  139. event.returnValue = false;
  140. }
  141. //if it is a standard browser get target otherwise it is IE then adapt syntax and get target
  142. if (typeof event.target != 'undefined') {
  143. targetElement = event.target;
  144. } else {
  145. targetElement = event.srcElement;
  146. }
  147. return targetElement;
  148. },
  149. add_filters: function(subject) {
  150. Admin.log('[core|add_filters] configure filters on', subject);
  151. jQuery('a.sonata-toggle-filter', subject).on('click', function(e) {
  152. e.preventDefault();
  153. e.stopPropagation();
  154. if (jQuery(e.target).attr('sonata-filter') == 'false') {
  155. return;
  156. }
  157. Admin.log('[core|add_filters] handle filter container: ', jQuery(e.target).attr('filter-container'))
  158. var filters_container = jQuery('#' + jQuery(e.currentTarget).attr('filter-container'));
  159. if (jQuery('div[sonata-filter="true"]:visible', filters_container).length == 0) {
  160. jQuery(filters_container).slideDown();
  161. }
  162. var targetSelector = jQuery(e.currentTarget).attr('filter-target'),
  163. target = jQuery('div[id="' + targetSelector + '"]', filters_container),
  164. filterToggler = jQuery('i', '.sonata-toggle-filter[filter-target="' + targetSelector + '"]')
  165. ;
  166. if (jQuery(target).is(":visible")) {
  167. filterToggler
  168. .removeClass('fa-check-square-o')
  169. .addClass('fa-square-o')
  170. ;
  171. target.hide();
  172. } else {
  173. filterToggler
  174. .removeClass('fa-square-o')
  175. .addClass('fa-check-square-o')
  176. ;
  177. target.show();
  178. }
  179. if (jQuery('div[sonata-filter="true"]:visible', filters_container).length > 0) {
  180. jQuery(filters_container).slideDown();
  181. } else {
  182. jQuery(filters_container).slideUp();
  183. }
  184. });
  185. jQuery('.sonata-filter-form', subject).on('submit', function () {
  186. jQuery(this).find('[sonata-filter="true"]:hidden :input').val('');
  187. });
  188. /* Advanced filters */
  189. if (jQuery('.advanced-filter :input:visible', subject).filter(function () { return jQuery(this).val() }).length === 0) {
  190. jQuery('.advanced-filter').hide();
  191. };
  192. jQuery('[data-toggle="advanced-filter"]', subject).click(function() {
  193. jQuery('.advanced-filter').toggle();
  194. });
  195. },
  196. /**
  197. * Change object field value
  198. * @param subject
  199. */
  200. set_object_field_value: function(subject) {
  201. Admin.log('[core|set_object_field_value] set value field on', subject);
  202. this.log(jQuery('a.sonata-ba-edit-inline', subject));
  203. jQuery('a.sonata-ba-edit-inline', subject).click(function(event) {
  204. Admin.stopEvent(event);
  205. var subject = jQuery(this);
  206. jQuery.ajax({
  207. url: subject.attr('href'),
  208. type: 'POST',
  209. success: function(json) {
  210. if(json.status === "OK") {
  211. var elm = jQuery(subject).parent();
  212. elm.children().remove();
  213. // fix issue with html comment ...
  214. elm.html(jQuery(json.content.replace(/<!--[\s\S]*?-->/g, "")).html());
  215. elm.effect("highlight", {'color' : '#57A957'}, 2000);
  216. Admin.set_object_field_value(elm);
  217. } else {
  218. jQuery(subject).parent().effect("highlight", {'color' : '#C43C35'}, 2000);
  219. }
  220. }
  221. });
  222. });
  223. },
  224. setup_collection_counter: function(subject) {
  225. Admin.log('[core|setup_collection_counter] setup collection counter', subject);
  226. // Count and save element of each collection
  227. var highestCounterRegexp = new RegExp('_([0-9]+)[^0-9]*$');
  228. jQuery(subject).find('[data-prototype]').each(function() {
  229. var collection = jQuery(this);
  230. var counter = 0;
  231. collection.children().each(function() {
  232. var matches = highestCounterRegexp.exec(jQuery('[id^="sonata-ba-field-container"]', this).attr('id'));
  233. if (matches && matches[1] && matches[1] > counter) {
  234. counter = parseInt(matches[1], 10);
  235. }
  236. });
  237. Admin.collectionCounters[collection.attr('id')] = counter;
  238. });
  239. },
  240. setup_collection_buttons: function(subject) {
  241. jQuery(subject).on('click', '.sonata-collection-add', function(event) {
  242. Admin.stopEvent(event);
  243. var container = jQuery(this).closest('[data-prototype]');
  244. var counter = ++Admin.collectionCounters[container.attr('id')];
  245. var proto = container.attr('data-prototype');
  246. var protoName = container.attr('data-prototype-name') || '__name__';
  247. // Set field id
  248. var idRegexp = new RegExp(container.attr('id')+'_'+protoName,'g');
  249. proto = proto.replace(idRegexp, container.attr('id')+'_'+counter);
  250. // Set field name
  251. var parts = container.attr('id').split('_');
  252. var nameRegexp = new RegExp(parts[parts.length-1]+'\\]\\['+protoName,'g');
  253. proto = proto.replace(nameRegexp, parts[parts.length-1]+']['+counter);
  254. jQuery(proto)
  255. .insertBefore(jQuery(this).parent())
  256. .trigger('sonata-admin-append-form-element')
  257. ;
  258. jQuery(this).trigger('sonata-collection-item-added');
  259. });
  260. jQuery(subject).on('click', '.sonata-collection-delete', function(event) {
  261. Admin.stopEvent(event);
  262. jQuery(this).trigger('sonata-collection-item-deleted');
  263. jQuery(this).closest('.sonata-collection-row').remove();
  264. jQuery(document).trigger('sonata-collection-item-deleted-successful');
  265. });
  266. },
  267. setup_per_page_switcher: function(subject) {
  268. Admin.log('[core|setup_per_page_switcher] setup page switcher', subject);
  269. jQuery('select.per-page').change(function(event) {
  270. jQuery('input[type=submit]').hide();
  271. window.top.location.href=this.options[this.selectedIndex].value;
  272. });
  273. },
  274. setup_form_tabs_for_errors: function(subject) {
  275. Admin.log('[core|setup_form_tabs_for_errors] setup form tab\'s errors', subject);
  276. // Switch to first tab with server side validation errors on page load
  277. jQuery('form', subject).each(function() {
  278. Admin.show_form_first_tab_with_errors(jQuery(this), '.sonata-ba-field-error');
  279. });
  280. // Switch to first tab with HTML5 errors on form submit
  281. jQuery(subject)
  282. .on('click', 'form [type="submit"]', function() {
  283. Admin.show_form_first_tab_with_errors(jQuery(this).closest('form'), ':invalid');
  284. })
  285. .on('keypress', 'form [type="text"]', function(e) {
  286. if (13 === e.which) {
  287. Admin.show_form_first_tab_with_errors(jQuery(this), ':invalid');
  288. }
  289. })
  290. ;
  291. },
  292. show_form_first_tab_with_errors: function(form, errorSelector) {
  293. Admin.log('[core|show_form_first_tab_with_errors] show first tab with errors', form);
  294. var tabs = form.find('.nav-tabs a'), firstTabWithErrors;
  295. tabs.each(function() {
  296. var id = jQuery(this).attr('href'),
  297. tab = jQuery(this),
  298. icon = tab.find('.has-errors');
  299. if (jQuery(id).find(errorSelector).length > 0) {
  300. // Only show first tab with errors
  301. if (!firstTabWithErrors) {
  302. tab.tab('show');
  303. firstTabWithErrors = tab;
  304. }
  305. icon.removeClass('hide');
  306. } else {
  307. icon.addClass('hide');
  308. }
  309. });
  310. },
  311. setup_inline_form_errors: function(subject) {
  312. Admin.log('[core|setup_inline_form_errors] show first tab with errors', subject);
  313. var deleteCheckboxSelector = '.sonata-ba-field-inline-table [id$="_delete"][type="checkbox"]';
  314. jQuery(deleteCheckboxSelector, subject).each(function() {
  315. Admin.switch_inline_form_errors(jQuery(this));
  316. });
  317. jQuery(subject).on('change', deleteCheckboxSelector, function() {
  318. Admin.switch_inline_form_errors(jQuery(this));
  319. });
  320. },
  321. /**
  322. * Disable inline form errors when the row is marked for deletion
  323. */
  324. switch_inline_form_errors: function(subject) {
  325. Admin.log('[core|switch_inline_form_errors] switch_inline_form_errors', subject);
  326. var row = subject.closest('.sonata-ba-field-inline-table'),
  327. errors = row.find('.sonata-ba-field-error-messages')
  328. ;
  329. if (subject.is(':checked')) {
  330. row
  331. .find('[required]')
  332. .removeAttr('required')
  333. .attr('data-required', 'required')
  334. ;
  335. errors.hide();
  336. } else {
  337. row
  338. .find('[data-required]')
  339. .attr('required', 'required')
  340. ;
  341. errors.show();
  342. }
  343. },
  344. setup_tree_view: function(subject) {
  345. Admin.log('[core|setup_tree_view] setup tree view', subject);
  346. jQuery('ul.js-treeview', subject).treeView();
  347. },
  348. /** Return the width for simple and sortable select2 element **/
  349. get_select2_width: function(element){
  350. var ereg = /width:(auto|(([-+]?([0-9]*\.)?[0-9]+)(px|em|ex|%|in|cm|mm|pt|pc)))/i;
  351. // this code is an adaptation of select2 code (initContainerWidth function)
  352. var style = element.attr('style');
  353. //console.log("main style", style);
  354. if (style !== undefined) {
  355. var attrs = style.split(';');
  356. for (i = 0, l = attrs.length; i < l; i = i + 1) {
  357. var matches = attrs[i].replace(/\s/g, '').match(ereg);
  358. if (matches !== null && matches.length >= 1)
  359. return matches[1];
  360. }
  361. }
  362. style = element.css('width');
  363. if (style.indexOf("%") > 0) {
  364. return style;
  365. }
  366. return '100%';
  367. },
  368. setup_sortable_select2: function(subject, data) {
  369. var transformedData = [];
  370. for (var i = 0 ; i < data.length ; i++) {
  371. transformedData[i] = {id: data[i].data, text: data[i].label};
  372. }
  373. subject.select2({
  374. width: function(){
  375. // Select2 v3 and v4 BC. If window.Select2 is defined, then the v3 is installed.
  376. // NEXT_MAJOR: Remove Select2 v3 support.
  377. return Admin.get_select2_width(window.Select2 ? this.element : jQuery(this));
  378. },
  379. dropdownAutoWidth: true,
  380. data: transformedData,
  381. multiple: true
  382. });
  383. subject.select2("container").find("ul.select2-choices").sortable({
  384. containment: 'parent',
  385. start: function () {
  386. subject.select2("onSortStart");
  387. },
  388. update: function () {
  389. subject.select2("onSortEnd");
  390. }
  391. });
  392. // On form submit, transform value to match what is expected by server
  393. subject.parents('form:first').submit(function (event) {
  394. var values = subject.val().trim();
  395. if (values !== '') {
  396. var baseName = subject.attr('name');
  397. values = values.split(',');
  398. baseName = baseName.substring(0, baseName.length-1);
  399. for (var i=0; i<values.length; i++) {
  400. jQuery('<input>')
  401. .attr('type', 'hidden')
  402. .attr('name', baseName+i+']')
  403. .val(values[i])
  404. .appendTo(subject.parents('form:first'));
  405. }
  406. }
  407. subject.remove();
  408. });
  409. },
  410. setup_sticky_elements: function(subject) {
  411. if (window.SONATA_CONFIG && window.SONATA_CONFIG.USE_STICKYFORMS) {
  412. Admin.log('[core|setup_sticky_elements] setup sticky elements on', subject);
  413. var wrapper = jQuery(subject).find('.content-wrapper');
  414. var navbar = jQuery(wrapper).find('nav.navbar');
  415. var footer = jQuery(wrapper).find('.sonata-ba-form-actions');
  416. if (navbar.length) {
  417. new Waypoint.Sticky({
  418. element: navbar[0],
  419. offset: 50,
  420. handler: function( direction ) {
  421. if (direction == 'up') {
  422. jQuery(navbar).width('auto');
  423. } else {
  424. jQuery(navbar).width(jQuery(wrapper).outerWidth());
  425. }
  426. }
  427. });
  428. }
  429. if (footer.length) {
  430. new Waypoint({
  431. element: wrapper[0],
  432. offset: 'bottom-in-view',
  433. handler: function(direction) {
  434. var position = jQuery('.sonata-ba-form form > .row').outerHeight() + jQuery(footer).outerHeight() - 2;
  435. if (position < jQuery(footer).offset().top) {
  436. jQuery(footer).removeClass('stuck');
  437. }
  438. if (direction == 'up') {
  439. jQuery(footer).addClass('stuck');
  440. }
  441. }
  442. });
  443. }
  444. Admin.handleScroll(footer, navbar, wrapper);
  445. }
  446. },
  447. handleScroll: function(footer, navbar, wrapper) {
  448. if (footer.length && jQuery(window).scrollTop() + jQuery(window).height() != jQuery(document).height()) {
  449. jQuery(footer).addClass('stuck');
  450. }
  451. jQuery(window).scroll(
  452. Admin.debounce(function() {
  453. if (footer.length && jQuery(window).scrollTop() + jQuery(window).height() == jQuery(document).height()) {
  454. jQuery(footer).removeClass('stuck');
  455. }
  456. if (navbar.length && jQuery(window).scrollTop() === 0) {
  457. jQuery(navbar).removeClass('stuck');
  458. }
  459. }, 250)
  460. );
  461. jQuery('body').on('expanded.pushMenu collapsed.pushMenu', function() {
  462. Admin.handleResize(footer, navbar, wrapper);
  463. });
  464. jQuery(window).resize(
  465. Admin.debounce(function() {
  466. Admin.handleResize(footer, navbar, wrapper);
  467. }, 250)
  468. );
  469. },
  470. handleResize: function(footer, navbar, wrapper) {
  471. setTimeout(function() {
  472. if (navbar.length && jQuery(navbar).hasClass('stuck')) {
  473. jQuery(navbar).width(jQuery(wrapper).outerWidth());
  474. }
  475. if (footer.length && jQuery(footer).hasClass('stuck')) {
  476. jQuery(footer).width(jQuery(wrapper).outerWidth());
  477. }
  478. }, 350); // the animation take 0.3s to execute, so we have to take the width, just after the animation ended
  479. },
  480. // http://davidwalsh.name/javascript-debounce-function
  481. debounce: function (func, wait, immediate) {
  482. var timeout;
  483. return function() {
  484. var context = this,
  485. args = arguments;
  486. var later = function() {
  487. timeout = null;
  488. if (!immediate) {
  489. func.apply(context, args);
  490. }
  491. };
  492. var callNow = immediate && !timeout;
  493. clearTimeout(timeout);
  494. timeout = setTimeout(later, wait);
  495. if (callNow) {
  496. func.apply(context, args);
  497. }
  498. };
  499. },
  500. setup_readmore_elements: function(subject) {
  501. Admin.log('[core|setup_readmore_elements] setup readmore elements on', subject);
  502. jQuery(subject).find('.sonata-readmore').each(function(i, ui){
  503. jQuery(this).readmore({
  504. collapsedHeight: parseInt(jQuery(this).data('readmore-height')),
  505. moreLink: '<a href="#">'+jQuery(this).data('readmore-more')+'</a>',
  506. lessLink: '<a href="#">'+jQuery(this).data('readmore-less')+'</a>'
  507. });
  508. });
  509. },
  510. handle_inline_delete_checkboxes: function() {
  511. var eventType = window.SONATA_CONFIG.USE_ICHECK ? 'ifChanged': 'change';
  512. $('.sonata-ba-form').on(eventType, '.sonata-admin-type-delete-checkbox', function() {
  513. var id = jQuery(this).prop('id');
  514. jQuery('[id^=' + id.split('__')[0] + ']:not("#' + id + '")')
  515. .prop('disabled', jQuery(this).is(':checked'))
  516. ;
  517. });
  518. }
  519. };
  520. jQuery(document).ready(function() {
  521. jQuery('html').removeClass('no-js');
  522. if (window.SONATA_CONFIG && window.SONATA_CONFIG.CONFIRM_EXIT) {
  523. jQuery('.sonata-ba-form form').each(function () { jQuery(this).confirmExit(); });
  524. }
  525. Admin.setup_per_page_switcher(document);
  526. Admin.setup_collection_buttons(document);
  527. Admin.shared_setup(document);
  528. Admin.handle_inline_delete_checkboxes();
  529. });
  530. jQuery(document).on('sonata-admin-append-form-element', function(e) {
  531. Admin.setup_select2(e.target);
  532. Admin.setup_icheck(e.target);
  533. Admin.setup_collection_counter(e.target);
  534. });