2024-05-14 08:45:41 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-07-09 08:13:12 +02:00
|
|
|
import { ContextMenuModel } from "@/app/store/contextmenu";
|
2024-05-14 18:37:41 +02:00
|
|
|
import { Markdown } from "@/element/markdown";
|
2024-07-18 03:42:49 +02:00
|
|
|
import { globalStore, useBlockAtom } from "@/store/global";
|
2024-06-12 02:42:10 +02:00
|
|
|
import * as services from "@/store/services";
|
2024-05-28 21:12:28 +02:00
|
|
|
import * as WOS from "@/store/wos";
|
2024-07-18 03:49:27 +02:00
|
|
|
import { getWebServerEndpoint } from "@/util/endpoints";
|
2024-05-14 21:29:41 +02:00
|
|
|
import * as util from "@/util/util";
|
2024-06-04 03:22:26 +02:00
|
|
|
import clsx from "clsx";
|
2024-05-28 21:12:28 +02:00
|
|
|
import * as jotai from "jotai";
|
2024-07-09 00:04:48 +02:00
|
|
|
import { loadable } from "jotai/utils";
|
2024-06-24 19:17:35 +02:00
|
|
|
import { useRef } from "react";
|
2024-05-17 07:48:23 +02:00
|
|
|
import { CenteredDiv } from "../element/quickelems";
|
2024-07-09 08:13:12 +02:00
|
|
|
import { CodeEdit } from "./codeedit/codeedit";
|
2024-06-24 19:17:35 +02:00
|
|
|
import { CSVView } from "./csvview";
|
2024-05-29 09:00:36 +02:00
|
|
|
import { DirectoryPreview } from "./directorypreview";
|
2024-05-14 08:45:41 +02:00
|
|
|
import "./view.less";
|
|
|
|
|
2024-05-17 07:48:23 +02:00
|
|
|
const MaxFileSize = 1024 * 1024 * 10; // 10MB
|
2024-07-09 00:04:48 +02:00
|
|
|
const MaxCSVSize = 1024 * 1024 * 1; // 1MB
|
|
|
|
|
2024-07-18 01:11:16 +02:00
|
|
|
function isTextFile(mimeType: string): boolean {
|
|
|
|
return (
|
|
|
|
mimeType.startsWith("text/") ||
|
|
|
|
mimeType == "application/sql" ||
|
|
|
|
(mimeType.startsWith("application/") &&
|
|
|
|
(mimeType.includes("json") || mimeType.includes("yaml") || mimeType.includes("toml"))) ||
|
|
|
|
mimeType == "application/pem-certificate-chain"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-07-09 00:04:48 +02:00
|
|
|
export class PreviewModel implements ViewModel {
|
|
|
|
blockId: string;
|
|
|
|
blockAtom: jotai.Atom<Block>;
|
2024-07-09 08:13:12 +02:00
|
|
|
viewIcon: jotai.Atom<string | HeaderIconButton>;
|
2024-07-09 00:04:48 +02:00
|
|
|
viewName: jotai.Atom<string>;
|
|
|
|
viewText: jotai.Atom<string>;
|
2024-07-09 08:13:12 +02:00
|
|
|
preIconButton: jotai.Atom<HeaderIconButton>;
|
|
|
|
endIconButtons: jotai.Atom<HeaderIconButton[]>;
|
2024-07-09 00:04:48 +02:00
|
|
|
|
|
|
|
fileName: jotai.WritableAtom<string, [string], void>;
|
|
|
|
statFile: jotai.Atom<Promise<FileInfo>>;
|
|
|
|
fullFile: jotai.Atom<Promise<FullFile>>;
|
|
|
|
fileMimeType: jotai.Atom<Promise<string>>;
|
|
|
|
fileMimeTypeLoadable: jotai.Atom<Loadable<string>>;
|
|
|
|
fileContent: jotai.Atom<Promise<string>>;
|
|
|
|
|
2024-07-09 01:36:30 +02:00
|
|
|
showHiddenFiles: jotai.PrimitiveAtom<boolean>;
|
2024-07-09 08:13:12 +02:00
|
|
|
refreshVersion: jotai.PrimitiveAtom<number>;
|
2024-07-09 01:36:30 +02:00
|
|
|
refreshCallback: () => void;
|
|
|
|
|
2024-07-09 00:04:48 +02:00
|
|
|
setPreviewFileName(fileName: string) {
|
|
|
|
services.ObjectService.UpdateObjectMeta(`block:${this.blockId}`, { file: fileName });
|
|
|
|
}
|
|
|
|
|
|
|
|
constructor(blockId: string) {
|
|
|
|
this.blockId = blockId;
|
2024-07-09 01:36:30 +02:00
|
|
|
this.showHiddenFiles = jotai.atom(true);
|
2024-07-09 08:13:12 +02:00
|
|
|
this.refreshVersion = jotai.atom(0);
|
2024-07-09 00:04:48 +02:00
|
|
|
this.blockAtom = WOS.getWaveObjectAtom<Block>(`block:${blockId}`);
|
|
|
|
this.viewIcon = jotai.atom((get) => {
|
|
|
|
let blockData = get(this.blockAtom);
|
|
|
|
if (blockData?.meta?.icon) {
|
|
|
|
return blockData.meta.icon;
|
|
|
|
}
|
|
|
|
const mimeType = util.jotaiLoadableValue(get(this.fileMimeTypeLoadable), "");
|
2024-07-09 08:13:12 +02:00
|
|
|
if (mimeType == "directory") {
|
|
|
|
return {
|
|
|
|
elemtype: "iconbutton",
|
|
|
|
icon: "folder-open",
|
|
|
|
longClick: (e: React.MouseEvent<any>) => {
|
|
|
|
let menuItems: ContextMenuItem[] = [];
|
|
|
|
menuItems.push({ label: "Go to Home", click: () => globalStore.set(this.fileName, "~") });
|
|
|
|
menuItems.push({
|
|
|
|
label: "Go to Desktop",
|
|
|
|
click: () => globalStore.set(this.fileName, "~/Desktop"),
|
|
|
|
});
|
|
|
|
menuItems.push({
|
|
|
|
label: "Go to Downloads",
|
|
|
|
click: () => globalStore.set(this.fileName, "~/Downloads"),
|
|
|
|
});
|
|
|
|
menuItems.push({
|
|
|
|
label: "Go to Documents",
|
|
|
|
click: () => globalStore.set(this.fileName, "~/Documents"),
|
|
|
|
});
|
|
|
|
menuItems.push({ label: "Go to Root", click: () => globalStore.set(this.fileName, "/") });
|
|
|
|
ContextMenuModel.showContextMenu(menuItems, e);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
2024-07-09 00:04:48 +02:00
|
|
|
const fileName = get(this.fileName);
|
|
|
|
return iconForFile(mimeType, fileName);
|
|
|
|
});
|
|
|
|
this.viewName = jotai.atom("Preview");
|
|
|
|
this.viewText = jotai.atom((get) => {
|
|
|
|
return get(this.fileName);
|
|
|
|
});
|
2024-07-09 01:36:30 +02:00
|
|
|
this.preIconButton = jotai.atom((get) => {
|
|
|
|
const mimeType = util.jotaiLoadableValue(get(this.fileMimeTypeLoadable), "");
|
|
|
|
if (mimeType == "directory") {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return {
|
2024-07-09 08:13:12 +02:00
|
|
|
elemtype: "iconbutton",
|
2024-07-09 01:36:30 +02:00
|
|
|
icon: "chevron-left",
|
|
|
|
click: this.onBack.bind(this),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
this.endIconButtons = jotai.atom((get) => {
|
2024-07-09 00:04:48 +02:00
|
|
|
const mimeType = util.jotaiLoadableValue(get(this.fileMimeTypeLoadable), "");
|
|
|
|
if (mimeType == "directory") {
|
2024-07-09 01:36:30 +02:00
|
|
|
let showHiddenFiles = get(this.showHiddenFiles);
|
|
|
|
return [
|
|
|
|
{
|
2024-07-09 08:13:12 +02:00
|
|
|
elemtype: "iconbutton",
|
2024-07-09 01:36:30 +02:00
|
|
|
icon: showHiddenFiles ? "eye" : "eye-slash",
|
|
|
|
click: () => {
|
|
|
|
globalStore.set(this.showHiddenFiles, (prev) => !prev);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2024-07-09 08:13:12 +02:00
|
|
|
elemtype: "iconbutton",
|
2024-07-09 01:36:30 +02:00
|
|
|
icon: "arrows-rotate",
|
|
|
|
click: () => this.refreshCallback?.(),
|
|
|
|
},
|
|
|
|
];
|
2024-07-09 00:04:48 +02:00
|
|
|
}
|
2024-07-09 01:36:30 +02:00
|
|
|
return null;
|
2024-07-09 00:04:48 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
this.fileName = jotai.atom<string, [string], void>(
|
|
|
|
(get) => {
|
|
|
|
return get(this.blockAtom)?.meta?.file;
|
|
|
|
},
|
|
|
|
(get, set, update) => {
|
|
|
|
services.ObjectService.UpdateObjectMeta(`block:${blockId}`, { file: update });
|
|
|
|
}
|
|
|
|
);
|
|
|
|
this.statFile = jotai.atom<Promise<FileInfo>>(async (get) => {
|
|
|
|
const fileName = get(this.fileName);
|
|
|
|
if (fileName == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
// const statFile = await FileService.StatFile(fileName);
|
|
|
|
console.log("PreviewModel calling StatFile", fileName);
|
|
|
|
const statFile = await services.FileService.StatFile(fileName);
|
|
|
|
return statFile;
|
|
|
|
});
|
|
|
|
this.fullFile = jotai.atom<Promise<FullFile>>(async (get) => {
|
|
|
|
const fileName = get(this.fileName);
|
|
|
|
if (fileName == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
// const file = await FileService.ReadFile(fileName);
|
|
|
|
const file = await services.FileService.ReadFile(fileName);
|
|
|
|
return file;
|
|
|
|
});
|
|
|
|
this.fileMimeType = jotai.atom<Promise<string>>(async (get) => {
|
|
|
|
const fileInfo = await get(this.statFile);
|
|
|
|
return fileInfo?.mimetype;
|
|
|
|
});
|
|
|
|
this.fileMimeTypeLoadable = loadable(this.fileMimeType);
|
|
|
|
this.fileContent = jotai.atom<Promise<string>>(async (get) => {
|
|
|
|
const fullFile = await get(this.fullFile);
|
|
|
|
return util.base64ToString(fullFile?.data64);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.onBack = this.onBack.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
onBack() {
|
|
|
|
const fileName = globalStore.get(this.fileName);
|
|
|
|
if (fileName == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const splitPath = fileName.split("/");
|
|
|
|
console.log("splitPath-1", splitPath);
|
|
|
|
splitPath.pop();
|
|
|
|
console.log("splitPath-2", splitPath);
|
|
|
|
const newPath = splitPath.join("/");
|
|
|
|
globalStore.set(this.fileName, newPath);
|
|
|
|
}
|
2024-07-09 00:32:10 +02:00
|
|
|
|
|
|
|
getSettingsMenuItems(): ContextMenuItem[] {
|
|
|
|
const menuItems: ContextMenuItem[] = [];
|
|
|
|
menuItems.push({
|
|
|
|
label: "Copy Full Path",
|
|
|
|
click: () => {
|
|
|
|
const fileName = globalStore.get(this.fileName);
|
|
|
|
if (fileName == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
navigator.clipboard.writeText(fileName);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
menuItems.push({
|
|
|
|
label: "Copy File Name",
|
|
|
|
click: () => {
|
|
|
|
let fileName = globalStore.get(this.fileName);
|
|
|
|
if (fileName == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (fileName.endsWith("/")) {
|
|
|
|
fileName = fileName.substring(0, fileName.length - 1);
|
|
|
|
}
|
|
|
|
const splitPath = fileName.split("/");
|
|
|
|
const baseName = splitPath[splitPath.length - 1];
|
|
|
|
navigator.clipboard.writeText(baseName);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
return menuItems;
|
|
|
|
}
|
2024-07-09 00:04:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function makePreviewModel(blockId: string): PreviewModel {
|
|
|
|
const previewModel = new PreviewModel(blockId);
|
|
|
|
return previewModel;
|
|
|
|
}
|
2024-05-17 07:48:23 +02:00
|
|
|
|
2024-06-04 00:50:51 +02:00
|
|
|
function DirNav({ cwdAtom }: { cwdAtom: jotai.WritableAtom<string, [string], void> }) {
|
|
|
|
const [cwd, setCwd] = jotai.useAtom(cwdAtom);
|
2024-06-13 23:41:28 +02:00
|
|
|
if (cwd == null || cwd == "") {
|
|
|
|
return null;
|
|
|
|
}
|
2024-06-04 00:50:51 +02:00
|
|
|
let splitNav = [cwd];
|
|
|
|
let remaining = cwd;
|
|
|
|
|
|
|
|
let idx = remaining.lastIndexOf("/");
|
|
|
|
while (idx !== -1) {
|
|
|
|
remaining = remaining.substring(0, idx);
|
|
|
|
splitNav.unshift(remaining);
|
|
|
|
|
|
|
|
idx = remaining.lastIndexOf("/");
|
|
|
|
}
|
|
|
|
if (splitNav.length === 0) {
|
|
|
|
splitNav = [cwd];
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<div className="view-nav">
|
2024-06-04 03:22:26 +02:00
|
|
|
{splitNav.map((item, idx) => {
|
2024-06-04 00:50:51 +02:00
|
|
|
let splitPath = item.split("/");
|
|
|
|
if (splitPath.length === 0) {
|
|
|
|
splitPath = [item];
|
|
|
|
}
|
2024-06-04 03:22:26 +02:00
|
|
|
const isLast = idx == splitNav.length - 1;
|
|
|
|
let baseName = splitPath[splitPath.length - 1];
|
|
|
|
if (!isLast) {
|
|
|
|
baseName += "/";
|
|
|
|
}
|
2024-06-04 00:50:51 +02:00
|
|
|
return (
|
2024-06-04 03:22:26 +02:00
|
|
|
<div
|
|
|
|
className={clsx("view-nav-item", isLast ? "current-file" : "clickable")}
|
|
|
|
key={`nav-item-${item}`}
|
|
|
|
onClick={isLast ? null : () => setCwd(item)}
|
|
|
|
>
|
2024-06-04 00:50:51 +02:00
|
|
|
{baseName}
|
2024-06-04 03:22:26 +02:00
|
|
|
</div>
|
2024-06-04 00:50:51 +02:00
|
|
|
);
|
|
|
|
})}
|
2024-06-04 03:22:26 +02:00
|
|
|
<div className="flex-spacer"></div>
|
2024-06-04 00:50:51 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-05-17 07:48:23 +02:00
|
|
|
function MarkdownPreview({ contentAtom }: { contentAtom: jotai.Atom<Promise<string>> }) {
|
2024-05-16 09:29:58 +02:00
|
|
|
const readmeText = jotai.useAtomValue(contentAtom);
|
2024-05-14 18:37:41 +02:00
|
|
|
return (
|
|
|
|
<div className="view-preview view-preview-markdown">
|
2024-05-14 21:29:41 +02:00
|
|
|
<Markdown text={readmeText} />
|
2024-05-14 18:37:41 +02:00
|
|
|
</div>
|
|
|
|
);
|
2024-05-17 07:48:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function StreamingPreview({ fileInfo }: { fileInfo: FileInfo }) {
|
|
|
|
const filePath = fileInfo.path;
|
2024-07-18 03:49:27 +02:00
|
|
|
const streamingUrl = getWebServerEndpoint() + "/wave/stream-file?path=" + encodeURIComponent(filePath);
|
2024-06-03 22:13:41 +02:00
|
|
|
if (fileInfo.mimetype == "application/pdf") {
|
|
|
|
return (
|
|
|
|
<div className="view-preview view-preview-pdf">
|
2024-06-14 20:10:54 +02:00
|
|
|
<iframe src={streamingUrl} width="95%" height="95%" name="pdfview" />
|
2024-06-03 22:13:41 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
2024-05-17 07:48:23 +02:00
|
|
|
if (fileInfo.mimetype.startsWith("video/")) {
|
|
|
|
return (
|
|
|
|
<div className="view-preview view-preview-video">
|
|
|
|
<video controls>
|
|
|
|
<source src={streamingUrl} />
|
|
|
|
</video>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (fileInfo.mimetype.startsWith("audio/")) {
|
|
|
|
return (
|
|
|
|
<div className="view-preview view-preview-audio">
|
|
|
|
<audio controls>
|
|
|
|
<source src={streamingUrl} />
|
|
|
|
</audio>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (fileInfo.mimetype.startsWith("image/")) {
|
|
|
|
return (
|
|
|
|
<div className="view-preview view-preview-image">
|
|
|
|
<img src={streamingUrl} />
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return <CenteredDiv>Preview Not Supported</CenteredDiv>;
|
|
|
|
}
|
|
|
|
|
2024-06-22 01:22:59 +02:00
|
|
|
function CodeEditPreview({
|
|
|
|
contentAtom,
|
|
|
|
filename,
|
|
|
|
readonly,
|
|
|
|
}: {
|
|
|
|
contentAtom: jotai.Atom<Promise<string>>;
|
|
|
|
filename: string;
|
|
|
|
readonly: boolean;
|
|
|
|
}) {
|
|
|
|
const fileContent = jotai.useAtomValue(contentAtom);
|
|
|
|
return <CodeEdit readonly={true} text={fileContent} filename={filename} />;
|
|
|
|
}
|
|
|
|
|
2024-06-24 19:17:35 +02:00
|
|
|
function CSVViewPreview({
|
|
|
|
parentRef,
|
|
|
|
contentAtom,
|
|
|
|
filename,
|
|
|
|
readonly,
|
|
|
|
}: {
|
|
|
|
parentRef: React.MutableRefObject<HTMLDivElement>;
|
|
|
|
contentAtom: jotai.Atom<Promise<string>>;
|
|
|
|
filename: string;
|
|
|
|
readonly: boolean;
|
|
|
|
}) {
|
|
|
|
const fileContent = jotai.useAtomValue(contentAtom);
|
|
|
|
return <CSVView parentRef={parentRef} readonly={true} content={fileContent} filename={filename} />;
|
|
|
|
}
|
|
|
|
|
2024-06-22 01:40:24 +02:00
|
|
|
function iconForFile(mimeType: string, fileName: string): string {
|
2024-07-09 00:04:48 +02:00
|
|
|
if (mimeType == null) {
|
|
|
|
mimeType = "unknown";
|
|
|
|
}
|
2024-06-22 01:40:24 +02:00
|
|
|
if (mimeType == "application/pdf") {
|
|
|
|
return "file-pdf";
|
|
|
|
} else if (mimeType.startsWith("image/")) {
|
|
|
|
return "image";
|
|
|
|
} else if (mimeType.startsWith("video/")) {
|
|
|
|
return "film";
|
|
|
|
} else if (mimeType.startsWith("audio/")) {
|
|
|
|
return "headphones";
|
|
|
|
} else if (mimeType.startsWith("text/markdown")) {
|
|
|
|
return "file-lines";
|
2024-06-26 18:39:41 +02:00
|
|
|
} else if (mimeType == "text/csv") {
|
|
|
|
return "file-csv";
|
2024-06-22 01:40:24 +02:00
|
|
|
} else if (
|
|
|
|
mimeType.startsWith("text/") ||
|
2024-07-09 00:04:48 +02:00
|
|
|
mimeType == "application/sql" ||
|
2024-06-22 01:40:24 +02:00
|
|
|
(mimeType.startsWith("application/") &&
|
|
|
|
(mimeType.includes("json") || mimeType.includes("yaml") || mimeType.includes("toml")))
|
|
|
|
) {
|
|
|
|
return "file-code";
|
|
|
|
} else if (mimeType === "directory") {
|
|
|
|
if (fileName == "~" || fileName == "~/") {
|
|
|
|
return "home";
|
|
|
|
}
|
2024-07-09 00:04:48 +02:00
|
|
|
return "folder-open";
|
2024-06-22 01:40:24 +02:00
|
|
|
} else {
|
|
|
|
return "file";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-09 00:04:48 +02:00
|
|
|
function PreviewView({ blockId, model }: { blockId: string; model: PreviewModel }) {
|
2024-07-11 00:06:19 +02:00
|
|
|
const contentRef = useRef<HTMLDivElement>(null);
|
2024-05-29 09:00:36 +02:00
|
|
|
const blockAtom = WOS.getWaveObjectAtom<Block>(`block:${blockId}`);
|
2024-07-09 00:04:48 +02:00
|
|
|
const fileNameAtom = model.fileName;
|
|
|
|
const statFileAtom = model.statFile;
|
|
|
|
const fullFileAtom = model.fullFile;
|
|
|
|
const fileMimeTypeAtom = model.fileMimeType;
|
|
|
|
const fileContentAtom = model.fileContent;
|
2024-05-16 09:29:58 +02:00
|
|
|
let mimeType = jotai.useAtomValue(fileMimeTypeAtom);
|
|
|
|
if (mimeType == null) {
|
|
|
|
mimeType = "";
|
|
|
|
}
|
2024-06-06 23:52:01 +02:00
|
|
|
let fileName = jotai.useAtomValue(fileNameAtom);
|
2024-05-17 07:48:23 +02:00
|
|
|
const fileInfo = jotai.useAtomValue(statFileAtom);
|
|
|
|
|
|
|
|
// handle streaming files here
|
2024-06-04 00:50:51 +02:00
|
|
|
let specializedView: React.ReactNode;
|
2024-06-22 01:40:24 +02:00
|
|
|
let blockIcon = iconForFile(mimeType, fileName);
|
2024-06-03 22:13:41 +02:00
|
|
|
if (
|
|
|
|
mimeType == "application/pdf" ||
|
|
|
|
mimeType.startsWith("video/") ||
|
|
|
|
mimeType.startsWith("audio/") ||
|
|
|
|
mimeType.startsWith("image/")
|
|
|
|
) {
|
2024-06-04 00:50:51 +02:00
|
|
|
specializedView = <StreamingPreview fileInfo={fileInfo} />;
|
|
|
|
} else if (fileInfo == null) {
|
2024-06-13 23:41:28 +02:00
|
|
|
specializedView = (
|
|
|
|
<CenteredDiv>File Not Found{util.isBlank(fileName) ? null : JSON.stringify(fileName)}</CenteredDiv>
|
|
|
|
);
|
2024-06-04 00:50:51 +02:00
|
|
|
} else if (fileInfo.size > MaxFileSize) {
|
|
|
|
specializedView = <CenteredDiv>File Too Large to Preview</CenteredDiv>;
|
|
|
|
} else if (mimeType === "text/markdown") {
|
|
|
|
specializedView = <MarkdownPreview contentAtom={fileContentAtom} />;
|
2024-06-24 19:17:35 +02:00
|
|
|
} else if (mimeType === "text/csv") {
|
2024-07-09 00:04:48 +02:00
|
|
|
if (fileInfo.size > MaxCSVSize) {
|
|
|
|
specializedView = <CenteredDiv>CSV File Too Large to Preview (1MB Max)</CenteredDiv>;
|
|
|
|
} else {
|
|
|
|
specializedView = (
|
2024-07-11 00:06:19 +02:00
|
|
|
<CSVViewPreview
|
|
|
|
parentRef={contentRef}
|
|
|
|
contentAtom={fileContentAtom}
|
|
|
|
filename={fileName}
|
|
|
|
readonly={true}
|
|
|
|
/>
|
2024-07-09 00:04:48 +02:00
|
|
|
);
|
|
|
|
}
|
2024-07-18 01:11:16 +02:00
|
|
|
} else if (isTextFile(mimeType)) {
|
2024-06-22 01:22:59 +02:00
|
|
|
specializedView = <CodeEditPreview readonly={true} contentAtom={fileContentAtom} filename={fileName} />;
|
2024-06-04 00:50:51 +02:00
|
|
|
} else if (mimeType === "directory") {
|
2024-07-09 01:36:30 +02:00
|
|
|
specializedView = <DirectoryPreview fileNameAtom={fileNameAtom} model={model} />;
|
2024-06-04 00:50:51 +02:00
|
|
|
} else {
|
|
|
|
specializedView = (
|
|
|
|
<div className="view-preview">
|
|
|
|
<div>Preview ({mimeType})</div>
|
|
|
|
</div>
|
|
|
|
);
|
2024-05-14 18:37:41 +02:00
|
|
|
}
|
2024-06-22 01:01:51 +02:00
|
|
|
setTimeout(() => {
|
|
|
|
const blockIconOverrideAtom = useBlockAtom<string>(blockId, "blockicon:override", () => {
|
|
|
|
return jotai.atom<string>(null);
|
2024-06-22 01:11:34 +02:00
|
|
|
}) as jotai.PrimitiveAtom<string>;
|
2024-06-22 01:01:51 +02:00
|
|
|
globalStore.set(blockIconOverrideAtom, blockIcon);
|
|
|
|
}, 10);
|
2024-06-04 00:50:51 +02:00
|
|
|
|
2024-05-14 08:45:41 +02:00
|
|
|
return (
|
2024-07-11 00:06:19 +02:00
|
|
|
<div className="full-preview scrollbar-hide-until-hover">
|
|
|
|
<div ref={contentRef} className="full-preview-content">
|
|
|
|
{specializedView}
|
|
|
|
</div>
|
2024-05-14 08:45:41 +02:00
|
|
|
</div>
|
|
|
|
);
|
2024-05-17 07:48:23 +02:00
|
|
|
}
|
2024-05-14 08:45:41 +02:00
|
|
|
|
2024-07-09 00:04:48 +02:00
|
|
|
export { PreviewView, makePreviewModel };
|