-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathplayback_element.ts
108 lines (91 loc) · 4.08 KB
/
playback_element.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
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
import { MidiBuffer, TuneObject, renderAbc, synth } from 'abcjs';
import { MarkdownRenderChild } from 'obsidian';
import { AUDIO_PARAMS, DEFAULT_OPTIONS, OPTIONS_REGEX, PLAYBACK_CONTROLS_ID, SYNTH_INIT_OPTIONS } from './cfg';
import { NoteHighlighter, togglePlayingHighlight } from './note_highlighter';
/**
* This class abstraction is needed to support load/unload hooks
*
* "If your post processor requires lifecycle management, for example, to clear an interval, kill a subprocess, etc when this element is removed from the app..."
* https://marcus.se.net/obsidian-plugin-docs/reference/typescript/interfaces/MarkdownPostProcessorContext#addchild
*/
export class PlaybackElement extends MarkdownRenderChild {
private readonly abortController = new AbortController();
private readonly midiBuffer: MidiBuffer = new synth.CreateSynth();
private readonly synthCtrl = new synth.SynthController();
constructor(
private readonly el: HTMLElement,
private readonly markdownSource: string,
) {
super(el); // important
}
onload() {
const { userOptions, source } = this.parseOptionsAndSource();
const renderResp = renderAbc(this.el, source, Object.assign(DEFAULT_OPTIONS, userOptions));
this.enableAudioPlayback(renderResp[0]);
}
/**
* Stop the music and clean things up.
*
* (Tested) Called when:
* 1. Cursor focus goes into the text area (which switches from preview to edit mode)
* 2. A tab containing this is closed (very important)
*
* Not called when:
* 1. Switching tabs to a different one (audio keeps playing)
*/
onunload() {
this.abortController.abort(); // dom event listeners
// A lot of steps, but I think all these things need to happen to really stop in-progress audio playback for ABCjs.
this.synthCtrl.restart();
this.synthCtrl.pause();
this.midiBuffer.stop(); // doesn't stop the music by itself?
}
parseOptionsAndSource(): { userOptions: {}, source: string } {
let userOptions = {};
const optionsMatch = this.markdownSource.match(OPTIONS_REGEX);
let source = this.markdownSource; // can be modified, removes the options portion.
if (optionsMatch !== null) {
source = optionsMatch.groups["source"];
try {
userOptions = JSON.parse(optionsMatch.groups["options"]);
} catch (e) {
console.error(e);
this.renderError(`<strong>Failed to parse user-options</strong>\n\t${e}`);
}
}
return { userOptions, source };
}
renderError(error?: string) {
if (error == null) return;
const errorNode = document.createElement('div');
errorNode.innerHTML = error;
errorNode.addClass("obsidian-plugin-abcjs-error");
this.el.appendChild(errorNode);
}
// Audio playback features
// Many variants, options, and guidance here: https://paulrosen.github.io/abcjs/audio/synthesized-sound.html
enableAudioPlayback(visualObj: TuneObject) {
if (!synth.supportsAudio()) return;
// We need the SynthController to drive NoteHighlighter (CursorControl), even though we don't want the UI controls from SynthController
this.synthCtrl.load(
`#${PLAYBACK_CONTROLS_ID}`, //controlsEl, // can be an HTMLElement reference or css selector
new NoteHighlighter(this.el), // an implementation of a `CursorControl`
);
this.midiBuffer.init({ visualObj, options: SYNTH_INIT_OPTIONS })
.then(() => this.synthCtrl.setTune(visualObj, false, AUDIO_PARAMS))
.catch(console.warn.bind(console));
const signal = this.abortController.signal; // for event cleanup
this.el.addEventListener('click', this.togglePlayback, { signal });
this.el.addEventListener('dblclick', this.restartPlayback, { signal });
}
private readonly togglePlayback = () => {
// private access. Can improve when https://github.com/paulrosen/abcjs/pull/917 merges
const isPlaying = (this.midiBuffer as any)?.isRunning;
isPlaying ? this.synthCtrl.pause() : this.synthCtrl.play();
togglePlayingHighlight(this.el)(isPlaying);
};
// start again at the begining of the tune
private readonly restartPlayback = () => {
this.synthCtrl.restart();
};
}