forked from p34709852/monkey
UI: refactor TelemetryLog component out of MapPage and migrate it to hook with typescript
This commit is contained in:
parent
7f903efb07
commit
7425bf1bbd
|
@ -0,0 +1,96 @@
|
||||||
|
import React, {useEffect, useRef, useState} from 'react';
|
||||||
|
import AuthComponent from '../AuthComponent';
|
||||||
|
|
||||||
|
|
||||||
|
const authComponent = new AuthComponent({});
|
||||||
|
|
||||||
|
const TelemetryLog = (props: { onStatusChange: Function }) => {
|
||||||
|
|
||||||
|
let [telemetryUpdateInProgress, setTelemetryUpdateInProgress] = useState(false);
|
||||||
|
let [telemetry, setTelemetry] = useState([]);
|
||||||
|
let [lastTelemetryTimestamp, setLastTelemetryTimestamp] = useState(null);
|
||||||
|
let [isScrolledUp, setIsScrolledUp] = useState(false);
|
||||||
|
let [telemetryLines, setTelemetryLines] = useState(0);
|
||||||
|
let [telemetryCurrentLine, setTelemetryCurrentLine] = useState(0);
|
||||||
|
|
||||||
|
let scrollTop = 0;
|
||||||
|
const telemetryConsole = useRef(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const interval = setInterval(updateTelemetryFromServer, 5000);
|
||||||
|
return function cleanup() {
|
||||||
|
clearInterval(interval);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
function updateTelemetryFromServer() {
|
||||||
|
if (telemetryUpdateInProgress) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
setTelemetryUpdateInProgress(true);
|
||||||
|
authComponent.authFetch('/api/telemetry-feed?timestamp=' + lastTelemetryTimestamp)
|
||||||
|
.then(res => res.json())
|
||||||
|
.then(res => {
|
||||||
|
if ('telemetries' in res) {
|
||||||
|
let newTelem = telemetry.concat(res['telemetries']);
|
||||||
|
setTelemetry(newTelem);
|
||||||
|
setLastTelemetryTimestamp(res['timestamp']);
|
||||||
|
setTelemetryUpdateInProgress(false);
|
||||||
|
props.onStatusChange();
|
||||||
|
|
||||||
|
let telemConsoleRef = telemetryConsole.current;
|
||||||
|
if (!isScrolledUp) {
|
||||||
|
telemConsoleRef.scrollTop = telemConsoleRef.scrollHeight - telemConsoleRef.clientHeight;
|
||||||
|
scrollTop = telemConsoleRef.scrollTop;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleScroll(e) {
|
||||||
|
let element = e.target;
|
||||||
|
|
||||||
|
let telemetryStyle = window.getComputedStyle(element);
|
||||||
|
let telemetryLineHeight = parseInt((telemetryStyle.lineHeight).replace('px', ''));
|
||||||
|
|
||||||
|
setIsScrolledUp((element.scrollTop < scrollTop));
|
||||||
|
setTelemetryCurrentLine(Math.trunc(element.scrollTop / telemetryLineHeight) + 1);
|
||||||
|
setTelemetryLines(Math.trunc(element.scrollHeight / telemetryLineHeight));
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderTelemetryConsole() {
|
||||||
|
return (
|
||||||
|
<div className="telemetry-console" onScroll={handleScroll} ref={telemetryConsole}>
|
||||||
|
{
|
||||||
|
telemetry.map(renderTelemetryEntry)
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderTelemetryEntry(telemetry) {
|
||||||
|
return (
|
||||||
|
<div key={telemetry.id}>
|
||||||
|
<span className="date">{telemetry.timestamp}</span>
|
||||||
|
<span className="source"> {telemetry.hostname}:</span>
|
||||||
|
<span className="event"> {telemetry.brief}</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderTelemetryLineCount() {
|
||||||
|
return (
|
||||||
|
<div className="telemetry-lines">
|
||||||
|
<b>[{telemetryCurrentLine}/{telemetryLines}]</b>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{renderTelemetryLineCount()}
|
||||||
|
{renderTelemetryConsole()}
|
||||||
|
</>);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default TelemetryLog;
|
|
@ -10,6 +10,7 @@ import {getOptions, edgeGroupToColor} from 'components/map/MapOptions';
|
||||||
import AuthComponent from '../AuthComponent';
|
import AuthComponent from '../AuthComponent';
|
||||||
import '../../styles/components/Map.scss';
|
import '../../styles/components/Map.scss';
|
||||||
import {faInfoCircle} from '@fortawesome/free-solid-svg-icons/faInfoCircle';
|
import {faInfoCircle} from '@fortawesome/free-solid-svg-icons/faInfoCircle';
|
||||||
|
import TelemetryLog from '../map/TelemetryLog';
|
||||||
|
|
||||||
class MapPageComponent extends AuthComponent {
|
class MapPageComponent extends AuthComponent {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
|
@ -20,17 +21,8 @@ class MapPageComponent extends AuthComponent {
|
||||||
selected: null,
|
selected: null,
|
||||||
selectedType: null,
|
selectedType: null,
|
||||||
killPressed: false,
|
killPressed: false,
|
||||||
showKillDialog: false,
|
showKillDialog: false
|
||||||
telemetry: [],
|
|
||||||
telemetryLastTimestamp: null,
|
|
||||||
isScrolledUp: false,
|
|
||||||
telemetryLines: 0,
|
|
||||||
telemetryCurrentLine: 0,
|
|
||||||
telemetryUpdateInProgress: false
|
|
||||||
};
|
};
|
||||||
this.telemConsole = React.createRef();
|
|
||||||
this.handleScroll = this.handleScroll.bind(this);
|
|
||||||
this.scrollTop = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
events = {
|
events = {
|
||||||
|
@ -40,7 +32,7 @@ class MapPageComponent extends AuthComponent {
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this.getNodeStateListFromServer();
|
this.getNodeStateListFromServer();
|
||||||
this.updateMapFromServer();
|
this.updateMapFromServer();
|
||||||
this.interval = setInterval(this.timedEvents, 5000);
|
this.interval = setInterval(this.updateMapFromServer, 5000);
|
||||||
}
|
}
|
||||||
|
|
||||||
componentWillUnmount() {
|
componentWillUnmount() {
|
||||||
|
@ -55,11 +47,6 @@ class MapPageComponent extends AuthComponent {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
timedEvents = () => {
|
|
||||||
this.updateMapFromServer();
|
|
||||||
this.updateTelemetryFromServer();
|
|
||||||
};
|
|
||||||
|
|
||||||
updateMapFromServer = () => {
|
updateMapFromServer = () => {
|
||||||
this.authFetch('/api/netmap')
|
this.authFetch('/api/netmap')
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
|
@ -74,33 +61,6 @@ class MapPageComponent extends AuthComponent {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
updateTelemetryFromServer = () => {
|
|
||||||
if (this.state.telemetryUpdateInProgress) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
this.setState({telemetryUpdateInProgress: true});
|
|
||||||
this.authFetch('/api/telemetry-feed?timestamp=' + this.state.telemetryLastTimestamp)
|
|
||||||
.then(res => res.json())
|
|
||||||
.then(res => {
|
|
||||||
if ('telemetries' in res) {
|
|
||||||
let newTelem = this.state.telemetry.concat(res['telemetries']);
|
|
||||||
this.setState(
|
|
||||||
{
|
|
||||||
telemetry: newTelem,
|
|
||||||
telemetryLastTimestamp: res['timestamp'],
|
|
||||||
telemetryUpdateInProgress: false
|
|
||||||
});
|
|
||||||
this.props.onStatusChange();
|
|
||||||
|
|
||||||
let telemConsoleRef = this.telemConsole.current;
|
|
||||||
if (!this.state.isScrolledUp) {
|
|
||||||
telemConsoleRef.scrollTop = telemConsoleRef.scrollHeight - telemConsoleRef.clientHeight;
|
|
||||||
this.scrollTop = telemConsoleRef.scrollTop;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
selectionChanged(event) {
|
selectionChanged(event) {
|
||||||
if (event.nodes.length === 1) {
|
if (event.nodes.length === 1) {
|
||||||
this.authFetch('/api/netmap/node?id=' + event.nodes[0])
|
this.authFetch('/api/netmap/node?id=' + event.nodes[0])
|
||||||
|
@ -157,46 +117,6 @@ class MapPageComponent extends AuthComponent {
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
renderTelemetryEntry(telemetry) {
|
|
||||||
return (
|
|
||||||
<div key={telemetry.id}>
|
|
||||||
<span className="date">{telemetry.timestamp}</span>
|
|
||||||
<span className="source"> {telemetry.hostname}:</span>
|
|
||||||
<span className="event"> {telemetry.brief}</span>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
handleScroll(e) {
|
|
||||||
let element = e.target;
|
|
||||||
|
|
||||||
let telemetryStyle = window.getComputedStyle(element);
|
|
||||||
let telemetryLineHeight = parseInt((telemetryStyle.lineHeight).replace('px', ''));
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
isScrolledUp: (element.scrollTop < this.scrollTop),
|
|
||||||
telemetryCurrentLine: Math.trunc(element.scrollTop / telemetryLineHeight) + 1,
|
|
||||||
telemetryLines: Math.trunc(element.scrollHeight / telemetryLineHeight)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
renderTelemetryConsole() {
|
|
||||||
return (
|
|
||||||
<div className="telemetry-console" onScroll={this.handleScroll} ref={this.telemConsole}>
|
|
||||||
{
|
|
||||||
this.state.telemetry.map(this.renderTelemetryEntry)
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
renderTelemetryLineCount() {
|
|
||||||
return (
|
|
||||||
<div className="telemetry-lines">
|
|
||||||
<b>[{this.state.telemetryCurrentLine}/{this.state.telemetryLines}]</b>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
return (
|
return (
|
||||||
|
@ -220,10 +140,9 @@ class MapPageComponent extends AuthComponent {
|
||||||
<span>Island Communication <FontAwesomeIcon icon={faMinus} size="lg" style={{color: '#a9aaa9'}}/></span>
|
<span>Island Communication <FontAwesomeIcon icon={faMinus} size="lg" style={{color: '#a9aaa9'}}/></span>
|
||||||
</div>
|
</div>
|
||||||
<div style={{height: '80vh'}} className={'map-window'}>
|
<div style={{height: '80vh'}} className={'map-window'}>
|
||||||
{this.renderTelemetryLineCount()}
|
|
||||||
{this.renderTelemetryConsole()}
|
|
||||||
<ReactiveGraph graph={this.state.graph} options={getOptions(this.state.nodeStateList)}
|
<ReactiveGraph graph={this.state.graph} options={getOptions(this.state.nodeStateList)}
|
||||||
events={this.events}/>
|
events={this.events}/>
|
||||||
|
<TelemetryLog onStatusChange={this.props.onStatusChange}/>
|
||||||
</div>
|
</div>
|
||||||
</Col>
|
</Col>
|
||||||
<Col xs={4}>
|
<Col xs={4}>
|
||||||
|
|
Loading…
Reference in New Issue