Added Logs navigation
This commit is contained in:
		@@ -1,24 +1,17 @@
 | 
				
			|||||||
import { Box, Container, IconButton, Menu, MenuItem, MenuList, Paper, Grow, Popper } from '@material-ui/core';
 | 
					import { Box, Paper } from '@material-ui/core';
 | 
				
			||||||
import AccountCircle from '@material-ui/icons/AccountCircle';
 | 
					import { blueGrey, grey, orange } from '@material-ui/core/colors';
 | 
				
			||||||
import AppBar from '@material-ui/core/AppBar';
 | 
					 | 
				
			||||||
import { positions } from '@material-ui/system';
 | 
					 | 
				
			||||||
import { createMuiTheme, createStyles, makeStyles, Theme } from '@material-ui/core/styles';
 | 
					import { createMuiTheme, createStyles, makeStyles, Theme } from '@material-ui/core/styles';
 | 
				
			||||||
import Toolbar from '@material-ui/core/Toolbar';
 | 
					 | 
				
			||||||
import Typography from '@material-ui/core/Typography';
 | 
					 | 
				
			||||||
import { ThemeProvider } from '@material-ui/styles';
 | 
					import { ThemeProvider } from '@material-ui/styles';
 | 
				
			||||||
import React, { useState, } from 'react';
 | 
					import React, { useState } from 'react';
 | 
				
			||||||
import { BrowserRouter, NavLink, Redirect, Route, Switch, Link } from 'react-router-dom';
 | 
					import { BrowserRouter, Redirect, Route, Switch } from 'react-router-dom';
 | 
				
			||||||
import BirdmapTitle from './components/appBar/BirdmapTitle';
 | 
					import BirdmapBar from './components/appBar/BirdmapBar';
 | 
				
			||||||
import Auth from './components/auth/Auth';
 | 
					import Auth from './components/auth/Auth';
 | 
				
			||||||
import AuthService from './components/auth/AuthService';
 | 
					import AuthService from './components/auth/AuthService';
 | 
				
			||||||
import { ClickAwayListener } from '@material-ui/core';
 | 
					 | 
				
			||||||
import MapContainer from './components/heatmap/Heatmap';
 | 
					 | 
				
			||||||
import Devices from './components/devices/Devices';
 | 
					 | 
				
			||||||
import { blueGrey, blue, orange, grey } from '@material-ui/core/colors';
 | 
					 | 
				
			||||||
import DevicesContextProvider from './contexts/DevicesContextProvider'
 | 
					 | 
				
			||||||
import Dashboard from './components/dashboard/Dashboard';
 | 
					import Dashboard from './components/dashboard/Dashboard';
 | 
				
			||||||
 | 
					import Devices from './components/devices/Devices';
 | 
				
			||||||
 | 
					import MapContainer from './components/heatmap/Heatmap';
 | 
				
			||||||
import Logs from './components/logs/Logs';
 | 
					import Logs from './components/logs/Logs';
 | 
				
			||||||
 | 
					import DevicesContextProvider from './contexts/DevicesContextProvider';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const theme = createMuiTheme({
 | 
					const theme = createMuiTheme({
 | 
				
			||||||
    palette: {
 | 
					    palette: {
 | 
				
			||||||
@@ -27,14 +20,13 @@ const theme = createMuiTheme({
 | 
				
			|||||||
            dark: grey[400],
 | 
					            dark: grey[400],
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        secondary: {
 | 
					        secondary: {
 | 
				
			||||||
            main: orange[200],
 | 
					            main: blueGrey[700],
 | 
				
			||||||
            dark: blueGrey[50],
 | 
					            dark: blueGrey[50],
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function App() {
 | 
					function App() {
 | 
				
			||||||
 | 
					 | 
				
			||||||
    const [authenticated, setAuthenticated] = useState(AuthService.isAuthenticated());
 | 
					    const [authenticated, setAuthenticated] = useState(AuthService.isAuthenticated());
 | 
				
			||||||
    const [isAdmin, setIsAdmin] = useState(AuthService.isAdmin());
 | 
					    const [isAdmin, setIsAdmin] = useState(AuthService.isAdmin());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -61,6 +53,7 @@ function App() {
 | 
				
			|||||||
        return <Devices isAdmin={isAdmin}/>;
 | 
					        return <Devices isAdmin={isAdmin}/>;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const HeatmapComponent = () => {
 | 
					    const HeatmapComponent = () => {
 | 
				
			||||||
        return (
 | 
					        return (
 | 
				
			||||||
            <Paper elevation={0}>
 | 
					            <Paper elevation={0}>
 | 
				
			||||||
@@ -69,16 +62,46 @@ function App() {
 | 
				
			|||||||
        );
 | 
					        );
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const HeaderComponent = () => {
 | 
				
			||||||
 | 
					        return (
 | 
				
			||||||
 | 
					            <BirdmapBar onLogout={AuthService.logout} isAdmin={isAdmin} isAuthenticated={authenticated}/>
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const PredicateRoute = ({ component: Component, predicate: Predicate, ...rest }: { [x: string]: any, component: any, predicate: any }) => {
 | 
				
			||||||
 | 
					        return (
 | 
				
			||||||
 | 
					            <PredicateRouteInternal {...rest} header={HeaderComponent} body={Component} predicate={Predicate}/>
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const PublicRoute = ({ component: Component, ...rest }: { [x: string]: any, component: any }) => {
 | 
				
			||||||
 | 
					        return (
 | 
				
			||||||
 | 
					            <PredicateRoute {...rest} component={Component} predicate={true}/>
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const PrivateRoute = ({ component: Component, ...rest }: { [x: string]: any, component: any }) => {
 | 
				
			||||||
 | 
					        return (
 | 
				
			||||||
 | 
					            <PredicateRoute {...rest} component={Component} predicate={authenticated}/>
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const AdminRoute = ({ component: Component, ...rest }: { [x: string]: any, component: any }) => {
 | 
				
			||||||
 | 
					        return (
 | 
				
			||||||
 | 
					            <PredicateRoute {...rest} component={Component} predicate={authenticated && isAdmin}/>
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return (
 | 
					    return (
 | 
				
			||||||
        <ThemeProvider theme={theme}>
 | 
					        <ThemeProvider theme={theme}>
 | 
				
			||||||
            <BrowserRouter>
 | 
					            <BrowserRouter>
 | 
				
			||||||
                <Switch>
 | 
					                <Switch>
 | 
				
			||||||
                    <PublicRoute path="/login" exact component={AuthComponent} />
 | 
					                    <PublicRoute      exact path="/login"           component={AuthComponent} />
 | 
				
			||||||
                    <AdminRoute path="/logs" exact authenticated={authenticated} isAdmin={isAdmin} component={LogsComponent} />
 | 
					                    <AdminRoute       exact path="/logs"            component={LogsComponent} />
 | 
				
			||||||
                    <DevicesContextProvider>
 | 
					                    <DevicesContextProvider>
 | 
				
			||||||
                        <PrivateRoute path="/" exact authenticated={authenticated} component={DashboardComponent} />
 | 
					                        <PrivateRoute exact path="/"                component={DashboardComponent} />
 | 
				
			||||||
                        <PrivateRoute path="/devices/:id?" exact authenticated={authenticated} component={DevicesComponent} />
 | 
					                        <PrivateRoute exact path="/devices/:id?"    component={DevicesComponent} />
 | 
				
			||||||
                        <PrivateRoute path="/heatmap" exact authenticated={authenticated} component={HeatmapComponent} />
 | 
					                        <PrivateRoute exact path="/heatmap"         component={HeatmapComponent} />
 | 
				
			||||||
                    </DevicesContextProvider>
 | 
					                    </DevicesContextProvider>
 | 
				
			||||||
                </Switch>
 | 
					                </Switch>
 | 
				
			||||||
            </BrowserRouter>
 | 
					            </BrowserRouter>
 | 
				
			||||||
@@ -88,123 +111,26 @@ function App() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
export default App;
 | 
					export default App;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const PublicRoute = ({ component: Component, ...rest }: { [x: string]: any, component: any}) => {
 | 
					const PredicateRouteInternal = ({ header: HeaderComponent, body: BodyComponent, predicate: Predicate, ...rest }: { [x: string]: any, header: any, body: any, predicate: any }) => {
 | 
				
			||||||
    return (
 | 
					    return (
 | 
				
			||||||
        <Route {...rest} render={matchProps => (
 | 
					        <Route {...rest} render={matchProps => (
 | 
				
			||||||
            <DefaultLayout component={Component} authenticated={false} isAdmin={false} {...matchProps} />
 | 
					            Predicate
 | 
				
			||||||
        )} />
 | 
					                ? <DefaultLayoutInternal header={HeaderComponent} body={BodyComponent} {...matchProps} />
 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const AdminRoute = ({ component: Component, authenticated: Authenticated, isAdmin: IsAdmin, ...rest }: { [x: string]: any, component: any, authenticated: any, isAdmin: any }) => {
 | 
					 | 
				
			||||||
    return (
 | 
					 | 
				
			||||||
        <Route {...rest} render={matchProps => (
 | 
					 | 
				
			||||||
            Authenticated && IsAdmin
 | 
					 | 
				
			||||||
                ? <DefaultLayout component={Component} authenticated={Authenticated} {...matchProps} />
 | 
					 | 
				
			||||||
                : <Redirect to='/login' />
 | 
					                : <Redirect to='/login' />
 | 
				
			||||||
        )} />
 | 
					        )} />
 | 
				
			||||||
    );
 | 
					    );
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const PrivateRoute = ({ component: Component, authenticated: Authenticated, ...rest }: { [x: string]: any, component: any, authenticated: any }) => {
 | 
					const DefaultLayoutInternal = ({ header: HeaderComponent, body: BodyComponent, ...rest }: { [x: string]: any, header: any, body: any }) => {
 | 
				
			||||||
    return (
 | 
					 | 
				
			||||||
        <Route {...rest} render={matchProps => (
 | 
					 | 
				
			||||||
            Authenticated
 | 
					 | 
				
			||||||
                ? <DefaultLayout component={Component} authenticated={Authenticated} {...matchProps} />
 | 
					 | 
				
			||||||
                : <Redirect to='/login' />
 | 
					 | 
				
			||||||
        )} />
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const DefaultLayout = ({ component: Component, authenticated: Authenticated, ...rest }: { [x: string]: any, component: any, authenticated: any }) => {
 | 
					 | 
				
			||||||
    const classes = useDefaultLayoutStyles();
 | 
					    const classes = useDefaultLayoutStyles();
 | 
				
			||||||
    const [open, setOpen] = React.useState(false);
 | 
					 | 
				
			||||||
    const anchorRef = React.useRef<HTMLButtonElement>(null);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const handleToggle = () => {
 | 
					 | 
				
			||||||
        setOpen((prevOpen) => !prevOpen);
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const handleClose = (event: React.MouseEvent<EventTarget>) => {
 | 
					 | 
				
			||||||
        if (anchorRef.current && anchorRef.current.contains(event.target as HTMLElement)) {
 | 
					 | 
				
			||||||
            return;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        setOpen(false);
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const handleLogout = (event: React.MouseEvent<EventTarget>) => {
 | 
					 | 
				
			||||||
        if (anchorRef.current && anchorRef.current.contains(event.target as HTMLElement)) {
 | 
					 | 
				
			||||||
            return;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        AuthService.logout();
 | 
					 | 
				
			||||||
        setOpen(false);
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    function handleListKeyDown(event: React.KeyboardEvent) {
 | 
					 | 
				
			||||||
        if (event.key === 'Tab') {
 | 
					 | 
				
			||||||
            event.preventDefault();
 | 
					 | 
				
			||||||
            setOpen(false);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const prevOpen = React.useRef(open);
 | 
					 | 
				
			||||||
    React.useEffect(() => {
 | 
					 | 
				
			||||||
        if (prevOpen.current === true && open === false) {
 | 
					 | 
				
			||||||
            anchorRef.current!.focus();
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        prevOpen.current = open;
 | 
					 | 
				
			||||||
    }, [open]);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    const renderNavLinks = () => {
 | 
					 | 
				
			||||||
        return Authenticated
 | 
					 | 
				
			||||||
            ? <Container className={classes.nav_menu}>
 | 
					 | 
				
			||||||
                <NavLink exact to="/" className={classes.nav_menu_item} activeClassName={classes.nav_menu_item_active}>Dashboard</NavLink>
 | 
					 | 
				
			||||||
                {}
 | 
					 | 
				
			||||||
                <NavLink to="/devices" className={classes.nav_menu_item} activeClassName={classes.nav_menu_item_active}>Devices</NavLink>
 | 
					 | 
				
			||||||
                <NavLink exact to="/heatmap" className={classes.nav_menu_item} activeClassName={classes.nav_menu_item_active}>Heatmap</NavLink>
 | 
					 | 
				
			||||||
                <IconButton className={classes.nav_menu_icon}
 | 
					 | 
				
			||||||
                    ref={anchorRef}
 | 
					 | 
				
			||||||
                    aria-haspopup="true"
 | 
					 | 
				
			||||||
                    aria-controls={open ? 'menu-list-grow' : undefined}
 | 
					 | 
				
			||||||
                    aria-label="account of current user"
 | 
					 | 
				
			||||||
                    onClick={handleToggle}>
 | 
					 | 
				
			||||||
                    <AccountCircle/>
 | 
					 | 
				
			||||||
                </IconButton>
 | 
					 | 
				
			||||||
                <Popper open={open} anchorEl={anchorRef.current} role={undefined} transition disablePortal>
 | 
					 | 
				
			||||||
                    {({ TransitionProps, placement }) => (
 | 
					 | 
				
			||||||
                        <Grow
 | 
					 | 
				
			||||||
                            {...TransitionProps}
 | 
					 | 
				
			||||||
                            style={{ transformOrigin: placement === 'bottom' ? 'center top' : 'center bottom' }}>
 | 
					 | 
				
			||||||
                            <Paper>
 | 
					 | 
				
			||||||
                                <ClickAwayListener onClickAway={handleClose}>
 | 
					 | 
				
			||||||
                                    <MenuList autoFocusItem={open} id="menu-list-grow" onKeyDown={handleListKeyDown}>
 | 
					 | 
				
			||||||
                                        <MenuItem onClick={handleLogout} component={Link} {...{ to: '/login' }}>Logout</MenuItem>
 | 
					 | 
				
			||||||
                                    </MenuList>
 | 
					 | 
				
			||||||
                                </ClickAwayListener>
 | 
					 | 
				
			||||||
                            </Paper>
 | 
					 | 
				
			||||||
                        </Grow>
 | 
					 | 
				
			||||||
                    )}
 | 
					 | 
				
			||||||
                </Popper>
 | 
					 | 
				
			||||||
            </Container>
 | 
					 | 
				
			||||||
            : null;
 | 
					 | 
				
			||||||
    };
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return (
 | 
					    return (
 | 
				
			||||||
        <React.Fragment>
 | 
					        <React.Fragment>
 | 
				
			||||||
            <AppBar position="static" className={classes.bar_root}>
 | 
					            <Box className={classes.header}>
 | 
				
			||||||
                <Toolbar>
 | 
					                <HeaderComponent />
 | 
				
			||||||
                    <BirdmapTitle />
 | 
					            </Box>
 | 
				
			||||||
                    <Typography component={'span'} className={classes.typo}>
 | 
					            <Box className={classes.body}>
 | 
				
			||||||
                        {renderNavLinks()}
 | 
					                <BodyComponent {...rest} />
 | 
				
			||||||
                    </Typography>
 | 
					 | 
				
			||||||
                </Toolbar>
 | 
					 | 
				
			||||||
            </AppBar>
 | 
					 | 
				
			||||||
            <Box zIndex="modal" className={classes.box_root}>
 | 
					 | 
				
			||||||
                <Component {...rest} />
 | 
					 | 
				
			||||||
            </Box>
 | 
					            </Box>
 | 
				
			||||||
        </React.Fragment>
 | 
					        </React.Fragment>
 | 
				
			||||||
    );
 | 
					    );
 | 
				
			||||||
@@ -212,46 +138,12 @@ const DefaultLayout = ({ component: Component, authenticated: Authenticated, ...
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const useDefaultLayoutStyles = makeStyles((theme: Theme) =>
 | 
					const useDefaultLayoutStyles = makeStyles((theme: Theme) =>
 | 
				
			||||||
    createStyles({
 | 
					    createStyles({
 | 
				
			||||||
        bar_root: {
 | 
					        header: {
 | 
				
			||||||
            height: '7%',
 | 
					            height: '7%',
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        box_root: {
 | 
					        body: {
 | 
				
			||||||
            backgroundColor: theme.palette.primary.dark,
 | 
					            backgroundColor: theme.palette.primary.dark,
 | 
				
			||||||
            height: '93%',
 | 
					            height: '93%',
 | 
				
			||||||
        },
 | 
					        }
 | 
				
			||||||
        typo: {
 | 
					 | 
				
			||||||
            marginLeft: 'auto',
 | 
					 | 
				
			||||||
            color: 'white',
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        nav_menu: {
 | 
					 | 
				
			||||||
            display: 'flex',
 | 
					 | 
				
			||||||
            flexDirection: 'row',
 | 
					 | 
				
			||||||
            alignItems: 'center',
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        nav_menu_icon: {
 | 
					 | 
				
			||||||
            color: 'inherit',
 | 
					 | 
				
			||||||
            marginLeft: '24px',
 | 
					 | 
				
			||||||
            '&:hover': {
 | 
					 | 
				
			||||||
                color: 'inherit',
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        nav_menu_item: {
 | 
					 | 
				
			||||||
            textDecoration: 'none',
 | 
					 | 
				
			||||||
            fontWeight: 'normal',
 | 
					 | 
				
			||||||
            color: 'inherit',
 | 
					 | 
				
			||||||
            marginLeft: '24px',
 | 
					 | 
				
			||||||
            '&:hover': {
 | 
					 | 
				
			||||||
                color: 'inherit',
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        nav_menu_item_active: {
 | 
					 | 
				
			||||||
            textDecoration: 'underline',
 | 
					 | 
				
			||||||
            fontWeight: 'bold',
 | 
					 | 
				
			||||||
            color: 'inherit',
 | 
					 | 
				
			||||||
            marginLeft: '24px',
 | 
					 | 
				
			||||||
            '&:hover': {
 | 
					 | 
				
			||||||
                color: 'inherit',
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
    }),
 | 
					    }),
 | 
				
			||||||
);
 | 
					);
 | 
				
			||||||
							
								
								
									
										136
									
								
								Birdmap.API/ClientApp/src/components/appBar/BirdmapBar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								Birdmap.API/ClientApp/src/components/appBar/BirdmapBar.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,136 @@
 | 
				
			|||||||
 | 
					import { ClickAwayListener, Container, createStyles, Grow, IconButton, makeStyles, MenuItem, MenuList, Paper, Popper, Theme } from '@material-ui/core';
 | 
				
			||||||
 | 
					import AppBar from '@material-ui/core/AppBar';
 | 
				
			||||||
 | 
					import Toolbar from '@material-ui/core/Toolbar';
 | 
				
			||||||
 | 
					import Typography from '@material-ui/core/Typography';
 | 
				
			||||||
 | 
					import AccountCircle from '@material-ui/icons/AccountCircle';
 | 
				
			||||||
 | 
					import React from 'react';
 | 
				
			||||||
 | 
					import { Link, NavLink } from 'react-router-dom';
 | 
				
			||||||
 | 
					import BirdmapTitle from './BirdmapTitle';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default function BirdmapBar(props: { onLogout: () => void; isAuthenticated: any; isAdmin: any; }) {
 | 
				
			||||||
 | 
					    const classes = useAppbarStyles();
 | 
				
			||||||
 | 
					    const [open, setOpen] = React.useState(false);
 | 
				
			||||||
 | 
					    const anchorRef = React.useRef<HTMLButtonElement>(null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const handleToggle = () => {
 | 
				
			||||||
 | 
					        setOpen((prevOpen) => !prevOpen);
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const handleClose = (event: React.MouseEvent<EventTarget>) => {
 | 
				
			||||||
 | 
					        if (anchorRef.current && anchorRef.current.contains(event.target as HTMLElement)) {
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        setOpen(false);
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const handleLogout = (event: React.MouseEvent<EventTarget>) => {
 | 
				
			||||||
 | 
					        if (anchorRef.current && anchorRef.current.contains(event.target as HTMLElement)) {
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        props.onLogout();
 | 
				
			||||||
 | 
					        setOpen(false);
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function handleListKeyDown(event: React.KeyboardEvent) {
 | 
				
			||||||
 | 
					        if (event.key === 'Tab') {
 | 
				
			||||||
 | 
					            event.preventDefault();
 | 
				
			||||||
 | 
					            setOpen(false);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const prevOpen = React.useRef(open);
 | 
				
			||||||
 | 
					    React.useEffect(() => {
 | 
				
			||||||
 | 
					        if (prevOpen.current === true && open === false) {
 | 
				
			||||||
 | 
					            anchorRef.current!.focus();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        prevOpen.current = open;
 | 
				
			||||||
 | 
					    }, [open]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const renderNavLinks = () => {
 | 
				
			||||||
 | 
					        return props.isAuthenticated
 | 
				
			||||||
 | 
					            ? <Container className={classes.nav_menu}>
 | 
				
			||||||
 | 
					                <NavLink exact to="/" className={classes.nav_menu_item} activeClassName={classes.nav_menu_item_active}>Dashboard</NavLink>
 | 
				
			||||||
 | 
					                {props.isAdmin ? <NavLink exact to="/logs" className={classes.nav_menu_item} activeClassName={classes.nav_menu_item_active}>Logs</NavLink> : null}
 | 
				
			||||||
 | 
					                <NavLink to="/devices" className={classes.nav_menu_item} activeClassName={classes.nav_menu_item_active}>Devices</NavLink>
 | 
				
			||||||
 | 
					                <NavLink exact to="/heatmap" className={classes.nav_menu_item} activeClassName={classes.nav_menu_item_active}>Heatmap</NavLink>
 | 
				
			||||||
 | 
					                <IconButton className={classes.nav_menu_icon}
 | 
				
			||||||
 | 
					                    ref={anchorRef}
 | 
				
			||||||
 | 
					                    aria-haspopup="true"
 | 
				
			||||||
 | 
					                    aria-controls={open ? 'menu-list-grow' : undefined}
 | 
				
			||||||
 | 
					                    aria-label="account of current user"
 | 
				
			||||||
 | 
					                    onClick={handleToggle}>
 | 
				
			||||||
 | 
					                    <AccountCircle />
 | 
				
			||||||
 | 
					                </IconButton>
 | 
				
			||||||
 | 
					                <Popper open={open} anchorEl={anchorRef.current} role={undefined} transition disablePortal>
 | 
				
			||||||
 | 
					                    {({ TransitionProps, placement }) => (
 | 
				
			||||||
 | 
					                        <Grow
 | 
				
			||||||
 | 
					                            {...TransitionProps}
 | 
				
			||||||
 | 
					                            style={{ transformOrigin: placement === 'bottom' ? 'center top' : 'center bottom' }}>
 | 
				
			||||||
 | 
					                            <Paper>
 | 
				
			||||||
 | 
					                                <ClickAwayListener onClickAway={handleClose}>
 | 
				
			||||||
 | 
					                                    <MenuList autoFocusItem={open} id="menu-list-grow" onKeyDown={handleListKeyDown}>
 | 
				
			||||||
 | 
					                                        <MenuItem onClick={handleLogout} component={Link} {...{ to: '/login' }}>Logout</MenuItem>
 | 
				
			||||||
 | 
					                                    </MenuList>
 | 
				
			||||||
 | 
					                                </ClickAwayListener>
 | 
				
			||||||
 | 
					                            </Paper>
 | 
				
			||||||
 | 
					                        </Grow>
 | 
				
			||||||
 | 
					                    )}
 | 
				
			||||||
 | 
					                </Popper>
 | 
				
			||||||
 | 
					            </Container>
 | 
				
			||||||
 | 
					            : null;
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return (
 | 
				
			||||||
 | 
					        <AppBar position="static">
 | 
				
			||||||
 | 
					            <Toolbar>
 | 
				
			||||||
 | 
					                <BirdmapTitle />
 | 
				
			||||||
 | 
					                <Typography component={'span'} className={classes.typo}>
 | 
				
			||||||
 | 
					                    {renderNavLinks()}
 | 
				
			||||||
 | 
					                </Typography>
 | 
				
			||||||
 | 
					            </Toolbar>
 | 
				
			||||||
 | 
					        </AppBar>
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const useAppbarStyles = makeStyles((theme: Theme) =>
 | 
				
			||||||
 | 
					    createStyles({
 | 
				
			||||||
 | 
					        typo: {
 | 
				
			||||||
 | 
					            marginLeft: 'auto',
 | 
				
			||||||
 | 
					            color: 'white',
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        nav_menu: {
 | 
				
			||||||
 | 
					            display: 'flex',
 | 
				
			||||||
 | 
					            flexDirection: 'row',
 | 
				
			||||||
 | 
					            alignItems: 'center',
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        nav_menu_icon: {
 | 
				
			||||||
 | 
					            color: 'inherit',
 | 
				
			||||||
 | 
					            marginLeft: '24px',
 | 
				
			||||||
 | 
					            '&:hover': {
 | 
				
			||||||
 | 
					                color: 'inherit',
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        nav_menu_item: {
 | 
				
			||||||
 | 
					            textDecoration: 'none',
 | 
				
			||||||
 | 
					            fontWeight: 'normal',
 | 
				
			||||||
 | 
					            color: 'inherit',
 | 
				
			||||||
 | 
					            marginLeft: '24px',
 | 
				
			||||||
 | 
					            '&:hover': {
 | 
				
			||||||
 | 
					                color: 'inherit',
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        nav_menu_item_active: {
 | 
				
			||||||
 | 
					            textDecoration: 'underline',
 | 
				
			||||||
 | 
					            fontWeight: 'bold',
 | 
				
			||||||
 | 
					            color: 'inherit',
 | 
				
			||||||
 | 
					            marginLeft: '24px',
 | 
				
			||||||
 | 
					            '&:hover': {
 | 
				
			||||||
 | 
					                color: 'inherit',
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    }),
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
@@ -1,4 +1,5 @@
 | 
				
			|||||||
/*global google*/
 | 
					/*global google*/
 | 
				
			||||||
 | 
					import { Box, withStyles } from '@material-ui/core';
 | 
				
			||||||
import GoogleMapReact from 'google-map-react';
 | 
					import GoogleMapReact from 'google-map-react';
 | 
				
			||||||
import React, { Component } from 'react';
 | 
					import React, { Component } from 'react';
 | 
				
			||||||
import C from '../../common/Constants';
 | 
					import C from '../../common/Constants';
 | 
				
			||||||
@@ -8,7 +9,14 @@ import DeviceMarker from './DeviceMarker';
 | 
				
			|||||||
const lat_offset = 0.000038;
 | 
					const lat_offset = 0.000038;
 | 
				
			||||||
const lng_offset = -0.000058;
 | 
					const lng_offset = -0.000058;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default class MapContainer extends Component {
 | 
					const styles = theme => ({
 | 
				
			||||||
 | 
					    root: { 
 | 
				
			||||||
 | 
					        height: '93vh', 
 | 
				
			||||||
 | 
					        width: '100%', 
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class MapContainer extends Component {
 | 
				
			||||||
    constructor(props) {
 | 
					    constructor(props) {
 | 
				
			||||||
        super(props);
 | 
					        super(props);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -64,6 +72,8 @@ export default class MapContainer extends Component {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    render() {
 | 
					    render() {
 | 
				
			||||||
 | 
					        const {classes} = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        const heatMapData = {
 | 
					        const heatMapData = {
 | 
				
			||||||
            positions: this.state.heatmapPoints,
 | 
					            positions: this.state.heatmapPoints,
 | 
				
			||||||
            options: {
 | 
					            options: {
 | 
				
			||||||
@@ -92,7 +102,7 @@ export default class MapContainer extends Component {
 | 
				
			|||||||
        ));
 | 
					        ));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return (
 | 
					        return (
 | 
				
			||||||
            <div style={{ height: '93vh', width: '100%' }}>
 | 
					            <Box className={classes.root}>
 | 
				
			||||||
                <GoogleMapReact
 | 
					                <GoogleMapReact
 | 
				
			||||||
                    bootstrapURLKeys={{
 | 
					                    bootstrapURLKeys={{
 | 
				
			||||||
                        key: ["AIzaSyCZ51VFfxqZ2GkCmVrcNZdUKsM0fuBQUCY"],
 | 
					                        key: ["AIzaSyCZ51VFfxqZ2GkCmVrcNZdUKsM0fuBQUCY"],
 | 
				
			||||||
@@ -106,7 +116,9 @@ export default class MapContainer extends Component {
 | 
				
			|||||||
                    defaultCenter={this.state.center}>
 | 
					                    defaultCenter={this.state.center}>
 | 
				
			||||||
                    {Markers}
 | 
					                    {Markers}
 | 
				
			||||||
                </GoogleMapReact>
 | 
					                </GoogleMapReact>
 | 
				
			||||||
            </div>
 | 
					            </Box>
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default withStyles(styles)(MapContainer);
 | 
				
			||||||
@@ -1,8 +1,19 @@
 | 
				
			|||||||
import { Button, Checkbox, List, ListItem, ListItemIcon, ListItemText, Paper } from '@material-ui/core';
 | 
					import { Box, Button, Checkbox, List, ListItem, ListItemIcon, ListItemText, Paper, withStyles } from '@material-ui/core';
 | 
				
			||||||
 | 
					import { blueGrey } from '@material-ui/core/colors';
 | 
				
			||||||
import React, { Component } from 'react';
 | 
					import React, { Component } from 'react';
 | 
				
			||||||
import LogService from './LogService';
 | 
					import LogService from './LogService';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class Logs extends Component {
 | 
					const styles = theme => ({
 | 
				
			||||||
 | 
					    root: {
 | 
				
			||||||
 | 
					        padding: '64px',
 | 
				
			||||||
 | 
					        backgroundColor: theme.palette.primary.dark,
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    paper: {
 | 
				
			||||||
 | 
					        backgroundColor: blueGrey[50],
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Logs extends Component {
 | 
				
			||||||
    constructor(props) {
 | 
					    constructor(props) {
 | 
				
			||||||
        super(props)
 | 
					        super(props)
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
@@ -10,6 +21,7 @@ export class Logs extends Component {
 | 
				
			|||||||
            service: null,
 | 
					            service: null,
 | 
				
			||||||
            files: [],
 | 
					            files: [],
 | 
				
			||||||
            checked: [],
 | 
					            checked: [],
 | 
				
			||||||
 | 
					            selectAllChecked: false,
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
@@ -35,7 +47,18 @@ export class Logs extends Component {
 | 
				
			|||||||
        this.setState({checked: newChecked});
 | 
					        this.setState({checked: newChecked});
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    downloadChecked = () => {
 | 
					    handleSelectAllToggle = () => {
 | 
				
			||||||
 | 
					        this.setState({selectAllChecked: !this.state.selectAllChecked});
 | 
				
			||||||
 | 
					        if (this.state.selectAllChecked) {
 | 
				
			||||||
 | 
					            this.setState({checked: []});
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            const newChecked = [...this.state.files];
 | 
				
			||||||
 | 
					            this.setState({checked: newChecked});
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    onDownload = () => {
 | 
				
			||||||
        this.state.service.getFiles(this.state.checked)
 | 
					        this.state.service.getFiles(this.state.checked)
 | 
				
			||||||
        .then(result => {
 | 
					        .then(result => {
 | 
				
			||||||
            const filename = `Logs-${new Date().toISOString()}.zip`;
 | 
					            const filename = `Logs-${new Date().toISOString()}.zip`;
 | 
				
			||||||
@@ -47,11 +70,15 @@ export class Logs extends Component {
 | 
				
			|||||||
            document.body.appendChild(element);
 | 
					            document.body.appendChild(element);
 | 
				
			||||||
            element.click();
 | 
					            element.click();
 | 
				
			||||||
            document.body.removeChild(element);
 | 
					            document.body.removeChild(element);
 | 
				
			||||||
 | 
					            this.setState({checked: []});
 | 
				
			||||||
 | 
					            this.setState({selectAllChecked: false});
 | 
				
			||||||
        })
 | 
					        })
 | 
				
			||||||
        .catch(ex => console.log(ex));
 | 
					        .catch(ex => console.log(ex));
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    render() {
 | 
					    render() {
 | 
				
			||||||
 | 
					        const { classes } = this.props;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        const Files = this.state.files.map((value) => {
 | 
					        const Files = this.state.files.map((value) => {
 | 
				
			||||||
            const labelId = `checkbox-list-label-${value}`;
 | 
					            const labelId = `checkbox-list-label-${value}`;
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
@@ -72,16 +99,30 @@ export class Logs extends Component {
 | 
				
			|||||||
        })
 | 
					        })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return (
 | 
					        return (
 | 
				
			||||||
            <Paper>
 | 
					            <Box className={classes.root}>
 | 
				
			||||||
                <List >
 | 
					                <Paper className={classes.paper}>
 | 
				
			||||||
                    {Files}
 | 
					                    <List className={classes.paper}>
 | 
				
			||||||
                </List>
 | 
					                        <ListItem key="Select-all" role={undefined} dense button onClick={this.handleSelectAllToggle}>
 | 
				
			||||||
                <Button onClick={this.downloadChecked}>
 | 
					                            <ListItemIcon>
 | 
				
			||||||
                    Download
 | 
					                            <Checkbox
 | 
				
			||||||
                </Button>
 | 
					                                edge="start"
 | 
				
			||||||
            </Paper>
 | 
					                                checked={this.state.selectAllChecked}
 | 
				
			||||||
 | 
					                                tabIndex={-1}
 | 
				
			||||||
 | 
					                                disableRipple
 | 
				
			||||||
 | 
					                                inputProps={{ 'aria-labelledby': "Select-all" }}
 | 
				
			||||||
 | 
					                            />
 | 
				
			||||||
 | 
					                            </ListItemIcon>
 | 
				
			||||||
 | 
					                            <ListItemText id="checkbox-list-label-Select-all" primary={(this.state.selectAllChecked ? "Uns" : "S") + "elect all"} />
 | 
				
			||||||
 | 
					                        </ListItem>
 | 
				
			||||||
 | 
					                        {Files}
 | 
				
			||||||
 | 
					                    </List>
 | 
				
			||||||
 | 
					                    <Button onClick={this.onDownload}>
 | 
				
			||||||
 | 
					                        Download
 | 
				
			||||||
 | 
					                    </Button>
 | 
				
			||||||
 | 
					                </Paper>
 | 
				
			||||||
 | 
					            </Box>
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default Logs
 | 
					export default withStyles(styles)(Logs);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1 +1 @@
 | 
				
			|||||||
[{"Topic":"devices/output","Payload":"eyJ0YWciOiIxN2NmOGI4Mi1lZDQ4LTQ4MDctOTI3MS0xZDlmMGY5ZDViMWYiLCJwcm9iYWJpbGl0eSI6MC43MjA0MzAyMzAxMjU5ODUyfQ==","QualityOfServiceLevel":1,"Retain":true,"UserProperties":null,"ContentType":null,"ResponseTopic":null,"PayloadFormatIndicator":null,"MessageExpiryInterval":null,"TopicAlias":null,"CorrelationData":null,"SubscriptionIdentifiers":null}]
 | 
					[{"Topic":"devices/output","Payload":"eyJ0YWciOiJkNDhhODAwOC0wYjU2LTRkYzAtODYxMy0zMWY0MDY4OTk0ZDciLCJwcm9iYWJpbGl0eSI6MC4yNTMxNDI4NDgyNjMwOTU1fQ==","QualityOfServiceLevel":1,"Retain":true,"UserProperties":null,"ContentType":null,"ResponseTopic":null,"PayloadFormatIndicator":null,"MessageExpiryInterval":null,"TopicAlias":null,"CorrelationData":null,"SubscriptionIdentifiers":null}]
 | 
				
			||||||
		Reference in New Issue
	
	Block a user