refactor(nvim-tree): abstract away file tree building

This commit is contained in:
Pihkaal
2024-01-28 17:27:32 +01:00
parent 5542dce881
commit 965fff6226
3 changed files with 76 additions and 58 deletions

View File

@@ -1,12 +1,16 @@
import { useApp } from "~/context/AppContext";
import { NvimStatusBar } from "./NvimStatusBar"; import { NvimStatusBar } from "./NvimStatusBar";
import { NvimTree } from "./NvimTree"; import { NvimTree } from "./NvimTree";
import { buildFileTree } from "~/utils/filesystem";
export const Nvim = () => { export const Nvim = () => {
const manifest = useApp();
return ( return (
<div> <div>
<div className="flex"> <div className="flex">
<div className="w-fit"> <div className="w-fit">
<NvimTree /> <NvimTree files={buildFileTree(manifest)} />
</div> </div>
<div className="flex-1"></div> <div className="flex-1"></div>
</div> </div>

View File

@@ -1,11 +1,14 @@
import { useState, useEffect } from "react"; import { useState, useEffect } from "react";
import { useApp } from "~/context/AppContext";
import { useTerminal } from "~/context/TerminalContext"; import { useTerminal } from "~/context/TerminalContext";
import { FILE_STYLES, type File } from "~/utils/filesystem"; import {
DEFAULT_FILE_STYLE,
FILE_STYLES,
getExtension,
type File,
} from "~/utils/filesystem";
import { type Cell } from "~/utils/terminal/cell"; import { type Cell } from "~/utils/terminal/cell";
import { TerminalRenderer } from "~/utils/terminal/renderer"; import { TerminalRenderer } from "~/utils/terminal/renderer";
import { theme } from "~/utils/terminal/theme"; import { theme } from "~/utils/terminal/theme";
import { type Manifest } from "~/utils/types";
const PATH_FOLDED: Cell = { const PATH_FOLDED: Cell = {
char: "", char: "",
@@ -17,39 +20,9 @@ const PATH_UNFOLDED: Cell = {
foreground: theme.blue, foreground: theme.blue,
}; };
const buildFileTree = (manifest: Manifest): Array<File> => { export const NvimTree = (props: { files: Array<File> }) => {
if (manifest === undefined) return [];
const files: Array<File> = [];
manifest.projects.forEach(project => {
if (project.name === "pihkaal") {
project.files.forEach(file => {
files.push({
name: file,
type: "md",
});
});
} else {
files.push({
name: project.name,
type: "directory",
folded: true,
children: project.files.map(file => ({
name: file,
type: "md",
})),
});
}
});
return files;
};
export const NvimTree = () => {
const manifest = useApp();
const [selected, setSelected] = useState(0); const [selected, setSelected] = useState(0);
const [files, setFiles] = useState(buildFileTree(manifest)); const [files, setFiles] = useState(props.files);
const { cols: width, rows: height } = useTerminal(); const { cols: width, rows: height } = useTerminal();
const canvas = new TerminalRenderer(width * 0.2, height - 2, { const canvas = new TerminalRenderer(width * 0.2, height - 2, {
@@ -63,12 +36,15 @@ export const NvimTree = () => {
let indent = 0; let indent = 0;
const renderTree = (files: Array<File>) => { const renderTree = (files: Array<File>) => {
files.forEach(file => { files.forEach(file => {
tree.apply(2 + indent * 2, y, FILE_STYLES[file.type]);
if (file.type === "directory") { if (file.type === "directory") {
tree.apply(2 + indent * 2, y, {
char: file.folded ? "\ue6ad" : "\ueaf6",
foreground: theme.blue,
});
tree.apply(indent * 2, y, file.folded ? PATH_FOLDED : PATH_UNFOLDED); tree.apply(indent * 2, y, file.folded ? PATH_FOLDED : PATH_UNFOLDED);
tree.write(4 + indent * 2, y, file.name, { tree.write(4 + indent * 2, y, file.name, {
foreground: FILE_STYLES.directory.foreground, foreground: theme.blue,
}); });
y++; y++;
@@ -78,6 +54,10 @@ export const NvimTree = () => {
indent--; indent--;
} }
} else { } else {
const style =
FILE_STYLES[getExtension(file.name)] ?? DEFAULT_FILE_STYLE;
tree.apply(2 + indent * 2, y, style);
if (file.name === "README.md") { if (file.name === "README.md") {
tree.write(4 + indent * 2, y, file.name, { tree.write(4 + indent * 2, y, file.name, {
foreground: theme.yellow, foreground: theme.yellow,
@@ -146,13 +126,3 @@ export const NvimTree = () => {
return <p>{canvas.render()}</p>; return <p>{canvas.render()}</p>;
}; };
/*
.sort((a, b) => a.name.localeCompare(b.name)).sort((a, b) =>
a.type === "directory" && b.type !== "directory"
? -1
: a.type !== "directory" && b.type === "directory"
? 1
: 0,
),
*/

View File

@@ -1,11 +1,18 @@
import { type Cell } from "./terminal/cell"; import { type Cell } from "./terminal/cell";
import { theme } from "./terminal/theme"; import { theme } from "./terminal/theme";
import { type Manifest } from "./types";
export const FILE_STYLES = { export const getExtension = (path: string) => {
directory: { const parts = path.split(".");
char: "\ue6ad", // \ue6ad ||| \ueaf6 return parts[Math.max(0, parts.length - 1)] ?? "";
foreground: theme.blue, };
},
export const DEFAULT_FILE_STYLE: Cell = {
char: "F",
foreground: theme.white,
};
export const FILE_STYLES: Record<string, Cell> = {
md: { md: {
char: "\ue73e", char: "\ue73e",
foreground: theme.blue, foreground: theme.blue,
@@ -14,15 +21,13 @@ export const FILE_STYLES = {
char: "\uf43d", char: "\uf43d",
foreground: theme.yellow, foreground: theme.yellow,
}, },
} as const satisfies Record<string, Cell>; };
export type FileType = keyof typeof FILE_STYLES;
export type File = { export type File = {
name: string; name: string;
} & ( } & (
| { | {
type: Exclude<FileType, "directory">; type: "file";
} }
| { | {
type: "directory"; type: "directory";
@@ -30,3 +35,42 @@ export type File = {
folded: boolean; folded: boolean;
} }
); );
const sortFiles = (files: Array<File>) =>
files
.sort((a, b) => a.name.localeCompare(b.name))
.sort((a, b) =>
a.type === "directory" && b.type !== "directory"
? -1
: a.type !== "directory" && b.type === "directory"
? 1
: 0,
);
export const buildFileTree = (manifest: Manifest): Array<File> => {
const files: Array<File> = [];
manifest.projects.forEach(project => {
if (project.name === "pihkaal") {
project.files.forEach(file => {
files.push({
name: file,
type: "file",
});
});
} else {
files.push({
name: project.name,
type: "directory",
folded: true,
children: sortFiles(
project.files.map(file => ({
name: file,
type: "file",
})),
),
});
}
});
return sortFiles(files);
};