User:Eufalesio/How to build edos in DAWs: Difference between revisions
No edit summary |
|||
| (2 intermediate revisions by the same user not shown) | |||
| Line 23: | Line 23: | ||
* Disadvantages: Only possible with 12n edos. Edos above 72 or 84 may become unwieldy or RAM-intensive, due to the abundance of many instrument instances. | * Disadvantages: Only possible with 12n edos. Edos above 72 or 84 may become unwieldy or RAM-intensive, due to the abundance of many instrument instances. | ||
Good edos: [[24edo|24]], [[72edo|72]], [[84edo|84]] - Reasoning: Efficient approximations of JI / 24edo is an entry-level microtonal edo, great [[2.3.11 subgroup| | Good edos: [[24edo|24]], [[72edo|72]], [[84edo|84]] - Reasoning: Efficient approximations of JI / 24edo is an entry-level microtonal edo, great [[2.3.11 subgroup|ila]] tuning. | ||
== Edos >12 == | == Edos >12 == | ||
| Line 43: | Line 43: | ||
* Advantages: Allows use of very fine-grained edos, 10.<u>3</u> octave range, practical to compose in. | * Advantages: Allows use of very fine-grained edos, 10.<u>3</u> octave range, practical to compose in. | ||
* Disadvantages: Harder to set up, less transposing-friendly because of wolf intervals. | * Disadvantages: Harder to set up, less transposing-friendly because of wolf intervals. Not usable live. | ||
Good edos: [[41edo|41]], [[53edo|53]], [[94edo|94]], [[130edo|130]], [[159edo|159]] - Reasoning: Astounding fifths and/or Great 13-limit approximations. | Good edos: [[41edo|41]], [[53edo|53]], [[94edo|94]], [[130edo|130]], [[159edo|159]] - Reasoning: Astounding fifths and/or Great 13-limit approximations. | ||
94edo is my favorite here, because it combines two top-tier edos into one great jack-of-all trades 23-odd-limit tuning. 159edo, [[Dawson Berry|Aura]]'s favorite, is also a top-tier edo, making an ''astonishingly good'' | 94edo is my favorite here, because it combines two top-tier edos into one great jack-of-all trades 23-odd-limit tuning. 159edo, [[Dawson Berry|Aura]]'s favorite, is also a top-tier edo, making an ''astonishingly good'' ila tuning, and/or an overall a very good jack-of-all trades tuning, both of them with a graspable gamut. | ||
=== Preset-scale deviations - MIDI and articulations === | === Preset-scale deviations - MIDI and articulations === | ||
| Line 55: | Line 55: | ||
* Advantages: Near total freedom of pitch and tuning options, allows use of extremely fine edos, 10.<u>3</u> octave range, practical to compose in. | * Advantages: Near total freedom of pitch and tuning options, allows use of extremely fine edos, 10.<u>3</u> octave range, practical to compose in. | ||
* Disadvantages: (Probably) Non-transposing friendly. Requires MPE or else instrument must be monophonic. Hard to setup, very technical, prone to error. | * Disadvantages: (Probably) Non-transposing friendly. Requires MPE or else instrument must be monophonic. Hard to setup, very technical, prone to error, cannot use articulations for their intended purpose. | ||
Good edos: [[217edo|217]], [[270edo|270]], [[311edo|311 | Good edos: [[217edo|217]], [[270edo|270]], [[311edo|311]] - Reasoning: High-prime-limit support / Unbeatably accurate yazalathana. | ||
Excessive edos: Anything beyond 311. Highly recommend you use superfine 12n edos for these ones. I don't go here. | |||
==== About edo usage and JI approximations (my experience) ==== | |||
[[User:Eufalesio/My temperament|Ultimate]] is my temperament of temperaments, which supports {{Edos|12e, 41, 53, 94, 217, 270, 311}}, and of those my best choices are undoubtedly 94edo and 270edo. They have respectively, very good and extremely good 13-limit, and are even, so I can split the [[Pythagorean comma|poma]] in halves for easier navigation. | |||
===== Vibe-coded Javascript script ===== | I aim for 13-limit JI, and if you also aim for that, then these tunings are the best for you. Check the Ultimate article for more info. Note, I don't care about essential tempering, VAOs, MOSes beyond the theoretical, or structural exploiting. | ||
<syntaxhighlight lang="javascript" line="1">// Generalized EDO | |||
// | If you still want to go even finer than that... then try 612edo and 2460; S-tier 12n edos, so you get transposing-friendly hyperfine microtonality. But, since their step sizes are smaller than the melodic JND, you can't tell notes edosteps apart. You are likely wasting your time here, however. | ||
// | If you STILL want to go finer... you are surely wasting your time. Just use [[JI]] scales and forget about edos and temperaments altogether. Remember when I said you could ''theoretically'' compose in these insane edos? That's because PB resolution is going to be your main bottleneck. The other one is your paranoia. | ||
// | ===== Vibe-coded Javascript script (V7) ===== | ||
<syntaxhighlight lang="javascript" line="1">// Generalized EDO Retuner with TRUE POLYPHONIC Micropitch Deviation (CC27 per-note) | |||
/ | // FIXED: simultaneous chords with master-channel CC27 now work correctly | ||
var NeedsTimingInfo = true; | |||
// Output pool | |||
var memberStart = 1, memberEnd = 16; | |||
// Scale-bend | |||
var scaleBendEnabled = true; | |||
var mpeMasterCCFeedsMembers = true; // set false if your controller sends CC27 on member channels | |||
// ===== Defaults / State ===== | // ===== Defaults / State ===== | ||
var | var EDO = 94; | ||
var | var rootKeyPC = 0; | ||
var ccNumber = 27; | var refMidiNote = 69; | ||
var | var refHz = 440.0; | ||
var | var ch1Mode = 0; | ||
var n = 8; | |||
var maxDeviation = 4800; | |||
var ccNumber = 27; | |||
var MidiChDeviationSplit = 1; | |||
var noteDelayMs = 2; | |||
var offsetScopeGlobal = false; // Per-Channel recommended for independent per-note micro | |||
var swallowCC = true; | |||
var debug = false; | |||
// Sustain and tail protection | |||
var | var sustainDown = false; | ||
var | var releaseHoldMs = 300; | ||
var | |||
var | // Panic | ||
var panicCCNumber = 83; | |||
var panicThreshold = 63; | |||
// Derived | // Derived | ||
| Line 93: | Line 111: | ||
// Micro offset memory | // Micro offset memory | ||
var globalMicro = 0; | var globalMicro = 0; | ||
var | var masterMicro = 0; // last CC27 received on MPE master (ch1) | ||
var currentMicroPerInCh = Array(17).fill(0); | |||
var microTouchedByInCh = Array(17).fill(false); | |||
// | // Output-channel state | ||
var channelBusy = Array(17).fill(0); | var channelBusy = Array(17).fill(0); | ||
var centered = Array(17).fill(true); | var centered = Array(17).fill(true); | ||
var noteMap = {}; | var releaseUntil = Array(17).fill(0); | ||
var lastPB = Array(17).fill(999999); | |||
// Note tracking | |||
var noteMap = {}; | |||
var uniq = 1; | var uniq = 1; | ||
// Allocator | |||
var pitchKeyToCh = {}; | |||
var chToPitchKey = Array(17).fill(""); | |||
var pitchKeyActive = {}; | |||
var rrNext = 1; | |||
// Scale-bend state | |||
var gSteps = 0; | |||
var scaleBend = Array(12).fill(0.0); | |||
var masterOffsetCents = 0.0; | |||
var FIFTH_OFFSETS = [0, -5, +2, -3, +4, -1, +6, +1, -4, +3, -2, +5]; | |||
// Safe Trace | |||
function T(s){ | |||
if (!debug) return; | |||
s = String(s); | |||
if (s.length > 120) s = s.slice(0, 120); | |||
Trace(s); | |||
} | |||
// ===== UI ===== | // ===== UI ===== | ||
var PluginParameters = [ | var PluginParameters = [ | ||
{ name:"EDO", type:"lin", minValue:1, maxValue:65535, numberOfSteps:65534, defaultValue:311 }, | { name:"EDO", type:"lin", minValue:1, maxValue:65535, numberOfSteps:65534, defaultValue:311 }, | ||
{ name:"Root key (PC)", type:"menu", valueStrings:["C","C#","D","D#","E","F","F#","G","G#","A","A#","B"], defaultValue:0 }, | |||
{ name:"Channel at which step=0 (n)", type:"lin", minValue: | { name:"Reference MIDI note", type:"lin", minValue:0, maxValue:127, numberOfSteps:127, defaultValue:69 }, | ||
{ name:"Reference Hz", type:"float", minValue:1, maxValue:20000, numberOfSteps:19999, defaultValue:440 }, | |||
{ name:"Ch 1 treatment", type:"menu", valueStrings:["Normal","ScaleBendOnly","Bypass (12edo)","Nullify"], defaultValue:0 }, | |||
{ name:"Channel at which step=0 (n)", type:"lin", minValue:1, maxValue:16, numberOfSteps:15, defaultValue:8 }, | |||
{ name:"PB range (± cents)", type:"float", minValue:1, maxValue:4800, numberOfSteps:4799, defaultValue:200 }, | { name:"PB range (± cents)", type:"float", minValue:1, maxValue:4800, numberOfSteps:4799, defaultValue:200 }, | ||
{ name:"Articulation CC#", type:"lin", minValue:0, maxValue:127, numberOfSteps:127, defaultValue:27 }, | { name:"Articulation CC#", type:"lin", minValue:0, maxValue:127, numberOfSteps:127, defaultValue:27 }, | ||
{ name:"MidiChDeviationSplit", type:"lin", minValue:0, maxValue:63, numberOfSteps:63, defaultValue:2 }, | { name:"MidiChDeviationSplit", type:"lin", minValue:0, maxValue:63, numberOfSteps:63, defaultValue:2 }, | ||
{ name:"Release Hold (ms)", type:"lin", minValue:0, maxValue:20000, numberOfSteps:20000, defaultValue:300 }, | |||
{ name:"Offset scope", type:"menu", valueStrings:["Per-Channel","Global-Last"], defaultValue: | { name:"Panic CC#", type:"lin", minValue:0, maxValue:127, numberOfSteps:127, defaultValue:83 }, | ||
{ name:"Panic thresh (>)", type:"lin", minValue:0, maxValue:127, numberOfSteps:127, defaultValue:63 }, | |||
{ name:"PANIC (kill all)", type:"menu", valueStrings:["Off","TRIGGER"], defaultValue:0 }, | |||
{ name:"Offset scope", type:"menu", valueStrings:["Per-Channel","Global-Last"], defaultValue:0 }, | |||
{ name:"Swallow CC to synth", type:"menu", valueStrings:["No","Yes"], defaultValue:1 }, | { name:"Swallow CC to synth", type:"menu", valueStrings:["No","Yes"], defaultValue:1 }, | ||
{ name:"NoteOn Delay (ms)", type:"lin", minValue:0, maxValue:10, numberOfSteps:10, defaultValue:2 }, | { name:"NoteOn Delay (ms)", type:"lin", minValue:0, maxValue:10, numberOfSteps:10, defaultValue:2 }, | ||
{ name:"Debug Trace", type:"menu", valueStrings:["Off","On"], defaultValue:0 } | { name:"Debug Trace", type:"menu", valueStrings:["Off","On"], defaultValue:0 }, | ||
{ name:"Output ch start", type:"lin", minValue:1, maxValue:16, numberOfSteps:15, defaultValue:1 }, | |||
{ name:"Output ch end", type:"lin", minValue:1, maxValue:16, numberOfSteps:15, defaultValue:16 }, | |||
{ name:"Scale-bend or 12EDO", type:"menu", valueStrings:["On","Off"], defaultValue:0 } | |||
]; | ]; | ||
function ParameterChanged(p,v){ | function ParameterChanged(p,v){ | ||
if (p===0){ | if (p===0){ EDO = Math.max(1, Math.min(65535, Math.floor(v))); STEP_CENTS = 1200.0 / EDO; rebuildScaleAndRetuneAssignedChannels(); } | ||
else if (p===1){ rootKeyPC = (v|0) % 12; rebuildScaleAndRetuneAssignedChannels(); } | |||
else if (p===2){ refMidiNote = Math.max(0, Math.min(127, v|0)); rebuildScaleAndRetuneAssignedChannels(); } | |||
else if (p===3){ refHz = Math.max(1.0, v); rebuildScaleAndRetuneAssignedChannels(); } | |||
else if (p===4){ ch1Mode = (v|0) & 3; } | |||
else if (p===5){ n = Math.max(1, Math.min(16, v|0)); } | |||
else if (p===6){ maxDeviation = Math.max(1, v); rebuildScaleAndRetuneAssignedChannels(); } | |||
else if (p===7){ ccNumber = Math.floor(v); } | |||
} | else if (p===8){ MidiChDeviationSplit = Math.max(0, Math.min(63, Math.floor(v))); } | ||
else if (p=== | else if (p===9){ releaseHoldMs = Math.max(0, Math.min(20000, Math.floor(v))); } | ||
else if (p=== | else if (p===10){ panicCCNumber = Math.max(0, Math.min(127, v|0)); } | ||
else if (p=== | else if (p===11){ panicThreshold = Math.max(0, Math.min(127, v|0)); } | ||
else if (p=== | else if (p===12){ if ((v|0)===1) panicAll(); } | ||
else if (p===13){ offsetScopeGlobal = (v|0)===1; } | |||
else if (p===14){ swallowCC = (v|0)===1; } | |||
else if (p===15){ noteDelayMs = Math.max(0, Math.min(10, v|0)); } | |||
else if (p===16){ debug = (v|0)===1; } | |||
else if (p===17){ setOutputPool(v|0, memberEnd); allocatorReset(); rebuildScaleAndRetuneAssignedChannels(); } | |||
else if (p===18){ setOutputPool(memberStart, v|0); allocatorReset(); rebuildScaleAndRetuneAssignedChannels(); } | |||
else if (p=== | else if (p===19){ scaleBendEnabled = ((v|0) === 0); rebuildScaleAndRetuneAssignedChannels(); } | ||
else if (p=== | |||
else if (p=== | |||
else if (p=== | |||
else if (p=== | |||
else if (p=== | |||
} | } | ||
// ===== Helpers ===== | // ===== Helpers ===== | ||
function safeCh(ch){ ch=(ch|0)||1; if (ch<1) ch=1; if (ch>16) ch=16; return ch; } | function safeCh(ch){ ch=(ch|0)||1; if (ch<1) ch=1; if (ch>16) ch=16; return ch; } | ||
function mod(a,m){ var r=a%m; if (r<0) r+=m; return r; } | |||
function | function log2(x){ return Math.log(x) / Math.log(2.0); } | ||
function setOutputPool(a, b){ | |||
return ( | a = safeCh(a); b = safeCh(b); if (a > b){ var t=a; a=b; b=t; } | ||
memberStart = a; memberEnd = b; | |||
if (rrNext < memberStart || rrNext > memberEnd) rrNext = memberStart; | |||
T("Pool " + memberStart + ".." + memberEnd); | |||
} | |||
function makePitchKey(inCh, pitch, step){ return String(inCh|0) + ":" + String(pitch|0) + ":" + String(step|0); } | |||
function GetNowMs(){ | |||
if (typeof GetTimingInfo !== "function") return 0; | |||
var t = GetTimingInfo(); | |||
if (!t) return 0; | |||
var tempo = (t.tempo && t.tempo > 0) ? t.tempo : 120.0; | |||
var beat = t.blockStartBeat !== undefined ? t.blockStartBeat : (t.playhead !== undefined ? t.playhead : 0); | |||
return (beat * 60000.0) / tempo; | |||
} | } | ||
function baseStepFromInChannel(ch){ return (ch - n) * MidiChDeviationSplit; } | |||
function microFromCC(val){ | function microFromCC(val){ | ||
if (MidiChDeviationSplit === 1) return 0; | if (MidiChDeviationSplit === 1) return 0; | ||
var m = (val|0) - 64; | var m = (val|0) - 64; | ||
if (m < -64) m = -64; | if (m < -64) m = -64; if (m > 63) m = 63; | ||
return m; | return m; | ||
} | } | ||
function centsToPB(c){ | function centsToPB(c){ | ||
var v = (c / maxDeviation) * 8191.0; | var v = (c / maxDeviation) * 8191.0; | ||
if (v > 8191) v = 8191; | if (v > 8191) v = 8191; if (v < -8192) v = -8192; | ||
return Math.round(v); | return Math.round(v); | ||
} | |||
function sendPB(ch, value){ | |||
ch = safeCh(ch); | |||
var pb = new PitchBend(); pb.channel = ch; pb.value = value; pb.send(); | |||
centered[ch] = (value === 0); lastPB[ch] = value; | |||
} | |||
function sendCC(ch, num, val){ | |||
var cc = new ControlChange(); cc.channel = safeCh(ch); cc.number = num|0; cc.value = val|0; cc.send(); | |||
} | } | ||
function | // ===== Scale bend ===== | ||
var | function rebuildScale(){ | ||
STEP_CENTS = 1200.0 / EDO; | |||
var gReal = EDO * (log2(3.0) - 1.0); | |||
gSteps = Math.round(gReal); | |||
var rawBend = Array(12).fill(0.0); | |||
for (var i=0; i<12; i++){ | |||
var st = mod(FIFTH_OFFSETS[i] * gSteps, EDO); | |||
rawBend[i] = st * STEP_CENTS - (i * 100.0); | |||
} | |||
var stdHz = 440.0 * Math.pow(2.0, (refMidiNote - 69) / 12.0); | |||
masterOffsetCents = 1200.0 * log2(refHz / stdHz); | |||
var refDeg = mod((refMidiNote|0) - rootKeyPC, 12); | |||
var anchor = rawBend[refDeg]; | |||
for (var k=0; k<12; k++) scaleBend[k] = rawBend[k] - anchor; | |||
T("Scale g="+gSteps+" EDO="+EDO); | |||
} | |||
function rebuildScaleAndRetuneAssignedChannels(){ | |||
rebuildScale(); | |||
for (var ch=memberStart; ch<=memberEnd; ch++){ | |||
var pk = chToPitchKey[ch]; | |||
if (pk && pk.length){ | |||
var p = parsePitchKey(pk); // note: parsePitchKey is defined below | |||
var deg = degreeForPitch(p.pitch); | |||
var sb = scaleBendEnabled ? (scaleBend[deg] || 0.0) : 0.0; | |||
var cents = masterOffsetCents + sb + p.step * STEP_CENTS; | |||
var pb = centsToPB(cents); | |||
if (lastPB[ch] !== pb) sendPB(ch, pb); | |||
} | |||
} | |||
} | |||
function parsePitchKey(pk){ | |||
var parts = String(pk).split(":"); | |||
return { inCh: (parts[0]|0), pitch: (parts[1]|0), step: (parts[2]|0) }; | |||
} | } | ||
function degreeForPitch(pitch){ return mod((pitch|0) - rootKeyPC, 12); } | |||
// ===== Note tracking ===== | |||
function key(inCh, pitch, id){ return ((safeCh(inCh)&0xFF)<<16) | ((pitch&0x7F)<<8) | (id&0xFF); } | function key(inCh, pitch, id){ return ((safeCh(inCh)&0xFF)<<16) | ((pitch&0x7F)<<8) | (id&0xFF); } | ||
function findMostRecent(inCh, pitch){ | |||
var foundKey = null, info = null, bestId = -1; | |||
for (var k in noteMap){ | |||
var packed = k|0; | |||
var kCh = (packed>>16)&0xFF; | |||
var kPitch = (packed>>8)&0x7F; | |||
var val = noteMap[k]; | |||
if (kCh===inCh && kPitch===(pitch|0) && val.id>bestId){ | |||
bestId = val.id; foundKey = k; info = val; | |||
} | |||
} | |||
return { foundKey: foundKey, info: info }; | |||
} | |||
function | // ===== Tail protection & allocator ===== | ||
function maybeCenterIdleChannels(now){ | |||
for (var ch=memberStart; ch<=memberEnd; ch++){ | for (var ch=memberStart; ch<=memberEnd; ch++){ | ||
if (channelBusy[ch]===0) return ch; | if (channelBusy[ch]===0 && now >= releaseUntil[ch] && !centered[ch]) sendPB(ch, 0); | ||
} | |||
} | |||
function nextChInPool(ch){ | |||
ch = (ch|0); if (ch < memberStart || ch > memberEnd) return memberStart; | |||
ch++; if (ch > memberEnd) ch = memberStart; return ch; | |||
} | |||
function freePitchKeyIfInactive(pk){ | |||
var c = pitchKeyActive[pk] || 0; | |||
if (c <= 0){ | |||
var ch = pitchKeyToCh[pk]; | |||
if (ch){ if (chToPitchKey[ch] === pk) chToPitchKey[ch] = ""; delete pitchKeyToCh[pk]; } | |||
delete pitchKeyActive[pk]; | |||
} | |||
} | |||
function hardKillChannel(ch){ | |||
ch = safeCh(ch); | |||
sendCC(ch, 120, 0); sendCC(ch, 123, 0); sendPB(ch, 0); | |||
for (var k in noteMap){ | |||
var info = noteMap[k]; | |||
if (info && info.ch === ch){ | |||
var pk = info.pitchKey; | |||
if (pitchKeyActive[pk]) pitchKeyActive[pk] = Math.max(0, (pitchKeyActive[pk]|0) - 1); | |||
delete noteMap[k]; | |||
} | |||
} | |||
channelBusy[ch] = 0; centered[ch] = true; releaseUntil[ch] = GetNowMs(); lastPB[ch] = 0; | |||
var oldPk = chToPitchKey[ch]; | |||
if (oldPk && oldPk.length){ chToPitchKey[ch] = ""; if (pitchKeyToCh[oldPk] === ch) delete pitchKeyToCh[oldPk]; freePitchKeyIfInactive(oldPk); } | |||
} | |||
function allocChannelForPitchKey(pk, now){ | |||
var existing = pitchKeyToCh[pk]; | |||
if (existing) return existing|0; | |||
var ch = rrNext; var span = (memberEnd - memberStart + 1); | |||
for (var i=0; i<span; i++){ | |||
if (!chToPitchKey[ch] && channelBusy[ch]===0 && now >= releaseUntil[ch]){ | |||
pitchKeyToCh[pk] = ch; chToPitchKey[ch] = pk; rrNext = nextChInPool(ch); return ch; | |||
} | |||
ch = nextChInPool(ch); | |||
} | |||
var steal = rrNext; rrNext = nextChInPool(steal); | |||
hardKillChannel(steal); | |||
pitchKeyToCh[pk] = steal; chToPitchKey[steal] = pk; | |||
return steal; | |||
} | |||
// ===== Panic & Reset ===== | |||
function allocatorReset(){ | |||
var now = GetNowMs(); | |||
pitchKeyToCh = {}; chToPitchKey = Array(17).fill(""); pitchKeyActive = {}; rrNext = memberStart; | |||
for (var ch=1; ch<=16; ch++){ | |||
sendCC(ch, 123, 0); sendPB(ch, 0); | |||
channelBusy[ch] = 0; centered[ch] = true; releaseUntil[ch] = now; lastPB[ch] = 0; | |||
} | |||
noteMap = {}; uniq = 1; | |||
T("ALLOC RESET"); | |||
} | |||
function panicAll(){ | |||
var now = GetNowMs(); | |||
for (var ch=1; ch<=16; ch++){ | |||
sendCC(ch, 120, 0); sendCC(ch, 123, 0); sendPB(ch, 0); | |||
channelBusy[ch] = 0; centered[ch] = true; releaseUntil[ch] = now; lastPB[ch] = 0; | |||
} | } | ||
noteMap = {}; uniq = 1; | |||
for (var | globalMicro = 0; masterMicro = 0; | ||
if (channelBusy[ | for (var i=1; i<=16; i++){ currentMicroPerInCh[i]=0; microTouchedByInCh[i]=false; } | ||
pitchKeyToCh = {}; chToPitchKey = Array(17).fill(""); pitchKeyActive = {}; rrNext = memberStart; | |||
sustainDown = false; | |||
T("PANIC ALL"); | |||
} | |||
function flushSustained(now){ | |||
var any = false; | |||
for (var k in noteMap){ | |||
var info = noteMap[k]; | |||
if (info && info.sustained){ | |||
any = true; | |||
var ch = info.ch; var pk = info.pitchKey; | |||
channelBusy[ch] = Math.max(0, channelBusy[ch]-1); | |||
if (pitchKeyActive[pk]) pitchKeyActive[pk] = Math.max(0, (pitchKeyActive[pk]|0) - 1); | |||
delete noteMap[k]; | |||
if (channelBusy[ch]===0) releaseUntil[ch] = now + releaseHoldMs; | |||
freePitchKeyIfInactive(pk); | |||
} | |||
} | } | ||
if (any) T("SustainUp: flushed"); | |||
} | } | ||
// ===== Main ===== | // ===== Main ===== | ||
function HandleMIDI(e){ | function HandleMIDI(e){ | ||
var now = GetNowMs(); | |||
maybeCenterIdleChannels(now); | |||
if (e instanceof ControlChange && e.number === (panicCCNumber|0)){ | |||
if ((e.value|0) > (panicThreshold|0)){ allocatorReset(); return; } | |||
e.send(); return; | |||
} | |||
if (e instanceof ControlChange && e.number === 64){ | |||
var v = e.value|0; var was = sustainDown; | |||
sustainDown = (v >= 64); e.send(); | |||
if (was && !sustainDown) flushSustained(now); | |||
return; | |||
} | |||
// CC27 capture (no live retune) | |||
if (e instanceof ControlChange && e.number === ccNumber){ | if (e instanceof ControlChange && e.number === ccNumber){ | ||
var | var chCC = safeCh(e.channel); | ||
var micro = microFromCC(e.value|0); | var micro = microFromCC(e.value|0); | ||
if ( | currentMicroPerInCh[chCC] = micro; | ||
if ( | microTouchedByInCh[chCC] = true; | ||
if (chCC === 1){ | |||
masterMicro = micro; // applies to all future member notes | |||
} | |||
if (offsetScopeGlobal){ | |||
globalMicro = micro; | |||
T("CC"+ccNumber+" ch"+chCC+" → GLOBAL m="+micro); | |||
} else { | |||
T("CC"+ccNumber+" ch"+chCC+" → captured m="+micro); | |||
} | |||
if (swallowCC) return; | if (swallowCC) return; | ||
e.send(); | e.send(); | ||
| Line 215: | Line 415: | ||
} | } | ||
// NoteOn — capture micro at exact moment of arrival | |||
if (e instanceof NoteOn){ | if (e instanceof NoteOn){ | ||
if ((e.velocity|0) === 0){ HandleMIDI(new NoteOff(e)); return; } | |||
var inCh = safeCh(e.channel); | var inCh = safeCh(e.channel); | ||
if (inCh === 1){ | |||
if (ch1Mode === 3) return; | |||
if (ch1Mode === 2){ e.send(); return; } | |||
} | |||
var deg = degreeForPitch(e.pitch|0); | |||
var base = baseStepFromInChannel(inCh); | var base = baseStepFromInChannel(inCh); | ||
var | |||
var | var microN = 0; | ||
var cents = | if (offsetScopeGlobal){ | ||
microN = globalMicro; | |||
} else { | |||
microN = currentMicroPerInCh[inCh] || 0; | |||
// MPE master fallback — now applies to ALL simultaneous notes | |||
if (inCh !== 1 && mpeMasterCCFeedsMembers && !microTouchedByInCh[inCh]){ | |||
microN = masterMicro; | |||
} | |||
} | |||
if (inCh === 1 && ch1Mode === 1){ base = 0; microN = 0; } | |||
var stepOffset = (base + microN) | 0; | |||
var pk = makePitchKey(inCh, (e.pitch|0), stepOffset); | |||
var tgt = allocChannelForPitchKey(pk, now); | |||
var sb = scaleBendEnabled ? (scaleBend[deg] || 0.0) : 0.0; | |||
var cents = masterOffsetCents + sb + stepOffset * STEP_CENTS; | |||
var pb = centsToPB(cents); | var pb = centsToPB(cents); | ||
T("NOTE in="+inCh+" out="+tgt+" pitch="+(e.pitch|0)+" micro="+microN+" cents="+cents.toFixed(3)+" pb="+pb); | |||
sendPB(tgt, pb); | if (lastPB[tgt] !== pb) sendPB(tgt, pb); | ||
var on = new NoteOn(e); | var on = new NoteOn(e); on.channel = tgt; | ||
if (noteDelayMs > 0) on.sendAfterMilliseconds(noteDelayMs); else on.send(); | |||
if (noteDelayMs>0) on.sendAfterMilliseconds(noteDelayMs); | |||
channelBusy[tgt]++; | channelBusy[tgt]++; pitchKeyActive[pk] = ((pitchKeyActive[pk]|0) + 1) | 0; | ||
var id = (uniq = (uniq+1)&0xFF) || 1; | var id = (uniq = (uniq+1)&0xFF) || 1; | ||
noteMap[key(inCh, e.pitch, id)] = { ch:tgt, id:id }; | noteMap[key(inCh, e.pitch, id)] = { ch:tgt, id:id, pitchKey:pk, sustained:false }; | ||
return; | return; | ||
} | } | ||
if (e instanceof NoteOff){ | if (e instanceof NoteOff){ | ||
var | var inChOff = safeCh(e.channel); | ||
var | var r = findMostRecent(inChOff, e.pitch|0); | ||
if (r.info){ | |||
var | var info = r.info; var chOut = info.ch; var pk2 = info.pitchKey; | ||
if ( | var off = new NoteOff(e); off.channel = chOut; off.send(); | ||
if (sustainDown){ | |||
info.sustained = true; noteMap[r.foundKey] = info; | |||
} else { | |||
delete noteMap[r.foundKey]; | |||
channelBusy[chOut] = Math.max(0, channelBusy[chOut]-1); | |||
if (pitchKeyActive[pk2]) pitchKeyActive[pk2] = Math.max(0, (pitchKeyActive[pk2]|0) - 1); | |||
if (channelBusy[chOut]===0) releaseUntil[chOut] = now + releaseHoldMs; | |||
freePitchKeyIfInactive(pk2); | |||
} | |||
return; | |||
} | } | ||
if (inChOff === 1){ | |||
if (ch1Mode === 3) return; | |||
if ( | if (ch1Mode === 2){ e.send(); return; } | ||
if ( | |||
} | } | ||
return; | e.send(); return; | ||
} | } | ||
e.send(); | e.send(); | ||
} | } | ||
function Reset(){ | function Reset(){ | ||
for (var | sustainDown = false; globalMicro = 0; masterMicro = 0; | ||
for (var i=1; i<=16; i++){ currentMicroPerInCh[i]=0; microTouchedByInCh[i]=false; } | |||
panicAll(); | |||
rebuildScaleAndRetuneAssignedChannels(); | |||
} | |||
} | |||
Parameters: | // init | ||
rebuildScale(); | |||
allocatorReset();</syntaxhighlight>Parameters: | |||
* EDO: Obviously, the edo you'll be working with. | * EDO: Obviously, the edo you'll be working with. | ||
* Root key: modes of 5L 7s 6|5 (built from the best fifth), transposing from C | |||
* Reference MIDI note, reference Hz: Reference pitch at NOTE = FREQUENCY. Default 69 = 440, but you can make it anything. | |||
* Channel 1 treatment: | |||
** Normal: CH1 acts normally as one midiCH deviation step. | |||
** ScaleBendOnly: CH1 Matches the 5L 7s of the tuning, but ignores midiCH and microdeviations. | |||
** Bypass: CH1 notes ignore the script completely. | |||
** Nullify: CH1 notes are killed. | |||
* Channel at which step=0 (n): from 3 to 9. The midpoint of your edostep deviations. Channel 1 is not used here. If you choose 5, then your edosteps will be 2:-3, 3:-2, 4:-1, 5:0, 6:+1, 7:+2, 8:+3, so the amount of deviations you have at your disposal is 2n+1. Note that no matter the input channel (the one in your piano roll), it will be sent as Channel 1, or allocated to another channel for MPE. | * Channel at which step=0 (n): from 3 to 9. The midpoint of your edostep deviations. Channel 1 is not used here. If you choose 5, then your edosteps will be 2:-3, 3:-2, 4:-1, 5:0, 6:+1, 7:+2, 8:+3, so the amount of deviations you have at your disposal is 2n+1. Note that no matter the input channel (the one in your piano roll), it will be sent as Channel 1, or allocated to another channel for MPE. | ||
* PB range (± cents): The range of your plugin's PB. Recommend values like 200 or 1200. For MPE, it is often 4800, but if you're using very fine grained edos, you might want to bring this down to get more resolution. | * PB range (± cents): The range of your plugin's PB. Recommend values like 200 or 1200. For MPE, it is often 4800, but if you're using very fine grained edos, you might want to bring this down to get more resolution. | ||
* Articulation CC#: The CC at which articulations will be sent, you can change it if it causes conflicts with your plugin, but then you need to change all the articulation set's CC. Default 27, Not recommended to change it. | * Articulation CC#: The CC at which articulations will be sent, you can change it if it causes conflicts with your plugin, but then you need to change all the articulation set's CC. Default 27, Not recommended to change it. | ||
* MidiChDeviationSplit: Multiplies the edosteps deviations of the MIDI channels by this number. at n=9, and this=16, then 10:+16, 8:-16, 11:+32, 7:-32... etc. If you are using HUGE edos (or edos with sharpness higher than 15), you will need to set this to a value bigger than 1. If you set this to 1, you can use it for smaller edos without duplicating instances '''(Preset-scale deviations - MIDI channels only, 12n)''' | * MidiChDeviationSplit: Multiplies the edosteps deviations of the MIDI channels by this number. at n=9, and this=16, then 10:+16, 8:-16, 11:+32, 7:-32... etc. If you are using HUGE edos (or edos with sharpness higher than 15), you will need to set this to a value bigger than 1. If you set this to 1, you can use it for smaller edos without duplicating instances '''(Preset-scale deviations - MIDI channels only, 12n)''' | ||
* Offset scope: I don't know what this | * Release Hold (ms): The duration that the MPE allocator holds each output channel's pitch after NoteOff messages. Recommended it be long for sounds with a long tail. | ||
* Panic CC#: Default 83. Exceeding {Panic Threshold} in this will kill all MIDI events, you can use this if the script behaves erratically. | |||
* Offset scope: I don't know exactly what this does. | |||
* Swallow CC to synth: I suppose this makes the synth recieve CC27...? | * Swallow CC to synth: I suppose this makes the synth recieve CC27...? | ||
* Member Ch Start: The lowest working channel for MPE. | * Member Ch Start: The lowest working channel for MPE. If you hear muted notes when you play, put it at 2. | ||
* Member Ch End: The highest working channel for MPE. | * Member Ch End: The highest working channel for MPE. No reason to put it lower than 16. | ||
* NoteOn Delay (ms): Set higher than 0 if the notes aren't being detuned correctly. Obviously, don't set it too high. Default 2. | * NoteOn Delay (ms): Set higher than 0 if the notes aren't being detuned correctly. Obviously, don't set it too high. Default 2. | ||
* Debug Trace: If something is not working, check this box to see debugging output in the Scripter. | * Debug Trace: If something is not working, check this box to see debugging output in the Scripter. | ||
* Scale-bend or 12edo: This makes only sense to deactivate with hyperfine 12n edos, to use 12edo instead of the near-pyth 5L 7s. | |||
===== .plist Articulation set ===== | ===== .plist Articulation set (alternate) ===== | ||
<syntaxhighlight lang="xml" line="1"> | <syntaxhighlight lang="xml" line="1"> | ||
<?xml version="1.0" encoding="UTF-8"?> | <?xml version="1.0" encoding="UTF-8"?> | ||
| Line 299: | Line 530: | ||
<key>Articulations</key> | <key>Articulations</key> | ||
<array> | <array> | ||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>1</integer> | |||
<key>ID</key> | |||
<integer>1001</integer> | |||
<key>Name</key> | |||
<string>0</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>64</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>2</integer> | |||
<key>ID</key> | |||
<integer>1002</integer> | |||
<key>Name</key> | |||
<string>+1</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>65</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>3</integer> | |||
<key>ID</key> | |||
<integer>1003</integer> | |||
<key>Name</key> | |||
<string>-1</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>63</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>4</integer> | |||
<key>ID</key> | |||
<integer>1004</integer> | |||
<key>Name</key> | |||
<string>+2</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>66</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>5</integer> | |||
<key>ID</key> | |||
<integer>1005</integer> | |||
<key>Name</key> | |||
<string>-2</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>62</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>6</integer> | |||
<key>ID</key> | |||
<integer>1006</integer> | |||
<key>Name</key> | |||
<string>+3</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>67</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>7</integer> | |||
<key>ID</key> | |||
<integer>1007</integer> | |||
<key>Name</key> | |||
<string>-3</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>61</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>8</integer> | |||
<key>ID</key> | |||
<integer>1008</integer> | |||
<key>Name</key> | |||
<string>+4</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>68</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>9</integer> | |||
<key>ID</key> | |||
<integer>1009</integer> | |||
<key>Name</key> | |||
<string>-4</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>60</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>10</integer> | |||
<key>ID</key> | |||
<integer>1010</integer> | |||
<key>Name</key> | |||
<string>+5</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>69</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>11</integer> | |||
<key>ID</key> | |||
<integer>1011</integer> | |||
<key>Name</key> | |||
<string>-5</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>59</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>12</integer> | |||
<key>ID</key> | |||
<integer>1012</integer> | |||
<key>Name</key> | |||
<string>+6</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>70</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>13</integer> | |||
<key>ID</key> | |||
<integer>1013</integer> | |||
<key>Name</key> | |||
<string>-6</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>58</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>14</integer> | |||
<key>ID</key> | |||
<integer>1014</integer> | |||
<key>Name</key> | |||
<string>+7</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
<integer>71</integer> | |||
</dict> | |||
</array> | |||
</dict> | |||
<dict> | |||
<key>ArticulationID</key> | |||
<integer>15</integer> | |||
<key>ID</key> | |||
<integer>1015</integer> | |||
<key>Name</key> | |||
<string>-7</string> | |||
<key>Output</key> | |||
<array> | |||
<dict> | |||
<key>MB1</key> | |||
<integer>27</integer> | |||
<key>Status</key> | |||
<string>Controller</string> | |||
<key>ValueLow</key> | |||
</array> | </array> | ||
<key>Name</key> | <key>Name</key> | ||
| Line 434: | Line 2,969: | ||
</dict> | </dict> | ||
</plist> | </plist> | ||
</syntaxhighlight> | </syntaxhighlight> | ||