2016-12-09 14:02:36 +01:00
|
|
|
/* eslint react/no-multi-comp:off */
|
|
|
|
import React, { Component, PureComponent } from 'react';
|
2016-12-05 15:15:01 +01:00
|
|
|
|
|
|
|
import { Link } from 'react-router';
|
2016-12-13 20:54:53 +01:00
|
|
|
import {
|
2017-02-14 12:11:18 +01:00
|
|
|
Grid, Cell, Card, CardTitle, CardText, CardMenu,
|
2016-12-13 20:54:53 +01:00
|
|
|
List, ListItem, ListItemContent,
|
|
|
|
Textfield, Icon, ProgressBar,
|
|
|
|
Tabs, Tab,
|
2016-12-13 22:46:56 +01:00
|
|
|
Switch,
|
2016-12-13 20:54:53 +01:00
|
|
|
} from 'react-mdl';
|
2017-02-14 12:11:18 +01:00
|
|
|
import { IconLink, shorten, styles as commonStyles } from '../common';
|
2017-01-06 11:41:52 +01:00
|
|
|
import { formatFullDateTime } from '../common/util';
|
2016-12-05 15:15:01 +01:00
|
|
|
|
2016-12-09 14:02:36 +01:00
|
|
|
class StatefulTextfield extends Component {
|
|
|
|
constructor (props) {
|
|
|
|
super(props);
|
|
|
|
this.state = { value: props.value };
|
|
|
|
this.setValue = function setValue (e) {
|
|
|
|
this.setState({ value: e.target.value });
|
|
|
|
}.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
render () {
|
2016-12-10 15:39:03 +01:00
|
|
|
return (<Textfield
|
|
|
|
style={{ width: '100%' }}
|
|
|
|
label={this.props.label}
|
2016-12-09 14:02:36 +01:00
|
|
|
floatingLabel
|
|
|
|
rows={this.props.rows}
|
|
|
|
value={this.state.value}
|
|
|
|
onChange={this.setValue}
|
|
|
|
onBlur={this.props.onBlur} />
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-13 20:54:53 +01:00
|
|
|
class ClientApplications extends PureComponent {
|
|
|
|
constructor (props) {
|
|
|
|
super(props);
|
|
|
|
this.state = { activeTab: 0 };
|
|
|
|
}
|
|
|
|
|
2016-12-05 15:15:01 +01:00
|
|
|
componentDidMount () {
|
|
|
|
this.props.fetchApplication(this.props.appName);
|
|
|
|
}
|
|
|
|
|
2016-12-05 16:01:09 +01:00
|
|
|
render () {
|
2016-12-05 15:15:01 +01:00
|
|
|
if (!this.props.application) {
|
2016-12-10 14:19:52 +01:00
|
|
|
return <ProgressBar indeterminate />;
|
2016-12-05 15:15:01 +01:00
|
|
|
}
|
2016-12-09 14:02:36 +01:00
|
|
|
const {
|
|
|
|
application,
|
|
|
|
storeApplicationMetaData,
|
|
|
|
} = this.props;
|
2016-12-05 15:15:01 +01:00
|
|
|
const {
|
|
|
|
appName,
|
|
|
|
instances,
|
|
|
|
strategies,
|
|
|
|
seenToggles,
|
2016-12-09 22:11:05 +01:00
|
|
|
url,
|
|
|
|
description,
|
|
|
|
icon = 'apps',
|
|
|
|
color,
|
2016-12-09 14:02:36 +01:00
|
|
|
} = application;
|
2016-12-05 16:01:09 +01:00
|
|
|
|
2016-12-13 20:54:53 +01:00
|
|
|
const content = this.state.activeTab === 0 ? (
|
|
|
|
<Grid>
|
2016-12-22 18:42:33 +01:00
|
|
|
<Cell col={6} tablet={4} phone={12}>
|
2016-12-13 20:54:53 +01:00
|
|
|
<h6> Toggles</h6>
|
|
|
|
<hr />
|
|
|
|
<List>
|
2016-12-13 22:46:56 +01:00
|
|
|
{seenToggles.map(({ name, description, enabled, notFound }, i) =>
|
2016-12-14 12:50:23 +01:00
|
|
|
(notFound ?
|
2016-12-13 22:46:56 +01:00
|
|
|
<ListItem twoLine key={i}>
|
|
|
|
<ListItemContent icon={'report'} subtitle={'Missing, want to create?'}>
|
|
|
|
<Link to={`/features/create?name=${name}`}>
|
|
|
|
{name}
|
|
|
|
</Link>
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem> :
|
|
|
|
<ListItem twoLine key={i}>
|
2016-12-23 08:52:09 +01:00
|
|
|
<ListItemContent
|
|
|
|
icon={<span><Switch disabled checked={!!enabled} /></span>}
|
|
|
|
subtitle={shorten(description, 60)}>
|
2016-12-22 10:43:25 +01:00
|
|
|
<Link to={`/features/view/${name}`}>
|
2016-12-23 08:48:29 +01:00
|
|
|
{shorten(name, 50)}
|
2016-12-13 20:54:53 +01:00
|
|
|
</Link>
|
|
|
|
</ListItemContent>
|
2016-12-14 12:50:23 +01:00
|
|
|
</ListItem>)
|
|
|
|
)}
|
2016-12-13 20:54:53 +01:00
|
|
|
</List>
|
|
|
|
</Cell>
|
2016-12-22 18:42:33 +01:00
|
|
|
<Cell col={6} tablet={4} phone={12}>
|
2016-12-13 20:54:53 +01:00
|
|
|
<h6>Implemented strategies</h6>
|
|
|
|
<hr />
|
|
|
|
<List>
|
2016-12-13 22:46:56 +01:00
|
|
|
{strategies.map(({ name, description, notFound }, i) => (
|
|
|
|
notFound ?
|
|
|
|
<ListItem twoLine key={`${name}-${i}`}>
|
|
|
|
<ListItemContent icon={'report'} subtitle={'Missing, want to create?'}>
|
|
|
|
<Link to={`/strategies/create?name=${name}`}>
|
|
|
|
{name}
|
|
|
|
</Link>
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem> :
|
|
|
|
<ListItem twoLine key={`${name}-${i}`}>
|
2016-12-23 08:48:29 +01:00
|
|
|
<ListItemContent icon={'extension'} subtitle={shorten(description, 60)}>
|
2016-12-13 20:54:53 +01:00
|
|
|
<Link to={`/strategies/view/${name}`}>
|
2016-12-23 08:48:29 +01:00
|
|
|
{shorten(name, 50)}
|
2016-12-09 14:02:36 +01:00
|
|
|
</Link>
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem>
|
|
|
|
))}
|
2016-12-13 20:54:53 +01:00
|
|
|
</List>
|
|
|
|
</Cell>
|
2016-12-22 18:42:33 +01:00
|
|
|
<Cell col={12} tablet={12}>
|
2016-12-22 09:56:16 +01:00
|
|
|
<h6>{instances.length} Instances registered</h6>
|
2016-12-13 20:54:53 +01:00
|
|
|
<hr />
|
|
|
|
<List>
|
|
|
|
{instances.map(({ instanceId, clientIp, lastSeen }, i) => (
|
|
|
|
<ListItem key={i} twoLine>
|
|
|
|
<ListItemContent
|
|
|
|
icon="timeline"
|
|
|
|
subtitle={
|
2017-01-06 11:41:52 +01:00
|
|
|
<span>{clientIp} last seen at <small>{formatFullDateTime(lastSeen)}</small></span>
|
2016-12-13 20:54:53 +01:00
|
|
|
}>
|
|
|
|
{instanceId}
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem>
|
|
|
|
))}
|
|
|
|
</List>
|
|
|
|
</Cell>
|
|
|
|
</Grid>) : (
|
|
|
|
<Grid>
|
|
|
|
<Cell col={12}>
|
|
|
|
<h5>Edit app meta data</h5>
|
|
|
|
</Cell>
|
2016-12-17 22:24:47 +01:00
|
|
|
<Cell col={6} tablet={12}>
|
2016-12-13 20:54:53 +01:00
|
|
|
<StatefulTextfield
|
|
|
|
value={url} label="URL" onBlur={(e) => storeApplicationMetaData(appName, 'url', e.target.value)} /><br />
|
|
|
|
<StatefulTextfield
|
|
|
|
value={description}
|
|
|
|
label="Description" rows={5} onBlur={(e) => storeApplicationMetaData(appName, 'description', e.target.value)} />
|
|
|
|
</Cell>
|
2016-12-17 22:24:47 +01:00
|
|
|
<Cell col={6} tablet={12}>
|
2016-12-13 20:54:53 +01:00
|
|
|
<StatefulTextfield
|
|
|
|
value={icon} label="Select icon" onBlur={(e) => storeApplicationMetaData(appName, 'icon', e.target.value)} />
|
|
|
|
<StatefulTextfield
|
|
|
|
value={color} label="Select color" onBlur={(e) => storeApplicationMetaData(appName, 'color', e.target.value)} />
|
|
|
|
</Cell>
|
|
|
|
</Grid>);
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
2017-02-14 12:11:18 +01:00
|
|
|
<Card shadow={0} className={commonStyles.fullwidth}>
|
|
|
|
<CardTitle style={{ paddingTop: '24px', paddingRight: '64px', wordBreak: 'break-all' }}>
|
|
|
|
<Icon name={icon} /> {appName}
|
|
|
|
</CardTitle>
|
|
|
|
{description &&
|
|
|
|
<CardText>{description}</CardText>
|
|
|
|
}
|
|
|
|
{url &&
|
|
|
|
<CardMenu><IconLink url={url} icon="link"/></CardMenu>
|
|
|
|
}
|
|
|
|
<hr className={commonStyles.divider}/>
|
|
|
|
<Tabs activeTab={this.state.activeTab} onChange={(tabId) => this.setState({ activeTab: tabId })} ripple
|
|
|
|
tabBarProps={{ style: { width: '100%' } }} className="mdl-color--grey-100">
|
|
|
|
<Tab>Details</Tab>
|
|
|
|
<Tab>Edit</Tab>
|
|
|
|
</Tabs>
|
2016-12-09 14:02:36 +01:00
|
|
|
|
2017-02-14 12:11:18 +01:00
|
|
|
{content}
|
|
|
|
</Card>
|
2016-12-05 15:15:01 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-13 20:54:53 +01:00
|
|
|
export default ClientApplications;
|