2022-01-02 00:19:18 +00:00
|
|
|
import React, { Component } from "react";
|
2022-01-18 15:23:57 +00:00
|
|
|
import { Alert, Paper, CircularProgress, Button , Grid, Typography, List, ListItem, ListItemIcon, ListItemText, ListItemAvatar, Avatar, Divider, Box, LinearProgress, Dialog, DialogActions, DialogContent, DialogContentText, DialogTitle} from "@mui/material"
|
2022-01-14 00:43:26 +00:00
|
|
|
import Countdown, { zeroPad, calcTimeDelta } from 'react-countdown';
|
2022-01-08 13:08:03 +00:00
|
|
|
import TradeBox from "./TradeBox";
|
2022-01-16 18:32:34 +00:00
|
|
|
import getFlags from './getFlags'
|
2022-01-02 00:19:18 +00:00
|
|
|
|
2022-01-14 12:00:53 +00:00
|
|
|
// icons
|
|
|
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
|
|
|
import NumbersIcon from '@mui/icons-material/Numbers';
|
|
|
|
import PriceChangeIcon from '@mui/icons-material/PriceChange';
|
|
|
|
import PaymentsIcon from '@mui/icons-material/Payments';
|
|
|
|
import MoneyIcon from '@mui/icons-material/Money';
|
|
|
|
import ArticleIcon from '@mui/icons-material/Article';
|
2022-01-18 15:23:57 +00:00
|
|
|
import ContentCopy from "@mui/icons-material/ContentCopy";
|
2022-01-08 15:34:09 +00:00
|
|
|
|
2022-01-05 00:13:08 +00:00
|
|
|
function getCookie(name) {
|
|
|
|
let cookieValue = null;
|
|
|
|
if (document.cookie && document.cookie !== '') {
|
|
|
|
const cookies = document.cookie.split(';');
|
|
|
|
for (let i = 0; i < cookies.length; i++) {
|
|
|
|
const cookie = cookies[i].trim();
|
|
|
|
// Does this cookie string begin with the name we want?
|
|
|
|
if (cookie.substring(0, name.length + 1) === (name + '=')) {
|
|
|
|
cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return cookieValue;
|
|
|
|
}
|
|
|
|
const csrftoken = getCookie('csrftoken');
|
|
|
|
|
2022-01-03 14:27:25 +00:00
|
|
|
// pretty numbers
|
|
|
|
function pn(x) {
|
|
|
|
return x.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ",");
|
|
|
|
}
|
|
|
|
|
2022-01-02 00:19:18 +00:00
|
|
|
export default class OrderPage extends Component {
|
2022-01-02 12:59:48 +00:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
this.state = {
|
2022-01-02 13:35:31 +00:00
|
|
|
isExplicit: false,
|
2022-01-15 12:00:11 +00:00
|
|
|
delay: 60000, // Refresh every 60 seconds by default
|
2022-01-14 00:43:26 +00:00
|
|
|
currencies_dict: {"1":"USD"},
|
|
|
|
total_secs_expiry: 300,
|
2022-01-14 14:19:25 +00:00
|
|
|
loading: true,
|
2022-01-18 15:23:57 +00:00
|
|
|
openCancel: false,
|
2022-01-02 12:59:48 +00:00
|
|
|
};
|
|
|
|
this.orderId = this.props.match.params.orderId;
|
2022-01-08 12:48:03 +00:00
|
|
|
this.getCurrencyDict();
|
2022-01-02 13:24:35 +00:00
|
|
|
this.getOrderDetails();
|
2022-01-15 12:00:11 +00:00
|
|
|
|
2022-01-15 15:12:26 +00:00
|
|
|
// Refresh delais according to Order status
|
2022-01-15 12:00:11 +00:00
|
|
|
this.statusToDelay = {
|
|
|
|
"0": 3000, //'Waiting for maker bond'
|
|
|
|
"1": 30000, //'Public'
|
2022-01-15 14:22:07 +00:00
|
|
|
"2": 9999999, //'Deleted'
|
2022-01-15 12:00:11 +00:00
|
|
|
"3": 3000, //'Waiting for taker bond'
|
2022-01-15 14:22:07 +00:00
|
|
|
"4": 9999999, //'Cancelled'
|
2022-01-15 12:00:11 +00:00
|
|
|
"5": 999999, //'Expired'
|
|
|
|
"6": 3000, //'Waiting for trade collateral and buyer invoice'
|
|
|
|
"7": 3000, //'Waiting only for seller trade collateral'
|
|
|
|
"8": 10000, //'Waiting only for buyer invoice'
|
|
|
|
"9": 10000, //'Sending fiat - In chatroom'
|
|
|
|
"10": 15000, //'Fiat sent - In chatroom'
|
2022-01-17 23:11:41 +00:00
|
|
|
"11": 60000, //'In dispute'
|
2022-01-15 14:22:07 +00:00
|
|
|
"12": 9999999,//'Collaboratively cancelled'
|
2022-01-18 13:20:19 +00:00
|
|
|
"13": 3000, //'Sending satoshis to buyer'
|
2022-01-15 14:22:07 +00:00
|
|
|
"14": 9999999,//'Sucessful trade'
|
|
|
|
"15": 10000, //'Failed lightning network routing'
|
|
|
|
"16": 9999999,//'Maker lost dispute'
|
|
|
|
"17": 9999999,//'Taker lost dispute'
|
|
|
|
}
|
2022-01-02 12:59:48 +00:00
|
|
|
}
|
2022-01-02 00:19:18 +00:00
|
|
|
|
2022-01-02 13:24:35 +00:00
|
|
|
getOrderDetails() {
|
2022-01-09 01:23:13 +00:00
|
|
|
this.setState(null)
|
2022-01-02 13:24:35 +00:00
|
|
|
fetch('/api/order' + '?order_id=' + this.orderId)
|
2022-01-02 12:59:48 +00:00
|
|
|
.then((response) => response.json())
|
2022-01-09 14:07:05 +00:00
|
|
|
.then((data) => {console.log(data) &
|
2022-01-02 12:59:48 +00:00
|
|
|
this.setState({
|
2022-01-14 14:19:25 +00:00
|
|
|
loading: false,
|
2022-01-15 14:22:07 +00:00
|
|
|
delay: this.setDelay(data.status),
|
2022-01-09 14:07:05 +00:00
|
|
|
id: data.id,
|
2022-01-03 12:11:33 +00:00
|
|
|
statusCode: data.status,
|
|
|
|
statusText: data.status_message,
|
2022-01-02 12:59:48 +00:00
|
|
|
type: data.type,
|
|
|
|
currency: data.currency,
|
2022-01-05 00:13:08 +00:00
|
|
|
currencyCode: this.getCurrencyCode(data.currency),
|
2022-01-02 12:59:48 +00:00
|
|
|
amount: data.amount,
|
|
|
|
paymentMethod: data.payment_method,
|
2022-01-02 13:35:31 +00:00
|
|
|
isExplicit: data.is_explicit,
|
2022-01-03 12:11:33 +00:00
|
|
|
premium: data.premium,
|
|
|
|
satoshis: data.satoshis,
|
|
|
|
makerId: data.maker,
|
|
|
|
isParticipant: data.is_participant,
|
2022-01-13 19:22:54 +00:00
|
|
|
urNick: data.ur_nick,
|
2022-01-03 12:11:33 +00:00
|
|
|
makerNick: data.maker_nick,
|
|
|
|
takerId: data.taker,
|
|
|
|
takerNick: data.taker_nick,
|
2022-01-08 13:08:03 +00:00
|
|
|
isMaker: data.is_maker,
|
|
|
|
isTaker: data.is_taker,
|
|
|
|
isBuyer: data.is_buyer,
|
|
|
|
isSeller: data.is_seller,
|
2022-01-10 12:10:32 +00:00
|
|
|
penalty: data.penalty,
|
2022-01-08 13:08:03 +00:00
|
|
|
expiresAt: data.expires_at,
|
|
|
|
badRequest: data.bad_request,
|
|
|
|
bondInvoice: data.bond_invoice,
|
|
|
|
bondSatoshis: data.bond_satoshis,
|
2022-01-08 17:19:30 +00:00
|
|
|
escrowInvoice: data.escrow_invoice,
|
|
|
|
escrowSatoshis: data.escrow_satoshis,
|
2022-01-09 12:14:11 +00:00
|
|
|
invoiceAmount: data.invoice_amount,
|
2022-01-14 12:00:53 +00:00
|
|
|
total_secs_expiry: data.total_secs_exp,
|
|
|
|
numSimilarOrders: data.num_similar_orders,
|
|
|
|
priceNow: data.price_now,
|
|
|
|
premiumNow: data.premium_now,
|
|
|
|
robotsInBook: data.robots_in_book,
|
|
|
|
premiumPercentile: data.premium_percentile,
|
|
|
|
numSimilarOrders: data.num_similar_orders
|
2022-01-11 20:49:53 +00:00
|
|
|
})
|
2022-01-02 12:59:48 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-01-09 01:23:13 +00:00
|
|
|
// These are used to refresh the data
|
|
|
|
componentDidMount() {
|
|
|
|
this.interval = setInterval(this.tick, this.state.delay);
|
|
|
|
}
|
2022-01-15 12:00:11 +00:00
|
|
|
componentDidUpdate() {
|
|
|
|
clearInterval(this.interval);
|
2022-01-09 01:23:13 +00:00
|
|
|
this.interval = setInterval(this.tick, this.state.delay);
|
|
|
|
}
|
2022-01-15 12:00:11 +00:00
|
|
|
|
2022-01-09 01:23:13 +00:00
|
|
|
componentWillUnmount() {
|
|
|
|
clearInterval(this.interval);
|
|
|
|
}
|
|
|
|
tick = () => {
|
|
|
|
this.getOrderDetails();
|
|
|
|
}
|
|
|
|
|
2022-01-04 15:00:34 +00:00
|
|
|
// Fix to use proper react props
|
|
|
|
handleClickBackButton=()=>{
|
|
|
|
window.history.back();
|
|
|
|
}
|
2022-01-05 00:13:08 +00:00
|
|
|
|
2022-01-14 00:43:26 +00:00
|
|
|
// Countdown Renderer callback with condition
|
|
|
|
countdownRenderer = ({ total, hours, minutes, seconds, completed }) => {
|
|
|
|
if (completed) {
|
|
|
|
// Render a completed state
|
2022-01-18 15:23:57 +00:00
|
|
|
return (<span> The order has expired</span>);
|
|
|
|
|
2022-01-14 00:43:26 +00:00
|
|
|
} else {
|
|
|
|
var col = 'black'
|
|
|
|
var fraction_left = (total/1000) / this.state.total_secs_expiry
|
2022-01-14 12:00:53 +00:00
|
|
|
// Make orange at 25% of time left
|
2022-01-14 00:43:26 +00:00
|
|
|
if (fraction_left < 0.25){col = 'orange'}
|
|
|
|
// Make red at 10% of time left
|
|
|
|
if (fraction_left < 0.1){col = 'red'}
|
2022-01-14 12:00:53 +00:00
|
|
|
// Render a countdown, bold when less than 25%
|
2022-01-14 00:43:26 +00:00
|
|
|
return (
|
|
|
|
fraction_left < 0.25 ? <b><span style={{color:col}}>{hours}h {zeroPad(minutes)}m {zeroPad(seconds)}s </span></b>
|
|
|
|
:<span style={{color:col}}>{hours}h {zeroPad(minutes)}m {zeroPad(seconds)}s </span>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
LinearDeterminate =()=> {
|
|
|
|
const [progress, setProgress] = React.useState(0);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
const timer = setInterval(() => {
|
|
|
|
setProgress((oldProgress) => {
|
|
|
|
var left = calcTimeDelta( new Date(this.state.expiresAt)).total /1000;
|
|
|
|
return (left / this.state.total_secs_expiry) * 100;
|
|
|
|
});
|
|
|
|
}, 1000);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
clearInterval(timer);
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Box sx={{ width: '100%' }}>
|
|
|
|
<LinearProgress variant="determinate" value={progress} />
|
|
|
|
</Box>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-05 00:13:08 +00:00
|
|
|
handleClickTakeOrderButton=()=>{
|
|
|
|
console.log(this.state)
|
|
|
|
const requestOptions = {
|
|
|
|
method: 'POST',
|
2022-01-06 20:33:40 +00:00
|
|
|
headers: {'Content-Type':'application/json', 'X-CSRFToken': getCookie('csrftoken'),},
|
|
|
|
body: JSON.stringify({
|
|
|
|
'action':'take',
|
|
|
|
}),
|
2022-01-05 00:13:08 +00:00
|
|
|
};
|
|
|
|
fetch('/api/order/' + '?order_id=' + this.orderId, requestOptions)
|
|
|
|
.then((response) => response.json())
|
2022-01-11 20:49:53 +00:00
|
|
|
.then((data) => (this.setState({badRequest:data.bad_request})
|
|
|
|
& console.log(data)
|
|
|
|
& this.getOrderDetails(data.id)));
|
2022-01-05 00:13:08 +00:00
|
|
|
}
|
2022-01-08 12:48:03 +00:00
|
|
|
getCurrencyDict() {
|
2022-01-09 14:29:10 +00:00
|
|
|
fetch('/static/assets/currencies.json')
|
2022-01-08 12:48:03 +00:00
|
|
|
.then((response) => response.json())
|
|
|
|
.then((data) =>
|
|
|
|
this.setState({
|
|
|
|
currencies_dict: data
|
|
|
|
}));
|
|
|
|
}
|
2022-01-09 12:35:19 +00:00
|
|
|
|
2022-01-15 15:12:26 +00:00
|
|
|
// set delay to the one matching the order status. If null order status, delay goes to 9999999.
|
|
|
|
setDelay = (status)=>{
|
|
|
|
return status >= 0 ? this.statusToDelay[status.toString()] : 99999999;
|
2022-01-15 14:22:07 +00:00
|
|
|
}
|
|
|
|
|
2022-01-08 12:48:03 +00:00
|
|
|
getCurrencyCode(val){
|
2022-01-09 15:28:12 +00:00
|
|
|
let code = val ? this.state.currencies_dict[val.toString()] : ""
|
|
|
|
return code
|
2022-01-08 12:48:03 +00:00
|
|
|
}
|
2022-01-05 00:13:08 +00:00
|
|
|
|
2022-01-18 15:23:57 +00:00
|
|
|
handleClickConfirmCancelButton=()=>{
|
2022-01-08 15:34:09 +00:00
|
|
|
console.log(this.state)
|
|
|
|
const requestOptions = {
|
|
|
|
method: 'POST',
|
|
|
|
headers: {'Content-Type':'application/json', 'X-CSRFToken': getCookie('csrftoken'),},
|
|
|
|
body: JSON.stringify({
|
|
|
|
'action':'cancel',
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
fetch('/api/order/' + '?order_id=' + this.orderId, requestOptions)
|
|
|
|
.then((response) => response.json())
|
|
|
|
.then((data) => (console.log(data) & this.getOrderDetails(data.id)));
|
2022-01-18 15:23:57 +00:00
|
|
|
this.handleClickCloseConfirmCancelDialog();
|
|
|
|
}
|
|
|
|
|
|
|
|
handleClickOpenConfirmCancelDialog = () => {
|
|
|
|
this.setState({openCancel: true});
|
|
|
|
};
|
|
|
|
handleClickCloseConfirmCancelDialog = () => {
|
|
|
|
this.setState({openCancel: false});
|
|
|
|
};
|
|
|
|
|
|
|
|
CancelDialog =() =>{
|
|
|
|
return(
|
|
|
|
<Dialog
|
|
|
|
open={this.state.openCancel}
|
|
|
|
onClose={this.handleClickCloseConfirmCancelDialog}
|
|
|
|
aria-labelledby="cancel-dialog-title"
|
|
|
|
aria-describedby="cancel-dialog-description"
|
|
|
|
>
|
|
|
|
<DialogTitle id="cancel-dialog-title">
|
|
|
|
{"Cancel the order?"}
|
|
|
|
</DialogTitle>
|
|
|
|
<DialogContent>
|
|
|
|
<DialogContentText id="cancel-dialog-description">
|
|
|
|
If the order is cancelled now you will lose your bond.
|
|
|
|
</DialogContentText>
|
|
|
|
</DialogContent>
|
|
|
|
<DialogActions>
|
|
|
|
<Button onClick={this.handleClickCloseConfirmCancelDialog} autoFocus>Go back</Button>
|
|
|
|
<Button onClick={this.handleClickConfirmCancelButton}> Confirm Cancel </Button>
|
|
|
|
</DialogActions>
|
|
|
|
</Dialog>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
CancelButton = () => {
|
|
|
|
|
|
|
|
// If maker and Waiting for Bond. Or if taker and Waiting for bond.
|
|
|
|
// Simply allow to cancel without showing the cancel dialog.
|
|
|
|
if ((this.state.isMaker & this.state.statusCode == 0) || this.state.isTaker & this.state.statusCode == 3){
|
|
|
|
return(
|
|
|
|
<Grid item xs={12} align="center">
|
|
|
|
<Button variant='contained' color='secondary' onClick={this.handleClickConfirmCancelButton}>Cancel</Button>
|
|
|
|
</Grid>
|
|
|
|
)}
|
|
|
|
// If the order does not yet have an escrow deposited. Show dialog
|
|
|
|
// to confirm forfeiting the bond
|
|
|
|
if (this.state.statusCode < 8){
|
|
|
|
return(
|
|
|
|
<Grid item xs={12} align="center">
|
|
|
|
<this.CancelDialog/>
|
|
|
|
<Button variant='contained' color='secondary' onClick={this.handleClickOpenConfirmCancelDialog}>Cancel</Button>
|
|
|
|
</Grid>
|
|
|
|
)}
|
|
|
|
|
|
|
|
// TODO If the escrow is Locked, show the collaborative cancel button.
|
|
|
|
|
|
|
|
// If none of the above do not return a cancel button.
|
|
|
|
return(null)
|
2022-01-08 15:34:09 +00:00
|
|
|
}
|
|
|
|
|
2022-01-08 13:08:03 +00:00
|
|
|
orderBox=()=>{
|
|
|
|
return(
|
2022-01-03 12:11:33 +00:00
|
|
|
<Grid container spacing={1}>
|
|
|
|
<Grid item xs={12} align="center">
|
|
|
|
<Typography component="h5" variant="h5">
|
2022-01-17 16:41:55 +00:00
|
|
|
Order Details
|
2022-01-03 12:11:33 +00:00
|
|
|
</Typography>
|
2022-01-03 19:15:13 +00:00
|
|
|
<Paper elevation={12} style={{ padding: 8,}}>
|
2022-01-05 00:13:08 +00:00
|
|
|
<List dense="true">
|
2022-01-05 02:03:03 +00:00
|
|
|
<ListItem >
|
2022-01-03 12:11:33 +00:00
|
|
|
<ListItemAvatar sx={{ width: 56, height: 56 }}>
|
|
|
|
<Avatar
|
|
|
|
alt={this.state.makerNick}
|
|
|
|
src={window.location.origin +'/static/assets/avatars/' + this.state.makerNick + '.png'}
|
|
|
|
/>
|
|
|
|
</ListItemAvatar>
|
2022-01-09 14:07:05 +00:00
|
|
|
<ListItemText primary={this.state.makerNick + (this.state.type ? " (Seller)" : " (Buyer)")} secondary="Order maker" align="right"/>
|
2022-01-03 12:11:33 +00:00
|
|
|
</ListItem>
|
|
|
|
<Divider />
|
2022-01-05 02:03:03 +00:00
|
|
|
|
|
|
|
{this.state.isParticipant ?
|
|
|
|
<>
|
|
|
|
{this.state.takerNick!='None' ?
|
|
|
|
<>
|
|
|
|
<ListItem align="left">
|
2022-01-09 14:07:05 +00:00
|
|
|
<ListItemText primary={this.state.takerNick + (this.state.type ? " (Buyer)" : " (Seller)")} secondary="Order taker"/>
|
2022-01-05 02:03:03 +00:00
|
|
|
<ListItemAvatar >
|
|
|
|
<Avatar
|
|
|
|
alt={this.state.makerNick}
|
|
|
|
src={window.location.origin +'/static/assets/avatars/' + this.state.takerNick + '.png'}
|
|
|
|
/>
|
|
|
|
</ListItemAvatar>
|
|
|
|
</ListItem>
|
|
|
|
<Divider />
|
|
|
|
</>:
|
2022-01-05 12:18:54 +00:00
|
|
|
""
|
2022-01-05 02:03:03 +00:00
|
|
|
}
|
2022-01-05 12:18:54 +00:00
|
|
|
<ListItem>
|
2022-01-14 12:00:53 +00:00
|
|
|
<ListItemIcon>
|
|
|
|
<ArticleIcon/>
|
|
|
|
</ListItemIcon>
|
2022-01-05 12:18:54 +00:00
|
|
|
<ListItemText primary={this.state.statusText} secondary="Order status"/>
|
|
|
|
</ListItem>
|
|
|
|
<Divider />
|
2022-01-05 02:03:03 +00:00
|
|
|
</>
|
|
|
|
:""
|
|
|
|
}
|
|
|
|
|
2022-01-03 12:11:33 +00:00
|
|
|
<ListItem>
|
2022-01-14 12:00:53 +00:00
|
|
|
<ListItemIcon>
|
2022-01-16 18:32:34 +00:00
|
|
|
{getFlags(this.state.currencyCode)}
|
2022-01-14 12:00:53 +00:00
|
|
|
</ListItemIcon>
|
2022-01-16 18:32:34 +00:00
|
|
|
<ListItemText primary={parseFloat(parseFloat(this.state.amount).toFixed(4))
|
|
|
|
+" "+this.state.currencyCode} secondary="Amount"/>
|
2022-01-03 12:11:33 +00:00
|
|
|
</ListItem>
|
|
|
|
<Divider />
|
|
|
|
<ListItem>
|
2022-01-14 12:00:53 +00:00
|
|
|
<ListItemIcon>
|
|
|
|
<PaymentsIcon/>
|
|
|
|
</ListItemIcon>
|
2022-01-03 12:11:33 +00:00
|
|
|
<ListItemText primary={this.state.paymentMethod} secondary="Accepted payment methods"/>
|
|
|
|
</ListItem>
|
|
|
|
<Divider />
|
2022-01-14 12:00:53 +00:00
|
|
|
|
|
|
|
{/* If there is live Price and Premium data, show it. Otherwise show the order maker settings */}
|
2022-01-03 12:11:33 +00:00
|
|
|
<ListItem>
|
2022-01-14 12:00:53 +00:00
|
|
|
<ListItemIcon>
|
|
|
|
<PriceChangeIcon/>
|
|
|
|
</ListItemIcon>
|
|
|
|
{this.state.priceNow?
|
|
|
|
<ListItemText primary={pn(this.state.priceNow)+" "+this.state.currencyCode+"/BTC - Premium: "+this.state.premiumNow+"%"} secondary="Price and Premium"/>
|
|
|
|
:
|
|
|
|
(this.state.isExplicit ?
|
|
|
|
<ListItemText primary={pn(this.state.satoshis)} secondary="Amount of Satoshis"/>
|
|
|
|
:
|
|
|
|
<ListItemText primary={parseFloat(parseFloat(this.state.premium).toFixed(2))+"%"} secondary="Premium over market price"/>
|
|
|
|
)
|
|
|
|
}
|
2022-01-03 12:11:33 +00:00
|
|
|
</ListItem>
|
|
|
|
<Divider />
|
2022-01-05 02:03:03 +00:00
|
|
|
|
2022-01-03 12:11:33 +00:00
|
|
|
<ListItem>
|
2022-01-14 12:00:53 +00:00
|
|
|
<ListItemIcon>
|
|
|
|
<NumbersIcon/>
|
|
|
|
</ListItemIcon>
|
|
|
|
<ListItemText primary={this.orderId} secondary="Order ID"/>
|
2022-01-03 12:11:33 +00:00
|
|
|
</ListItem>
|
2022-01-05 02:03:03 +00:00
|
|
|
<Divider />
|
|
|
|
<ListItem>
|
2022-01-14 12:00:53 +00:00
|
|
|
<ListItemIcon>
|
|
|
|
<AccessTimeIcon/>
|
|
|
|
</ListItemIcon>
|
|
|
|
<ListItemText secondary="Expires in">
|
2022-01-14 14:19:25 +00:00
|
|
|
<Countdown date={new Date(this.state.expiresAt)} renderer={this.countdownRenderer} />
|
2022-01-14 00:43:26 +00:00
|
|
|
</ListItemText>
|
2022-01-05 02:03:03 +00:00
|
|
|
</ListItem>
|
2022-01-14 00:43:26 +00:00
|
|
|
<this.LinearDeterminate />
|
2022-01-03 12:11:33 +00:00
|
|
|
</List>
|
2022-01-10 12:10:32 +00:00
|
|
|
|
|
|
|
{/* If the user has a penalty/limit */}
|
|
|
|
{this.state.penalty ?
|
|
|
|
<>
|
|
|
|
<Divider />
|
|
|
|
<Grid item xs={12} align="center">
|
|
|
|
<Alert severity="warning" sx={{maxWidth:360}}>
|
|
|
|
You cannot take an order yet! Wait {this.state.penalty} seconds
|
|
|
|
</Alert>
|
|
|
|
</Grid>
|
|
|
|
</>
|
|
|
|
: null}
|
2022-01-05 02:03:03 +00:00
|
|
|
|
2022-01-05 00:13:08 +00:00
|
|
|
</Paper>
|
2022-01-08 13:08:03 +00:00
|
|
|
</Grid>
|
2022-01-03 12:11:33 +00:00
|
|
|
|
2022-01-18 15:23:57 +00:00
|
|
|
{/* Participants can see the "Cancel" Button, but cannot see the "Back" or "Take Order" buttons */}
|
|
|
|
{this.state.isParticipant ?
|
|
|
|
<this.CancelButton/>
|
|
|
|
:
|
2022-01-08 13:08:03 +00:00
|
|
|
<>
|
|
|
|
<Grid item xs={12} align="center">
|
|
|
|
<Button variant='contained' color='primary' onClick={this.handleClickTakeOrderButton}>Take Order</Button>
|
|
|
|
</Grid>
|
|
|
|
<Grid item xs={12} align="center">
|
|
|
|
<Button variant='contained' color='secondary' onClick={this.handleClickBackButton}>Back</Button>
|
|
|
|
</Grid>
|
|
|
|
</>
|
|
|
|
}
|
2022-01-08 15:34:09 +00:00
|
|
|
|
2022-01-18 15:23:57 +00:00
|
|
|
</Grid>
|
2022-01-08 13:08:03 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-01-09 20:05:19 +00:00
|
|
|
orderDetailsPage (){
|
|
|
|
return(
|
2022-01-08 13:08:03 +00:00
|
|
|
this.state.badRequest ?
|
|
|
|
<div align='center'>
|
|
|
|
<Typography component="subtitle2" variant="subtitle2" color="secondary" >
|
|
|
|
{this.state.badRequest}<br/>
|
|
|
|
</Typography>
|
|
|
|
<Button variant='contained' color='secondary' onClick={this.handleClickBackButton}>Back</Button>
|
|
|
|
</div>
|
|
|
|
:
|
|
|
|
(this.state.isParticipant ?
|
|
|
|
<Grid container xs={12} align="center" spacing={2}>
|
|
|
|
<Grid item xs={6} align="left">
|
|
|
|
{this.orderBox()}
|
|
|
|
</Grid>
|
|
|
|
<Grid item xs={6} align="left">
|
|
|
|
<TradeBox data={this.state}/>
|
|
|
|
</Grid>
|
2022-01-03 12:11:33 +00:00
|
|
|
</Grid>
|
2022-01-08 13:08:03 +00:00
|
|
|
:
|
2022-01-04 15:00:34 +00:00
|
|
|
<Grid item xs={12} align="center">
|
2022-01-08 13:08:03 +00:00
|
|
|
{this.orderBox()}
|
|
|
|
</Grid>)
|
2022-01-09 20:05:19 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
render (){
|
|
|
|
return (
|
|
|
|
// Only so nothing shows while requesting the first batch of data
|
2022-01-14 14:19:25 +00:00
|
|
|
this.state.loading ? <CircularProgress /> : this.orderDetailsPage()
|
2022-01-02 12:59:48 +00:00
|
|
|
);
|
|
|
|
}
|
2022-01-09 14:29:10 +00:00
|
|
|
}
|