Swatch

<div class="swatch ">

    <div class="swatch__wrapper" aria-label="Size" tabindex="0" aria-activedescendant="opt-size-s" aria-required="true" role="listbox" aria-invalid="false">
        <div class="swatch__option-container " id="opt-size-xs" tabindex="0" role="option" aria-selected="false">
            <div class="swatch__option ">
                XS
            </div>
        </div>
        <div class="swatch__option-container selected" id="opt-size-s" tabindex="0" role="option" aria-selected="true">
            <div class="swatch__option ">
                S
            </div>
        </div>
        <div class="swatch__option-container " id="opt-size-m" tabindex="0" role="option" aria-selected="false">
            <div class="swatch__option ">
                M
            </div>
        </div>
        <div class="swatch__option-container " id="opt-size-l" tabindex="0" role="option" aria-selected="false">
            <div class="swatch__option ">
                L
            </div>
        </div>
        <div class="swatch__option-container " id="opt-size-xl" tabindex="0" role="option" aria-selected="false">
            <div class="swatch__option ">
                XL
            </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": false,
  "wrapper": {
    "attributes": "aria-label=\"Size\" tabindex=\"0\" aria-activedescendant=\"opt-size-s\" aria-required=\"true\" role=\"listbox\" aria-invalid=\"false\""
  },
  "options": [
    {
      "id": "opt-size-xs",
      "attributes": "tabindex=\"0\" role=\"option\" aria-selected=\"false\"",
      "text": "XS"
    },
    {
      "id": "opt-size-s",
      "attributes": "tabindex=\"0\" role=\"option\" aria-selected=\"true\"",
      "class": "selected",
      "text": "S"
    },
    {
      "id": "opt-size-m",
      "attributes": "tabindex=\"0\" role=\"option\" aria-selected=\"false\"",
      "text": "M"
    },
    {
      "id": "opt-size-l",
      "attributes": "tabindex=\"0\" role=\"option\" aria-selected=\"false\"",
      "text": "L"
    },
    {
      "id": "opt-size-xl",
      "attributes": "tabindex=\"0\" role=\"option\" aria-selected=\"false\"",
      "text": "XL"
    }
  ]
}
  • Content:
    $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;
        }
    }
    
  • URL: /components/raw/swatch/_swatch.scss
  • Filesystem Path: build/components/02-elements/swatch/_swatch.scss
  • Size: 2.1 KB
  • Content:
    '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');
        })
      })
    })();
    
  • URL: /components/raw/swatch/swatch.js
  • Filesystem Path: build/components/02-elements/swatch/swatch.js
  • Size: 617 Bytes

Swatches accessibility

Swatches require following aria structure to be accessible for assistive technology and keyboard:

  1. Swatches wrapper .swatch__wrapper should have following aria attributes:

    • role="listbox" to easy list swatch’s options
    • aria-activedescendant with value of selected swatch option
    • tabindex="0" to make element focusable
    • aria-required with value true or false depending if swatch option is required
    • aria-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)
  2. Swatch option .swatch__option should have following aria attributes:

    • role="option" to show options of swatch
    • tabindex="0" to make element focusable
    • aria-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 exist
    • aria-selected with value true if the option was selected