dpkit
Version:
Fast TypeScript data management framework built on top of the Data Package standard and Polars DataFrames
157 lines • 20.6 kB
JavaScript
import { useApp, useInput } from "ink";
import { Box, Text } from "ink";
import pc from "picocolors";
import { useEffect, useState } from "react";
import React from "react";
import { DataGrid } from "./DataGrid.js";
const PAGE_SIZE = 10;
export function TableGrid(props) {
const { table, schema, borderColor } = props;
const { exit } = useApp();
const [col, setCol] = useState(0);
const [row, setRow] = useState(0);
const [page, setPage] = useState(1);
const [order, setOrder] = useState();
const [records, setRecords] = useState();
const handleColChange = async (newCol) => {
if (newCol <= 0)
return;
if (newCol > table.columns.length)
return;
setCol(newCol);
};
const handleRowChange = async (row) => {
if (row > PAGE_SIZE) {
handlePageChange(page + 1);
row = 1;
}
else if (row < 1) {
if (page === 1)
return;
handlePageChange(page - 1);
row = 10;
}
else if (records && row > records.length) {
return;
}
setRow(row);
};
const handleOrderChange = async (order) => {
setOrder(order);
if (order) {
handlePageChange(1, order);
}
};
const handlePageChange = async (page, newOrder) => {
const thisOrder = newOrder ?? order;
if (page === 0)
return;
let ldf = table;
if (thisOrder) {
const name = table.columns[thisOrder.col - 1];
if (name) {
ldf = ldf.sort(name, thisOrder.dir === "desc");
}
}
const offset = (page - 1) * PAGE_SIZE;
const df = await ldf.slice(offset, PAGE_SIZE).collect();
const records = df.toRecords();
if (records.length) {
setPage(page);
setRecords(records);
}
};
useEffect(() => {
handlePageChange(1);
}, [table]);
useEffect(() => {
if (records && props.quit)
exit();
}, [records]);
useInput((input, key) => {
if (key.escape || input === "q") {
exit();
}
if (key.pageUp || input === "p") {
handlePageChange(page - 1);
}
if (key.pageDown || input === "n") {
handlePageChange(page + 1);
}
if (key.downArrow || input === "j") {
handleRowChange(row + 1);
}
if (key.upArrow || input === "k") {
handleRowChange(row - 1);
}
if (key.leftArrow || input === "h") {
handleColChange(col - 1);
}
if (key.rightArrow || input === "l") {
handleColChange(col + 1);
}
if (key.return || input === "o") {
let nextOrder = { col, dir: "desc" };
if (order?.col === col) {
if (order?.dir === "desc")
nextOrder = { col, dir: "asc" };
if (order?.dir === "asc")
nextOrder = undefined;
}
handleOrderChange(nextOrder);
}
});
if (!records) {
return null;
}
return (React.createElement(Box, { flexDirection: "column" },
React.createElement(DataGrid, { records: records, schema: schema, col: col, row: row, order: order, rowHeight: 2, borderColor: borderColor, withTypes: props.withTypes }),
React.createElement(Help, { page: page })));
}
function Help(props) {
const { exit } = useApp();
const [isOpen, setIsOpen] = useState(false);
useInput((input, key) => {
if (key.escape || input === "q") {
exit();
}
if (input === "d") {
setIsOpen(!isOpen);
}
});
if (!isOpen) {
return (React.createElement(Box, { paddingLeft: 1 },
React.createElement(PageItem, { page: props.page }),
React.createElement(Text, null, ", "),
React.createElement(HelpItem, { button: "d", description: "to toggle docs" }),
React.createElement(Text, null, ", "),
React.createElement(HelpItem, { button: "q", description: "to quit" })));
}
return (React.createElement(Box, { flexDirection: "column", paddingLeft: 1 },
React.createElement(Text, { bold: true }, "Table Usage"),
React.createElement(HelpItem, { button: "p, pgUp", description: "for prev page" }),
React.createElement(HelpItem, { button: "n, pgDown", description: "for next page" }),
React.createElement(HelpItem, { button: "k, up", description: "for prev row" }),
React.createElement(HelpItem, { button: "j, down", description: "for next row" }),
React.createElement(HelpItem, { button: "h, left", description: "for prev column" }),
React.createElement(HelpItem, { button: "l, right", description: "for next column" }),
React.createElement(HelpItem, { button: "o, enter", description: "for order" }),
React.createElement(HelpItem, { button: "q, esc", description: "for quit" })));
}
// It has weird Text.dimColor bug so we use picocolors here
function PageItem(props) {
return (React.createElement(Text, null,
React.createElement(Text, null,
pc.dim("page"),
" "),
React.createElement(Text, { bold: true }, props.page)));
}
function HelpItem(props) {
return (React.createElement(Text, null,
React.createElement(Text, { dimColor: true }, "press"),
" ",
React.createElement(Text, { bold: true }, props.button),
" ",
React.createElement(Text, { dimColor: true }, props.description)));
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TableGrid.js","sourceRoot":"","sources":["../../components/TableGrid.tsx"],"names":[],"mappings":"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","sourcesContent":["import type { DataRecord, Schema, Table } from \"@dpkit/all\"\nimport { useApp, useInput } from \"ink\"\nimport { Box, Text } from \"ink\"\nimport pc from \"picocolors\"\nimport { useEffect, useState } from \"react\"\nimport React from \"react\"\nimport type { Order } from \"./DataGrid.tsx\"\nimport { DataGrid } from \"./DataGrid.tsx\"\n\nconst PAGE_SIZE = 10\n\nexport function TableGrid(props: {\n  table: Table\n  schema?: Schema\n  borderColor?: \"green\" | \"red\"\n  withTypes?: boolean\n  quit?: boolean\n}) {\n  const { table, schema, borderColor } = props\n\n  const { exit } = useApp()\n  const [col, setCol] = useState(0)\n  const [row, setRow] = useState(0)\n  const [page, setPage] = useState(1)\n  const [order, setOrder] = useState<Order>()\n  const [records, setRecords] = useState<DataRecord[]>()\n\n  const handleColChange = async (newCol: number) => {\n    if (newCol <= 0) return\n    if (newCol > table.columns.length) return\n\n    setCol(newCol)\n  }\n\n  const handleRowChange = async (row: number) => {\n    if (row > PAGE_SIZE) {\n      handlePageChange(page + 1)\n      row = 1\n    } else if (row < 1) {\n      if (page === 1) return\n      handlePageChange(page - 1)\n      row = 10\n    } else if (records && row > records.length) {\n      return\n    }\n\n    setRow(row)\n  }\n\n  const handleOrderChange = async (order?: Order) => {\n    setOrder(order)\n\n    if (order) {\n      handlePageChange(1, order)\n    }\n  }\n\n  const handlePageChange = async (page: number, newOrder?: Order) => {\n    const thisOrder = newOrder ?? order\n    if (page === 0) return\n\n    let ldf = table\n    if (thisOrder) {\n      const name = table.columns[thisOrder.col - 1]\n      if (name) {\n        ldf = ldf.sort(name, thisOrder.dir === \"desc\")\n      }\n    }\n\n    const offset = (page - 1) * PAGE_SIZE\n    const df = await ldf.slice(offset, PAGE_SIZE).collect()\n    const records = df.toRecords()\n\n    if (records.length) {\n      setPage(page)\n      setRecords(records)\n    }\n  }\n\n  useEffect(() => {\n    handlePageChange(1)\n  }, [table])\n\n  useEffect(() => {\n    if (records && props.quit) exit()\n  }, [records])\n\n  useInput((input, key) => {\n    if (key.escape || input === \"q\") {\n      exit()\n    }\n\n    if (key.pageUp || input === \"p\") {\n      handlePageChange(page - 1)\n    }\n\n    if (key.pageDown || input === \"n\") {\n      handlePageChange(page + 1)\n    }\n\n    if (key.downArrow || input === \"j\") {\n      handleRowChange(row + 1)\n    }\n\n    if (key.upArrow || input === \"k\") {\n      handleRowChange(row - 1)\n    }\n\n    if (key.leftArrow || input === \"h\") {\n      handleColChange(col - 1)\n    }\n\n    if (key.rightArrow || input === \"l\") {\n      handleColChange(col + 1)\n    }\n\n    if (key.return || input === \"o\") {\n      let nextOrder: Order | undefined = { col, dir: \"desc\" }\n\n      if (order?.col === col) {\n        if (order?.dir === \"desc\") nextOrder = { col, dir: \"asc\" }\n        if (order?.dir === \"asc\") nextOrder = undefined\n      }\n\n      handleOrderChange(nextOrder)\n    }\n  })\n\n  if (!records) {\n    return null\n  }\n\n  return (\n    <Box flexDirection=\"column\">\n      <DataGrid\n        records={records}\n        schema={schema}\n        col={col}\n        row={row}\n        order={order}\n        rowHeight={2}\n        borderColor={borderColor}\n        withTypes={props.withTypes}\n      />\n      <Help page={page} />\n    </Box>\n  )\n}\n\nfunction Help(props: { page: number }) {\n  const { exit } = useApp()\n  const [isOpen, setIsOpen] = useState(false)\n\n  useInput((input, key) => {\n    if (key.escape || input === \"q\") {\n      exit()\n    }\n\n    if (input === \"d\") {\n      setIsOpen(!isOpen)\n    }\n  })\n\n  if (!isOpen) {\n    return (\n      <Box paddingLeft={1}>\n        <PageItem page={props.page} />\n        <Text>{\", \"}</Text>\n        <HelpItem button=\"d\" description=\"to toggle docs\" />\n        <Text>{\", \"}</Text>\n        <HelpItem button=\"q\" description=\"to quit\" />\n      </Box>\n    )\n  }\n\n  return (\n    <Box flexDirection=\"column\" paddingLeft={1}>\n      <Text bold>Table Usage</Text>\n      <HelpItem button=\"p, pgUp\" description=\"for prev page\" />\n      <HelpItem button=\"n, pgDown\" description=\"for next page\" />\n      <HelpItem button=\"k, up\" description=\"for prev row\" />\n      <HelpItem button=\"j, down\" description=\"for next row\" />\n      <HelpItem button=\"h, left\" description=\"for prev column\" />\n      <HelpItem button=\"l, right\" description=\"for next column\" />\n      <HelpItem button=\"o, enter\" description=\"for order\" />\n      <HelpItem button=\"q, esc\" description=\"for quit\" />\n    </Box>\n  )\n}\n\n// It has weird Text.dimColor bug so we use picocolors here\nfunction PageItem(props: { page: number }) {\n  return (\n    <Text>\n      <Text>{pc.dim(\"page\")} </Text>\n      <Text bold>{props.page}</Text>\n    </Text>\n  )\n}\n\nfunction HelpItem(props: { button: string; description: string }) {\n  return (\n    <Text>\n      <Text dimColor>press</Text> <Text bold>{props.button}</Text>{\" \"}\n      <Text dimColor>{props.description}</Text>\n    </Text>\n  )\n}\n"]}