Merge pull request #1563 from RoboSats/generate-robot-and-alert

Generate Robot and Order
This commit is contained in:
KoalaSat 2024-10-19 20:54:24 +00:00 committed by GitHub
commit 1eac225853
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
10 changed files with 81 additions and 83 deletions

View File

@ -1,27 +1,20 @@
import React, { useContext, useState } from 'react'; import React, { useContext, useState } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Button, Grid, ButtonGroup, Dialog, Box } from '@mui/material'; import { Button, Grid, ButtonGroup } from '@mui/material';
import { useNavigate } from 'react-router-dom'; import { useNavigate } from 'react-router-dom';
import DepthChart from '../../components/Charts/DepthChart'; import DepthChart from '../../components/Charts/DepthChart';
import { NoRobotDialog } from '../../components/Dialogs';
import MakerForm from '../../components/MakerForm';
import BookTable from '../../components/BookTable'; import BookTable from '../../components/BookTable';
// Icons // Icons
import { BarChart, FormatListBulleted, Map } from '@mui/icons-material'; import { BarChart, FormatListBulleted, Map } from '@mui/icons-material';
import { AppContext, type UseAppStoreType } from '../../contexts/AppContext'; import { AppContext, type UseAppStoreType } from '../../contexts/AppContext';
import MapChart from '../../components/Charts/MapChart'; import MapChart from '../../components/Charts/MapChart';
import { GarageContext, type UseGarageStoreType } from '../../contexts/GarageContext';
const BookPage = (): JSX.Element => { const BookPage = (): JSX.Element => {
const { windowSize } = useContext<UseAppStoreType>(AppContext); const { windowSize } = useContext<UseAppStoreType>(AppContext);
const { garage } = useContext<UseGarageStoreType>(GarageContext);
const { t } = useTranslation(); const { t } = useTranslation();
const navigate = useNavigate(); const navigate = useNavigate();
const [view, setView] = useState<'list' | 'depth' | 'map'>('list'); const [view, setView] = useState<'list' | 'depth' | 'map'>('list');
const [openMaker, setOpenMaker] = useState<boolean>(false);
const [openNoRobot, setOpenNoRobot] = useState<boolean>(false);
const doubleView = windowSize.width > 115; const doubleView = windowSize.width > 115;
const width = windowSize.width * 0.9; const width = windowSize.width * 0.9;
@ -29,11 +22,7 @@ const BookPage = (): JSX.Element => {
const chartWidthEm = width - maxBookTableWidth; const chartWidthEm = width - maxBookTableWidth;
const onOrderClicked = function (id: number, shortAlias: string): void { const onOrderClicked = function (id: number, shortAlias: string): void {
if (garage.getSlot()?.hashId) {
navigate(`/order/${shortAlias}/${id}`); navigate(`/order/${shortAlias}/${id}`);
} else {
setOpenNoRobot(true);
}
}; };
const NavButtons = function (): JSX.Element { const NavButtons = function (): JSX.Element {
@ -42,7 +31,7 @@ const BookPage = (): JSX.Element => {
<Button <Button
color='primary' color='primary'
onClick={() => { onClick={() => {
setOpenMaker(true); navigate('/create');
}} }}
> >
{t('Create')} {t('Create')}
@ -80,32 +69,6 @@ const BookPage = (): JSX.Element => {
return ( return (
<Grid container direction='column' alignItems='center' spacing={1} sx={{ minWidth: 400 }}> <Grid container direction='column' alignItems='center' spacing={1} sx={{ minWidth: 400 }}>
<NoRobotDialog
open={openNoRobot}
onClose={() => {
setOpenNoRobot(false);
}}
onClickGenerateRobot={() => {
navigate('/garage');
}}
/>
{openMaker ? (
<Dialog
open={openMaker}
onClose={() => {
setOpenMaker(false);
}}
>
<Box sx={{ maxWidth: '18em', padding: '0.5em' }}>
<MakerForm
onClickGenerateRobot={() => {
navigate('/garage');
}}
/>
</Box>
</Dialog>
) : null}
<Grid item xs={12}> <Grid item xs={12}>
{doubleView ? ( {doubleView ? (
<Grid <Grid

View File

@ -2,7 +2,7 @@ import React, { useContext, useMemo, useState } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { useNavigate } from 'react-router-dom'; import { useNavigate } from 'react-router-dom';
import { Grid, Paper, Collapse, Typography } from '@mui/material'; import { Grid, Paper, Collapse, Typography } from '@mui/material';
import { filterOrders } from '../../utils'; import { filterOrders, genBase62Token } from '../../utils';
import MakerForm from '../../components/MakerForm'; import MakerForm from '../../components/MakerForm';
import BookTable from '../../components/BookTable'; import BookTable from '../../components/BookTable';
@ -22,6 +22,7 @@ const MakerPage = (): JSX.Element => {
const maxHeight = (windowSize.height - navbarHeight) * 0.85 - 3; const maxHeight = (windowSize.height - navbarHeight) * 0.85 - 3;
const [showMatches, setShowMatches] = useState<boolean>(false); const [showMatches, setShowMatches] = useState<boolean>(false);
const [openNoRobot, setOpenNoRobot] = useState<boolean>(false); const [openNoRobot, setOpenNoRobot] = useState<boolean>(false);
const [clickedOrder, setClickedOrder] = useState<{ id: number; shortAlias: string }>();
const matches = useMemo(() => { const matches = useMemo(() => {
return filterOrders({ return filterOrders({
@ -55,6 +56,7 @@ const MakerPage = (): JSX.Element => {
if (garage.getSlot()?.hashId) { if (garage.getSlot()?.hashId) {
navigate(`/order/${shortAlias}/${id}`); navigate(`/order/${shortAlias}/${id}`);
} else { } else {
setClickedOrder({ id, shortAlias });
setOpenNoRobot(true); setOpenNoRobot(true);
} }
}; };
@ -67,7 +69,16 @@ const MakerPage = (): JSX.Element => {
setOpenNoRobot(false); setOpenNoRobot(false);
}} }}
onClickGenerateRobot={() => { onClickGenerateRobot={() => {
navigate('/garage'); const token = genBase62Token(36);
garage
.createRobot(federation, token)
.then(() => {
setOpenNoRobot(true);
if (clickedOrder) navigate(`/order/${clickedOrder?.shortAlias}/${clickedOrder?.id}`);
})
.catch((e) => {
console.log(e);
});
}} }}
/> />
<Grid item> <Grid item>
@ -111,9 +122,6 @@ const MakerPage = (): JSX.Element => {
setShowMatches(false); setShowMatches(false);
}} }}
submitButtonLabel={matches.length > 0 && !showMatches ? 'Submit' : 'Create order'} submitButtonLabel={matches.length > 0 && !showMatches ? 'Submit' : 'Create order'}
onClickGenerateRobot={() => {
navigate('/garage');
}}
/> />
</Paper> </Paper>
</Grid> </Grid>

View File

@ -8,9 +8,10 @@ import OrderDetails from '../../components/OrderDetails';
import { AppContext, type UseAppStoreType } from '../../contexts/AppContext'; import { AppContext, type UseAppStoreType } from '../../contexts/AppContext';
import { FederationContext, type UseFederationStoreType } from '../../contexts/FederationContext'; import { FederationContext, type UseFederationStoreType } from '../../contexts/FederationContext';
import { WarningDialog } from '../../components/Dialogs'; import { NoRobotDialog, WarningDialog } from '../../components/Dialogs';
import { Order, type Slot } from '../../models'; import { Order, type Slot } from '../../models';
import { type UseGarageStoreType, GarageContext } from '../../contexts/GarageContext'; import { type UseGarageStoreType, GarageContext } from '../../contexts/GarageContext';
import { genBase62Token } from '../../utils';
const OrderPage = (): JSX.Element => { const OrderPage = (): JSX.Element => {
const { windowSize, setOpen, acknowledgedWarning, setAcknowledgedWarning, navbarHeight } = const { windowSize, setOpen, acknowledgedWarning, setAcknowledgedWarning, navbarHeight } =
@ -27,6 +28,7 @@ const OrderPage = (): JSX.Element => {
const [tab, setTab] = useState<'order' | 'contract'>('contract'); const [tab, setTab] = useState<'order' | 'contract'>('contract');
const [currentOrder, setCurrentOrder] = useState<Order | null>(null); const [currentOrder, setCurrentOrder] = useState<Order | null>(null);
const [openNoRobot, setOpenNoRobot] = useState<boolean>(false);
useEffect(() => { useEffect(() => {
paramsRef.current = params; paramsRef.current = params;
@ -43,12 +45,14 @@ const OrderPage = (): JSX.Element => {
void order.fecth(federation, slot).then((updatedOrder) => { void order.fecth(federation, slot).then((updatedOrder) => {
updateSlotFromOrder(updatedOrder, slot); updateSlotFromOrder(updatedOrder, slot);
}); });
} else {
setOpenNoRobot(true);
} }
return () => { return () => {
setCurrentOrder(null); setCurrentOrder(null);
}; };
}, [params.orderId]); }, [params.orderId, openNoRobot]);
const updateSlotFromOrder = (updatedOrder: Order, slot: Slot): void => { const updateSlotFromOrder = (updatedOrder: Order, slot: Slot): void => {
if ( if (
@ -77,9 +81,6 @@ const OrderPage = (): JSX.Element => {
shortAlias={String(currentOrder.shortAlias)} shortAlias={String(currentOrder.shortAlias)}
currentOrder={currentOrder} currentOrder={currentOrder}
onClickCoordinator={onClickCoordinator} onClickCoordinator={onClickCoordinator}
onClickGenerateRobot={() => {
navigate('/garage');
}}
/> />
) : ( ) : (
<></> <></>
@ -100,6 +101,23 @@ const OrderPage = (): JSX.Element => {
}} }}
longAlias={federation.getCoordinator(params.shortAlias ?? '')?.longAlias} longAlias={federation.getCoordinator(params.shortAlias ?? '')?.longAlias}
/> />
<NoRobotDialog
open={openNoRobot}
onClose={() => {
setOpenNoRobot(false);
}}
onClickGenerateRobot={() => {
const token = genBase62Token(36);
garage
.createRobot(federation, token)
.then(() => {
setOpenNoRobot(false);
})
.catch((e) => {
console.log(e);
});
}}
/>
{!currentOrder?.maker_hash_id && <CircularProgress />} {!currentOrder?.maker_hash_id && <CircularProgress />}
{currentOrder?.bad_request && currentOrder.status !== 5 ? ( {currentOrder?.bad_request && currentOrder.status !== 5 ? (
<Typography align='center' variant='subtitle2' color='secondary'> <Typography align='center' variant='subtitle2' color='secondary'>

View File

@ -116,7 +116,7 @@ const Onboarding = ({ setView, inputToken, setInputToken }: OnboardingProps): JS
<Button <Button
onClick={() => { onClick={() => {
setStep('2'); setStep('2');
garage.createRobot(federation, inputToken); void garage.createRobot(federation, inputToken);
}} }}
variant='contained' variant='contained'
size='large' size='large'

View File

@ -58,7 +58,7 @@ const RobotProfile = ({
const handleAddRobot = (): void => { const handleAddRobot = (): void => {
const token = genBase62Token(36); const token = genBase62Token(36);
garage.createRobot(federation, token); void garage.createRobot(federation, token);
setInputToken(token); setInputToken(token);
setLoading(true); setLoading(true);
}; };

View File

@ -121,7 +121,7 @@ const Welcome = ({ setView, width, setInputToken }: WelcomeProps): JSX.Element =
color='primary' color='primary'
onClick={() => { onClick={() => {
const token = genBase62Token(36); const token = genBase62Token(36);
garage.createRobot(federation, token); void garage.createRobot(federation, token);
setInputToken(token); setInputToken(token);
navigate('/create'); navigate('/create');
setPage('create'); setPage('create');

View File

@ -25,7 +25,7 @@ const RecoveryDialog = ({ setInputToken, setView }: Props): JSX.Element => {
}, [open.recovery]); }, [open.recovery]);
const onClickRecover = (): void => { const onClickRecover = (): void => {
garage.createRobot(federation, recoveryToken); void garage.createRobot(federation, recoveryToken);
setInputToken(recoveryToken.trim()); setInputToken(recoveryToken.trim());
setView('profile'); setView('profile');
setOpen((open) => { setOpen((open) => {

View File

@ -34,7 +34,7 @@ import { FlagWithProps } from '../Icons';
import AutocompletePayments from './AutocompletePayments'; import AutocompletePayments from './AutocompletePayments';
import AmountRange from './AmountRange'; import AmountRange from './AmountRange';
import currencyDict from '../../../static/assets/currencies.json'; import currencyDict from '../../../static/assets/currencies.json';
import { amountToString, computeSats, pn } from '../../utils'; import { amountToString, computeSats, genBase62Token, pn } from '../../utils';
import { SelfImprovement, Lock, HourglassTop, DeleteSweep, Edit, Map } from '@mui/icons-material'; import { SelfImprovement, Lock, HourglassTop, DeleteSweep, Edit, Map } from '@mui/icons-material';
import { LoadingButton } from '@mui/lab'; import { LoadingButton } from '@mui/lab';
@ -52,7 +52,6 @@ interface MakerFormProps {
onSubmit?: () => void; onSubmit?: () => void;
onReset?: () => void; onReset?: () => void;
submitButtonLabel?: string; submitButtonLabel?: string;
onClickGenerateRobot?: () => void;
} }
const MakerForm = ({ const MakerForm = ({
@ -62,7 +61,6 @@ const MakerForm = ({
onSubmit = () => {}, onSubmit = () => {},
onReset = () => {}, onReset = () => {},
submitButtonLabel = 'Create Order', submitButtonLabel = 'Create Order',
onClickGenerateRobot = () => null,
}: MakerFormProps): JSX.Element => { }: MakerFormProps): JSX.Element => {
const { fav, setFav } = useContext<UseAppStoreType>(AppContext); const { fav, setFav } = useContext<UseAppStoreType>(AppContext);
const { federation, federationUpdatedAt } = useContext<UseFederationStoreType>(FederationContext); const { federation, federationUpdatedAt } = useContext<UseFederationStoreType>(FederationContext);
@ -582,7 +580,18 @@ const MakerForm = ({
}} }}
onClickDone={handleCreateOrder} onClickDone={handleCreateOrder}
hasRobot={Boolean(garage.getSlot()?.hashId)} hasRobot={Boolean(garage.getSlot()?.hashId)}
onClickGenerateRobot={onClickGenerateRobot} onClickGenerateRobot={() => {
setOpenDialogs(false);
const token = genBase62Token(36);
garage
.createRobot(federation, token)
.then(() => {
setOpenDialogs(true);
})
.catch((e) => {
console.log(e);
});
}}
/> />
<F2fMapDialog <F2fMapDialog
interactive interactive

View File

@ -2,8 +2,6 @@ import React, { createContext, useEffect, useState, useContext, type ReactNode }
import { Federation, Settings } from '../models'; import { Federation, Settings } from '../models';
import { federationLottery } from '../utils';
import { AppContext, type UseAppStoreType } from './AppContext'; import { AppContext, type UseAppStoreType } from './AppContext';
import { GarageContext, type UseGarageStoreType } from './GarageContext'; import { GarageContext, type UseGarageStoreType } from './GarageContext';
import type Coordinator from '../models/Coordinator.model'; import type Coordinator from '../models/Coordinator.model';

View File

@ -129,12 +129,15 @@ class Garage {
}; };
// Robots // Robots
createRobot: (federation: Federation, token: string) => void = (federation, token) => { createRobot: (federation: Federation, token: string) => Promise<void> = async (
federation,
token,
) => {
if (!token) return; if (!token) return;
if (this.getSlot(token) === null) { if (this.getSlot(token) === null) {
genKey(token) try {
.then((key) => { const key = await genKey(token);
const robotAttributes = { const robotAttributes = {
token, token,
pubKey: key.publicKeyArmored, pubKey: key.publicKeyArmored,
@ -152,10 +155,9 @@ class Garage {
); );
void this.fetchRobot(federation, token); void this.fetchRobot(federation, token);
this.save(); this.save();
}) } catch (error) {
.catch((error) => {
console.error('Error:', error); console.error('Error:', error);
}); }
} }
}; };