<div class="swatch ">
<span id="option-label-color-1" class="swatch__title ">
Label
</span>
<div class="swatch__wrapper" aria-label="Color" tabindex="0" aria-activedescendant="opt-color-red-3" aria-required="true" role="listbox" aria-invalid="false">
<div class="swatch__option-container " id="opt-color-orange-1" aria-label="Orange" tabindex="0" role="option" aria-selected="false" aria-describedby="option-label-color-1">
<div class="swatch__option swatch__option--image" style="background-image: url(../../images/swatch/swatch-orange.jpg)">
</div>
</div>
<div class="swatch__option-container " id="opt-color-green-2" aria-label="Green" tabindex="0" role="option" aria-selected="false" aria-describedby="option-label-color-1">
<div class="swatch__option swatch__option--image" style="background-image: url(../../images/swatch/swatch-green.jpg)">
</div>
</div>
<div class="swatch__option-container selected" id="opt-color-red-3" aria-label="Red" tabindex="0" role="option" aria-selected="true" aria-describedby="option-label-color-1">
<div class="swatch__option swatch__option--image" style="background-image: url(../../images/swatch/swatch-red.jpg)">
</div>
</div>
<div class="swatch__option-container " id="opt-color-white-4" aria-label="White" tabindex="0" role="option" aria-selected="false" aria-describedby="option-label-color-1">
<div class="swatch__option swatch__option--image" style="background-image: url(../../images/swatch/swatch-white.jpg)">
</div>
</div>
</div>
</div>
<script src="/components/raw/swatch/swatch.js"></script>
<div class="swatch {{ class }}">
{{#if heading}}
<{{{ heading.tag }}}
id="{{ heading.id }}"
class="swatch__title {{ heading.class }}"
>
{{ heading.text }}
</{{{ heading.tag }}}>
{{/if}}
<div
class="swatch__wrapper"
{{{ wrapper.attributes }}}
>
{{#each options }}
<div
class="swatch__option-container {{ this.class }}"
id="{{ this.id }}"
{{{ this.attributes }}}
{{#if this.headingId}}
aria-describedby="{{ this.headingId }}"
{{/if}}
>
<div
class="swatch__option {{ this.option.class}}"
{{{ this.option.attributes }}}
>
{{ this.text }}
</div>
</div>
{{/each }}
</div>
</div>
{{#if script}}
<script src="{{static 'swatch.js' }}"></script>
{{/if}}
{
"script": true,
"heading": {
"id": "option-label-color-1",
"tag": "span",
"text": "Label"
},
"wrapper": {
"attributes": "aria-label=\"Color\" tabindex=\"0\" aria-activedescendant=\"opt-color-red-3\" aria-required=\"true\" role=\"listbox\" aria-invalid=\"false\""
},
"options": [
{
"headingId": "option-label-color-1",
"id": "opt-color-orange-1",
"attributes": "aria-label=\"Orange\" tabindex=\"0\" role=\"option\" aria-selected=\"false\"",
"option": {
"class": "swatch__option--image",
"attributes": "style=\"background-image: url(../../images/swatch/swatch-orange.jpg)\""
}
},
{
"headingId": "option-label-color-1",
"id": "opt-color-green-2",
"attributes": "aria-label=\"Green\" tabindex=\"0\" role=\"option\" aria-selected=\"false\"",
"option": {
"class": "swatch__option--image",
"attributes": "style=\"background-image: url(../../images/swatch/swatch-green.jpg)\""
}
},
{
"headingId": "option-label-color-1",
"id": "opt-color-red-3",
"class": "selected",
"attributes": "aria-label=\"Red\" tabindex=\"0\" role=\"option\" aria-selected=\"true\"",
"option": {
"class": "swatch__option--image",
"attributes": "style=\"background-image: url(../../images/swatch/swatch-red.jpg)\""
}
},
{
"headingId": "option-label-color-1",
"id": "opt-color-white-4",
"attributes": "aria-label=\"White\" tabindex=\"0\" role=\"option\" aria-selected=\"false\"",
"option": {
"class": "swatch__option--image",
"attributes": "style=\"background-image: url(../../images/swatch/swatch-white.jpg)\""
}
}
]
}
$swatch__transition : $transition-base !default;
$swatch__title-margin-bottom : $spacer !default;
$swatch__title-font-size : $font-size-base !default;
$swatch__option-size : 40px !default;
$swatch__option-size--image : 48px !default;
$swatch__option-margin : $spacer !default;
$swatch__option-background : $white !default;
$swatch__option-color : $gray-dark !default;
$swatch__option-border : 4px solid $white !default;
$swatch__option-border--white : 1px solid $gray-lighter !default;
$swatch__option-border-color--active : $color-primary !default;
.swatch {
display: flex;
flex-wrap: wrap;
&__wrapper {
display: flex;
flex-wrap: wrap;
width: auto;
}
&__option-container {
box-sizing: border-box;
border: $swatch__option-border;
margin-right: $swatch__option-margin;
transition: $swatch__transition;
cursor: pointer;
&:hover,
&:focus,
&.selected {
outline: none;
border-color: $swatch__option-border-color--active;
.swatch__option--white {
border: 0;
}
}
&:last-child {
margin-right: 0;
}
}
&__selected-option {
display: none;
}
&__title {
flex: 0 0 100%;
margin-bottom: $swatch__title-margin-bottom;
font-size: $swatch__title-font-size;
}
&__option {
display: flex;
justify-content: center;
align-items: center;
min-width: $swatch__option-size;
min-height: $swatch__option-size;
background-color: $swatch__option-background;
color: $swatch__option-color;
&--image {
min-height: $swatch__option-size--image;
background-size: cover;
background-position: top center;
}
&--white {
border: $swatch__option-border--white;
}
}
&__input {
@include visually-hidden;
}
}
'use strict';
(function() { // eslint-disable-line
const swatchWrapper = document.querySelector('.swatch__wrapper'),
swatchOptions = swatchWrapper.querySelectorAll('.swatch__option-container');
swatchOptions.forEach(option => {
option.addEventListener('click', () => {
swatchOptions.forEach(option => {
option.classList.remove('selected');
option.setAttribute('aria-selected', 'false');
})
option.classList.add('selected');
swatchWrapper.setAttribute('aria-activedescendant', option.id);
option.setAttribute('aria-selected', 'true');
})
})
})();
Swatches require following aria structure to be accessible for assistive technology and keyboard:
Swatches wrapper .swatch__wrapper
should have following aria attributes:
role="listbox"
to easy list swatch’s optionsaria-activedescendant
with value of selected swatch optiontabindex="0"
to make element focusablearia-required
with value true
or false
depending if swatch option is requiredaria-invalid
with value true
if required field is not filled (if none of options is selected) or with the value false
if required field is field (option is selected)Swatch option .swatch__option
should have following aria attributes:
role="option"
to show options of swatchtabindex="0"
to make element focusablearia-label
with value of the name/label of swatch option if the swatch option is not a text but graphic element - icon or image (color option - ex: “Green”)aria-describedby
with value of the listbox label/heading if existaria-selected
with value true
if the option was selected