mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-10 19:58:00 +01:00
138 lines
4.2 KiB
TypeScript
138 lines
4.2 KiB
TypeScript
// Copyright 2024, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
import { ModalsRenderer } from "@/app/modals/modalsrenderer";
|
|
import { TabBar } from "@/app/tab/tabbar";
|
|
import { TabContent } from "@/app/tab/tabcontent";
|
|
import { atoms, createBlock } from "@/store/global";
|
|
import * as util from "@/util/util";
|
|
import * as jotai from "jotai";
|
|
import * as React from "react";
|
|
import { CenteredDiv } from "../element/quickelems";
|
|
|
|
import { ErrorBoundary } from "@/app/element/errorboundary";
|
|
import "./workspace.less";
|
|
|
|
const iconRegex = /^[a-z0-9-]+$/;
|
|
|
|
function keyLen(obj: Object): number {
|
|
if (obj == null) {
|
|
return 0;
|
|
}
|
|
return Object.keys(obj).length;
|
|
}
|
|
|
|
function sortByDisplayOrder(wmap: { [key: string]: WidgetConfigType }): WidgetConfigType[] {
|
|
if (wmap == null) {
|
|
return [];
|
|
}
|
|
const wlist = Object.values(wmap);
|
|
wlist.sort((a, b) => {
|
|
return a["display:order"] - b["display:order"];
|
|
});
|
|
return wlist;
|
|
}
|
|
|
|
const Widgets = React.memo(() => {
|
|
const fullConfig = jotai.useAtomValue(atoms.fullConfigAtom);
|
|
const newWidgetModalVisible = React.useState(false);
|
|
const helpWidget: WidgetConfigType = {
|
|
icon: "circle-question",
|
|
label: "help",
|
|
blockdef: {
|
|
meta: {
|
|
view: "help",
|
|
},
|
|
},
|
|
};
|
|
const tipsWidget: WidgetConfigType = {
|
|
icon: "lightbulb",
|
|
label: "tips",
|
|
blockdef: {
|
|
meta: {
|
|
view: "tips",
|
|
},
|
|
},
|
|
};
|
|
const showHelp = fullConfig?.settings?.["widget:showhelp"] ?? true;
|
|
const showDivider = keyLen(fullConfig?.defaultwidgets) > 0 && keyLen(fullConfig?.widgets) > 0;
|
|
const defaultWidgets = sortByDisplayOrder(fullConfig?.defaultwidgets);
|
|
const widgets = sortByDisplayOrder(fullConfig?.widgets);
|
|
return (
|
|
<div className="workspace-widgets">
|
|
{defaultWidgets.map((data, idx) => (
|
|
<Widget key={`defwidget-${idx}`} widget={data} />
|
|
))}
|
|
{showDivider ? <div className="widget-divider" /> : null}
|
|
{widgets?.map((data, idx) => <Widget key={`widget-${idx}`} widget={data} />)}
|
|
{showHelp ? (
|
|
<>
|
|
<div className="widget-spacer" />
|
|
<Widget key="tips" widget={tipsWidget} />
|
|
<Widget key="help" widget={helpWidget} />
|
|
</>
|
|
) : null}
|
|
</div>
|
|
);
|
|
});
|
|
|
|
async function handleWidgetSelect(blockDef: BlockDef) {
|
|
createBlock(blockDef);
|
|
}
|
|
|
|
function isIconValid(icon: string): boolean {
|
|
if (util.isBlank(icon)) {
|
|
return false;
|
|
}
|
|
return icon.match(iconRegex) != null;
|
|
}
|
|
|
|
function getIconClass(icon: string): string {
|
|
if (!isIconValid(icon)) {
|
|
return "fa fa-regular fa-browser fa-fw";
|
|
}
|
|
return `fa fa-solid fa-${icon} fa-fw`;
|
|
}
|
|
|
|
const Widget = React.memo(({ widget }: { widget: WidgetConfigType }) => {
|
|
return (
|
|
<div
|
|
className="widget"
|
|
onClick={() => handleWidgetSelect(widget.blockdef)}
|
|
title={widget.description || widget.label}
|
|
>
|
|
<div className="widget-icon" style={{ color: widget.color }}>
|
|
<i className={getIconClass(widget.icon)}></i>
|
|
</div>
|
|
{!util.isBlank(widget.label) ? <div className="widget-label">{widget.label}</div> : null}
|
|
</div>
|
|
);
|
|
});
|
|
|
|
const WorkspaceElem = React.memo(() => {
|
|
const windowData = jotai.useAtomValue(atoms.waveWindow);
|
|
const activeTabId = windowData?.activetabid;
|
|
const ws = jotai.useAtomValue(atoms.workspace);
|
|
|
|
return (
|
|
<div className="workspace">
|
|
<TabBar key={ws.oid} workspace={ws} />
|
|
<div className="workspace-tabcontent">
|
|
<ErrorBoundary key={activeTabId}>
|
|
{activeTabId == "" ? (
|
|
<CenteredDiv>No Active Tab</CenteredDiv>
|
|
) : (
|
|
<>
|
|
<TabContent key={activeTabId} tabId={activeTabId} />
|
|
<Widgets />
|
|
<ModalsRenderer />
|
|
</>
|
|
)}
|
|
</ErrorBoundary>
|
|
</div>
|
|
</div>
|
|
);
|
|
});
|
|
|
|
export { WorkspaceElem as Workspace };
|