mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
102 lines
2.8 KiB
TypeScript
102 lines
2.8 KiB
TypeScript
import { Widget, WidgetOptions } from "./Widget";
|
|
import { create_element, el } from "./dom";
|
|
import { LazyWidget } from "./LazyWidget";
|
|
import { Resizable } from "./Resizable";
|
|
import { ResizableWidget } from "./ResizableWidget";
|
|
import "./TabContainer.css";
|
|
|
|
export type Tab = {
|
|
title: string;
|
|
key: string;
|
|
create_view: () => Promise<Widget & Resizable>;
|
|
};
|
|
|
|
export type TabContainerOptions = WidgetOptions & {
|
|
tabs: Tab[];
|
|
};
|
|
|
|
type TabInfo = Tab & { tab_element: HTMLSpanElement; lazy_view: LazyWidget };
|
|
|
|
const BAR_HEIGHT = 28;
|
|
|
|
export class TabContainer extends ResizableWidget {
|
|
readonly element = el.div({ class: "core_TabContainer" });
|
|
|
|
private tabs: TabInfo[] = [];
|
|
private bar_element = el.div({ class: "core_TabContainer_Bar" });
|
|
private panes_element = el.div({ class: "core_TabContainer_Panes" });
|
|
|
|
constructor(options: TabContainerOptions) {
|
|
super(options);
|
|
|
|
this.bar_element.onmousedown = this.bar_mousedown;
|
|
|
|
for (const tab of options.tabs) {
|
|
const tab_element = create_element("span", {
|
|
class: "core_TabContainer_Tab",
|
|
text: tab.title,
|
|
data: { key: tab.key },
|
|
});
|
|
this.bar_element.append(tab_element);
|
|
|
|
const lazy_view = new LazyWidget(tab.create_view);
|
|
|
|
this.tabs.push({
|
|
...tab,
|
|
tab_element,
|
|
lazy_view,
|
|
});
|
|
|
|
this.panes_element.append(lazy_view.element);
|
|
this.disposable(lazy_view);
|
|
}
|
|
|
|
if (this.tabs.length) {
|
|
this.activate(this.tabs[0].key);
|
|
}
|
|
|
|
this.element.append(this.bar_element, this.panes_element);
|
|
|
|
this.finalize_construction(TabContainer.prototype);
|
|
}
|
|
|
|
resize(width: number, height: number): this {
|
|
super.resize(width, height);
|
|
|
|
this.bar_element.style.width = `${width}px`;
|
|
this.bar_element.style.height = `${BAR_HEIGHT}px`;
|
|
|
|
const tab_pane_height = height - BAR_HEIGHT;
|
|
|
|
this.panes_element.style.width = `${width}px`;
|
|
this.panes_element.style.height = `${tab_pane_height}px`;
|
|
|
|
for (const tabs of this.tabs) {
|
|
tabs.lazy_view.resize(width, tab_pane_height);
|
|
}
|
|
|
|
return this;
|
|
}
|
|
|
|
private bar_mousedown = (e: MouseEvent) => {
|
|
if (e.target instanceof HTMLElement) {
|
|
const key = e.target.dataset["key"];
|
|
if (key) this.activate(key);
|
|
}
|
|
};
|
|
|
|
private activate(key: string): void {
|
|
for (const tab of this.tabs) {
|
|
const active = tab.key === key;
|
|
|
|
if (active) {
|
|
tab.tab_element.classList.add("active");
|
|
} else {
|
|
tab.tab_element.classList.remove("active");
|
|
}
|
|
|
|
tab.lazy_view.visible.val = active;
|
|
}
|
|
}
|
|
}
|