-
Notifications
You must be signed in to change notification settings - Fork 705
/
Copy pathslides.js
111 lines (89 loc) · 2.58 KB
/
slides.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
export class Slide extends HTMLElement {
#slideFunction;
#slideIterator;
#currentState = -1;
#queueChain = Promise.resolve();
#done = false;
useTransitions = false;
constructor(slideFunction = async function* () {}) {
super();
this.#slideFunction = slideFunction;
this.goto(0);
}
async #unqueuedGoto(targetState) {
this.innerHTML = "";
this.#done = false;
this.#slideIterator = this.#slideFunction(this);
this.#currentState = -1;
while (this.#currentState !== targetState) {
await this.#advance({ useTransitions: false });
if (!this.hasNext) return;
}
}
#queue(callback) {
return (this.#queueChain = this.#queueChain.finally(callback));
}
goto(targetState) {
return this.#queue(() => this.#unqueuedGoto(targetState));
}
async #advance({ useTransitions }) {
if (this.#done) return;
this.useTransitions = useTransitions;
this.#currentState++;
const { done } = await this.#slideIterator.next();
this.#done = done;
}
previous() {
return this.#queue(() => {
if (this.#currentState === 0) return;
return this.#unqueuedGoto(this.#currentState - 1);
});
}
next() {
return this.#queue(() => this.#advance({ useTransitions: true }));
}
get hasNext() {
return !this.#done;
}
get hasPrevious() {
return this.#currentState > 0;
}
}
customElements.define("spec-slide", Slide);
/**
* @param {HTMLElement} element
* @param {Keyframe[] | PropertyIndexedKeyframes} from
* @param {KeyframeAnimationOptions} options
*/
export function transitionFrom(element, from, options) {
const slide = element.closest("spec-slide");
if (!slide) throw Error("Transitioning element must be within spec-slide");
from = Array.isArray(from) ? from : { ...from, offset: 0 };
const anim = element.animate(from, {
...options,
fill: "backwards",
duration: slide.useTransitions ? options.duration : 0,
delay: slide.useTransitions ? options.delay : 0,
});
return anim;
}
/**
* @param {HTMLElement} element
* @param {Keyframe[] | PropertyIndexedKeyframes} to
* @param {KeyframeAnimationOptions} options
*/
export function transition(element, to, options) {
const slide = element.closest("spec-slide");
if (!slide) throw Error("Transitioning element must be within spec-slide");
const anim = element.animate(to, {
...options,
fill: "both",
duration: slide.useTransitions ? options.duration : 0,
delay: slide.useTransitions ? options.delay : 0,
});
anim.finished.then(() => {
anim.commitStyles();
anim.cancel();
});
return anim;
}