<section class="carousel carousel--image-text 
    carousel__container" id="carousel-647079861" aria-labelledby="carouselheading" data-carousel>
    <h3 class="u-sr-only" id="carouselheading">Carousel heading.</h3>

    <ul class="u-list-unstyled u-list-inline carousel__slides">
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/600/451" class="slide__image" alt="Random image">
                </div>
                <figcaption class="u-align-self-start slide__caption">
                    This is another optional image caption
                </figcaption>
            </figure>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/600/452" class="slide__image" alt="Random image">
                </div>
            </figure>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/600/453" class="slide__image" alt="Random image">
                </div>
                <figcaption class="u-align-self-start slide__caption">
                    Vitae elementum curabitur vitae nunc. Egestas fringilla phasellus faucibus scelerisque eleifend. Vitae elementum curabitur vitae nunc.
                </figcaption>
            </figure>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/600/454" class="slide__image" alt="Random image">
                </div>
                <figcaption class="u-align-self-start slide__caption">
                    This is another optional image caption
                </figcaption>
            </figure>
        </li>
    </ul>
</section>
<section class="carousel carousel--image 
    carousel__container" id="carousel-956998705" aria-labelledby="carouselheading" data-carousel>
    <h3 class="u-sr-only" id="carouselheading">Carousel heading.</h3>

    <ul class="u-list-unstyled u-list-inline carousel__slides">
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/650/450" class="slide__image" alt="Random image">
                </div>
            </figure>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/600/400" class="slide__image" alt="Random image">
                </div>
            </figure>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/700/450" class="slide__image" alt="Random image">
                </div>
            </figure>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="https://picsum.photos/620/450" class="slide__image" alt="Random image">
                </div>
            </figure>
        </li>
    </ul>
</section>
<section class="carousel carousel--text 
    carousel__container" id="carousel-684212178" aria-labelledby="carouselheading" data-carousel>
    <h3 class="u-sr-only" id="carouselheading">Carousel heading.</h3>

    <ul class="u-list-unstyled u-list-inline carousel__slides">
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <article class="slide__inner">
                <h4>This is a slide heading</h4>
                <p>This is the body text Fermentum dui faucibus in ornare quam viverra orci sagittis. Facilisis sed odio morbi quis. Egestas diam in arcu cursus euismod quis viverra nibh.</p>
            </article>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <article class="slide__inner">
                <p>This slide has no heading. This is the body text of the slide, maximum of # characters.</p>
            </article>
        </li>
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            <article class="slide__inner">
                <h4>This is a slide heading</h4>
                <p>Tempor id eu nisl nunc mi. Ut tellus elementum sagittis vitae. Enim facilisis gravida neque convallis a. Augue lacus viverra vitae congue eu consequat ac felis. Vel fringilla est ullamcorper eget nulla. Urna id volutpat lacus laoreet non.
                    Proin fermentum leo vel orci porta non pulvinar.</p>
            </article>
        </li>
    </ul>
</section>
{% set slides = carousels.slides %}
{% set type = {
    imageText: ' carousel--image-text',
    image: ' carousel--image',
    text: ' carousel--text',
} %}

{% for carousel in carousels %}
<section  class="carousel carousel--
    {%- if carousel.type -%}
        {{- carousel.type }}
    {%- endif %} 
    carousel__container"
    id="carousel-{{ random() }}"
    aria-labelledby="carouselheading" 
    data-carousel
    >
    <h3 class="u-sr-only" id="carouselheading" >Carousel heading.</h3>

    <ul class="u-list-unstyled u-list-inline carousel__slides">
        {% for slide in carousel.slides %}
        <li class="carousel__slide" data-slide-item aria-hidden="true">
            {% if slide.image %}
            <figure class="slide__inner">
                <div class="image__container">
                    <img src="{{ slide.image.src }}" class="slide__image" alt="{{ slide.image.alt }}">
                </div>
                {% if slide.image.caption %}
                <figcaption class="u-align-self-start slide__caption">
                    {{ slide.image.caption }}
                </figcaption>
                {% endif %}
            </figure>
            {% endif %}
            {% if slide.text %}
            <article class="slide__inner">
                {% if slide.heading %}
                <h4>{{slide.heading}}</h4>
                {% endif %}
                <p>{{ slide.text }}</p>
            </article>
            {% endif %}
        </li>
        {% endfor %}
    </ul>
</section>
{% endfor %}
{
  "carousels": [
    {
      "type": [
        "image-text"
      ],
      "slides": [
        {
          "image": {
            "src": "https://picsum.photos/600/451",
            "alt": "Random image",
            "caption": "This is another optional image caption"
          }
        },
        {
          "image": {
            "src": "https://picsum.photos/600/452",
            "alt": "Random image",
            "caption": ""
          }
        },
        {
          "image": {
            "src": "https://picsum.photos/600/453",
            "alt": "Random image",
            "caption": "Vitae elementum curabitur vitae nunc. Egestas fringilla phasellus faucibus scelerisque eleifend. Vitae elementum curabitur vitae nunc."
          }
        },
        {
          "image": {
            "src": "https://picsum.photos/600/454",
            "alt": "Random image",
            "caption": "This is another optional image caption"
          }
        }
      ]
    },
    {
      "type": [
        "image"
      ],
      "slides": [
        {
          "image": {
            "src": "https://picsum.photos/650/450",
            "alt": "Random image"
          }
        },
        {
          "image": {
            "src": "https://picsum.photos/600/400",
            "alt": "Random image"
          }
        },
        {
          "image": {
            "src": "https://picsum.photos/700/450",
            "alt": "Random image"
          }
        },
        {
          "image": {
            "src": "https://picsum.photos/620/450",
            "alt": "Random image"
          }
        }
      ]
    },
    {
      "type": [
        "text"
      ],
      "slides": [
        {
          "heading": "This is a slide heading",
          "text": "This is the body text  Fermentum dui faucibus in ornare quam viverra orci sagittis. Facilisis sed odio morbi quis. Egestas diam in arcu cursus euismod quis viverra nibh."
        },
        {
          "text": "This slide has no heading. This is the body text of the slide, maximum of # characters."
        },
        {
          "heading": "This is a slide heading",
          "text": "Tempor id eu nisl nunc mi. Ut tellus elementum sagittis vitae. Enim facilisis gravida neque convallis a. Augue lacus viverra vitae congue eu consequat ac felis. Vel fringilla est ullamcorper eget nulla. Urna id volutpat lacus laoreet non. Proin fermentum leo vel orci porta non pulvinar."
        }
      ]
    }
  ]
}
  • Content:
    .carousel {
      height: auto;
    
      @include bem-m('text') {
    
        .slidenav {
    
            @media (min-width: $breakpoint-sm) { margin-top: 35px; }
        }
      }
    
      @include bem-m('image-text') {
    
        .slidenav {
    
            @media (min-width: $breakpoint-sm) { margin-top: 35px; }
        }
      }
    
      @include bem-m('image') {
    
        .slidenav {
    
            @media (min-width: $breakpoint-sm) { margin-top: 15px; }
        }
      }
    
      &__container {
        position: relative;
        display: flex;
        flex-direction: column;
        align-items: center;
        justify-content: space-between;
        margin: 0 auto 50px;
        width: 100%;
        height: 100%;
        overflow: hidden;
    
        @media (min-width: $breakpoint-sm) { background-color: color(default, light); }
    
        @media (min-width: $breakpoint-lg) { max-width: 800px; }
      }
    
      &__slides {
        position: relative;
        margin: 0 auto;
        width: 100%;
        height: 100%;
    
        @media (min-width: $breakpoint-md) { width: 600px; }
    
        li[aria-hidden='true'] { visibility: hidden; }
    
        .carousel--image-text & {
          min-height: 380px; // to account for a long caption, (2 lines at desktop width)
    
          @media (min-width: $breakpoint-sm) { min-height: 430px; }
    
          @media (min-width: $breakpoint-md) { min-height: 500px; }
        }
    
        .carousel--image & {
          min-height: 270px;
    
          @media (min-width: $breakpoint-sm) { min-height: 350px; }
    
          @media (min-width: $breakpoint-md) { min-height: 440px; }
        }
    
        .carousel--text & {
          min-height: 290px;
    
          @media (min-width: $breakpoint-md) { min-height: 440px; }
        }
      }
    
      &__slide {
        position: absolute;
        top: 0;
        left: 0;
        display: block;
        width: 100%;
        height: auto;
        opacity: 0;
        visibility: hidden;
        transition: left .7s ease-in-out, opacity .7s ease-in-out, visibility .7s ease-in-out;
    
        &.in-transition { visibility: visible; }
    
        .slide__inner {
    
          .carousel--text & {
            display: flex;
            flex-direction: column;
            justify-content: center;
            margin: 0 auto;
            padding: $base-spacing;
            width: 80%;
            background-color: color(neutral,lightest);
    
            @media (min-width: $breakpoint-md) { width: 100%; }
          }
    
          h4,
          p { text-align: center; }
        }
    
        .carousel--text & {
          display: flex;
          height: 100%;
        }
    
        .image__container {
          position: relative;
          width: 100%;
          height: 240px;
          overflow: hidden;
          background-color: color(neutral, lighest);
    
          @media (min-width: $breakpoint-sm) { height: 350px; }
    
          @media (min-width: $breakpoint-md) { height: 440px; }
    
          .carousel--image-text &,
          .carousel--image & { background-color: color(neutral, lightest); }
        }
    
        .slide__image {
          position: absolute;
          top: 50%;
          width: 100%;
          height: auto;
          min-height: 210px;
          transform: translateY(-50%);
        }
    
        .slide__caption {
          @include font-size($base-font-size);
    
          padding: $base-spacing;
          min-height: 60px;
          background-color: color(neutral, lightest);
        }
    
        &.next { left: 100%; }
    
        &.prev { left: -100%; }
    
        &.current {
          opacity: 1;
          visibility: visible;
        }
      }
    
      &__nav {
        position: absolute;
        top: 0;
        right: 0;
        left: 0;
        display: flex;
        flex-direction: row;
        align-items: center;
        justify-content: space-between;
        margin-bottom: 0;
        width: 100%;
        height: 240px; // matches max-height of image container
    
        @media (min-width: $breakpoint-sm) { height: 350px; } // matches max-height of image container
    
        @media (min-width: $breakpoint-md) { height: 440px; } // matches max-height of image container
    
        li { margin-top: 0; }
      }
    
      & .btn__carousel {
        border-radius: 0;
        padding: 10px;
        min-width: 44px;
        background-color: color(default, transparent-medium);
        cursor: pointer;
        color: color(default, light);
    
        @media (min-width: $breakpoint-md) {
          background-color: color(default, light);
          color: color(default, dark);
        }
    
      &:focus,
      &:hover {
          background-color: color(neutral, base);
          color: color(default, light);
        }
    
        .carousel--text & {
          background-color: transparent;
          color: color(default, dark);
        }
      }
    
      // prev and next arrows 
      &__arrow {
        width: 26px;
        height: 26px;
        color: currentColor;
    
        &--prev { transform: rotate(90deg); }
    
        &--next { transform: rotate(-90deg); }
      }
    
    
      .slidenav > li > button {
        position: relative;
        border-radius: 0;
        border: 1px solid color(neutral, lighter);
        min-width: 44px;
        height: 44px;
        background-color: color(default,light);
        font-size: 16px;
        color: color(default, dark);
    
        &.current,
        &:hover {
          background-color: color(neutral, light);
          color: color(default, light);
        }
    
        &.btn__action {
          margin-right: $small-spacing;
          padding: 0;
          min-width: 44px;
          background-color: color(neutral, base);
          color: color(default, light);
    
          svg {
            margin-bottom: 3px;
            width: 22px;
            height: 22px;
            pointer-events: none;
          }
        }
      }
    }
    
  • URL: /components/raw/carousel/_carousel.scss
  • Filesystem Path: components/carousel/_carousel.scss
  • Size: 5.1 KB
  • Content:
    /* Focusin/out event polyfill (for Firefox) by nuxodin
     * Source: https://gist.github.com/nuxodin/9250e56a3ce6c0446efa
     */
    
    !function(){
      var w = window,
      d = w.document;
    
      if( w.onfocusin === undefined ){
        d.addEventListener('focus' ,addPolyfill ,true);
        d.addEventListener('blur' ,addPolyfill ,true);
        d.addEventListener('focusin' ,removePolyfill ,true);
        d.addEventListener('focusout' ,removePolyfill ,true);
      }
      function addPolyfill(e){
        var type = e.type === 'focus' ? 'focusin' : 'focusout';
        var event = new CustomEvent(type, { bubbles:true, cancelable:false });
        event.c1Generated = true;
        e.target.dispatchEvent( event );
      }
      function removePolyfill(e){
        if(!e.c1Generated){ // focus after focusin, so chrome will the first time trigger tow times focusin
          d.removeEventListener('focus' ,addPolyfill ,true);
          d.removeEventListener('blur' ,addPolyfill ,true);
          d.removeEventListener('focusin' ,removePolyfill ,true);
          d.removeEventListener('focusout' ,removePolyfill ,true);
        }
        setTimeout(function(){
          d.removeEventListener('focusin' ,removePolyfill ,true);
          d.removeEventListener('focusout' ,removePolyfill ,true);
        });
      }
    }();
    
    /*
    Carousel Prototype
    Eric Eggert for W3C
    https://www.w3.org/WAI/tutorials/carousels/full-code/
    */
    
    var myCarousel = (function() {
    
      "use strict";
    
      // Initial variables
      var carousel, slides, index, slidenav, settings, timer, setFocus, animationSuspended;
    
      // Helper function: Iterates over an array of elements
      function forEachElement(elements, fn) {
        for (var i = 0; i < elements.length; i++)
        fn(elements[i], i);
      }
    
      // Helper function: Remove Class
      function removeClass(el, className) {
        if (el.classList) {
          el.classList.remove(className);
        } else {
          el.className = el.className.replace(new RegExp('(^|\\b)' + className.split(' ').join('|') + '(\\b|$)', 'gi'), ' ');
        }
      }
    
      // Helper function: Test if element has a specific class
      function hasClass(el, className) {
        if (el.classList) {
          return el.classList.contains(className);
        } else {
          return new RegExp('(^| )' + className + '( |$)', 'gi').test(el.className);
        }
      }
    
      // Initialization for the carousel
      // Argument: set = an object of settings
      // Possible settings:
      // id <string> ID of the carousel wrapper element (required).
      // slidenav <bool> If true, a list of slides is shown.
      // animate <bool> If true, the slides can be animated.
      // startAnimated <bool> If true, the animation begins immediately.
      //                      If false, the animation needsto be initiated by clicking the play button.
      function init(set) {
    
        // Make settings available to all functions
        settings = set;
    
        // Select the element and the individual slides
        carousel = document.getElementById(settings.id);
        slides = carousel.querySelectorAll('[data-slide-item]');
    
        carousel.classList.add('active');
    
        // Create unordered list for controls, and attach click events for previous and next slide
        var ctrls = document.createElement('ul');
        ctrls.className = 'u-list-unstyled carousel__nav controls';
        ctrls.innerHTML = '<li>' +
        '<button type="button" class="btn btn__carousel btn-prev" data-button-previous>' + 
            '<span class="u-sr-only visuallyhidden">Go to previous item.</span>' + 
            '<svg class="carousel__arrow carousel__arrow--prev" viewBox="0 0 407.437 407.437" role="presentation" aria-hidden="true"><polygon points="386.258,91.567 203.718,273.512 21.179,91.567 0,112.815 203.718,315.87 407.437,112.815" fill="currentColor"/></svg>' +
            '</button>' +
          '</li>' +
          '<li>' +
            '<button type="button" class="btn btn__carousel btn-next" data-button-next>' +
              '<span class="u-sr-only visuallyhidden">Go to next item.</span>' +
              '<svg class="carousel__arrow carousel__arrow--next" viewBox="0 0 407.437 407.437" role="presentation" aria-hidden="true"><polygon points="386.258,91.567 203.718,273.512 21.179,91.567 0,112.815 203.718,315.87 407.437,112.815" fill="currentColor"/></svg>' + 
            '</button>' +
          '</li>';
    
        ctrls.querySelector('[data-button-previous]')
          .addEventListener('click', function () {
          prevSlide(true);
        });
        ctrls.querySelector('[data-button-next]')
          .addEventListener('click', function () {
          nextSlide(true);
        });
    
        carousel.appendChild(ctrls);
    
        // If the carousel is animated or a slide navigation is requested in the settings, another unordered list that contains those elements is added. (Note that you cannot supress the navigation when it is animated.)
        if (settings.slidenav || settings.animate) {
          console.log(settings);
          slidenav = document.createElement('ul');
          
          slidenav.className = 'u-list-inline slidenav';
    
          if (settings.animate) {
            var li = document.createElement('li');
    
            if (settings.startAnimated) {
              li.innerHTML = '<button class="btn btn__action" data-action="stop"><span class="u-sr-only visuallyhidden">Stop Animation</span><svg viewBox="0 0 44 44" xmlns="http://www.w3.org/2000/svg"><path fill="currentColor" d="M11.3 11.3h21.5v21.5H11.3z"/></svg></button>';
            } else {
              li.innerHTML = '<button class="btn btn__action" data-action="start"><span class="u-sr-only visuallyhidden">Start Animation</span><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 44 44"><path fill="currentColor" d="M10.6 6.4l25.7 14.3c.1.1.2.2.2.4s-.1.3-.2.4L10.8 37.6c-.1.1-.3.1-.6 0-.2-.1-.3-.3-.3-.4V6.8c0-.2.1-.3.3-.4.1-.1.3-.1.4 0z"/></svg></button>';
            }
    
            slidenav.appendChild(li);
          } 
    
          if (settings.slidenav) {
            forEachElement(slides, function(el, i){
            var li = document.createElement('li');
            
            var klass = (i===0) ? 'class="btn current" ' : 'btn ';
            var kurrent = (i===0) ? ' <span class="u-sr-only visuallyhidden">(Current Item)</span>' : '';
    
            li.innerHTML = '<button '+ klass +'data-slide="' + i + '"><span class="u-sr-only visuallyhidden">News</span> ' + (i+1) + kurrent + '</button>';
            slidenav.appendChild(li);
            });
          }
    
          slidenav.addEventListener('click', function(event) {
            var button = event.target;
            
            if (button.localName == 'button') {
              if (button.getAttribute('data-slide')) {
                setSlides(button.getAttribute('data-slide'), true);
              }
              
              if (button.getAttribute('data-action') == "stop") {
                stopAnimation();
              } else if (button.getAttribute('data-action') == "start") {
                startAnimation();
              }
            }
          }, true);
    
          carousel.classList.add('has-slidenav');
          carousel.appendChild(slidenav);
        } 
    
        // Add a live region to announce the slide number when using the previous/next buttons
        var liveregion = document.createElement('div');
        liveregion.setAttribute('aria-live', 'polite');
        liveregion.setAttribute('aria-atomic', 'true');
        liveregion.setAttribute('class', 'liveregion u-sr-only visuallyhidden');
        carousel.appendChild(liveregion);
    
        // After the slide has transitioned, remove the in-transition class. If focus should be set, set the tabindex attribute to -1 to prevent keyboard navigation to this while it is not visible, and focus the slide.
        slides[0].parentNode.addEventListener('transitionend', function (event) {
          var slide = event.target;
          removeClass(slide, 'in-transition');
          if (hasClass(slide, 'current'))  {
            if(setFocus) {
              slide.setAttribute('tabindex', '-1');
              slide.focus();
              setFocus = true;
            }
          }
        });
    
        // When the mouse enters the carousel, suspend the animation.
        carousel.addEventListener('mouseenter', suspendAnimation);
    
        // When the mouse leaves the carousel, and the animation is suspended, start the animation.
        carousel.addEventListener('mouseleave', function(event) {
          if (animationSuspended) {
            startAnimation();
          }
        });
    
        // When the focus enters the carousel, suspend the animation
        carousel.addEventListener('focusin', function(event) {
          if (!hasClass(event.target, 'slide')) {
            suspendAnimation();
          }
        });
    
        // When the focus leaves the carousel, and the animation is suspended, start the animation
        carousel.addEventListener('focusout', function(event) {
          if (!hasClass(event.target, 'slide') && animationSuspended) {
            startAnimation();
          }
        });
    
        // Set the index (= current slide) to 0 – the first slide
        index = 0;
        setSlides(index);
    
        // If the carousel is animated, advance to the
        // next slide after 5s
        if (settings.startAnimated) {
          timer = setTimeout(nextSlide, 5000);
        }
      }
    
      // Function to set a slide the current slide
      function setSlides(new_current, setFocusHere, transition, announceItemHere) {
        // Focus, transition and announceItem are optional parameters.
        // Focus denotes if the focus should be set after the
        // carousel advanced to slide number new_current.
        // Transition denotes if the transition is going into the
        // next or previous direction.
        // If announceItem is set to true, the live region’s text is changed (and announced)
        // Here defaults are set:
        let announceItem;
        setFocus = typeof setFocusHere !== 'undefined' ? setFocusHere : true;
        transition = typeof transition !== 'undefined' ? transition : 'none';
        announceItem = typeof announceItemHere !== 'undefined' ? announceItemHere : true;
    
        new_current = parseFloat(new_current);
    
        var length = slides.length;
        var new_next = new_current+1;
        var new_prev = new_current-1;
    
        // If the next slide number is equal to the length,
        // the next slide should be the first one of the slides.
        // If the previous slide number is less than 0.
        // the previous slide is the last of the slides.
        if(new_next === length) {
          new_next = 0;
        } else if(new_prev < 0) {
          new_prev = length-1;
        }
    
        // Reset slide classes
        for (var i = slides.length - 1; i >= 0; i--) {
          slides[i].className = "carousel__slide";
        }
    
        // Add classes to the previous, next and current slide
        slides[new_next].className = 'next carousel__slide' + ((transition == 'next') ? ' in-transition' : '');
        slides[new_next].setAttribute('aria-hidden', 'true');
    
        slides[new_prev].className = 'prev carousel__slide' + ((transition == 'prev') ? ' in-transition' : '');
        slides[new_prev].setAttribute('aria-hidden', 'true');
    
        slides[new_current].className = 'current carousel__slide';
        slides[new_current].removeAttribute('aria-hidden');
    
        // Update the text in the live region which is then announced by screen readers.
        if (announceItem) {
          carousel.querySelector('.liveregion').textContent = 'Item ' + (new_current + 1) + ' of ' + slides.length;
        }
    
        // Update the buttons in the slider navigation to match the currently displayed item
        if(settings.slidenav) {
          var buttons = carousel.querySelectorAll('.slidenav button[data-slide]');
          for (var j = buttons.length - 1; j >= 0; j--) {
            buttons[j].className = '';
            buttons[j].innerHTML = '<span class="u-sr-only visuallyhidden">Carousel Item </span> ' + (j+1);
          }
          buttons[new_current].className = "current";
          buttons[new_current].innerHTML = '<span class="u-sr-only visuallyhidden">Carousel Item </span> ' + (new_current+1) + ' <span class="u-sr-only visuallyhidden">(Current Item)</span>';
        }
    
        // Set the global index to the new current value
        index = new_current;
      }
    
      // Function to advance to the next slide
      function nextSlide(announceItem) {
        announceItem = typeof announceItem !== 'undefined' ? announceItem : true;
    
        var length = slides.length,
        new_current = index + 1;
    
        if(new_current === length) {
          new_current = 0;
        }
    
        // If we advance to the next slide, the previous needs to be
        // visible to the user, so the third parameter is 'prev', not
        // next.
        setSlides(new_current, false, 'prev', announceItem);
    
        // If the carousel is animated, advance to the next
        // slide after 5s
        if (settings.animate) {
          timer = setTimeout(nextSlide, 5000);
        }
      }
    
      // Function to advance to the previous slide
      function prevSlide(announceItem) {
        announceItem = typeof announceItem !== 'undefined' ? announceItem : true;
    
        var length = slides.length,
        new_current = index - 1;
    
        // If we are already on the first slide, show the last slide instead.
        if(new_current < 0) {
          new_current = length-1;
        }
    
        // If we advance to the previous slide, the next needs to be
        // visible to the user, so the third parameter is 'next', not
        // prev.
        setSlides(new_current, false, 'next', announceItem);
      }
    
      // Function to stop the animation
      function stopAnimation() {
        let _this;
        clearTimeout(timer);
        settings.animate = false;
        animationSuspended = false;
        _this = carousel.querySelector('[data-action]');
        _this.innerHTML = '<span class="u-sr-only visuallyhidden">Start Animation </span><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 44 44"><path fill="currentColor" d="M10.6 6.4l25.7 14.3c.1.1.2.2.2.4s-.1.3-.2.4L10.8 37.6c-.1.1-.3.1-.6 0-.2-.1-.3-.3-.3-.4V6.8c0-.2.1-.3.3-.4.1-.1.3-.1.4 0z"/></svg>';
        _this.setAttribute('data-action', 'start');
      }
    
      // Function to start the animation
      function startAnimation() {
        let _this;
        settings.animate = true;
        animationSuspended = false;
        timer = setTimeout(nextSlide, 5000);
        _this = carousel.querySelector('[data-action]');
        _this.innerHTML = '<span class="u-sr-only visuallyhidden">Stop Animation </span><svg viewBox="0 0 44 44" xmlns="http://www.w3.org/2000/svg"><path fill="currentColor" d="M11.3 11.3h21.5v21.5H11.3z"/></svg>';
        _this.setAttribute('data-action', 'stop');
      }
    
      // Function to suspend the animation
      function suspendAnimation() {
        if(settings.animate) {
          clearTimeout(timer);
          settings.animate = false;
          animationSuspended = true;
        }
      }
    
      // Making some functions public
      return {
        init:init,
        next:nextSlide,
        prev:prevSlide,
        goto:setSlides,
        stop:stopAnimation,
        start:startAnimation
      };
    });
    
    // check for multiple carousels on a page
    var carouselIds = document.querySelectorAll('[data-carousel]');
    
    for (let i = 0; i < carouselIds.length; i++) {
      let id;
      var carousel = new myCarousel();
      id = carouselIds[i].id; 
      carousel.init({
        id: id,
        slidenav: true,
        animate: false, 
        startAnimated: false, 
        announceItem: true,
      });
    }
    
  • URL: /components/raw/carousel/carousel.js
  • Filesystem Path: components/carousel/carousel.js
  • Size: 14.7 KB

Usage

Carousels show a collection of items one at a time. They are also known as “slideshows” and “sliders”. Typical uses of carousels include scrolling news headlines, featured articles on home pages, and image galleries. This carousel has been adapted from the following example: Web Accessibility Tutorials / Carousels.

This carousel includes modifier classes to accomodate various types of content:

  • carousel--image should be added to the outer container when the carousel includes only images.
  • carousel--image-text should be added to the outer container when the carousel includes images with text.
  • carousel--text should be added to the outer container when the carousel includes text-only content.

If the carousel is configured to include additional navigation buttons below the slides, has-slidenav will be added to the outermost container.

JavaScript Configuration Settings

This carousel comes with the following settings that can be configured in the accompanying JavaScript file:

  • slidenav: Set this attribute to true to add additional navigation buttons for each item in the carousel, allowing screen reader users to get an overview of carousel content, where they are in the sequence and will enable them to navigate directly to any item.
  • animate: Set this attribute to true to include a play/stop button in the slide navigation. Otherwise, it should be set to false.
  • startAnimated: Set this attribute to true to autoplay the carousel on page load. Otherwise, it should be set to false.
  • announceItem: Set this attribute to true to include an aria-live="polite" attribute on the carousel navigation, informing screen reader users what item is currently active.

Labelling Expectations

  • Each carousel should have a unique id attribute.
  • Each carousel should be enclosed in a labeled region with an aria-labelledby attribute to define the region.
  • Each inactive carousel item should include the aria-hidden="true" attribute. This attribute should be removed when a carousel item is active.
  • When a carousel item is focused, it should have a tabindex=“-1".

Focus Expectations

  • All navigation buttons should have a visible keyboard focus state.
  • Keyboard focus does not move when the carousel advances automatically.
  • Keyboard focus does not move when the previous or next buttons are used.
  • When users interact with navigation buttons focus should be on the selected slide.

Keyboard Expectations

  • Tab = Move to next focusable element
  • Shift + Tab = Move to previous focusable element
  • Enter or Space = Navigate to the Previous or Next slide
  • Control + Option = Navigate through item content with screen reader

Screenreader Expectations

  • Screenreaders will announce all buttons and actionable elements.
  • When the previous or next button is focused, screenreaders will announce the item number in sequence.
  • When using the slide navigation buttons, screenreaders will announce the item number and read the content within the slide.

Tab Order Expectations

Once inside the section containing of the carousel, the tab order will be as follows:

  1. Previous button
  2. Next button
  3. Buttons in the slide navigation