mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 16:28:30 +08:00
95 lines
2.6 KiB
TypeScript
95 lines
2.6 KiB
TypeScript
![]() |
import { View } from "./View";
|
||
|
import { create_el } from "./dom";
|
||
|
import { LazyView } from "./LazyView";
|
||
|
import { Resizable } from "./Resizable";
|
||
|
import { ResizableView } from "./ResizableView";
|
||
|
import "./TabContainer.css";
|
||
|
|
||
|
export type Tab = {
|
||
|
title: string;
|
||
|
key: string;
|
||
|
create_view: () => Promise<View & Resizable>;
|
||
|
};
|
||
|
|
||
|
type TabInfo = Tab & { tab_element: HTMLSpanElement; lazy_view: LazyView };
|
||
|
|
||
|
const BAR_HEIGHT = 28;
|
||
|
|
||
|
export class TabContainer extends ResizableView {
|
||
|
element = create_el("div", "core_TabContainer");
|
||
|
|
||
|
private tabs: TabInfo[] = [];
|
||
|
private bar_element = create_el("div", "core_TabContainer_Bar");
|
||
|
private panes_element = create_el("div", "core_TabContainer_Panes");
|
||
|
|
||
|
constructor(...tabs: Tab[]) {
|
||
|
super();
|
||
|
|
||
|
this.bar_element.onclick = this.bar_click;
|
||
|
|
||
|
for (const tab of tabs) {
|
||
|
const tab_element = create_el("span", "core_TabContainer_Tab", tab_element => {
|
||
|
tab_element.textContent = tab.title;
|
||
|
tab_element.dataset["key"] = tab.key;
|
||
|
});
|
||
|
this.bar_element.append(tab_element);
|
||
|
|
||
|
const lazy_view = new LazyView(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);
|
||
|
}
|
||
|
|
||
|
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_click = (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 = active;
|
||
|
}
|
||
|
}
|
||
|
}
|