Tone.js/Tone/instrument/PolySynth.test.ts

388 lines
11 KiB
TypeScript
Raw Normal View History

import { expect } from "chai";
import { BasicTests, warns } from "../../test/helper/Basic.js";
import { CompareToFile } from "../../test/helper/CompareToFile.js";
import { atTime, Offline } from "../../test/helper/Offline.js";
import { OutputAudio } from "../../test/helper/OutputAudio.js";
import { PolySynth } from "./PolySynth.js";
import { Synth } from "./Synth.js";
import { FMSynth } from "./FMSynth.js";
import { PluckSynth } from "./PluckSynth.js";
import { MetalSynth } from "./MetalSynth.js";
import { MembraneSynth } from "./MembraneSynth.js";
describe("PolySynth", () => {
BasicTests(PolySynth);
it("matches a file", () => {
return CompareToFile(
() => {
const synth = new PolySynth().toDestination();
synth.triggerAttackRelease("C4", 0.2, 0);
synth.triggerAttackRelease("C4", 0.1, 0.1);
synth.triggerAttackRelease("E4", 0.1, 0.2);
synth.triggerAttackRelease("E4", 0.1, 0.3);
synth.triggerAttackRelease("G4", 0.1, 0.4);
synth.triggerAttackRelease("B4", 0.1, 0.4);
synth.triggerAttackRelease("C4", 0.2, 0.5);
},
"polySynth.wav",
0.6
);
});
it("matches another file", () => {
return CompareToFile(
() => {
const synth = new PolySynth().toDestination();
synth.triggerAttackRelease(["C4", "E4", "G4", "B4"], 0.2, 0);
synth.triggerAttackRelease(["C4", "E4", "G4", "B4"], 0.2, 0.3);
},
"polySynth2.wav",
0.6
);
});
it("matches a file and chooses the right voice", () => {
return CompareToFile(
() => {
const synth = new PolySynth().toDestination();
synth.triggerAttackRelease(["C4", "E4"], 1, 0);
synth.triggerAttackRelease("G4", 0.1, 0.2);
synth.triggerAttackRelease("B4", 0.1, 0.4);
synth.triggerAttackRelease("G4", 0.1, 0.6);
},
"polySynth3.wav",
0.5
);
});
it("can be constructed with monophonic synths", () => {
expect(() => {
const polySynth = new PolySynth(Synth);
polySynth.dispose();
}).to.not.throw(Error);
expect(() => {
const polySynth = new PolySynth(FMSynth);
polySynth.dispose();
}).to.not.throw(Error);
expect(() => {
const polySynth = new PolySynth(MetalSynth);
polySynth.dispose();
}).to.not.throw(Error);
expect(() => {
const polySynth = new PolySynth(MembraneSynth);
polySynth.dispose();
}).to.not.throw(Error);
});
context("Playing Notes", () => {
it("triggerAttackRelease can take an array of durations", () => {
return OutputAudio(() => {
const polySynth = new PolySynth();
polySynth.toDestination();
polySynth.triggerAttackRelease(["C4", "D4"], [0.1, 0.2]);
});
});
it("triggerAttack and triggerRelease can be invoked without arrays", () => {
return Offline(() => {
const polySynth = new PolySynth();
2019-09-14 22:12:44 +00:00
polySynth.set({ envelope: { release: 0.1 } });
polySynth.toDestination();
polySynth.triggerAttack("C4", 0);
polySynth.triggerRelease("C4", 0.1);
}, 0.3).then((buffer) => {
expect(buffer.getTimeOfFirstSound()).to.be.closeTo(0, 0.01);
expect(buffer.getValueAtTime(0.2)).to.be.closeTo(0, 0.01);
});
});
it("can stop all of the currently playing sounds", () => {
return Offline(() => {
const polySynth = new PolySynth();
2019-09-14 22:12:44 +00:00
polySynth.set({ envelope: { release: 0.1 } });
polySynth.toDestination();
polySynth.triggerAttack(["C4", "E4", "G4", "B4"], 0);
return atTime(0.1, () => {
polySynth.releaseAll();
});
}, 0.3).then((buffer) => {
expect(buffer.getTimeOfFirstSound()).to.be.closeTo(0, 0.01);
expect(buffer.getTimeOfLastSound()).to.be.closeTo(0.2, 0.01);
});
});
it("is silent before being triggered", () => {
return Offline(() => {
const polySynth = new PolySynth();
polySynth.toDestination();
}).then((buffer) => {
expect(buffer.isSilent()).to.be.true;
});
});
it("can be scheduled to start in the future", () => {
return Offline(() => {
const polySynth = new PolySynth();
polySynth.toDestination();
polySynth.triggerAttack("C4", 0.1);
}, 0.3).then((buffer) => {
expect(buffer.getTimeOfFirstSound()).to.be.closeTo(0.1, 0.01);
});
});
it("can stop all sounds scheduled to start in the future when disposed", () => {
return Offline(() => {
const polySynth = new PolySynth();
polySynth.set({ envelope: { release: 0.1 } });
polySynth.toDestination();
polySynth.triggerAttackRelease(["C4", "E4", "G4", "B4"], 0.2);
return atTime(0.1, () => {
polySynth.dispose();
});
}, 0.3).then((buffer) => {
expect(buffer.isSilent()).to.be.true;
});
});
it("disposes voices when they are no longer used", () => {
return Offline(() => {
const polySynth = new PolySynth(Synth, {
2019-09-14 22:12:44 +00:00
envelope: {
release: 0.1,
},
});
polySynth.toDestination();
polySynth.triggerAttackRelease(
["C4", "E4", "G4", "B4", "D5"],
0.1,
0
);
return [
atTime(0, () => {
expect(polySynth.activeVoices).to.equal(5);
}),
atTime(0.3, () => {
expect(polySynth.activeVoices).to.equal(0);
}),
];
}, 10);
});
2019-08-12 17:21:55 +00:00
it("warns when too much polyphony is attempted and notes are dropped", () => {
warns(() => {
return Offline(() => {
const polySynth = new PolySynth({
maxPolyphony: 2,
});
polySynth.toDestination();
polySynth.triggerAttack(["C4", "D4", "G4"], 0.1);
}, 0.3);
2019-08-12 17:21:55 +00:00
});
});
it("reports the active notes", () => {
return Offline(() => {
const polySynth = new PolySynth();
2019-09-14 22:12:44 +00:00
polySynth.set({ envelope: { release: 0.1 } });
polySynth.toDestination();
polySynth.triggerAttackRelease("C4", 0.1, 0.1);
polySynth.triggerAttackRelease("D4", 0.1, 0.2);
polySynth.triggerAttackRelease("C4", 0.1, 0.5);
polySynth.triggerAttackRelease("C4", 0.1, 0.6);
return [
atTime(0, () => {
expect(polySynth.activeVoices).to.equal(0);
}),
atTime(0.1, () => {
expect(polySynth.activeVoices).to.equal(1);
}),
atTime(0.2, () => {
expect(polySynth.activeVoices).to.equal(2);
}),
atTime(0.3, () => {
expect(polySynth.activeVoices).to.equal(1);
}),
atTime(0.4, () => {
expect(polySynth.activeVoices).to.equal(0);
}),
atTime(0.5, () => {
expect(polySynth.activeVoices).to.equal(1);
}),
atTime(0.6, () => {
expect(polySynth.activeVoices).to.equal(2);
}),
atTime(0.7, () => {
expect(polySynth.activeVoices).to.equal(1);
}),
atTime(0.8, () => {
expect(polySynth.activeVoices).to.equal(0);
}),
];
}, 1);
});
it("can trigger another attack before the release has ended", () => {
// compute the end time
return Offline(() => {
const synth = new PolySynth(Synth, {
2019-09-14 22:12:44 +00:00
envelope: {
release: 0.1,
},
});
synth.toDestination();
synth.triggerAttack("C4", 0.05);
synth.triggerRelease("C4", 0.1);
synth.triggerAttack("C4", 0.15);
synth.triggerRelease("C4", 0.2);
}, 1).then((buffer) => {
expect(buffer.getTimeOfLastSound()).to.be.closeTo(0.3, 0.01);
});
});
it("can trigger another attack right after the release has ended", () => {
// compute the end time
return Offline(() => {
const synth = new PolySynth(Synth, {
2019-09-14 22:12:44 +00:00
envelope: {
release: 0.1,
},
});
synth.toDestination();
synth.triggerAttack("C4", 0.05);
synth.triggerRelease("C4", 0.1);
synth.triggerAttack("C4", 0.2);
synth.triggerRelease("C4", 0.3);
return atTime(0.41, () => {
expect(synth.activeVoices).to.equal(0);
});
}, 1).then((buffer) => {
expect(buffer.getTimeOfLastSound()).to.be.closeTo(0.4, 0.01);
});
});
});
context("Transport sync", () => {
it("can be synced to the transport", () => {
return Offline(({ transport }) => {
const polySynth = new PolySynth(Synth, {
envelope: {
release: 0.1,
},
}).sync();
polySynth.toDestination();
polySynth.triggerAttackRelease("C4", 0.1, 0.1);
polySynth.triggerAttackRelease("E4", 0.1, 0.3);
transport.start(0.1);
}, 0.8).then((buffer) => {
expect(buffer.getTimeOfFirstSound()).to.be.closeTo(0.2, 0.01);
expect(buffer.getTimeOfLastSound()).to.be.closeTo(0.6, 0.01);
});
});
it("is silent until the transport is started", () => {
return Offline(({ transport }) => {
const synth = new PolySynth(Synth).sync().toDestination();
synth.triggerAttackRelease("C4", 0.5);
transport.start(0.5);
}, 1).then((buffer) => {
expect(buffer.getTimeOfFirstSound()).is.closeTo(0.5, 0.1);
});
});
it("stops when the transport is stopped", () => {
return Offline(({ transport }) => {
const synth = new PolySynth(Synth, {
envelope: {
release: 0,
},
})
.sync()
.toDestination();
synth.triggerAttackRelease("C4", 0.5);
transport.start(0.5).stop(1);
}, 1.5).then((buffer) => {
expect(buffer.getTimeOfLastSound()).is.closeTo(1, 0.1);
});
});
it("goes silent at the loop boundary", () => {
return Offline(({ transport }) => {
const synth = new PolySynth(Synth, {
envelope: {
release: 0,
},
})
.sync()
.toDestination();
synth.triggerAttackRelease("C4", 0.8, 0.5);
transport.loopEnd = 1;
transport.loop = true;
transport.start();
}, 2).then((buffer) => {
expect(buffer.getRmsAtTime(0)).to.be.closeTo(0, 0.05);
expect(buffer.getRmsAtTime(0.6)).to.be.closeTo(0.2, 0.05);
expect(buffer.getRmsAtTime(1.1)).to.be.closeTo(0, 0.05);
expect(buffer.getRmsAtTime(1.6)).to.be.closeTo(0.2, 0.05);
});
});
it("can unsync", () => {
return Offline(({ transport }) => {
const synth = new PolySynth(Synth, {
envelope: {
sustain: 1,
release: 0,
},
})
.sync()
.toDestination()
.unsync();
synth.triggerAttackRelease("C4", 1, 0.5);
transport.start().stop(1);
}, 2).then((buffer) => {
expect(buffer.getRmsAtTime(0)).to.be.closeTo(0, 0.05);
expect(buffer.getRmsAtTime(0.6)).to.be.closeTo(0.6, 0.05);
expect(buffer.getRmsAtTime(1.4)).to.be.closeTo(0.6, 0.05);
expect(buffer.getRmsAtTime(1.6)).to.be.closeTo(0, 0.05);
});
});
});
context("API", () => {
it("can be constructed with an options object", () => {
const polySynth = new PolySynth(Synth, {
2019-09-14 22:12:44 +00:00
envelope: {
sustain: 0.3,
},
});
expect(polySynth.get().envelope.sustain).to.equal(0.3);
polySynth.dispose();
});
it("throws an error when used without a monophonic synth", () => {
expect(() => {
// @ts-ignore
new PolySynth(PluckSynth);
}).throws(Error);
});
it("can pass in the volume", () => {
const polySynth = new PolySynth({
2019-09-14 22:12:44 +00:00
volume: -12,
});
expect(polySynth.volume.value).to.be.closeTo(-12, 0.1);
polySynth.dispose();
});
it("can get/set attributes", () => {
const polySynth = new PolySynth();
polySynth.set({
2019-09-14 22:12:44 +00:00
envelope: { decay: 3 },
});
expect(polySynth.get().envelope.decay).to.equal(3);
polySynth.dispose();
});
});
});