557 lines
34 KiB
JavaScript
557 lines
34 KiB
JavaScript
function AudioKeys(options) {
|
|
var self = this;
|
|
|
|
self._setState(options);
|
|
|
|
// all listeners are stored in arrays in their respective properties.
|
|
// e.g. self._listeners.down = [fn1, fn2, ... ]
|
|
self._listeners = {};
|
|
|
|
// bind DOM events
|
|
self._bind();
|
|
}
|
|
|
|
// Play well with require so that we can run a test suite and use browserify.
|
|
if(typeof module !== 'undefined') {
|
|
module.exports = AudioKeys;
|
|
}
|
|
|
|
AudioKeys.prototype._setState = function(options) {
|
|
var self = this;
|
|
|
|
if(!options) {
|
|
options = {};
|
|
}
|
|
|
|
// the state is kept in this object
|
|
self._state = {};
|
|
|
|
// set some defaults ...
|
|
self._extendState({
|
|
polyphony: 4,
|
|
rows: 1,
|
|
priority: 'last',
|
|
rootNote: 60,
|
|
octaveControls: true,
|
|
octave: 0,
|
|
velocityControls: true,
|
|
velocity: 127,
|
|
keys: [],
|
|
buffer: []
|
|
});
|
|
|
|
// ... and override them with options.
|
|
self._extendState(options);
|
|
};
|
|
|
|
AudioKeys.prototype._extendState = function(options) {
|
|
var self = this;
|
|
|
|
for(var o in options) {
|
|
self._state[o] = options[o];
|
|
}
|
|
};
|
|
|
|
AudioKeys.prototype.set = function(/* options || property, value */) {
|
|
var self = this;
|
|
|
|
if(arguments.length === 1) {
|
|
self._extendState(arguments[0]);
|
|
} else {
|
|
self._state[arguments[0]] = arguments[1];
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
AudioKeys.prototype.get = function(property) {
|
|
var self = this;
|
|
|
|
return self._state[property];
|
|
};
|
|
|
|
// ================================================================
|
|
// Event Listeners
|
|
// ================================================================
|
|
|
|
// AudioKeys has a very simple event handling system. Internally
|
|
// we'll call self._trigger('down', argument) when we want to fire
|
|
// an event for the user.
|
|
|
|
AudioKeys.prototype.down = function(fn) {
|
|
var self = this;
|
|
|
|
// add the function to our list of listeners
|
|
self._listeners.down = (self._listeners.down || []).concat(fn);
|
|
};
|
|
|
|
AudioKeys.prototype.up = function(fn) {
|
|
var self = this;
|
|
|
|
// add the function to our list of listeners
|
|
self._listeners.up = (self._listeners.up || []).concat(fn);
|
|
};
|
|
|
|
AudioKeys.prototype._trigger = function(action /* args */) {
|
|
var self = this;
|
|
|
|
// if we have any listeners by this name ...
|
|
if(self._listeners[action] && self._listeners[action].length) {
|
|
// grab the arguments to pass to the listeners ...
|
|
var args = Array.prototype.slice.call(arguments);
|
|
args.splice(0, 1);
|
|
// and call them!
|
|
self._listeners[action].forEach( function(fn) {
|
|
fn.apply(self, args);
|
|
});
|
|
}
|
|
};
|
|
|
|
// ================================================================
|
|
// DOM Bindings
|
|
// ================================================================
|
|
|
|
AudioKeys.prototype._bind = function() {
|
|
var self = this;
|
|
|
|
if(typeof window !== 'undefined' && window.document) {
|
|
window.document.addEventListener('keydown', function(e) {
|
|
self._addKey(e);
|
|
});
|
|
window.document.addEventListener('keyup', function(e) {
|
|
self._removeKey(e);
|
|
});
|
|
|
|
var lastFocus = true;
|
|
setInterval( function() {
|
|
if(window.document.hasFocus() === lastFocus) {
|
|
return;
|
|
}
|
|
lastFocus = !lastFocus;
|
|
if(!lastFocus) {
|
|
self.clear();
|
|
}
|
|
}, 100);
|
|
}
|
|
};
|
|
|
|
// _map returns the midi note for a given keyCode.
|
|
AudioKeys.prototype._map = function(keyCode) {
|
|
return this._keyMap[this._state.rows][keyCode] + this._offset();
|
|
};
|
|
|
|
AudioKeys.prototype._offset = function() {
|
|
return this._state.rootNote - this._keyMap[this._state.rows].root + (this._state.octave * 12);
|
|
};
|
|
|
|
// _isNote determines whether a keyCode is a note or not.
|
|
AudioKeys.prototype._isNote = function(keyCode) {
|
|
return !!this._keyMap[this._state.rows][keyCode];
|
|
};
|
|
|
|
// convert a midi note to a frequency. we assume here that _map has
|
|
// already been called (to account for a potential rootNote offset)
|
|
AudioKeys.prototype._toFrequency = function(note) {
|
|
return ( Math.pow(2, ( note-69 ) / 12) ) * 440.0;
|
|
};
|
|
|
|
// the object keys correspond to `rows`, so `_keyMap[rows]` should
|
|
// retrieve that particular mapping.
|
|
AudioKeys.prototype._keyMap = {
|
|
1: {
|
|
root: 60,
|
|
// starting with the 'a' key
|
|
65: 60,
|
|
87: 61,
|
|
83: 62,
|
|
69: 63,
|
|
68: 64,
|
|
70: 65,
|
|
84: 66,
|
|
71: 67,
|
|
89: 68,
|
|
72: 69,
|
|
85: 70,
|
|
74: 71,
|
|
75: 72,
|
|
79: 73,
|
|
76: 74,
|
|
80: 75,
|
|
186: 76,
|
|
222: 77
|
|
},
|
|
2: {
|
|
root: 60,
|
|
// bottom row
|
|
90: 60,
|
|
83: 61,
|
|
88: 62,
|
|
68: 63,
|
|
67: 64,
|
|
86: 65,
|
|
71: 66,
|
|
66: 67,
|
|
72: 68,
|
|
78: 69,
|
|
74: 70,
|
|
77: 71,
|
|
188: 72,
|
|
76: 73,
|
|
190: 74,
|
|
186: 75,
|
|
191: 76,
|
|
// top row
|
|
81: 72,
|
|
50: 73,
|
|
87: 74,
|
|
51: 75,
|
|
69: 76,
|
|
82: 77,
|
|
53: 78,
|
|
84: 79,
|
|
54: 80,
|
|
89: 81,
|
|
55: 82,
|
|
85: 83,
|
|
73: 84,
|
|
57: 85,
|
|
79: 86,
|
|
48: 87,
|
|
80: 88,
|
|
219: 89,
|
|
187: 90,
|
|
221: 91
|
|
}
|
|
};
|
|
|
|
// ================================================================
|
|
// KEY BUFFER
|
|
// ================================================================
|
|
|
|
// The process is:
|
|
|
|
// key press
|
|
// add to self._state.keys
|
|
// (an accurate representation of keys currently pressed)
|
|
// resolve self.buffer
|
|
// based on polyphony and priority, determine the notes
|
|
// that get triggered for the user
|
|
|
|
AudioKeys.prototype._addKey = function(e) {
|
|
var self = this;
|
|
// if the keyCode is one that can be mapped and isn't
|
|
// already pressed, add it to the key object.
|
|
if(self._isNote(e.keyCode) && !self._isPressed(e.keyCode)) {
|
|
var newKey = self._makeNote(e.keyCode);
|
|
// add the newKey to the list of keys
|
|
self._state.keys = (self._state.keys || []).concat(newKey);
|
|
// reevaluate the active notes based on our priority rules.
|
|
// give it the new note to use if there is an event to trigger.
|
|
self._update();
|
|
} else if(self._isSpecialKey(e.keyCode)) {
|
|
self._specialKey(e.keyCode);
|
|
}
|
|
};
|
|
|
|
AudioKeys.prototype._removeKey = function(e) {
|
|
var self = this;
|
|
// if the keyCode is active, remove it from the key object.
|
|
if(self._isPressed(e.keyCode)) {
|
|
var keyToRemove;
|
|
for(var i = 0; i < self._state.keys.length; i++) {
|
|
if(self._state.keys[i].keyCode === e.keyCode) {
|
|
keyToRemove = self._state.keys[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
// remove the key from _keys
|
|
self._state.keys.splice(self._state.keys.indexOf(keyToRemove), 1);
|
|
self._update();
|
|
}
|
|
};
|
|
|
|
AudioKeys.prototype._isPressed = function(keyCode) {
|
|
var self = this;
|
|
|
|
if(!self._state.keys || !self._state.keys.length) {
|
|
return false;
|
|
}
|
|
|
|
for(var i = 0; i < self._state.keys.length; i++) {
|
|
if(self._state.keys[i].keyCode === keyCode) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
// turn a key object into a note object for the event listeners.
|
|
AudioKeys.prototype._makeNote = function(keyCode) {
|
|
var self = this;
|
|
return {
|
|
keyCode: keyCode,
|
|
note: self._map(keyCode),
|
|
frequency: self._toFrequency( self._map(keyCode) ),
|
|
velocity: self._state.velocity
|
|
};
|
|
};
|
|
|
|
// clear any active notes
|
|
AudioKeys.prototype.clear = function() {
|
|
var self = this;
|
|
// trigger note off for the notes in the buffer before
|
|
// removing them.
|
|
self._state.buffer.forEach( function(key) {
|
|
self._trigger('up', key);
|
|
});
|
|
self._state.keys = [];
|
|
self._state.buffer = [];
|
|
};
|
|
|
|
// ================================================================
|
|
// NOTE BUFFER
|
|
// ================================================================
|
|
|
|
// every time a change is made to _keys due to a key on or key off
|
|
// we need to call `_update`. It compares the `_keys` array to the
|
|
// `buffer` array, which is the array of notes that are really
|
|
// being played, makes the necessary changes to `buffer` and
|
|
// triggers any events that need triggering.
|
|
|
|
AudioKeys.prototype._update = function() {
|
|
var self = this;
|
|
|
|
// a key has been added to self._state.keys.
|
|
// stash the old buffer
|
|
var oldBuffer = self._state.buffer;
|
|
// set the new priority in self.state._keys
|
|
self._prioritize();
|
|
// compare the buffers and trigger events based on
|
|
// the differences.
|
|
self._diff(oldBuffer);
|
|
};
|
|
|
|
AudioKeys.prototype._diff = function(oldBuffer) {
|
|
var self = this;
|
|
|
|
// if it's not in the OLD buffer, it's a note ON.
|
|
// if it's not in the NEW buffer, it's a note OFF.
|
|
|
|
var oldNotes = oldBuffer.map( function(key) {
|
|
return key.keyCode;
|
|
});
|
|
|
|
var newNotes = self._state.buffer.map( function(key) {
|
|
return key.keyCode;
|
|
});
|
|
|
|
// check for old (removed) notes
|
|
var notesToRemove = [];
|
|
oldNotes.forEach( function(key) {
|
|
if(newNotes.indexOf(key) === -1) {
|
|
notesToRemove.push(key);
|
|
}
|
|
});
|
|
|
|
// check for new notes
|
|
var notesToAdd = [];
|
|
newNotes.forEach( function(key) {
|
|
if(oldNotes.indexOf(key) === -1) {
|
|
notesToAdd.push(key);
|
|
}
|
|
});
|
|
|
|
notesToAdd.forEach( function(key) {
|
|
for(var i = 0; i < self._state.buffer.length; i++) {
|
|
if(self._state.buffer[i].keyCode === key) {
|
|
self._trigger('down', self._state.buffer[i]);
|
|
break;
|
|
}
|
|
}
|
|
});
|
|
|
|
notesToRemove.forEach( function(key) {
|
|
// these need to fire the entire object
|
|
for(var i = 0; i < oldBuffer.length; i++) {
|
|
if(oldBuffer[i].keyCode === key) {
|
|
self._trigger('up', oldBuffer[i]);
|
|
break;
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
AudioKeys.prototype._prioritize = function() {
|
|
var self = this;
|
|
|
|
// if all the keys have been turned off, no need
|
|
// to do anything here.
|
|
if(!self._state.keys.length) {
|
|
self._state.buffer = [];
|
|
return;
|
|
}
|
|
|
|
|
|
if(self._state.polyphony >= self._state.keys.length) {
|
|
// every note is active
|
|
self._state.keys = self._state.keys.map( function(key) {
|
|
key.isActive = true;
|
|
return key;
|
|
});
|
|
} else {
|
|
// set all keys to inactive.
|
|
self._state.keys = self._state.keys.map( function(key) {
|
|
key.isActive = false;
|
|
return key;
|
|
});
|
|
|
|
self['_' + self._state.priority]();
|
|
}
|
|
|
|
// now take the isActive keys and set the new buffer.
|
|
self._state.buffer = [];
|
|
|
|
self._state.keys.forEach( function(key) {
|
|
if(key.isActive) {
|
|
self._state.buffer.push(key);
|
|
}
|
|
});
|
|
|
|
// done.
|
|
};
|
|
|
|
AudioKeys.prototype._last = function() {
|
|
var self = this;
|
|
// set the last bunch to active based on the polyphony.
|
|
for(var i = self._state.keys.length - self._state.polyphony; i < self._state.keys.length; i++) {
|
|
self._state.keys[i].isActive = true;
|
|
}
|
|
};
|
|
|
|
AudioKeys.prototype._first = function() {
|
|
var self = this;
|
|
// set the last bunch to active based on the polyphony.
|
|
for(var i = 0; i < self._state.polyphony; i++) {
|
|
self._state.keys[i].isActive = true;
|
|
}
|
|
};
|
|
|
|
AudioKeys.prototype._highest = function() {
|
|
var self = this;
|
|
// get the highest notes and set them to active
|
|
var notes = self._state.keys.map( function(key) {
|
|
return key.note;
|
|
});
|
|
|
|
notes.sort( function(b,a) {
|
|
if(a === b) {
|
|
return 0;
|
|
}
|
|
return a < b ? -1 : 1;
|
|
});
|
|
|
|
notes.splice(self._state.polyphony, Number.MAX_VALUE);
|
|
|
|
self._state.keys.forEach( function(key) {
|
|
if(notes.indexOf(key.note) !== -1) {
|
|
key.isActive = true;
|
|
}
|
|
});
|
|
};
|
|
|
|
AudioKeys.prototype._lowest = function() {
|
|
var self = this;
|
|
// get the lowest notes and set them to active
|
|
var notes = self._state.keys.map( function(key) {
|
|
return key.note;
|
|
});
|
|
|
|
notes.sort( function(a,b) {
|
|
if(a === b) {
|
|
return 0;
|
|
}
|
|
return a < b ? -1 : 1;
|
|
});
|
|
|
|
notes.splice(self._state.polyphony, Number.MAX_VALUE);
|
|
|
|
self._state.keys.forEach( function(key) {
|
|
if(notes.indexOf(key.note) !== -1) {
|
|
key.isActive = true;
|
|
}
|
|
});
|
|
};
|
|
|
|
// This file maps special keys to the state— octave shifting and
|
|
// velocity selection, both available when `rows` = 1.
|
|
|
|
AudioKeys.prototype._isSpecialKey = function(keyCode) {
|
|
return (this._state.rows === 1 && this._specialKeyMap[keyCode]);
|
|
};
|
|
|
|
AudioKeys.prototype._specialKey = function(keyCode) {
|
|
var self = this;
|
|
if(self._specialKeyMap[keyCode].type === 'octave' && self._state.octaveControls) {
|
|
// shift the state of the `octave`
|
|
self._state.octave += self._specialKeyMap[keyCode].value;
|
|
} else if(self._specialKeyMap[keyCode].type === 'velocity' && self._state.velocityControls) {
|
|
// set the `velocity` to a new value
|
|
self._state.velocity = self._specialKeyMap[keyCode].value;
|
|
}
|
|
};
|
|
|
|
AudioKeys.prototype._specialKeyMap = {
|
|
// octaves
|
|
90: {
|
|
type: 'octave',
|
|
value: -1
|
|
},
|
|
88: {
|
|
type: 'octave',
|
|
value: 1
|
|
},
|
|
// velocity
|
|
49: {
|
|
type: 'velocity',
|
|
value: 1
|
|
},
|
|
50: {
|
|
type: 'velocity',
|
|
value: 14
|
|
},
|
|
51: {
|
|
type: 'velocity',
|
|
value: 28
|
|
},
|
|
52: {
|
|
type: 'velocity',
|
|
value: 42
|
|
},
|
|
53: {
|
|
type: 'velocity',
|
|
value: 56
|
|
},
|
|
54: {
|
|
type: 'velocity',
|
|
value: 70
|
|
},
|
|
55: {
|
|
type: 'velocity',
|
|
value: 84
|
|
},
|
|
56: {
|
|
type: 'velocity',
|
|
value: 98
|
|
},
|
|
57: {
|
|
type: 'velocity',
|
|
value: 112
|
|
},
|
|
48: {
|
|
type: 'velocity',
|
|
value: 127
|
|
},
|
|
};
|
|
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["AudioKeys.js","AudioKeys.state.js","AudioKeys.events.js","AudioKeys.mapping.js","AudioKeys.buffer.js","AudioKeys.priority.js","AudioKeys.special.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AChEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACpGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"audiokeys.js","sourcesContent":["function AudioKeys(options) {\n  var self = this;\n\n  self._setState(options);\n\n  // all listeners are stored in arrays in their respective properties.\n  // e.g. self._listeners.down = [fn1, fn2, ... ]\n  self._listeners = {};\n\n  // bind DOM events\n  self._bind();\n}\n\n// Play well with require so that we can run a test suite and use browserify.\nif(typeof module !== 'undefined') {\n  module.exports = AudioKeys;\n}\n","AudioKeys.prototype._setState = function(options) {\n  var self = this;\n\n  if(!options) {\n    options = {};\n  }\n\n  // the state is kept in this object\n  self._state = {};\n\n  // set some defaults ...\n  self._extendState({\n    polyphony: 4,\n    rows: 1,\n    priority: 'last',\n    rootNote: 60,\n    octaveControls: true,\n    octave: 0,\n    velocityControls: true,\n    velocity: 127,\n    keys: [],\n    buffer: []\n  });\n\n  // ... and override them with options.\n  self._extendState(options);\n};\n\nAudioKeys.prototype._extendState = function(options) {\n  var self = this;\n\n  for(var o in options) {\n    self._state[o] = options[o];\n  }\n};\n\nAudioKeys.prototype.set = function(/* options || property, value */) {\n  var self = this;\n\n  if(arguments.length === 1) {\n    self._extendState(arguments[0]);\n  } else {\n    self._state[arguments[0]] = arguments[1];\n  }\n\n  return this;\n};\n\nAudioKeys.prototype.get = function(property) {\n  var self = this;\n\n  return self._state[property];\n};\n","// ================================================================\n// Event Listeners\n// ================================================================\n\n// AudioKeys has a very simple event handling system. Internally\n// we'll call self._trigger('down', argument) when we want to fire\n// an event for the user.\n\nAudioKeys.prototype.down = function(fn) {\n  var self = this;\n\n  // add the function to our list of listeners\n  self._listeners.down = (self._listeners.down || []).concat(fn);\n};\n\nAudioKeys.prototype.up = function(fn) {\n  var self = this;\n\n  // add the function to our list of listeners\n  self._listeners.up = (self._listeners.up || []).concat(fn);\n};\n\nAudioKeys.prototype._trigger = function(action /* args */) {\n  var self = this;\n\n  // if we have any listeners by this name ...\n  if(self._listeners[action] && self._listeners[action].length) {\n    // grab the arguments to pass to the listeners ...\n    var args = Array.prototype.slice.call(arguments);\n    args.splice(0, 1);\n    // and call them!\n    self._listeners[action].forEach( function(fn) {\n      fn.apply(self, args);\n    });\n  }\n};\n\n// ================================================================\n// DOM Bindings\n// ================================================================\n\nAudioKeys.prototype._bind = function() {\n  var self = this;\n\n  if(typeof window !== 'undefined' && window.document) {\n    window.document.addEventListener('keydown', function(e) {\n      self._addKey(e);\n    });\n    window.document.addEventListener('keyup', function(e) {\n      self._removeKey(e);\n    });\n\n    var lastFocus = true;\n    setInterval( function() {\n      if(window.document.hasFocus() === lastFocus) {\n        return;\n      }\n      lastFocus = !lastFocus;\n      if(!lastFocus) {\n        self.clear();\n      }\n    }, 100);\n  }\n};\n","// _map returns the midi note for a given keyCode.\nAudioKeys.prototype._map = function(keyCode) {\n  return this._keyMap[this._state.rows][keyCode] + this._offset();\n};\n\nAudioKeys.prototype._offset = function() {\n  return this._state.rootNote - this._keyMap[this._state.rows].root + (this._state.octave * 12);\n};\n\n// _isNote determines whether a keyCode is a note or not.\nAudioKeys.prototype._isNote = function(keyCode) {\n  return !!this._keyMap[this._state.rows][keyCode];\n};\n\n// convert a midi note to a frequency. we assume here that _map has\n// already been called (to account for a potential rootNote offset)\nAudioKeys.prototype._toFrequency = function(note) {\n  return ( Math.pow(2, ( note-69 ) / 12) ) * 440.0;\n};\n\n// the object keys correspond to `rows`, so `_keyMap[rows]` should\n// retrieve that particular mapping.\nAudioKeys.prototype._keyMap = {\n  1: {\n    root: 60,\n    // starting with the 'a' key\n    65:  60,\n    87:  61,\n    83:  62,\n    69:  63,\n    68:  64,\n    70:  65,\n    84:  66,\n    71:  67,\n    89:  68,\n    72:  69,\n    85:  70,\n    74:  71,\n    75:  72,\n    79:  73,\n    76:  74,\n    80:  75,\n    186: 76,\n    222: 77\n  },\n  2: {\n    root: 60,\n    // bottom row\n    90:  60,\n    83:  61,\n    88:  62,\n    68:  63,\n    67:  64,\n    86:  65,\n    71:  66,\n    66:  67,\n    72:  68,\n    78:  69,\n    74:  70,\n    77:  71,\n    188: 72,\n    76:  73,\n    190: 74,\n    186: 75,\n    191: 76,\n    // top row\n    81:  72,\n    50:  73,\n    87:  74,\n    51:  75,\n    69:  76,\n    82:  77,\n    53:  78,\n    84:  79,\n    54:  80,\n    89:  81,\n    55:  82,\n    85:  83,\n    73:  84,\n    57:  85,\n    79:  86,\n    48:  87,\n    80:  88,\n    219: 89,\n    187: 90,\n    221: 91\n  }\n};\n","// ================================================================\n// KEY BUFFER\n// ================================================================\n\n// The process is:\n\n// key press\n//   add to self._state.keys\n//   (an accurate representation of keys currently pressed)\n// resolve self.buffer\n//   based on polyphony and priority, determine the notes\n//   that get triggered for the user\n\nAudioKeys.prototype._addKey = function(e) {\n  var self = this;\n  // if the keyCode is one that can be mapped and isn't\n  // already pressed, add it to the key object.\n  if(self._isNote(e.keyCode) && !self._isPressed(e.keyCode)) {\n    var newKey = self._makeNote(e.keyCode);\n    // add the newKey to the list of keys\n    self._state.keys = (self._state.keys || []).concat(newKey);\n    // reevaluate the active notes based on our priority rules.\n    // give it the new note to use if there is an event to trigger.\n    self._update();\n  } else if(self._isSpecialKey(e.keyCode)) {\n    self._specialKey(e.keyCode);\n  }\n};\n\nAudioKeys.prototype._removeKey = function(e) {\n  var self = this;\n  // if the keyCode is active, remove it from the key object.\n  if(self._isPressed(e.keyCode)) {\n    var keyToRemove;\n    for(var i = 0; i < self._state.keys.length; i++) {\n      if(self._state.keys[i].keyCode === e.keyCode) {\n        keyToRemove = self._state.keys[i];\n        break;\n      }\n    }\n\n    // remove the key from _keys\n    self._state.keys.splice(self._state.keys.indexOf(keyToRemove), 1);\n    self._update();\n  }\n};\n\nAudioKeys.prototype._isPressed = function(keyCode) {\n  var self = this;\n\n  if(!self._state.keys || !self._state.keys.length) {\n    return false;\n  }\n\n  for(var i = 0; i < self._state.keys.length; i++) {\n    if(self._state.keys[i].keyCode === keyCode) {\n      return true;\n    }\n  }\n  return false;\n};\n\n// turn a key object into a note object for the event listeners.\nAudioKeys.prototype._makeNote = function(keyCode) {\n  var self = this;\n  return {\n    keyCode: keyCode,\n    note: self._map(keyCode),\n    frequency: self._toFrequency( self._map(keyCode) ),\n    velocity: self._state.velocity\n  };\n};\n\n// clear any active notes\nAudioKeys.prototype.clear = function() {\n  var self = this;\n  // trigger note off for the notes in the buffer before\n  // removing them.\n  self._state.buffer.forEach( function(key) {\n    self._trigger('up', key);\n  });\n  self._state.keys = [];\n  self._state.buffer = [];\n};\n\n// ================================================================\n// NOTE BUFFER\n// ================================================================\n\n// every time a change is made to _keys due to a key on or key off\n// we need to call `_update`. It compares the `_keys` array to the\n// `buffer` array, which is the array of notes that are really\n// being played, makes the necessary changes to `buffer` and\n// triggers any events that need triggering.\n\nAudioKeys.prototype._update = function() {\n  var self = this;\n\n  // a key has been added to self._state.keys.\n  // stash the old buffer\n  var oldBuffer = self._state.buffer;\n  // set the new priority in self.state._keys\n  self._prioritize();\n  // compare the buffers and trigger events based on\n  // the differences.\n  self._diff(oldBuffer);\n};\n\nAudioKeys.prototype._diff = function(oldBuffer) {\n  var self = this;\n\n  // if it's not in the OLD buffer, it's a note ON.\n  // if it's not in the NEW buffer, it's a note OFF.\n\n  var oldNotes = oldBuffer.map( function(key) {\n    return key.keyCode;\n  });\n\n  var newNotes = self._state.buffer.map( function(key) {\n    return key.keyCode;\n  });\n\n  // check for old (removed) notes\n  var notesToRemove = [];\n  oldNotes.forEach( function(key) {\n    if(newNotes.indexOf(key) === -1) {\n      notesToRemove.push(key);\n    }\n  });\n\n  // check for new notes\n  var notesToAdd = [];\n  newNotes.forEach( function(key) {\n    if(oldNotes.indexOf(key) === -1) {\n      notesToAdd.push(key);\n    }\n  });\n\n  notesToAdd.forEach( function(key) {\n    for(var i = 0; i < self._state.buffer.length; i++) {\n      if(self._state.buffer[i].keyCode === key) {\n        self._trigger('down', self._state.buffer[i]);\n        break;\n      }\n    }\n  });\n\n  notesToRemove.forEach( function(key) {\n    // these need to fire the entire object\n    for(var i = 0; i < oldBuffer.length; i++) {\n      if(oldBuffer[i].keyCode === key) {\n        self._trigger('up', oldBuffer[i]);\n        break;\n      }\n    }\n  });\n};\n","AudioKeys.prototype._prioritize = function() {\n  var self = this;\n\n  // if all the keys have been turned off, no need\n  // to do anything here.\n  if(!self._state.keys.length) {\n    self._state.buffer = [];\n    return;\n  }\n\n\n  if(self._state.polyphony >= self._state.keys.length) {\n    // every note is active\n    self._state.keys = self._state.keys.map( function(key) {\n      key.isActive = true;\n      return key;\n    });\n  } else {\n    // set all keys to inactive.\n    self._state.keys = self._state.keys.map( function(key) {\n      key.isActive = false;\n      return key;\n    });\n\n    self['_' + self._state.priority]();\n  }\n\n  // now take the isActive keys and set the new buffer.\n  self._state.buffer = [];\n\n  self._state.keys.forEach( function(key) {\n    if(key.isActive) {\n      self._state.buffer.push(key);\n    }\n  });\n\n  // done.\n};\n\nAudioKeys.prototype._last = function() {\n  var self = this;\n  // set the last bunch to active based on the polyphony.\n  for(var i = self._state.keys.length - self._state.polyphony; i < self._state.keys.length; i++) {\n    self._state.keys[i].isActive = true;\n  }\n};\n\nAudioKeys.prototype._first = function() {\n  var self = this;\n  // set the last bunch to active based on the polyphony.\n  for(var i = 0; i < self._state.polyphony; i++) {\n    self._state.keys[i].isActive = true;\n  }\n};\n\nAudioKeys.prototype._highest = function() {\n  var self = this;\n  // get the highest notes and set them to active\n  var notes = self._state.keys.map( function(key) {\n    return key.note;\n  });\n\n  notes.sort( function(b,a) {\n    if(a === b) {\n      return 0;\n    }\n    return a < b ? -1 : 1;\n  });\n\n  notes.splice(self._state.polyphony, Number.MAX_VALUE);\n\n  self._state.keys.forEach( function(key) {\n    if(notes.indexOf(key.note) !== -1) {\n      key.isActive = true;\n    }\n  });\n};\n\nAudioKeys.prototype._lowest = function() {\n  var self = this;\n  // get the lowest notes and set them to active\n  var notes = self._state.keys.map( function(key) {\n    return key.note;\n  });\n\n  notes.sort( function(a,b) {\n    if(a === b) {\n      return 0;\n    }\n    return a < b ? -1 : 1;\n  });\n\n  notes.splice(self._state.polyphony, Number.MAX_VALUE);\n\n  self._state.keys.forEach( function(key) {\n    if(notes.indexOf(key.note) !== -1) {\n      key.isActive = true;\n    }\n  });\n};\n","// This file maps special keys to the state— octave shifting and\n// velocity selection, both available when `rows` = 1.\n\nAudioKeys.prototype._isSpecialKey = function(keyCode) {\n  return (this._state.rows === 1 && this._specialKeyMap[keyCode]);\n};\n\nAudioKeys.prototype._specialKey = function(keyCode) {\n  var self = this;\n  if(self._specialKeyMap[keyCode].type === 'octave' && self._state.octaveControls) {\n    // shift the state of the `octave`\n    self._state.octave += self._specialKeyMap[keyCode].value;\n  } else if(self._specialKeyMap[keyCode].type === 'velocity' && self._state.velocityControls) {\n    // set the `velocity` to a new value\n    self._state.velocity = self._specialKeyMap[keyCode].value;\n  }\n};\n\nAudioKeys.prototype._specialKeyMap = {\n  // octaves\n  90: {\n    type: 'octave',\n    value: -1\n  },\n  88: {\n    type: 'octave',\n    value: 1\n  },\n  // velocity\n  49: {\n    type: 'velocity',\n    value: 1\n  },\n  50: {\n    type: 'velocity',\n    value: 14\n  },\n  51: {\n    type: 'velocity',\n    value: 28\n  },\n  52: {\n    type: 'velocity',\n    value: 42\n  },\n  53: {\n    type: 'velocity',\n    value: 56\n  },\n  54: {\n    type: 'velocity',\n    value: 70\n  },\n  55: {\n    type: 'velocity',\n    value: 84\n  },\n  56: {\n    type: 'velocity',\n    value: 98\n  },\n  57: {\n    type: 'velocity',\n    value: 112\n  },\n  48: {\n    type: 'velocity',\n    value: 127\n  },\n};\n"],"sourceRoot":"/source/"}
|