(function(Modules) { "use strict"; Modules.TemplateFolderForm = function() { this.start = function(templateFolderForm) { this.$form = $(templateFolderForm); // remove the hidden unknown button - if you've got JS enabled then the action you want to do is implied by // which field is visible. this.$form.find('button[value=unknown]').remove(); this.$stickyBottom = this.$form.find('#sticky_template_forms'); this.$liveRegionCounter = this.$form.find('.selection-counter'); this.$stickyBottom.append(this.nothingSelectedButtons); this.$stickyBottom.append(this.itemsSelectedButtons); // all the diff states that we want to show or hide this.states = [ {key: 'nothing-selected-buttons', $el: this.$form.find('#nothing_selected'), cancellable: false}, {key: 'items-selected-buttons', $el: this.$form.find('#items_selected'), cancellable: false}, {key: 'move-to-existing-folder', $el: this.$form.find('#move_to_folder_radios'), cancellable: true, setFocus: this.getFocusRoutine('#move_to_folder_radios legend', true)}, {key: 'move-to-new-folder', $el: this.$form.find('#move_to_new_folder_form'), cancellable: true, setFocus: this.getFocusRoutine('#move_to_new_folder_name', false)}, {key: 'add-new-folder', $el: this.$form.find('#add_new_folder_form'), cancellable: true, setFocus: this.getFocusRoutine('#add_new_folder_name', false)}, {key: 'add-new-template', $el: this.$form.find('#add_new_template_form'), cancellable: true, setFocus: this.getFocusRoutine('#add_new_template_form legend', true)} ]; // cancel/clear buttons only relevant if JS enabled, so this.states.filter(state => state.cancellable).forEach((x) => this.addCancelButton(x)); this.states.filter(state => state.key === 'items-selected-buttons').forEach(x => this.addClearButton(x)); // activate stickiness of elements in each state this.activateStickyElements(); // first off show the new template / new folder buttons this.currentState = this.$form.data('prev-state') || 'unknown'; if (this.currentState === 'unknown') { this.selectActionButtons(); } else { this.render(); } this.$form.on('click', 'button.button-secondary', (event) => this.actionButtonClicked(event)); this.$form.on('change', 'input[type=checkbox]', () => this.templateFolderCheckboxChanged()); }; this.getFocusRoutine = function (selector, setTabindex) { return function () { let $el = $(selector); let removeTabindex = (e) => { $(e.target) .removeAttr('tabindex') .off('blur', removeTabindex); }; if (setTabindex) { $el.attr('tabindex', '-1'); $el.on('blur', removeTabindex); } $el.focus(); }; }; this.activateStickyElements = function() { var oldClass = 'js-will-stick-at-bottom-when-scrolling'; var newClass = 'js-stick-at-bottom-when-scrolling'; this.states.forEach(state => { state.$el .find('.' + oldClass) .removeClass(oldClass) .addClass(newClass); }); }; this.addCancelButton = function(state) { let selector = `[value=${state.key}]`; let $cancel = this.makeButton('Cancel', () => { // clear existing data state.$el.find('input:radio').prop('checked', false); state.$el.find('input:text').val(''); // go back to action buttons this.selectActionButtons(selector); }, selector); state.$el.find('[type=submit]').after($cancel); }; this.addClearButton = function(state) { let selector = 'button[value=add-new-template]'; let $clear = this.makeButton('Clear', () => { // uncheck all templates and folders this.$form.find('input:checkbox').prop('checked', false); // go back to action buttons this.selectActionButtons(selector); }); state.$el.find('.template-list-selected-counter').append($clear); }; this.makeButton = (text, fn, cancelSelector) => $('') .html(text) .addClass('js-cancel') .data('target', cancelSelector) // isn't set if cancelSelector is undefined .attr('tabindex', '0') .on('click keydown', event => { // space, enter or no keyCode (must be mouse input) if ([13, 32, undefined].indexOf(event.keyCode) > -1) { event.preventDefault(); fn(); } }); this.selectActionButtons = function (targetSelector) { // If we want to show one of the grey choose actions state, we can pretend we're in the choose actions state, // and then pretend a checkbox was clicked to work out whether to show zero or non-zero options. // This calls a render at the end this.currentState = 'nothing-selected-buttons'; this.templateFolderCheckboxChanged(); if (targetSelector) { let setFocus = this.getFocusRoutine(targetSelector, false); setFocus(); } }; this.actionButtonClicked = function(event) { event.preventDefault(); this.currentState = $(event.currentTarget).val(); this.render(); }; this.selectionStatus = { 'default': 'Nothing selected', 'selected': numSelected => `${numSelected} selected`, 'update': numSelected => { let liveRegionHTML = (numSelected > 0) ? this.selectionStatus.selected(numSelected) : this.selectionStatus.default; $('.template-list-selected-counter-count').html(numSelected); this.$liveRegionCounter.html(liveRegionHTML); } }; this.templateFolderCheckboxChanged = function() { let numSelected = this.countSelectedCheckboxes(); if (this.currentState === 'nothing-selected-buttons' && numSelected !== 0) { // user has just selected first item this.currentState = 'items-selected-buttons'; } else if (this.currentState === 'items-selected-buttons' && numSelected === 0) { // user has just deselected last item this.currentState = 'nothing-selected-buttons'; } this.render(); this.selectionStatus.update(numSelected); $('.template-list-selected-counter').toggle(this.hasCheckboxes()); }; this.hasCheckboxes = function() { return !!this.$form.find('input:checkbox').length; }; this.countSelectedCheckboxes = function() { return this.$form.find('input:checkbox:checked').length; }; this.render = function() { let mode = 'default'; let currentStateObj = this.states.filter(state => { return (state.key === this.currentState); })[0]; // detach everything, unless they are the currentState this.states.forEach( state => (state.key === this.currentState ? this.$stickyBottom.append(state.$el) : state.$el.detach()) ); // use dialog mode for states which contain more than one form control if (['move-to-existing-folder', 'add-new-template'].indexOf(this.currentState) !== -1) { mode = 'dialog'; } GOVUK.stickAtBottomWhenScrolling.setMode(mode); // make sticky JS recalculate its cache of the element's position GOVUK.stickAtBottomWhenScrolling.recalculate(); if (currentStateObj && ('setFocus' in currentStateObj)) { currentStateObj.setFocus(); } }; this.nothingSelectedButtons = $(`
${this.selectionStatus.default}
`).get(0); this.itemsSelectedButtons = $(`
${this.selectionStatus.selected(1)}
`).get(0); }; })(window.GOVUK.Modules);