2019-08-23 23:00:39 +08:00
|
|
|
/* eslint-disable no-dupe-class-members */
|
2019-08-28 06:50:38 +08:00
|
|
|
import { LabelledControl, LabelledControlOptions } from "./LabelledControl";
|
2019-08-29 03:36:45 +08:00
|
|
|
import { create_element, el } from "./dom";
|
2019-08-30 00:24:03 +08:00
|
|
|
import { WritableProperty } from "../observable/property/WritableProperty";
|
|
|
|
import { is_any_property, Property } from "../observable/property/Property";
|
2019-08-23 23:00:39 +08:00
|
|
|
import "./Input.css";
|
2019-08-30 00:24:03 +08:00
|
|
|
import { WidgetProperty } from "../observable/property/WidgetProperty";
|
2019-08-28 06:50:38 +08:00
|
|
|
|
|
|
|
export type InputOptions = LabelledControlOptions;
|
2019-08-23 23:00:39 +08:00
|
|
|
|
2019-08-29 03:36:45 +08:00
|
|
|
export abstract class Input<T> extends LabelledControl<HTMLElement> {
|
2019-08-23 23:00:39 +08:00
|
|
|
readonly value: WritableProperty<T>;
|
|
|
|
|
2019-08-30 00:24:03 +08:00
|
|
|
protected readonly input_element: HTMLInputElement;
|
2019-08-23 23:00:39 +08:00
|
|
|
|
2019-08-28 06:50:38 +08:00
|
|
|
private readonly _value: WidgetProperty<T>;
|
|
|
|
|
2019-08-23 23:00:39 +08:00
|
|
|
protected constructor(
|
2019-08-28 06:50:38 +08:00
|
|
|
value: T,
|
2019-08-23 23:00:39 +08:00
|
|
|
class_name: string,
|
|
|
|
input_type: string,
|
|
|
|
input_class_name: string,
|
2019-08-28 06:50:38 +08:00
|
|
|
options?: InputOptions,
|
2019-08-23 23:00:39 +08:00
|
|
|
) {
|
2019-08-29 03:36:45 +08:00
|
|
|
super(el.span({ class: `${class_name} core_Input` }), options);
|
2019-08-23 23:00:39 +08:00
|
|
|
|
2019-08-28 06:50:38 +08:00
|
|
|
this._value = new WidgetProperty<T>(this, value, this.set_value);
|
|
|
|
this.value = this._value;
|
2019-08-23 23:00:39 +08:00
|
|
|
|
2019-08-30 00:24:03 +08:00
|
|
|
this.input_element = create_element("input", {
|
2019-08-23 23:00:39 +08:00
|
|
|
class: `${input_class_name} core_Input_inner`,
|
|
|
|
});
|
2019-08-30 00:24:03 +08:00
|
|
|
this.input_element.type = input_type;
|
|
|
|
this.input_element.onchange = () => {
|
2019-08-29 03:36:45 +08:00
|
|
|
this._value.set_val(this.get_value(), { silent: false });
|
2019-08-27 20:50:16 +08:00
|
|
|
};
|
2019-08-23 23:00:39 +08:00
|
|
|
|
2019-08-30 00:24:03 +08:00
|
|
|
this.element.append(this.input_element);
|
2019-08-27 20:50:16 +08:00
|
|
|
}
|
|
|
|
|
2019-08-28 06:50:38 +08:00
|
|
|
protected set_enabled(enabled: boolean): void {
|
|
|
|
super.set_enabled(enabled);
|
2019-08-30 00:24:03 +08:00
|
|
|
this.input_element.disabled = !enabled;
|
2019-08-27 20:50:16 +08:00
|
|
|
}
|
|
|
|
|
2019-08-28 06:50:38 +08:00
|
|
|
protected abstract get_value(): T;
|
2019-08-23 23:00:39 +08:00
|
|
|
|
2019-08-28 06:50:38 +08:00
|
|
|
protected abstract set_value(value: T): void;
|
2019-08-23 23:00:39 +08:00
|
|
|
|
|
|
|
protected set_attr<T>(attr: InputAttrsOfType<T>, value?: T | Property<T>): void;
|
|
|
|
protected set_attr<T, U>(
|
|
|
|
attr: InputAttrsOfType<U>,
|
|
|
|
value: T | Property<T> | undefined,
|
|
|
|
convert: (value: T) => U,
|
|
|
|
): void;
|
|
|
|
protected set_attr<T, U>(
|
|
|
|
attr: InputAttrsOfType<U>,
|
|
|
|
value?: T | Property<T>,
|
|
|
|
convert?: (value: T) => U,
|
|
|
|
): void {
|
|
|
|
if (value == undefined) return;
|
|
|
|
|
2019-08-30 00:24:03 +08:00
|
|
|
const input = this.input_element as any;
|
2019-08-23 23:00:39 +08:00
|
|
|
const cvt = convert ? convert : (v: T) => (v as any) as U;
|
|
|
|
|
|
|
|
if (is_any_property(value)) {
|
|
|
|
input[attr] = cvt(value.val);
|
2019-08-26 21:42:12 +08:00
|
|
|
this.disposable(value.observe(({ value }) => (input[attr] = cvt(value))));
|
2019-08-23 23:00:39 +08:00
|
|
|
} else {
|
|
|
|
input[attr] = cvt(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type InputAttrsOfType<T> = {
|
|
|
|
[K in keyof HTMLInputElement]: T extends HTMLInputElement[K] ? K : never;
|
|
|
|
}[keyof HTMLInputElement];
|