music-fns is a JavaScript music utility library that contains small music notation related functions.
Installation
npm install music-fns
or
yarn add music-fns
Usage
All methods are available as named exports.
example:
; console;// > 440
API
This is an overview of all methods & constants.
music-fns
uses scientific pitch notation, you can use symbols (♭, ♯) or letters (b, #) as accidentals. The functions preserve your style (flat/sharp, letter/symbol).
Documentation is still WIP, PRs welcome.
🎵 Notes
CONSTANTS
NOTES
+ An nested array that contains a chromatic C scale with its possible note variations.
; console;/* > [ [ 'C', 'B#' ], [ 'C#', 'Db' ], [ 'D' ], [ 'D#', 'Eb' ], [ 'E', 'Fb' ], [ 'F', 'E#' ], [ 'F#', 'Gb' ], [ 'G' ], [ 'G#', 'Ab' ], [ 'A' ], [ 'A#', 'Bb' ], [ 'B', 'Cb' ] ]*/
METHODS
getRoot(note):note
+ Returns the root (only note, no accidental) from a note.
; console;// > 'A'
getNote(note):note
+ Returns the full note (note + accidental) from a note.
; console;// > 'A#'
getOctave(note):?octave
+ Returns the octave information (or undefined
) from a note.
; console;// > 4 console;// > undefined
getAccidental(note):?accidental
+ Returns the accidental (or undefined
) from a note.
; console;// > '#' console;// > undefined
hasOctave(note):boolean
+ Returns true
if the note has octave information.
; console;// > true console;// > false
hasAccidental(note):boolean
+ Returns true
if the note has an accidental.
; console;// > true console;// > false
hasAccidentalLetter(note):boolean
+ Returns true
if the note has an accidental as a letter (b, #).
; console;// > true console;// > false
hasAccidentalSymbol(note):boolean
+ Returns true
if the note has an accidental as a symbol (♭, ♯).
; console;// > true console;// > false
isFlat(note):boolean
+ Returns true
if the note is flat (b, ♭).
; console;// > true console;// > true console;// > false
isSharp(note):boolean
+ Returns true
if the note is sharp (#, ♯).
; console;// > true console;// > true console;// > false
isNatural(note):boolean
+ Returns true
if the note is natural (no accidental).
; console;// > true console;// > false
flatToSharp(note):note
+ Converts a flat to its sharp equivalent, this function preserves the accidental style (letter or symbol). If no flat is found the function returns the unmodified input.
; console;// > 'F#3'
sharpToFlat(note):note
+ Converts a sharp to its flat equivalent, this function preserves the accidental style (letter or symbol). If no sharp is found the function returns the unmodified input.
; console;// > 'Gb3'
areEqual(notes):boolean
+ Returns true
if the provided notes are the same notes.
; // same notes in flat & sharp, symbol and letterconsole;// > true console;// > false
getChromaticCPosition(note):position
+ Returns the (0-indexed) position of the specific root within a chromatic C scale (equals the NOTES
constant).
; console;// > 2 console;// > 8
haveSameOctave(notes):boolean
+ Returns true
if all notes share the same octave information.
; console;// > true console;// > false
noteToFrequency(note, [options]):frequency
+ Converts a note to a frequency (in Hz). You can use a different base frequency for A4 via standard
.
; console;// > 440 console;// > 138.59131548843604 console;// > 442
noteToMidi(note, [options]):midiNumber
+ Converts a note to its MIDI number. C4 = 60 in our implementation. You can provide a different middle C via standard
.
; console;// > 60 console;// > 81 console;// > 60
noteToObject(note):noteObject
+ Converts a note to an object describing the note.
The object can contain the following keys:
- root: a root note (
C
,D
,E
,F
,G
,A
,B
) - [accidental]:
SHARP
,FLAT
- [accidentalType]:
SYMBOL
,LETTER
- [octave]: number
; console;/* > { root: 'A', octave: 4} */ console;/* > { root: 'G', accidental: 'SHARP', accidentalType: 'SYMBOL', octave: 1} */ console; /* > { root: 'A', accidental: 'FLAT', accidentalType: 'LETTER', octave: 4} */
objectToNote(noteObject):note
+ Converts an object describing the note to a note.
The object can contain following keys:
- root: a root note (
C
,D
,E
,F
,G
,A
,B
) - [accidental]:
SHARP
/FLAT
- [accidentalType]:
SYMBOL
/LETTER
- [octave]: number
; console; // > 'A4' console; // > 'G♯1'
isNote(value):boolean
+ Returns true
if the provided value is a valid note in scientific pitch notation.
; console;// > true; console;// > false;
accidentalToLetter(note):note
+ Converts a note which has an accidental as a symbol (♭, ♯) to a note with the accidental as a letter (b, #).
; console;// > 'A#4'
accidentalToSymbol(note):note
+ Converts a note which has an accidental as a letter (b, #) to a note with the accidental as a symbol (♭, ♯).
; console;// > 'A♯4'
transferAccidental(note, referenceNote):note
+ Transfer the accidental type (flat or sharp) from a provided reference note
In this example, we've converted F♯3
to Gb3
because the reference note is Bb4
(a flat instead of sharp).
; console;// > 'Gb3';// = same as 'F#3', different style of notation
transferAccidentalStyle(note, referenceNote):note
+ Transfer the accidental notation style (letter or symbol) from the provided reference note
In this example, we've converted F♯3
to F♯3
because the reference note is B♭4
(a symbol instead of a letter).
; console;// > 'F♯3';// = same as 'F#3', different style of notation
transferStyle(note, referenceNote):note
+ Combination of transferAccidental
and transferAccidentalStyle
.
In this example, we've converted F#3
to G♭3
because the reference note is B♭4
(a flat symbol instead of a sharp letter).
; console;// > 'G♭3';// = same as 'F#3', different style of notation
🎹 Intervals
CONSTANTS
Interval
+ An object that contains different interval constants. Feel free to open a PR to add more interval constants.
(interval amount between the brackets ())
short:
(0)
ROOT / R(1)
SEMITONE / S(2)
TONE / T(6)
TRITONE / TT(12)
OCTAVE / O
minor / major:
(0)
PERFECT_UNISON / P1(1)
MINOR_SECOND / m2(2)
MAJOR_SECOND / M2(3)
MINOR_THIRD / m3(4)
MAJOR_THIRD / M3(5)
PERFECT_FOURTH / P4(7)
PERFECT_FIFTH / P5(8)
MINOR_SIXTH / m6(9)
MAJOR_SIXTH / M6(10)
MINOR_SEVENTH / m7(11)
MAJOR_SEVENTH / M7(12)
PERFECT_OCTAVE / P8
augmented / diminished:
(0)
DIMINISHED_SECOND / d2(1)
AUGMENTED_UNISON / A1(2)
DIMINISHED_THIRD / d3(3)
AUGMENTED_SECOND / A2(4)
DIMINISHED_FOURTH / d4(5)
AUGMENTED_THIRD / A3(6)
DIMINISHED_FIFTH / d5(6)
AUGMENTED_FOURTH / A4(7)
DIMINISHED_SIXTH / d6(8)
AUGMENTED_FIFTH / A5(9)
DIMINISHED_SEVENTH / d7(10)
AUGMENTED_SIXTH / A6(11)
DIMINISHED_EIGHTH / d8(11)
DIMINISHED_OCTAVE(12)
AUGMENTED_SEVENTH / A7
; console;// > 5 console;// > 10 console;// > 4 console;// > 1 console;// > 12
METHODS
transpose(note, interval):note
+ Transpose a note by a specific interval (use the Interval
constant). An interval can also be negative.
; console;// > 'A♯3' console;// > 'C♯2'
getIntervals(notes, [options]):intervals
+ Returns one or more intervals between the provided notes. You can pass fromRoot
to calculate the interval from the first note (= root).
; console;// > [ 9 ] // interval 'C♯3' -> 'A♯3' and 'A♯3' -> 'B3'console;// > [ 9, 1 ] // interval 'C♯3' -> 'A♯3' and 'C♯3' -> 'B3'console;// > [ 9, 10 ]
isSemitone(interval):boolean
+ Returns true
if the interval is a semitone (1)
.
; console;// > true;
isTone(interval):boolean
+ Returns true
if the interval is a tone (2)
.
; console;// > true;
isThird(interval):boolean
+ Returns true
if the interval is a third (diminished, minor, major or augmented) (2, 3, 4, 5)
.
; console;// > true; console;// > true; console;// > true;
isFifth(interval):boolean
+ Returns true
if the interval is a fifth (diminished, perfect or augmented) (6, 7, 8)
.
; console;// > true; console;// > true; console;// > true;
isOctave(number):boolean
+ Returns true
if the interval is an octave (12)
.
; console;// > true;
🎹 Scales & Modes
CONSTANTS
Scale
+ An object containing scale constants (array of intervals). Feel free to open PRs to add more scales.
- ALGERIAN:
[ 2, 1, 2, 1, 1, 1, 3, 1 ]
- ALTERED:
[ 1, 2, 1, 2, 2, 2, 2 ]
- ARABIC:
[ 1, 3, 1, 2, 1, 3, 1 ]
- AUGMENTED:
[ 3, 1, 3, 1, 3, 1 ]
- BALINESE:
[ 1, 2, 4, 1, 4 ]
- BLUES:
[ 3, 2, 1, 1, 3, 2 ]
- BYZANTINE:
[ 1, 3, 1, 2, 1, 3, 1 ]
- CHINESE:
[ 4, 2, 1, 4, 1 ]
- CHROMATIC:
[ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 ]
- DOUBLE_HARMONIC:
[ 1, 3, 1, 2, 1, 3, 1 ]
- ENIGMATIC:
[ 1, 3, 2, 2, 2, 1, 1 ]
- GYPSY_MAJOR:
[ 1, 3, 1, 2, 1, 3, 1 ]
- HARMONIC_MINOR:
[ 2, 1, 2, 2, 1, 3, 1 ]
- HIRAJOSHI:
[ 1, 4, 1, 4, 2 ]
- IN_SEN:
[ 1, 4, 2, 3, 2 ]
- JAPANESE:
[ 1, 4, 2, 3, 2 ]
- MAJOR_PENTATONIC:
[ 2, 2, 3, 2, 3 ]
- MAJOR:
[ 2, 2, 1, 2, 2, 2, 1 ]
- MELODIC_MINOR:
[ 2, 1, 2, 2, 2, 2, 1 ]
- MINOR_PENTATONIC:
[ 3, 2, 2, 3, 2 ]
- MONGOLIAN:
[ 2, 2, 3, 2, 3 ]
- NATURAL_MINOR:
[ 2, 1, 2, 2, 1, 2, 2 ]
- PELOG:
[ 1, 2, 4, 1, 4 ]
- PROMETHEUS:
[ 2, 2, 2, 3, 1, 2 ]
- WHOLE_TONE:
[ 2, 2, 2, 2, 2, 2 ]
- YO:
[ 2, 3, 2, 2, 3 ]
; console;// > [ 2, 1, 2, 1, 1, 1, 3, 1 ] console;// > [ 1, 2, 4, 1, 4 ] console;// > [ 1, 2, 4, 1, 4 ]
Mode
+ An object containing mode constants (array of intervals). White keys on a piano from note -> note an octave up.
- IONIAN: (C - C)
[ 2, 2, 1, 2, 2, 2, 1 ]
- DORIAN: (D - D)
[ 2, 1, 2, 2, 2, 1, 2 ]
- PHRYGIAN: (E - E)
[ 1, 2, 2, 2, 1, 2, 2 ]
- LYDIAN: (F - F)
[ 2, 2, 2, 1, 2, 2, 1 ]
- MIXOLYDIAN: (G - G)
[ 2, 2, 1, 2, 2, 1, 2 ]
- AEOLIAN: (A - A)
[ 2, 1, 2, 2, 1, 2, 2 ]
- LOCRIAN: (B - B)
[ 1, 2, 2, 1, 2, 2, 2 ]
; console;// > [ 2, 2, 1, 2, 2, 2, 1 ] console;// > [ 2, 2, 1, 2, 2, 2, 1 ]
METHODS
createScale(root, intervals, [options]):scale
+ Creates a scale (or mode) by providing a root note and an intervals (use the Scale
or Mode
constant). You can provide includeRootEnd
to also include the root note transposed an octave up.
; console;// > [ 'C2', 'C#2', 'E2', 'F2', 'G2', 'G#2', 'B2' ] console;// > [ 'G', 'A', 'B', 'C', 'D', 'E', 'F' ] console;// > [ 'F2', 'G#2', 'A#2', 'B2', 'C3', 'D#3', 'F3' ]
isScale(notes, [options]):boolean
+ Returns true
when the array of notes is a scale.
; console;// > true console;// > false
isMode(notes, [options]):boolean
+ Returns true
when the provided array of notes is a mode (Ionian, Dorian, Phrygian, Lydian, Mixolydian, Aeolian or Locrian).
; console;// > true console;// > false
isAscending(scale, [options]):boolean
+ Returns true
if the scale is ascending.
; console;// > true console;// > false
isDescending(scale, [options]):boolean
+ Returns true
if the scale is descending.
; console;// > true console;// > false
hasIntervalAmount(scale, intervalAmount, [options]):boolean
+ Returns true
if a scale has the provided interval amount.
; console;// > true console;// > true console;// > false
isPentatonic(scale, [options]):boolean
+ Returns true
if the scale is pentatonic (5 notes per octave).
; console;// > true console;// > true console;// > false
isHexatonic(scale, [options]):boolean
+ Returns true
if the scale is hexatonic (6 notes per octave).
; console;// > true console;// > true console;// > false
isHeptatonic(scale, [options]):boolean
+ Returns true
if the scale is heptatonic (7 notes per octave).
; console;// > true console;// > true console;// > false
isOctatonic(scale, [options]):boolean
+ Returns true
if the scale is octatonic (8 notes per octave).
; console;// > true console;// > true console;// > false
isDiatonic(scale, [options]):boolean
+ Returns true
if the scale is diatonic (5 tones & 2 semitones, where the semitones are separated at least 2 steps from each other).
; console;// > true console;// > false
isHemitonic(scale, [options]):boolean
+ Returns true
if the scale is hemitonic (contains 1 or more semitones).
; console;// > true console;// > false
isAnhemitonic(scale, [options]):boolean
+ Returns true
if the scale is anhemitonic (does not contain semitones).
; console;// > true console;// > false
isCohemitonic(scale, [options]):boolean
+ Returns true
if the scale is cohemitonic (contains 2 or more semitones that appear consecutively in scale order).
; // semitones between F# - G and G - Abconsole;// > true console;// > false
getNoteOnDegree(scale, degree, [options]):note
+ Returns the note on the provided scale degree.
; console;// > E1 console;// > C
getTonic(diatonicScale), [options]:note
+ Returns the note on scale degree 1
in a diatonic scale.
; console;// > C
getSupertonic(diatonicScale, [options]):note
+ Returns the note on scale degree 2
in a diatonic scale.
; console;// > D
getMediant(diatonicScale, [options]):note
+ Returns the note on scale degree 3
in a diatonic scale.
; console;// > E
getSubdominant(diatonicScale, [options]):note
+ Returns the note on scale degree 4
in a diatonic scale.
; console;// > F
getDominant(diatonicScale, [options]):note
+ Returns the note on scale degree 5
in a diatonic scale.
; console;// > G
getSubmediant(diatonicScale, [options]):note
+ Returns the note on scale degree 6
in a diatonic scale.
; console;// > A
getLeadingTone(diatonicScale, [options]):note
+ Returns the note on scale degree 7
in a diatonic scale.
; console;// > B
normalize(scale, [options]): scale
+ Normalize a scale by making sure it's ascending & has a root end.
; console;// > ['C1', 'D1', 'E1', 'F1', 'G1', 'C2'] console;// > ['C1', 'D1', 'E1', 'F1', 'G1', 'C2']
🎹 Chords
CONSTANTS
Chord
+ An object containing chord constants (array of intervals from root, root included). Feel free to open PRs to add more chords.
- AUGMENTED:
[ 0, 4, 8 ]
- DIMINISHED:
[ 0, 3, 6 ]
- MAJOR_SEVENTH:
[ 0, 4, 7, 11 ]
- MAJOR:
[ 0, 4, 7 ]
- MINOR_MAJOR_SEVENTH:
[ 0, 3, 7, 11 ]
- MINOR_SEVENTH:
[ 0, 3, 7, 10 ]
- MINOR:
[ 0, 3, 7 ]
- SEVENTH:
[ 0, 4, 7, 10 ]
; console;// > [ 0, 4, 7 ] console;// > [ 0, 3, 7, 11 ]
METHODS
createChord(root, intervals):chord
+ Creates a chord by providing a root note and intervals (use the Chord
constant).
; console;// > [ 'C2', 'E2', 'G2' ] console;// > [ 'D#3', 'G3', 'A#3', 'D4' ]
isTriad(chord):boolean
+ Returns true
if the chord is a triad (a set of three notes that can be stacked in thirds).
; console;// > true console;// > false
Melody
METHODS
createMelody(notes, pattern):notes
+ Creates a melody using a provided array of notes and a pattern.
; console;// > [ 'C2', 'E2', 'C2', 'D2', 'C2', 'F2' ]
Thanks.
Thanks to our employer madewithlove for letting us work on this during our weekly Friyays.