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];
|
|
|
|
|
2024-08-03 05:28:53 +02:00
|
|
|
// XXX: global variables
|
|
|
|
let timeUnit: TimeUnit;
|
|
|
|
let volume: number;
|
2024-08-03 07:26:54 +02:00
|
|
|
let sample = 0;
|
2024-08-03 05:28:53 +02:00
|
|
|
let playSpeed = 0;
|
|
|
|
let songData: {
|
|
|
|
sampleRate: number;
|
|
|
|
mode: SongMode;
|
|
|
|
} = {
|
|
|
|
sampleRate: 8000, // float32
|
|
|
|
mode: "Bytebeat",
|
|
|
|
};
|
|
|
|
|
2022-04-29 00:23:07 +02:00
|
|
|
const searchParams = new URLSearchParams(location.search);
|
|
|
|
|
2024-03-14 06:49:35 +01:00
|
|
|
const audioPromise = initAudioContext();
|
2022-01-10 14:06:18 +01:00
|
|
|
|
2024-08-02 05:07:45 +02:00
|
|
|
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");
|
|
|
|
}
|
2023-12-28 20:07:07 +01:00
|
|
|
|
2024-03-14 06:49:35 +01:00
|
|
|
const { audioCtx, audioGain, audioWorklet } = await audioPromise;
|
|
|
|
|
|
|
|
loadSettings();
|
|
|
|
|
2024-08-03 05:28:53 +02:00
|
|
|
{
|
|
|
|
let urlData: Song | undefined;
|
|
|
|
if (location.hash && globalThis.useUrlData !== false) {
|
|
|
|
urlData = await fromUrlData(location.hash);
|
|
|
|
}
|
2023-12-28 21:34:32 +01:00
|
|
|
|
2024-08-03 05:28:53 +02:00
|
|
|
setSong(urlData, false);
|
|
|
|
updateCounterValue();
|
|
|
|
}
|
2023-12-28 21:34:32 +01:00
|
|
|
|
2024-08-03 05:28:53 +02:00
|
|
|
// XXX: global variables
|
2024-03-14 06:49:35 +01:00
|
|
|
let nextErrType = null;
|
|
|
|
let nextErr = null;
|
|
|
|
let nextErrPriority = undefined;
|
|
|
|
let errorPriority = -Infinity;
|
2023-12-28 20:07:07 +01:00
|
|
|
|
|
|
|
async function initAudioContext() {
|
2024-03-13 23:11:33 +01:00
|
|
|
let audioContextSampleRate = Number(searchParams.get("baseSampleRate"));
|
2024-08-03 05:28:53 +02:00
|
|
|
// also true for NaN
|
2023-12-28 20:07:07 +01:00
|
|
|
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",
|
2023-12-28 20:07:07 +01:00
|
|
|
sampleRate: audioContextSampleRate,
|
|
|
|
});
|
|
|
|
|
2024-03-14 06:49:35 +01:00
|
|
|
const audioGain = new GainNode(audioCtx);
|
2023-12-28 20:07:07 +01:00
|
|
|
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", {
|
2023-12-28 20:07:07 +01:00
|
|
|
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 };
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
2024-08-02 05:40:17 +02:00
|
|
|
|
|
|
|
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?
|
2024-08-02 05:07:45 +02:00
|
|
|
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();
|
2024-08-03 05:28:53 +02:00
|
|
|
setPlaySpeed(1);
|
2024-08-02 05:07:45 +02:00
|
|
|
}
|
|
|
|
}
|
2024-08-02 05:40:17 +02:00
|
|
|
/// call refresh after calling this
|
2024-08-02 05:07:45 +02:00
|
|
|
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());
|
|
|
|
}
|
|
|
|
}
|
2024-08-02 05:40:17 +02:00
|
|
|
/// call refresh after calling this
|
2024-08-02 05:07:45 +02:00
|
|
|
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>) {
|
2023-12-28 20:07:07 +01:00
|
|
|
if (isPlainObject(e.data)) {
|
|
|
|
const data = e.data;
|
|
|
|
if (data.clearCanvas) {
|
2024-03-14 06:04:54 +01:00
|
|
|
osc.clear();
|
2023-12-28 20:07:07 +01:00
|
|
|
} else if (data.clearDrawBuffer) {
|
2024-03-14 06:04:54 +01:00
|
|
|
osc.clearBuffer();
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
|
2024-08-03 07:26:54 +02:00
|
|
|
if (typeof data.sample === "number") {
|
|
|
|
setByteSample(data.sample, false);
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
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
|
|
|
}
|
2022-05-11 18:04:06 +02: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
|
|
|
}
|
2022-05-11 18:04:06 +02:00
|
|
|
|
2023-12-28 20:07:07 +01:00
|
|
|
if (data.errorMessage !== undefined) {
|
|
|
|
if (isPlainObject(data.errorMessage)) {
|
2023-12-25 23:40:54 +01:00
|
|
|
if (
|
2023-12-28 20:07:07 +01:00
|
|
|
typeof data.errorMessage.type === "string" &&
|
|
|
|
typeof data.errorMessage.err === "string" &&
|
|
|
|
typeof (data.errorMessage.priority ?? 0) === "number"
|
2023-12-25 23:40:54 +01:00
|
|
|
) {
|
2024-08-03 05:28:53 +02:00
|
|
|
if (playSpeed) {
|
2023-12-28 20:07:07 +01:00
|
|
|
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
|
|
|
}
|
2023-12-28 20:07:07 +01:00
|
|
|
} else {
|
|
|
|
hideErrorMessage();
|
2021-11-18 14:36:00 +01:00
|
|
|
}
|
2022-05-11 18:04:06 +02:00
|
|
|
}
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-02 05:07:45 +02: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
|
|
|
|
2024-08-02 05:07:45 +02:00
|
|
|
osc.showTimeCursor(timeCursorShouldBeVisible());
|
2024-08-03 07:26:54 +02:00
|
|
|
osc.moveTimeCursor(sample, playSpeed >= 0);
|
2024-08-02 05:07:45 +02:00
|
|
|
saveSettings();
|
|
|
|
}
|
|
|
|
}
|
2023-12-28 20:07:07 +01:00
|
|
|
|
|
|
|
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();
|
2023-12-28 20:07:07 +01:00
|
|
|
|
|
|
|
nextErr = null;
|
|
|
|
nextErrType = null;
|
|
|
|
nextErrPriority = undefined;
|
|
|
|
errorPriority = priority;
|
|
|
|
|
|
|
|
if (audioWorklet) {
|
|
|
|
audioWorklet.port.postMessage({ displayedError: true });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-08-02 05:07:45 +02:00
|
|
|
function hideErrorMessage() {
|
|
|
|
if (elements.error) {
|
|
|
|
elements.error.innerText = "";
|
|
|
|
|
|
|
|
nextErr = null;
|
|
|
|
nextErrType = null;
|
|
|
|
nextErrPriority = undefined;
|
|
|
|
errorPriority = -Infinity;
|
|
|
|
}
|
|
|
|
}
|
2023-12-28 20:07:07 +01:00
|
|
|
|
|
|
|
function resetTime() {
|
|
|
|
setByteSample(0, true, true);
|
2024-08-03 05:28:53 +02:00
|
|
|
if (!playSpeed) {
|
2023-12-28 21:34:32 +01:00
|
|
|
elements.canvasTogglePlay.classList.add("canvas-toggleplay-show");
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
}
|
2024-08-03 05:28:53 +02:00
|
|
|
/// TODO document
|
2024-04-23 23:16:19 +02:00
|
|
|
function setByteSample(value: number, send = true, clear = false) {
|
2023-12-28 20:07:07 +01:00
|
|
|
if (audioWorklet && isFinite(value)) {
|
2024-08-03 07:26:54 +02:00
|
|
|
sample = value;
|
2023-12-28 20:07:07 +01:00
|
|
|
updateCounterValue();
|
|
|
|
if (send) {
|
|
|
|
audioWorklet.port.postMessage({ setByteSample: [value, clear] });
|
|
|
|
}
|
2024-08-03 07:26:54 +02:00
|
|
|
osc.moveTimeCursor(sample, playSpeed >= 0);
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
}
|
2024-08-02 05:40:17 +02:00
|
|
|
|
2023-12-28 20:07:07 +01:00
|
|
|
function setPlaySpeed(value: number) {
|
|
|
|
if (audioWorklet && value !== playSpeed) {
|
2024-08-03 05:28:53 +02:00
|
|
|
elements.canvasTogglePlay.classList.toggle("canvas-toggleplay-pause", !!value);
|
|
|
|
if (value) {
|
|
|
|
elements.canvasTogglePlay.classList.remove("canvas-toggleplay-show");
|
|
|
|
audioCtx.resume();
|
|
|
|
startAnimation();
|
|
|
|
} else {
|
|
|
|
//audioCtx.suspend();
|
|
|
|
stopAnimation();
|
|
|
|
}
|
2023-12-28 20:07:07 +01:00
|
|
|
playSpeed = value;
|
|
|
|
audioWorklet.port.postMessage({ playSpeed, updateSampleRatio: true });
|
|
|
|
}
|
|
|
|
}
|
2024-08-03 05:28:53 +02:00
|
|
|
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);
|
2024-08-03 05:28:53 +02:00
|
|
|
if (nextErr) {
|
|
|
|
showErrorMessage(nextErrType, nextErr, nextErrPriority);
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
2024-08-03 05:28:53 +02:00
|
|
|
|
|
|
|
startAnimation();
|
|
|
|
}
|
|
|
|
function startAnimation() {
|
|
|
|
request = requestAnimationFrame(() => animationFrame());
|
|
|
|
}
|
|
|
|
function stopAnimation() {
|
|
|
|
animationFrame();
|
|
|
|
cancelAnimationFrame(request);
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
2024-04-23 23:16:19 +02:00
|
|
|
function convertFromUnit(value: number, unit = timeUnit) {
|
2023-12-28 20:07:07 +01:00
|
|
|
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) {
|
2023-12-28 20:07:07 +01:00
|
|
|
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) {
|
2023-12-28 20:07:07 +01:00
|
|
|
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;
|
2023-12-28 20:07:07 +01:00
|
|
|
} else {
|
2023-12-28 21:34:32 +01:00
|
|
|
timeUnit = elements.timeUnitLabel.innerText;
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
2023-12-28 20:07:07 +01:00
|
|
|
|
|
|
|
updateCounterValue();
|
|
|
|
saveSettings();
|
|
|
|
}
|
2024-03-14 06:04:54 +01:00
|
|
|
|
2024-08-02 05:40:17 +02: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();
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
|
|
|
}
|
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
|
|
|
};
|
2023-12-28 20:07:07 +01:00
|
|
|
try {
|
2024-04-23 23:16:19 +02:00
|
|
|
// FIXME: this should be validated
|
|
|
|
settings = JSON.parse(localStorage["settings"]);
|
2023-12-28 20:07:07 +01:00
|
|
|
} catch (err) {
|
2024-04-23 23:16:19 +02:00
|
|
|
console.error("Couldn't load settings!", localStorage["settings"]);
|
2023-12-28 20:07:07 +01:00
|
|
|
localStorage.clear();
|
|
|
|
loadDefaultSettings();
|
|
|
|
return;
|
2022-05-11 18:04:06 +02:00
|
|
|
}
|
|
|
|
|
2024-04-23 23:16:19 +02:00
|
|
|
if (settings.drawSettings) {
|
2024-03-14 06:04:54 +01:00
|
|
|
osc.settings = settings.drawSettings;
|
2023-12-28 20:07:07 +01:00
|
|
|
} else {
|
2024-03-14 06:04:54 +01:00
|
|
|
osc.settings.scale = 5;
|
2022-05-11 18:04:06 +02:00
|
|
|
}
|
|
|
|
|
2024-03-13 23:11:33 +01:00
|
|
|
setVolume(false, settings.volume);
|
2022-01-08 19:04:55 +01:00
|
|
|
|
2024-04-23 23:16:19 +02:00
|
|
|
if (settings.timeUnit) {
|
2023-12-28 20:07:07 +01:00
|
|
|
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,
|
|
|
|
});
|
|
|
|
}
|
2023-12-28 20:07:07 +01:00
|
|
|
}
|
2022-05-11 18:04:06 +02:00
|
|
|
|
2023-12-28 20:07:07 +01:00
|
|
|
// used in html
|
|
|
|
Object.defineProperty(globalThis, "bytebeat", {
|
|
|
|
value: {
|
|
|
|
changeTimeUnit,
|
|
|
|
setByteSample,
|
|
|
|
convertFromUnit,
|
|
|
|
togglePlay,
|
|
|
|
resetTime,
|
|
|
|
setPlaySpeed,
|
|
|
|
setVolume,
|
|
|
|
changeScale,
|
2024-08-02 05:07:45 +02:00
|
|
|
setSongMode,
|
2023-12-28 20:07:07 +01:00
|
|
|
setSampleRate,
|
|
|
|
setSampleRateDivisor,
|
2024-08-02 05:07:45 +02:00
|
|
|
refresh,
|
2022-05-11 18:04:06 +02:00
|
|
|
},
|
2023-12-28 20:07:07 +01:00
|
|
|
});
|
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`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|