mirror of
https://github.com/GSA/notifications-admin.git
synced 2026-02-07 11:53:52 -05:00
223 lines
7.3 KiB
JavaScript
223 lines
7.3 KiB
JavaScript
(function(global) {
|
||
|
||
"use strict";
|
||
|
||
var Modules = global.GOVUK.Modules;
|
||
|
||
// Template functions for rendering component states
|
||
let renderStates = {
|
||
'initial': function(data) {
|
||
return `
|
||
${data.showNowAsDefault ? `
|
||
<div class="radio-select__column margin-y-2">
|
||
<div class="usa-radio">
|
||
<input class="usa-radio__input" checked="checked" id="${data.name}-0" name="${data.name}" type="radio" value="">
|
||
<label class="usa-radio__label" for="${data.name}-0">Now</label>
|
||
</div>
|
||
</div>
|
||
` : ''}
|
||
<div class="radio-select__column margin-y-2">
|
||
${data.categories.map(category =>
|
||
`<input type='button' class='usa-button usa-button--outline radio-select__button--category' aria-expanded="false" value='${category}' />`
|
||
).join('')}
|
||
</div>
|
||
`;
|
||
},
|
||
'choose': function(data) {
|
||
return `
|
||
${data.showNowAsDefault ? `
|
||
<div class="radio-select__column margin-y-2">
|
||
<div class="usa-radio">
|
||
<input class="usa-radio__input" checked="checked" id="${data.name}-0" name="${data.name}" type="radio" value="">
|
||
<label class="usa-radio__label" for="${data.name}-0">Now</label>
|
||
</div>
|
||
</div>
|
||
` : ''}
|
||
<div class="radio-select__column margin-y-2">
|
||
${data.choices.map(choice => `
|
||
<div class="usa-radio js-option">
|
||
<input class="usa-radio__input" type="radio" value="${choice.value}" id="${choice.id}" name="${data.name}" />
|
||
<label class="usa-radio__label" for="${choice.id}">${choice.label}</label>
|
||
</div>
|
||
`).join('')}
|
||
<input type='button' class='usa-button usa-button--outline radio-select__button--done margin-top-4' aria-expanded='true' value='Back to select a new time' />
|
||
</div>
|
||
`;
|
||
},
|
||
'chosen': function(data) {
|
||
return `
|
||
${data.showNowAsDefault ? `
|
||
<div class="radio-select__column margin-y-2">
|
||
<div class="usa-radio">
|
||
<input class="usa-radio__input" id="${data.name}-0" name="${data.name}" type="radio" value="">
|
||
<label class="usa-radio__label" for="${data.name}-0">Now</label>
|
||
</div>
|
||
</div>
|
||
` : ''}
|
||
<div class="radio-select__column margin-y-2">
|
||
${data.choices.map(choice => `
|
||
<div class="usa-radio">
|
||
<input class="usa-radio__input" checked="checked" type="radio" value="${choice.value}" id="${choice.id}" name="${data.name}" />
|
||
<label class="usa-radio__label" for="${choice.id}">${choice.label}</label>
|
||
</div>
|
||
`).join('')}
|
||
</div>
|
||
<div class="radio-select__column margin-y-2">
|
||
<input type='button' class='usa-button usa-button--outline radio-select__button--reset' aria-expanded='false' value='Choose a different time' />
|
||
</div>
|
||
`;
|
||
}
|
||
};
|
||
|
||
let shiftFocus = function(elementToFocus, component) {
|
||
// The first option is always the default
|
||
if (elementToFocus === 'default') {
|
||
$('[type=radio]', component).eq(0).focus();
|
||
}
|
||
if (elementToFocus === 'option') {
|
||
$('[type=radio]', component).eq(1).focus();
|
||
}
|
||
};
|
||
|
||
Modules.RadioSelect = function() {
|
||
|
||
this.start = function(component) {
|
||
|
||
let $component = $(component);
|
||
let render = (state, data) => {
|
||
$component.html(renderStates[state](data));
|
||
};
|
||
// store array of all options in component
|
||
let choices = $('label', $component).toArray().map(function(element) {
|
||
let $element = $(element);
|
||
return {
|
||
'id': $element.attr('for'),
|
||
'label': $.trim($element.text()),
|
||
'value': $element.prev('input').attr('value')
|
||
};
|
||
});
|
||
let categories = $component.data('categories').split(',');
|
||
let name = $component.find('input').eq(0).attr('name');
|
||
let mousedownOption = null;
|
||
let showNowAsDefault = (
|
||
$component.data('show-now-as-default').toString() === 'true' ?
|
||
{'name': name} : false
|
||
);
|
||
|
||
// functions for changing the state of the component's HTML
|
||
const reset = () => {
|
||
render('initial', {
|
||
'categories': categories,
|
||
'name': name,
|
||
'showNowAsDefault': showNowAsDefault
|
||
});
|
||
shiftFocus('default', component);
|
||
};
|
||
const selectOption = (value) => {
|
||
render('chosen', {
|
||
'choices': choices.filter(
|
||
element => element.value == value
|
||
),
|
||
'name': name,
|
||
'showNowAsDefault': showNowAsDefault
|
||
});
|
||
shiftFocus('option', component);
|
||
};
|
||
|
||
// use mousedown + mouseup event sequence to confirm option selection
|
||
const trackMouseup = (event) => {
|
||
const parentNode = event.target.parentNode;
|
||
|
||
if (parentNode === mousedownOption) {
|
||
const value = $('input', parentNode).attr('value');
|
||
|
||
selectOption(value);
|
||
|
||
// clear tracking
|
||
mousedownOption = null;
|
||
$(document).off('mouseup', trackMouseup);
|
||
}
|
||
};
|
||
|
||
// set events
|
||
$component
|
||
.on('click', '.radio-select__button--category', function(event) {
|
||
|
||
event.preventDefault();
|
||
let wordsInDay = $(this).attr('value').split(' ');
|
||
let day = wordsInDay[wordsInDay.length - 1].toLowerCase();
|
||
render('choose', {
|
||
'choices': choices.filter(
|
||
element => element.label.toLowerCase().indexOf(day) > -1
|
||
),
|
||
'name': name,
|
||
'showNowAsDefault': showNowAsDefault
|
||
});
|
||
shiftFocus('option', component);
|
||
|
||
})
|
||
.on('mousedown', '.js-option', function(event) {
|
||
mousedownOption = this;
|
||
|
||
// mouseup on the same option completes the click action
|
||
$(document).on('mouseup', trackMouseup);
|
||
})
|
||
// space and enter, clicked on a radio confirm that option was selected
|
||
.on('keydown', 'input[type=radio]', function(event) {
|
||
|
||
// allow keypresses which aren’t enter or space through
|
||
if (event.which !== 13 && event.which !== 32) {
|
||
return true;
|
||
}
|
||
|
||
event.preventDefault();
|
||
let value = $(this).attr('value');
|
||
selectOption(value);
|
||
|
||
})
|
||
.on('click', '.radio-select__button--done', function(event) {
|
||
|
||
event.preventDefault();
|
||
let $selection = $('input[type=radio]:checked', this.parentNode);
|
||
if ($selection.length) {
|
||
|
||
render('chosen', {
|
||
'choices': choices.filter(
|
||
element => element.value == $selection.eq(0).attr('value')
|
||
),
|
||
'name': name,
|
||
'showNowAsDefault': showNowAsDefault
|
||
});
|
||
shiftFocus('option', component);
|
||
|
||
} else {
|
||
|
||
reset();
|
||
shiftFocus('default', component);
|
||
|
||
}
|
||
|
||
})
|
||
.on('click', '.radio-select__button--reset', function(event) {
|
||
|
||
event.preventDefault();
|
||
reset();
|
||
shiftFocus('default', component);
|
||
|
||
});
|
||
|
||
// set HTML to initial state
|
||
render('initial', {
|
||
'categories': categories,
|
||
'name': name,
|
||
'showNowAsDefault': showNowAsDefault
|
||
});
|
||
|
||
$component.css({'height': 'auto'});
|
||
|
||
};
|
||
|
||
};
|
||
|
||
})(window);
|