<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."
}
]
}
]
}
.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;
}
}
}
}
/* 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,
});
}
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.
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.id
attribute.aria-labelledby
attribute to define the region.aria-hidden="true"
attribute. This attribute should be removed when a carousel item is active.tabindex=“-1"
.Tab
= Move to next focusable elementShift + Tab
= Move to previous focusable elementEnter
or Space
= Navigate to the Previous or Next slideControl + Option
←
→
= Navigate through item content with screen readerOnce inside the section containing of the carousel, the tab order will be as follows: