-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnote_highlighter.ts
46 lines (37 loc) · 1.62 KB
/
note_highlighter.ts
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
import { CursorControl, NoteTimingEvent } from 'abcjs';
/**
* Monitors playback and provides event hooks.
*
* Used to:
* 1. indicate song start/end
* 2. highlight the currently playing note
*
* https://paulrosen.github.io/abcjs/audio/synthesized-sound.html#cursorcontrol-object
*/
export class NoteHighlighter implements CursorControl {
beatSubdivisions = 2;
constructor(private readonly el: HTMLElement) { }
onStart = () => togglePlayingHighlight(this.el)(true);
onFinished = () => rmAllHighlights(this.el)();
// On every note event, clear highlights and redraw with current note highlighted.
onEvent(event: NoteTimingEvent) {
// this was the second part of a tie across a measure line. Just ignore it.
if (event.measureStart && event.left === null) return;
// Select the currently selected notes.
rmNoteHighlights(this.el)();
event.elements.flat().forEach(el => el.classList.add("abcjs-highlight"));
}
}
// -------------------------------------------------------------- 2nd order fn library: Visualization
// outline region when playing
export const togglePlayingHighlight = (el: HTMLElement) => (is: boolean) => el.parentElement.classList.toggle('is-playing', is);
// clear all the note highlights
export const rmNoteHighlights = (parentEl: HTMLElement) => () => {
const selected = Array.from(parentEl.querySelectorAll(".abcjs-highlight"));
selected.forEach(el => el.classList.remove("abcjs-highlight"));
};
// remove both playing and note highlights
export const rmAllHighlights = (parentEl: HTMLElement) => () => {
togglePlayingHighlight(parentEl)(false);
rmNoteHighlights(parentEl)();
}