org.clojars.triss/midi-data0.2.0-SNAPSHOTHandles parsing of MIDI data tuples to maps containing friendly representations of the events they represent. dependencies
| (this space intentionally left almost blank) | |||
MIDI data is often presented as a list of three values: This namespace handle's parsing of MIDI data tuples to maps that represnt there content and back. The values
| (ns midi-data.core (:require [clojure.set :refer [map-invert]])) | |||
MIDI Message types | ||||
Maps MIDI message types to the numbers used to represent them. | (def midi-msg-type->n {:clock 0xF8 :active 0xFE :sysex 0xF0 :eox 0xF7 :start 0xFA :stop 0xFC :continue 0xFB :reset 0xFF :note-on 0x90 :note-off 0x80 :channel-at 0xD0 :poly-at 0xA0 :program 0xC0 :control 0xB0 :pitchbend 0xE0 :mtc 0xF1 :song-pos 0xF2 :song-sel 0xF3 :tune 0xF6}) | |||
Map of numbers to midi msg types. Channel numbers often need to be removed with a fn like quantize before they can be looked up using this. | (def inverse-n->midi-msg-type (map-invert midi-msg-type->n)) | |||
Returns channel indicated by value. Typically used on first value of MIDI data tuple. | (defn n->channel [n] (bit-and n 0x0F)) | |||
(defn strip-channel [n] (bit-and-not n 0xF)) | ||||
Returns key representing meaning of numerical value. If n has channel number encoded within it it is discarded. | (defn n->midi-msg-type [n] (or (inverse-n->midi-msg-type n) (inverse-n->midi-msg-type (strip-channel n)))) | |||
MIDI Message paramater types | ||||
Maps midi message type to a to the names that | (def midi-msg-type->msg-params {:note-on [:note :velocity] :note-off [:note :velocity] :control [:number :value]}) | |||
Decodeing | ||||
Decodes MIDI message and transforms it to a map containing a representation of the MIDI message being sent. | (defn decode [[a b c :as data]] (let [msg-type (n->midi-msg-type a)] (merge {:midi-msg msg-type :channel (n->channel a)} (or (zipmap (midi-msg-type->msg-params msg-type) [b c]) {:data data})))) | |||
Encoding | ||||
Takes representation of MIDI event provided as a map and turns it back to
a data tuple of | (defn encode [{:keys [midi-msg channel] :or {channel 0} :as m}] (cons (+ (midi-msg-type->n midi-msg) channel) ((apply juxt (midi-msg-type->msg-params midi-msg)) m))) | |||
Provide's functions for quickly creating MIDI messages of various types. | (ns midi-data.messages) | |||
Construct's a note on message. | (defn note-on [note velocity & [channel]] {:midi-msg :note-on :note note :velocity velocity :channel (or channel 0)}) | |||
Construct's a note off message. | (defn note-off [note velocity & [channel]] {:midi-msg :note-off :note note :velocity velocity :channel (or channel 0)}) | |||
Construct's a control (sometimes called cc) message. | (defn control [number value & [channel]] {:midi-msg :control :number number :value value :channel (or channel 0)}) | |||
Tests for midi-data.core. | (ns midi-data.core-test (:use midje.sweet) (:use [midi-data.core])) | |||
Represents a note-on on channel 1 with velocity 100 and note of 60. | (def test-map {:channel 1 :midi-msg :note-on :note 60 :velocity 100}) | |||
(facts "about `decode`" (fact "provides representation of MIDI data tuple as a map." (decode [0x91 60 100]) => test-map) (fact "`decode` is practically inverse of `encode`" (decode (encode test-map)) => test-map)) | ||||
(facts "about `encode`" (fact "Converts data in map back to MIDI data tuple." (encode test-map) => [0x91 60 100]) (fact "`encode` is practically inverse of `decode`" (encode (decode [0x91 60 100])) => [0x91 60 100])) | ||||