2019-05-23 18:00:49 +00:00
|
|
|
import { expect } from "chai";
|
2024-05-03 18:31:14 +00:00
|
|
|
import { ConstantOutput } from "../../../test/helper/ConstantOutput.js";
|
|
|
|
import { Offline } from "../../../test/helper/Offline.js";
|
|
|
|
import { TransportClass } from "../clock/Transport.js";
|
|
|
|
import { getContext } from "../Global.js";
|
|
|
|
import { createAudioContext } from "./AudioContext.js";
|
|
|
|
import { Context } from "./Context.js";
|
|
|
|
import { DestinationClass } from "./Destination.js";
|
|
|
|
import { ListenerClass } from "./Listener.js";
|
|
|
|
import { DrawClass } from "../util/Draw.js";
|
|
|
|
import { connect } from "./ToneAudioNode.js";
|
2019-05-23 18:00:49 +00:00
|
|
|
|
|
|
|
describe("Context", () => {
|
2019-10-31 15:41:21 +00:00
|
|
|
it("creates and disposes the classes attached to the context", async () => {
|
2019-12-22 03:05:24 +00:00
|
|
|
const ac = createAudioContext();
|
2019-10-31 15:41:21 +00:00
|
|
|
const context = new Context(ac);
|
|
|
|
const ctxDest = context.destination;
|
|
|
|
const ctxDraw = context.draw;
|
|
|
|
const ctxTransport = context.transport;
|
2019-12-22 03:05:24 +00:00
|
|
|
const ctxListener = context.listener;
|
2024-04-28 17:05:26 +00:00
|
|
|
expect(context.destination).is.instanceOf(DestinationClass);
|
|
|
|
expect(context.draw).is.instanceOf(DrawClass);
|
|
|
|
expect(context.listener).is.instanceOf(ListenerClass);
|
2019-10-31 15:41:21 +00:00
|
|
|
await context.close();
|
|
|
|
expect(ctxDest.disposed).to.be.true;
|
|
|
|
expect(ctxDraw.disposed).to.be.true;
|
|
|
|
expect(ctxTransport.disposed).to.be.true;
|
2019-12-22 03:05:24 +00:00
|
|
|
expect(ctxListener.disposed).to.be.true;
|
2019-10-31 15:41:21 +00:00
|
|
|
context.dispose();
|
|
|
|
});
|
|
|
|
|
2019-05-23 18:00:49 +00:00
|
|
|
context("AudioContext", () => {
|
|
|
|
it("extends the AudioContext methods", () => {
|
2019-10-14 03:20:42 +00:00
|
|
|
const ctx = new Context(createAudioContext());
|
2019-05-23 18:00:49 +00:00
|
|
|
expect(ctx).to.have.property("createGain");
|
2019-08-19 18:15:53 +00:00
|
|
|
expect(ctx.createGain()).to.have.property("gain");
|
2019-05-23 18:00:49 +00:00
|
|
|
expect(ctx).to.have.property("createOscillator");
|
2019-08-19 18:15:53 +00:00
|
|
|
expect(ctx.createOscillator()).to.be.have.property("frequency");
|
2019-05-23 18:00:49 +00:00
|
|
|
expect(ctx).to.have.property("createDelay");
|
2019-08-19 18:15:53 +00:00
|
|
|
expect(ctx.createDelay()).to.be.have.property("delayTime");
|
2019-05-23 18:00:49 +00:00
|
|
|
expect(ctx).to.have.property("createConstantSource");
|
|
|
|
ctx.dispose();
|
2020-05-12 16:31:17 +00:00
|
|
|
return ctx.close();
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
|
2021-01-01 01:53:38 +00:00
|
|
|
it("can be stringified", () => {
|
|
|
|
const ctx = new Context(createAudioContext());
|
|
|
|
expect(JSON.stringify(ctx)).to.equal("{}");
|
|
|
|
ctx.dispose();
|
|
|
|
return ctx.close();
|
|
|
|
});
|
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("clock is running", (done) => {
|
|
|
|
const interval = setInterval(() => {
|
|
|
|
if (getContext().currentTime > 0.5) {
|
|
|
|
clearInterval(interval);
|
|
|
|
done();
|
|
|
|
}
|
|
|
|
}, 20);
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
|
|
|
it("has a rawContext", () => {
|
2019-10-14 03:20:42 +00:00
|
|
|
const ctx = new Context(createAudioContext());
|
2019-08-19 18:15:53 +00:00
|
|
|
expect(ctx.rawContext).has.property("destination");
|
|
|
|
expect(ctx.rawContext).has.property("sampleRate");
|
2019-10-14 03:20:42 +00:00
|
|
|
ctx.dispose();
|
2020-05-12 16:31:17 +00:00
|
|
|
return ctx.close();
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("can be constructed with an options object", () => {
|
|
|
|
const ctx = new Context({
|
|
|
|
clockSource: "timeout",
|
2020-05-12 16:31:17 +00:00
|
|
|
latencyHint: "playback",
|
2019-05-23 18:00:49 +00:00
|
|
|
lookAhead: 0.2,
|
2024-05-03 18:31:14 +00:00
|
|
|
updateInterval: 0.1,
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
expect(ctx.lookAhead).to.equal(0.2);
|
2021-12-01 17:19:03 +00:00
|
|
|
expect(ctx.updateInterval).to.equal(0.1);
|
2020-05-12 16:31:17 +00:00
|
|
|
expect(ctx.latencyHint).to.equal("playback");
|
2019-05-23 18:00:49 +00:00
|
|
|
expect(ctx.clockSource).to.equal("timeout");
|
2019-10-14 03:20:42 +00:00
|
|
|
ctx.dispose();
|
2020-05-12 16:31:17 +00:00
|
|
|
return ctx.close();
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
2020-05-12 16:31:17 +00:00
|
|
|
|
2019-10-29 18:29:52 +00:00
|
|
|
it("returns 'now' and 'immediate' time", () => {
|
|
|
|
const ctx = new Context();
|
|
|
|
expect(ctx.now()).to.be.a("number");
|
|
|
|
expect(ctx.immediate()).to.be.a("number");
|
|
|
|
ctx.dispose();
|
2020-05-12 16:31:17 +00:00
|
|
|
return ctx.close();
|
2019-10-29 18:29:52 +00:00
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
context("state", () => {
|
2019-09-16 03:32:40 +00:00
|
|
|
it("can suspend and resume the state", async () => {
|
2019-12-22 03:05:24 +00:00
|
|
|
const ac = createAudioContext();
|
2019-05-23 18:00:49 +00:00
|
|
|
const context = new Context(ac);
|
|
|
|
expect(context.rawContext).to.equal(ac);
|
|
|
|
await ac.suspend();
|
|
|
|
expect(context.state).to.equal("suspended");
|
|
|
|
await context.resume();
|
|
|
|
expect(context.state).to.equal("running");
|
|
|
|
context.dispose();
|
2019-10-31 15:41:21 +00:00
|
|
|
return context.close();
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("invokes the statechange event", async () => {
|
2019-12-22 03:05:24 +00:00
|
|
|
const ac = createAudioContext();
|
2019-05-23 18:00:49 +00:00
|
|
|
const context = new Context(ac);
|
|
|
|
let triggerChange = false;
|
2021-01-01 01:53:38 +00:00
|
|
|
context.on("statechange", (state) => {
|
2019-05-23 18:00:49 +00:00
|
|
|
if (!triggerChange) {
|
|
|
|
triggerChange = true;
|
|
|
|
expect(state).to.equal("running");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
await context.resume();
|
2021-10-13 17:32:29 +00:00
|
|
|
await new Promise<void>((done) => setTimeout(() => done(), 10));
|
2019-05-23 18:00:49 +00:00
|
|
|
expect(triggerChange).to.equal(true);
|
2022-03-01 17:38:26 +00:00
|
|
|
return context.dispose();
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
context("clockSource", () => {
|
|
|
|
let ctx;
|
|
|
|
beforeEach(() => {
|
|
|
|
ctx = new Context();
|
|
|
|
return ctx.resume();
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
afterEach(() => {
|
|
|
|
ctx.dispose();
|
|
|
|
return ctx.close();
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("defaults to 'worker'", () => {
|
|
|
|
expect(ctx.clockSource).to.equal("worker");
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("provides callback", (done) => {
|
|
|
|
expect(ctx.clockSource).to.equal("worker");
|
|
|
|
ctx.setTimeout(() => {
|
|
|
|
done();
|
|
|
|
}, 0.1);
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("can be set to 'timeout'", (done) => {
|
|
|
|
ctx.clockSource = "timeout";
|
|
|
|
expect(ctx.clockSource).to.equal("timeout");
|
|
|
|
ctx.setTimeout(() => {
|
|
|
|
done();
|
|
|
|
}, 0.1);
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("can be set to 'offline'", (done) => {
|
|
|
|
ctx.clockSource = "offline";
|
|
|
|
expect(ctx.clockSource).to.equal("offline");
|
|
|
|
// provides no callback
|
|
|
|
ctx.setTimeout(() => {
|
|
|
|
throw new Error("shouldn't be called");
|
|
|
|
}, 0.1);
|
|
|
|
setTimeout(() => {
|
|
|
|
done();
|
|
|
|
}, 200);
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
2024-05-03 18:31:14 +00:00
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
context("setTimeout", () => {
|
2024-05-03 18:31:14 +00:00
|
|
|
let ctx;
|
|
|
|
beforeEach(() => {
|
|
|
|
ctx = new Context();
|
|
|
|
return ctx.resume();
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
afterEach(() => {
|
|
|
|
ctx.dispose();
|
|
|
|
return ctx.close();
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("can set a timeout", (done) => {
|
|
|
|
ctx.setTimeout(() => {
|
|
|
|
done();
|
|
|
|
}, 0.1);
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("returns an id", () => {
|
|
|
|
expect(ctx.setTimeout(() => {}, 0.1)).to.be.a("number");
|
|
|
|
// try clearing a random ID, shouldn't cause any errors
|
|
|
|
ctx.clearTimeout(-2);
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("timeout is not invoked when cancelled", (done) => {
|
|
|
|
const id = ctx.setTimeout(() => {
|
|
|
|
throw new Error("shouldn't be invoked");
|
|
|
|
}, 0.01);
|
|
|
|
ctx.clearTimeout(id);
|
|
|
|
ctx.setTimeout(() => {
|
|
|
|
done();
|
|
|
|
}, 0.02);
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("order is maintained", (done) => {
|
|
|
|
let wasInvoked = false;
|
|
|
|
ctx.setTimeout(() => {
|
|
|
|
expect(wasInvoked).to.equal(true);
|
|
|
|
done();
|
|
|
|
}, 0.02);
|
|
|
|
ctx.setTimeout(() => {
|
|
|
|
wasInvoked = true;
|
|
|
|
}, 0.01);
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
|
|
|
|
it("is invoked in the offline context", () => {
|
2021-01-01 01:53:38 +00:00
|
|
|
return Offline((context) => {
|
2024-04-28 17:05:26 +00:00
|
|
|
const transport = new TransportClass({ context });
|
2019-05-23 18:00:49 +00:00
|
|
|
transport.context.setTimeout(() => {
|
2019-06-18 01:50:43 +00:00
|
|
|
expect(transport.now()).to.be.closeTo(0.01, 0.005);
|
2019-05-23 18:00:49 +00:00
|
|
|
}, 0.01);
|
|
|
|
}, 0.05);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2019-09-06 18:46:44 +00:00
|
|
|
context("setInterval", () => {
|
2024-05-03 18:31:14 +00:00
|
|
|
let ctx;
|
|
|
|
beforeEach(() => {
|
|
|
|
ctx = new Context();
|
|
|
|
return ctx.resume();
|
|
|
|
});
|
2019-09-06 18:46:44 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
afterEach(() => {
|
|
|
|
ctx.dispose();
|
|
|
|
return ctx.close();
|
|
|
|
});
|
2019-09-06 18:46:44 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("can set an interval", (done) => {
|
|
|
|
ctx.setInterval(() => {
|
|
|
|
done();
|
|
|
|
}, 0.1);
|
|
|
|
});
|
2019-09-06 18:46:44 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("returns an id", () => {
|
|
|
|
expect(ctx.setInterval(() => {}, 0.1)).to.be.a("number");
|
|
|
|
// try clearing a random ID, shouldn't cause any errors
|
|
|
|
ctx.clearInterval(-2);
|
|
|
|
});
|
2019-09-06 18:46:44 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("timeout is not invoked when cancelled", (done) => {
|
|
|
|
const id = ctx.setInterval(() => {
|
|
|
|
throw new Error("shouldn't be invoked");
|
|
|
|
}, 0.01);
|
|
|
|
ctx.clearInterval(id);
|
|
|
|
ctx.setInterval(() => {
|
|
|
|
done();
|
|
|
|
}, 0.02);
|
|
|
|
});
|
2019-09-06 18:46:44 +00:00
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
it("order is maintained", (done) => {
|
|
|
|
let wasInvoked = false;
|
|
|
|
ctx.setInterval(() => {
|
|
|
|
expect(wasInvoked).to.equal(true);
|
|
|
|
done();
|
|
|
|
}, 0.02);
|
|
|
|
ctx.setInterval(() => {
|
|
|
|
wasInvoked = true;
|
|
|
|
}, 0.01);
|
|
|
|
});
|
2019-09-06 18:46:44 +00:00
|
|
|
|
|
|
|
it("is invoked in the offline context", () => {
|
|
|
|
let invocationCount = 0;
|
2021-01-01 01:53:38 +00:00
|
|
|
return Offline((context) => {
|
2019-09-06 18:46:44 +00:00
|
|
|
context.setInterval(() => {
|
|
|
|
invocationCount++;
|
|
|
|
}, 0.01);
|
|
|
|
}, 0.051).then(() => {
|
|
|
|
expect(invocationCount).to.equal(4);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("is invoked in with the right interval", () => {
|
|
|
|
let numberOfInvocations = 0;
|
2021-01-01 01:53:38 +00:00
|
|
|
return Offline((context) => {
|
2019-09-06 18:46:44 +00:00
|
|
|
let intervalTime = context.now();
|
|
|
|
context.setInterval(() => {
|
2021-01-01 01:53:38 +00:00
|
|
|
expect(context.now() - intervalTime).to.be.closeTo(
|
|
|
|
0.01,
|
|
|
|
0.005
|
|
|
|
);
|
2019-09-06 18:46:44 +00:00
|
|
|
intervalTime = context.now();
|
|
|
|
numberOfInvocations++;
|
|
|
|
}, 0.01);
|
|
|
|
}, 0.051).then(() => {
|
|
|
|
expect(numberOfInvocations).to.equal(4);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2019-05-23 18:00:49 +00:00
|
|
|
context("get/set", () => {
|
|
|
|
let ctx;
|
|
|
|
beforeEach(() => {
|
|
|
|
ctx = new Context();
|
|
|
|
return ctx.resume();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
2019-10-14 03:20:42 +00:00
|
|
|
ctx.dispose();
|
|
|
|
return ctx.close();
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("can set the lookAhead", () => {
|
|
|
|
ctx.lookAhead = 0.05;
|
|
|
|
expect(ctx.lookAhead).to.equal(0.05);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("can set the updateInterval", () => {
|
|
|
|
ctx.updateInterval = 0.05;
|
|
|
|
expect(ctx.updateInterval).to.equal(0.05);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("gets a constant signal", () => {
|
2021-01-01 01:53:38 +00:00
|
|
|
return ConstantOutput((context) => {
|
2019-08-19 18:15:53 +00:00
|
|
|
const bufferSrc = context.getConstant(1);
|
|
|
|
connect(bufferSrc, context.destination);
|
|
|
|
}, 1);
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("multiple calls return the same buffer source", () => {
|
|
|
|
const bufferA = ctx.getConstant(2);
|
|
|
|
const bufferB = ctx.getConstant(2);
|
|
|
|
expect(bufferA).to.equal(bufferB);
|
|
|
|
});
|
|
|
|
});
|
2020-10-02 14:28:57 +00:00
|
|
|
|
|
|
|
context("Methods", () => {
|
|
|
|
let ctx;
|
|
|
|
beforeEach(() => {
|
|
|
|
ctx = new Context();
|
|
|
|
return ctx.resume();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
ctx.dispose();
|
|
|
|
return ctx.close();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("can create a MediaElementAudioSourceNode", () => {
|
|
|
|
const audioNode = document.createElement("audio");
|
|
|
|
const node = ctx.createMediaElementSource(audioNode);
|
|
|
|
expect(node).is.not.undefined;
|
|
|
|
});
|
|
|
|
});
|
2019-05-23 18:00:49 +00:00
|
|
|
});
|