Animated carousel

This commit is contained in:
a 2020-09-14 15:24:32 +02:00
parent 7cef893930
commit 02f8f27839
2 changed files with 145 additions and 31 deletions

View file

@ -1,5 +1,4 @@
import { css, html, LitElement } from "lit-element"; import { animationPromise, cyclicArray, Alarms, html } from "./utils";
import { cyclicArray } from "./utils";
const items = [ const items = [
{ {
@ -23,39 +22,125 @@ const items = [
}, },
]; ];
class Carousel extends LitElement { class Carousel extends HTMLElement {
currentIndex = 0; items = cyclicArray(items);
currentItem = items[this.currentIndex]; shadow;
constructor() { constructor() {
super(); super();
this.shadow = this.attachShadow({ mode: "open" });
} }
render() { connectedCallback() {
return html` this.shadow.innerHTML = html`
<button class="left">${'<'}</button> <button class="next">${"<"}</button>
<button class="right">${'>'}</button> <div class="container">
<img src="${this.items.next().value.img}" />
<img src=${this.currentItem.img}></img> </div>
<button class="prev">${">"}</button>
<style>
${Carousel.css}
</style>
`; `;
const next = this.shadow.querySelector(".next") as HTMLButtonElement;
const prev = this.shadow.querySelector(".prev") as HTMLButtonElement;
next.addEventListener("click", () => this.rotate());
prev.addEventListener("click", () => this.rotate(true));
Alarms.create(6000, "rotateAlarm").then(() => this.rotate());
} }
static get styles() { rotate(reverse = false) {
return css` Alarms.create(6000, "rotateAlarm").then(() => this.rotate());
const container = this.shadow.querySelector(".container") as HTMLDivElement;
const oldItem = container.querySelector(
"img:last-of-type"
) as HTMLImageElement; //In case we are waiting for animation to finish
const newItem = document.createElement("img") as HTMLImageElement;
const inClass = reverse ? ["reverse", "left"] : ["right"];
const outClass = reverse ? ["reverse", "right"] : ["left"];
newItem.src = this.items.next(reverse).value.img;
container.appendChild(newItem);
animationPromise(oldItem).finally(() => {
container.removeChild(oldItem);
});
oldItem.classList.add(...outClass);
animationPromise(newItem).finally(() =>
newItem.classList.remove(...inClass)
);
newItem.classList.add(...inClass);
}
static get css() {
return `
:host { :host {
max-width: 100%; width: 100%;
height: 100%; height: 100%;
display: flex;
align-items: center;
justify-content: center;
}
.container {
height: 100%; height: 100%;
width: 60%;
clip-path: margin-box;
position: relative;
display: flex;
align-items: center;
justify-content: center;
overflow: hidden;
} }
img { img {
max-height: 100%; animation-duration: 1.5s;
box-shadow: 0px 0px 5px 0px var(--background-opposite-1); object-fit: scale-down;
transition: 0.2s; width: 100%;
height: 100%;
} }
img:hover, button:hover~img { @keyframes slide-to-left {
transform: scale(1.05); from {
transform: translateX(0%);
}
to {
transform: translateX(-200%);
}
}
@keyframes slide-from-right {
from {
transform: translateX(200%);
}
to {
transform: translateX(0%);
}
}
.right {
animation-name: slide-from-right;
}
.left {
animation-name: slide-to-left;
position: absolute;
}
.reverse {
animation-direction: reverse;
}
.left.reverse {
position: static;
}
.right.reverse {
position: absolute;
} }
button { button {
@ -63,18 +148,10 @@ class Carousel extends LitElement {
border-radius: 50%; border-radius: 50%;
background-color: var(--background-primary-2); background-color: var(--background-primary-2);
cursor: pointer; cursor: pointer;
position: relative;
top:-50%;
z-index: 1; z-index: 1;
color: var(--text-color-3);
margin: -20%;
} }
.left {
left: 20%;
}
.right {
left: 80%;
}
`; `;
} }
} }

View file

@ -1,5 +1,12 @@
/// A generator that infinitely loops around an array /**
export function* cyclicArray<T>(array: T[]) { A generator that infinitely loops around an array
@param array The array to loop around
@yields Yields items from the array wrapping around to the start.
@next Decideds if the iterator should go backwards
*/
export function* cyclicArray<T>(array: T[]): Generator<T, never, boolean> {
let index = 0; let index = 0;
while (true) { while (true) {
const reverse = yield array[index]; const reverse = yield array[index];
@ -11,12 +18,23 @@ export function* cyclicArray<T>(array: T[]) {
} }
} }
/**
Rotates an array
@param arr The array to rotate
@param reverse Should the array be rotated in the opposite direction
@returns The rotated array. For example by passing in [1,2,3] you will recieve [3,1,2]
*/
export const rotateArray = <T>(arr: T[], reverse = false): T[] => { export const rotateArray = <T>(arr: T[], reverse = false): T[] => {
if (reverse) arr.unshift(arr.pop()!); if (reverse) arr.unshift(arr.pop()!);
else arr.push(arr.shift()!); else arr.push(arr.shift()!);
return arr; return arr;
}; };
/**
Ergonomic wrapper for creating new promises
@returns A tuple of the created promise and it's resolve and reject functions
*/
type resolveCallback<T> = (value: T) => void; type resolveCallback<T> = (value: T) => void;
type rejectCallback = (reason?: unknown) => void; type rejectCallback = (reason?: unknown) => void;
export const newPromise = <T>(): [ export const newPromise = <T>(): [
@ -60,3 +78,22 @@ export namespace Alarms {
}; };
export const clearAll = () => alarmsMap.forEach((a) => a[1]()); export const clearAll = () => alarmsMap.forEach((a) => a[1]());
} }
/**
@returns Promise that resolves when an animation on the element finishes
*/
export const animationPromise = (element: HTMLElement) => {
const [promise, reject, resolve] = newPromise<void>();
element.addEventListener("animationend", () => resolve(), true);
element.addEventListener("animationcancel", () => reject(), true);
return promise;
};
/**
Escapes all the keys in the template string
*/
export const html = (strings: TemplateStringsArray, ...keys: string[]) =>
strings.reduce((p, c, i) => p + escape(keys[i - 1]) + c);
export const escape = (string: string) =>
string.replace(/</g, "&lt").replace(/>/g, "&gt");