mirror of
https://github.com/RoboSats/robosats.git
synced 2024-12-15 11:56:23 +00:00
312 lines
10 KiB
JavaScript
312 lines
10 KiB
JavaScript
import React, { Component } from "react";
|
|
import { Paper, Button , Grid, Typography, List, ListItem, ListItemText, ListItemAvatar, Avatar, Divider, Box, LinearProgress} from "@material-ui/core"
|
|
import TradeBox from "./TradeBox";
|
|
|
|
function msToTime(duration) {
|
|
var seconds = Math.floor((duration / 1000) % 60),
|
|
minutes = Math.floor((duration / (1000 * 60)) % 60),
|
|
hours = Math.floor((duration / (1000 * 60 * 60)) % 24);
|
|
|
|
minutes = (minutes < 10) ? "0" + minutes : minutes;
|
|
seconds = (seconds < 10) ? "0" + seconds : seconds;
|
|
|
|
return hours + "h " + minutes + "m " + seconds + "s";
|
|
}
|
|
|
|
// TO DO fix Progress bar to go from 100 to 0, from total_expiration time, showing time_left
|
|
function LinearDeterminate() {
|
|
const [progress, setProgress] = React.useState(0);
|
|
|
|
React.useEffect(() => {
|
|
const timer = setInterval(() => {
|
|
setProgress((oldProgress) => {
|
|
if (oldProgress === 0) {
|
|
return 100;
|
|
}
|
|
const diff = 1;
|
|
return Math.max(oldProgress - diff, 0);
|
|
});
|
|
}, 500);
|
|
|
|
return () => {
|
|
clearInterval(timer);
|
|
};
|
|
}, []);
|
|
|
|
return (
|
|
<Box sx={{ width: '100%' }}>
|
|
<LinearProgress variant="determinate" value={progress} />
|
|
</Box>
|
|
);
|
|
}
|
|
|
|
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');
|
|
|
|
// pretty numbers
|
|
function pn(x) {
|
|
return x.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ",");
|
|
}
|
|
|
|
export default class OrderPage extends Component {
|
|
constructor(props) {
|
|
super(props);
|
|
this.state = {
|
|
isExplicit: false,
|
|
delay: 5000, // Refresh every 5 seconds
|
|
currencies_dict: {"1":"USD"}
|
|
};
|
|
this.orderId = this.props.match.params.orderId;
|
|
this.getCurrencyDict();
|
|
this.getOrderDetails();
|
|
}
|
|
|
|
getOrderDetails() {
|
|
this.setState(null)
|
|
fetch('/api/order' + '?order_id=' + this.orderId)
|
|
.then((response) => response.json())
|
|
.then((data) => {
|
|
this.setState({
|
|
statusCode: data.status,
|
|
statusText: data.status_message,
|
|
type: data.type,
|
|
currency: data.currency,
|
|
currencyCode: this.getCurrencyCode(data.currency),
|
|
amount: data.amount,
|
|
paymentMethod: data.payment_method,
|
|
isExplicit: data.is_explicit,
|
|
premium: data.premium,
|
|
satoshis: data.satoshis,
|
|
makerId: data.maker,
|
|
isParticipant: data.is_participant,
|
|
makerNick: data.maker_nick,
|
|
takerId: data.taker,
|
|
takerNick: data.taker_nick,
|
|
isMaker: data.is_maker,
|
|
isTaker: data.is_taker,
|
|
isBuyer: data.is_buyer,
|
|
isSeller: data.is_seller,
|
|
expiresAt: data.expires_at,
|
|
badRequest: data.bad_request,
|
|
bondInvoice: data.bond_invoice,
|
|
bondSatoshis: data.bond_satoshis,
|
|
escrowInvoice: data.escrow_invoice,
|
|
escrowSatoshis: data.escrow_satoshis,
|
|
invoiceAmount: data.invoice_amount,
|
|
badRequest: data.bad_request,
|
|
});
|
|
});
|
|
}
|
|
|
|
// These are used to refresh the data
|
|
componentDidMount() {
|
|
this.interval = setInterval(this.tick, this.state.delay);
|
|
}
|
|
componentDidUpdate(prevProps, prevState) {
|
|
if (prevState.delay !== this.state.delay) {
|
|
clearInterval(this.interval);
|
|
this.interval = setInterval(this.tick, this.state.delay);
|
|
}
|
|
}
|
|
componentWillUnmount() {
|
|
clearInterval(this.interval);
|
|
}
|
|
tick = () => {
|
|
this.getOrderDetails();
|
|
}
|
|
handleDelayChange = (e) => {
|
|
this.setState({ delay: Number(e.target.value) });
|
|
}
|
|
|
|
// Fix to use proper react props
|
|
handleClickBackButton=()=>{
|
|
window.history.back();
|
|
}
|
|
|
|
handleClickTakeOrderButton=()=>{
|
|
console.log(this.state)
|
|
const requestOptions = {
|
|
method: 'POST',
|
|
headers: {'Content-Type':'application/json', 'X-CSRFToken': getCookie('csrftoken'),},
|
|
body: JSON.stringify({
|
|
'action':'take',
|
|
}),
|
|
};
|
|
fetch('/api/order/' + '?order_id=' + this.orderId, requestOptions)
|
|
.then((response) => response.json())
|
|
.then((data) => (console.log(data) & this.getOrderDetails(data.id)));
|
|
}
|
|
getCurrencyDict() {
|
|
fetch('/static/assets/currencies.json')
|
|
.then((response) => response.json())
|
|
.then((data) =>
|
|
this.setState({
|
|
currencies_dict: data
|
|
}));
|
|
}
|
|
|
|
getCurrencyCode(val){
|
|
return this.state.currencies_dict[val.toString()]
|
|
}
|
|
|
|
handleClickCancelOrderButton=()=>{
|
|
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)));
|
|
}
|
|
|
|
orderBox=()=>{
|
|
return(
|
|
<Grid container spacing={1}>
|
|
<Grid item xs={12} align="center">
|
|
<Typography component="h5" variant="h5">
|
|
{this.state.type ? "Sell " : "Buy "} Order Details
|
|
</Typography>
|
|
<Paper elevation={12} style={{ padding: 8,}}>
|
|
<List dense="true">
|
|
<ListItem >
|
|
<ListItemAvatar sx={{ width: 56, height: 56 }}>
|
|
<Avatar
|
|
alt={this.state.makerNick}
|
|
src={window.location.origin +'/static/assets/avatars/' + this.state.makerNick + '.png'}
|
|
/>
|
|
</ListItemAvatar>
|
|
<ListItemText primary={this.state.makerNick + (this.state.type ? " (Buyer)" : " (Seller)")} secondary="Order maker" align="right"/>
|
|
</ListItem>
|
|
<Divider />
|
|
|
|
{this.state.isParticipant ?
|
|
<>
|
|
{this.state.takerNick!='None' ?
|
|
<>
|
|
<ListItem align="left">
|
|
<ListItemText primary={this.state.takerNick + (this.state.type ? " (Seller)" : " (Buyer)")} secondary="Order taker"/>
|
|
<ListItemAvatar >
|
|
<Avatar
|
|
alt={this.state.makerNick}
|
|
src={window.location.origin +'/static/assets/avatars/' + this.state.takerNick + '.png'}
|
|
/>
|
|
</ListItemAvatar>
|
|
</ListItem>
|
|
<Divider />
|
|
</>:
|
|
""
|
|
}
|
|
<ListItem>
|
|
<ListItemText primary={this.state.statusText} secondary="Order status"/>
|
|
</ListItem>
|
|
<Divider />
|
|
</>
|
|
:""
|
|
}
|
|
|
|
<ListItem>
|
|
<ListItemText primary={parseFloat(parseFloat(this.state.amount).toFixed(4))+" "+this.state.currencyCode} secondary="Amount"/>
|
|
</ListItem>
|
|
<Divider />
|
|
<ListItem>
|
|
<ListItemText primary={this.state.paymentMethod} secondary="Accepted payment methods"/>
|
|
</ListItem>
|
|
<Divider />
|
|
<ListItem>
|
|
{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"/>
|
|
}
|
|
</ListItem>
|
|
<Divider />
|
|
|
|
<ListItem>
|
|
<ListItemText primary={'#'+this.orderId} secondary="Order ID"/>
|
|
</ListItem>
|
|
<Divider />
|
|
<ListItem>
|
|
<ListItemText primary={msToTime( new Date(this.state.expiresAt) - Date.now())} secondary="Expires"/>
|
|
</ListItem>
|
|
<LinearDeterminate />
|
|
</List>
|
|
|
|
</Paper>
|
|
</Grid>
|
|
|
|
{/* Participants cannot see the Back or Take Order buttons */}
|
|
{this.state.isParticipant ? "" :
|
|
<>
|
|
<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>
|
|
</>
|
|
}
|
|
|
|
{/* Makers can cancel before commiting the bond (status 0)*/}
|
|
{this.state.isMaker & this.state.statusCode == 0 ?
|
|
<Grid item xs={12} align="center">
|
|
<Button variant='contained' color='secondary' onClick={this.handleClickCancelOrderButton}>Cancel</Button>
|
|
</Grid>
|
|
:""}
|
|
|
|
{/* Takers can cancel before commiting the bond (status 3)*/}
|
|
{this.state.isTaker & this.state.statusCode == 3 ?
|
|
<Grid item xs={12} align="center">
|
|
<Button variant='contained' color='secondary' onClick={this.handleClickCancelOrderButton}>Cancel</Button>
|
|
</Grid>
|
|
:""}
|
|
|
|
</Grid>
|
|
)
|
|
}
|
|
|
|
|
|
render (){
|
|
return (
|
|
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>
|
|
</Grid>
|
|
:
|
|
<Grid item xs={12} align="center">
|
|
{this.orderBox()}
|
|
</Grid>)
|
|
);
|
|
}
|
|
}
|