2019-09-13 20:30:29 +08:00
|
|
|
import { LabelledControl, LabelledControlOptions } from "./LabelledControl";
|
2020-01-05 08:07:35 +08:00
|
|
|
import { bind_attr, Icon, icon, input, span } from "./dom";
|
2019-09-13 20:30:29 +08:00
|
|
|
import "./ComboBox.css";
|
|
|
|
import "./Input.css";
|
|
|
|
import { Menu } from "./Menu";
|
|
|
|
import { Property } from "../observable/property/Property";
|
|
|
|
import { WritableProperty } from "../observable/property/WritableProperty";
|
|
|
|
import { WidgetProperty } from "../observable/property/WidgetProperty";
|
|
|
|
|
|
|
|
export type ComboBoxOptions<T> = LabelledControlOptions & {
|
2019-10-08 00:26:45 +08:00
|
|
|
items: readonly T[] | Property<readonly T[]>;
|
2019-09-14 19:16:13 +08:00
|
|
|
to_label(item: T): string;
|
2019-09-13 20:30:29 +08:00
|
|
|
placeholder_text?: string;
|
2019-09-14 19:16:13 +08:00
|
|
|
filter?(text: string): void;
|
2019-09-13 20:30:29 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
export class ComboBox<T> extends LabelledControl {
|
2019-12-27 07:55:32 +08:00
|
|
|
readonly element = span({ className: "core_ComboBox core_Input" });
|
2019-09-16 01:32:34 +08:00
|
|
|
|
2019-09-13 20:30:29 +08:00
|
|
|
readonly preferred_label_position = "left";
|
|
|
|
|
|
|
|
readonly selected: WritableProperty<T | undefined>;
|
|
|
|
|
|
|
|
private readonly to_label: (element: T) => string;
|
|
|
|
private readonly menu: Menu<T>;
|
2019-12-27 07:55:32 +08:00
|
|
|
private readonly input_element: HTMLInputElement = input();
|
2019-09-13 20:30:29 +08:00
|
|
|
private readonly _selected: WidgetProperty<T | undefined>;
|
|
|
|
|
|
|
|
constructor(options: ComboBoxOptions<T>) {
|
2019-09-16 01:32:34 +08:00
|
|
|
super(options);
|
2019-09-13 20:30:29 +08:00
|
|
|
|
|
|
|
this.to_label = options.to_label;
|
|
|
|
|
|
|
|
this._selected = new WidgetProperty<T | undefined>(this, undefined, this.set_selected);
|
|
|
|
this.selected = this._selected;
|
|
|
|
|
2019-12-24 10:04:18 +08:00
|
|
|
this.menu = this.disposable(
|
|
|
|
new Menu({
|
|
|
|
items: options.items,
|
|
|
|
to_label: options.to_label,
|
|
|
|
related_element: this.element,
|
|
|
|
}),
|
|
|
|
);
|
2019-09-13 20:30:29 +08:00
|
|
|
this.menu.element.onmousedown = e => e.preventDefault();
|
|
|
|
|
|
|
|
this.input_element.placeholder = options.placeholder_text || "";
|
|
|
|
this.input_element.onmousedown = () => {
|
2019-10-26 23:03:12 +08:00
|
|
|
this.menu.visible.set_val(true, { silent: false });
|
2019-09-13 20:30:29 +08:00
|
|
|
};
|
2019-09-14 19:16:13 +08:00
|
|
|
|
2019-09-14 02:53:31 +08:00
|
|
|
this.input_element.onkeydown = (e: Event) => {
|
|
|
|
const key = (e as KeyboardEvent).key;
|
|
|
|
|
|
|
|
switch (key) {
|
|
|
|
case "ArrowDown":
|
|
|
|
e.preventDefault();
|
|
|
|
this.menu.hover_next();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "ArrowUp":
|
|
|
|
e.preventDefault();
|
|
|
|
this.menu.hover_prev();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case "Enter":
|
|
|
|
this.menu.select_hovered();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
2019-09-14 19:16:13 +08:00
|
|
|
|
|
|
|
const filter = options.filter;
|
|
|
|
|
|
|
|
if (filter) {
|
|
|
|
let input_value = "";
|
|
|
|
|
|
|
|
this.input_element.onkeyup = () => {
|
|
|
|
if (this.input_element.value !== input_value) {
|
|
|
|
input_value = this.input_element.value;
|
|
|
|
filter(input_value);
|
|
|
|
|
|
|
|
if (this.menu.visible.val || input_value) {
|
|
|
|
this.menu.hover_next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-09-13 20:30:29 +08:00
|
|
|
this.input_element.onblur = () => {
|
2019-10-26 23:03:12 +08:00
|
|
|
this.menu.visible.set_val(false, { silent: false });
|
2019-09-13 20:30:29 +08:00
|
|
|
};
|
|
|
|
|
2019-12-27 23:56:48 +08:00
|
|
|
const down_arrow_element = icon(Icon.TriangleDown);
|
|
|
|
const up_arrow_element = icon(Icon.TriangleUp);
|
2019-12-27 07:55:32 +08:00
|
|
|
const button_element = span(
|
|
|
|
{ className: "core_ComboBox_button" },
|
2019-09-13 20:30:29 +08:00
|
|
|
down_arrow_element,
|
|
|
|
up_arrow_element,
|
|
|
|
);
|
|
|
|
button_element.onmousedown = e => {
|
|
|
|
e.preventDefault();
|
2019-10-26 23:03:12 +08:00
|
|
|
this.menu.visible.set_val(!this.menu.visible.val, { silent: false });
|
2019-09-13 20:30:29 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
this.element.append(
|
2019-12-27 07:55:32 +08:00
|
|
|
span(
|
|
|
|
{ className: "core_ComboBox_inner core_Input_inner" },
|
2019-09-13 20:30:29 +08:00
|
|
|
this.input_element,
|
|
|
|
button_element,
|
|
|
|
),
|
|
|
|
this.menu.element,
|
|
|
|
);
|
|
|
|
|
|
|
|
this.disposables(
|
2019-10-26 23:03:12 +08:00
|
|
|
this.menu.visible.observe(({ value: visible }) => {
|
2019-09-14 19:16:13 +08:00
|
|
|
if (visible) {
|
|
|
|
this.menu.hover_next();
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
|
2019-09-13 20:30:29 +08:00
|
|
|
this.menu.selected.observe(({ value }) => {
|
|
|
|
this.selected.set_val(value, { silent: false });
|
|
|
|
this.input_element.focus();
|
|
|
|
}),
|
2020-01-05 08:07:35 +08:00
|
|
|
|
|
|
|
bind_attr(
|
|
|
|
up_arrow_element,
|
|
|
|
"hidden",
|
|
|
|
this.menu.visible.map(v => !v),
|
|
|
|
),
|
|
|
|
|
|
|
|
bind_attr(down_arrow_element, "hidden", this.menu.visible),
|
2019-09-13 20:30:29 +08:00
|
|
|
);
|
2019-09-14 21:15:59 +08:00
|
|
|
|
2019-12-20 01:54:01 +08:00
|
|
|
this.finalize_construction();
|
2019-09-13 20:30:29 +08:00
|
|
|
}
|
|
|
|
|
2020-04-30 02:48:47 +08:00
|
|
|
protected set_enabled(enabled: boolean): void {
|
|
|
|
super.set_enabled(enabled);
|
|
|
|
this.input_element.disabled = !enabled;
|
|
|
|
this.menu.enabled.val = enabled;
|
|
|
|
}
|
|
|
|
|
2019-09-13 20:30:29 +08:00
|
|
|
protected set_selected(selected?: T): void {
|
|
|
|
this.input_element.value = selected ? this.to_label(selected) : "";
|
|
|
|
this.menu.selected.val = selected;
|
|
|
|
}
|
|
|
|
}
|