2019-05-23 18:00:49 +00:00
|
|
|
import { Compare, Plot } from "@tonejs/plot";
|
2019-04-12 14:37:47 +00:00
|
|
|
import { expect } from "chai";
|
2019-05-23 18:00:49 +00:00
|
|
|
import { BasicTests, testAudioContext } from "test/helper/Basic";
|
|
|
|
import { Offline } from "test/helper/Offline";
|
|
|
|
import { SCHEDULE_RAMP_AFTER_SET_TARGET } from "test/helper/Supports";
|
2019-06-23 19:02:38 +00:00
|
|
|
import { getContext } from "../Global";
|
2019-05-23 18:00:49 +00:00
|
|
|
import { Param } from "./Param";
|
2019-04-12 14:37:47 +00:00
|
|
|
|
2019-06-23 19:02:38 +00:00
|
|
|
const audioContext = getContext();
|
2019-04-12 14:37:47 +00:00
|
|
|
|
|
|
|
describe("Param", () => {
|
|
|
|
|
|
|
|
BasicTests(Param, {
|
|
|
|
param: testAudioContext.createOscillator().frequency,
|
|
|
|
});
|
|
|
|
|
|
|
|
context("constructor", () => {
|
|
|
|
it("can be created and disposed", async () => {
|
|
|
|
await Offline(context => {
|
|
|
|
const param = new Param<"time">({
|
|
|
|
context,
|
|
|
|
param: context.createConstantSource().offset,
|
|
|
|
units: "time",
|
|
|
|
});
|
|
|
|
expect(param.getValueAtTime(0)).to.equal(1);
|
|
|
|
param.dispose();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("can pass in a value", async () => {
|
|
|
|
await Offline(context => {
|
|
|
|
const param = new Param({
|
|
|
|
context,
|
|
|
|
param: context.createConstantSource().offset,
|
|
|
|
value : 1.1,
|
|
|
|
});
|
|
|
|
expect(param.getValueAtTime(0)).to.equal(1.1);
|
|
|
|
param.dispose();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it ("requires a param in the constructor", () => {
|
|
|
|
expect(() => {
|
|
|
|
const param = new Param({
|
|
|
|
value: 1.1,
|
|
|
|
});
|
|
|
|
}).throws(Error);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
context("Scheduling Curves", () => {
|
|
|
|
|
|
|
|
const sampleRate = 11025;
|
|
|
|
function matchesOutputCurve(param, outBuffer): void {
|
|
|
|
outBuffer.toArray()[0].forEach((sample, index) => {
|
|
|
|
try {
|
|
|
|
expect(param.getValueAtTime(index / sampleRate)).to.be.closeTo(sample, 0.1);
|
|
|
|
} catch (e) {
|
|
|
|
console.log(index / sampleRate);
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SCHEDULE_RAMP_AFTER_SET_TARGET) {
|
|
|
|
|
|
|
|
it("correctly handles setTargetAtTime followed by a ramp", async () => {
|
|
|
|
let param;
|
|
|
|
// this fails on FF
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
});
|
|
|
|
param.setTargetAtTime(2, 0.5, 0.1);
|
|
|
|
expect(param.getValueAtTime(0.6)).to.be.closeTo(1.6, 0.1);
|
|
|
|
param.linearRampToValueAtTime(0.5, 0.7);
|
|
|
|
expect(param.getValueAtTime(0.6)).to.be.closeTo(0.75, 0.1);
|
2019-06-18 01:51:10 +00:00
|
|
|
}, 1.5, 1, sampleRate);
|
2019-04-12 14:37:47 +00:00
|
|
|
document.body.appendChild(await Plot.signal(testBuffer));
|
|
|
|
matchesOutputCurve(param, testBuffer);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("schedules a value curve", async () => {
|
|
|
|
let param;
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
units: "number",
|
|
|
|
value : 0,
|
|
|
|
});
|
|
|
|
param.setValueCurveAtTime([0, 0.5, 0, 1, 1.5], 0.1, 0.8, 0.5);
|
|
|
|
expect(param.getValueAtTime(0.91)).to.be.closeTo(0.75, 0.01);
|
2019-06-18 01:51:10 +00:00
|
|
|
}, 1, 1, sampleRate);
|
2019-04-12 14:37:47 +00:00
|
|
|
document.body.appendChild(await Plot.signal(testBuffer));
|
|
|
|
matchesOutputCurve(param, testBuffer);
|
|
|
|
});
|
|
|
|
|
|
|
|
it ("a mixture of scheduling curves", async () => {
|
|
|
|
let param;
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
value : 0.1,
|
|
|
|
});
|
|
|
|
param.setValueAtTime(0, 0);
|
|
|
|
param.setValueAtTime(1, 0.1);
|
|
|
|
param.linearRampToValueAtTime(3, 0.2);
|
|
|
|
param.exponentialRampToValueAtTime(0.01, 0.3);
|
|
|
|
param.setTargetAtTime(-1, 0.35, 0.2);
|
|
|
|
param.cancelAndHoldAtTime(0.6);
|
|
|
|
param.rampTo(1.1, 0.2, 0.7);
|
|
|
|
param.exponentialRampTo(0, 0.1, 0.85);
|
|
|
|
param.setValueAtTime(0, 1);
|
|
|
|
param.linearRampTo(1, 0.2, 1);
|
|
|
|
param.targetRampTo(0, 0.1, 1.1);
|
|
|
|
param.setValueAtTime(4, 1.2);
|
|
|
|
param.cancelScheduledValues(1.2);
|
|
|
|
param.linearRampToValueAtTime(1, 1.3);
|
2019-06-18 01:51:10 +00:00
|
|
|
}, 1.5, 1, sampleRate);
|
2019-04-12 14:37:47 +00:00
|
|
|
document.body.appendChild(await Plot.signal(testBuffer));
|
|
|
|
matchesOutputCurve(param, testBuffer);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("can cancel and hold", async () => {
|
|
|
|
let param;
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
value: 0.1,
|
|
|
|
});
|
|
|
|
param.setValueAtTime(0, 0);
|
|
|
|
param.setValueAtTime(1, 0.2);
|
|
|
|
param.cancelAndHoldAtTime(0.1);
|
|
|
|
param.linearRampToValueAtTime(1, 0.3);
|
|
|
|
param.cancelAndHoldAtTime(0.2);
|
|
|
|
expect(param.getValueAtTime(0.2)).to.be.closeTo(0.5, 0.001);
|
|
|
|
param.exponentialRampToValueAtTime(0, 0.4);
|
|
|
|
param.cancelAndHoldAtTime(0.25);
|
|
|
|
expect(param.getValueAtTime(0.25)).to.be.closeTo(0.033, 0.001);
|
|
|
|
param.setTargetAtTime(1, 0.3, 0.1);
|
|
|
|
param.cancelAndHoldAtTime(0.4);
|
|
|
|
expect(param.getValueAtTime(0.4)).to.be.closeTo(0.644, 0.001);
|
|
|
|
param.setValueAtTime(0, 0.45);
|
|
|
|
param.setValueAtTime(1, 0.48);
|
|
|
|
param.cancelAndHoldAtTime(0.45);
|
|
|
|
expect(param.getValueAtTime(0.45)).to.be.closeTo(0, 0.001);
|
2019-06-18 01:51:10 +00:00
|
|
|
}, 0.5, 1, sampleRate);
|
2019-04-12 14:37:47 +00:00
|
|
|
matchesOutputCurve(param, testBuffer);
|
|
|
|
// document.body.appendChild(await Plot.signal(testBuffer));
|
|
|
|
});
|
|
|
|
|
2019-05-23 18:00:49 +00:00
|
|
|
// it ("matches known values", async () => {
|
|
|
|
// await Compare.toFile(context => {
|
|
|
|
// const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
// source.connect(context.rawContext.destination);
|
2019-05-23 18:00:49 +00:00
|
|
|
// source.start(0);
|
|
|
|
// const param = new Param({
|
|
|
|
// context,
|
|
|
|
// param: source.offset,
|
|
|
|
// value: 0.1,
|
|
|
|
// });
|
|
|
|
// param.setValueAtTime(0, 0);
|
|
|
|
// param.setValueAtTime(1, 0.2);
|
|
|
|
// param.cancelAndHoldAtTime(0.1);
|
|
|
|
// param.linearRampToValueAtTime(1, 0.3);
|
|
|
|
// param.cancelAndHoldAtTime(0.2);
|
|
|
|
// param.exponentialRampToValueAtTime(0, 0.4);
|
|
|
|
// param.cancelAndHoldAtTime(0.25);
|
|
|
|
// param.setTargetAtTime(1, 0.3, 0.1);
|
|
|
|
// param.cancelAndHoldAtTime(0.4);
|
|
|
|
// }, "/base/test/audio/param/curve_0.wav", 0.01, 0.5, 1, 11025);
|
|
|
|
// });
|
2019-04-12 14:37:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
context("Units", () => {
|
|
|
|
|
|
|
|
it("can be created with specific units", () => {
|
|
|
|
const gain = audioContext.createGain();
|
|
|
|
const param = new Param<"bpm">({
|
|
|
|
context: audioContext,
|
|
|
|
param : gain.gain,
|
|
|
|
units : "bpm",
|
|
|
|
});
|
|
|
|
expect(param.units).to.equal("bpm");
|
|
|
|
param.dispose();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("can evaluate the given units", () => {
|
|
|
|
const gain = audioContext.createGain();
|
|
|
|
const param = new Param<"decibels">({
|
|
|
|
context: audioContext,
|
|
|
|
param: gain.gain,
|
|
|
|
units: "decibels",
|
|
|
|
});
|
|
|
|
param.value = 0.5;
|
|
|
|
expect(param.value).to.be.closeTo(0.5, 0.001);
|
|
|
|
param.dispose();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("can be forced to not convert", async () => {
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
const param = new Param({
|
|
|
|
context,
|
|
|
|
convert : false,
|
|
|
|
param: source.offset,
|
|
|
|
units : "decibels",
|
|
|
|
});
|
|
|
|
param.value = -10;
|
|
|
|
expect(param.value).to.be.closeTo(-10, 0.01);
|
|
|
|
}, 0.001, 1);
|
|
|
|
expect(testBuffer.getValueAtTime(0)).to.be.closeTo(-10, 0.01);
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
context("Unit Conversions", () => {
|
|
|
|
function testUnitConversion(units: Unit, inputValue: any, inputVerification: number, outputValue: number): void {
|
|
|
|
it(`converts to ${units}`, async () => {
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
const param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
units,
|
|
|
|
});
|
|
|
|
param.value = inputValue;
|
|
|
|
expect(param.value).to.be.closeTo(inputVerification, 0.01);
|
|
|
|
}, 0.001, 1);
|
|
|
|
expect(testBuffer.getValueAtTime(0)).to.be.closeTo(outputValue, 0.01);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
testUnitConversion("number", 3, 3, 3);
|
|
|
|
testUnitConversion("decibels", -10, -10, 0.31);
|
|
|
|
testUnitConversion("decibels", -20, -20, 0.1);
|
|
|
|
testUnitConversion("decibels", -100, -100, 0);
|
|
|
|
testUnitConversion("gain", 1.2, 1.2, 1.2);
|
|
|
|
testUnitConversion("positive", 1.5, 1.5, 1.5);
|
|
|
|
testUnitConversion("positive", -1.5, 0, 0);
|
|
|
|
testUnitConversion("time", 2, 2, 2);
|
|
|
|
testUnitConversion("time", 0, 0, 0);
|
|
|
|
testUnitConversion("frequency", 20, 20, 20);
|
|
|
|
testUnitConversion("frequency", 0.1, 0.1, 0.1);
|
|
|
|
testUnitConversion("normalRange", -1, 0, 0);
|
|
|
|
testUnitConversion("normalRange", 0.5, 0.5, 0.5);
|
|
|
|
testUnitConversion("normalRange", 1.5, 1, 1);
|
|
|
|
testUnitConversion("audioRange", -1.1, -1, -1);
|
|
|
|
testUnitConversion("audioRange", 0.5, 0.5, 0.5);
|
|
|
|
testUnitConversion("audioRange", 1.5, 1, 1);
|
|
|
|
});
|
|
|
|
|
|
|
|
context("min/maxValue", () => {
|
|
|
|
function testMinMaxValue(units: Unit, min, max): void {
|
|
|
|
it(`has proper min/max for ${units}`, () => {
|
|
|
|
const source = audioContext.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(audioContext.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
const param = new Param({
|
|
|
|
context : audioContext,
|
|
|
|
param: source.offset,
|
|
|
|
units,
|
|
|
|
});
|
|
|
|
expect(param.minValue).to.be.equal(min);
|
|
|
|
expect(param.maxValue).to.be.equal(max);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// number, decibels, normalRange, audioRange, gain
|
|
|
|
// positive, time, frequency, transportTime, ticks, bpm, degrees, samples, hertz
|
|
|
|
const rangeMax = 3.4028234663852886e+38;
|
|
|
|
testMinMaxValue("number", -rangeMax, rangeMax);
|
|
|
|
testMinMaxValue("decibels", -Infinity, rangeMax);
|
|
|
|
testMinMaxValue("normalRange", 0, 1);
|
|
|
|
testMinMaxValue("audioRange", -1, 1);
|
|
|
|
testMinMaxValue("gain", -rangeMax, rangeMax);
|
|
|
|
testMinMaxValue("positive", 0, rangeMax);
|
|
|
|
testMinMaxValue("time", 0, rangeMax);
|
|
|
|
testMinMaxValue("frequency", 0, rangeMax);
|
|
|
|
testMinMaxValue("transportTime", 0, rangeMax);
|
|
|
|
testMinMaxValue("ticks", 0, rangeMax);
|
|
|
|
testMinMaxValue("bpm", 0, rangeMax);
|
|
|
|
testMinMaxValue("degrees", -rangeMax, rangeMax);
|
|
|
|
testMinMaxValue("samples", 0, rangeMax);
|
|
|
|
testMinMaxValue("hertz", 0, rangeMax);
|
|
|
|
});
|
|
|
|
|
|
|
|
// const allSchedulingMethods = ['setValueAtTime', 'linearRampToValueAtTime', 'exponentialRampToValueAtTime']
|
|
|
|
|
|
|
|
context("setValueAtTime", () => {
|
|
|
|
function testSetValueAtTime(units: Unit, value0, value1, value2): void {
|
|
|
|
it(`can schedule value with units ${units}`, async () => {
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
const param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
units,
|
|
|
|
});
|
|
|
|
param.setValueAtTime(value0, 0);
|
|
|
|
param.setValueAtTime(value1, 0.01);
|
|
|
|
param.setValueAtTime(value2, 0.02);
|
|
|
|
|
|
|
|
expect(param.getValueAtTime(0)).to.be.closeTo(value0, 0.01);
|
|
|
|
expect(param.getValueAtTime(0.01)).to.be.closeTo(value1, 0.01);
|
|
|
|
expect(param.getValueAtTime(0.02)).to.be.closeTo(value2, 0.01);
|
|
|
|
}, 0.022, 1);
|
|
|
|
expect(testBuffer.getValueAtTime(0)).to.be.closeTo(0, 0.01);
|
|
|
|
expect(testBuffer.getValueAtTime(0.011)).to.be.closeTo(1, 0.01);
|
|
|
|
expect(testBuffer.getValueAtTime(0.021)).to.be.closeTo(0.5, 0.01);
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
const allUnits: Unit[] = ["number", "decibels", "normalRange", "audioRange", "gain",
|
|
|
|
"positive", "time", "frequency", "transportTime", "ticks", "bpm", "degrees", "samples", "hertz"];
|
|
|
|
|
|
|
|
allUnits.forEach(unit => {
|
|
|
|
if (unit === "decibels") {
|
|
|
|
testSetValueAtTime(unit, -100, 0, -6);
|
|
|
|
} else {
|
|
|
|
testSetValueAtTime(unit, 0, 1, 0.5);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
["linearRampToValueAtTime", "exponentialRampToValueAtTime"].forEach(method => {
|
|
|
|
|
|
|
|
context(method, () => {
|
|
|
|
function testRampToValueAtTime(units: Unit, value0, value1, value2): void {
|
|
|
|
it(`can schedule value with units ${units}`, async () => {
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
const param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
units,
|
|
|
|
});
|
|
|
|
param.setValueAtTime(value0, 0);
|
|
|
|
param[method](value1, 0.01);
|
|
|
|
param[method](value2, 0.02);
|
|
|
|
|
|
|
|
expect(param.getValueAtTime(0)).to.be.closeTo(value0, 0.01);
|
|
|
|
expect(param.getValueAtTime(0.01)).to.be.closeTo(value1, 0.01);
|
|
|
|
expect(param.getValueAtTime(0.02)).to.be.closeTo(value2, 0.01);
|
|
|
|
}, 0.022, 1);
|
|
|
|
expect(testBuffer.getValueAtTime(0)).to.be.closeTo(1, 0.01);
|
|
|
|
expect(testBuffer.getValueAtTime(0.01)).to.be.closeTo(0.7, 0.01);
|
|
|
|
expect(testBuffer.getValueAtTime(0.02)).to.be.closeTo(0, 0.01);
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
const allUnits: Unit[] = ["number", "decibels", "normalRange", "audioRange", "gain",
|
|
|
|
"positive", "time", "frequency", "transportTime", "ticks", "bpm", "degrees", "samples", "hertz"];
|
|
|
|
|
|
|
|
allUnits.forEach(unit => {
|
|
|
|
if (unit === "decibels") {
|
|
|
|
testRampToValueAtTime(unit, 0, -3, -100);
|
|
|
|
} else {
|
|
|
|
testRampToValueAtTime(unit, 1, 0.7, 0);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
["linearRampTo", "exponentialRampTo", "rampTo", "targetRampTo"].forEach(method => {
|
|
|
|
|
|
|
|
context(method, () => {
|
|
|
|
function testRampToValueAtTime(units: Unit, value0, value1, value2): void {
|
|
|
|
it(`can schedule value with units ${units}`, async () => {
|
|
|
|
const testBuffer = await Offline(context => {
|
|
|
|
const source = context.createConstantSource();
|
2019-06-23 19:02:38 +00:00
|
|
|
source.connect(context.rawContext.destination);
|
2019-04-12 14:37:47 +00:00
|
|
|
source.start(0);
|
|
|
|
const param = new Param({
|
|
|
|
context,
|
|
|
|
param: source.offset,
|
|
|
|
units,
|
|
|
|
value: value0,
|
|
|
|
});
|
|
|
|
param[method](value1, 0.009, 0);
|
|
|
|
param[method](value2, 0.01, 0.01);
|
|
|
|
|
|
|
|
expect(param.getValueAtTime(0)).to.be.closeTo(value0, 0.02);
|
|
|
|
expect(param.getValueAtTime(0.01)).to.be.closeTo(value1, 0.02);
|
|
|
|
if (units !== "decibels") {
|
|
|
|
expect(param.getValueAtTime(0.025)).to.be.closeTo(value2, 0.01);
|
|
|
|
}
|
|
|
|
}, 0.021, 1);
|
|
|
|
// document.body.appendChild(await Plot.signal(testBuffer));
|
|
|
|
expect(testBuffer.getValueAtTime(0)).to.be.closeTo(1, 0.01);
|
|
|
|
expect(testBuffer.getValueAtTime(0.01)).to.be.closeTo(0.7, 0.01);
|
|
|
|
expect(testBuffer.getValueAtTime(0.02)).to.be.closeTo(0, 0.01);
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
const allUnits: Unit[] = ["number", "decibels", "normalRange", "audioRange", "gain",
|
|
|
|
"positive", "time", "frequency", "transportTime", "ticks", "bpm", "degrees", "samples", "hertz"];
|
|
|
|
|
|
|
|
allUnits.forEach(unit => {
|
|
|
|
if (unit === "decibels") {
|
|
|
|
testRampToValueAtTime(unit, 0, -3, -100);
|
|
|
|
} else {
|
|
|
|
testRampToValueAtTime(unit, 1, 0.7, 0);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|