bytebeat-composer/src/bytebeat.ts

446 lines
11 KiB
TypeScript
Raw Normal View History

2024-04-23 23:16:19 +02:00
import { f32ToRound10, isPlainObject, Song, SongMode, StrongPartial } from "./common.ts";
2024-03-14 06:49:35 +01:00
import elements from "./elements.ts";
2024-03-11 18:25:55 +01:00
import { fromUrlData, setUrlData } from "./url/mod.ts";
2024-03-14 06:49:35 +01:00
import { CodeEditor } from "./code-editor/mod.ts";
2024-03-14 23:32:54 +01:00
import { Oscillioscope } from "./oscillioscope.ts";
2021-11-18 14:36:00 +01:00
2024-03-14 06:49:35 +01:00
const timeUnits = ["t", "s"] as const;
type TimeUnit = (typeof timeUnits)[number];
// XXX: global variables
let timeUnit: TimeUnit;
let volume: number;
2024-08-03 07:26:54 +02:00
let sample = 0;
let playSpeed = 0;
let songData: {
sampleRate: number;
mode: SongMode;
} = {
sampleRate: 8000, // float32
mode: "Bytebeat",
};
const searchParams = new URLSearchParams(location.search);
2024-03-14 06:49:35 +01:00
const audioPromise = initAudioContext();
const codeEditor = new CodeEditor(refresh);
2024-03-14 06:49:35 +01:00
const osc = new Oscillioscope(elements.canvas, elements.timeCursor);
2024-08-03 05:55:27 +02:00
2024-03-14 06:49:35 +01:00
if (globalThis.loadLibrary !== false) {
import("./library/load.ts");
}
2024-03-14 06:49:35 +01:00
const { audioCtx, audioGain, audioWorklet } = await audioPromise;
loadSettings();
{
let urlData: Song | undefined;
if (location.hash && globalThis.useUrlData !== false) {
urlData = await fromUrlData(location.hash);
}
2023-12-28 21:34:32 +01:00
setSong(urlData, false);
updateCounterValue();
}
2023-12-28 21:34:32 +01:00
// XXX: global variables
2024-03-14 06:49:35 +01:00
let nextErrType = null;
let nextErr = null;
let nextErrPriority = undefined;
let errorPriority = -Infinity;
async function initAudioContext() {
2024-03-13 23:11:33 +01:00
let audioContextSampleRate = Number(searchParams.get("baseSampleRate"));
// also true for NaN
if (!(audioContextSampleRate > 0)) {
audioContextSampleRate = 48000; // TODO this should be set to an lcm > 44100
}
2024-03-14 06:49:35 +01:00
const audioCtx = new AudioContext({
latencyHint: (searchParams.get("latencyHint") as AudioContextLatencyCategory) ?? "balanced",
sampleRate: audioContextSampleRate,
});
2024-03-14 06:49:35 +01:00
const audioGain = new GainNode(audioCtx);
audioGain.connect(audioCtx.destination);
await audioCtx.audioWorklet.addModule("/audio-worklet.js");
2024-03-14 06:49:35 +01:00
const audioWorklet = new AudioWorkletNode(audioCtx, "bytebeatProcessor", {
outputChannelCount: [2],
});
audioWorklet.port.addEventListener("message", e => handleMessage(e));
audioWorklet.port.start();
audioWorklet.connect(audioGain);
console.info(
`started audio with latency ${audioCtx.baseLatency * audioCtx.sampleRate} at ${
audioCtx.sampleRate
}Hz`,
);
2024-03-14 06:49:35 +01:00
return { audioCtx, audioGain, audioWorklet };
}
function refresh() {
if (audioWorklet) {
audioWorklet.port.postMessage({ code: codeEditor.getCode().trim() });
}
}
2024-08-03 05:55:27 +02:00
// TODO: bug, this calls refresh twice every time because
// setting codemirror programmatically also calls refresh
// the code text input might also have this issue?
export function setSong(songData?: StrongPartial<Song>, play = true) {
let code, sampleRate, mode;
if (songData) {
({ code, sampleRate, mode } = songData);
codeEditor.setCode(code);
}
setSampleRate(sampleRate ?? 8000);
setSongMode(mode ?? "Bytebeat");
refresh();
if (play) {
resetTime();
setPlaySpeed(1);
}
}
/// call refresh after calling this
function setSampleRate(sampleRate: number) {
if (audioWorklet) {
const rate = Math.min(Math.max(Math.fround(sampleRate), 2), 2 ** 24 - 1);
// implicit cast
elements.sampleRate.value = f32ToRound10(rate) as unknown as string;
songData.sampleRate = rate;
// TODO if funcbeat change time so that sec stays the same
audioWorklet.port.postMessage({ songData, updateSampleRatio: true });
osc.showTimeCursor(timeCursorShouldBeVisible());
}
}
/// call refresh after calling this
function setSongMode(mode: SongMode) {
if (audioWorklet) {
elements.playbackMode.value = mode;
songData.mode = mode;
audioWorklet.port.postMessage({ songData });
}
}
export function getSong(): Song {
return { code: codeEditor.getCode(), ...songData };
}
2024-03-14 06:49:35 +01:00
function handleMessage(e: MessageEvent<any>) {
if (isPlainObject(e.data)) {
const data = e.data;
if (data.clearCanvas) {
2024-03-14 06:04:54 +01:00
osc.clear();
} else if (data.clearDrawBuffer) {
2024-03-14 06:04:54 +01:00
osc.clearBuffer();
}
2024-08-03 07:26:54 +02:00
if (typeof data.sample === "number") {
setByteSample(data.sample, false);
}
if (Array.isArray(data.drawBuffer)) {
2024-08-03 07:26:54 +02:00
osc.addToBuffer(data.drawBuffer, sample);
2023-12-25 23:40:54 +01:00
}
2024-03-11 18:25:55 +01:00
if (data.updateUrl && globalThis.useUrlData !== false) {
setUrlData(getSong());
2023-12-25 23:40:54 +01:00
}
if (data.errorMessage !== undefined) {
if (isPlainObject(data.errorMessage)) {
2023-12-25 23:40:54 +01:00
if (
typeof data.errorMessage.type === "string" &&
typeof data.errorMessage.err === "string" &&
typeof (data.errorMessage.priority ?? 0) === "number"
2023-12-25 23:40:54 +01:00
) {
if (playSpeed) {
nextErrType = data.errorMessage.type;
nextErr = data.errorMessage.err;
nextErrPriority = data.errorMessage.priority;
} else {
showErrorMessage(
data.errorMessage.type,
data.errorMessage.err,
data.errorMessage.priority,
);
2021-11-18 14:36:00 +01:00
}
2023-12-25 23:40:54 +01:00
}
} else {
hideErrorMessage();
2021-11-18 14:36:00 +01:00
}
}
}
}
function timeCursorShouldBeVisible(): boolean {
return songData.sampleRate >> osc.settings.scale < 3950;
}
function changeScale(amount: number) {
if (amount) {
osc.settings.scale = Math.max(osc.settings.scale + amount, 0);
osc.clear(false);
if (osc.settings.scale <= 0) {
elements.scaleDown.setAttribute("disabled", true);
} else {
elements.scaleDown.removeAttribute("disabled");
}
2024-03-14 06:49:35 +01:00
osc.showTimeCursor(timeCursorShouldBeVisible());
2024-08-03 07:26:54 +02:00
osc.moveTimeCursor(sample, playSpeed >= 0);
saveSettings();
}
}
function showErrorMessage(errType, err, priority = 0) {
2023-12-28 21:34:32 +01:00
if (elements.error && (errorPriority < 2 || priority > 0)) {
elements.error.dataset.errType = errType;
elements.error.innerText = err.toString();
nextErr = null;
nextErrType = null;
nextErrPriority = undefined;
errorPriority = priority;
if (audioWorklet) {
audioWorklet.port.postMessage({ displayedError: true });
}
}
}
function hideErrorMessage() {
if (elements.error) {
elements.error.innerText = "";
nextErr = null;
nextErrType = null;
nextErrPriority = undefined;
errorPriority = -Infinity;
}
}
function resetTime() {
setByteSample(0, true, true);
if (!playSpeed) {
2023-12-28 21:34:32 +01:00
elements.canvasTogglePlay.classList.add("canvas-toggleplay-show");
}
}
/// TODO document
2024-04-23 23:16:19 +02:00
function setByteSample(value: number, send = true, clear = false) {
if (audioWorklet && isFinite(value)) {
2024-08-03 07:26:54 +02:00
sample = value;
updateCounterValue();
if (send) {
audioWorklet.port.postMessage({ setByteSample: [value, clear] });
}
2024-08-03 07:26:54 +02:00
osc.moveTimeCursor(sample, playSpeed >= 0);
}
}
function setPlaySpeed(value: number) {
if (audioWorklet && value !== playSpeed) {
elements.canvasTogglePlay.classList.toggle("canvas-toggleplay-pause", !!value);
if (value) {
elements.canvasTogglePlay.classList.remove("canvas-toggleplay-show");
audioCtx.resume();
startAnimation();
} else {
//audioCtx.suspend();
stopAnimation();
}
playSpeed = value;
audioWorklet.port.postMessage({ playSpeed, updateSampleRatio: true });
}
}
function togglePlay() {
setPlaySpeed(+!playSpeed);
}
{
let request = undefined as unknown as number;
function animationFrame() {
2024-08-03 07:26:54 +02:00
osc.draw(sample, playSpeed >= 0);
osc.moveTimeCursor(sample, playSpeed >= 0);
if (nextErr) {
showErrorMessage(nextErrType, nextErr, nextErrPriority);
}
startAnimation();
}
function startAnimation() {
request = requestAnimationFrame(() => animationFrame());
}
function stopAnimation() {
animationFrame();
cancelAnimationFrame(request);
}
}
function updateCounterValue() {
2024-04-23 23:16:19 +02:00
// implicit cast
2024-08-03 07:26:54 +02:00
elements.timeValue.placeholder = convertToUnit(sample) as unknown as string;
}
2024-04-23 23:16:19 +02:00
function convertFromUnit(value: number, unit = timeUnit) {
switch (unit) {
case "t":
return value;
case "s":
return value * songData.sampleRate;
}
}
// IMPORTANT: this function is ONLY used for text formatting, does not work for many conversions, and is inaccurate.
2024-03-14 06:04:54 +01:00
function convertToUnit(value: number, unit = timeUnit) {
switch (unit) {
case "t":
return value;
case "s":
return (value / songData.sampleRate).toFixed(3);
}
}
2024-04-23 23:16:19 +02:00
function setTimeUnit(value?: number | TimeUnit, userInput = true) {
if (value !== undefined) {
if (typeof value === "number") {
timeUnit = timeUnits[value];
} else {
timeUnit = value;
}
2023-12-28 21:34:32 +01:00
elements.timeUnitLabel.innerText = timeUnit;
} else {
2023-12-28 21:34:32 +01:00
timeUnit = elements.timeUnitLabel.innerText;
}
if (userInput) {
updateCounterValue();
saveSettings();
}
}
function changeTimeUnit() {
timeUnit = timeUnits[(timeUnits.indexOf(timeUnit) + 1) % timeUnits.length];
2023-12-28 21:34:32 +01:00
elements.timeUnitLabel.innerText = timeUnit;
updateCounterValue();
saveSettings();
}
2024-03-14 06:04:54 +01:00
function setSampleRateDivisor(sampleRateDivisor: number) {
if (audioWorklet && sampleRateDivisor > 0) {
audioWorklet.port.postMessage({ sampleRateDivisor, updateSampleRatio: true });
}
}
2024-03-14 06:04:54 +01:00
function setVolume(save = true, value?: number) {
if (value !== undefined) {
volume = value;
2024-04-23 23:16:19 +02:00
// implicit cast
elements.volume.value = volume as unknown as string;
2024-03-14 06:04:54 +01:00
}
volume = elements.volume.valueAsNumber;
2024-08-03 05:55:27 +02:00
audioGain.gain.value = volume ** 2;
2024-03-14 06:04:54 +01:00
if (save) {
saveSettings();
}
}
2024-03-13 23:11:33 +01:00
type Settings = {
2024-03-14 06:49:35 +01:00
drawSettings: typeof osc.settings;
volume: number;
timeUnit: TimeUnit;
};
2024-03-14 06:04:54 +01:00
function loadDefaultSettings() {
setVolume(false);
setTimeUnit(undefined, false);
}
2024-03-13 23:11:33 +01:00
function loadSettings(): Settings {
2024-04-23 23:16:19 +02:00
if (globalThis.useLocalStorage !== false && localStorage["settings"]) {
2024-03-13 23:11:33 +01:00
let settings: {
2024-04-23 23:16:19 +02:00
drawSettings?: typeof osc.settings;
volume?: typeof volume;
timeUnit?: typeof timeUnit;
2024-03-13 23:11:33 +01:00
};
try {
2024-04-23 23:16:19 +02:00
// FIXME: this should be validated
settings = JSON.parse(localStorage["settings"]);
} catch (err) {
2024-04-23 23:16:19 +02:00
console.error("Couldn't load settings!", localStorage["settings"]);
localStorage.clear();
loadDefaultSettings();
return;
}
2024-04-23 23:16:19 +02:00
if (settings.drawSettings) {
2024-03-14 06:04:54 +01:00
osc.settings = settings.drawSettings;
} else {
2024-03-14 06:04:54 +01:00
osc.settings.scale = 5;
}
2024-03-13 23:11:33 +01:00
setVolume(false, settings.volume);
2024-04-23 23:16:19 +02:00
if (settings.timeUnit) {
setTimeUnit(settings.timeUnit, false);
} else {
setTimeUnit(undefined, false);
}
} else {
loadDefaultSettings();
}
}
2024-03-14 06:04:54 +01:00
function saveSettings() {
if (globalThis.useLocalStorage !== false) {
2024-04-23 23:16:19 +02:00
localStorage["settings"] = JSON.stringify({
2024-03-14 06:04:54 +01:00
drawSettings: osc.settings,
volume,
timeUnit,
});
}
}
// used in html
Object.defineProperty(globalThis, "bytebeat", {
value: {
changeTimeUnit,
setByteSample,
convertFromUnit,
togglePlay,
resetTime,
setPlaySpeed,
setVolume,
changeScale,
setSongMode,
setSampleRate,
setSampleRateDivisor,
refresh,
},
});
2024-08-03 05:55:27 +02:00
// TODO: move this to a different file
autoSizeCanvas(true);
addEventListener("resize", () => autoSizeCanvas());
export function autoSizeCanvas(force = false) {
if (!elements.canvas.dataset["forcedWidth"]) {
// 768 is halfway between 512 and 1024, 3 added for outline
if (innerWidth >= 768 + 3) {
let width = 1024;
while (innerWidth - 516 >= width * 2) {
// 516px = 4px (outline) + 512px (library)
width *= 2;
}
setCanvasWidth(width, force);
} else {
setCanvasWidth(512, force);
}
}
}
export function setCanvasWidth(width: number, force = false) {
if (elements.canvas) {
if (width !== elements.canvas.width || force) {
elements.canvas.width = width;
// TODO: see if it's possible to get rid of this
elements.main.style.maxWidth = `${width + 4}px`;
}
}
}