2018-03-05 16:58:46 +00:00
|
|
|
define(["Tone/core/Tone", "Tone/core/Transport", "Tone/core/Buffer", "Tone/core/OfflineContext", "Tone/core/Master"], function(Tone){
|
2018-01-22 21:19:26 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Because of a bug in iOS causing the currentTime to increment
|
|
|
|
* before the rendering is started, sometimes it takes multiple
|
2018-03-05 16:58:46 +00:00
|
|
|
* attempts to render the audio correctly.
|
2018-01-22 21:19:26 +00:00
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
function attemptRender(callback, duration, sampleRate, tries){
|
|
|
|
tries = Tone.defaultArg(tries, 0);
|
|
|
|
var context = new Tone.OfflineContext(2, duration, sampleRate);
|
|
|
|
Tone.context = context;
|
|
|
|
|
|
|
|
//invoke the callback/scheduling
|
|
|
|
var response = callback(Tone.Transport);
|
|
|
|
|
|
|
|
if (context.currentTime > 0 && tries < 1000){
|
|
|
|
return attemptRender(callback, duration, sampleRate, ++tries);
|
|
|
|
} else {
|
|
|
|
return {
|
|
|
|
"response" : response,
|
|
|
|
"context" : context
|
2018-01-22 21:34:47 +00:00
|
|
|
};
|
2018-01-22 21:19:26 +00:00
|
|
|
}
|
|
|
|
}
|
2017-02-19 16:50:34 +00:00
|
|
|
|
|
|
|
/**
|
2017-10-21 23:02:46 +00:00
|
|
|
* Generate a buffer by rendering all of the Tone.js code within the callback using the OfflineAudioContext.
|
|
|
|
* The OfflineAudioContext is capable of rendering much faster than real time in many cases.
|
2017-03-22 15:15:46 +00:00
|
|
|
* The callback function also passes in an offline instance of Tone.Transport which can be used
|
2018-04-06 23:56:11 +00:00
|
|
|
* to schedule events along the Transport. **NOTE** OfflineAudioContext has the same restrictions
|
|
|
|
* as the AudioContext in that on certain platforms (like iOS) it must be invoked by an explicit
|
|
|
|
* user action like a click or tap.
|
2017-02-19 16:50:34 +00:00
|
|
|
* @param {Function} callback All Tone.js nodes which are created and scheduled within this callback are recorded into the output Buffer.
|
|
|
|
* @param {Time} duration the amount of time to record for.
|
|
|
|
* @return {Promise} The promise which is invoked with the Tone.Buffer of the recorded output.
|
|
|
|
* @example
|
|
|
|
* //render 2 seconds of the oscillator
|
|
|
|
* Tone.Offline(function(){
|
|
|
|
* //only nodes created in this callback will be recorded
|
|
|
|
* var oscillator = new Tone.Oscillator().toMaster().start(0)
|
|
|
|
* //schedule their events
|
|
|
|
* }, 2).then(function(buffer){
|
|
|
|
* //do something with the output buffer
|
|
|
|
* })
|
|
|
|
* @example
|
|
|
|
* //can also schedule events along the Transport
|
2017-03-22 15:15:46 +00:00
|
|
|
* //using the passed in Offline Transport
|
|
|
|
* Tone.Offline(function(Transport){
|
2017-03-26 16:32:41 +00:00
|
|
|
* var osc = new Tone.Oscillator().toMaster()
|
2017-03-22 15:15:46 +00:00
|
|
|
* Transport.schedule(function(time){
|
2017-02-19 16:50:34 +00:00
|
|
|
* osc.start(time).stop(time + 0.1)
|
|
|
|
* }, 1)
|
2017-03-22 15:15:46 +00:00
|
|
|
* Transport.start(0.2)
|
2017-02-19 16:50:34 +00:00
|
|
|
* }, 4).then(function(buffer){
|
|
|
|
* //do something with the output buffer
|
|
|
|
* })
|
|
|
|
*/
|
|
|
|
Tone.Offline = function(callback, duration){
|
|
|
|
//set the OfflineAudioContext
|
|
|
|
var sampleRate = Tone.context.sampleRate;
|
|
|
|
var originalContext = Tone.context;
|
|
|
|
|
2018-01-22 21:19:26 +00:00
|
|
|
var renderRet = attemptRender(callback, duration, sampleRate);
|
|
|
|
var response = renderRet.response;
|
|
|
|
var context = renderRet.context;
|
2017-02-19 16:50:34 +00:00
|
|
|
|
2018-01-03 17:02:10 +00:00
|
|
|
var ret;
|
|
|
|
if (response instanceof Promise){
|
|
|
|
//wait for the promise to resolve
|
|
|
|
ret = response.then(function(){
|
|
|
|
//then render the audio
|
|
|
|
return context.render();
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
//process the audio
|
|
|
|
ret = context.render();
|
|
|
|
}
|
2017-02-19 16:50:34 +00:00
|
|
|
|
|
|
|
//return the original AudioContext
|
|
|
|
Tone.context = originalContext;
|
|
|
|
|
|
|
|
//return the audio
|
2018-01-03 17:02:10 +00:00
|
|
|
return ret.then(function(buffer){
|
2017-02-19 16:50:34 +00:00
|
|
|
//wrap it in a Tone.Buffer
|
|
|
|
return new Tone.Buffer(buffer);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
return Tone.Offline;
|
2017-10-21 23:02:46 +00:00
|
|
|
});
|