import { disposable_listener, el, Icon } from "./dom"; import "./DropDown.css"; import { Property } from "../observable/property/Property"; import { Button, ButtonOptions } from "./Button"; import { Menu } from "./Menu"; import { Control } from "./Control"; import { Observable } from "../observable/Observable"; import { Emitter } from "../observable/Emitter"; import { emitter } from "../observable"; export type DropDownOptions = ButtonOptions; export class DropDown extends Control { readonly chosen: Observable; private readonly button: Button; private readonly menu: Menu; private readonly _chosen: Emitter; private just_opened: boolean; constructor( text: string, items: T[] | Property, to_label: (element: T) => string, options?: DropDownOptions, ) { const element = el.div({ class: "core_DropDown" }); const button = new Button(text, { icon_left: options && options.icon_left, icon_right: Icon.TriangleDown, }); const menu = new Menu(items, to_label, element); super(element, options); this.button = this.disposable(button); this.menu = this.disposable(menu); this.element.append(this.button.element, this.menu.element); this._chosen = emitter(); this.chosen = this._chosen; this.just_opened = false; this.disposables( disposable_listener(button.element, "mousedown", () => this.button_mousedown(), { capture: true, }), button.mouseup.observe(() => this.button_mouseup()), this.menu.selected.observe(({ value }) => { if (value) { this._chosen.emit({ value }); this.menu.selected.val = undefined; } }), ); } protected set_enabled(enabled: boolean): void { super.set_enabled(enabled); this.button.enabled.val = enabled; } private button_mousedown(): void { this.just_opened = !this.menu.visible.val; this.menu.visible.val = true; } private button_mouseup(): void { if (this.just_opened) { this.menu.focus(); } else { this.menu.visible.val = false; } this.just_opened = false; } }