mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-25 10:05:02 +00:00
6dd22e752f
addresses #959
987 lines
27 KiB
TypeScript
987 lines
27 KiB
TypeScript
import { expect } from "chai";
|
|
import { atTime, Offline, whenBetween } from "test/helper/Offline";
|
|
import { Time } from "Tone/core/type/Time";
|
|
import { noOp } from "Tone/core/util/Interface";
|
|
import { Signal } from "../../signal/Signal";
|
|
import { TransportTime } from "../type/TransportTime";
|
|
import { Transport } from "./Transport";
|
|
// importing for side affects
|
|
import "../context/Destination";
|
|
import { warns } from "test/helper/Basic";
|
|
import { Synth } from "Tone/instrument/Synth";
|
|
|
|
describe("Transport", () => {
|
|
|
|
context("BPM and timeSignature", () => {
|
|
|
|
it("can get and set bpm", () => {
|
|
return Offline((context) => {
|
|
const transport = new Transport({ context });
|
|
transport.bpm.value = 125;
|
|
expect(transport.bpm.value).to.be.closeTo(125, 0.001);
|
|
transport.bpm.value = 120;
|
|
expect(transport.bpm.value).to.equal(120);
|
|
});
|
|
});
|
|
|
|
it("can get and set timeSignature as both an array or number", () => {
|
|
return Offline((context) => {
|
|
const transport = new Transport({ context });
|
|
transport.timeSignature = [6, 8];
|
|
expect(transport.timeSignature).to.equal(3);
|
|
transport.timeSignature = 5;
|
|
expect(transport.timeSignature).to.equal(5);
|
|
});
|
|
});
|
|
|
|
it("can get and set timeSignature as both an array or number", () => {
|
|
return Offline((context) => {
|
|
const transport = new Transport({ context });
|
|
transport.timeSignature = [6, 8];
|
|
expect(transport.timeSignature).to.equal(3);
|
|
transport.timeSignature = 5;
|
|
expect(transport.timeSignature).to.equal(5);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
context("looping", () => {
|
|
|
|
it("can get and set loop points", () => {
|
|
return Offline((context) => {
|
|
const transport = new Transport({ context });
|
|
transport.loopStart = 0.2;
|
|
transport.loopEnd = 0.4;
|
|
expect(transport.loopStart).to.be.closeTo(0.2, 0.01);
|
|
expect(transport.loopEnd).to.be.closeTo(0.4, 0.01);
|
|
transport.setLoopPoints(0, "1m");
|
|
expect(transport.loopStart).to.be.closeTo(0, 0.01);
|
|
expect(transport.loopEnd).to.be.closeTo(transport.toSeconds("1m"), 0.01);
|
|
});
|
|
});
|
|
|
|
it("can loop events scheduled on the transport", () => {
|
|
let invocations = 0;
|
|
return Offline((context) => {
|
|
const transport = new Transport({ context });
|
|
transport.schedule((time) => {
|
|
invocations++;
|
|
}, 0);
|
|
transport.setLoopPoints(0, 0.1).start(0);
|
|
transport.loop = true;
|
|
}, 0.41).then(() => {
|
|
expect(invocations).to.equal(5);
|
|
});
|
|
});
|
|
|
|
it("jumps to the loopStart after the loopEnd point", () => {
|
|
let looped = false;
|
|
return Offline((context) => {
|
|
const transport = new Transport({ context });
|
|
transport.on("loop", () => {
|
|
looped = true;
|
|
});
|
|
transport.loop = true;
|
|
transport.loopEnd = 1;
|
|
transport.seconds = 2;
|
|
transport.start();
|
|
}, 0.4).then(() => {
|
|
expect(looped).to.equal(true);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
context("nextSubdivision", () => {
|
|
|
|
it("returns 0 if the transports not started", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
expect(transport.nextSubdivision()).to.equal(0);
|
|
});
|
|
});
|
|
|
|
it("can get the next subdivision of the transport", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.start(0);
|
|
return time => {
|
|
whenBetween(time, 0.05, 0.07, () => {
|
|
expect(transport.nextSubdivision(0.5)).to.be.closeTo(0.5, 0.01);
|
|
expect(transport.nextSubdivision(0.04)).to.be.closeTo(0.08, 0.01);
|
|
expect(transport.nextSubdivision(2)).to.be.closeTo(2, 0.01);
|
|
});
|
|
whenBetween(time, 0.09, 0.1, () => {
|
|
expect(transport.nextSubdivision(0.04)).to.be.closeTo(0.12, 0.01);
|
|
expect(transport.nextSubdivision("8n")).to.be.closeTo(0.25, 0.01);
|
|
});
|
|
};
|
|
}, 0.1);
|
|
});
|
|
|
|
});
|
|
|
|
context("PPQ", () => {
|
|
|
|
it("can get and set pulses per quarter", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.PPQ = 96;
|
|
expect(transport.PPQ).to.equal(96);
|
|
});
|
|
});
|
|
|
|
it("schedules a quarter note at the same time with a different PPQ", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.PPQ = 1;
|
|
const id = transport.schedule(time => {
|
|
expect(time).to.be.closeTo(transport.toSeconds("4n"), 0.1);
|
|
transport.clear(id);
|
|
}, "4n");
|
|
transport.start();
|
|
});
|
|
});
|
|
|
|
it("invokes the right number of ticks with a different PPQ", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.bpm.value = 120;
|
|
const ppq = 20;
|
|
transport.PPQ = ppq;
|
|
transport.start();
|
|
|
|
return time => {
|
|
if (time > 0.5) {
|
|
expect(transport.ticks).to.be.within(ppq, ppq * 1.2);
|
|
}
|
|
};
|
|
}, 0.55);
|
|
});
|
|
|
|
});
|
|
|
|
context("position", () => {
|
|
|
|
it("can jump to a specific tick number", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.ticks = 200;
|
|
expect(transport.ticks).to.equal(200);
|
|
transport.start(0);
|
|
let tested = false;
|
|
return () => {
|
|
if (!tested) {
|
|
expect(transport.ticks).to.at.least(200);
|
|
tested = true;
|
|
}
|
|
};
|
|
}, 0.1);
|
|
});
|
|
|
|
it("can get the current position in BarsBeatsSixteenths", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
expect(transport.position).to.equal("0:0:0");
|
|
transport.start(0);
|
|
return atTime(0.05, () => {
|
|
expect(transport.position).to.not.equal("0:0:0");
|
|
});
|
|
}, 0.1);
|
|
});
|
|
|
|
it("can get the current position in seconds", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
expect(transport.seconds).to.equal(0);
|
|
transport.start(0.05);
|
|
return time => {
|
|
if (time > 0.05) {
|
|
expect(transport.seconds).to.be.closeTo(time - 0.05, 0.01);
|
|
}
|
|
};
|
|
}, 0.1);
|
|
});
|
|
|
|
it("can get the current position in seconds during a bpm ramp", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
expect(transport.seconds).to.equal(0);
|
|
transport.start(0.05);
|
|
transport.bpm.linearRampTo(60, 0.5, 0.5);
|
|
return time => {
|
|
if (time > 0.05) {
|
|
expect(transport.seconds).to.be.closeTo(time - 0.05, 0.01);
|
|
}
|
|
};
|
|
}, 0.7);
|
|
});
|
|
|
|
it("can set the current position in seconds", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
expect(transport.seconds).to.equal(0);
|
|
transport.seconds = 3;
|
|
expect(transport.seconds).to.be.closeTo(3, 0.01);
|
|
});
|
|
});
|
|
|
|
it("can set the current position in BarsBeatsSixteenths", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
expect(transport.position).to.equal("0:0:0");
|
|
transport.position = "3:0";
|
|
expect(transport.position).to.equal("3:0:0");
|
|
transport.position = "0:0";
|
|
expect(transport.position).to.equal("0:0:0");
|
|
});
|
|
});
|
|
|
|
it("can get the progress of the loop", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.setLoopPoints(0, "1m").start();
|
|
transport.loop = true;
|
|
expect(transport.progress).to.be.equal(0);
|
|
transport.position = "2n";
|
|
expect(transport.progress).to.be.closeTo(0.5, 0.001);
|
|
transport.position = Time("2n").valueOf() + Time("4n").valueOf();
|
|
expect(transport.progress).to.be.closeTo(0.75, 0.001);
|
|
});
|
|
});
|
|
|
|
it("progress is always 0 when not looping", () => {
|
|
return Offline(({ transport }) => {
|
|
transport.loop = false;
|
|
transport.start();
|
|
return atTime(0.1, () => {
|
|
expect(transport.progress).to.be.equal(0);
|
|
});
|
|
}, 0.2);
|
|
});
|
|
|
|
});
|
|
|
|
context("state", () => {
|
|
|
|
it("can start, pause, and restart", () => {
|
|
return Offline(({ transport }) => {
|
|
transport.start(0).pause(0.2).start(0.4);
|
|
|
|
const pulse = new Signal(0).toDestination();
|
|
|
|
transport.schedule(time => {
|
|
pulse.setValueAtTime(1, time);
|
|
pulse.setValueAtTime(0, time + 0.1);
|
|
}, 0);
|
|
|
|
transport.schedule(time => {
|
|
pulse.setValueAtTime(1, time);
|
|
pulse.setValueAtTime(0, time + 0.1);
|
|
}, 0.3);
|
|
|
|
return time => {
|
|
whenBetween(time, 0, 0.2, () => {
|
|
expect(transport.state).to.equal("started");
|
|
});
|
|
|
|
whenBetween(time, 0.2, 0.4, () => {
|
|
expect(transport.state).to.equal("paused");
|
|
});
|
|
|
|
whenBetween(time, 0.4, Infinity, () => {
|
|
expect(transport.state).to.equal("started");
|
|
});
|
|
};
|
|
}, 0.6).then(buffer => {
|
|
|
|
buffer.forEach((sample, time) => {
|
|
whenBetween(time, 0, 0.01, () => {
|
|
expect(sample).to.equal(1);
|
|
});
|
|
whenBetween(time, 0.1, 0.11, () => {
|
|
expect(sample).to.equal(0);
|
|
});
|
|
whenBetween(time, 0.502, 0.51, () => {
|
|
expect(sample).to.equal(1);
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
context("ticks", () => {
|
|
|
|
it("resets ticks on stop but not on pause", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.start(0).pause(0.1).stop(0.2);
|
|
expect(transport.getTicksAtTime(0)).to.be.equal(Math.floor(transport.PPQ * 0));
|
|
expect(transport.getTicksAtTime(0.05)).to.be.equal(Math.floor(transport.PPQ * 0.1));
|
|
expect(transport.getTicksAtTime(0.1)).to.be.equal(Math.floor(transport.PPQ * 0.2));
|
|
expect(transport.getTicksAtTime(0.15)).to.be.equal(Math.floor(transport.PPQ * 0.2));
|
|
expect(transport.getTicksAtTime(0.2)).to.be.equal(0);
|
|
}, 0.3);
|
|
});
|
|
|
|
it("tracks ticks after start", () => {
|
|
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.bpm.value = 120;
|
|
const ppq = transport.PPQ;
|
|
transport.start();
|
|
|
|
return time => {
|
|
if (time > 0.5) {
|
|
expect(transport.ticks).to.at.least(ppq);
|
|
}
|
|
};
|
|
}, 0.6);
|
|
});
|
|
|
|
it("can start with a tick offset", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.start(0, "200i");
|
|
|
|
return time => {
|
|
if (time < 0.01) {
|
|
expect(transport.ticks).to.at.least(200);
|
|
}
|
|
};
|
|
}, 0.1);
|
|
});
|
|
|
|
it("can toggle the state of the transport", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.toggle(0);
|
|
transport.toggle(0.2);
|
|
|
|
return time => {
|
|
whenBetween(time, 0, 0.2, () => {
|
|
expect(transport.state).to.equal("started");
|
|
});
|
|
|
|
whenBetween(time, 0.2, Infinity, () => {
|
|
expect(transport.state).to.equal("stopped");
|
|
});
|
|
};
|
|
}, 0.1);
|
|
});
|
|
|
|
it("tracks ticks correctly with a different PPQ and BPM", () => {
|
|
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.PPQ = 96;
|
|
transport.bpm.value = 90;
|
|
transport.start();
|
|
|
|
return time => {
|
|
if (time > 0.5) {
|
|
expect(transport.ticks).to.at.least(72);
|
|
}
|
|
};
|
|
}, 0.6);
|
|
});
|
|
|
|
it("can set the ticks while started", () => {
|
|
let invocations = 0;
|
|
const times = [0, 1.5];
|
|
return Offline(({ transport }) => {
|
|
transport.PPQ = 1;
|
|
transport.schedule(time => {
|
|
expect(time).to.be.closeTo(times[invocations], 0.01);
|
|
invocations++;
|
|
}, 0);
|
|
transport.start(0);
|
|
return atTime(1.1, () => {
|
|
transport.ticks = 0;
|
|
});
|
|
}, 2.5).then(() => {
|
|
expect(invocations).to.equal(2);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
context("schedule", () => {
|
|
|
|
it("can schedule an event on the timeline", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const eventID = transport.schedule(() => { }, 0);
|
|
expect(eventID).to.be.a("number");
|
|
});
|
|
});
|
|
|
|
it("scheduled event gets invoked with the time of the event", () => {
|
|
let wasCalled = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = 0.1;
|
|
transport.schedule(time => {
|
|
expect(time).to.be.closeTo(startTime, 0.01);
|
|
wasCalled = true;
|
|
}, 0);
|
|
transport.start(startTime);
|
|
}, 0.2).then(() => {
|
|
expect(wasCalled).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can schedule events with TransportTime", () => {
|
|
let wasCalled = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = 0.1;
|
|
const eighth = transport.toSeconds("8n");
|
|
transport.schedule(time => {
|
|
expect(time).to.be.closeTo(startTime + eighth, 0.01);
|
|
wasCalled = true;
|
|
}, TransportTime("8n"));
|
|
transport.start(startTime);
|
|
}, 0.5).then(() => {
|
|
expect(wasCalled).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can clear a scheduled event", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const eventID = transport.schedule(() => {
|
|
throw new Error("should not call this function");
|
|
}, 0);
|
|
transport.clear(eventID);
|
|
transport.start();
|
|
});
|
|
});
|
|
|
|
it("can cancel the timeline of scheduled object", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.schedule(() => {
|
|
throw new Error("should not call this");
|
|
}, 0);
|
|
transport.cancel(0);
|
|
transport.start(0);
|
|
});
|
|
});
|
|
|
|
it("can cancel the timeline of scheduleOnce object", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.scheduleOnce(() => {
|
|
throw new Error("should not call this");
|
|
}, 0);
|
|
transport.cancel(0);
|
|
transport.start(0);
|
|
});
|
|
});
|
|
|
|
it("scheduled event anywhere along the timeline", () => {
|
|
let wasCalled = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = transport.now();
|
|
transport.schedule(time => {
|
|
expect(time).to.be.closeTo(startTime + 0.5, 0.001);
|
|
wasCalled = true;
|
|
}, 0.5);
|
|
transport.start(startTime);
|
|
}, 0.6).then(() => {
|
|
expect(wasCalled).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can schedule multiple events and invoke them in the right order", () => {
|
|
let wasCalled = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
let first = false;
|
|
transport.schedule(() => {
|
|
first = true;
|
|
}, 0.1);
|
|
transport.schedule(() => {
|
|
expect(first).to.equal(true);
|
|
wasCalled = true;
|
|
}, 0.11);
|
|
transport.start();
|
|
}, 0.2).then(() => {
|
|
expect(wasCalled).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("invokes the event again if the timeline is restarted", () => {
|
|
let iterations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.schedule(() => {
|
|
iterations++;
|
|
}, 0.05);
|
|
transport.start(0).stop(0.1).start(0.2);
|
|
}, 0.3).then(() => {
|
|
expect(iterations).to.be.equal(2);
|
|
});
|
|
});
|
|
|
|
it("can add an event after the Transport is started", () => {
|
|
let wasCalled = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.start(0);
|
|
let wasScheduled = false;
|
|
return time => {
|
|
if (time > 0.1 && !wasScheduled) {
|
|
wasScheduled = true;
|
|
transport.schedule(() => {
|
|
wasCalled = true;
|
|
}, 0.15);
|
|
}
|
|
};
|
|
}, 0.3).then(() => {
|
|
expect(wasCalled).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("warns if the scheduled time was not used in the callback", async () => {
|
|
return Offline(({ transport }) => {
|
|
const synth = new Synth();
|
|
transport.schedule(() => {
|
|
warns(() => {
|
|
synth.triggerAttackRelease("C2", 0.1);
|
|
});
|
|
}, 0);
|
|
transport.start(0);
|
|
}, 0.3).then(() => {
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
context("scheduleRepeat", () => {
|
|
|
|
it("can schedule a repeated event", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const eventID = transport.scheduleRepeat(noOp, 1);
|
|
expect(eventID).to.be.a("number");
|
|
});
|
|
});
|
|
|
|
it("scheduled event gets invoked with the time of the event", () => {
|
|
let invoked = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = 0.1;
|
|
const eventID = transport.scheduleRepeat(time => {
|
|
expect(time).to.be.closeTo(startTime, 0.01);
|
|
invoked = true;
|
|
transport.clear(eventID);
|
|
}, 1, 0);
|
|
transport.start(startTime);
|
|
}, 0.3).then(() => {
|
|
expect(invoked).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can cancel the timeline of scheduleRepeat", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.scheduleRepeat(() => {
|
|
throw new Error("should not call this");
|
|
}, 0.01, 0);
|
|
transport.cancel(0);
|
|
transport.start(0);
|
|
});
|
|
});
|
|
|
|
it("can schedule events with TransportTime", () => {
|
|
let invoked = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = 0.1;
|
|
const eighth = transport.toSeconds("8n");
|
|
transport.scheduleRepeat(time => {
|
|
expect(time).to.be.closeTo(startTime + eighth, 0.01);
|
|
invoked = true;
|
|
}, "1n", TransportTime("8n"));
|
|
transport.start(startTime);
|
|
}, 0.4).then(() => {
|
|
expect(invoked).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can clear a scheduled event", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const eventID = transport.scheduleRepeat(() => {
|
|
throw new Error("should not call this function");
|
|
}, 1, 0);
|
|
transport.clear(eventID);
|
|
transport.stop();
|
|
});
|
|
});
|
|
|
|
it("can be scheduled in the future", () => {
|
|
let invoked = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = 0.1;
|
|
const eventID = transport.scheduleRepeat(time => {
|
|
transport.clear(eventID);
|
|
expect(time).to.be.closeTo(startTime + 0.2, 0.01);
|
|
invoked = true;
|
|
}, 1, 0.2);
|
|
transport.start(startTime);
|
|
}, 0.5).then(() => {
|
|
expect(invoked).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("repeats a repeat event", () => {
|
|
let invocations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.scheduleRepeat(() => {
|
|
invocations++;
|
|
}, 0.1, 0);
|
|
transport.start();
|
|
}, 0.51).then(() => {
|
|
expect(invocations).to.equal(6);
|
|
});
|
|
});
|
|
|
|
it("repeats at the repeat interval", () => {
|
|
let wasCalled = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
let repeatTime = -1;
|
|
transport.scheduleRepeat(time => {
|
|
if (repeatTime !== -1) {
|
|
expect(time - repeatTime).to.be.closeTo(0.1, 0.01);
|
|
}
|
|
repeatTime = time;
|
|
wasCalled = true;
|
|
}, 0.1, 0);
|
|
transport.start();
|
|
}, 0.5).then(() => {
|
|
expect(wasCalled).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can schedule multiple events and invoke them in the right order", () => {
|
|
let first = false;
|
|
let second = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const firstID = transport.scheduleRepeat(() => {
|
|
first = true;
|
|
transport.clear(firstID);
|
|
}, 1, 0.1);
|
|
const secondID = transport.scheduleRepeat(() => {
|
|
transport.clear(secondID);
|
|
expect(first).to.equal(true);
|
|
second = true;
|
|
}, 1, 0.11);
|
|
transport.start();
|
|
}, 0.3).then(() => {
|
|
expect(first);
|
|
expect(second);
|
|
});
|
|
});
|
|
|
|
it("repeats for the given interval", () => {
|
|
let repeatCount = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.scheduleRepeat(time => {
|
|
repeatCount++;
|
|
}, 0.1, 0, 0.5);
|
|
transport.start();
|
|
}, 0.61).then(() => {
|
|
expect(repeatCount).to.equal(5);
|
|
});
|
|
});
|
|
|
|
it("can add an event after the Transport is started", () => {
|
|
let invocations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.start(0);
|
|
let wasScheduled = false;
|
|
const times = [0.15, 0.3];
|
|
return time => {
|
|
if (time > 0.1 && !wasScheduled) {
|
|
wasScheduled = true;
|
|
transport.scheduleRepeat(repeatedTime => {
|
|
expect(repeatedTime).to.be.closeTo(times[invocations], 0.01);
|
|
invocations++;
|
|
}, 0.15, 0.15);
|
|
}
|
|
};
|
|
}, 0.31).then(() => {
|
|
expect(invocations).to.equal(2);
|
|
});
|
|
});
|
|
|
|
it("can add an event to the past after the Transport is started", () => {
|
|
let invocations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.start(0);
|
|
let wasScheduled = false;
|
|
const times = [0.15, 0.25];
|
|
return time => {
|
|
if (time >= 0.12 && !wasScheduled) {
|
|
wasScheduled = true;
|
|
transport.scheduleRepeat(repeatedTime => {
|
|
expect(repeatedTime).to.be.closeTo(times[invocations], 0.01);
|
|
invocations++;
|
|
}, 0.1, 0.05);
|
|
}
|
|
};
|
|
}, 0.3).then(() => {
|
|
expect(invocations).to.equal(2);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
context("scheduleOnce", () => {
|
|
|
|
it("can schedule a single event on the timeline", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const eventID = transport.scheduleOnce(() => {}, 0);
|
|
expect(eventID).to.be.a("number");
|
|
});
|
|
});
|
|
|
|
it("scheduled event gets invoked with the time of the event", () => {
|
|
let invoked = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = 0.1;
|
|
const eventID = transport.scheduleOnce(time => {
|
|
invoked = true;
|
|
transport.clear(eventID);
|
|
expect(time).to.be.closeTo(startTime, 0.01);
|
|
}, 0);
|
|
transport.start(startTime);
|
|
}, 0.2).then(() => {
|
|
expect(invoked).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can schedule events with TransportTime", () => {
|
|
let invoked = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = 0.1;
|
|
const eighth = transport.toSeconds("8n");
|
|
transport.scheduleOnce(time => {
|
|
expect(time).to.be.closeTo(startTime + eighth, 0.01);
|
|
invoked = true;
|
|
}, TransportTime("8n"));
|
|
transport.start(startTime);
|
|
}, 0.5).then(() => {
|
|
expect(invoked).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("can clear a scheduled event", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const eventID = transport.scheduleOnce(() => {
|
|
throw new Error("should not call this function");
|
|
}, 0);
|
|
transport.clear(eventID);
|
|
transport.start();
|
|
});
|
|
});
|
|
|
|
it("can be scheduled in the future", () => {
|
|
let invoked = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const startTime = transport.now() + 0.1;
|
|
const eventID = transport.scheduleOnce(time => {
|
|
transport.clear(eventID);
|
|
expect(time).to.be.closeTo(startTime + 0.3, 0.01);
|
|
invoked = true;
|
|
}, 0.3);
|
|
transport.start(startTime);
|
|
}, 0.5).then(() => {
|
|
expect(invoked).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("the event is removed after is is invoked", () => {
|
|
let iterations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.scheduleOnce(() => {
|
|
iterations++;
|
|
}, 0);
|
|
transport.start().stop("+0.1").start("+0.2");
|
|
}, 0.5).then(() => {
|
|
expect(iterations).to.be.lessThan(2);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
context("events", () => {
|
|
|
|
it("invokes start/stop/pause events", () => {
|
|
let invocations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.on("start", () => {
|
|
invocations++;
|
|
});
|
|
transport.on("stop", () => {
|
|
invocations++;
|
|
});
|
|
transport.on("pause", () => {
|
|
invocations++;
|
|
});
|
|
transport.start().stop(0.1).start(0.2);
|
|
}, 0.5).then(() => {
|
|
expect(invocations).to.equal(3);
|
|
});
|
|
});
|
|
|
|
it("invokes start event with correct offset", () => {
|
|
let wasCalled = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.on("start", (time, offset) => {
|
|
expect(time).to.be.closeTo(0.2, 0.01);
|
|
expect(offset).to.be.closeTo(0.5, 0.001);
|
|
wasCalled = true;
|
|
});
|
|
transport.start(0.2, "4n");
|
|
}, 0.3).then(() => {
|
|
expect(wasCalled).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("invokes the event just before the scheduled time", () => {
|
|
let invoked = false;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.on("start", (time, offset) => {
|
|
expect(time - transport.context.currentTime).to.be.closeTo(0, 0.01);
|
|
expect(offset).to.equal(0);
|
|
invoked = true;
|
|
});
|
|
transport.start(0.2);
|
|
}, 0.3).then(() => {
|
|
expect(invoked).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it("passes in the time argument to the events", () => {
|
|
let invocations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const now = transport.now();
|
|
transport.on("start", time => {
|
|
invocations++;
|
|
expect(time).to.be.closeTo(now + 0.1, 0.01);
|
|
});
|
|
transport.on("stop", time => {
|
|
invocations++;
|
|
expect(time).to.be.closeTo(now + 0.2, 0.01);
|
|
});
|
|
transport.start("+0.1").stop("+0.2");
|
|
}, 0.3).then(() => {
|
|
expect(invocations).to.equal(2);
|
|
});
|
|
});
|
|
|
|
it("invokes the 'loop' method on loop", () => {
|
|
let loops = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
const sixteenth = transport.toSeconds("16n");
|
|
transport.setLoopPoints(0, sixteenth);
|
|
transport.loop = true;
|
|
let lastLoop = -1;
|
|
transport.on("loop", time => {
|
|
loops++;
|
|
if (lastLoop !== -1) {
|
|
expect(time - lastLoop).to.be.closeTo(sixteenth, 0.001);
|
|
}
|
|
lastLoop = time;
|
|
});
|
|
transport.start(0).stop(sixteenth * 5.1);
|
|
}, 0.7).then(() => {
|
|
expect(loops).to.equal(5);
|
|
});
|
|
});
|
|
});
|
|
|
|
context("swing", () => {
|
|
|
|
it("can get/set the swing subdivision", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.swingSubdivision = "8n";
|
|
expect(transport.swingSubdivision).to.equal("8n");
|
|
transport.swingSubdivision = "4n";
|
|
expect(transport.swingSubdivision).to.equal("4n");
|
|
});
|
|
});
|
|
|
|
it("can get/set the swing amount", () => {
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.swing = 0.5;
|
|
expect(transport.swing).to.equal(0.5);
|
|
transport.swing = 0;
|
|
expect(transport.swing).to.equal(0);
|
|
});
|
|
});
|
|
|
|
it("can swing", () => {
|
|
let invocations = 0;
|
|
return Offline(context => {
|
|
const transport = new Transport({ context });
|
|
transport.swing = 1;
|
|
transport.swingSubdivision = "8n";
|
|
const eightNote = transport.toSeconds("8n");
|
|
// downbeat, no swing
|
|
transport.schedule(time => {
|
|
invocations++;
|
|
expect(time).is.closeTo(0, 0.001);
|
|
}, 0);
|
|
// eighth note has swing
|
|
transport.schedule(time => {
|
|
invocations++;
|
|
expect(time).is.closeTo(eightNote * 5 / 3, 0.001);
|
|
}, "8n");
|
|
// sixteenth note is also swung
|
|
transport.schedule(time => {
|
|
invocations++;
|
|
expect(time).is.closeTo(eightNote, 0.05);
|
|
}, "16n");
|
|
// no swing on the quarter
|
|
transport.schedule(time => {
|
|
invocations++;
|
|
expect(time).is.closeTo(eightNote * 2, 0.001);
|
|
}, "4n");
|
|
transport.start(0).stop(0.7);
|
|
}, 0.7).then(() => {
|
|
expect(invocations).to.equal(4);
|
|
});
|
|
});
|
|
});
|
|
|
|
});
|