import { LabelledControl, LabelledControlOptions, LabelPosition } from "./LabelledControl"; import { disposable_listener, el, Icon } from "./dom"; import "./Select.css"; import { is_any_property, Property } from "../observable/property/Property"; import { Button } from "./Button"; import { WritableProperty } from "../observable/property/WritableProperty"; import { WidgetProperty } from "../observable/property/WidgetProperty"; import { Menu } from "./Menu"; export type SelectOptions = LabelledControlOptions & { selected: T | Property; }; export class Select extends LabelledControl { readonly preferred_label_position: LabelPosition; readonly selected: WritableProperty; private readonly to_label: (element: T) => string; private readonly button: Button; private readonly menu: Menu; private readonly _selected: WidgetProperty; private just_opened: boolean; constructor( items: T[] | Property, to_label: (element: T) => string, options?: SelectOptions, ) { const element = el.div({ class: "core_Select" }); const button = new Button("", { icon_right: Icon.TriangleDown, }); const menu = new Menu(items, to_label, element); super(element, options); this.preferred_label_position = "left"; this.to_label = to_label; this.button = this.disposable(button); this.menu = this.disposable(menu); this.element.append(this.button.element, this.menu.element); this._selected = new WidgetProperty(this, undefined, this.set_selected); this.selected = this._selected; this.just_opened = false; this.disposables( disposable_listener(button.element, "mousedown", e => this.button_mousedown(e)), button.mouseup.observe(() => this.button_mouseup()), this.menu.selected.observe(({ value }) => this._selected.set_val(value, { silent: false }), ), ); if (options) { if (is_any_property(options.selected)) { this.selected.bind_to(options.selected); } else if (options.selected) { this.selected.val = options.selected; } } } protected set_enabled(enabled: boolean): void { super.set_enabled(enabled); this.button.enabled.val = enabled; } protected set_selected(selected?: T): void { this.button.text.val = selected ? this.to_label(selected) : ""; this.menu.selected.val = selected; } private button_mousedown(e: Event): void { e.stopPropagation(); this.just_opened = !this.menu.visible.val; this.menu.visible.val = true; } private button_mouseup(): void { if (!this.just_opened) { this.menu.visible.val = false; } this.just_opened = false; } }