2024-05-03 18:31:14 +00:00
|
|
|
import { Signal, SignalOptions } from "./Signal.js";
|
|
|
|
import {
|
|
|
|
NormalRange,
|
|
|
|
Seconds,
|
|
|
|
Time,
|
|
|
|
TransportTime,
|
|
|
|
UnitMap,
|
|
|
|
UnitName,
|
|
|
|
} from "../core/type/Units.js";
|
|
|
|
import { optionsFromArguments } from "../core/util/Defaults.js";
|
|
|
|
import { TransportTimeClass } from "../core/type/TransportTime.js";
|
|
|
|
import { ToneConstantSource } from "./ToneConstantSource.js";
|
|
|
|
import { OutputNode } from "../core/context/ToneAudioNode.js";
|
|
|
|
import type { TransportClass } from "../core/clock/Transport.js";
|
2019-11-14 21:55:25 +00:00
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* Adds the ability to synchronize the signal to the {@link TransportClass}
|
2024-04-28 17:05:26 +00:00
|
|
|
* @category Signal
|
2019-11-14 21:55:25 +00:00
|
|
|
*/
|
2024-05-03 18:31:14 +00:00
|
|
|
export class SyncedSignal<
|
|
|
|
TypeName extends UnitName = "number",
|
|
|
|
> extends Signal<TypeName> {
|
2019-11-16 21:36:51 +00:00
|
|
|
readonly name: string = "SyncedSignal";
|
2024-05-03 18:31:14 +00:00
|
|
|
|
2019-11-14 21:55:25 +00:00
|
|
|
/**
|
|
|
|
* Don't override when something is connected to the input
|
|
|
|
*/
|
|
|
|
readonly override = false;
|
|
|
|
|
|
|
|
readonly output: OutputNode;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Keep track of the last value as an optimization.
|
|
|
|
*/
|
|
|
|
private _lastVal: UnitMap[TypeName];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The ID returned from scheduleRepeat
|
|
|
|
*/
|
|
|
|
private _synced: number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remember the callback value
|
|
|
|
*/
|
|
|
|
private _syncedCallback: () => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param value Initial value of the signal
|
|
|
|
* @param units The unit name, e.g. "frequency"
|
|
|
|
*/
|
|
|
|
constructor(value?: UnitMap[TypeName], units?: TypeName);
|
|
|
|
constructor(options?: Partial<SignalOptions<TypeName>>);
|
|
|
|
constructor() {
|
2024-05-03 18:31:14 +00:00
|
|
|
super(
|
|
|
|
optionsFromArguments(Signal.getDefaults(), arguments, [
|
|
|
|
"value",
|
|
|
|
"units",
|
|
|
|
])
|
|
|
|
);
|
|
|
|
const options = optionsFromArguments(Signal.getDefaults(), arguments, [
|
|
|
|
"value",
|
|
|
|
"units",
|
|
|
|
]) as SignalOptions<TypeName>;
|
2019-11-14 21:55:25 +00:00
|
|
|
|
|
|
|
this._lastVal = options.value;
|
2024-05-03 18:31:14 +00:00
|
|
|
this._synced = this.context.transport.scheduleRepeat(
|
|
|
|
this._onTick.bind(this),
|
|
|
|
"1i"
|
|
|
|
);
|
2019-11-14 21:55:25 +00:00
|
|
|
|
|
|
|
this._syncedCallback = this._anchorValue.bind(this);
|
|
|
|
this.context.transport.on("start", this._syncedCallback);
|
|
|
|
this.context.transport.on("pause", this._syncedCallback);
|
|
|
|
this.context.transport.on("stop", this._syncedCallback);
|
|
|
|
|
|
|
|
// disconnect the constant source from the output and replace it with another one
|
|
|
|
this._constantSource.disconnect();
|
|
|
|
this._constantSource.stop(0);
|
|
|
|
|
|
|
|
// create a new one
|
2024-05-03 18:31:14 +00:00
|
|
|
this._constantSource = this.output = new ToneConstantSource<TypeName>({
|
2019-11-14 21:55:25 +00:00
|
|
|
context: this.context,
|
|
|
|
offset: options.value,
|
|
|
|
units: options.units,
|
|
|
|
}).start(0);
|
|
|
|
this.setValueAtTime(options.value, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback which is invoked every tick.
|
|
|
|
*/
|
|
|
|
private _onTick(time: Seconds): void {
|
|
|
|
const val = super.getValueAtTime(this.context.transport.seconds);
|
|
|
|
// approximate ramp curves with linear ramps
|
|
|
|
if (this._lastVal !== val) {
|
|
|
|
this._lastVal = val;
|
|
|
|
this._constantSource.offset.setValueAtTime(val, time);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Anchor the value at the start and stop of the Transport
|
|
|
|
*/
|
|
|
|
private _anchorValue(time: Seconds): void {
|
|
|
|
const val = super.getValueAtTime(this.context.transport.seconds);
|
|
|
|
this._lastVal = val;
|
|
|
|
this._constantSource.offset.cancelAndHoldAtTime(time);
|
|
|
|
this._constantSource.offset.setValueAtTime(val, time);
|
|
|
|
}
|
|
|
|
|
|
|
|
getValueAtTime(time: TransportTime): UnitMap[TypeName] {
|
2024-05-03 18:31:14 +00:00
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
time
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
return super.getValueAtTime(computedTime);
|
|
|
|
}
|
2024-05-03 18:31:14 +00:00
|
|
|
|
2019-11-14 21:55:25 +00:00
|
|
|
setValueAtTime(value: UnitMap[TypeName], time: TransportTime) {
|
2024-05-03 18:31:14 +00:00
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
time
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.setValueAtTime(value, computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
linearRampToValueAtTime(value: UnitMap[TypeName], time: TransportTime) {
|
2024-05-03 18:31:14 +00:00
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
time
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.linearRampToValueAtTime(value, computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
exponentialRampToValueAtTime(
|
|
|
|
value: UnitMap[TypeName],
|
|
|
|
time: TransportTime
|
|
|
|
) {
|
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
time
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.exponentialRampToValueAtTime(value, computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
setTargetAtTime(
|
|
|
|
value,
|
|
|
|
startTime: TransportTime,
|
|
|
|
timeConstant: number
|
|
|
|
): this {
|
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
startTime
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.setTargetAtTime(value, computedTime, timeConstant);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
cancelScheduledValues(startTime: TransportTime): this {
|
2024-05-03 18:31:14 +00:00
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
startTime
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.cancelScheduledValues(computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
setValueCurveAtTime(
|
|
|
|
values: UnitMap[TypeName][],
|
|
|
|
startTime: TransportTime,
|
|
|
|
duration: Time,
|
|
|
|
scaling: NormalRange
|
|
|
|
): this {
|
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
startTime
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
duration = this.toSeconds(duration);
|
|
|
|
super.setValueCurveAtTime(values, computedTime, duration, scaling);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
cancelAndHoldAtTime(time: TransportTime): this {
|
2024-05-03 18:31:14 +00:00
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
time
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.cancelAndHoldAtTime(computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
2024-05-03 18:31:14 +00:00
|
|
|
|
2019-11-14 21:55:25 +00:00
|
|
|
setRampPoint(time: TransportTime): this {
|
2024-05-03 18:31:14 +00:00
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
time
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.setRampPoint(computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
2024-05-03 18:31:14 +00:00
|
|
|
|
|
|
|
exponentialRampTo(
|
|
|
|
value: UnitMap[TypeName],
|
|
|
|
rampTime: Time,
|
|
|
|
startTime?: TransportTime
|
|
|
|
): this {
|
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
startTime
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.exponentialRampTo(value, rampTime, computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
2024-05-03 18:31:14 +00:00
|
|
|
|
|
|
|
linearRampTo(
|
|
|
|
value: UnitMap[TypeName],
|
|
|
|
rampTime: Time,
|
|
|
|
startTime?: TransportTime
|
|
|
|
): this {
|
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
startTime
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.linearRampTo(value, rampTime, computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2024-05-03 18:31:14 +00:00
|
|
|
targetRampTo(
|
|
|
|
value: UnitMap[TypeName],
|
|
|
|
rampTime: Time,
|
|
|
|
startTime?: TransportTime
|
|
|
|
): this {
|
|
|
|
const computedTime = new TransportTimeClass(
|
|
|
|
this.context,
|
|
|
|
startTime
|
|
|
|
).toSeconds();
|
2019-11-14 21:55:25 +00:00
|
|
|
super.targetRampTo(value, rampTime, computedTime);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this.context.transport.clear(this._synced);
|
|
|
|
this.context.transport.off("start", this._syncedCallback);
|
|
|
|
this.context.transport.off("pause", this._syncedCallback);
|
|
|
|
this.context.transport.off("stop", this._syncedCallback);
|
|
|
|
this._constantSource.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|