Compare commits
22 Commits
feat/jsdoc
...
fix/binary
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d05180d148 | ||
|
|
bfaa6be17d | ||
|
|
9e368dec84 | ||
|
|
25eec6980f | ||
|
|
8e2f20c5ac | ||
|
|
a3d094e873 | ||
|
|
ef70bfb13a | ||
|
|
c26c7c13d1 | ||
|
|
fc461ddd0d | ||
|
|
c7001f6089 | ||
|
|
243cbfec08 | ||
|
|
1295e7fa41 | ||
|
|
793623d216 | ||
|
|
0cde0eb240 | ||
|
|
e2acb48e03 | ||
|
|
3fcbac5ed9 | ||
|
|
3af2bad536 | ||
|
|
4f1b877db0 | ||
|
|
53afb1d3d1 | ||
|
|
31ff7c0e28 | ||
|
|
dfa35df465 | ||
|
|
f163b052e1 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -32,3 +32,4 @@ yarn-error.log*
|
||||
|
||||
# vercel
|
||||
.vercel
|
||||
.vscode
|
||||
|
||||
@@ -31,6 +31,7 @@ import transactionsData from "../content/transactions.json";
|
||||
import { SetHookDialog } from "./SetHookDialog";
|
||||
import { addFunds } from "../state/actions/addFaucetAccount";
|
||||
import { deleteHook } from "../state/actions/deployHook";
|
||||
import { capitalize } from "../utils/helpers";
|
||||
|
||||
export const AccountDialog = ({
|
||||
activeAccountAddress,
|
||||
@@ -42,12 +43,12 @@ export const AccountDialog = ({
|
||||
const snap = useSnapshot(state);
|
||||
const [showSecret, setShowSecret] = useState(false);
|
||||
const activeAccount = snap.accounts.find(
|
||||
(account) => account.address === activeAccountAddress
|
||||
account => account.address === activeAccountAddress
|
||||
);
|
||||
return (
|
||||
<Dialog
|
||||
open={Boolean(activeAccountAddress)}
|
||||
onOpenChange={(open) => {
|
||||
onOpenChange={open => {
|
||||
setShowSecret(false);
|
||||
!open && setActiveAccountAddress(null);
|
||||
}}
|
||||
@@ -99,7 +100,7 @@ export const AccountDialog = ({
|
||||
tabIndex={-1}
|
||||
onClick={() => {
|
||||
const index = state.accounts.findIndex(
|
||||
(acc) => acc.address === activeAccount?.address
|
||||
acc => acc.address === activeAccount?.address
|
||||
);
|
||||
state.accounts.splice(index, 1);
|
||||
}}
|
||||
@@ -165,7 +166,7 @@ export const AccountDialog = ({
|
||||
}}
|
||||
ghost
|
||||
size="xs"
|
||||
onClick={() => setShowSecret((curr) => !curr)}
|
||||
onClick={() => setShowSecret(curr => !curr)}
|
||||
>
|
||||
{showSecret ? "Hide" : "Show"}
|
||||
</Button>
|
||||
@@ -252,7 +253,7 @@ export const AccountDialog = ({
|
||||
}}
|
||||
>
|
||||
{activeAccount && activeAccount.hooks.length > 0
|
||||
? activeAccount.hooks.map((i) => {
|
||||
? activeAccount.hooks.map(i => {
|
||||
return (
|
||||
<a
|
||||
key={i}
|
||||
@@ -301,7 +302,7 @@ interface AccountProps {
|
||||
showHookStats?: boolean;
|
||||
}
|
||||
|
||||
const Accounts: FC<AccountProps> = (props) => {
|
||||
const Accounts: FC<AccountProps> = props => {
|
||||
const snap = useSnapshot(state);
|
||||
const [activeAccountAddress, setActiveAccountAddress] = useState<
|
||||
string | null
|
||||
@@ -309,7 +310,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
useEffect(() => {
|
||||
const fetchAccInfo = async () => {
|
||||
if (snap.clientStatus === "online") {
|
||||
const requests = snap.accounts.map((acc) =>
|
||||
const requests = snap.accounts.map(acc =>
|
||||
snap.client?.send({
|
||||
id: `hooks-builder-req-info-${acc.address}`,
|
||||
command: "account_info",
|
||||
@@ -322,7 +323,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
const balance = res?.account_data?.Balance as string;
|
||||
const sequence = res?.account_data?.Sequence as number;
|
||||
const accountToUpdate = state.accounts.find(
|
||||
(acc) => acc.address === address
|
||||
acc => acc.address === address
|
||||
);
|
||||
if (accountToUpdate) {
|
||||
accountToUpdate.xrp = balance;
|
||||
@@ -330,7 +331,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
accountToUpdate.error = null;
|
||||
} else {
|
||||
const oldAccount = state.accounts.find(
|
||||
(acc) => acc.address === res?.account
|
||||
acc => acc.address === res?.account
|
||||
);
|
||||
if (oldAccount) {
|
||||
oldAccount.xrp = "0";
|
||||
@@ -341,7 +342,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
}
|
||||
}
|
||||
});
|
||||
const objectRequests = snap.accounts.map((acc) => {
|
||||
const objectRequests = snap.accounts.map(acc => {
|
||||
return snap.client?.send({
|
||||
id: `hooks-builder-req-objects-${acc.address}`,
|
||||
command: "account_objects",
|
||||
@@ -352,7 +353,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
objectResponses.forEach((res: any) => {
|
||||
const address = res?.account as string;
|
||||
const accountToUpdate = state.accounts.find(
|
||||
(acc) => acc.address === address
|
||||
acc => acc.address === address
|
||||
);
|
||||
if (accountToUpdate) {
|
||||
accountToUpdate.hooks =
|
||||
@@ -416,9 +417,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
<Wallet size="15px" /> <Text css={{ lineHeight: 1 }}>Accounts</Text>
|
||||
</Heading>
|
||||
<Flex css={{ ml: "auto", gap: "$3", marginRight: "$3" }}>
|
||||
<Button ghost size="sm" onClick={() => addFaucetAccount(true)}>
|
||||
Create
|
||||
</Button>
|
||||
<ImportAccountDialog type="create" />
|
||||
<ImportAccountDialog />
|
||||
</Flex>
|
||||
</Flex>
|
||||
@@ -435,7 +434,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
overflowY: "auto",
|
||||
}}
|
||||
>
|
||||
{snap.accounts.map((account) => (
|
||||
{snap.accounts.map(account => (
|
||||
<Flex
|
||||
column
|
||||
key={account.address + account.name}
|
||||
@@ -488,7 +487,7 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
{!props.hideDeployBtn && (
|
||||
<div
|
||||
className="hook-deploy-button"
|
||||
onClick={(e) => {
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
}}
|
||||
>
|
||||
@@ -514,32 +513,71 @@ const Accounts: FC<AccountProps> = (props) => {
|
||||
);
|
||||
};
|
||||
|
||||
export const transactionsOptions = transactionsData.map((tx) => ({
|
||||
export const transactionsOptions = transactionsData.map(tx => ({
|
||||
value: tx.TransactionType,
|
||||
label: tx.TransactionType,
|
||||
}));
|
||||
|
||||
const ImportAccountDialog = () => {
|
||||
const [value, setValue] = useState("");
|
||||
const ImportAccountDialog = ({
|
||||
type = "import",
|
||||
}: {
|
||||
type?: "import" | "create";
|
||||
}) => {
|
||||
const [secret, setSecret] = useState("");
|
||||
const [name, setName] = useState("");
|
||||
|
||||
const btnText = type === "import" ? "Import" : "Create";
|
||||
const title = type === "import" ? "Import Account" : "Create Account";
|
||||
|
||||
const handleSubmit = async () => {
|
||||
if (type === "create") {
|
||||
const value = capitalize(name);
|
||||
await addFaucetAccount(value, true);
|
||||
setName("");
|
||||
setSecret("");
|
||||
return;
|
||||
}
|
||||
importAccount(secret, name);
|
||||
setName("");
|
||||
setSecret("");
|
||||
};
|
||||
return (
|
||||
<Dialog>
|
||||
<DialogTrigger asChild>
|
||||
<Button ghost size="sm">
|
||||
Import
|
||||
{btnText}
|
||||
</Button>
|
||||
</DialogTrigger>
|
||||
<DialogContent>
|
||||
<DialogTitle>Import account</DialogTitle>
|
||||
<DialogDescription>
|
||||
<Label>Add account secret</Label>
|
||||
<Input
|
||||
name="secret"
|
||||
type="password"
|
||||
autoComplete="new-password"
|
||||
value={value}
|
||||
onChange={(e) => setValue(e.target.value)}
|
||||
/>
|
||||
</DialogDescription>
|
||||
<DialogContent aria-describedby={undefined}>
|
||||
<DialogTitle css={{ mb: "$4" }}>{title}</DialogTitle>
|
||||
<Flex column>
|
||||
<Box css={{ mb: "$2" }}>
|
||||
<Label>
|
||||
Account name <Text muted>(optional)</Text>
|
||||
</Label>
|
||||
<Input
|
||||
name="name"
|
||||
type="text"
|
||||
autoComplete="off"
|
||||
autoCapitalize="on"
|
||||
value={name}
|
||||
onChange={e => setName(e.target.value)}
|
||||
/>
|
||||
</Box>
|
||||
{type === "import" && (
|
||||
<Box>
|
||||
<Label>Account secret</Label>
|
||||
<Input
|
||||
required
|
||||
name="secret"
|
||||
type="password"
|
||||
autoComplete="new-password"
|
||||
value={secret}
|
||||
onChange={e => setSecret(e.target.value)}
|
||||
/>
|
||||
</Box>
|
||||
)}
|
||||
</Flex>
|
||||
|
||||
<Flex
|
||||
css={{
|
||||
@@ -552,14 +590,8 @@ const ImportAccountDialog = () => {
|
||||
<Button outline>Cancel</Button>
|
||||
</DialogClose>
|
||||
<DialogClose asChild>
|
||||
<Button
|
||||
variant="primary"
|
||||
onClick={() => {
|
||||
importAccount(value);
|
||||
setValue("");
|
||||
}}
|
||||
>
|
||||
Import account
|
||||
<Button type="submit" variant="primary" onClick={handleSubmit}>
|
||||
{title}
|
||||
</Button>
|
||||
</DialogClose>
|
||||
</Flex>
|
||||
|
||||
@@ -87,7 +87,7 @@ const addListeners = (account: ISelect | null) => {
|
||||
if (streamState.socket) {
|
||||
interval = setInterval(() => {
|
||||
streamState.socket?.send("");
|
||||
}, 10000);
|
||||
}, 45000);
|
||||
}
|
||||
|
||||
streamState.socket.addEventListener("open", () => onOpen(account));
|
||||
|
||||
@@ -1,7 +1,6 @@
|
||||
import React, { useRef, useState } from "react";
|
||||
import { useSnapshot, ref } from "valtio";
|
||||
import Editor, { loader } from "@monaco-editor/react";
|
||||
import type monaco from "monaco-editor";
|
||||
import React, { useState } from "react";
|
||||
import { useSnapshot } from "valtio";
|
||||
|
||||
import { useTheme } from "next-themes";
|
||||
import { useRouter } from "next/router";
|
||||
import NextLink from "next/link";
|
||||
@@ -10,24 +9,16 @@ import filesize from "filesize";
|
||||
|
||||
import Box from "./Box";
|
||||
import Container from "./Container";
|
||||
import dark from "../theme/editor/amy.json";
|
||||
import light from "../theme/editor/xcode_default.json";
|
||||
import state from "../state";
|
||||
import wat from "../utils/wat-highlight";
|
||||
|
||||
import EditorNavigation from "./EditorNavigation";
|
||||
import { Button, Text, Link, Flex } from ".";
|
||||
|
||||
loader.config({
|
||||
paths: {
|
||||
vs: "https://cdn.jsdelivr.net/npm/monaco-editor@0.30.1/min/vs",
|
||||
},
|
||||
});
|
||||
import Monaco from "./Monaco";
|
||||
|
||||
const FILESIZE_BREAKPOINTS: [number, number] = [2 * 1024, 5 * 1024];
|
||||
|
||||
const DeployEditor = () => {
|
||||
const editorRef = useRef<monaco.editor.IStandaloneCodeEditor>();
|
||||
const snap = useSnapshot(state);
|
||||
const router = useRouter();
|
||||
const { theme } = useTheme();
|
||||
@@ -36,7 +27,7 @@ const DeployEditor = () => {
|
||||
|
||||
const activeFile = snap.files[snap.active]?.compiledContent
|
||||
? snap.files[snap.active]
|
||||
: snap.files.filter((file) => file.compiledContent)[0];
|
||||
: snap.files.filter(file => file.compiledContent)[0];
|
||||
const compiledSize = activeFile?.compiledContent?.byteLength || 0;
|
||||
const color =
|
||||
compiledSize > FILESIZE_BREAKPOINTS[1]
|
||||
@@ -45,6 +36,10 @@ const DeployEditor = () => {
|
||||
? "$warning"
|
||||
: "$success";
|
||||
|
||||
const isContentChanged =
|
||||
activeFile && activeFile.compiledValueSnapshot !== activeFile.content;
|
||||
// const hasDeployErros = activeFile && activeFile.containsErrors;
|
||||
|
||||
const CompiledStatView = activeFile && (
|
||||
<Flex
|
||||
column
|
||||
@@ -80,6 +75,12 @@ const DeployEditor = () => {
|
||||
<Button variant="link" onClick={() => setShowContent(true)}>
|
||||
View as WAT-file
|
||||
</Button>
|
||||
{isContentChanged && (
|
||||
<Text warning>
|
||||
File contents were changed after last compile, compile again to
|
||||
incorporate your latest changes in the build.
|
||||
</Text>
|
||||
)}
|
||||
</Flex>
|
||||
);
|
||||
const NoContentView = !snap.loading && router.isReady && (
|
||||
@@ -99,7 +100,7 @@ const DeployEditor = () => {
|
||||
</Text>
|
||||
);
|
||||
const isContent =
|
||||
snap.files?.filter((file) => file.compiledWatContent).length > 0 &&
|
||||
snap.files?.filter(file => file.compiledWatContent).length > 0 &&
|
||||
router.isReady;
|
||||
return (
|
||||
<Box
|
||||
@@ -126,32 +127,38 @@ const DeployEditor = () => {
|
||||
) : !showContent ? (
|
||||
CompiledStatView
|
||||
) : (
|
||||
<Editor
|
||||
<Monaco
|
||||
className="hooks-editor"
|
||||
defaultLanguage={"wat"}
|
||||
language={"wat"}
|
||||
path={`file://tmp/c/${activeFile?.name}.wat`}
|
||||
value={activeFile?.compiledWatContent || ""}
|
||||
beforeMount={(monaco) => {
|
||||
beforeMount={monaco => {
|
||||
monaco.languages.register({ id: "wat" });
|
||||
monaco.languages.setLanguageConfiguration("wat", wat.config);
|
||||
monaco.languages.setMonarchTokensProvider("wat", wat.tokens);
|
||||
if (!state.editorCtx) {
|
||||
state.editorCtx = ref(monaco.editor);
|
||||
// @ts-expect-error
|
||||
monaco.editor.defineTheme("dark", dark);
|
||||
// @ts-expect-error
|
||||
monaco.editor.defineTheme("light", light);
|
||||
}
|
||||
}}
|
||||
onMount={(editor, monaco) => {
|
||||
editorRef.current = editor;
|
||||
onMount={editor => {
|
||||
editor.updateOptions({
|
||||
glyphMargin: true,
|
||||
readOnly: true,
|
||||
});
|
||||
}}
|
||||
theme={theme === "dark" ? "dark" : "light"}
|
||||
overlay={
|
||||
<Flex
|
||||
css={{
|
||||
m: "$1",
|
||||
ml: "auto",
|
||||
fontSize: "$sm",
|
||||
color: "$textMuted",
|
||||
}}
|
||||
>
|
||||
<Link onClick={() => setShowContent(false)}>
|
||||
Exit editor mode
|
||||
</Link>
|
||||
</Flex>
|
||||
}
|
||||
/>
|
||||
)}
|
||||
</Container>
|
||||
|
||||
@@ -1,6 +1,5 @@
|
||||
import React, { useEffect, useRef } from "react";
|
||||
import { useSnapshot, ref } from "valtio";
|
||||
import Editor from "@monaco-editor/react";
|
||||
import type monaco from "monaco-editor";
|
||||
import { ArrowBendLeftUp } from "phosphor-react";
|
||||
import { useTheme } from "next-themes";
|
||||
@@ -8,8 +7,6 @@ import { useRouter } from "next/router";
|
||||
|
||||
import Box from "./Box";
|
||||
import Container from "./Container";
|
||||
import dark from "../theme/editor/amy.json";
|
||||
import light from "../theme/editor/xcode_default.json";
|
||||
import { saveFile } from "../state/actions";
|
||||
import { apiHeaderFiles } from "../state/constants";
|
||||
import state from "../state";
|
||||
@@ -22,10 +19,12 @@ import { listen } from "@codingame/monaco-jsonrpc";
|
||||
import ReconnectingWebSocket from "reconnecting-websocket";
|
||||
|
||||
import docs from "../xrpl-hooks-docs/docs";
|
||||
import Monaco from "./Monaco";
|
||||
import { saveAllFiles } from '../state/actions/saveFile';
|
||||
|
||||
const validateWritability = (editor: monaco.editor.IStandaloneCodeEditor) => {
|
||||
const currPath = editor.getModel()?.uri.path;
|
||||
if (apiHeaderFiles.find((h) => currPath?.endsWith(h))) {
|
||||
if (apiHeaderFiles.find(h => currPath?.endsWith(h))) {
|
||||
editor.updateOptions({ readOnly: true });
|
||||
} else {
|
||||
editor.updateOptions({ readOnly: false });
|
||||
@@ -42,7 +41,7 @@ const setMarkers = (monacoE: typeof monaco) => {
|
||||
.getModelMarkers({})
|
||||
// Filter out the markers that are hooks specific
|
||||
.filter(
|
||||
(marker) =>
|
||||
marker =>
|
||||
typeof marker?.code === "string" &&
|
||||
// Take only markers that starts with "hooks-"
|
||||
marker?.code?.includes("hooks-")
|
||||
@@ -56,16 +55,16 @@ const setMarkers = (monacoE: typeof monaco) => {
|
||||
// Add decoration (aka extra hoverMessages) to markers in the
|
||||
// exact same range (location) where the markers are
|
||||
const models = monacoE.editor.getModels();
|
||||
models.forEach((model) => {
|
||||
models.forEach(model => {
|
||||
decorations[model.id] = model?.deltaDecorations(
|
||||
decorations?.[model.id] || [],
|
||||
markers
|
||||
.filter((marker) =>
|
||||
.filter(marker =>
|
||||
marker?.resource.path
|
||||
.split("/")
|
||||
.includes(`${state.files?.[state.active]?.name}`)
|
||||
)
|
||||
.map((marker) => ({
|
||||
.map(marker => ({
|
||||
range: new monacoE.Range(
|
||||
marker.startLineNumber,
|
||||
marker.startColumn,
|
||||
@@ -113,6 +112,13 @@ const HooksEditor = () => {
|
||||
setMarkers(monacoRef.current);
|
||||
}
|
||||
}, [snap.active]);
|
||||
useEffect(() => {
|
||||
return () => {
|
||||
saveAllFiles();
|
||||
};
|
||||
}, []);
|
||||
|
||||
const file = snap.files[snap.active];
|
||||
return (
|
||||
<Box
|
||||
css={{
|
||||
@@ -127,16 +133,16 @@ const HooksEditor = () => {
|
||||
>
|
||||
<EditorNavigation />
|
||||
{snap.files.length > 0 && router.isReady ? (
|
||||
<Editor
|
||||
className="hooks-editor"
|
||||
<Monaco
|
||||
keepCurrentModel
|
||||
defaultLanguage={snap.files?.[snap.active]?.language}
|
||||
language={snap.files?.[snap.active]?.language}
|
||||
path={`file:///work/c/${snap.files?.[snap.active]?.name}`}
|
||||
defaultValue={snap.files?.[snap.active]?.content}
|
||||
beforeMount={(monaco) => {
|
||||
defaultLanguage={file?.language}
|
||||
language={file?.language}
|
||||
path={`file:///work/c/${file?.name}`}
|
||||
defaultValue={file?.content}
|
||||
// onChange={val => (state.files[snap.active].content = val)} // Auto save?
|
||||
beforeMount={monaco => {
|
||||
if (!snap.editorCtx) {
|
||||
snap.files.forEach((file) =>
|
||||
snap.files.forEach(file =>
|
||||
monaco.editor.createModel(
|
||||
file.content,
|
||||
file.language,
|
||||
@@ -161,7 +167,7 @@ const HooksEditor = () => {
|
||||
// listen when the web socket is opened
|
||||
listen({
|
||||
webSocket: webSocket as WebSocket,
|
||||
onConnection: (connection) => {
|
||||
onConnection: connection => {
|
||||
// create and start the language client
|
||||
const languageClient = createLanguageClient(connection);
|
||||
const disposable = languageClient.start();
|
||||
@@ -177,7 +183,6 @@ const HooksEditor = () => {
|
||||
});
|
||||
}
|
||||
|
||||
// // hook editor to global state
|
||||
// editor.updateOptions({
|
||||
// minimap: {
|
||||
// enabled: false,
|
||||
@@ -186,10 +191,6 @@ const HooksEditor = () => {
|
||||
// });
|
||||
if (!state.editorCtx) {
|
||||
state.editorCtx = ref(monaco.editor);
|
||||
// @ts-expect-error
|
||||
monaco.editor.defineTheme("dark", dark);
|
||||
// @ts-expect-error
|
||||
monaco.editor.defineTheme("light", light);
|
||||
}
|
||||
}}
|
||||
onMount={(editor, monaco) => {
|
||||
@@ -217,13 +218,13 @@ const HooksEditor = () => {
|
||||
});
|
||||
|
||||
// Hacky way to hide Peek menu
|
||||
editor.onContextMenu((e) => {
|
||||
editor.onContextMenu(e => {
|
||||
const host =
|
||||
document.querySelector<HTMLElement>(".shadow-root-host");
|
||||
|
||||
const contextMenuItems =
|
||||
host?.shadowRoot?.querySelectorAll("li.action-item");
|
||||
contextMenuItems?.forEach((k) => {
|
||||
contextMenuItems?.forEach(k => {
|
||||
// If menu item contains "Peek" lets hide it
|
||||
if (k.querySelector(".action-label")?.textContent === "Peek") {
|
||||
// @ts-expect-error
|
||||
|
||||
75
components/Monaco.tsx
Normal file
75
components/Monaco.tsx
Normal file
@@ -0,0 +1,75 @@
|
||||
import Editor, { loader, EditorProps, Monaco } from "@monaco-editor/react";
|
||||
import { CSS } from "@stitches/react";
|
||||
import type monaco from "monaco-editor";
|
||||
import { useTheme } from "next-themes";
|
||||
import { FC, MutableRefObject, ReactNode } from "react";
|
||||
import { Flex } from ".";
|
||||
import dark from "../theme/editor/amy.json";
|
||||
import light from "../theme/editor/xcode_default.json";
|
||||
|
||||
export type MonacoProps = EditorProps & {
|
||||
id?: string;
|
||||
rootProps?: { css: CSS } & Record<string, any>;
|
||||
overlay?: ReactNode;
|
||||
editorRef?: MutableRefObject<monaco.editor.IStandaloneCodeEditor>;
|
||||
monacoRef?: MutableRefObject<typeof monaco>;
|
||||
};
|
||||
|
||||
loader.config({
|
||||
paths: {
|
||||
vs: "https://cdn.jsdelivr.net/npm/monaco-editor@0.30.1/min/vs",
|
||||
},
|
||||
});
|
||||
|
||||
const Monaco: FC<MonacoProps> = ({
|
||||
id,
|
||||
path = `file:///${id}`,
|
||||
className = id,
|
||||
language = "json",
|
||||
overlay,
|
||||
editorRef,
|
||||
monacoRef,
|
||||
beforeMount,
|
||||
rootProps,
|
||||
...rest
|
||||
}) => {
|
||||
const { theme } = useTheme();
|
||||
const setTheme = (monaco: Monaco) => {
|
||||
monaco.editor.defineTheme("dark", dark as any);
|
||||
monaco.editor.defineTheme("light", light as any);
|
||||
};
|
||||
return (
|
||||
<Flex
|
||||
fluid
|
||||
column
|
||||
{...rootProps}
|
||||
css={{
|
||||
position: "relative",
|
||||
height: "100%",
|
||||
...rootProps?.css,
|
||||
}}
|
||||
>
|
||||
<Editor
|
||||
className={className}
|
||||
language={language}
|
||||
path={path}
|
||||
beforeMount={monaco => {
|
||||
beforeMount?.(monaco);
|
||||
|
||||
setTheme(monaco);
|
||||
}}
|
||||
theme={theme === "dark" ? "dark" : "light"}
|
||||
{...rest}
|
||||
/>
|
||||
{overlay && (
|
||||
<Flex
|
||||
css={{ position: "absolute", bottom: 0, right: 0, width: "100%" }}
|
||||
>
|
||||
{overlay}
|
||||
</Flex>
|
||||
)}
|
||||
</Flex>
|
||||
);
|
||||
};
|
||||
|
||||
export default Monaco;
|
||||
@@ -20,6 +20,11 @@ const Text = styled("span", {
|
||||
color: "$error",
|
||||
},
|
||||
},
|
||||
warning: {
|
||||
true: {
|
||||
color: "$warning",
|
||||
},
|
||||
},
|
||||
monospace: {
|
||||
true: {
|
||||
fontFamily: "$monospace",
|
||||
@@ -28,8 +33,8 @@ const Text = styled("span", {
|
||||
block: {
|
||||
true: {
|
||||
display: "block",
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
|
||||
@@ -1,9 +1,4 @@
|
||||
import Editor, { loader, useMonaco } from "@monaco-editor/react";
|
||||
import { FC, useCallback, useEffect, useState } from "react";
|
||||
import { useTheme } from "next-themes";
|
||||
|
||||
import dark from "../../theme/editor/amy.json";
|
||||
import light from "../../theme/editor/xcode_default.json";
|
||||
import { useSnapshot } from "valtio";
|
||||
import state, {
|
||||
prepareState,
|
||||
@@ -11,18 +6,13 @@ import state, {
|
||||
TransactionState,
|
||||
} from "../../state";
|
||||
import Text from "../Text";
|
||||
import Flex from "../Flex";
|
||||
import { Link } from "..";
|
||||
import { Flex, Link } from "..";
|
||||
import { showAlert } from "../../state/actions/showAlert";
|
||||
import { parseJSON } from "../../utils/json";
|
||||
import { extractSchemaProps } from "../../utils/schema";
|
||||
import amountSchema from "../../content/amount-schema.json";
|
||||
|
||||
loader.config({
|
||||
paths: {
|
||||
vs: "https://cdn.jsdelivr.net/npm/monaco-editor@0.30.1/min/vs",
|
||||
},
|
||||
});
|
||||
import Monaco from "../Monaco";
|
||||
import type monaco from "monaco-editor";
|
||||
|
||||
interface JsonProps {
|
||||
value?: string;
|
||||
@@ -40,7 +30,6 @@ export const TxJson: FC<JsonProps> = ({
|
||||
}) => {
|
||||
const { editorSettings, accounts } = useSnapshot(state);
|
||||
const { editorValue = value, estimatedFee } = txState;
|
||||
const { theme } = useTheme();
|
||||
const [hasUnsaved, setHasUnsaved] = useState(false);
|
||||
const [currTxType, setCurrTxType] = useState<string | undefined>(
|
||||
txState.selectedTransaction?.value
|
||||
@@ -95,9 +84,6 @@ export const TxJson: FC<JsonProps> = ({
|
||||
});
|
||||
};
|
||||
|
||||
const path = `file:///${header}`;
|
||||
const monaco = useMonaco();
|
||||
|
||||
const getSchemas = useCallback(async (): Promise<any[]> => {
|
||||
const txObj = transactionsData.find(
|
||||
td => td.TransactionType === currTxType
|
||||
@@ -177,55 +163,63 @@ export const TxJson: FC<JsonProps> = ({
|
||||
];
|
||||
}, [accounts, currTxType, estimatedFee, header]);
|
||||
|
||||
const [monacoInst, setMonacoInst] = useState<typeof monaco>();
|
||||
useEffect(() => {
|
||||
if (!monaco) return;
|
||||
if (!monacoInst) return;
|
||||
getSchemas().then(schemas => {
|
||||
monaco.languages.json.jsonDefaults.setDiagnosticsOptions({
|
||||
monacoInst.languages.json.jsonDefaults.setDiagnosticsOptions({
|
||||
validate: true,
|
||||
schemas,
|
||||
});
|
||||
});
|
||||
}, [getSchemas, monaco]);
|
||||
}, [getSchemas, monacoInst]);
|
||||
|
||||
return (
|
||||
<Flex
|
||||
fluid
|
||||
column
|
||||
css={{ height: "calc(100% - 45px)", position: "relative" }}
|
||||
>
|
||||
<Editor
|
||||
className="hooks-editor"
|
||||
language={"json"}
|
||||
path={path}
|
||||
height="100%"
|
||||
beforeMount={monaco => {
|
||||
monaco.editor.defineTheme("dark", dark as any);
|
||||
monaco.editor.defineTheme("light", light as any);
|
||||
}}
|
||||
value={editorValue}
|
||||
onChange={val => setState({ editorValue: val })}
|
||||
onMount={(editor, monaco) => {
|
||||
editor.updateOptions({
|
||||
minimap: { enabled: false },
|
||||
glyphMargin: true,
|
||||
tabSize: editorSettings.tabSize,
|
||||
dragAndDrop: true,
|
||||
fontSize: 14,
|
||||
});
|
||||
<Monaco
|
||||
rootProps={{
|
||||
css: { height: "calc(100% - 45px)" },
|
||||
}}
|
||||
language={"json"}
|
||||
id={header}
|
||||
height="100%"
|
||||
value={editorValue}
|
||||
onChange={val => setState({ editorValue: val })}
|
||||
onMount={(editor, monaco) => {
|
||||
editor.updateOptions({
|
||||
minimap: { enabled: false },
|
||||
glyphMargin: true,
|
||||
tabSize: editorSettings.tabSize,
|
||||
dragAndDrop: true,
|
||||
fontSize: 14,
|
||||
});
|
||||
|
||||
// register onExit cb
|
||||
const model = editor.getModel();
|
||||
model?.onWillDispose(() => onExit(model.getValue()));
|
||||
}}
|
||||
theme={theme === "dark" ? "dark" : "light"}
|
||||
/>
|
||||
{hasUnsaved && (
|
||||
<Text muted small css={{ position: "absolute", bottom: 0, right: 0 }}>
|
||||
This file has unsaved changes.{" "}
|
||||
<Link onClick={() => saveState(editorValue, currTxType)}>save</Link>{" "}
|
||||
<Link onClick={discardChanges}>discard</Link>
|
||||
</Text>
|
||||
)}
|
||||
</Flex>
|
||||
setMonacoInst(monaco);
|
||||
// register onExit cb
|
||||
const model = editor.getModel();
|
||||
model?.onWillDispose(() => onExit(model.getValue()));
|
||||
}}
|
||||
overlay={
|
||||
hasUnsaved ? (
|
||||
<Flex
|
||||
row
|
||||
align="center"
|
||||
css={{ fontSize: "$xs", color: "$textMuted", ml: 'auto' }}
|
||||
>
|
||||
<Text muted small>
|
||||
This file has unsaved changes.
|
||||
</Text>
|
||||
<Link
|
||||
css={{ ml: "$1" }}
|
||||
onClick={() => saveState(editorValue, currTxType)}
|
||||
>
|
||||
save
|
||||
</Link>
|
||||
<Link css={{ ml: "$1" }} onClick={discardChanges}>
|
||||
discard
|
||||
</Link>
|
||||
</Flex>
|
||||
) : undefined
|
||||
}
|
||||
/>
|
||||
);
|
||||
};
|
||||
|
||||
@@ -38,22 +38,23 @@ export const TxUI: FC<UIProps> = ({
|
||||
txFields,
|
||||
} = txState;
|
||||
|
||||
const transactionsOptions = transactionsData.map((tx) => ({
|
||||
|
||||
const transactionsOptions = transactionsData.map(tx => ({
|
||||
value: tx.TransactionType,
|
||||
label: tx.TransactionType,
|
||||
}));
|
||||
|
||||
const accountOptions: SelectOption[] = accounts.map((acc) => ({
|
||||
const accountOptions: SelectOption[] = accounts.map(acc => ({
|
||||
label: acc.name,
|
||||
value: acc.address,
|
||||
}));
|
||||
|
||||
const destAccountOptions: SelectOption[] = accounts
|
||||
.map((acc) => ({
|
||||
.map(acc => ({
|
||||
label: acc.name,
|
||||
value: acc.address,
|
||||
}))
|
||||
.filter((acc) => acc.value !== selectedAccount?.value);
|
||||
.filter(acc => acc.value !== selectedAccount?.value);
|
||||
|
||||
const [feeLoading, setFeeLoading] = useState(false);
|
||||
|
||||
@@ -97,32 +98,37 @@ export const TxUI: FC<UIProps> = ({
|
||||
[estimateFee, handleSetField]
|
||||
);
|
||||
|
||||
const handleChangeTxType = (tt: SelectOption) => {
|
||||
setState({ selectedTransaction: tt });
|
||||
const handleChangeTxType = useCallback(
|
||||
(tt: SelectOption) => {
|
||||
setState({ selectedTransaction: tt });
|
||||
|
||||
const newState = resetOptions(tt.value);
|
||||
const newState = resetOptions(tt.value);
|
||||
|
||||
handleEstimateFee(newState, true);
|
||||
};
|
||||
handleEstimateFee(newState, true);
|
||||
},
|
||||
[handleEstimateFee, resetOptions, setState]
|
||||
);
|
||||
|
||||
const specialFields = ["TransactionType", "Account", "Destination"];
|
||||
|
||||
const otherFields = Object.keys(txFields).filter(
|
||||
(k) => !specialFields.includes(k)
|
||||
k => !specialFields.includes(k)
|
||||
) as [keyof TxFields];
|
||||
|
||||
const switchToJson = () =>
|
||||
setState({ editorSavedValue: null, viewType: "json" });
|
||||
|
||||
// default tx
|
||||
useEffect(() => {
|
||||
if (selectedTransaction?.value) return;
|
||||
|
||||
const defaultOption = transactionsOptions.find(
|
||||
(tt) => tt.value === "Payment"
|
||||
tt => tt.value === "Payment"
|
||||
);
|
||||
if (defaultOption) {
|
||||
handleChangeTxType(defaultOption);
|
||||
}
|
||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||
}, []);
|
||||
}, [handleChangeTxType, selectedTransaction?.value, transactionsOptions]);
|
||||
|
||||
return (
|
||||
<Container
|
||||
@@ -204,7 +210,7 @@ export const TxUI: FC<UIProps> = ({
|
||||
/>
|
||||
</Flex>
|
||||
)}
|
||||
{otherFields.map((field) => {
|
||||
{otherFields.map(field => {
|
||||
let _value = txFields[field];
|
||||
|
||||
let value: string | undefined;
|
||||
@@ -255,7 +261,7 @@ export const TxUI: FC<UIProps> = ({
|
||||
<Input
|
||||
type={isFee ? "number" : "text"}
|
||||
value={value}
|
||||
onChange={(e) => {
|
||||
onChange={e => {
|
||||
if (isFee) {
|
||||
const val = e.target.value
|
||||
.replaceAll(".", "")
|
||||
@@ -267,7 +273,7 @@ export const TxUI: FC<UIProps> = ({
|
||||
}}
|
||||
onKeyPress={
|
||||
isFee
|
||||
? (e) => {
|
||||
? e => {
|
||||
if (e.key === "." || e.key === ",") {
|
||||
e.preventDefault();
|
||||
}
|
||||
|
||||
@@ -40,9 +40,9 @@
|
||||
{
|
||||
"label": "Token",
|
||||
"body": {
|
||||
"currency": "${1:13.1}",
|
||||
"value": "${2:FOO}",
|
||||
"description": "${3:rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpns}"
|
||||
"currency": "${1:USD}",
|
||||
"value": "${2:100}",
|
||||
"issuer": "${3:rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpns}"
|
||||
}
|
||||
}
|
||||
]
|
||||
|
||||
@@ -79,25 +79,40 @@
|
||||
"Condition": "A0258020E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855810100",
|
||||
"Fulfillment": "A0028000"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenMint",
|
||||
"Account": "rvYAfWj5gh67oV6fW32ZzP3Aw4Eubs59B",
|
||||
"Fee": "10",
|
||||
"NFTokenTaxon": 0,
|
||||
"URI": "697066733A2F2F516D614374444B5A4656767666756676626479346573745A626851483744586831364354707631686F776D424779"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenBurn",
|
||||
"Account": "rvYAfWj5gh67oV6fW32ZzP3Aw4Eubs59B",
|
||||
"Fee": "10",
|
||||
"TokenID": "000B013A95F14B0044F78A264E41713C64B5F89242540EE208C3098E00000D65"
|
||||
"NFTokenID": "000B013A95F14B0044F78A264E41713C64B5F89242540EE208C3098E00000D65"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenAcceptOffer",
|
||||
"Fee": "10"
|
||||
"Account": "rvYAfWj5gh67oV6fW32ZzP3Aw4Eubs59B",
|
||||
"Fee": "10",
|
||||
"NFTokenSellOffer": "A2FA1A9911FE2AEF83DAB05F437768E26A301EF899BD31EB85E704B3D528FF18",
|
||||
"NFTokenBuyOffer": "4AAAEEA76E3C8148473CB3840CE637676E561FB02BD4CA22CA59729EA815B862",
|
||||
"NFTokenBrokerFee": "10"
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenCancelOffer",
|
||||
"Account": "ra5nK24KXen9AHvsdFTKHSANinZseWnPcX",
|
||||
"TokenIDs": "000100001E962F495F07A990F4ED55ACCFEEF365DBAA76B6A048C0A200000007"
|
||||
"Fee": "10",
|
||||
"NFTokenOffers": {
|
||||
"$type": "json",
|
||||
"$value": ["4AAAEEA76E3C8148473CB3840CE637676E561FB02BD4CA22CA59729EA815B862"]
|
||||
}
|
||||
},
|
||||
{
|
||||
"TransactionType": "NFTokenCreateOffer",
|
||||
"Account": "rs8jBmmfpwgmrSPgwMsh7CvKRmRt1JTVSX",
|
||||
"TokenID": "000100001E962F495F07A990F4ED55ACCFEEF365DBAA76B6A048C0A200000007",
|
||||
"NFTokenID": "000100001E962F495F07A990F4ED55ACCFEEF365DBAA76B6A048C0A200000007",
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
@@ -212,9 +227,13 @@
|
||||
"Fee": "12",
|
||||
"Flags": 262144,
|
||||
"LastLedgerSequence": 8007750,
|
||||
"Amount": {
|
||||
"$value": "100",
|
||||
"$type": "xrp"
|
||||
"LimitAmount": {
|
||||
"$type": "json",
|
||||
"$value": {
|
||||
"currency": "USD",
|
||||
"issuer": "rsP3mgGb2tcYUrxiLFiHJiQXhsziegtwBc",
|
||||
"value": "100"
|
||||
}
|
||||
},
|
||||
"Sequence": 12
|
||||
}
|
||||
|
||||
@@ -61,7 +61,7 @@
|
||||
"vscode-languageserver": "^7.0.0",
|
||||
"vscode-uri": "^3.0.2",
|
||||
"wabt": "1.0.16",
|
||||
"xrpl-accountlib": "^1.3.2",
|
||||
"xrpl-accountlib": "^1.5.2",
|
||||
"xrpl-client": "^1.9.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -27,7 +27,7 @@ export const names = [
|
||||
* new account with 10 000 XRP. Hooks Testnet /newcreds endpoint
|
||||
* is protected with CORS so that's why we did our own endpoint
|
||||
*/
|
||||
export const addFaucetAccount = async (showToast: boolean = false) => {
|
||||
export const addFaucetAccount = async (name?: string, showToast: boolean = false) => {
|
||||
// Lets limit the number of faucet accounts to 5 for now
|
||||
if (state.accounts.length > 5) {
|
||||
return toast.error("You can only have maximum 6 accounts");
|
||||
@@ -52,7 +52,7 @@ export const addFaucetAccount = async (showToast: boolean = false) => {
|
||||
}
|
||||
const currNames = state.accounts.map(acc => acc.name);
|
||||
state.accounts.push({
|
||||
name: names.filter(name => !currNames.includes(name))[0],
|
||||
name: name || names.filter(name => !currNames.includes(name))[0],
|
||||
xrp: (json.xrp || 0 * 1000000).toString(),
|
||||
address: json.address,
|
||||
secret: json.secret,
|
||||
|
||||
@@ -14,19 +14,21 @@ import { ref } from "valtio";
|
||||
*/
|
||||
export const compileCode = async (activeId: number) => {
|
||||
// Save the file to global state
|
||||
saveFile(false);
|
||||
saveFile(false, activeId);
|
||||
if (!process.env.NEXT_PUBLIC_COMPILE_API_ENDPOINT) {
|
||||
throw Error("Missing env!");
|
||||
}
|
||||
// Bail out if we're already compiling
|
||||
if (state.compiling) {
|
||||
// if compiling is ongoing return
|
||||
// if compiling is ongoing return // TODO Inform user about it.
|
||||
return;
|
||||
}
|
||||
// Set loading state to true
|
||||
state.compiling = true;
|
||||
state.logs = []
|
||||
const file = state.files[activeId]
|
||||
try {
|
||||
file.containsErrors = false
|
||||
const res = await fetch(process.env.NEXT_PUBLIC_COMPILE_API_ENDPOINT, {
|
||||
method: "POST",
|
||||
headers: {
|
||||
@@ -40,8 +42,8 @@ export const compileCode = async (activeId: number) => {
|
||||
{
|
||||
type: "c",
|
||||
options: state.compileOptions.optimizationLevel || '-O2',
|
||||
name: state.files[activeId].name,
|
||||
src: state.files[activeId].content,
|
||||
name: file.name,
|
||||
src: file.content,
|
||||
},
|
||||
],
|
||||
}),
|
||||
@@ -49,15 +51,15 @@ export const compileCode = async (activeId: number) => {
|
||||
const json = await res.json();
|
||||
state.compiling = false;
|
||||
if (!json.success) {
|
||||
state.logs.push({ type: "error", message: json.message });
|
||||
const errors = [json.message]
|
||||
if (json.tasks && json.tasks.length > 0) {
|
||||
json.tasks.forEach((task: any) => {
|
||||
if (!task.success) {
|
||||
state.logs.push({ type: "error", message: task?.console });
|
||||
errors.push(task?.console)
|
||||
}
|
||||
});
|
||||
}
|
||||
return toast.error(`Couldn't compile!`, { position: "bottom-center" });
|
||||
throw errors
|
||||
}
|
||||
state.logs.push({
|
||||
type: "success",
|
||||
@@ -67,8 +69,9 @@ export const compileCode = async (activeId: number) => {
|
||||
});
|
||||
// Decode base64 encoded wasm that is coming back from the endpoint
|
||||
const bufferData = await decodeBinary(json.output);
|
||||
state.files[state.active].compiledContent = ref(bufferData);
|
||||
state.files[state.active].lastCompiled = new Date();
|
||||
file.compiledContent = ref(bufferData);
|
||||
file.lastCompiled = new Date();
|
||||
file.compiledValueSnapshot = file.content
|
||||
// Import wabt from and create human readable version of wasm file and
|
||||
// put it into state
|
||||
import("wabt").then((wabt) => {
|
||||
@@ -84,10 +87,23 @@ export const compileCode = async (activeId: number) => {
|
||||
});
|
||||
} catch (err) {
|
||||
console.log(err);
|
||||
state.logs.push({
|
||||
type: "error",
|
||||
message: "Error occured while compiling!",
|
||||
});
|
||||
|
||||
if (err instanceof Array && typeof err[0] === 'string') {
|
||||
err.forEach(message => {
|
||||
state.logs.push({
|
||||
type: "error",
|
||||
message,
|
||||
});
|
||||
})
|
||||
}
|
||||
else {
|
||||
state.logs.push({
|
||||
type: "error",
|
||||
message: "Something went wrong, check your connection try again later!",
|
||||
});
|
||||
}
|
||||
state.compiling = false;
|
||||
toast.error(`Error occurred while compiling!`, { position: "bottom-center" });
|
||||
file.containsErrors = true
|
||||
}
|
||||
};
|
||||
|
||||
@@ -189,7 +189,7 @@ export const deployHook = async (
|
||||
console.log(err);
|
||||
state.deployLogs.push({
|
||||
type: "error",
|
||||
message: "Error occured while deploying",
|
||||
message: "Error occurred while deploying",
|
||||
});
|
||||
}
|
||||
if (currentAccount) {
|
||||
@@ -272,10 +272,10 @@ export const deleteHook = async (account: IAccount & { name?: string }) => {
|
||||
}
|
||||
} catch (err) {
|
||||
console.log(err);
|
||||
toast.error("Error occured while deleting hoook", { id: toastId });
|
||||
toast.error("Error occurred while deleting hook", { id: toastId });
|
||||
state.deployLogs.push({
|
||||
type: "error",
|
||||
message: "Error occured while deleting hook",
|
||||
message: "Error occurred while deleting hook",
|
||||
});
|
||||
}
|
||||
if (currentAccount) {
|
||||
|
||||
@@ -13,7 +13,7 @@ export const downloadAsZip = async () => {
|
||||
const zipFileName = guessZipFileName(files);
|
||||
zipped.saveFile(zipFileName);
|
||||
} catch (error) {
|
||||
toast.error('Error occured while creating zip file, try again later')
|
||||
toast.error('Error occurred while creating zip file, try again later')
|
||||
} finally {
|
||||
state.zipLoading = false
|
||||
}
|
||||
|
||||
@@ -5,7 +5,7 @@ import state from '../index';
|
||||
import { names } from './addFaucetAccount';
|
||||
|
||||
// Adds test account to global state with secret key
|
||||
export const importAccount = (secret: string) => {
|
||||
export const importAccount = (secret: string, name?: string) => {
|
||||
if (!secret) {
|
||||
return toast.error("You need to add secret!");
|
||||
}
|
||||
@@ -19,7 +19,7 @@ export const importAccount = (secret: string) => {
|
||||
if (err?.message) {
|
||||
toast.error(err.message)
|
||||
} else {
|
||||
toast.error('Error occured while importing account')
|
||||
toast.error('Error occurred while importing account')
|
||||
}
|
||||
return;
|
||||
}
|
||||
@@ -27,7 +27,7 @@ export const importAccount = (secret: string) => {
|
||||
return toast.error(`Couldn't create account!`);
|
||||
}
|
||||
state.accounts.push({
|
||||
name: names[state.accounts.length],
|
||||
name: name || names[state.accounts.length],
|
||||
address: account.address || "",
|
||||
secret: account.secret.familySeed || "",
|
||||
xrp: "0",
|
||||
|
||||
@@ -2,14 +2,15 @@ import toast from "react-hot-toast";
|
||||
import state from '../index';
|
||||
|
||||
// Saves the current editor content to global state
|
||||
export const saveFile = (showToast: boolean = true) => {
|
||||
export const saveFile = (showToast: boolean = true, activeId?: number) => {
|
||||
const editorModels = state.editorCtx?.getModels();
|
||||
const sought = '/' + state.files[state.active].name;
|
||||
const currentModel = editorModels?.find((editorModel) => {
|
||||
return editorModel.uri.path.endsWith(sought);
|
||||
});
|
||||
const file = state.files[activeId || state.active]
|
||||
if (state.files.length > 0) {
|
||||
state.files[state.active].content = currentModel?.getValue() || "";
|
||||
file.content = currentModel?.getValue() || "";
|
||||
}
|
||||
if (showToast) {
|
||||
toast.success("Saved successfully", { position: "bottom-center" });
|
||||
|
||||
@@ -24,7 +24,6 @@ export const sendTransaction = async (account: IAccount, txOptions: TransactionO
|
||||
Fee, // TODO auto-fillable default
|
||||
...opts
|
||||
};
|
||||
|
||||
const { logPrefix = '' } = options || {}
|
||||
try {
|
||||
const signedAccount = derive.familySeed(account.secret);
|
||||
|
||||
@@ -13,9 +13,11 @@ export interface IFile {
|
||||
name: string;
|
||||
language: string;
|
||||
content: string;
|
||||
compiledValueSnapshot?: string
|
||||
compiledContent?: ArrayBuffer | null;
|
||||
compiledWatContent?: string | null;
|
||||
lastCompiled?: Date
|
||||
containsErrors?: boolean
|
||||
}
|
||||
|
||||
export interface FaucetAccountRes {
|
||||
|
||||
@@ -118,7 +118,7 @@ export const prepareTransaction = (data: any) => {
|
||||
// handle type: `json`
|
||||
if (_value && typeof _value === "object" && _value.$type === "json") {
|
||||
if (typeof _value.$value === "object") {
|
||||
options[field] = _value.$value as any;
|
||||
options[field] = _value.$value;
|
||||
} else {
|
||||
try {
|
||||
options[field] = JSON.parse(_value.$value);
|
||||
@@ -131,7 +131,7 @@ export const prepareTransaction = (data: any) => {
|
||||
}
|
||||
}
|
||||
|
||||
// delete unneccesary fields
|
||||
// delete unnecessary fields
|
||||
if (options[field] === undefined) {
|
||||
delete options[field];
|
||||
}
|
||||
|
||||
@@ -19,6 +19,6 @@ export const getErrors = (source?: string): Error | undefined => {
|
||||
);
|
||||
if (!probs.length) return undefined
|
||||
const errors = probs.map(prob => `[${prob.code}] on line ${prob.line}: ${prob.message}`)
|
||||
const error = new Error(`The following error(s) occured while parsing JSDOC: \n${errors.join('\n')}`)
|
||||
const error = new Error(`The following error(s) occurred while parsing JSDOC: \n${errors.join('\n')}`)
|
||||
return error
|
||||
}
|
||||
@@ -6,4 +6,10 @@ export const guessZipFileName = (files: File[]) => {
|
||||
let parts = (files.filter(f => f.name.endsWith('.c'))[0]?.name || 'hook').split('.')
|
||||
parts = parts.length > 1 ? parts.slice(0, -1) : parts
|
||||
return parts.join('')
|
||||
}
|
||||
|
||||
export const capitalize = (value?: string) => {
|
||||
if (!value) return '';
|
||||
|
||||
return value[0].toLocaleUpperCase() + value.slice(1);
|
||||
}
|
||||
@@ -18,13 +18,18 @@ export const tts = {
|
||||
ttDEPOSIT_PREAUTH: 19,
|
||||
ttTRUST_SET: 20,
|
||||
ttACCOUNT_DELETE: 21,
|
||||
ttHOOK_SET: 22
|
||||
ttHOOK_SET: 22,
|
||||
ttNFTOKEN_MINT: 25,
|
||||
ttNFTOKEN_BURN: 26,
|
||||
ttNFTOKEN_CREATE_OFFER: 27,
|
||||
ttNFTOKEN_CANCEL_OFFER: 28,
|
||||
ttNFTOKEN_ACCEPT_OFFER: 29
|
||||
};
|
||||
|
||||
export type TTS = typeof tts;
|
||||
|
||||
const calculateHookOn = (arr: (keyof TTS)[]) => {
|
||||
let start = '0x00000000003ff5bf';
|
||||
let start = '0x000000003e3ff5bf';
|
||||
arr.forEach(n => {
|
||||
let v = BigInt(start);
|
||||
v ^= (BigInt(1) << BigInt(tts[n as keyof TTS]));
|
||||
|
||||
73
yarn.lock
73
yarn.lock
@@ -1280,14 +1280,6 @@ babel-plugin-macros@^2.6.1:
|
||||
cosmiconfig "^6.0.0"
|
||||
resolve "^1.12.0"
|
||||
|
||||
babel-runtime@^6.26.0:
|
||||
version "6.26.0"
|
||||
resolved "https://registry.npmjs.org/babel-runtime/-/babel-runtime-6.26.0.tgz"
|
||||
integrity sha1-llxwWGaOgrVde/4E/yM3vItWR/4=
|
||||
dependencies:
|
||||
core-js "^2.4.0"
|
||||
regenerator-runtime "^0.11.0"
|
||||
|
||||
balanced-match@^1.0.0:
|
||||
version "1.0.2"
|
||||
resolved "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz"
|
||||
@@ -1552,11 +1544,6 @@ core-js-pure@^3.20.2:
|
||||
resolved "https://registry.npmjs.org/core-js-pure/-/core-js-pure-3.21.1.tgz"
|
||||
integrity sha512-12VZfFIu+wyVbBebyHmRTuEE/tZrB4tJToWcwAMcsp3h4+sHR+fMJWbKpYiCRWlhFBq+KNyO8rIV9rTkeVmznQ==
|
||||
|
||||
core-js@^2.4.0:
|
||||
version "2.6.12"
|
||||
resolved "https://registry.npmjs.org/core-js/-/core-js-2.6.12.tgz"
|
||||
integrity sha512-Kb2wC0fvsWfQrgk8HU5lW6U/Lcs8+9aaYcy4ZFc6DDlo4nZ7n70dEgE5rtR0oG6ufKDUnrwfWL1mXR5ljDatrQ==
|
||||
|
||||
core-util-is@~1.0.0:
|
||||
version "1.0.3"
|
||||
resolved "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz"
|
||||
@@ -3545,11 +3532,6 @@ reconnecting-websocket@^4.4.0:
|
||||
resolved "https://registry.npmjs.org/reconnecting-websocket/-/reconnecting-websocket-4.4.0.tgz"
|
||||
integrity sha512-D2E33ceRPga0NvTDhJmphEgJ7FUYF0v4lr1ki0csq06OdlxKfugGzN0dSkxM/NfqCxYELK4KcaTOUOjTV6Dcng==
|
||||
|
||||
regenerator-runtime@^0.11.0:
|
||||
version "0.11.1"
|
||||
resolved "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz"
|
||||
integrity sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg==
|
||||
|
||||
regenerator-runtime@^0.13.4:
|
||||
version "0.13.9"
|
||||
resolved "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.9.tgz"
|
||||
@@ -3648,20 +3630,15 @@ ripple-address-codec@^4.1.0, ripple-address-codec@^4.1.1, ripple-address-codec@^
|
||||
base-x "3.0.9"
|
||||
create-hash "^1.1.2"
|
||||
|
||||
ripple-binary-codec@^0.2.4:
|
||||
version "0.2.7"
|
||||
resolved "https://registry.npmjs.org/ripple-binary-codec/-/ripple-binary-codec-0.2.7.tgz"
|
||||
integrity sha512-VD+sHgZK76q3kmO765klFHPDCEveS5SUeg/bUNVpNrj7w2alyDNkbF17XNbAjFv+kSYhfsUudQanoaSs2Y6uzw==
|
||||
ripple-address-codec@^4.2.4:
|
||||
version "4.2.4"
|
||||
resolved "https://registry.yarnpkg.com/ripple-address-codec/-/ripple-address-codec-4.2.4.tgz#a56c2168c8bb81269ea4d15ed96d6824c5a866f8"
|
||||
integrity sha512-roAOjKz94+FboTItey1XRh5qynwt4xvfBLvbbcx+FiR94Yw2x3LrKLF2GVCMCSAh5I6PkcpADg6AbYsUbGN3nA==
|
||||
dependencies:
|
||||
babel-runtime "^6.26.0"
|
||||
bn.js "^5.1.1"
|
||||
create-hash "^1.2.0"
|
||||
decimal.js "^10.2.0"
|
||||
inherits "^2.0.4"
|
||||
lodash "^4.17.15"
|
||||
ripple-address-codec "^4.1.0"
|
||||
base-x "3.0.9"
|
||||
create-hash "^1.1.2"
|
||||
|
||||
ripple-binary-codec@^1.1.3, ripple-binary-codec@^1.3.0:
|
||||
ripple-binary-codec@^1.1.3:
|
||||
version "1.3.2"
|
||||
resolved "https://registry.npmjs.org/ripple-binary-codec/-/ripple-binary-codec-1.3.2.tgz"
|
||||
integrity sha512-8VG1vfb3EM1J7ZdPXo9E57Zv2hF4cxT64gP6rGSQzODVgMjiBCWozhN3729qNTGtHItz0e82Oix8v95vWYBQ3A==
|
||||
@@ -3673,6 +3650,18 @@ ripple-binary-codec@^1.1.3, ripple-binary-codec@^1.3.0:
|
||||
decimal.js "^10.2.0"
|
||||
ripple-address-codec "^4.2.3"
|
||||
|
||||
ripple-binary-codec@^1.4.2:
|
||||
version "1.4.2"
|
||||
resolved "https://registry.yarnpkg.com/ripple-binary-codec/-/ripple-binary-codec-1.4.2.tgz#cdc35353e4bc7c3a704719247c82b4c4d0b57dd3"
|
||||
integrity sha512-EDKIyZMa/6Ay/oNgCwjD9b9CJv0zmBreeHVQeG4BYwy+9GPnIQjNeT5e/aB6OjAnhcmpgbPeBmzwmNVwzxlt0w==
|
||||
dependencies:
|
||||
assert "^2.0.0"
|
||||
big-integer "^1.6.48"
|
||||
buffer "5.6.0"
|
||||
create-hash "^1.2.0"
|
||||
decimal.js "^10.2.0"
|
||||
ripple-address-codec "^4.2.4"
|
||||
|
||||
ripple-bs58@^4.0.0:
|
||||
version "4.0.1"
|
||||
resolved "https://registry.npmjs.org/ripple-bs58/-/ripple-bs58-4.0.1.tgz"
|
||||
@@ -3696,7 +3685,7 @@ ripple-hashes@^0.3.4, ripple-hashes@latest:
|
||||
bignumber.js "^4.1.0"
|
||||
create-hash "^1.1.2"
|
||||
ripple-address-codec "^3.0.4"
|
||||
ripple-binary-codec "^0.2.4"
|
||||
ripple-binary-codec "^1.4.2"
|
||||
|
||||
ripple-keypairs@^1.0.3, ripple-keypairs@latest:
|
||||
version "1.1.3"
|
||||
@@ -4328,10 +4317,10 @@ ws@^7.2.0:
|
||||
resolved "https://registry.npmjs.org/ws/-/ws-7.5.7.tgz"
|
||||
integrity sha512-KMvVuFzpKBuiIXW3E4u3mySRO2/mCHSyZDJQM5NQ9Q9KHWHWh0NHgfbRMLLrceUK5qAL4ytALJbpRMjixFZh8A==
|
||||
|
||||
xrpl-accountlib@^1.3.2:
|
||||
version "1.3.2"
|
||||
resolved "https://registry.npmjs.org/xrpl-accountlib/-/xrpl-accountlib-1.3.2.tgz"
|
||||
integrity sha512-mXwoumGp0xUiZ7Ty/1o4FHVRK4uLnqngxdYmikZs50drMjlgCUP6GXun2Vf4Uus1fnVnxhXIw+E7peH5OjiOJA==
|
||||
xrpl-accountlib@^1.5.2:
|
||||
version "1.5.2"
|
||||
resolved "https://registry.yarnpkg.com/xrpl-accountlib/-/xrpl-accountlib-1.5.2.tgz#8f16abe449fd60ba9ed75597f6ce3f0c45dfff43"
|
||||
integrity sha512-lieY2/5G9DySqdtgQ0AD/aMMG5Sy/MLAmbIsmsCaF06scM5DpR8s4SsEzgHni7dOG68Wjnb2Uz6tf5aV+l4/Kg==
|
||||
dependencies:
|
||||
assert "^2.0.0"
|
||||
bip32 "^2.0.5"
|
||||
@@ -4340,13 +4329,13 @@ xrpl-accountlib@^1.3.2:
|
||||
elliptic "6.5.4"
|
||||
hash.js "^1.1.7"
|
||||
ripple-address-codec "^4.1.0"
|
||||
ripple-binary-codec "^1.3.0"
|
||||
ripple-binary-codec "^1.4.2"
|
||||
ripple-hashes "^0.3.4"
|
||||
ripple-keypairs "^1.0.3"
|
||||
ripple-lib "^1.6.4"
|
||||
ripple-secret-codec "^1.0.2"
|
||||
xrpl-secret-numbers "^0.3.3"
|
||||
xrpl-sign-keypairs "^2.0.1"
|
||||
xrpl-sign-keypairs "^2.1.1"
|
||||
|
||||
xrpl-client@^1.9.4:
|
||||
version "1.9.4"
|
||||
@@ -4365,13 +4354,13 @@ xrpl-secret-numbers@^0.3.3:
|
||||
brorand "^1.1.0"
|
||||
ripple-keypairs "^1.0.3"
|
||||
|
||||
xrpl-sign-keypairs@^2.0.1:
|
||||
version "2.0.1"
|
||||
resolved "https://registry.npmjs.org/xrpl-sign-keypairs/-/xrpl-sign-keypairs-2.0.1.tgz"
|
||||
integrity sha512-84QbE3trxetaw0hqDADCWMx0HH1VAWnTJp0TGoKTGRf1jzTqjI7eNNNw5lmcay2MH8bW/waNzJIF8vSAJSkVrQ==
|
||||
xrpl-sign-keypairs@^2.1.1:
|
||||
version "2.1.1"
|
||||
resolved "https://registry.yarnpkg.com/xrpl-sign-keypairs/-/xrpl-sign-keypairs-2.1.1.tgz#2f7f2855799c5d4ba091007963825eef1db21a4e"
|
||||
integrity sha512-rKQmUCx+x7gjjJ5zv/Z7bOYR+8I36JwUCFlpuD9UzYD4w2msGQDG0rmxVENyZSfThDBVQ1kEArVn6SMDMe9LUQ==
|
||||
dependencies:
|
||||
big-integer latest
|
||||
ripple-binary-codec "^1.3.0"
|
||||
ripple-binary-codec "^1.4.2"
|
||||
ripple-bs58check latest
|
||||
ripple-hashes latest
|
||||
ripple-keypairs latest
|
||||
|
||||
Reference in New Issue
Block a user