mirror of
https://github.com/eliasstepanik/strudel.git
synced 2026-01-11 21:58:37 +00:00
136 lines
4.9 KiB
JavaScript
136 lines
4.9 KiB
JavaScript
/*
|
|
midi.mjs - <short description TODO>
|
|
Copyright (C) 2022 Strudel contributors - see <https://github.com/tidalcycles/strudel/blob/main/packages/midi/midi.mjs>
|
|
This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import * as _WebMidi from 'webmidi';
|
|
import { Pattern, isPattern, logger } from '@strudel.cycles/core';
|
|
import { getAudioContext } from '@strudel.cycles/webaudio';
|
|
import { noteToMidi } from '@strudel.cycles/core';
|
|
|
|
// if you use WebMidi from outside of this package, make sure to import that instance:
|
|
export const { WebMidi } = _WebMidi;
|
|
|
|
function supportsMidi() {
|
|
return typeof navigator.requestMIDIAccess === 'function';
|
|
}
|
|
|
|
export function enableWebMidi(options = {}) {
|
|
const { onReady, onConnected, onDisconnected } = options;
|
|
|
|
if (!supportsMidi()) {
|
|
throw new Error('Your Browser does not support WebMIDI.');
|
|
}
|
|
return new Promise((resolve, reject) => {
|
|
if (WebMidi.enabled) {
|
|
// if already enabled, just resolve WebMidi
|
|
resolve(WebMidi);
|
|
return;
|
|
}
|
|
WebMidi.enable((err) => {
|
|
if (err) {
|
|
reject(err);
|
|
}
|
|
WebMidi.addListener('connected', (e) => {
|
|
onConnected?.(WebMidi);
|
|
});
|
|
// Reacting when a device becomes unavailable
|
|
WebMidi.addListener('disconnected', (e) => {
|
|
onDisconnected?.(WebMidi, e);
|
|
});
|
|
onReady?.(WebMidi);
|
|
resolve(WebMidi);
|
|
});
|
|
});
|
|
}
|
|
// const outputByName = (name: string) => WebMidi.getOutputByName(name);
|
|
const outputByName = (name) => WebMidi.getOutputByName(name);
|
|
|
|
let midiReady;
|
|
|
|
// output?: string | number, outputs: typeof WebMidi.outputs
|
|
function getDevice(output, outputs) {
|
|
if (!outputs.length) {
|
|
throw new Error(`🔌 No MIDI devices found. Connect a device or enable IAC Driver.`);
|
|
}
|
|
if (typeof output === 'number') {
|
|
return outputs[output];
|
|
}
|
|
if (typeof output === 'string') {
|
|
return outputByName(output);
|
|
}
|
|
return outputs[0];
|
|
}
|
|
|
|
// Pattern.prototype.midi = function (output: string | number, channel = 1) {
|
|
Pattern.prototype.midi = function (output) {
|
|
if (!supportsMidi()) {
|
|
throw new Error(`🎹 WebMidi is not enabled. Supported Browsers: https://caniuse.com/?search=webmidi`);
|
|
}
|
|
/* await */ enableWebMidi({
|
|
onConnected: ({ outputs }) =>
|
|
logger(`Midi device connected! Available: ${outputs.map((o) => `'${o.name}'`).join(', ')}`),
|
|
onDisconnected: ({ outputs }) =>
|
|
logger(`Midi device disconnected! Available: ${outputs.map((o) => `'${o.name}'`).join(', ')}`),
|
|
onReady: ({ outputs }) => {
|
|
const device = getDevice(output, outputs);
|
|
const otherOutputs = outputs
|
|
.filter((o) => o.name !== device.name)
|
|
.map((o) => `'${o.name}'`)
|
|
.join(' | ');
|
|
midiReady = true;
|
|
logger(`Midi connected! Using "${device.name}". ${otherOutputs ? `Also available: ${otherOutputs}` : ''}`);
|
|
},
|
|
});
|
|
if (isPattern(output)) {
|
|
throw new Error(
|
|
`.midi does not accept Pattern input. Make sure to pass device name with single quotes. Example: .midi('${
|
|
WebMidi.outputs?.[0]?.name || 'IAC Driver Bus 1'
|
|
}')`,
|
|
);
|
|
}
|
|
return this.onTrigger((time, hap) => {
|
|
if (!midiReady) {
|
|
return;
|
|
}
|
|
const device = getDevice(output, WebMidi.outputs);
|
|
if (!device) {
|
|
throw new Error(
|
|
`🔌 MIDI device '${output ? output : ''}' not found. Use one of ${WebMidi.outputs
|
|
.map((o) => `'${o.name}'`)
|
|
.join(' | ')}`,
|
|
);
|
|
}
|
|
hap.ensureObjectValue();
|
|
|
|
// calculate time
|
|
const timingOffset = WebMidi.time - getAudioContext().getOutputTimestamp().contextTime * 1000;
|
|
time = time * 1000 + timingOffset;
|
|
|
|
// destructure value
|
|
const { note, nrpnn, nrpv, ccn, ccv, midichan = 1 } = hap.value;
|
|
const velocity = hap.context?.velocity ?? 0.9; // TODO: refactor velocity
|
|
const duration = hap.duration.valueOf() * 1000 - 5;
|
|
|
|
if (note) {
|
|
const midiNumber = noteToMidi(note);
|
|
device.playNote(midiNumber, midichan, {
|
|
time,
|
|
duration,
|
|
attack: velocity,
|
|
});
|
|
}
|
|
if (ccv && ccn) {
|
|
if (typeof ccv !== 'number' || ccv < 0 || ccv > 1) {
|
|
throw new Error('expected ccv to be a number between 0 and 1');
|
|
}
|
|
if (!['string', 'number'].includes(typeof ccn)) {
|
|
throw new Error('expected ccn to be a number or a string');
|
|
}
|
|
const scaled = Math.round(ccv * 127);
|
|
device.sendControlChange(ccn, scaled, midichan, { time });
|
|
}
|
|
});
|
|
};
|