import React, { useState } from 'react'; import { FormControl, InputLabel, Select, MenuItem, Table, TableHead, TableRow, TableCell, TableBody, Link, IconButton, } from '@mui/material'; import GpsFixedIcon from '@mui/icons-material/GpsFixed'; import LocationSearchingIcon from '@mui/icons-material/LocationSearching'; import { useSelector } from 'react-redux'; import { formatSpeed, formatTime } from '../common/util/formatter'; import ReportFilter from './components/ReportFilter'; import { prefixString } from '../common/util/stringUtils'; import { useTranslation } from '../common/components/LocalizationProvider'; import PageLayout from '../common/components/PageLayout'; import ReportsMenu from './components/ReportsMenu'; import usePersistedState from '../common/util/usePersistedState'; import ColumnSelect from './components/ColumnSelect'; import { useCatch, useEffectAsync } from '../reactHelper'; import useReportStyles from './common/useReportStyles'; import TableShimmer from '../common/components/TableShimmer'; import { useAttributePreference, usePreference } from '../common/util/preferences'; import MapView from '../map/core/MapView'; import MapGeofence from '../map/MapGeofence'; import MapPositions from '../map/MapPositions'; import MapCamera from '../map/MapCamera'; const columnsArray = [ ['eventTime', 'positionFixTime'], ['type', 'sharedType'], ['geofenceId', 'sharedGeofence'], ['maintenanceId', 'sharedMaintenance'], ['attributes', 'commandData'], ]; const columnsMap = new Map(columnsArray); const EventReportPage = () => { const classes = useReportStyles(); const t = useTranslation(); const devices = useSelector((state) => state.devices.items); const geofences = useSelector((state) => state.geofences.items); const speedUnit = useAttributePreference('speedUnit'); const hours12 = usePreference('twelveHourFormat'); const [allEventTypes, setAllEventTypes] = useState([['allEvents', 'eventAll']]); const [columns, setColumns] = usePersistedState('eventColumns', ['eventTime', 'type', 'attributes']); const [eventTypes, setEventTypes] = useState(['allEvents']); const [items, setItems] = useState([]); const [loading, setLoading] = useState(false); const [selectedItem, setSelectedItem] = useState(null); const [position, setPosition] = useState(null); useEffectAsync(async () => { if (selectedItem) { const response = await fetch(`/api/positions?id=${selectedItem.positionId}`); if (response.ok) { const positions = await response.json(); if (positions.length > 0) { setPosition(positions[0]); } } else { throw Error(await response.text()); } } else { setPosition(null); } }, [selectedItem]); useEffectAsync(async () => { const response = await fetch('/api/notifications/types'); if (response.ok) { const types = await response.json(); setAllEventTypes([...allEventTypes, ...types.map((it) => [it.type, prefixString('event', it.type)])]); } else { throw Error(await response.text()); } }, []); const handleSubmit = useCatch(async ({ deviceId, from, to, type }) => { const query = new URLSearchParams({ deviceId, from, to }); eventTypes.forEach((it) => query.append('type', it)); if (type === 'export') { window.location.assign(`/api/reports/events/xlsx?${query.toString()}`); } else if (type === 'mail') { const response = await fetch(`/api/reports/events/mail?${query.toString()}`); if (!response.ok) { throw Error(await response.text()); } } else { setLoading(true); try { const response = await fetch(`/api/reports/events?${query.toString()}`, { headers: { Accept: 'application/json' }, }); if (response.ok) { setItems(await response.json()); } else { throw Error(await response.text()); } } finally { setLoading(false); } } }); const formatValue = (item, key) => { switch (key) { case 'eventTime': return formatTime(item[key], 'seconds', hours12); case 'type': return t(prefixString('event', item[key])); case 'geofenceId': if (item[key] > 0) { const geofence = geofences[item[key]]; return geofence && geofence.name; } return null; case 'maintenanceId': return item[key] > 0 ? item[key] > 0 : null; case 'attributes': switch (item.type) { case 'alarm': return t(prefixString('alarm', item.attributes.alarm)); case 'deviceOverspeed': return formatSpeed(item.attributes.speed, speedUnit, t); case 'driverChanged': return item.attributes.driverUniqueId; case 'media': return ({item.attributes.file}); case 'commandResult': return item.attributes.result; default: return ''; } default: return item[key]; } }; return ( } breadcrumbs={['reportTitle', 'reportEvents']}>
{selectedItem && (
{position && } {position && }
)}
{t('reportEventTypes')}
{columns.map((key) => ({t(columnsMap.get(key))}))} {!loading ? items.map((item) => ( {item.positionId ? selectedItem === item ? ( setSelectedItem(null)}> ) : ( setSelectedItem(item)}> ) : ''} {columns.map((key) => ( {formatValue(item, key)} ))} )) : ()}
); }; export default EventReportPage;