2024-05-03 14:10:40 +00:00
|
|
|
import { AutomationEvent, Param, ParamOptions } from "../context/Param.js";
|
|
|
|
import { Seconds, Ticks, Time, UnitMap, UnitName } from "../type/Units.js";
|
|
|
|
import { optionsFromArguments } from "../util/Defaults.js";
|
|
|
|
import { Timeline } from "../util/Timeline.js";
|
|
|
|
import { isUndef } from "../util/TypeCheck.js";
|
2019-09-07 23:16:04 +00:00
|
|
|
|
|
|
|
type TickAutomationEvent = AutomationEvent & {
|
|
|
|
ticks: number;
|
|
|
|
};
|
|
|
|
|
2024-05-03 15:09:28 +00:00
|
|
|
interface TickParamOptions<TypeName extends UnitName>
|
|
|
|
extends ParamOptions<TypeName> {
|
2019-09-07 23:16:04 +00:00
|
|
|
multiplier: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* A Param class just for computing ticks. Similar to the {@link Param} class,
|
2019-09-07 23:16:04 +00:00
|
|
|
* but offers conversion to BPM values as well as ability to compute tick
|
|
|
|
* duration and elapsed ticks
|
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
export class TickParam<
|
|
|
|
TypeName extends "hertz" | "bpm",
|
|
|
|
> extends Param<TypeName> {
|
2019-09-07 23:16:04 +00:00
|
|
|
readonly name: string = "TickParam";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The timeline which tracks all of the automations.
|
|
|
|
*/
|
|
|
|
protected _events: Timeline<TickAutomationEvent> = new Timeline(Infinity);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The internal holder for the multiplier value
|
|
|
|
*/
|
2019-11-17 18:09:19 +00:00
|
|
|
private _multiplier = 1;
|
2019-09-07 23:16:04 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param param The AudioParam to wrap
|
|
|
|
* @param units The unit name
|
|
|
|
* @param convert Whether or not to convert the value to the target units
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @param value The initial value of the signal
|
|
|
|
*/
|
|
|
|
constructor(value?: number);
|
2019-10-28 15:37:53 +00:00
|
|
|
constructor(options: Partial<TickParamOptions<TypeName>>);
|
2019-09-07 23:16:04 +00:00
|
|
|
constructor() {
|
2024-05-03 15:09:28 +00:00
|
|
|
super(
|
|
|
|
optionsFromArguments(TickParam.getDefaults(), arguments, ["value"])
|
|
|
|
);
|
|
|
|
const options = optionsFromArguments(
|
|
|
|
TickParam.getDefaults(),
|
|
|
|
arguments,
|
|
|
|
["value"]
|
|
|
|
);
|
2019-09-07 23:16:04 +00:00
|
|
|
|
|
|
|
// set the multiplier
|
|
|
|
this._multiplier = options.multiplier;
|
|
|
|
|
|
|
|
// clear the ticks from the beginning
|
|
|
|
this._events.cancel(0);
|
|
|
|
// set an initial event
|
|
|
|
this._events.add({
|
|
|
|
ticks: 0,
|
2019-09-14 23:55:39 +00:00
|
|
|
time: 0,
|
|
|
|
type: "setValueAtTime",
|
2019-09-08 18:08:25 +00:00
|
|
|
value: this._fromType(options.value),
|
2019-09-07 23:16:04 +00:00
|
|
|
});
|
2019-09-08 18:08:25 +00:00
|
|
|
this.setValueAtTime(options.value, 0);
|
2019-09-07 23:16:04 +00:00
|
|
|
}
|
|
|
|
|
2019-09-08 18:08:25 +00:00
|
|
|
static getDefaults(): TickParamOptions<any> {
|
2019-09-07 23:16:04 +00:00
|
|
|
return Object.assign(Param.getDefaults(), {
|
|
|
|
multiplier: 1,
|
|
|
|
units: "hertz",
|
|
|
|
value: 1,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-05-03 15:09:28 +00:00
|
|
|
setTargetAtTime(
|
|
|
|
value: UnitMap[TypeName],
|
|
|
|
time: Time,
|
|
|
|
constant: number
|
|
|
|
): this {
|
2019-09-07 23:16:04 +00:00
|
|
|
// approximate it with multiple linear ramps
|
|
|
|
time = this.toSeconds(time);
|
|
|
|
this.setRampPoint(time);
|
|
|
|
const computedValue = this._fromType(value);
|
|
|
|
|
|
|
|
// start from previously scheduled value
|
|
|
|
const prevEvent = this._events.get(time) as TickAutomationEvent;
|
|
|
|
const segments = Math.round(Math.max(1 / constant, 1));
|
|
|
|
for (let i = 0; i <= segments; i++) {
|
|
|
|
const segTime = constant * i + time;
|
2024-05-03 15:09:28 +00:00
|
|
|
const rampVal = this._exponentialApproach(
|
|
|
|
prevEvent.time,
|
|
|
|
prevEvent.value,
|
|
|
|
computedValue,
|
|
|
|
constant,
|
|
|
|
segTime
|
|
|
|
);
|
2019-09-07 23:16:04 +00:00
|
|
|
this.linearRampToValueAtTime(this._toType(rampVal), segTime);
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2019-10-28 15:37:53 +00:00
|
|
|
setValueAtTime(value: UnitMap[TypeName], time: Time): this {
|
2019-09-07 23:16:04 +00:00
|
|
|
const computedTime = this.toSeconds(time);
|
|
|
|
super.setValueAtTime(value, time);
|
|
|
|
const event = this._events.get(computedTime) as TickAutomationEvent;
|
|
|
|
const previousEvent = this._events.previousEvent(event);
|
2024-05-03 15:09:28 +00:00
|
|
|
const ticksUntilTime = this._getTicksUntilEvent(
|
|
|
|
previousEvent,
|
|
|
|
computedTime
|
|
|
|
);
|
2019-09-07 23:16:04 +00:00
|
|
|
event.ticks = Math.max(ticksUntilTime, 0);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2019-10-28 15:37:53 +00:00
|
|
|
linearRampToValueAtTime(value: UnitMap[TypeName], time: Time): this {
|
2019-09-07 23:16:04 +00:00
|
|
|
const computedTime = this.toSeconds(time);
|
|
|
|
super.linearRampToValueAtTime(value, time);
|
|
|
|
const event = this._events.get(computedTime) as TickAutomationEvent;
|
|
|
|
const previousEvent = this._events.previousEvent(event);
|
2024-05-03 15:09:28 +00:00
|
|
|
const ticksUntilTime = this._getTicksUntilEvent(
|
|
|
|
previousEvent,
|
|
|
|
computedTime
|
|
|
|
);
|
2019-09-07 23:16:04 +00:00
|
|
|
event.ticks = Math.max(ticksUntilTime, 0);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2019-10-28 15:37:53 +00:00
|
|
|
exponentialRampToValueAtTime(value: UnitMap[TypeName], time: Time): this {
|
2019-09-07 23:16:04 +00:00
|
|
|
// aproximate it with multiple linear ramps
|
|
|
|
time = this.toSeconds(time);
|
|
|
|
const computedVal = this._fromType(value);
|
|
|
|
|
|
|
|
// start from previously scheduled value
|
|
|
|
const prevEvent = this._events.get(time) as TickAutomationEvent;
|
|
|
|
// approx 10 segments per second
|
|
|
|
const segments = Math.round(Math.max((time - prevEvent.time) * 10, 1));
|
2024-05-03 15:09:28 +00:00
|
|
|
const segmentDur = (time - prevEvent.time) / segments;
|
2019-09-07 23:16:04 +00:00
|
|
|
for (let i = 0; i <= segments; i++) {
|
|
|
|
const segTime = segmentDur * i + prevEvent.time;
|
2024-05-03 15:09:28 +00:00
|
|
|
const rampVal = this._exponentialInterpolate(
|
|
|
|
prevEvent.time,
|
|
|
|
prevEvent.value,
|
|
|
|
time,
|
|
|
|
computedVal,
|
|
|
|
segTime
|
|
|
|
);
|
2019-09-07 23:16:04 +00:00
|
|
|
this.linearRampToValueAtTime(this._toType(rampVal), segTime);
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the tick value at the time. Takes into account
|
|
|
|
* any automation curves scheduled on the signal.
|
2019-10-23 03:04:52 +00:00
|
|
|
* @param event The time to get the tick count at
|
2019-09-07 23:16:04 +00:00
|
|
|
* @return The number of ticks which have elapsed at the time given any automations.
|
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
private _getTicksUntilEvent(
|
|
|
|
event: TickAutomationEvent | null,
|
|
|
|
time: number
|
|
|
|
): Ticks {
|
2019-09-07 23:16:04 +00:00
|
|
|
if (event === null) {
|
|
|
|
event = {
|
2019-09-14 23:55:39 +00:00
|
|
|
ticks: 0,
|
|
|
|
time: 0,
|
2019-09-07 23:16:04 +00:00
|
|
|
type: "setValueAtTime",
|
|
|
|
value: 0,
|
|
|
|
};
|
|
|
|
} else if (isUndef(event.ticks)) {
|
|
|
|
const previousEvent = this._events.previousEvent(event);
|
|
|
|
event.ticks = this._getTicksUntilEvent(previousEvent, event.time);
|
|
|
|
}
|
|
|
|
const val0 = this._fromType(this.getValueAtTime(event.time));
|
|
|
|
let val1 = this._fromType(this.getValueAtTime(time));
|
|
|
|
// if it's right on the line, take the previous value
|
|
|
|
const onTheLineEvent = this._events.get(time);
|
2024-05-03 15:09:28 +00:00
|
|
|
if (
|
|
|
|
onTheLineEvent &&
|
|
|
|
onTheLineEvent.time === time &&
|
|
|
|
onTheLineEvent.type === "setValueAtTime"
|
|
|
|
) {
|
2019-09-07 23:16:04 +00:00
|
|
|
val1 = this._fromType(this.getValueAtTime(time - this.sampleTime));
|
|
|
|
}
|
|
|
|
return 0.5 * (time - event.time) * (val0 + val1) + event.ticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the tick value at the time. Takes into account
|
|
|
|
* any automation curves scheduled on the signal.
|
|
|
|
* @param time The time to get the tick count at
|
|
|
|
* @return The number of ticks which have elapsed at the time given any automations.
|
|
|
|
*/
|
|
|
|
getTicksAtTime(time: Time): Ticks {
|
|
|
|
const computedTime = this.toSeconds(time);
|
|
|
|
const event = this._events.get(computedTime);
|
|
|
|
return Math.max(this._getTicksUntilEvent(event, computedTime), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the elapsed time of the number of ticks from the given time
|
|
|
|
* @param ticks The number of ticks to calculate
|
|
|
|
* @param time The time to get the next tick from
|
|
|
|
* @return The duration of the number of ticks from the given time in seconds
|
|
|
|
*/
|
|
|
|
getDurationOfTicks(ticks: Ticks, time: Time): Seconds {
|
|
|
|
const computedTime = this.toSeconds(time);
|
|
|
|
const currentTick = this.getTicksAtTime(time);
|
|
|
|
return this.getTimeOfTick(currentTick + ticks) - computedTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given a tick, returns the time that tick occurs at.
|
|
|
|
* @return The time that the tick occurs.
|
|
|
|
*/
|
|
|
|
getTimeOfTick(tick: Ticks): Seconds {
|
|
|
|
const before = this._events.get(tick, "ticks");
|
|
|
|
const after = this._events.getAfter(tick, "ticks");
|
|
|
|
if (before && before.ticks === tick) {
|
|
|
|
return before.time;
|
2024-05-03 15:09:28 +00:00
|
|
|
} else if (
|
|
|
|
before &&
|
|
|
|
after &&
|
2019-09-07 23:16:04 +00:00
|
|
|
after.type === "linearRampToValueAtTime" &&
|
2024-05-03 15:09:28 +00:00
|
|
|
before.value !== after.value
|
|
|
|
) {
|
2019-09-07 23:16:04 +00:00
|
|
|
const val0 = this._fromType(this.getValueAtTime(before.time));
|
|
|
|
const val1 = this._fromType(this.getValueAtTime(after.time));
|
|
|
|
const delta = (val1 - val0) / (after.time - before.time);
|
2024-05-03 15:09:28 +00:00
|
|
|
const k = Math.sqrt(
|
|
|
|
Math.pow(val0, 2) - 2 * delta * (before.ticks - tick)
|
|
|
|
);
|
2019-09-07 23:16:04 +00:00
|
|
|
const sol1 = (-val0 + k) / delta;
|
|
|
|
const sol2 = (-val0 - k) / delta;
|
|
|
|
return (sol1 > 0 ? sol1 : sol2) + before.time;
|
|
|
|
} else if (before) {
|
|
|
|
if (before.value === 0) {
|
|
|
|
return Infinity;
|
|
|
|
} else {
|
|
|
|
return before.time + (tick - before.ticks) / before.value;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return tick / this._initialValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert some number of ticks their the duration in seconds accounting
|
|
|
|
* for any automation curves starting at the given time.
|
|
|
|
* @param ticks The number of ticks to convert to seconds.
|
|
|
|
* @param when When along the automation timeline to convert the ticks.
|
|
|
|
* @return The duration in seconds of the ticks.
|
|
|
|
*/
|
|
|
|
ticksToTime(ticks: Ticks, when: Time): Seconds {
|
|
|
|
return this.getDurationOfTicks(ticks, when);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* The inverse of {@link ticksToTime}. Convert a duration in
|
2019-09-07 23:16:04 +00:00
|
|
|
* seconds to the corresponding number of ticks accounting for any
|
|
|
|
* automation curves starting at the given time.
|
|
|
|
* @param duration The time interval to convert to ticks.
|
|
|
|
* @param when When along the automation timeline to convert the ticks.
|
|
|
|
* @return The duration in ticks.
|
|
|
|
*/
|
|
|
|
timeToTicks(duration: Time, when: Time): Ticks {
|
|
|
|
const computedTime = this.toSeconds(when);
|
|
|
|
const computedDuration = this.toSeconds(duration);
|
|
|
|
const startTicks = this.getTicksAtTime(computedTime);
|
|
|
|
const endTicks = this.getTicksAtTime(computedTime + computedDuration);
|
|
|
|
return endTicks - startTicks;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert from the type when the unit value is BPM
|
|
|
|
*/
|
2019-10-28 15:37:53 +00:00
|
|
|
protected _fromType(val: UnitMap[TypeName]): number {
|
2019-09-07 23:16:04 +00:00
|
|
|
if (this.units === "bpm" && this.multiplier) {
|
|
|
|
return 1 / (60 / val / this.multiplier);
|
|
|
|
} else {
|
|
|
|
return super._fromType(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Special case of type conversion where the units === "bpm"
|
|
|
|
*/
|
2019-10-28 15:37:53 +00:00
|
|
|
protected _toType(val: number): UnitMap[TypeName] {
|
2019-09-07 23:16:04 +00:00
|
|
|
if (this.units === "bpm" && this.multiplier) {
|
2024-05-03 15:09:28 +00:00
|
|
|
return ((val / this.multiplier) * 60) as UnitMap[TypeName];
|
2019-09-07 23:16:04 +00:00
|
|
|
} else {
|
|
|
|
return super._toType(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* A multiplier on the bpm value. Useful for setting a PPQ relative to the base frequency value.
|
|
|
|
*/
|
|
|
|
get multiplier(): number {
|
|
|
|
return this._multiplier;
|
|
|
|
}
|
|
|
|
set multiplier(m: number) {
|
|
|
|
// get and reset the current value with the new multiplier
|
|
|
|
// might be necessary to clear all the previous values
|
|
|
|
const currentVal = this.value;
|
|
|
|
this._multiplier = m;
|
2019-12-14 16:53:12 +00:00
|
|
|
this.cancelScheduledValues(0);
|
|
|
|
this.setValueAtTime(currentVal, 0);
|
2019-09-07 23:16:04 +00:00
|
|
|
}
|
|
|
|
}
|