define(["Tone/core/Tone", "Tone/core/Clock", "Tone/signal/Signal"], function(Tone){ "use strict"; /** * Time can be descibed in a number of ways. * Any Method which accepts Tone.Time as a parameter will accept: * * Numbers, which will be taken literally as the time (in seconds). * * Notation, ("4n", "8t") describes time in BPM and time signature relative values. * * Transport Time, ("4:3:2") will also provide tempo and time signature relative times * in the form BARS:QUARTERS:SIXTEENTHS. * * Frequency, ("8hz") is converted to the length of the cycle in seconds. * * Now-Relative, ("+1") prefix any of the above with "+" and it will be interpreted as * "the current time plus whatever expression follows". * * Expressions, ("3:0 + 2 - (1m / 7)") any of the above can also be combined * into a mathematical expression which will be evaluated to compute the desired time. * * No Argument, for methods which accept time, no argument will be interpreted as * 0 seconds or "now" (i.e. the currentTime) depending on the context. * * @typedef {number|string|undefined} Tone.Time */ /** * @class oscillator-based transport allows for simple musical timing * supports tempo curves and time changes * * @constructor * @extends {Tone} */ Tone.Transport = function(){ /** * watches the main oscillator for timing ticks * initially starts at 120bpm * * @private * @type {Tone.Clock} */ this._clock = new Tone.Clock(1, this._processTick.bind(this)); /** * @type {boolean} */ this.loop = false; /** * @type {TransportState} */ this.state = TransportState.STOPPED; }; Tone.extend(Tone.Transport); /** * @private * @type {number} */ var timelineTicks = 0; /** * @private * @type {number} */ var transportTicks = 0; /** * @private * @type {number} */ var tatum = 12; /** * controls which beat the swing is applied to * defaults to an 16th note * @private * @type {number} */ var swingTatum = 3; /** * controls which beat the swing is applied to * @private * @type {number} */ var swingAmount = 0; /** * @private * @type {number} */ var transportTimeSignature = 4; /** * @private * @type {number} */ var loopStart = 0; /** * @private * @type {number} */ var loopEnd = tatum * 4; /** * @private * @type {Array} */ var intervals = []; /** * @private * @type {Array} */ var timeouts = []; /** * @private * @type {Array} */ var transportTimeline = []; /** * @private * @type {number} */ var timelineProgress = 0; /** * All of the synced components * @private * @type {Array} */ var SyncedSources = []; /** * @enum */ var TransportState = { STARTED : "started", PAUSED : "paused", STOPPED : "stopped" }; /////////////////////////////////////////////////////////////////////////////// // TICKS /////////////////////////////////////////////////////////////////////////////// /** * called on every tick * @param {number} tickTime clock relative tick time * @private */ Tone.Transport.prototype._processTick = function(tickTime){ if (this.state === TransportState.STARTED){ if (swingAmount > 0 && timelineTicks % tatum !== 0 && timelineTicks % swingTatum === 0){ //add some swing tickTime += Tone.Transport.ticksToSeconds(swingTatum) * swingAmount; } processIntervals(tickTime); processTimeouts(tickTime); processTimeline(tickTime); transportTicks += 1; timelineTicks += 1; if (this.loop){ if (timelineTicks >= loopEnd){ this._setTicks(loopStart); } } } }; /** * jump to a specific tick in the timeline * updates the timeline callbacks * * @param {number} ticks the tick to jump to * @private */ Tone.Transport.prototype._setTicks = function(ticks){ timelineTicks = ticks; for (var i = 0; i < transportTimeline.length; i++){ var timeout = transportTimeline[i]; if (timeout.callbackTick() >= ticks){ timelineProgress = i; break; } } }; /////////////////////////////////////////////////////////////////////////////// // EVENT PROCESSING /////////////////////////////////////////////////////////////////////////////// /** * process the intervals * @param {number} time */ var processIntervals = function(time){ for (var i = 0, len = intervals.length; i transportTicks){ break; } } //remove the timeouts off the front of the array after they've been called timeouts.splice(0, removeTimeouts); }; /** * process the transportTimeline events * @param {number} time */ var processTimeline = function(time){ for (var i = timelineProgress, len = transportTimeline.length; i timelineTicks){ break; } } }; /////////////////////////////////////////////////////////////////////////////// // INTERVAL /////////////////////////////////////////////////////////////////////////////// /** * intervals are recurring events * * @example * //triggers a callback every 8th note with the exact time of the event * Tone.Transport.setInterval(function(time){ * envelope.triggerAttack(time); * }, "8n"); * * @param {function} callback * @param {Tone.Time} interval * @param {Object} ctx the context the function is invoked in * @return {number} the id of the interval */ Tone.Transport.prototype.setInterval = function(callback, interval, ctx){ var tickTime = this.toTicks(interval); var timeout = new TimelineEvent(callback, ctx, tickTime, transportTicks); intervals.push(timeout); return timeout.id; }; /** * clear an interval from the processing array * @param {number} rmInterval the interval to remove * @return {boolean} true if the event was removed */ Tone.Transport.prototype.clearInterval = function(rmInterval){ for (var i = 0; i < intervals.length; i++){ var interval = intervals[i]; if (interval.id === rmInterval){ intervals.splice(i, 1); return true; } } return false; }; /** * removes all of the intervals that are currently set */ Tone.Transport.prototype.clearIntervals = function(){ intervals = []; }; /////////////////////////////////////////////////////////////////////////////// // TIMEOUT /////////////////////////////////////////////////////////////////////////////// /** * set a timeout to occur after time from now. NB: the transport must be * running for this to be triggered. All timeout events are cleared when the * transport is stopped. * * @example * //trigger an event to happen 1 second from now * Tone.Transport.setTimeout(function(time){ * player.start(time); * }, 1) * * @param {function} callback * @param {Tone.Time} time * @param {Object} ctx the context to invoke the callback in * @return {number} the id of the timeout for clearing timeouts */ Tone.Transport.prototype.setTimeout = function(callback, time, ctx){ var ticks = this.toTicks(time); var timeout = new TimelineEvent(callback, ctx, ticks + transportTicks, 0); //put it in the right spot for (var i = 0, len = timeouts.length; i timeout.callbackTick()){ timeouts.splice(i, 0, timeout); return timeout.id; } } //otherwise push it on the end timeouts.push(timeout); return timeout.id; }; /** * clear the timeout based on it's ID * @param {number} timeoutID * @return {boolean} true if the timeout was removed */ Tone.Transport.prototype.clearTimeout = function(timeoutID){ for (var i = 0; i < timeouts.length; i++){ var testTimeout = timeouts[i]; if (testTimeout.id === timeoutID){ timeouts.splice(i, 1); return true; } } return false; }; /** * removes all of the timeouts that are currently set */ Tone.Transport.prototype.clearTimeouts = function(){ timeouts = []; }; /////////////////////////////////////////////////////////////////////////////// // TIMELINE /////////////////////////////////////////////////////////////////////////////// /** * Timeline events are synced to the transportTimeline of the Tone.Transport * Unlike Timeout, Timeline events will restart after the * Tone.Transport has been stopped and restarted. * * @example * //trigger the start of a part on the 16th measure * Tone.Transport.setTimeline(function(time){ * part.start(time); * }, "16m"); * * * @param {function} callback * @param {Tome.Time} timeout * @param {Object} ctx the context in which the funtion is called * @return {number} the id for clearing the transportTimeline event */ Tone.Transport.prototype.setTimeline = function(callback, timeout, ctx){ var ticks = this.toTicks(timeout); var timelineEvnt = new TimelineEvent(callback, ctx, ticks, 0); //put it in the right spot for (var i = timelineProgress, len = transportTimeline.length; i timelineEvnt.callbackTick()){ transportTimeline.splice(i, 0, timelineEvnt); return timelineEvnt.id; } } //otherwise push it on the end transportTimeline.push(timelineEvnt); return timelineEvnt.id; }; /** * clear the transportTimeline event from the * @param {number} timelineID * @return {boolean} true if it was removed */ Tone.Transport.prototype.clearTimeline = function(timelineID){ for (var i = 0; i < transportTimeline.length; i++){ var testTimeline = transportTimeline[i]; if (testTimeline.id === timelineID){ transportTimeline.splice(i, 1); return true; } } return false; }; /** * remove all events from the timeline */ Tone.Transport.prototype.clearTimelines = function(){ timelineProgress = 0; transportTimeline = []; }; /////////////////////////////////////////////////////////////////////////////// // TIME CONVERSIONS /////////////////////////////////////////////////////////////////////////////// /** * turns the time into * @param {Tone.Time} time * @return {number} */ Tone.Transport.prototype.toTicks = function(time){ //get the seconds var seconds = this.toSeconds(time); var quarter = this.notationToSeconds("4n"); var quarters = seconds / quarter; var tickNum = quarters * tatum; //quantize to tick value return Math.round(tickNum); }; /** * get the transport time * @return {string} in transportTime format (measures:beats:sixteenths) */ Tone.Transport.prototype.getTransportTime = function(){ var quarters = timelineTicks / tatum; var measures = Math.floor(quarters / transportTimeSignature); var sixteenths = Math.floor((quarters % 1) * 4); quarters = Math.floor(quarters) % transportTimeSignature; var progress = [measures, quarters, sixteenths]; return progress.join(":"); }; /** * set the transport time, jump to the position right away * * @param {Tone.Time} progress */ Tone.Transport.prototype.setTransportTime = function(progress){ var ticks = this.toTicks(progress); this._setTicks(ticks); }; /////////////////////////////////////////////////////////////////////////////// // START/STOP/PAUSE /////////////////////////////////////////////////////////////////////////////// /** * start the transport and all sources synced to the transport * * @param {Tone.Time} time * @param {Tone.Time=} offset the offset position to start */ Tone.Transport.prototype.start = function(time, offset){ if (this.state === TransportState.STOPPED || this.state === TransportState.PAUSED){ if (!this.isUndef(offset)){ this._setTicks(this.toTicks(offset)); } this.state = TransportState.STARTED; var startTime = this.toSeconds(time); this._clock.start(startTime); //call start on each of the synced sources for (var i = 0; i < SyncedSources.length; i++){ var source = SyncedSources[i].source; var delay = SyncedSources[i].delay; source.start(startTime + delay); } } }; /** * stop the transport and all sources synced to the transport * * @param {Tone.Time} time */ Tone.Transport.prototype.stop = function(time){ if (this.state === TransportState.STARTED || this.state === TransportState.PAUSED){ var stopTime = this.toSeconds(time); this._clock.stop(stopTime, this._onend.bind(this)); //call start on each of the synced sources for (var i = 0; i < SyncedSources.length; i++){ var source = SyncedSources[i].source; source.stop(stopTime); } } else { this._onend(); } }; /** * invoked when the transport is stopped * @private */ Tone.Transport.prototype._onend = function(){ transportTicks = 0; this._setTicks(0); this.clearTimeouts(); this.state = TransportState.STOPPED; }; /** * pause the transport and all sources synced to the transport * * @param {Tone.Time} time */ Tone.Transport.prototype.pause = function(time){ if (this.state === TransportState.STARTED){ this.state = TransportState.PAUSED; var stopTime = this.toSeconds(time); this._clock.stop(stopTime); //call pause on each of the synced sources for (var i = 0; i < SyncedSources.length; i++){ var source = SyncedSources[i].source; source.pause(stopTime); } } }; /////////////////////////////////////////////////////////////////////////////// // SETTERS/GETTERS /////////////////////////////////////////////////////////////////////////////// /** * set the BPM * optionally ramp to the bpm over some time * @param {number} bpm * @param {Tone.Time=} rampTime */ Tone.Transport.prototype.setBpm = function(bpm, rampTime){ var quarterTime = this.notationToSeconds(tatum.toString() + "n", bpm, transportTimeSignature) / 4; this._clock.setRate(quarterTime, rampTime); }; /** * return the current BPM * * @return {number} */ Tone.Transport.prototype.getBpm = function(){ //convert the current frequency of the oscillator to bpm var freq = this._clock.getRate(); return 60 * (freq / tatum); }; /** * set the time signature * * @example * this.setTimeSignature(3, 8); // 3/8 * this.setTimeSignature(4); // 4/4 * * @param {number} numerator the numerator of the time signature * @param {number=} [denominator=4] the denominator of the time signature. this should * be a multiple of 2. */ Tone.Transport.prototype.setTimeSignature = function(numerator, denominator){ denominator = this.defaultArg(denominator, 4); transportTimeSignature = numerator / (denominator / 4); }; /** * return the time signature as just the numerator over 4. * for example 4/4 would return 4 and 6/8 would return 3 * * @return {number} */ Tone.Transport.prototype.getTimeSignature = function(){ return transportTimeSignature; }; /** * set the loop start position * * @param {Tone.Time} startPosition */ Tone.Transport.prototype.setLoopStart = function(startPosition){ loopStart = this.toTicks(startPosition); }; /** * set the loop start position * * @param {Tone.Time} endPosition */ Tone.Transport.prototype.setLoopEnd = function(endPosition){ loopEnd = this.toTicks(endPosition); }; /** * shorthand loop setting * @param {Tone.Time} startPosition * @param {Tone.Time} endPosition */ Tone.Transport.prototype.setLoopPoints = function(startPosition, endPosition){ this.setLoopStart(startPosition); this.setLoopEnd(endPosition); }; /** * set the amount of swing which is applied to the subdivision (defaults to 16th notes) * @param {number} amount a value between 0-1 where 1 equal to the note + half the subdivision */ Tone.Transport.prototype.setSwing = function(amount){ //scale the values to a normal range swingAmount = amount * 0.5; }; /** * set the subdivision which the swing will be applied to. the starting values is a 16th note. * * @example * Tone.Transport.setSwingSubdivision("8n"); //the eight note will be swing by the "swing amount" * * @param {string} subdivision the subdivision in notation (i.e. 8n, 16n, 8t). * value must be less than a quarter note. */ Tone.Transport.prototype.setSwingSubdivision = function(subdivision){ swingTatum = this.toTicks(subdivision); }; /////////////////////////////////////////////////////////////////////////////// // SYNCING /////////////////////////////////////////////////////////////////////////////// /** * Sync a source to the transport so that * @param {Tone.Source} source the source to sync to the transport * @param {Tone.Time} delay (optionally) start the source with a delay from the transport */ Tone.Transport.prototype.syncSource = function(source, startDelay){ SyncedSources.push({ source : source, delay : this.toSeconds(this.defaultArg(startDelay, 0)) }); }; /** * remove the source from the list of Synced Sources * * @param {Tone.Source} source [description] */ Tone.Transport.prototype.unsyncSource = function(source){ for (var i = 0; i < SyncedSources.length; i++){ if (SyncedSources[i].source === source){ SyncedSources.splice(i, 1); } } }; /** * attaches the signal to the tempo control signal so that * any changes in the tempo will change the signal in the same * ratio * * @param {Tone.Signal} signal */ Tone.Transport.prototype.syncSignal = function(signal){ //overreaching. fix this. signal.sync(this._clock._controlSignal); }; /** * clean up */ Tone.Transport.prototype.dispose = function(){ this._clock.dispose(); this._clock = null; }; /////////////////////////////////////////////////////////////////////////////// // TIMELINE EVENT /////////////////////////////////////////////////////////////////////////////// /** * @static * @type {number} */ var TimelineEventIDCounter = 0; /** * A Timeline event * * @constructor * @internal * @param {function(number)} callback * @param {Object} context * @param {number} tickTime * @param {number} startTicks */ var TimelineEvent = function(callback, context, tickTime, startTicks){ this.startTicks = startTicks; this.tickTime = tickTime; this.callback = callback; this.context = context; this.id = TimelineEventIDCounter++; }; /** * invoke the callback in the correct context * passes in the playback time * * @param {number} playbackTime */ TimelineEvent.prototype.doCallback = function(playbackTime){ this.callback.call(this.context, playbackTime); }; /** * get the tick which the callback is supposed to occur on * * @return {number} */ TimelineEvent.prototype.callbackTick = function(){ return this.startTicks + this.tickTime; }; /** * test if the tick occurs on the interval * * @param {number} tick * @return {boolean} */ TimelineEvent.prototype.testInterval = function(tick){ return (tick - this.startTicks) % this.tickTime === 0; }; /////////////////////////////////////////////////////////////////////////////// // AUGMENT TONE'S PROTOTYPE TO INCLUDE TRANSPORT TIMING /////////////////////////////////////////////////////////////////////////////// /** * tests if a string is musical notation * i.e.: * 4n = quarter note * 2m = two measures * 8t = eighth-note triplet * defined in "Tone/core/Transport" * * @return {boolean} * @method isNotation * @lends Tone.prototype.isNotation */ Tone.prototype.isNotation = (function(){ var notationFormat = new RegExp(/[0-9]+[mnt]$/i); return function(note){ return notationFormat.test(note); }; })(); /** * tests if a string is in Tick notation * defined in "Tone/core/Transport" * @return {boolean} * @method isTicks * @lends Tone.prototype.isNotation */ Tone.prototype.isTicks = (function(){ var tickFormat = new RegExp(/[0-9]+[i]$/i); return function(tick){ return tickFormat.test(tick); }; })(); /** * tests if a string is transportTime * i.e. : * 1:2:0 = 1 measure + two quarter notes + 0 sixteenth notes * defined in "Tone/core/Transport" * * @return {boolean} * * @method isTransportTime * @lends Tone.prototype.isTransportTime */ Tone.prototype.isTransportTime = (function(){ var transportTimeFormat = new RegExp(/^\d+(\.\d+)?:\d+(\.\d+)?(:\d+(\.\d+)?)?$/); return function(transportTime){ return transportTimeFormat.test(transportTime); }; })(); /** * true if the input is in the format number+hz * i.e.: 10hz * defined in "Tone/core/Transport" * * @param {number} freq * @return {boolean} * * @method isFrequency * @lends Tone.prototype.isFrequency */ Tone.prototype.isFrequency = (function(){ var freqFormat = new RegExp(/[0-9]+hz$/i); return function(freq){ return freqFormat.test(freq); }; })(); /** * * convert notation format strings to seconds * defined in "Tone/core/Transport" * * @param {string} notation * @param {number=} bpm * @param {number=} timeSignature * @return {number} * */ Tone.prototype.notationToSeconds = function(notation, bpm, timeSignature){ bpm = this.defaultArg(bpm, Tone.Transport.getBpm()); timeSignature = this.defaultArg(timeSignature, transportTimeSignature); var beatTime = (60 / bpm); var subdivision = parseInt(notation, 10); var beats = 0; if (subdivision === 0){ beats = 0; } var lastLetter = notation.slice(-1); if (lastLetter === "t"){ beats = (4 / subdivision) * 2/3; } else if (lastLetter === "n"){ beats = 4 / subdivision; } else if (lastLetter === "m"){ beats = subdivision * timeSignature; } else { beats = 0; } return beatTime * beats; }; /** * convert transportTime into seconds * defined in "Tone/core/Transport" * * ie: 4:2:3 == 4 measures + 2 quarters + 3 sixteenths * * @param {string} transportTime * @param {number=} bpm * @param {number=} timeSignature * @return {number} seconds * * @lends Tone.prototype.transportTimeToSeconds */ Tone.prototype.transportTimeToSeconds = function(transportTime, bpm, timeSignature){ bpm = this.defaultArg(bpm, Tone.Transport.getBpm()); timeSignature = this.defaultArg(timeSignature, transportTimeSignature); var measures = 0; var quarters = 0; var sixteenths = 0; var split = transportTime.split(":"); if (split.length === 2){ measures = parseFloat(split[0]); quarters = parseFloat(split[1]); } else if (split.length === 1){ quarters = parseFloat(split[0]); } else if (split.length === 3){ measures = parseFloat(split[0]); quarters = parseFloat(split[1]); sixteenths = parseFloat(split[2]); } var beats = (measures * timeSignature + quarters + sixteenths / 4); return beats * this.notationToSeconds("4n"); }; /** * convert ticks into seconds * defined in "Tone/core/Transport" * * @param {number} ticks * @param {number=} bpm * @param {number=} timeSignature * @return {number} seconds */ Tone.prototype.ticksToSeconds = function(ticks, bpm, timeSignature){ ticks = Math.floor(ticks); var quater = this.notationToSeconds("4n", bpm, timeSignature); return (quater * ticks) / (tatum); }; /** * Convert seconds to the closest transportTime in the form * measures:quarters:sixteenths * defined in "Tone/core/Transport" * * @method toTransportTime * * @param {Tone.Time} seconds * @param {number=} bpm * @param {number=} timeSignature * @return {string} * * @lends Tone.prototype.toTransportTime */ Tone.prototype.toTransportTime = function(time, bpm, timeSignature){ var seconds = this.toSeconds(time, bpm, timeSignature); bpm = this.defaultArg(bpm, Tone.Transport.getBpm()); timeSignature = this.defaultArg(timeSignature, transportTimeSignature); var quarterTime = this.notationToSeconds("4n"); var quarters = seconds / quarterTime; var measures = Math.floor(quarters / timeSignature); var sixteenths = Math.floor((quarters % 1) * 4); quarters = Math.floor(quarters) % timeSignature; var progress = [measures, quarters, sixteenths]; return progress.join(":"); }; /** * convert a time to a frequency * defined in "Tone/core/Transport" * * @param {Tone.Time} time * @return {number} the time in hertz */ Tone.prototype.toFrequency = function(time, now){ if (this.isFrequency(time)){ return parseFloat(time); } else if (this.isNotation(time) || this.isTransportTime(time)) { return this.secondsToFrequency(this.toSeconds(time, now)); } else { return time; } }; /** * convert Tone.Time into seconds. * defined in "Tone/core/Transport" * * unlike the method which it overrides, this takes into account * transporttime and musical notation * * Time : 1.40 * Notation: 4n|1m|2t * TransportTime: 2:4:1 (measure:quarters:sixteens) * Now Relative: +3n * Math: 3n+16n or even very complicated expressions ((3n*2)/6 + 1) * Ticks: "146i" * * @override * @param {Tone.Time} time * @param {number=} now if passed in, this number will be * used for all 'now' relative timings * @return {number} */ Tone.prototype.toSeconds = function(time, now){ now = this.defaultArg(now, this.now()); if (typeof time === "number"){ return time; //assuming that it's seconds } else if (typeof time === "string"){ var plusTime = 0; if(time.charAt(0) === "+") { plusTime = now; time = time.slice(1); } var components = time.split(/[\(\)\-\+\/\*]/); if (components.length > 1){ var oringalTime = time; for(var i = 0; i < components.length; i++){ var symb = components[i]; if (symb !== ""){ var val = this.toSeconds(symb.trim()); time = time.replace(symb, val); } } try { //i know eval is evil, but i think it's safe here time = eval(time); // jshint ignore:line } catch (e){ throw new EvalError("problem evaluating Tone.Time: "+oringalTime); } } else if (this.isNotation(time)){ time = this.notationToSeconds(time); } else if (this.isTransportTime(time)){ time = this.transportTimeToSeconds(time); } else if (this.isFrequency(time)){ time = this.frequencyToSeconds(time); } else { time = parseFloat(time); } return time + plusTime; } else { return now; } }; var TransportConstructor = Tone.Transport; Tone._initAudioContext(function(){ if (typeof Tone.Transport === "function"){ //a single transport object Tone.Transport = new Tone.Transport(); Tone.Transport.setBpm(120); } else { //stop the clock Tone.Transport.stop(); //get the previous bpm var bpm = Tone.Transport.getBpm(); //destory the old clock Tone.Transport._clock.dispose(); //make new Transport insides TransportConstructor.call(Tone.Transport); //set the bpm Tone.Transport.setBpm(bpm); } }); return Tone.Transport; });