mirror of
https://github.com/RoboSats/robosats.git
synced 2025-01-18 20:21:35 +00:00
175 lines
6.5 KiB
JavaScript
175 lines
6.5 KiB
JavaScript
import React, { Component } from 'react';
|
|
import { withTranslation, Trans} from "react-i18next";
|
|
import {Button, Link, Badge, TextField, Grid, Container, Card, CardHeader, Paper, Avatar, FormHelperText, Typography} from "@mui/material";
|
|
import ReconnectingWebSocket from 'reconnecting-websocket';
|
|
import * as openpgp from 'openpgp/lightweight';
|
|
|
|
class Chat extends Component {
|
|
constructor(props) {
|
|
super(props);
|
|
}
|
|
|
|
state = {
|
|
messages: [],
|
|
value:'',
|
|
connected: false,
|
|
peer_connected: false,
|
|
};
|
|
|
|
rws = new ReconnectingWebSocket('ws://' + window.location.host + '/ws/chat/' + this.props.orderId + '/');
|
|
|
|
componentDidMount() {
|
|
this.rws.addEventListener('open', () => {
|
|
console.log('Connected!');
|
|
this.setState({connected: true});
|
|
this.rws.send(JSON.stringify({
|
|
type: "message",
|
|
message: 'just-connected',
|
|
nick: this.props.ur_nick,
|
|
}));
|
|
});
|
|
|
|
this.rws.addEventListener('message', (message) => {
|
|
|
|
const dataFromServer = JSON.parse(message.data);
|
|
console.log('Got reply!', dataFromServer.type);
|
|
|
|
if (dataFromServer){
|
|
if (dataFromServer.message != 'just-connected' & dataFromServer.message != 'peer-disconnected'){
|
|
this.setState((state) =>
|
|
({
|
|
messages: [...state.messages,
|
|
{
|
|
msg: dataFromServer.message,
|
|
userNick: dataFromServer.user_nick,
|
|
}],
|
|
})
|
|
)
|
|
}
|
|
this.setState({peer_connected: dataFromServer.peer_connected})
|
|
}
|
|
});
|
|
|
|
this.rws.addEventListener('close', () => {
|
|
console.log('Socket is closed. Reconnect will be attempted');
|
|
this.setState({connected: false});
|
|
});
|
|
|
|
this.rws.addEventListener('error', () => {
|
|
console.error('Socket encountered error: Closing socket');
|
|
});
|
|
}
|
|
|
|
componentDidUpdate() {
|
|
this.scrollToBottom();
|
|
}
|
|
|
|
scrollToBottom = () => {
|
|
this.messagesEnd.scrollIntoView({ behavior: "smooth" });
|
|
}
|
|
|
|
onButtonClicked = (e) => {
|
|
if(this.state.value!=''){
|
|
this.rws.send(JSON.stringify({
|
|
type: "message",
|
|
message: this.state.value,
|
|
nick: this.props.ur_nick,
|
|
}));
|
|
this.setState({value: ""});
|
|
}
|
|
e.preventDefault();
|
|
}
|
|
|
|
render() {
|
|
const { t } = this.props;
|
|
return (
|
|
<Container component="main" maxWidth="xs" >
|
|
<Grid container spacing={0.5}>
|
|
<Grid item xs={0.3}/>
|
|
<Grid item xs={5.5}>
|
|
<Paper elevation={1} style={this.state.connected ? {backgroundColor: '#e8ffe6'}: {backgroundColor: '#FFF1C5'}}>
|
|
<Typography variant='caption' sx={{color: '#111111'}}>
|
|
{t("You")+": "}{this.state.connected ? t("connected"): t("disconnected")}
|
|
</Typography>
|
|
</Paper>
|
|
</Grid>
|
|
<Grid item xs={0.4}/>
|
|
<Grid item xs={5.5}>
|
|
<Paper elevation={1} style={this.state.peer_connected ? {backgroundColor: '#e8ffe6'}: {backgroundColor: '#FFF1C5'}}>
|
|
<Typography variant='caption' sx={{color: '#111111'}}>
|
|
{t("Peer")+": "}{this.state.peer_connected ? t("connected"): t("disconnected")}
|
|
</Typography>
|
|
</Paper>
|
|
</Grid>
|
|
<Grid item xs={0.3}/>
|
|
</Grid>
|
|
<Paper elevation={1} style={{ height: '300px', maxHeight: '300px' , width: '280px' ,overflow: 'auto', backgroundColor: '#F7F7F7' }}>
|
|
{this.state.messages.map((message, index) =>
|
|
<li style={{listStyleType:"none"}} key={index}>
|
|
<Card elevation={5} align="left" >
|
|
{/* If message sender is not our nick, gray color, if it is our nick, green color */}
|
|
{message.userNick == this.props.ur_nick ?
|
|
<CardHeader sx={{color: '#111111'}}
|
|
avatar={
|
|
<Badge variant="dot" overlap="circular" badgeContent="" color={this.state.connected ? "success" : "error"}>
|
|
<Avatar className="flippedSmallAvatar"
|
|
alt={message.userNick}
|
|
src={window.location.origin +'/static/assets/avatars/' + message.userNick + '.png'}
|
|
/>
|
|
</Badge>
|
|
}
|
|
style={{backgroundColor: '#eeeeee'}}
|
|
title={message.userNick}
|
|
subheader={message.msg}
|
|
subheaderTypographyProps={{sx: {wordWrap: "break-word", width: '200px', color: '#444444'}}}
|
|
/>
|
|
:
|
|
<CardHeader sx={{color: '#111111'}}
|
|
avatar={
|
|
<Badge variant="dot" overlap="circular" badgeContent="" color={this.state.peer_connected ? "success" : "error"}>
|
|
<Avatar className="flippedSmallAvatar"
|
|
alt={message.userNick}
|
|
src={window.location.origin +'/static/assets/avatars/' + message.userNick + '.png'}
|
|
/>
|
|
</Badge>
|
|
}
|
|
style={{backgroundColor: '#fafafa'}}
|
|
title={message.userNick}
|
|
subheader={message.msg}
|
|
subheaderTypographyProps={{sx: {wordWrap: "break-word", width: '200px', color: '#444444'}}}
|
|
/>}
|
|
</Card>
|
|
</li>)}
|
|
<div style={{ float:"left", clear: "both" }} ref={(el) => { this.messagesEnd = el; }}></div>
|
|
</Paper>
|
|
<form noValidate onSubmit={this.onButtonClicked}>
|
|
<Grid alignItems="stretch" style={{ display: "flex" }}>
|
|
<Grid item alignItems="stretch" style={{ display: "flex"}}>
|
|
<TextField
|
|
label={t("Type a message")}
|
|
variant="standard"
|
|
size="small"
|
|
helperText={this.state.connected ? null : t("Connecting...")}
|
|
value={this.state.value}
|
|
onChange={e => {
|
|
this.setState({ value: e.target.value });
|
|
this.value = this.state.value;
|
|
}}
|
|
sx={{width: 214}}
|
|
/>
|
|
</Grid>
|
|
<Grid item alignItems="stretch" style={{ display: "flex" }}>
|
|
<Button sx={{'width':68}} disabled={!this.state.connected} type="submit" variant="contained" color="primary">{t("Send")} </Button>
|
|
</Grid>
|
|
</Grid>
|
|
</form>
|
|
<FormHelperText>
|
|
{t("The chat has no memory: if you leave, messages are lost.")} <Link target="_blank" href={t("PGP_guide_url")}> {t("Learn easy PGP encryption.")}</Link>
|
|
</FormHelperText>
|
|
</Container>
|
|
)
|
|
}
|
|
}
|
|
|
|
export default withTranslation()(Chat);
|