Use Single LogBox comp for scripts too
This commit is contained in:
@@ -6,7 +6,7 @@ import {
|
|||||||
useState,
|
useState,
|
||||||
useCallback,
|
useCallback,
|
||||||
} from "react";
|
} from "react";
|
||||||
import { Notepad, Prohibit } from "phosphor-react";
|
import { IconProps, Notepad, Prohibit } from "phosphor-react";
|
||||||
import useStayScrolled from "react-stay-scrolled";
|
import useStayScrolled from "react-stay-scrolled";
|
||||||
import NextLink from "next/link";
|
import NextLink from "next/link";
|
||||||
|
|
||||||
@@ -24,6 +24,7 @@ interface ILogBox {
|
|||||||
logs: ILog[];
|
logs: ILog[];
|
||||||
renderNav?: () => ReactNode;
|
renderNav?: () => ReactNode;
|
||||||
enhanced?: boolean;
|
enhanced?: boolean;
|
||||||
|
Icon?: FC<IconProps>
|
||||||
}
|
}
|
||||||
|
|
||||||
const LogBox: FC<ILogBox> = ({
|
const LogBox: FC<ILogBox> = ({
|
||||||
@@ -33,6 +34,7 @@ const LogBox: FC<ILogBox> = ({
|
|||||||
children,
|
children,
|
||||||
renderNav,
|
renderNav,
|
||||||
enhanced,
|
enhanced,
|
||||||
|
Icon = Notepad
|
||||||
}) => {
|
}) => {
|
||||||
const logRef = useRef<HTMLPreElement>(null);
|
const logRef = useRef<HTMLPreElement>(null);
|
||||||
const { stayScrolled /*, scrollBottom*/ } = useStayScrolled(logRef);
|
const { stayScrolled /*, scrollBottom*/ } = useStayScrolled(logRef);
|
||||||
@@ -82,14 +84,14 @@ const LogBox: FC<ILogBox> = ({
|
|||||||
gap: "$3",
|
gap: "$3",
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Notepad size="15px" /> <Text css={{ lineHeight: 1 }}>{title}</Text>
|
<Icon size="15px" /> <Text css={{ lineHeight: 1 }}>{title}</Text>
|
||||||
</Heading>
|
</Heading>
|
||||||
<Flex
|
<Flex
|
||||||
row
|
row
|
||||||
align="center"
|
align="center"
|
||||||
css={{
|
// css={{
|
||||||
width: "50%", // TODO make it max without breaking layout!
|
// maxWidth: "100%", // TODO make it max without breaking layout!
|
||||||
}}
|
// }}
|
||||||
>
|
>
|
||||||
{renderNav?.()}
|
{renderNav?.()}
|
||||||
</Flex>
|
</Flex>
|
||||||
|
|||||||
@@ -1,234 +0,0 @@
|
|||||||
import {
|
|
||||||
useRef,
|
|
||||||
useLayoutEffect,
|
|
||||||
ReactNode,
|
|
||||||
FC,
|
|
||||||
useState,
|
|
||||||
useCallback,
|
|
||||||
} from "react";
|
|
||||||
import { FileJs, Prohibit } from "phosphor-react";
|
|
||||||
import useStayScrolled from "react-stay-scrolled";
|
|
||||||
import NextLink from "next/link";
|
|
||||||
|
|
||||||
import Container from "./Container";
|
|
||||||
import LogText from "./LogText";
|
|
||||||
import state, { ILog } from "../state";
|
|
||||||
import { Pre, Link, Heading, Button, Text, Flex, Box } from ".";
|
|
||||||
import regexifyString from "regexify-string";
|
|
||||||
import { useSnapshot } from "valtio";
|
|
||||||
import { AccountDialog } from "./Accounts";
|
|
||||||
import RunScript from "./RunScript";
|
|
||||||
|
|
||||||
interface ILogBox {
|
|
||||||
title: string;
|
|
||||||
clearLog?: () => void;
|
|
||||||
logs: ILog[];
|
|
||||||
renderNav?: () => ReactNode;
|
|
||||||
enhanced?: boolean;
|
|
||||||
showButtons?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const LogBox: FC<ILogBox> = ({
|
|
||||||
title,
|
|
||||||
clearLog,
|
|
||||||
logs,
|
|
||||||
children,
|
|
||||||
renderNav,
|
|
||||||
enhanced,
|
|
||||||
showButtons = true,
|
|
||||||
}) => {
|
|
||||||
const logRef = useRef<HTMLPreElement>(null);
|
|
||||||
const { stayScrolled /*, scrollBottom*/ } = useStayScrolled(logRef);
|
|
||||||
const snap = useSnapshot(state);
|
|
||||||
useLayoutEffect(() => {
|
|
||||||
stayScrolled();
|
|
||||||
}, [stayScrolled, logs]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Flex
|
|
||||||
as="div"
|
|
||||||
css={{
|
|
||||||
display: "flex",
|
|
||||||
borderTop: "1px solid $mauve6",
|
|
||||||
background: "$mauve1",
|
|
||||||
position: "relative",
|
|
||||||
flex: 1,
|
|
||||||
height: "100%",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Container
|
|
||||||
css={{
|
|
||||||
px: 0,
|
|
||||||
height: "100%",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Flex
|
|
||||||
fluid
|
|
||||||
css={{
|
|
||||||
height: "48px",
|
|
||||||
alignItems: "center",
|
|
||||||
fontSize: "$sm",
|
|
||||||
fontWeight: 300,
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Heading
|
|
||||||
as="h3"
|
|
||||||
css={{
|
|
||||||
fontWeight: 300,
|
|
||||||
m: 0,
|
|
||||||
fontSize: "11px",
|
|
||||||
color: "$mauve12",
|
|
||||||
px: "$3",
|
|
||||||
textTransform: "uppercase",
|
|
||||||
alignItems: "center",
|
|
||||||
display: "inline-flex",
|
|
||||||
gap: "$3",
|
|
||||||
mr: "$3",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<FileJs size="15px" /> <Text css={{ lineHeight: 1 }}>{title}</Text>
|
|
||||||
</Heading>
|
|
||||||
{showButtons && (
|
|
||||||
<Flex css={{ gap: "$3" }}>
|
|
||||||
{snap.files
|
|
||||||
.filter((f) => f.name.endsWith(".js"))
|
|
||||||
.map((file) => (
|
|
||||||
<RunScript file={file} key={file.name} />
|
|
||||||
))}
|
|
||||||
</Flex>
|
|
||||||
)}
|
|
||||||
<Flex css={{ ml: "auto", gap: "$3", marginRight: "$3" }}>
|
|
||||||
{clearLog && (
|
|
||||||
<Button ghost size="xs" onClick={clearLog}>
|
|
||||||
<Prohibit size="14px" />
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</Flex>
|
|
||||||
</Flex>
|
|
||||||
<Box
|
|
||||||
as="pre"
|
|
||||||
ref={logRef}
|
|
||||||
css={{
|
|
||||||
margin: 0,
|
|
||||||
// display: "inline-block",
|
|
||||||
display: "flex",
|
|
||||||
flexDirection: "column",
|
|
||||||
width: "100%",
|
|
||||||
height: "calc(100% - 48px)", // 100% minus the logbox header height
|
|
||||||
overflowY: "auto",
|
|
||||||
fontSize: "13px",
|
|
||||||
fontWeight: "$body",
|
|
||||||
fontFamily: "$monospace",
|
|
||||||
px: "$3",
|
|
||||||
pb: "$2",
|
|
||||||
whiteSpace: "normal",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{logs?.map((log, index) => (
|
|
||||||
<Box
|
|
||||||
as="span"
|
|
||||||
key={log.type + index}
|
|
||||||
css={{
|
|
||||||
"@hover": {
|
|
||||||
"&:hover": {
|
|
||||||
backgroundColor: enhanced ? "$backgroundAlt" : undefined,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
p: enhanced ? "$1" : undefined,
|
|
||||||
my: enhanced ? "$1" : undefined,
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Log {...log} />
|
|
||||||
</Box>
|
|
||||||
))}
|
|
||||||
{children}
|
|
||||||
</Box>
|
|
||||||
</Container>
|
|
||||||
</Flex>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const Log: FC<ILog> = ({
|
|
||||||
type,
|
|
||||||
timestring,
|
|
||||||
message: _message,
|
|
||||||
link,
|
|
||||||
linkText,
|
|
||||||
defaultCollapsed,
|
|
||||||
jsonData: _jsonData,
|
|
||||||
}) => {
|
|
||||||
const [expanded, setExpanded] = useState(!defaultCollapsed);
|
|
||||||
const { accounts } = useSnapshot(state);
|
|
||||||
const [dialogAccount, setDialogAccount] = useState<string | null>(null);
|
|
||||||
|
|
||||||
const enrichAccounts = useCallback(
|
|
||||||
(str?: string): ReactNode => {
|
|
||||||
if (!str || !accounts.length) return null;
|
|
||||||
|
|
||||||
const pattern = `(${accounts.map((acc) => acc.address).join("|")})`;
|
|
||||||
const res = regexifyString({
|
|
||||||
pattern: new RegExp(pattern, "gim"),
|
|
||||||
decorator: (match, idx) => {
|
|
||||||
const name = accounts.find((acc) => acc.address === match)?.name;
|
|
||||||
return (
|
|
||||||
<Link
|
|
||||||
key={match + idx}
|
|
||||||
as="a"
|
|
||||||
onClick={() => setDialogAccount(match)}
|
|
||||||
title={match}
|
|
||||||
highlighted
|
|
||||||
>
|
|
||||||
{name || match}
|
|
||||||
</Link>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
input: str,
|
|
||||||
});
|
|
||||||
|
|
||||||
return <>{res}</>;
|
|
||||||
},
|
|
||||||
[accounts]
|
|
||||||
);
|
|
||||||
|
|
||||||
let message: ReactNode;
|
|
||||||
|
|
||||||
if (typeof _message === "string") {
|
|
||||||
_message = _message.trim().replace(/\n /gi, "\n");
|
|
||||||
message = enrichAccounts(_message);
|
|
||||||
} else {
|
|
||||||
message = _message;
|
|
||||||
}
|
|
||||||
|
|
||||||
const jsonData = enrichAccounts(_jsonData);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<AccountDialog
|
|
||||||
setActiveAccountAddress={setDialogAccount}
|
|
||||||
activeAccountAddress={dialogAccount}
|
|
||||||
/>
|
|
||||||
<LogText variant={type}>
|
|
||||||
{timestring && (
|
|
||||||
<Text muted monospace>
|
|
||||||
{timestring}{" "}
|
|
||||||
</Text>
|
|
||||||
)}
|
|
||||||
<Pre>{message} </Pre>
|
|
||||||
{link && (
|
|
||||||
<NextLink href={link} shallow passHref>
|
|
||||||
<Link as="a">{linkText}</Link>
|
|
||||||
</NextLink>
|
|
||||||
)}
|
|
||||||
{jsonData && (
|
|
||||||
<Link onClick={() => setExpanded(!expanded)} as="a">
|
|
||||||
{expanded ? "Collapse" : "Expand"}
|
|
||||||
</Link>
|
|
||||||
)}
|
|
||||||
{expanded && jsonData && <Pre block>{jsonData}</Pre>}
|
|
||||||
</LogText>
|
|
||||||
<br />
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default LogBox;
|
|
||||||
@@ -1,14 +1,13 @@
|
|||||||
import { Label } from "@radix-ui/react-label";
|
import { Label } from "@radix-ui/react-label";
|
||||||
import type { NextPage } from "next";
|
import type { NextPage } from "next";
|
||||||
import dynamic from "next/dynamic";
|
import dynamic from "next/dynamic";
|
||||||
import { Gear, Play } from "phosphor-react";
|
import { FileJs, Gear, Play } from "phosphor-react";
|
||||||
import Hotkeys from "react-hot-keys";
|
import Hotkeys from "react-hot-keys";
|
||||||
import Split from "react-split";
|
import Split from "react-split";
|
||||||
import { useSnapshot } from "valtio";
|
import { useSnapshot } from "valtio";
|
||||||
import { ButtonGroup, Flex } from "../../components";
|
import { ButtonGroup, Flex } from "../../components";
|
||||||
import Box from "../../components/Box";
|
import Box from "../../components/Box";
|
||||||
import Button from "../../components/Button";
|
import Button from "../../components/Button";
|
||||||
import LogBoxForScripts from "../../components/LogBoxForScripts";
|
|
||||||
import Popover from "../../components/Popover";
|
import Popover from "../../components/Popover";
|
||||||
import RunScript from "../../components/RunScript";
|
import RunScript from "../../components/RunScript";
|
||||||
import state from "../../state";
|
import state from "../../state";
|
||||||
@@ -244,8 +243,8 @@ const Home: NextPage = () => {
|
|||||||
flex: 1,
|
flex: 1,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<LogBoxForScripts
|
<LogBox
|
||||||
showButtons={false}
|
Icon={FileJs}
|
||||||
title="Script Log"
|
title="Script Log"
|
||||||
logs={snap.scriptLogs}
|
logs={snap.scriptLogs}
|
||||||
clearLog={() => (state.scriptLogs = [])}
|
clearLog={() => (state.scriptLogs = [])}
|
||||||
|
|||||||
@@ -6,8 +6,9 @@ import Transaction from "../../components/Transaction";
|
|||||||
import state from "../../state";
|
import state from "../../state";
|
||||||
import { getSplit, saveSplit } from "../../state/actions/persistSplits";
|
import { getSplit, saveSplit } from "../../state/actions/persistSplits";
|
||||||
import { transactionsState, modifyTransaction } from "../../state";
|
import { transactionsState, modifyTransaction } from "../../state";
|
||||||
import LogBoxForScripts from "../../components/LogBoxForScripts";
|
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
import { FileJs } from "phosphor-react";
|
||||||
|
import RunScript from '../../components/RunScript';
|
||||||
|
|
||||||
const DebugStream = dynamic(() => import("../../components/DebugStream"), {
|
const DebugStream = dynamic(() => import("../../components/DebugStream"), {
|
||||||
ssr: false,
|
ssr: false,
|
||||||
@@ -33,7 +34,17 @@ const Test = () => {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
const hasScripts = Boolean(
|
const hasScripts = Boolean(
|
||||||
snap.files.filter((f) => f.name.toLowerCase()?.endsWith(".js")).length
|
snap.files.filter(f => f.name.toLowerCase()?.endsWith(".js")).length
|
||||||
|
);
|
||||||
|
|
||||||
|
const renderNav = () => (
|
||||||
|
<Flex css={{ gap: "$3" }}>
|
||||||
|
{snap.files
|
||||||
|
.filter(f => f.name.endsWith(".js"))
|
||||||
|
.map(file => (
|
||||||
|
<RunScript file={file} key={file.name} />
|
||||||
|
))}
|
||||||
|
</Flex>
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -50,7 +61,7 @@ const Test = () => {
|
|||||||
gutterSize={4}
|
gutterSize={4}
|
||||||
gutterAlign="center"
|
gutterAlign="center"
|
||||||
style={{ height: "calc(100vh - 60px)" }}
|
style={{ height: "calc(100vh - 60px)" }}
|
||||||
onDragEnd={(e) => saveSplit("testVertical", e)}
|
onDragEnd={e => saveSplit("testVertical", e)}
|
||||||
>
|
>
|
||||||
<Flex
|
<Flex
|
||||||
row
|
row
|
||||||
@@ -72,11 +83,11 @@ const Test = () => {
|
|||||||
width: "100%",
|
width: "100%",
|
||||||
height: "100%",
|
height: "100%",
|
||||||
}}
|
}}
|
||||||
onDragEnd={(e) => saveSplit("testHorizontal", e)}
|
onDragEnd={e => saveSplit("testHorizontal", e)}
|
||||||
>
|
>
|
||||||
<Box css={{ width: "55%", px: "$2" }}>
|
<Box css={{ width: "55%", px: "$2" }}>
|
||||||
<Tabs
|
<Tabs
|
||||||
label='Transaction'
|
label="Transaction"
|
||||||
activeHeader={activeHeader}
|
activeHeader={activeHeader}
|
||||||
// TODO make header a required field
|
// TODO make header a required field
|
||||||
onChangeActive={(idx, header) => {
|
onChangeActive={(idx, header) => {
|
||||||
@@ -84,8 +95,8 @@ const Test = () => {
|
|||||||
}}
|
}}
|
||||||
keepAllAlive
|
keepAllAlive
|
||||||
defaultExtension="json"
|
defaultExtension="json"
|
||||||
allowedExtensions={['json']}
|
allowedExtensions={["json"]}
|
||||||
onCreateNewTab={(header) => modifyTransaction(header, {})}
|
onCreateNewTab={header => modifyTransaction(header, {})}
|
||||||
onCloseTab={(idx, header) =>
|
onCloseTab={(idx, header) =>
|
||||||
header && modifyTransaction(header, undefined)
|
header && modifyTransaction(header, undefined)
|
||||||
}
|
}
|
||||||
@@ -111,10 +122,12 @@ const Test = () => {
|
|||||||
flexDirection: "column",
|
flexDirection: "column",
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<LogBoxForScripts
|
<LogBox
|
||||||
|
Icon={FileJs}
|
||||||
title="Helper scripts"
|
title="Helper scripts"
|
||||||
logs={snap.scriptLogs}
|
logs={snap.scriptLogs}
|
||||||
clearLog={() => (state.scriptLogs = [])}
|
clearLog={() => (state.scriptLogs = [])}
|
||||||
|
renderNav={renderNav}
|
||||||
/>
|
/>
|
||||||
</Flex>
|
</Flex>
|
||||||
) : null}
|
) : null}
|
||||||
|
|||||||
Reference in New Issue
Block a user