+ simplify vanilla setup drastically

+ move repl + drawer boilerplate inside StrudelMirror
This commit is contained in:
Felix Roos 2023-05-05 23:43:07 +02:00
parent 5b67fccb1b
commit 9ec7109dc8
7 changed files with 101 additions and 103 deletions

View File

@ -5,6 +5,7 @@ import { syntaxHighlighting, defaultHighlightStyle } from '@codemirror/language'
import { javascript } from '@codemirror/lang-javascript';
import { StateField, StateEffect } from '@codemirror/state';
import { oneDark } from './themes/one-dark';
import { repl, Drawer } from '@strudel.cycles/core';
// https://codemirror.net/docs/guide/
export function initEditor({ initialCode = '', onChange, onEvaluate, onStop, theme = oneDark, root }) {
@ -127,22 +128,72 @@ export const flash = (view, ms = 200) => {
};
export class StrudelMirror {
constructor({ root, initialCode = '', onEvaluate, onStop }) {
constructor(options) {
const { root, initialCode = '', onDraw, drawTime = [-2, 2], prebake, ...replOptions } = options;
this.code = initialCode;
this.view = initEditor({
this.drawer = new Drawer((haps, time) => {
const currentFrame = haps.filter((hap) => time >= hap.whole.begin && time <= hap.whole.end);
this.highlight(currentFrame);
onDraw?.(haps, time, currentFrame);
}, drawTime);
const prebaked = prebake();
prebaked.then(async () => {
if (!onDraw) {
return;
}
const { scheduler, evaluate } = await this.repl;
// draw first frame instantly
prebaked.then(async () => {
await evaluate(this.code, false);
this.drawer.invalidate(scheduler);
onDraw?.(this.drawer.visibleHaps, 0, []);
});
});
this.repl = repl({
...replOptions,
onToggle: async (started) => {
replOptions?.onToggle?.(started);
const { scheduler } = await this.repl;
if (started) {
this.drawer.start(scheduler);
} else {
this.drawer.stop();
}
},
beforeEval: async () => {
await prebaked;
},
afterEval: (options) => {
replOptions?.afterEval?.(options);
this.drawer.invalidate();
},
});
this.editor = initEditor({
root,
initialCode,
onChange: (v) => {
this.code = v.state.doc.toString();
},
onEvaluate,
onStop,
onEvaluate: () => this.evaluate(),
onStop: () => this.stop(),
});
}
async evaluate() {
const { evaluate } = await this.repl;
this.flash();
await evaluate(this.code);
}
async stop() {
const { scheduler } = await this.repl;
scheduler.stop();
}
flash(ms) {
flash(this.view, ms);
flash(this.editor, ms);
}
highlight(haps) {
highlightHaps(this.view, haps);
highlightHaps(this.editor, haps);
}
}

View File

@ -38,7 +38,8 @@
"@codemirror/language": "^6.6.0",
"@codemirror/state": "^6.2.0",
"@codemirror/view": "^6.10.0",
"@lezer/highlight": "^1.1.4"
"@lezer/highlight": "^1.1.4",
"@strudel.cycles/core": "workspace:*"
},
"devDependencies": {
"vite": "^4.3.3"

View File

@ -97,19 +97,18 @@ export class Drawer {
},
);
}
check() {
if (!this.scheduler) {
throw new Error('no scheduler set..');
invalidate(scheduler = this.scheduler) {
if (!scheduler) {
return;
}
}
invalidate() {
this.check();
const t = this.scheduler.now();
this.scheduler = scheduler;
const t = scheduler.now();
let [_, lookahead] = this.drawTime;
const [begin, end] = [Math.max(t, 0), t + lookahead + 0.1];
// remove all future haps
this.visibleHaps = this.visibleHaps.filter((h) => h.whole.begin < t);
// query future haps
const futureHaps = this.scheduler.pattern.queryArc(Math.max(t, 0), t + lookahead + 0.1); // +0.1 = workaround for weird holes in query..
const futureHaps = scheduler.pattern.queryArc(begin, end); // +0.1 = workaround for weird holes in query..
// append future haps
this.visibleHaps = this.visibleHaps.concat(futureHaps);
}

View File

@ -1,69 +1,39 @@
// moved from sandbox: https://codesandbox.io/s/vanilla-codemirror-strudel-2wb7yw?file=/index.html:114-186
import { StrudelMirror } from '@strudel/codemirror';
import { initStrudel } from './strudel';
import { funk42 } from './tunes';
import { pianoroll, getDrawOptions, Drawer } from '@strudel.cycles/core';
import { drawPianoroll, evalScope, controls } from '@strudel.cycles/core';
import './style.css';
import { initAudioOnFirstClick } from '@strudel.cycles/webaudio';
import { transpiler } from '@strudel.cycles/transpiler';
import { getAudioContext, webaudioOutput, registerSynthSounds } from '@strudel.cycles/webaudio';
import { registerSoundfonts } from '@strudel.cycles/soundfonts';
const initAudio = initAudioOnFirstClick();
const editor = new StrudelMirror({
root: document.getElementById('editor'),
initialCode: funk42,
onEvaluate,
onStop,
});
async function drawFirstFrame(editor, repl) {
const { evaluate } = repl;
const pattern = await evaluate(editor.code, false);
const initialHaps = pattern.queryArc(0, drawTime[1]);
drawPianoroll(initialHaps, 0);
return repl;
}
const repl = initStrudel().then(async (repl) => {
await drawFirstFrame(editor, repl);
return repl;
});
// init canvas
const canvas = document.getElementById('roll');
canvas.width = canvas.width * 2;
canvas.height = canvas.height * 2;
const drawContext = canvas.getContext('2d');
const drawTime = [-2, 2]; // time window of drawn haps
async function onEvaluate() {
const { evaluate, scheduler } = await repl;
await initAudio;
editor.flash();
if (!scheduler.started) {
scheduler.stop();
await evaluate(editor.code);
drawer.start(scheduler);
} else {
await evaluate(editor.code);
drawer.invalidate(); // this is a bit mystic
}
}
const editor = new StrudelMirror({
defaultOutput: webaudioOutput,
getTime: () => getAudioContext().currentTime,
transpiler,
root: document.getElementById('editor'),
initialCode: funk42,
drawTime,
onDraw: (haps, time) => drawPianoroll({ haps, time, ctx: drawContext, drawTime, fold: 0 }),
prebake: async () => {
initAudioOnFirstClick(); // needed to make the browser happy (don't await this here..)
const loadModules = evalScope(
controls,
import('@strudel.cycles/core'),
import('@strudel.cycles/mini'),
import('@strudel.cycles/tonal'),
import('@strudel.cycles/webaudio'),
);
await Promise.all([loadModules, registerSynthSounds(), registerSoundfonts()]);
},
});
async function onStop() {
const { scheduler } = await repl;
scheduler.stop();
drawer.stop();
}
const drawTime = [-2, 2];
function drawPianoroll(haps, time) {
pianoroll({ ctx, time, haps, ...getDrawOptions(drawTime, { fold: 0 }) });
}
const ctx = canvas.getContext('2d');
let drawer = new Drawer((haps, time) => {
const currentFrame = haps.filter((hap) => time >= hap.whole.begin && time <= hap.whole.end);
editor.highlight(currentFrame);
drawPianoroll(haps, time);
}, drawTime);
document.getElementById('play').addEventListener('click', () => onEvaluate());
document.getElementById('stop').addEventListener('click', async () => onStop());
document.getElementById('play').addEventListener('click', () => editor.evaluate());
document.getElementById('stop').addEventListener('click', () => editor.stop());

View File

@ -1,28 +0,0 @@
import { getAudioContext, webaudioOutput, registerSynthSounds } from '@strudel.cycles/webaudio';
const ctx = getAudioContext();
export async function initStrudel(options = {}) {
const [{ controls, repl, evalScope }, { registerSoundfonts }, { transpiler }] = await Promise.all([
import('@strudel.cycles/core'),
import('@strudel.cycles/soundfonts'),
import('@strudel.cycles/transpiler'),
]);
const loadModules = evalScope(
controls,
import('@strudel.cycles/core'),
import('@strudel.cycles/mini'),
import('@strudel.cycles/tonal'),
import('@strudel.cycles/webaudio'),
);
await Promise.all([loadModules, registerSynthSounds(), registerSoundfonts()]);
return repl({
defaultOutput: webaudioOutput,
getTime: () => ctx.currentTime,
transpiler,
...options,
});
}

View File

@ -78,7 +78,7 @@ let drums = stack(
s("bd*2, ~ sd").bank('RolandTR707').room("0 .1"),
s("hh*4").begin(.2).release(.02).end(.25).release(.02)
.gain(.3).bank('RolandTR707').late(.02).room(.5),
s("shaker_small").struct("[x x*2]*2").speed(".8,.9").release(.02)
//s("shaker_small").struct("[x x*2]*2").speed(".8,.9").release(.02)
).fast(2)
let wurli = note(\`<

View File

@ -175,3 +175,8 @@ Pattern.prototype.pianoroll = function (options) {
pianoroll({ ctx, time, haps, ...getDrawOptions(drawTime, { fold: 0, ...options }) }),
);
};
export function drawPianoroll(options) {
const { drawTime, ...rest } = options;
pianoroll({ ...getDrawOptions(drawTime), ...rest });
}