mapcomplete/UI/Input/Slider.ts

60 lines
1.8 KiB
TypeScript
Raw Normal View History

2022-09-08 21:40:48 +02:00
import { InputElement } from "./InputElement"
import { UIEventSource } from "../../Logic/UIEventSource"
export default class Slider extends InputElement<number> {
private readonly _value: UIEventSource<number>
2022-09-08 21:40:48 +02:00
private readonly min: number
private readonly max: number
private readonly step: number
private readonly vertical: boolean
/**
* Constructs a slider input element for natural numbers
* @param min: the minimum value that is allowed, inclusive
* @param max: the max value that is allowed, inclusive
* @param options: value: injectable value; step: the step size of the slider
*/
2022-09-08 21:40:48 +02:00
constructor(
min: number,
max: number,
options?: {
value?: UIEventSource<number>
step?: 1 | number
vertical?: false | boolean
}
) {
super()
this.max = max
this.min = min
this._value = options?.value ?? new UIEventSource<number>(min)
2022-09-08 21:40:48 +02:00
this.step = options?.step ?? 1
this.vertical = options?.vertical ?? false
}
GetValue(): UIEventSource<number> {
2022-09-08 21:40:48 +02:00
return this._value
}
protected InnerConstructElement(): HTMLElement {
const el = document.createElement("input")
el.type = "range"
el.min = "" + this.min
el.max = "" + this.max
el.step = "" + this.step
const valuestore = this._value
el.oninput = () => {
valuestore.setData(Number(el.value))
}
2022-09-08 21:40:48 +02:00
if (this.vertical) {
2022-07-18 13:50:14 +02:00
el.classList.add("vertical")
2022-09-08 21:40:48 +02:00
el.setAttribute("orient", "vertical") // firefox only workaround...
2022-07-18 13:50:14 +02:00
}
2022-09-08 21:40:48 +02:00
valuestore.addCallbackAndRunD((v) => (el.value = "" + valuestore.data))
return el
}
IsValid(t: number): boolean {
2022-09-08 21:40:48 +02:00
return Math.round(t) == t && t >= this.min && t <= this.max
}
2022-09-08 21:40:48 +02:00
}