Tone.js/test/core/Clock.js
Yotam Mann 84c1affe83 Clock uses static lookahead
lookAhead is more conservatively computed and adjusts to the largest
value.
2016-11-21 00:05:19 -08:00

374 lines
No EOL
9.5 KiB
JavaScript

define(["Test", "Tone/core/Clock", "helper/Offline2"], function (Test, Clock, Offline) {
describe("Clock", function(){
it ("can be created and disposed", function(){
var clock = new Clock();
clock.dispose();
Test.wasDisposed(clock);
});
context("Get/Set values", function(){
it ("can get and set the frequency", function(){
var clock = new Clock(function(){}, 2);
expect(clock.frequency.value).to.equal(2);
clock.frequency.value = 0.2;
expect(clock.frequency.value).to.be.closeTo(0.2, 0.001);
clock.dispose();
});
it ("invokes the callback when started", function(done){
var clock = new Clock(function(){
clock.dispose();
done();
}, 10).start();
});
it ("can be constructed with an options object", function(done){
var clock = new Clock({
"callback" : function(){
clock.dispose();
done();
},
"frequency" : 8
}).start();
expect(clock.frequency.value).to.equal(8);
});
it ("can get and set it's values with the set/get", function(){
var clock = new Clock();
clock.set({
"frequency" : 2
});
var gotValues = clock.get();
expect(gotValues.frequency).to.equal(2);
clock.dispose();
});
});
context("State", function(){
it ("correctly returns the scheduled play state", function(done){
Offline(function(output, testFn, tearDown){
var clock = new Clock();
expect(clock.state).to.equal("stopped");
clock.start().stop(0.5);
expect(clock.state).to.equal("started");
tearDown(function(){
expect(clock.state).to.equal("stopped");
clock.dispose();
done();
});
}, 0.6);
});
it("can start, pause, and stop", function(done){
Offline(function(output, testFn, tearDown){
var clock = new Clock();
expect(clock.state).to.equal("stopped");
clock.start().pause(0.2).stop(0.4);
expect(clock.state).to.equal("started");
testFn(function(sample, time){
if (time >= 0.2 && time < 0.4){
expect(clock.state).to.equal("paused");
} else if (time >= 0.4){
expect(clock.state).to.equal("stopped");
}
});
tearDown(function(){
clock.dispose();
done();
});
}, 0.6);
});
it("can start and stop in close proximity", function(done){
Offline(function(output, testFn, tearDown){
var clock = new Clock();
expect(clock.state).to.equal("stopped");
clock.start(0).stop(0.001).start(0.002);
expect(clock.state).to.equal("started");
testFn(function(sample, time){
if (time < 0.001){
expect(clock.state).to.equal("started");
} else if (time >= 0.002){
expect(clock.state).to.equal("started");
} else {
expect(clock.state).to.equal("stopped");
}
});
tearDown(function(){
clock.dispose();
done();
});
}, 0.05);
});
it("can schedule multiple start and stops", function(done){
Offline(function(output, testFn, tearDown){
var clock = new Clock();
expect(clock.state).to.equal("stopped");
clock.start().pause(0.2).stop(0.4).start(0.6).stop(0.8);
expect(clock.state).to.equal("started");
testFn(function(sample, time){
if (time >= 0.2 && time < 0.4){
expect(clock.state).to.equal("paused");
} else if (time >= 0.4 && time < 0.6){
expect(clock.state).to.equal("stopped");
} else if (time >= 0.6 && time < 0.8){
expect(clock.state).to.equal("started");
} else if (time >= 0.8){
expect(clock.state).to.equal("stopped");
}
});
tearDown(function(){
clock.dispose();
done();
});
}, 0.9);
});
});
context("Scheduling", function(){
it ("passes a time to the callback", function(done){
var clock = new Clock(function(time){
expect(time).to.be.a.number;
clock.dispose();
done();
}, 10).start();
});
it ("invokes the callback with a time great than now", function(done){
var clock = new Clock(function(time){
clock.dispose();
expect(time).to.be.greaterThan(now);
done();
}, 10);
var now = clock.now();
var startTime = now + 0.1;
clock.start(startTime);
});
it ("invokes the first callback at the given start time", function(done){
var clock = new Clock(function(time){
clock.dispose();
expect(time).to.equal(startTime);
done();
}, 10);
var startTime = clock.now() + 0.1;
clock.start(startTime);
});
it ("can be scheduled to stop in the future", function(done){
Offline(function(output, testFn, tearDown){
var invokations = 0;
var clock = new Clock(function(){
invokations++;
}, 0.5).start(0);
tearDown(function(){
expect(invokations).to.equal(1);
clock.dispose();
done();
});
}, 0.6);
});
it ("invokes the right number of callbacks given the duration", function(done){
Offline(function(output, testFn, tearDown){
var invokations = 0;
var clock = new Clock(function(){
invokations++;
}, 10).start(0);
tearDown(function(){
expect(invokations).to.equal(5);
clock.dispose();
done();
});
}, 0.45);
});
it ("can schedule the frequency of the clock", function(done){
Offline(function(output, testFn, tearDown){
var invokations = 0;
var clock = new Clock(function(){
invokations++;
}, 2);
clock.start(0).stop(1.1);
clock.frequency.setValueAtTime(4, 0.5);
tearDown(function(){
expect(invokations).to.equal(4);
clock.dispose();
done();
});
}, 2);
});
});
context("Ticks", function(){
it ("has 0 ticks when first created", function(){
var clock = new Clock();
expect(clock.ticks).to.equal(0);
clock.dispose();
});
it ("increments 1 tick per callback", function(done){
Offline(function(output, testFn, tearDown){
var ticks = 0;
var clock = new Clock(function(){
ticks++;
}, 0.05).start();
tearDown(function(){
expect(ticks).to.equal(clock.ticks);
clock.dispose();
done();
});
}, 0.6);
});
it ("resets ticks on stop", function(done){
Offline(function(output, testFn, tearDown){
var clock = new Clock(function(){}, 20).start(0).stop(0.5);
testFn(function(sample, time){
if (time > 0.1 && time < 0.5){
expect(clock.ticks).to.be.above(0);
}
});
tearDown(function(){
expect(clock.ticks).to.equal(0);
clock.dispose();
done();
});
}, 0.6);
});
it ("does not reset ticks on pause but stops incrementing", function(done){
Offline(function(output, testFn, tearDown){
var clock = new Clock(function(){}, 20).start().pause(0.3);
var pausedTicks = 0;
testFn(function(sample, time){
if (time > 0.05 && time < 0.3){
expect(clock.ticks).to.be.above(0);
pausedTicks = clock.ticks;
} else if (time >= 0.3){
expect(clock.ticks).to.equal(pausedTicks);
}
});
tearDown(function(){
clock.dispose();
done();
});
}, 0.6);
});
it ("starts incrementing where it left off after pause", function(done){
Offline(function(output, testFn, tearDown){
var clock = new Clock(function(){}, 20).start(0).pause(0.3).start(0.5);
var pausedTicks = 0;
var restarted = false;
testFn(function(sample, time){
if (time < 0.3){
expect(clock.ticks).to.be.above(0);
pausedTicks = clock.ticks;
} else if (time >= 0.7 && !restarted){
restarted = true;
expect(clock.ticks).to.equal(pausedTicks + 1);
}
});
tearDown(function(){
clock.dispose();
done();
});
}, 0.6);
});
it ("can start with a tick offset", function(done){
var clock = new Clock(function(){
expect(clock.ticks).to.equal(4);
clock.dispose();
done();
}, 10);
expect(clock.ticks).to.equal(0);
clock.start(undefined, 4);
});
});
context("Events", function(){
it ("triggers the start event on start", function(done){
var clock = new Clock(function(){}, 20);
var startTime = clock.now() + 0.3;
clock.on("start", function(time, offset){
expect(time).to.be.closeTo(startTime, 0.01);
expect(clock.now()).to.be.closeTo(startTime, 0.1);
expect(offset).to.equal(0);
clock.dispose();
done();
});
clock.start(startTime);
});
it ("triggers the start event with an offset", function(done){
var clock = new Clock(function(){}, 20);
var startTime = clock.now() + 0.3;
clock.on("start", function(time, offset){
expect(time).to.be.closeTo(startTime, 0.01);
expect(clock.now()).to.be.closeTo(startTime, 0.1);
expect(offset).to.equal(2);
clock.dispose();
done();
});
clock.start(startTime, 2);
});
it ("triggers stop event", function(done){
var clock = new Clock(function(){}, 20);
var stopTime = clock.now() + 0.3;
clock.on("stop", function(time){
expect(time).to.be.closeTo(stopTime, 0.01);
expect(clock.now()).to.be.closeTo(stopTime, 0.1);
clock.dispose();
done();
});
clock.start().stop(stopTime);
});
it ("triggers pause stop event", function(done){
var clock = new Clock(function(){}, 20);
var now = clock.now();
clock.on("pause", function(time){
expect(time).to.be.closeTo(now + 0.1, 0.01);
}).on("stop", function(time){
expect(time).to.be.closeTo(now + 0.2, 0.01);
clock.dispose();
done();
});
clock.start().pause("+0.1").stop("+0.2");
});
});
});
});