2024-05-16 09:29:58 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
import * as React from "react";
|
|
|
|
import * as jotai from "jotai";
|
|
|
|
import { TabContent } from "@/app/tab/tab";
|
|
|
|
import { clsx } from "clsx";
|
2024-05-28 00:44:57 +02:00
|
|
|
import { atoms } from "@/store/global";
|
2024-05-27 22:59:58 +02:00
|
|
|
import * as WOS from "@/store/wos";
|
|
|
|
import { CenteredLoadingDiv, CenteredDiv } from "../element/quickelems";
|
2024-05-16 09:29:58 +02:00
|
|
|
|
|
|
|
import "./workspace.less";
|
|
|
|
|
2024-05-27 09:47:10 +02:00
|
|
|
function Tab({ tabId }: { tabId: string }) {
|
2024-05-27 22:59:58 +02:00
|
|
|
const windowData = jotai.useAtomValue(atoms.waveWindow);
|
|
|
|
const [tabData, tabLoading] = WOS.useWaveObjectValue<Tab>(WOS.makeORef("tab", tabId));
|
2024-05-28 01:33:31 +02:00
|
|
|
function setActiveTab() {
|
2024-05-27 23:31:12 +02:00
|
|
|
WOS.SetActiveTab(tabId);
|
2024-05-27 09:47:10 +02:00
|
|
|
}
|
2024-05-28 01:33:31 +02:00
|
|
|
function handleCloseTab() {
|
|
|
|
WOS.CloseTab(tabId);
|
|
|
|
}
|
2024-05-16 09:29:58 +02:00
|
|
|
return (
|
2024-05-27 09:47:10 +02:00
|
|
|
<div
|
|
|
|
className={clsx("tab", { active: tabData != null && windowData.activetabid === tabData.oid })}
|
2024-05-28 01:33:31 +02:00
|
|
|
onClick={() => setActiveTab()}
|
2024-05-27 09:47:10 +02:00
|
|
|
>
|
2024-05-28 01:33:31 +02:00
|
|
|
<div className="tab-close" onClick={() => handleCloseTab()}>
|
|
|
|
<div>
|
|
|
|
<i className="fa fa-solid fa-xmark" />
|
|
|
|
</div>
|
|
|
|
</div>
|
2024-05-27 09:47:10 +02:00
|
|
|
{tabData?.name ?? "..."}
|
2024-05-16 09:29:58 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-05-28 00:44:57 +02:00
|
|
|
function TabBar({ workspace }: { workspace: Workspace }) {
|
2024-05-16 09:29:58 +02:00
|
|
|
function handleAddTab() {
|
2024-05-27 22:59:58 +02:00
|
|
|
const newTabName = `Tab-${workspace.tabids.length + 1}`;
|
|
|
|
WOS.AddTabToWorkspace(newTabName, true);
|
2024-05-16 09:29:58 +02:00
|
|
|
}
|
2024-05-27 09:47:10 +02:00
|
|
|
const tabIds = workspace?.tabids ?? [];
|
2024-05-16 09:29:58 +02:00
|
|
|
return (
|
|
|
|
<div className="tab-bar">
|
2024-05-27 09:47:10 +02:00
|
|
|
{tabIds.map((tabid, idx) => {
|
|
|
|
return <Tab key={idx} tabId={tabid} />;
|
2024-05-16 09:29:58 +02:00
|
|
|
})}
|
|
|
|
<div className="tab-add" onClick={() => handleAddTab()}>
|
|
|
|
<i className="fa fa-solid fa-plus fa-fw" />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function Widgets() {
|
2024-05-27 22:59:58 +02:00
|
|
|
const windowData = jotai.useAtomValue(atoms.waveWindow);
|
2024-05-27 09:47:10 +02:00
|
|
|
const activeTabId = windowData.activetabid;
|
2024-05-16 09:29:58 +02:00
|
|
|
|
2024-05-28 00:44:57 +02:00
|
|
|
async function createBlock(blockDef: BlockDef) {
|
|
|
|
const rtOpts: RuntimeOpts = { termsize: { rows: 25, cols: 80 } };
|
|
|
|
await WOS.CreateBlock(blockDef, rtOpts);
|
2024-05-16 09:29:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async function clickTerminal() {
|
2024-05-28 00:44:57 +02:00
|
|
|
const termBlockDef = {
|
2024-05-16 09:29:58 +02:00
|
|
|
controller: "shell",
|
|
|
|
view: "term",
|
2024-05-28 00:44:57 +02:00
|
|
|
};
|
2024-05-16 09:29:58 +02:00
|
|
|
createBlock(termBlockDef);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function clickPreview(fileName: string) {
|
2024-05-28 00:44:57 +02:00
|
|
|
const markdownDef = {
|
2024-05-16 09:29:58 +02:00
|
|
|
view: "preview",
|
|
|
|
meta: { file: fileName },
|
2024-05-28 00:44:57 +02:00
|
|
|
};
|
2024-05-16 09:29:58 +02:00
|
|
|
createBlock(markdownDef);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function clickPlot() {
|
2024-05-28 00:44:57 +02:00
|
|
|
const plotDef: BlockDef = {
|
2024-05-16 22:54:15 +02:00
|
|
|
view: "plot",
|
2024-05-28 00:44:57 +02:00
|
|
|
};
|
2024-05-16 22:54:15 +02:00
|
|
|
createBlock(plotDef);
|
2024-05-16 09:29:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="workspace-widgets">
|
|
|
|
<div className="widget" onClick={() => clickTerminal()}>
|
|
|
|
<i className="fa fa-solid fa-square-terminal fa-fw" />
|
|
|
|
</div>
|
|
|
|
<div className="widget" onClick={() => clickPreview("README.md")}>
|
|
|
|
<i className="fa fa-solid fa-files fa-fw" />
|
|
|
|
</div>
|
|
|
|
<div className="widget" onClick={() => clickPreview("go.mod")}>
|
|
|
|
<i className="fa fa-solid fa-files fa-fw" />
|
|
|
|
</div>
|
2024-05-17 07:48:23 +02:00
|
|
|
<div className="widget" onClick={() => clickPreview("build/appicon.png")}>
|
|
|
|
<i className="fa fa-solid fa-files fa-fw" />
|
|
|
|
</div>
|
2024-05-20 20:39:23 +02:00
|
|
|
<div className="widget" onClick={() => clickPreview("~")}>
|
|
|
|
<i className="fa fa-solid fa-files fa-fw" />
|
|
|
|
</div>
|
2024-05-16 09:29:58 +02:00
|
|
|
<div className="widget" onClick={() => clickPlot()}>
|
|
|
|
<i className="fa fa-solid fa-chart-simple fa-fw" />
|
|
|
|
</div>
|
|
|
|
<div className="widget no-hover">
|
|
|
|
<i className="fa fa-solid fa-plus fa-fw" />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-05-24 23:08:24 +02:00
|
|
|
function WorkspaceElem() {
|
2024-05-27 22:59:58 +02:00
|
|
|
const windowData = jotai.useAtomValue(atoms.waveWindow);
|
2024-05-27 09:47:10 +02:00
|
|
|
const activeTabId = windowData?.activetabid;
|
2024-05-27 22:59:58 +02:00
|
|
|
const ws = jotai.useAtomValue(atoms.workspace);
|
2024-05-16 09:29:58 +02:00
|
|
|
return (
|
|
|
|
<div className="workspace">
|
2024-05-28 00:44:57 +02:00
|
|
|
<TabBar workspace={ws} />
|
2024-05-16 09:29:58 +02:00
|
|
|
<div className="workspace-tabcontent">
|
2024-05-28 01:33:31 +02:00
|
|
|
{activeTabId == "" ? (
|
|
|
|
<CenteredDiv>No Active Tab</CenteredDiv>
|
|
|
|
) : (
|
|
|
|
<>
|
|
|
|
<TabContent key={windowData.workspaceid} tabId={activeTabId} />
|
|
|
|
<Widgets />
|
|
|
|
</>
|
|
|
|
)}
|
2024-05-16 09:29:58 +02:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-05-24 23:08:24 +02:00
|
|
|
export { WorkspaceElem as Workspace };
|