mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-25 10:05:02 +00:00
665 lines
No EOL
16 KiB
JavaScript
665 lines
No EOL
16 KiB
JavaScript
define(["helper/Basic", "Tone/event/Part", "Tone/core/Tone",
|
|
"Tone/core/Transport", "Tone/event/Event", "helper/Offline2"],
|
|
function (Basic, Part, Tone, Transport, Event, Offline) {
|
|
|
|
describe("Part", function(){
|
|
|
|
Basic(Part);
|
|
|
|
function resetTransport(done){
|
|
Tone.Transport.cancel(0);
|
|
Tone.Transport.off("start stop pause loop");
|
|
Tone.Transport.stop();
|
|
Tone.Transport.loop = false;
|
|
Tone.Transport.PPQ = 48;
|
|
Tone.Transport.bpm.value = 120;
|
|
Tone.Transport.timeSignature = [4, 4];
|
|
setTimeout(done, 200);
|
|
}
|
|
|
|
context("Constructor", function(){
|
|
|
|
afterEach(resetTransport);
|
|
|
|
it ("takes a callback and an array of values", function(){
|
|
var callback = function(){};
|
|
var part = new Part(callback, [0, 1, 2]);
|
|
expect(part.callback).to.equal(callback);
|
|
expect(part.length).to.equal(3);
|
|
part.dispose();
|
|
});
|
|
|
|
it ("can be constructed with no arguments", function(){
|
|
var part = new Part();
|
|
expect(part.length).to.equal(0);
|
|
part.dispose();
|
|
});
|
|
|
|
it ("can pass in arguments in options object", function(){
|
|
var callback = function(){};
|
|
var part = new Part({
|
|
"callback" : callback,
|
|
"humanize" : true,
|
|
"events" : [0, 1, 2],
|
|
"loop" : true,
|
|
"loopEnd" : "4n",
|
|
"probability" : 0.3
|
|
});
|
|
expect(part.callback).to.equal(callback);
|
|
expect(part.length).to.equal(3);
|
|
expect(part.loop).to.be.true;
|
|
expect(part.loopEnd).to.equal("4n");
|
|
expect(part.probability).to.equal(0.3);
|
|
expect(part.humanize).to.be.true;
|
|
part.dispose();
|
|
});
|
|
});
|
|
|
|
context("Adding / Removing / Getting Events", function(){
|
|
|
|
it("can take events in the constructor as an array of times", function(){
|
|
var part = new Part(function(){}, ["0", "8n", "4n"]);
|
|
expect(part.length).to.equal(3);
|
|
part.dispose();
|
|
});
|
|
|
|
it("can take events in the constructor as an array of times and values", function(){
|
|
var part = new Part(function(){}, [["0", "C4"], ["8n", "D3"], ["4n", "E4"]]);
|
|
expect(part.length).to.equal(3);
|
|
part.dispose();
|
|
});
|
|
|
|
it ("can retrieve an event using 'at'", function(){
|
|
var part = new Part(function(){}, [["0", 0], ["8n", "C2"], ["4n", 2]]);
|
|
expect(part.length).to.equal(3);
|
|
expect(part.at(0)).to.be.instanceof(Event);
|
|
expect(part.at(0).value).to.equal(0);
|
|
expect(part.at("8n").value).to.equal("C2");
|
|
expect(part.at("4n").value).to.equal(2);
|
|
expect(part.at("2n")).to.be.null;
|
|
part.dispose();
|
|
});
|
|
|
|
it("can set the value of an existing event with 'at'", function(){
|
|
var part = new Part({
|
|
"events" : [[0, "C3"]]
|
|
});
|
|
expect(part.length).to.equal(1);
|
|
expect(part.at(0).value).to.equal("C3");
|
|
part.at(0, "C4");
|
|
expect(part.at(0).value).to.equal("C4");
|
|
part.dispose();
|
|
});
|
|
|
|
it("can take events in the constructor as an array of objects", function(){
|
|
var part = new Part(function(){}, [{
|
|
"time" : 0.3,
|
|
"note" : "C3"
|
|
},
|
|
{
|
|
"time" : 1,
|
|
"note" : "D3"
|
|
}]);
|
|
expect(part.length).to.equal(2);
|
|
expect(part.at(0.3).value).to.deep.equal({note : "C3"});
|
|
part.dispose();
|
|
});
|
|
|
|
it("can add an event as a time and value", function(){
|
|
var part = new Part();
|
|
expect(part.length).to.equal(0);
|
|
part.add(1, "D3");
|
|
expect(part.length).to.equal(1);
|
|
expect(part.at(1).value).to.equal("D3");
|
|
part.dispose();
|
|
});
|
|
|
|
it("can add an event as an object", function(){
|
|
var part = new Part();
|
|
expect(part.length).to.equal(0);
|
|
part.add({
|
|
"time" : 0.5,
|
|
"note" : "D4",
|
|
"duration" : "8n"
|
|
});
|
|
expect(part.length).to.equal(1);
|
|
expect(part.at(0.5).value).to.deep.equal({"note" : "D4", "duration" : "8n"});
|
|
part.dispose();
|
|
});
|
|
|
|
it("can add another part", function(){
|
|
var part = new Part();
|
|
expect(part.length).to.equal(0);
|
|
var subPart = new Part({
|
|
"events" : [0, 0.5]
|
|
});
|
|
part.add(0.2, subPart);
|
|
expect(part.length).to.equal(1);
|
|
expect(part.at(0.2)).to.equal(subPart);
|
|
part.dispose();
|
|
});
|
|
|
|
it("can remove an event by time", function(){
|
|
var part = new Part({
|
|
"events" : [[0.2, "C3"], [0.2, "C4"]]
|
|
});
|
|
expect(part.length).to.equal(2);
|
|
part.remove(0.2);
|
|
expect(part.length).to.equal(0);
|
|
part.dispose();
|
|
});
|
|
|
|
it("can remove an event by time and value", function(){
|
|
var part = new Part({
|
|
"events" : [[0.2, "C3"], [0.2, "C4"]]
|
|
});
|
|
expect(part.length).to.equal(2);
|
|
part.remove(0.2, "C4");
|
|
expect(part.length).to.equal(1);
|
|
part.dispose();
|
|
});
|
|
|
|
it("added events have the same settings as the parent", function(){
|
|
var part = new Part({
|
|
"loopEnd" : "1m",
|
|
"loopStart" : "4n",
|
|
"humanize" : 0.1,
|
|
"probability" : 0.2,
|
|
"events" : [[0.2, "C3"], [0.3, "C4"]]
|
|
});
|
|
var firstEvent = part.at(0.2);
|
|
expect(firstEvent.humanize).to.equal(0.1);
|
|
expect(firstEvent.probability).to.equal(0.2);
|
|
//loop duration is the same
|
|
expect(firstEvent.loopEnd).to.equal("1m");
|
|
expect(firstEvent.loopStart).to.equal("4n");
|
|
|
|
var secondEvent = part.at(0.3);
|
|
expect(secondEvent.humanize).to.equal(0.1);
|
|
expect(secondEvent.probability).to.equal(0.2);
|
|
//loop duration is the same
|
|
expect(secondEvent.loopEnd).to.equal("1m");
|
|
expect(secondEvent.loopStart).to.equal("4n");
|
|
part.dispose();
|
|
});
|
|
|
|
it("will create an event using at if one wasn't there at that time", function(){
|
|
var part = new Part();
|
|
expect(part.length).to.equal(0);
|
|
expect(part.at(0.1, "C4").value).to.equal("C4");
|
|
expect(part.length).to.equal(1);
|
|
part.dispose();
|
|
});
|
|
|
|
it("can remove all of the events", function(){
|
|
var part = new Part(function(){}, [0, 1, 2, 3, 4, 5]);
|
|
expect(part.length).to.equal(6);
|
|
part.removeAll();
|
|
expect(part.length).to.equal(0);
|
|
part.dispose();
|
|
});
|
|
|
|
});
|
|
|
|
context("Part callback", function(){
|
|
|
|
afterEach(resetTransport);
|
|
|
|
it ("does not invoke get invoked until started", function(done){
|
|
Offline(function(dest, test, after){
|
|
|
|
var part = new Event(function(){
|
|
throw new Error("shouldn't call this callback");
|
|
}, [0, 0.4]);
|
|
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.5);
|
|
});
|
|
|
|
it ("is invoked after it's started", function(done){
|
|
var part = new Part(function(){
|
|
part.dispose();
|
|
done();
|
|
}, [0, 1]).start(0);
|
|
Tone.Transport.start();
|
|
});
|
|
|
|
it ("passes in the scheduled time to the callback", function(done){
|
|
var now = Tone.Transport.now() + 0.1;
|
|
var part = new Part(function(time){
|
|
expect(time).to.be.a.number;
|
|
expect(time - now).to.be.closeTo(0.5, 0.01);
|
|
part.dispose();
|
|
done();
|
|
}, [0.3]);
|
|
part.start(0.2);
|
|
Tone.Transport.start(now);
|
|
});
|
|
|
|
it ("passes in the value to the callback", function(done){
|
|
var part = new Part(function(time, thing){
|
|
expect(time).to.be.a.number;
|
|
expect(thing).to.equal("thing");
|
|
part.dispose();
|
|
done();
|
|
}, [[0, "thing"]]).start();
|
|
Tone.Transport.start();
|
|
});
|
|
|
|
it ("can mute the callback", function(done){
|
|
Offline(function(dest, test, after){
|
|
|
|
var part = new Part(function(){
|
|
throw new Error("shouldn't call this callback");
|
|
}, [0, 0.1, 0.2, 0.3]).start();
|
|
|
|
part.mute = true;
|
|
expect(part.mute).to.be.true;
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it ("can trigger with some probability", function(done){
|
|
Offline(function(output, test, after){
|
|
|
|
var part = new Part(function(){
|
|
throw new Error("shouldn't call this callback");
|
|
}, [0, 1, 2, 3]).start();
|
|
|
|
part.probability = 0;
|
|
expect(part.probability).to.equal(0);
|
|
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.4);
|
|
});
|
|
|
|
|
|
it ("invokes all of the scheduled events", function(done){
|
|
Offline(function(output, test, after){
|
|
|
|
var count = 0;
|
|
var part = new Part(function(){
|
|
count++;
|
|
}, [0, 0.1, 0.2, 0.3]).start();
|
|
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
expect(count).to.equal(4);
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.4);
|
|
});
|
|
|
|
it ("invokes all of the scheduled events at the correct times", function(done){
|
|
Offline(function(output, test, after){
|
|
|
|
var count = 0;
|
|
var now = Tone.Transport.now() + 0.1;
|
|
|
|
var part = new Part(function(time, value){
|
|
count++;
|
|
expect(time - now).to.be.closeTo(value, 0.01);
|
|
}, [[0, 0], [0.1, 0.1], [0.2, 0.2]]).start();
|
|
|
|
Tone.Transport.start(now);
|
|
|
|
after(function(){
|
|
expect(count).to.equal(3);
|
|
part.dispose();
|
|
done();
|
|
});
|
|
}, 0.4);
|
|
});
|
|
|
|
it ("starts an event added after the part was started", function(done){
|
|
var part = new Part({
|
|
"loopEnd" : 0.2,
|
|
"loop" : true,
|
|
"callback" : function(time, value){
|
|
if (value === 1){
|
|
part.dispose();
|
|
done();
|
|
}
|
|
},
|
|
"events" : [[0, 0]]
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
setTimeout(function(){
|
|
part.add(0.1, 1);
|
|
}, 100);
|
|
});
|
|
|
|
it("can schedule a subpart", function(done){
|
|
var startTime = Tone.Transport.now() + 0.1;
|
|
var subPart = new Part({
|
|
"events" : [[0, 1], [0.3, 2]]
|
|
});
|
|
var part = new Part(function(time, value){
|
|
if (value === 0){
|
|
expect(time - startTime).to.be.closeTo(0, 0.01);
|
|
} else if (value === 1){
|
|
expect(time - startTime).to.be.closeTo(0.2, 0.01);
|
|
} else if (value === 2){
|
|
expect(time - startTime).to.be.closeTo(0.5, 0.01);
|
|
part.dispose();
|
|
done();
|
|
}
|
|
}).add(0.2, subPart).add(0, 0).start(0);
|
|
Tone.Transport.start(startTime);
|
|
});
|
|
|
|
it("can start with an offset", function(done){
|
|
var now = Tone.Transport.now() + 0.1;
|
|
var part = new Part(function(time, number){
|
|
expect(time - now).to.be.closeTo(0.1, 0.01);
|
|
expect(number).to.equal(1);
|
|
part.dispose();
|
|
done();
|
|
}, [[0, 0], [1, 1]]).start(0, 0.9);
|
|
Tone.Transport.start(now);
|
|
});
|
|
|
|
});
|
|
|
|
context("Looping", function(){
|
|
|
|
afterEach(resetTransport);
|
|
|
|
it ("can be set to loop", function(done){
|
|
|
|
Offline(function(output, test, after){
|
|
|
|
var callCount = 0;
|
|
var part = new Part({
|
|
"loopEnd" : 0.2,
|
|
"loop" : true,
|
|
"callback" : function(){
|
|
callCount++;
|
|
},
|
|
"events" : [[0, 1], [0.1, 2]]
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
expect(callCount).to.equal(6);
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.55);
|
|
});
|
|
|
|
it ("can be set to loop at a specific interval", function(done){
|
|
Offline(function(output, test, after){
|
|
|
|
var lastCall;
|
|
var part = new Part({
|
|
"loopEnd" : 0.25,
|
|
"loop" : true,
|
|
"callback" : function(time){
|
|
if (lastCall){
|
|
expect(time - lastCall).to.be.closeTo(0.25, 0.01);
|
|
}
|
|
lastCall = time;
|
|
},
|
|
events : [0]
|
|
}).start(0);
|
|
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.7);
|
|
});
|
|
|
|
it ("a started part will be stopped if it is after the loopEnd", function(done){
|
|
Offline(function(output, test, after){
|
|
|
|
var switched = false;
|
|
var part = new Part({
|
|
"loopEnd" : 0.5,
|
|
"loop" : true,
|
|
"callback" : function(time, value){
|
|
if (value === 1){
|
|
switched = true;
|
|
part.loopEnd = 0.2;
|
|
} else if (switched){
|
|
expect(value).to.equal(0);
|
|
}
|
|
},
|
|
events : [[0, 0], [0.25, 1]]
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.7);
|
|
});
|
|
|
|
it ("a started part will be stopped if it is before the loopStart", function(done){
|
|
|
|
Offline(function(output, test, after){
|
|
|
|
var switched = false;
|
|
var part = new Part({
|
|
"loopEnd" : 0.5,
|
|
"loop" : true,
|
|
"callback" : function(time, value){
|
|
if (value === 1){
|
|
switched = true;
|
|
part.loopStart = 0.2;
|
|
} else if (switched){
|
|
expect(value).to.equal(1);
|
|
}
|
|
},
|
|
events : [[0, 0], [0.25, 1]]
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.7);
|
|
});
|
|
|
|
|
|
it ("can loop a specific number of times", function(done){
|
|
|
|
Offline(function(output, test, after){
|
|
|
|
var callCount = 0;
|
|
var part = new Part({
|
|
"loopEnd" : 0.125,
|
|
"loop" : 3,
|
|
"callback" : function(){
|
|
callCount++;
|
|
},
|
|
events : [0, 0.1]
|
|
}).start(0.1);
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
expect(callCount).to.equal(6);
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.8);
|
|
});
|
|
|
|
it ("can loop between loopStart and loopEnd", function(done){
|
|
|
|
Offline(function(output, test, after){
|
|
var part = new Part({
|
|
"loopStart" : "8n",
|
|
"loopEnd" : "4n",
|
|
"loop" : true,
|
|
"callback" : function(time, value){
|
|
expect(value).to.be.at.least(1);
|
|
expect(value).to.be.at.most(2);
|
|
},
|
|
events : [[0, 0], ["8n", 1], ["8n + 16n", 2], ["4n", 3]]
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
|
|
}, 0.8);
|
|
});
|
|
|
|
|
|
it ("reports the progress of the loop", function(done){
|
|
|
|
Offline(function(output, test, after){
|
|
|
|
var callCount = 0;
|
|
var part = new Part({
|
|
"loopEnd" : 1,
|
|
"loop" : true,
|
|
"callback" : function(){
|
|
callCount++;
|
|
}
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
after(function(){
|
|
expect(part.progress).to.be.closeTo(0.8, 0.05);
|
|
part.dispose();
|
|
done();
|
|
});
|
|
}, 0.8);
|
|
});
|
|
|
|
it("can start a loop with an offset", function(done){
|
|
var iteration = 0;
|
|
var part = new Part(function(time, number){
|
|
if (iteration === 0){
|
|
expect(number).to.equal(1);
|
|
} else if (iteration === 1){
|
|
expect(number).to.equal(0);
|
|
part.dispose();
|
|
done();
|
|
}
|
|
iteration++;
|
|
}, [[0, 0], [0.25, 1]]);
|
|
part.loop = true;
|
|
part.loopEnd = 0.5;
|
|
part.start(0, 0.25);
|
|
Tone.Transport.start();
|
|
});
|
|
|
|
it("can start a loop with an offset before loop start", function(done){
|
|
var iteration = 0;
|
|
var part = new Part(function(time, number){
|
|
if (iteration === 0){
|
|
expect(number).to.equal(0);
|
|
} else if (iteration === 1){
|
|
expect(number).to.equal(1);
|
|
} else if (iteration === 2){
|
|
expect(number).to.equal(2);
|
|
} else if (iteration === 3){
|
|
expect(number).to.equal(1);
|
|
} else if (iteration === 4){
|
|
expect(number).to.equal(2);
|
|
part.dispose();
|
|
done();
|
|
}
|
|
iteration++;
|
|
}, [[0, 0], [0.25, 1], [0.30, 2]]);
|
|
part.loop = true;
|
|
part.loopStart = 0.25;
|
|
part.loopEnd = 0.5;
|
|
part.start(0, 0);
|
|
Tone.Transport.start();
|
|
});
|
|
|
|
});
|
|
|
|
context("playbackRate", function(){
|
|
|
|
afterEach(resetTransport);
|
|
|
|
it ("can adjust the playbackRate", function(done){
|
|
|
|
Offline(function(output, test, after){
|
|
var lastCall;
|
|
var part = new Part({
|
|
"playbackRate" : 2,
|
|
"loopEnd" : 1,
|
|
"loop" : true,
|
|
"events" : [0, 0.5],
|
|
"callback" : function(time){
|
|
if (lastCall){
|
|
expect(time - lastCall).to.be.closeTo(0.25, 0.01);
|
|
}
|
|
lastCall = time;
|
|
}
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
}, 0.7);
|
|
});
|
|
|
|
it ("can adjust the playbackRate after starting", function(done){
|
|
Offline(function(output, test, after){
|
|
|
|
var lastCall;
|
|
var part = new Part({
|
|
"playbackRate" : 1,
|
|
"loopEnd" : 0.5,
|
|
"loop" : true,
|
|
"events" : [0, 0.25],
|
|
"callback" : function(time){
|
|
if (lastCall){
|
|
expect(time - lastCall).to.be.closeTo(0.5, 0.01);
|
|
} else {
|
|
part.playbackRate = 0.5;
|
|
}
|
|
lastCall = time;
|
|
}
|
|
}).start(0);
|
|
Tone.Transport.start();
|
|
|
|
after(function(){
|
|
part.dispose();
|
|
done();
|
|
});
|
|
}, 0.8);
|
|
});
|
|
});
|
|
});
|
|
}); |