UNPKG

@strudel/midi

Version:

Midi API for strudel

512 lines (460 loc) 18.1 kB
/* midi.mjs - <short description TODO> Copyright (C) 2022 Strudel contributors - see <https://codeberg.org/uzu/strudel/src/branch/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, getEventOffsetMs, isPattern, logger, ref } from '@strudel/core'; import { noteToMidi, getControlName } from '@strudel/core'; import { Note } from 'webmidi'; // 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'; } function getMidiDeviceNamesString(devices) { return devices.map((o) => `'${o.name}'`).join(' | '); } export function enableWebMidi(options = {}) { const { onReady, onConnected, onDisconnected, onEnabled } = options; if (WebMidi.enabled) { return; } if (!supportsMidi()) { throw new Error('Your Browser does not support WebMIDI.'); } WebMidi.addListener('connected', () => { onConnected?.(WebMidi); }); WebMidi.addListener('enabled', () => { onEnabled?.(WebMidi); }); // Reacting when a device becomes unavailable WebMidi.addListener('disconnected', (e) => { onDisconnected?.(WebMidi, e); }); return new Promise((resolve, reject) => { if (WebMidi.enabled) { // if already enabled, just resolve WebMidi resolve(WebMidi); return; } WebMidi.enable( (err) => { if (err) { reject(err); } onReady?.(WebMidi); resolve(WebMidi); }, { sysex: true }, ); }); } function getDevice(indexOrName, devices) { if (!devices.length) { throw new Error(`🔌 No MIDI devices found. Connect a device or enable IAC Driver.`); } if (typeof indexOrName === 'number') { return devices[indexOrName]; } const byName = (name) => devices.find((output) => output.name.includes(name)); if (typeof indexOrName === 'string') { return byName(indexOrName); } // attempt to default to first IAC device if none is specified const IACOutput = byName('IAC'); const device = IACOutput ?? devices[0]; if (!device) { throw new Error( `🔌 MIDI device '${device ? device : ''}' not found. Use one of ${getMidiDeviceNamesString(devices)}`, ); } return IACOutput ?? devices[0]; } // send start/stop messages to outputs when repl starts/stops if (typeof window !== 'undefined') { window.addEventListener('message', (e) => { if (!WebMidi?.enabled) { return; } if (e.data === 'strudel-stop') { WebMidi.outputs.forEach((output) => output.sendStop()); } // cannot start here, since we have no timing info, see sendStart below }); } // registry for midi mappings, converting control names to cc messages export const midicontrolMap = new Map(); // takes midimap and converts each control key to the main control name function unifyMapping(mapping) { return Object.fromEntries( Object.entries(mapping).map(([key, mapping]) => { if (typeof mapping === 'number') { mapping = { ccn: mapping }; } return [getControlName(key), mapping]; }), ); } function githubPath(base, subpath = '') { if (!base.startsWith('github:')) { throw new Error('expected "github:" at the start of pseudoUrl'); } let [_, path] = base.split('github:'); path = path.endsWith('/') ? path.slice(0, -1) : path; if (path.split('/').length === 2) { // assume main as default branch if none set path += '/main'; } return `https://raw.githubusercontent.com/${path}/${subpath}`; } /** * configures the default midimap, which is used when no "midimap" port is set * @example * defaultmidimap({ lpf: 74 }) * $: note("c a f e").midi(); * $: lpf(sine.slow(4).segment(16)).midi(); */ export function defaultmidimap(mapping) { midicontrolMap.set('default', unifyMapping(mapping)); } let loadCache = {}; /** * Adds midimaps to the registry. Inside each midimap, control names (e.g. lpf) are mapped to cc numbers. * @example * midimaps({ mymap: { lpf: 74 } }) * $: note("c a f e") * .lpf(sine.slow(4)) * .midimap('mymap') * .midi() * @example * midimaps({ mymap: { * lpf: { ccn: 74, min: 0, max: 20000, exp: 0.5 } * }}) * $: note("c a f e") * .lpf(sine.slow(2).range(400,2000)) * .midimap('mymap') * .midi() */ export async function midimaps(map) { if (typeof map === 'string') { if (map.startsWith('github:')) { map = githubPath(map, 'midimap.json'); } if (!loadCache[map]) { loadCache[map] = fetch(map).then((res) => res.json()); } map = await loadCache[map]; } if (typeof map === 'object') { Object.entries(map).forEach(([name, mapping]) => midicontrolMap.set(name, unifyMapping(mapping))); } } // registry for midi sounds, converting sound names to controls export const midisoundMap = new Map(); // normalizes the given value from the given range and exponent function normalize(value = 0, min = 0, max = 1, exp = 1) { if (min === max) { throw new Error('min and max cannot be the same value'); } let normalized = (value - min) / (max - min); normalized = Math.min(1, Math.max(0, normalized)); return Math.pow(normalized, exp); } function mapCC(mapping, value) { return Object.keys(value) .filter((key) => !!mapping[getControlName(key)]) .map((key) => { const { ccn, min = 0, max = 1, exp = 1 } = mapping[key]; const ccv = normalize(value[key], min, max, exp); return { ccn, ccv }; }); } // sends a cc message to the given device on the given channel function sendCC(ccn, ccv, device, midichan, timeOffsetString) { 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: timeOffsetString }); } // sends a program change message to the given device on the given channel function sendProgramChange(progNum, device, midichan, timeOffsetString) { if (typeof progNum !== 'number' || progNum < 0 || progNum > 127) { throw new Error('expected progNum (program change) to be a number between 0 and 127'); } device.sendProgramChange(progNum, midichan, { time: timeOffsetString }); } // sends a sysex message to the given device on the given channel function sendSysex(sysexid, sysexdata, device, timeOffsetString) { if (Array.isArray(sysexid)) { if (!sysexid.every((byte) => Number.isInteger(byte) && byte >= 0 && byte <= 255)) { throw new Error('all sysexid bytes must be integers between 0 and 255'); } } else if (!Number.isInteger(sysexid) || sysexid < 0 || sysexid > 255) { throw new Error('A:sysexid must be an number between 0 and 255 or an array of such integers'); } if (!Array.isArray(sysexdata)) { throw new Error('expected sysex to be an array of numbers (0-255)'); } if (!sysexdata.every((byte) => Number.isInteger(byte) && byte >= 0 && byte <= 255)) { throw new Error('all sysex bytes must be integers between 0 and 255'); } device.sendSysex(sysexid, sysexdata, { time: timeOffsetString }); } // sends a NRPN message to the given device on the given channel function sendNRPN(nrpnn, nrpv, device, midichan, timeOffsetString) { if (Array.isArray(nrpnn)) { if (!nrpnn.every((byte) => Number.isInteger(byte) && byte >= 0 && byte <= 255)) { throw new Error('all nrpnn bytes must be integers between 0 and 255'); } } else if (!Number.isInteger(nrpv) || nrpv < 0 || nrpv > 255) { throw new Error('A:sysexid must be an number between 0 and 255 or an array of such integers'); } device.sendNRPN(nrpnn, nrpv, midichan, { time: timeOffsetString }); } // sends a pitch bend message to the given device on the given channel function sendPitchBend(midibend, device, midichan, timeOffsetString) { if (typeof midibend !== 'number' || midibend < -1 || midibend > 1) { throw new Error('expected midibend to be a number between -1 and 1'); } device.sendPitchBend(midibend, midichan, { time: timeOffsetString }); } // sends a channel aftertouch message to the given device on the given channel function sendAftertouch(miditouch, device, midichan, timeOffsetString) { if (typeof miditouch !== 'number' || miditouch < 0 || miditouch > 1) { throw new Error('expected miditouch to be a number between 0 and 1'); } device.sendChannelAftertouch(miditouch, midichan, { time: timeOffsetString }); } // sends a note message to the given device on the given channel function sendNote(note, velocity, duration, device, midichan, timeOffsetString) { if (note == null || note === '') { throw new Error('note cannot be null or empty'); } if (velocity != null && (typeof velocity !== 'number' || velocity < 0 || velocity > 1)) { throw new Error('velocity must be a number between 0 and 1'); } if (duration != null && (typeof duration !== 'number' || duration < 0)) { throw new Error('duration must be a positive number'); } const midiNumber = typeof note === 'number' ? note : noteToMidi(note); const midiNote = new Note(midiNumber, { attack: velocity, duration }); device.playNote(midiNote, midichan, { time: timeOffsetString, }); } /** * MIDI output: Opens a MIDI output port. * @param {string | number} midiport MIDI device name or index defaulting to 0 * @param {object} options Additional MIDI configuration options * @example * note("c4").midichan(1).midi('IAC Driver Bus 1') * @example * note("c4").midichan(1).midi('IAC Driver Bus 1', { controller: true, latency: 50 }) */ Pattern.prototype.midi = function (midiport, options = {}) { if (isPattern(midiport)) { throw new Error( `.midi does not accept Pattern input for midiport. Make sure to pass device name with single quotes. Example: .midi('${ WebMidi.outputs?.[0]?.name || 'IAC Driver Bus 1' }')`, ); } // For backward compatibility if (typeof midiport === 'object') { const { port, isController = false, ...configOptions } = midiport; options = { isController, ...configOptions, ...options, // Keep any options passed separately }; midiport = port; } let midiConfig = { // Default configuration values isController: false, // Disable sending notes for midi controllers latencyMs: 34, // Default latency to get audio engine to line up in ms noteOffsetMs: 10, // Default note-off offset to prevent glitching in ms midichannel: 1, // Default MIDI channel velocity: 0.9, // Default velocity gain: 1, // Default gain midimap: 'default', // Default MIDI map midiport: midiport, // Store the port in the config ...options, // Override defaults with provided options }; enableWebMidi({ onEnabled: ({ outputs }) => { const device = getDevice(midiConfig.midiport, outputs); const otherOutputs = outputs.filter((o) => o.name !== device.name); logger( `Midi enabled! Using "${device.name}". ${ otherOutputs?.length ? `Also available: ${getMidiDeviceNamesString(otherOutputs)}` : '' }`, ); }, onDisconnected: ({ outputs }) => logger(`Midi device disconnected! Available: ${getMidiDeviceNamesString(outputs)}`), }); return this.onTrigger((hap, currentTime, cps, targetTime) => { if (!WebMidi.enabled) { logger('Midi not enabled'); return; } hap.ensureObjectValue(); //magic number to get audio engine to line up, can probably be calculated somehow const latencyMs = midiConfig.latencyMs; // passing a string with a +num into the webmidi api adds an offset to the current time https://webmidijs.org/api/classes/Output const timeOffsetString = `+${getEventOffsetMs(targetTime, currentTime) + latencyMs}`; // midi event values from hap with configurable defaults let { note, nrpnn, nrpv, ccn, ccv, midichan = midiConfig.midichannel, midicmd, midibend, miditouch, polyTouch, gain = midiConfig.gain, velocity = midiConfig.velocity, progNum, sysexid, sysexdata, midimap = midiConfig.midimap, midiport = midiConfig.midiport, } = hap.value; const device = getDevice(midiport, WebMidi.outputs); if (!device) { logger( `[midi] midiport "${midiport}" not found! available: ${WebMidi.outputs.map((output) => `'${output.name}'`).join(', ')}`, ); return; } velocity = gain * velocity; // Handle midimap // if midimap is set, send a cc messages from defined controls if (midicontrolMap.has(midimap)) { const ccs = mapCC(midicontrolMap.get(midimap), hap.value); ccs.forEach(({ ccn, ccv }) => sendCC(ccn, ccv, device, midichan, timeOffsetString)); } else if (midimap !== 'default') { // Add warning when a non-existent midimap is specified logger(`[midi] midimap "${midimap}" not found! Available maps: ${[...midicontrolMap.keys()].join(', ')}`); } // Handle note if (note !== undefined && !midiConfig.isController) { // note off messages will often a few ms arrive late, // try to prevent glitching by subtracting noteOffsetMs from the duration length const duration = (hap.duration.valueOf() / cps) * 1000 - midiConfig.noteOffsetMs; sendNote(note, velocity, duration, device, midichan, timeOffsetString); } // Handle program change if (progNum !== undefined) { sendProgramChange(progNum, device, midichan, timeOffsetString); } // Handle sysex // sysex data is consist of 2 arrays, first is sysexid, second is sysexdata // sysexid is a manufacturer id it is either a number or an array of 3 numbers. // list of manufacturer ids can be found here : https://midi.org/sysexidtable // if sysexid is an array the first byte is 0x00 if (sysexid !== undefined && sysexdata !== undefined) { sendSysex(sysexid, sysexdata, device, timeOffsetString); } // Handle control change if (ccv !== undefined && ccn !== undefined) { sendCC(ccn, ccv, device, midichan, timeOffsetString); } // Handle NRPN non-registered parameter number if (nrpnn !== undefined && nrpv !== undefined) { sendNRPN(nrpnn, nrpv, device, midichan, timeOffsetString); } // Handle midibend if (midibend !== undefined) { sendPitchBend(midibend, device, midichan, timeOffsetString); } // Handle miditouch if (miditouch !== undefined) { sendAftertouch(miditouch, device, midichan, timeOffsetString); } // Handle midicmd if (hap.whole.begin + 0 === 0) { // we need to start here because we have the timing info device.sendStart({ time: timeOffsetString }); } if (['clock', 'midiClock'].includes(midicmd)) { device.sendClock({ time: timeOffsetString }); } else if (['start'].includes(midicmd)) { device.sendStart({ time: timeOffsetString }); } else if (['stop'].includes(midicmd)) { device.sendStop({ time: timeOffsetString }); } else if (['continue'].includes(midicmd)) { device.sendContinue({ time: timeOffsetString }); } else if (Array.isArray(midicmd)) { if (midicmd[0] === 'progNum') { sendProgramChange(midicmd[1], device, midichan, timeOffsetString); } else if (midicmd[0] === 'cc') { if (midicmd.length === 2) { sendCC(midicmd[0], midicmd[1] / 127, device, midichan, timeOffsetString); } } else if (midicmd[0] === 'sysex') { if (midicmd.length === 3) { const [_, id, data] = midicmd; sendSysex(id, data, device, timeOffsetString); } } } }); }; let listeners = {}; const refs = {}; /** * MIDI input: Opens a MIDI input port to receive MIDI control change messages. * @param {string | number} input MIDI device name or index defaulting to 0 * @returns {Function} * @example * let cc = await midin('IAC Driver Bus 1') * note("c a f e").lpf(cc(0).range(0, 1000)).lpq(cc(1).range(0, 10)).sound("sawtooth") */ export async function midin(input) { if (isPattern(input)) { throw new Error( `midin: does not accept Pattern as input. Make sure to pass device name with single quotes. Example: midin('${ WebMidi.outputs?.[0]?.name || 'IAC Driver Bus 1' }')`, ); } const initial = await enableWebMidi(); // only returns on first init const device = getDevice(input, WebMidi.inputs); if (!device) { throw new Error( `midiin: device "${input}" not found.. connected devices: ${getMidiDeviceNamesString(WebMidi.inputs)}`, ); } if (initial) { const otherInputs = WebMidi.inputs.filter((o) => o.name !== device.name); logger( `Midi enabled! Using "${device.name}". ${ otherInputs?.length ? `Also available: ${getMidiDeviceNamesString(otherInputs)}` : '' }`, ); } // ensure refs for this input are initialized if (!refs[input]) { refs[input] = {}; } const cc = (cc) => ref(() => refs[input][cc] || 0); listeners[input] && device.removeListener('midimessage', listeners[input]); listeners[input] = (e) => { const cc = e.dataBytes[0]; const v = e.dataBytes[1]; refs[input] && (refs[input][cc] = v / 127); }; device.addListener('midimessage', listeners[input]); return cc; }