2022-11-21 12:56:29 +00:00
|
|
|
import React, { useState, useEffect } from 'react';
|
|
|
|
import { useTranslation } from 'react-i18next';
|
|
|
|
import { Box, Divider, Grid } from '@mui/material';
|
|
|
|
|
|
|
|
import { apiClient } from '../../services/api';
|
|
|
|
import { getWebln, pn } from '../../utils';
|
|
|
|
|
|
|
|
import {
|
|
|
|
ConfirmCancelDialog,
|
|
|
|
ConfirmCollabCancelDialog,
|
|
|
|
ConfirmDisputeDialog,
|
2023-03-09 13:19:24 +00:00
|
|
|
ConfirmFiatSentDialog,
|
2023-04-28 09:19:18 +00:00
|
|
|
ConfirmUndoFiatSentDialog,
|
2022-11-21 12:56:29 +00:00
|
|
|
ConfirmFiatReceivedDialog,
|
|
|
|
WebLNDialog,
|
|
|
|
} from './Dialogs';
|
|
|
|
|
|
|
|
import Title from './Title';
|
|
|
|
import {
|
|
|
|
LockInvoicePrompt,
|
|
|
|
TakerFoundPrompt,
|
|
|
|
PublicWaitPrompt,
|
|
|
|
PausedPrompt,
|
|
|
|
ExpiredPrompt,
|
|
|
|
PayoutPrompt,
|
|
|
|
PayoutWaitPrompt,
|
|
|
|
EscrowWaitPrompt,
|
|
|
|
ChatPrompt,
|
|
|
|
DisputePrompt,
|
|
|
|
DisputeWaitPeerPrompt,
|
|
|
|
DisputeWaitResolutionPrompt,
|
|
|
|
SendingSatsPrompt,
|
|
|
|
SuccessfulPrompt,
|
|
|
|
RoutingFailedPrompt,
|
|
|
|
DisputeWinnerPrompt,
|
|
|
|
DisputeLoserPrompt,
|
|
|
|
} from './Prompts';
|
|
|
|
import BondStatus from './BondStatus';
|
|
|
|
import CancelButton from './CancelButton';
|
|
|
|
import {
|
|
|
|
defaultLightning,
|
|
|
|
LightningForm,
|
|
|
|
defaultOnchain,
|
|
|
|
OnchainForm,
|
|
|
|
DisputeForm,
|
|
|
|
defaultDispute,
|
|
|
|
} from './Forms';
|
|
|
|
|
2023-03-09 13:19:24 +00:00
|
|
|
import { Order, Robot, Settings } from '../../models';
|
2022-11-21 12:56:29 +00:00
|
|
|
import { EncryptedChatMessage } from './EncryptedChat';
|
|
|
|
import CollabCancelAlert from './CollabCancelAlert';
|
|
|
|
import { Bolt } from '@mui/icons-material';
|
2023-04-20 15:19:47 +00:00
|
|
|
import es from 'date-fns/esm/locale/es/index.js';
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
interface loadingButtonsProps {
|
|
|
|
cancel: boolean;
|
|
|
|
fiatSent: boolean;
|
|
|
|
fiatReceived: boolean;
|
|
|
|
submitInvoice: boolean;
|
|
|
|
submitAddress: boolean;
|
|
|
|
submitStatement: boolean;
|
|
|
|
openDispute: boolean;
|
|
|
|
pauseOrder: boolean;
|
|
|
|
renewOrder: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const noLoadingButtons: loadingButtonsProps = {
|
|
|
|
cancel: false,
|
|
|
|
fiatSent: false,
|
|
|
|
fiatReceived: false,
|
|
|
|
submitInvoice: false,
|
|
|
|
submitAddress: false,
|
|
|
|
submitStatement: false,
|
|
|
|
openDispute: false,
|
|
|
|
pauseOrder: false,
|
|
|
|
renewOrder: false,
|
|
|
|
};
|
|
|
|
|
|
|
|
interface OpenDialogProps {
|
|
|
|
confirmCancel: boolean;
|
|
|
|
confirmCollabCancel: boolean;
|
2023-03-09 13:19:24 +00:00
|
|
|
confirmFiatSent: boolean;
|
2023-04-28 09:19:18 +00:00
|
|
|
confirmUndoFiatSent: boolean;
|
2022-11-21 12:56:29 +00:00
|
|
|
confirmFiatReceived: boolean;
|
|
|
|
confirmDispute: boolean;
|
|
|
|
webln: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const closeAll: OpenDialogProps = {
|
|
|
|
confirmCancel: false,
|
|
|
|
confirmCollabCancel: false,
|
2023-03-09 13:19:24 +00:00
|
|
|
confirmFiatSent: false,
|
2023-04-28 09:19:18 +00:00
|
|
|
confirmUndoFiatSent: false,
|
2022-11-21 12:56:29 +00:00
|
|
|
confirmFiatReceived: false,
|
|
|
|
confirmDispute: false,
|
|
|
|
webln: false,
|
|
|
|
};
|
|
|
|
|
|
|
|
interface TradeBoxProps {
|
|
|
|
order: Order;
|
|
|
|
setOrder: (state: Order) => void;
|
2023-03-02 11:01:06 +00:00
|
|
|
robot: Robot;
|
2022-11-21 12:56:29 +00:00
|
|
|
setBadOrder: (state: string | undefined) => void;
|
|
|
|
onRenewOrder: () => void;
|
|
|
|
onStartAgain: () => void;
|
2022-11-24 17:42:30 +00:00
|
|
|
settings: Settings;
|
2022-11-21 12:56:29 +00:00
|
|
|
baseUrl: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
const TradeBox = ({
|
|
|
|
order,
|
|
|
|
setOrder,
|
2023-03-02 11:01:06 +00:00
|
|
|
robot,
|
2022-11-24 17:42:30 +00:00
|
|
|
settings,
|
2022-11-21 12:56:29 +00:00
|
|
|
baseUrl,
|
|
|
|
setBadOrder,
|
|
|
|
onRenewOrder,
|
|
|
|
onStartAgain,
|
|
|
|
}: TradeBoxProps): JSX.Element => {
|
|
|
|
const { t } = useTranslation();
|
|
|
|
|
|
|
|
// Buttons and Dialogs
|
|
|
|
const [loadingButtons, setLoadingButtons] = useState<loadingButtonsProps>(noLoadingButtons);
|
|
|
|
const [open, setOpen] = useState<OpenDialogProps>(closeAll);
|
|
|
|
const [waitingWebln, setWaitingWebln] = useState<boolean>(false);
|
|
|
|
const [lastOrderStatus, setLastOrderStatus] = useState<number>(-1);
|
|
|
|
|
|
|
|
// Forms
|
|
|
|
const [onchain, setOnchain] = useState<OnchainForm>(defaultOnchain);
|
|
|
|
const [lightning, setLightning] = useState<LightningForm>(defaultLightning);
|
|
|
|
const [dispute, setDispute] = useState<DisputeForm>(defaultDispute);
|
|
|
|
|
|
|
|
// Chat
|
|
|
|
const [messages, setMessages] = useState<EncryptedChatMessage[]>([]);
|
|
|
|
|
|
|
|
interface SubmitActionProps {
|
|
|
|
action:
|
|
|
|
| 'cancel'
|
|
|
|
| 'dispute'
|
|
|
|
| 'pause'
|
|
|
|
| 'confirm'
|
2023-04-28 09:19:18 +00:00
|
|
|
| 'undo_confirm'
|
2022-11-21 12:56:29 +00:00
|
|
|
| 'update_invoice'
|
|
|
|
| 'update_address'
|
|
|
|
| 'submit_statement'
|
|
|
|
| 'rate_platform';
|
|
|
|
invoice?: string;
|
2022-11-24 17:42:30 +00:00
|
|
|
routing_budget_ppm?: number;
|
2022-11-21 12:56:29 +00:00
|
|
|
address?: string;
|
|
|
|
mining_fee_rate?: number;
|
|
|
|
statement?: string;
|
|
|
|
rating?: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
const submitAction = function ({
|
|
|
|
action,
|
|
|
|
invoice,
|
2022-11-24 17:42:30 +00:00
|
|
|
routing_budget_ppm,
|
2022-11-21 12:56:29 +00:00
|
|
|
address,
|
|
|
|
mining_fee_rate,
|
|
|
|
statement,
|
|
|
|
rating,
|
|
|
|
}: SubmitActionProps) {
|
|
|
|
apiClient
|
|
|
|
.post(baseUrl, '/api/order/?order_id=' + order.id, {
|
|
|
|
action,
|
|
|
|
invoice,
|
2022-11-24 17:42:30 +00:00
|
|
|
routing_budget_ppm,
|
2022-11-21 12:56:29 +00:00
|
|
|
address,
|
|
|
|
mining_fee_rate,
|
|
|
|
statement,
|
|
|
|
rating,
|
|
|
|
})
|
|
|
|
.catch(() => {
|
|
|
|
setOpen(closeAll);
|
|
|
|
setLoadingButtons({ ...noLoadingButtons });
|
|
|
|
})
|
|
|
|
.then((data: Order) => {
|
|
|
|
setOpen(closeAll);
|
|
|
|
setLoadingButtons({ ...noLoadingButtons });
|
|
|
|
if (data.bad_request) {
|
|
|
|
setBadOrder(data.bad_request);
|
|
|
|
} else if (data.bad_address) {
|
|
|
|
setOnchain({ ...onchain, badAddress: data.bad_address });
|
|
|
|
} else if (data.bad_invoice) {
|
|
|
|
setLightning({ ...lightning, badInvoice: data.bad_invoice });
|
|
|
|
} else if (data.bad_statement) {
|
|
|
|
setDispute({ ...dispute, badStatement: data.bad_statement });
|
|
|
|
} else {
|
|
|
|
setOrder({ ...order, ...data });
|
|
|
|
setBadOrder(undefined);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const cancel = function () {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, cancel: true });
|
|
|
|
submitAction({ action: 'cancel' });
|
|
|
|
};
|
|
|
|
|
|
|
|
const openDispute = function () {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, openDispute: true });
|
|
|
|
submitAction({ action: 'dispute' });
|
|
|
|
};
|
|
|
|
|
|
|
|
const confirmFiatReceived = function () {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, fiatReceived: true });
|
|
|
|
submitAction({ action: 'confirm' });
|
|
|
|
};
|
|
|
|
|
|
|
|
const confirmFiatSent = function () {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, fiatSent: true });
|
|
|
|
submitAction({ action: 'confirm' });
|
|
|
|
};
|
|
|
|
|
2023-04-28 09:19:18 +00:00
|
|
|
const confirmUndoFiatSent = function () {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, undoFiatSent: true });
|
|
|
|
submitAction({ action: 'undo_confirm' });
|
|
|
|
};
|
|
|
|
|
2022-11-21 12:56:29 +00:00
|
|
|
const updateInvoice = function (invoice: string) {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, submitInvoice: true });
|
2022-11-24 17:42:30 +00:00
|
|
|
submitAction({
|
|
|
|
action: 'update_invoice',
|
|
|
|
invoice,
|
|
|
|
routing_budget_ppm: lightning.routingBudgetPPM,
|
|
|
|
});
|
2022-11-21 12:56:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const updateAddress = function () {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, submitAddress: true });
|
|
|
|
submitAction({
|
|
|
|
action: 'update_address',
|
|
|
|
address: onchain.address,
|
|
|
|
mining_fee_rate: onchain.miningFee,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const pauseOrder = function () {
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, pauseOrder: true });
|
|
|
|
submitAction({ action: 'pause' });
|
|
|
|
};
|
|
|
|
|
|
|
|
const submitStatement = function () {
|
|
|
|
let statement = dispute.statement;
|
|
|
|
if (dispute.attachLogs) {
|
2023-03-02 11:01:06 +00:00
|
|
|
const payload = { statement, messages, token: robot.token };
|
2022-11-21 12:56:29 +00:00
|
|
|
statement = JSON.stringify(payload, null, 2);
|
|
|
|
}
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, submitStatement: true });
|
|
|
|
submitAction({ action: 'submit_statement', statement });
|
|
|
|
};
|
|
|
|
const ratePlatform = function (rating: number) {
|
|
|
|
submitAction({ action: 'rate_platform', rating });
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleWebln = async (order: Order) => {
|
|
|
|
const webln = await getWebln().catch(() => console.log('WebLN not available'));
|
|
|
|
// If Webln implements locked payments compatibility, this logic might be simplier
|
|
|
|
if (webln == undefined) {
|
|
|
|
return null;
|
|
|
|
} else if (order.is_maker && order.status == 0) {
|
|
|
|
webln.sendPayment(order.bond_invoice);
|
|
|
|
setWaitingWebln(true);
|
|
|
|
setOpen({ ...open, webln: true });
|
|
|
|
} else if (order.is_taker && order.status == 3) {
|
|
|
|
webln.sendPayment(order.bond_invoice);
|
|
|
|
setWaitingWebln(true);
|
|
|
|
setOpen({ ...open, webln: true });
|
|
|
|
} else if (order.is_seller && (order.status == 6 || order.status == 7)) {
|
|
|
|
webln.sendPayment(order.escrow_invoice);
|
|
|
|
setWaitingWebln(true);
|
|
|
|
setOpen({ ...open, webln: true });
|
|
|
|
} else if (order.is_buyer && (order.status == 6 || order.status == 8)) {
|
|
|
|
setWaitingWebln(true);
|
|
|
|
setOpen({ ...open, webln: true });
|
|
|
|
webln
|
2022-11-24 17:42:30 +00:00
|
|
|
.makeInvoice(() => lightning.amount)
|
2022-11-21 12:56:29 +00:00
|
|
|
.then((invoice: any) => {
|
|
|
|
if (invoice) {
|
|
|
|
updateInvoice(invoice.paymentRequest);
|
|
|
|
setWaitingWebln(false);
|
|
|
|
setOpen(closeAll);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch(() => {
|
|
|
|
setWaitingWebln(false);
|
|
|
|
setOpen(closeAll);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
setWaitingWebln(false);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Effect on Order Status change (used for WebLN)
|
|
|
|
useEffect(() => {
|
|
|
|
if (order.status != lastOrderStatus) {
|
|
|
|
setLastOrderStatus(order.status);
|
|
|
|
handleWebln(order);
|
|
|
|
}
|
|
|
|
}, [order.status]);
|
|
|
|
|
|
|
|
const statusToContract = function (order: Order) {
|
2022-11-27 20:34:37 +00:00
|
|
|
const status = order.status;
|
2022-11-21 12:56:29 +00:00
|
|
|
const isBuyer = order.is_buyer;
|
|
|
|
const isMaker = order.is_maker;
|
|
|
|
|
|
|
|
let title: string = 'Unknown Order Status';
|
|
|
|
let titleVariables: object = {};
|
|
|
|
let titleColor: string = 'primary';
|
|
|
|
let titleIcon: () => JSX.Element = function () {
|
|
|
|
return <></>;
|
|
|
|
};
|
|
|
|
let prompt = () => <span>Wops!</span>;
|
|
|
|
let bondStatus: 'hide' | 'locked' | 'unlocked' | 'settled' = 'hide';
|
|
|
|
|
2023-04-20 15:19:47 +00:00
|
|
|
switch (status) {
|
|
|
|
// 0: 'Waiting for maker bond'
|
|
|
|
case 0:
|
|
|
|
if (isMaker) {
|
|
|
|
title = 'Lock {{amountSats}} Sats to PUBLISH order';
|
|
|
|
titleVariables = { amountSats: pn(order.bond_satoshis) };
|
|
|
|
prompt = () => {
|
|
|
|
return <LockInvoicePrompt order={order} concept={'bond'} />;
|
|
|
|
};
|
|
|
|
bondStatus = 'hide';
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
// 1: 'Public'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 1:
|
|
|
|
if (isMaker) {
|
|
|
|
title = 'Your order is public';
|
|
|
|
prompt = () => {
|
|
|
|
return (
|
|
|
|
<PublicWaitPrompt
|
|
|
|
order={order}
|
|
|
|
pauseLoading={loadingButtons.pauseOrder}
|
|
|
|
onClickPauseOrder={pauseOrder}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
bondStatus = 'locked';
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
// 2: 'Paused'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 2:
|
|
|
|
if (isMaker) {
|
|
|
|
title = 'Your order is paused';
|
|
|
|
prompt = () => {
|
|
|
|
return (
|
|
|
|
<PausedPrompt
|
|
|
|
pauseLoading={loadingButtons.pauseOrder}
|
|
|
|
onClickResumeOrder={pauseOrder}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
bondStatus = 'locked';
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 3: 'Waiting for taker bond'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 3:
|
|
|
|
if (isMaker) {
|
|
|
|
title = 'A taker has been found!';
|
|
|
|
prompt = () => {
|
|
|
|
return <TakerFoundPrompt />;
|
|
|
|
};
|
|
|
|
bondStatus = 'locked';
|
|
|
|
} else {
|
|
|
|
title = 'Lock {{amountSats}} Sats to TAKE order';
|
|
|
|
titleVariables = { amountSats: pn(order.bond_satoshis) };
|
|
|
|
prompt = () => {
|
|
|
|
return <LockInvoicePrompt order={order} concept={'bond'} />;
|
|
|
|
};
|
|
|
|
bondStatus = 'hide';
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 5: 'Expired'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 5:
|
|
|
|
title = 'The order has expired';
|
|
|
|
prompt = () => {
|
2022-11-21 12:56:29 +00:00
|
|
|
return (
|
2023-04-20 15:19:47 +00:00
|
|
|
<ExpiredPrompt
|
|
|
|
loadingRenew={loadingButtons.renewOrder}
|
2022-11-21 12:56:29 +00:00
|
|
|
order={order}
|
2023-04-20 15:19:47 +00:00
|
|
|
onClickRenew={() => {
|
|
|
|
onRenewOrder();
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, renewOrder: true });
|
|
|
|
}}
|
2022-11-21 12:56:29 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
2023-04-20 15:19:47 +00:00
|
|
|
bondStatus = 'hide'; // To do: show bond status according to expiry message.
|
2023-04-21 11:10:47 +00:00
|
|
|
break;
|
2023-04-20 15:19:47 +00:00
|
|
|
|
|
|
|
// 6: 'Waiting for trade collateral and buyer invoice'
|
|
|
|
case 6:
|
|
|
|
bondStatus = 'locked';
|
|
|
|
if (isBuyer) {
|
|
|
|
title = 'Submit payout info';
|
|
|
|
titleVariables = { amountSats: pn(order.invoice_amount) };
|
|
|
|
prompt = function () {
|
|
|
|
return (
|
|
|
|
<PayoutPrompt
|
|
|
|
order={order}
|
|
|
|
settings={settings}
|
|
|
|
onClickSubmitInvoice={updateInvoice}
|
|
|
|
loadingLightning={loadingButtons.submitInvoice}
|
|
|
|
lightning={lightning}
|
|
|
|
setLightning={setLightning}
|
|
|
|
onClickSubmitAddress={updateAddress}
|
|
|
|
loadingOnchain={loadingButtons.submitAddress}
|
|
|
|
onchain={onchain}
|
|
|
|
setOnchain={setOnchain}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
title = 'Lock {{amountSats}} Sats as collateral';
|
|
|
|
titleVariables = { amountSats: pn(order.escrow_satoshis) };
|
|
|
|
titleColor = 'warning';
|
|
|
|
prompt = () => {
|
|
|
|
return <LockInvoicePrompt order={order} concept={'escrow'} />;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 7: 'Waiting only for seller trade collateral'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 7:
|
|
|
|
bondStatus = 'locked';
|
|
|
|
if (isBuyer) {
|
|
|
|
title = 'Your info looks good!';
|
|
|
|
prompt = () => {
|
|
|
|
return <EscrowWaitPrompt />;
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
title = 'Lock {{amountSats}} Sats as collateral';
|
|
|
|
titleVariables = { amountSats: pn(order.escrow_satoshis) };
|
|
|
|
titleColor = 'warning';
|
|
|
|
prompt = () => {
|
|
|
|
return <LockInvoicePrompt order={order} concept={'escrow'} />;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 8: 'Waiting only for buyer invoice'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 8:
|
|
|
|
bondStatus = 'locked';
|
|
|
|
if (isBuyer) {
|
|
|
|
title = 'Submit payout info';
|
|
|
|
titleVariables = { amountSats: pn(order.invoice_amount) };
|
|
|
|
prompt = () => {
|
|
|
|
return (
|
|
|
|
<PayoutPrompt
|
|
|
|
order={order}
|
|
|
|
settings={settings}
|
|
|
|
onClickSubmitInvoice={updateInvoice}
|
|
|
|
loadingLightning={loadingButtons.submitInvoice}
|
|
|
|
lightning={lightning}
|
|
|
|
setLightning={setLightning}
|
|
|
|
onClickSubmitAddress={updateAddress}
|
|
|
|
loadingOnchain={loadingButtons.submitAddress}
|
|
|
|
onchain={onchain}
|
|
|
|
setOnchain={setOnchain}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
title = 'The trade collateral is locked!';
|
|
|
|
prompt = () => {
|
|
|
|
return <PayoutWaitPrompt />;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 9: 'Sending fiat - In chatroom'
|
|
|
|
// 10: 'Fiat sent - In chatroom'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 9:
|
|
|
|
case 10:
|
|
|
|
title = isBuyer ? 'Chat with the seller' : 'Chat with the buyer';
|
2022-11-21 12:56:29 +00:00
|
|
|
prompt = function () {
|
|
|
|
return (
|
2023-04-20 15:19:47 +00:00
|
|
|
<ChatPrompt
|
|
|
|
order={order}
|
|
|
|
robot={robot}
|
|
|
|
onClickConfirmSent={() => setOpen({ ...open, confirmFiatSent: true })}
|
2023-04-28 09:19:18 +00:00
|
|
|
onClickUndoConfirmSent={() => setOpen({ ...open, confirmUndoFiatSent: true })}
|
2023-04-20 15:19:47 +00:00
|
|
|
onClickConfirmReceived={() => setOpen({ ...open, confirmFiatReceived: true })}
|
|
|
|
loadingSent={loadingButtons.fiatSent}
|
2023-04-28 09:19:18 +00:00
|
|
|
loadingUndoSent={loadingButtons.undoFiatSent}
|
2023-04-20 15:19:47 +00:00
|
|
|
loadingReceived={loadingButtons.fiatReceived}
|
|
|
|
onClickDispute={() => setOpen({ ...open, confirmDispute: true })}
|
|
|
|
loadingDispute={loadingButtons.openDispute}
|
|
|
|
baseUrl={baseUrl}
|
|
|
|
messages={messages}
|
|
|
|
setMessages={setMessages}
|
2022-11-21 12:56:29 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
2023-04-20 15:19:47 +00:00
|
|
|
bondStatus = 'locked';
|
|
|
|
break;
|
|
|
|
|
|
|
|
// 11: 'In dispute'
|
|
|
|
case 11:
|
|
|
|
bondStatus = 'settled';
|
|
|
|
if (order.statement_submitted) {
|
|
|
|
title = 'We have received your statement';
|
|
|
|
prompt = function () {
|
|
|
|
return <DisputeWaitPeerPrompt />;
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
title = 'A dispute has been opened';
|
|
|
|
prompt = function () {
|
|
|
|
return (
|
|
|
|
<DisputePrompt
|
|
|
|
loading={loadingButtons.submitStatement}
|
|
|
|
dispute={dispute}
|
|
|
|
setDispute={setDispute}
|
|
|
|
onClickSubmit={submitStatement}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-11-21 12:56:29 +00:00
|
|
|
// 12: 'Collaboratively cancelled'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 12:
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
// 13: 'Sending satoshis to buyer'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 13:
|
|
|
|
if (isBuyer) {
|
|
|
|
bondStatus = 'unlocked';
|
|
|
|
title = 'Attempting Lightning Payment';
|
|
|
|
prompt = function () {
|
|
|
|
return <SendingSatsPrompt />;
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
title = 'Trade finished!';
|
|
|
|
titleColor = 'success';
|
|
|
|
titleIcon = function () {
|
|
|
|
return <Bolt xs={{ width: '1em', height: '1em' }} color='warning' />;
|
|
|
|
};
|
|
|
|
prompt = function () {
|
|
|
|
return (
|
|
|
|
<SuccessfulPrompt
|
|
|
|
baseUrl={baseUrl}
|
|
|
|
order={order}
|
|
|
|
ratePlatform={ratePlatform}
|
|
|
|
onClickStartAgain={onStartAgain}
|
|
|
|
loadingRenew={loadingButtons.renewOrder}
|
|
|
|
onClickRenew={() => {
|
|
|
|
onRenewOrder();
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, renewOrder: true });
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 14: 'Sucessful trade'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 14:
|
2022-11-21 12:56:29 +00:00
|
|
|
title = 'Trade finished!';
|
|
|
|
titleColor = 'success';
|
|
|
|
titleIcon = function () {
|
|
|
|
return <Bolt xs={{ width: '1em', height: '1em' }} color='warning' />;
|
|
|
|
};
|
|
|
|
prompt = function () {
|
|
|
|
return (
|
|
|
|
<SuccessfulPrompt
|
|
|
|
baseUrl={baseUrl}
|
|
|
|
order={order}
|
|
|
|
ratePlatform={ratePlatform}
|
|
|
|
onClickStartAgain={onStartAgain}
|
|
|
|
loadingRenew={loadingButtons.renewOrder}
|
|
|
|
onClickRenew={() => {
|
|
|
|
onRenewOrder();
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, renewOrder: true });
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
2023-04-20 15:19:47 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
// 15: 'Failed lightning network routing'
|
|
|
|
case 15:
|
|
|
|
if (isBuyer) {
|
|
|
|
bondStatus = 'unlocked';
|
|
|
|
title = 'Lightning Routing Failed';
|
|
|
|
prompt = function () {
|
|
|
|
return (
|
|
|
|
<RoutingFailedPrompt
|
|
|
|
order={order}
|
|
|
|
settings={settings}
|
|
|
|
onClickSubmitInvoice={updateInvoice}
|
|
|
|
loadingLightning={loadingButtons.submitInvoice}
|
|
|
|
lightning={lightning}
|
|
|
|
setLightning={setLightning}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
title = 'Trade finished!';
|
|
|
|
titleColor = 'success';
|
|
|
|
titleIcon = function () {
|
|
|
|
return <Bolt xs={{ width: '1em', height: '1em' }} color='warning' />;
|
|
|
|
};
|
|
|
|
prompt = function () {
|
|
|
|
return (
|
|
|
|
<SuccessfulPrompt
|
|
|
|
baseUrl={baseUrl}
|
|
|
|
order={order}
|
|
|
|
ratePlatform={ratePlatform}
|
|
|
|
onClickStartAgain={onStartAgain}
|
|
|
|
loadingRenew={loadingButtons.renewOrder}
|
|
|
|
onClickRenew={() => {
|
|
|
|
onRenewOrder();
|
|
|
|
setLoadingButtons({ ...noLoadingButtons, renewOrder: true });
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 16: 'Wait for dispute resolution'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 16:
|
|
|
|
bondStatus = 'settled';
|
|
|
|
title = 'We have the statements';
|
|
|
|
prompt = function () {
|
|
|
|
return <DisputeWaitResolutionPrompt />;
|
|
|
|
};
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
|
|
|
|
// 17: 'Maker lost dispute'
|
|
|
|
// 18: 'Taker lost dispute'
|
2023-04-20 15:19:47 +00:00
|
|
|
case 17:
|
|
|
|
case 18:
|
|
|
|
if ((status === 17 && isMaker) || (status === 18 && !isMaker)) {
|
|
|
|
title = 'You have lost the dispute';
|
|
|
|
prompt = function () {
|
|
|
|
return <DisputeLoserPrompt />;
|
|
|
|
};
|
|
|
|
bondStatus = 'settled';
|
|
|
|
} else if ((status === 17 && !isMaker) || (status === 18 && isMaker)) {
|
|
|
|
title = 'You have won the dispute';
|
|
|
|
prompt = function () {
|
|
|
|
return <DisputeWinnerPrompt />;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
2022-11-21 12:56:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return { title, titleVariables, titleColor, prompt, bondStatus, titleIcon };
|
|
|
|
};
|
|
|
|
|
|
|
|
const contract = statusToContract(order);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Box>
|
|
|
|
<WebLNDialog
|
|
|
|
open={open.webln}
|
|
|
|
onClose={() => setOpen(closeAll)}
|
|
|
|
waitingWebln={waitingWebln}
|
|
|
|
isBuyer={order.is_buyer}
|
|
|
|
/>
|
|
|
|
<ConfirmDisputeDialog
|
|
|
|
open={open.confirmDispute}
|
|
|
|
onClose={() => setOpen(closeAll)}
|
|
|
|
onAgreeClick={openDispute}
|
|
|
|
/>
|
|
|
|
<ConfirmCancelDialog
|
|
|
|
open={open.confirmCancel}
|
|
|
|
onClose={() => setOpen(closeAll)}
|
|
|
|
onCancelClick={cancel}
|
|
|
|
/>
|
|
|
|
<ConfirmCollabCancelDialog
|
|
|
|
open={open.confirmCollabCancel}
|
|
|
|
onClose={() => setOpen(closeAll)}
|
|
|
|
onCollabCancelClick={cancel}
|
|
|
|
loading={loadingButtons.cancel}
|
|
|
|
peerAskedCancel={order.pending_cancel}
|
|
|
|
/>
|
2023-03-09 13:19:24 +00:00
|
|
|
<ConfirmFiatSentDialog
|
|
|
|
open={open.confirmFiatSent}
|
|
|
|
order={order}
|
|
|
|
loadingButton={loadingButtons.fiatSent}
|
|
|
|
onClose={() => setOpen(closeAll)}
|
|
|
|
onConfirmClick={confirmFiatSent}
|
|
|
|
/>
|
2023-04-28 09:19:18 +00:00
|
|
|
<ConfirmUndoFiatSentDialog
|
|
|
|
open={open.confirmUndoFiatSent}
|
|
|
|
order={order}
|
|
|
|
loadingButton={loadingButtons.undoFiatSent}
|
|
|
|
onClose={() => setOpen(closeAll)}
|
|
|
|
onConfirmClick={confirmUndoFiatSent}
|
|
|
|
/>
|
2022-11-21 12:56:29 +00:00
|
|
|
<ConfirmFiatReceivedDialog
|
|
|
|
open={open.confirmFiatReceived}
|
|
|
|
order={order}
|
|
|
|
loadingButton={loadingButtons.fiatReceived}
|
|
|
|
onClose={() => setOpen(closeAll)}
|
|
|
|
onConfirmClick={confirmFiatReceived}
|
|
|
|
/>
|
|
|
|
<CollabCancelAlert order={order} />
|
|
|
|
<Grid
|
|
|
|
container
|
|
|
|
padding={1}
|
|
|
|
direction='column'
|
|
|
|
justifyContent='flex-start'
|
|
|
|
alignItems='center'
|
|
|
|
spacing={0}
|
|
|
|
>
|
|
|
|
<Grid item>
|
|
|
|
<Title
|
|
|
|
order={order}
|
|
|
|
text={contract.title}
|
|
|
|
color={contract.titleColor}
|
|
|
|
icon={contract.titleIcon}
|
|
|
|
variables={contract.titleVariables}
|
|
|
|
/>
|
|
|
|
</Grid>
|
|
|
|
<Divider />
|
|
|
|
|
|
|
|
<Grid item>{contract.prompt()}</Grid>
|
|
|
|
|
|
|
|
{contract.bondStatus != 'hide' ? (
|
|
|
|
<Grid item sx={{ width: '100%' }}>
|
|
|
|
<Divider />
|
|
|
|
<BondStatus status={contract.bondStatus} isMaker={order.is_maker} />
|
|
|
|
</Grid>
|
|
|
|
) : (
|
|
|
|
<></>
|
|
|
|
)}
|
|
|
|
|
|
|
|
<Grid item>
|
|
|
|
<CancelButton
|
|
|
|
order={order}
|
|
|
|
onClickCancel={cancel}
|
|
|
|
openCancelDialog={() => setOpen({ ...closeAll, confirmCancel: true })}
|
|
|
|
openCollabCancelDialog={() => setOpen({ ...closeAll, confirmCollabCancel: true })}
|
|
|
|
loading={loadingButtons.cancel}
|
|
|
|
/>
|
|
|
|
</Grid>
|
|
|
|
</Grid>
|
|
|
|
</Box>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default TradeBox;
|