mirror of
https://github.com/eliasstepanik/strudel.git
synced 2026-01-10 05:08:33 +00:00
* polish, rename, and document stepwise functions: `pace`, `take`, `drop`, `expand`, `contract`, `repeat`, `zip`, `grow`, `shrink`, and `tour`
254 lines
9.2 KiB
JavaScript
254 lines
9.2 KiB
JavaScript
/*
|
|
tonal.mjs - <short description TODO>
|
|
Copyright (C) 2022 Strudel contributors - see <https://github.com/tidalcycles/strudel/blob/main/packages/tonal/tonal.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 { Note, Interval, Scale } from '@tonaljs/tonal';
|
|
import { register, _mod, silence, logger, pure, isNote } from '@strudel/core';
|
|
import { stepInNamedScale } from './tonleiter.mjs';
|
|
|
|
const octavesInterval = (octaves) => (octaves <= 0 ? -1 : 1) + octaves * 7 + 'P';
|
|
|
|
function scaleStep(step, scale) {
|
|
scale = scale.replaceAll(':', ' ');
|
|
step = Math.ceil(step);
|
|
let { intervals, tonic, empty } = Scale.get(scale);
|
|
if ((empty && isNote(scale)) || (!empty && !tonic)) {
|
|
throw new Error(`incomplete scale. Make sure to use ":" instead of spaces, example: .scale("C:major")`);
|
|
} else if (empty) {
|
|
throw new Error(`invalid scale "${scale}"`);
|
|
}
|
|
tonic = tonic || 'C';
|
|
const { pc, oct = 3 } = Note.get(tonic);
|
|
const octaveOffset = Math.floor(step / intervals.length);
|
|
const scaleStep = _mod(step, intervals.length);
|
|
const interval = Interval.add(intervals[scaleStep], octavesInterval(octaveOffset));
|
|
return Note.transpose(pc + oct, interval);
|
|
}
|
|
|
|
// transpose note inside scale by offset steps
|
|
// function scaleOffset(scale: string, offset: number, note: string) {
|
|
function scaleOffset(scale, offset, note) {
|
|
let [tonic, scaleName] = Scale.tokenize(scale);
|
|
let { notes } = Scale.get(`${tonic} ${scaleName}`);
|
|
notes = notes.map((note) => Note.get(note).pc); // use only pc!
|
|
offset = Number(offset);
|
|
if (isNaN(offset)) {
|
|
throw new Error(`scale offset "${offset}" not a number`);
|
|
}
|
|
const { pc: fromPc, oct = 3 } = Note.get(note);
|
|
const noteIndex = notes.indexOf(fromPc);
|
|
if (noteIndex === -1) {
|
|
throw new Error(`note "${note}" is not in scale "${scale}"`);
|
|
}
|
|
let i = noteIndex,
|
|
o = oct,
|
|
n = fromPc;
|
|
const direction = Math.sign(offset);
|
|
// TODO: find way to do this smarter
|
|
while (Math.abs(i - noteIndex) < Math.abs(offset)) {
|
|
i += direction;
|
|
const index = _mod(i, notes.length);
|
|
if (direction < 0 && n[0] === 'C') {
|
|
o += direction;
|
|
}
|
|
n = notes[index];
|
|
if (direction > 0 && n[0] === 'C') {
|
|
o += direction;
|
|
}
|
|
}
|
|
return n + o;
|
|
}
|
|
|
|
// Pattern.prototype._transpose = function (intervalOrSemitones: string | number) {
|
|
/**
|
|
* Change the pitch of each value by the given amount. Expects numbers or note strings as values.
|
|
* The amount can be given as a number of semitones or as a string in interval short notation.
|
|
* If you don't care about enharmonic correctness, just use numbers. Otherwise, pass the interval of
|
|
* the form: ST where S is the degree number and T the type of interval with
|
|
*
|
|
* - M = major
|
|
* - m = minor
|
|
* - P = perfect
|
|
* - A = augmented
|
|
* - d = diminished
|
|
*
|
|
* Examples intervals:
|
|
*
|
|
* - 1P = unison
|
|
* - 3M = major third
|
|
* - 3m = minor third
|
|
* - 4P = perfect fourth
|
|
* - 4A = augmented fourth
|
|
* - 5P = perfect fifth
|
|
* - 5d = diminished fifth
|
|
*
|
|
* @param {string | number} amount Either number of semitones or interval string.
|
|
* @returns Pattern
|
|
* @memberof Pattern
|
|
* @name transpose
|
|
* @example
|
|
* "c2 c3".fast(2).transpose("<0 -2 5 3>".slow(2)).note()
|
|
* @example
|
|
* "c2 c3".fast(2).transpose("<1P -2M 4P 3m>".slow(2)).note()
|
|
*/
|
|
|
|
export const transpose = register('transpose', function (intervalOrSemitones, pat) {
|
|
return pat.withHap((hap) => {
|
|
const note = hap.value.note ?? hap.value;
|
|
if (typeof note === 'number') {
|
|
// note is a number, so just add the number semitones of the interval
|
|
let semitones;
|
|
if (typeof intervalOrSemitones === 'number') {
|
|
semitones = intervalOrSemitones;
|
|
} else if (typeof intervalOrSemitones === 'string') {
|
|
semitones = Interval.semitones(intervalOrSemitones) || 0;
|
|
}
|
|
const targetNote = note + semitones;
|
|
if (typeof hap.value === 'object') {
|
|
return hap.withValue(() => ({ ...hap.value, note: targetNote }));
|
|
}
|
|
return hap.withValue(() => targetNote);
|
|
}
|
|
if (typeof note !== 'string' || !isNote(note)) {
|
|
logger(`[tonal] transpose: not a note "${note}"`, 'warning');
|
|
return hap;
|
|
}
|
|
// note is a string, so we might be able to preserve harmonics if interval is a string as well
|
|
const interval = !isNaN(Number(intervalOrSemitones))
|
|
? Interval.fromSemitones(intervalOrSemitones)
|
|
: String(intervalOrSemitones);
|
|
// TODO: move simplify to player to preserve enharmonics
|
|
// tone.js doesn't understand multiple sharps flats e.g. F##3 has to be turned into G3
|
|
// TODO: check if this is still relevant..
|
|
const targetNote = Note.simplify(Note.transpose(note, interval));
|
|
if (typeof hap.value === 'object') {
|
|
return hap.withValue(() => ({ ...hap.value, note: targetNote }));
|
|
}
|
|
return hap.withValue(() => targetNote);
|
|
});
|
|
});
|
|
|
|
// example: transpose(3).late(0.2) will be equivalent to compose(transpose(3), late(0.2))
|
|
// e.g. `stack(c3).superimpose(transpose(slowcat(7, 5)))` or
|
|
// or even `stack(c3).superimpose(transpose.slowcat(7, 5))` or
|
|
|
|
/**
|
|
* Transposes notes inside the scale by the number of steps.
|
|
* Expected to be called on a Pattern which already has a {@link Pattern#scale}
|
|
*
|
|
* @memberof Pattern
|
|
* @name scaleTranspose
|
|
* @param {offset} offset number of steps inside the scale
|
|
* @returns Pattern
|
|
* @example
|
|
* "-8 [2,4,6]"
|
|
* .scale('C4 bebop major')
|
|
* .scaleTranspose("<0 -1 -2 -3 -4 -5 -6 -4>")
|
|
* .note()
|
|
*/
|
|
|
|
export const scaleTranspose = register('scaleTranspose', function (offset /* : number | string */, pat) {
|
|
return pat.withHap((hap) => {
|
|
if (!hap.context.scale) {
|
|
throw new Error('can only use scaleTranspose after .scale');
|
|
}
|
|
if (typeof hap.value === 'object')
|
|
return hap.withValue(() => ({
|
|
...hap.value,
|
|
note: scaleOffset(hap.context.scale, Number(offset), hap.value.note),
|
|
}));
|
|
if (typeof hap.value !== 'string') {
|
|
throw new Error('can only use scaleTranspose with notes');
|
|
}
|
|
return hap.withValue(() => scaleOffset(hap.context.scale, Number(offset), hap.value));
|
|
});
|
|
});
|
|
|
|
/**
|
|
* Turns numbers into notes in the scale (zero indexed). Also sets scale for other scale operations, like {@link Pattern#scaleTranspose}.
|
|
*
|
|
* A scale consists of a root note (e.g. `c4`, `c`, `f#`, `bb4`) followed by semicolon (':') and then a [scale type](https://github.com/tonaljs/tonal/blob/main/packages/scale-type/data.ts).
|
|
*
|
|
* The root note defaults to octave 3, if no octave number is given.
|
|
*
|
|
* @name scale
|
|
* @param {string} scale Name of scale
|
|
* @returns Pattern
|
|
* @example
|
|
* n("0 2 4 6 4 2").scale("C:major")
|
|
* @example
|
|
* n("[0,7] 4 [2,7] 4")
|
|
* .scale("C:<major minor>/2")
|
|
* .s("piano")
|
|
* @example
|
|
* n(rand.range(0,12).segment(8))
|
|
* .scale("C:ritusen")
|
|
* .s("piano")
|
|
*/
|
|
|
|
export const scale = register(
|
|
'scale',
|
|
function (scale, pat) {
|
|
// Supports ':' list syntax in mininotation
|
|
if (Array.isArray(scale)) {
|
|
scale = scale.flat().join(' ');
|
|
}
|
|
return (
|
|
pat
|
|
.fmap((value) => {
|
|
const isObject = typeof value === 'object';
|
|
let step = isObject ? value.n : value;
|
|
if (isObject) {
|
|
delete value.n; // remove n so it won't cause trouble
|
|
}
|
|
if (isNote(step)) {
|
|
// legacy..
|
|
return pure(step);
|
|
}
|
|
let asNumber = Number(step);
|
|
let semitones = 0;
|
|
if (isNaN(asNumber)) {
|
|
step = String(step);
|
|
if (!/^[-+]?\d+(#*|b*){1}$/.test(step)) {
|
|
logger(
|
|
`[tonal] invalid scale step "${step}", expected number or integer with optional # b suffixes`,
|
|
'error',
|
|
);
|
|
return silence;
|
|
}
|
|
const isharp = step.indexOf('#');
|
|
if (isharp >= 0) {
|
|
asNumber = Number(step.substring(0, isharp));
|
|
semitones = step.length - isharp;
|
|
} else {
|
|
const iflat = step.indexOf('b');
|
|
asNumber = Number(step.substring(0, iflat));
|
|
semitones = iflat - step.length;
|
|
}
|
|
}
|
|
try {
|
|
let note;
|
|
if (isObject && value.anchor) {
|
|
note = stepInNamedScale(asNumber, scale, value.anchor);
|
|
} else {
|
|
note = scaleStep(asNumber, scale);
|
|
}
|
|
if (semitones != 0) note = Note.transpose(note, Interval.fromSemitones(semitones));
|
|
value = pure(isObject ? { ...value, note } : note);
|
|
} catch (err) {
|
|
logger(`[tonal] ${err.message}`, 'error');
|
|
value = silence;
|
|
}
|
|
return value;
|
|
})
|
|
.outerJoin()
|
|
// legacy:
|
|
.withHap((hap) => hap.setContext({ ...hap.context, scale }))
|
|
);
|
|
},
|
|
true,
|
|
true, // preserve step count
|
|
);
|