2024-06-30 18:55:39 -04:00
|
|
|
import { Checklist, EventBusy, Timelapse } from '@mui/icons-material'
|
|
|
|
import {
|
|
|
|
Avatar,
|
|
|
|
Box,
|
|
|
|
Button,
|
|
|
|
CircularProgress,
|
|
|
|
Container,
|
|
|
|
Grid,
|
|
|
|
List,
|
|
|
|
ListItem,
|
|
|
|
ListItemContent,
|
|
|
|
ListItemDecorator,
|
|
|
|
Sheet,
|
|
|
|
Typography,
|
|
|
|
} from '@mui/joy'
|
|
|
|
import moment from 'moment'
|
|
|
|
import React, { useEffect, useState } from 'react'
|
|
|
|
import { Link, useParams } from 'react-router-dom'
|
|
|
|
import { API_URL } from '../../Config'
|
|
|
|
import { GetAllCircleMembers } from '../../utils/Fetcher'
|
|
|
|
import { Fetch } from '../../utils/TokenManager'
|
2024-07-09 17:39:16 -04:00
|
|
|
import HistoryCard from './HistoryCard'
|
2024-06-30 18:55:39 -04:00
|
|
|
|
|
|
|
const ChoreHistory = () => {
|
|
|
|
const [choreHistory, setChoresHistory] = useState([])
|
|
|
|
const [userHistory, setUserHistory] = useState([])
|
|
|
|
const [performers, setPerformers] = useState([])
|
|
|
|
const [historyInfo, setHistoryInfo] = useState([])
|
|
|
|
|
|
|
|
const [isLoading, setIsLoading] = useState(true) // Add loading state
|
|
|
|
const { choreId } = useParams()
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
setIsLoading(true) // Start loading
|
|
|
|
|
|
|
|
Promise.all([
|
|
|
|
Fetch(`${API_URL}/chores/${choreId}/history`).then(res => res.json()),
|
|
|
|
GetAllCircleMembers().then(res => res.json()),
|
|
|
|
])
|
|
|
|
.then(([historyData, usersData]) => {
|
|
|
|
setChoresHistory(historyData.res)
|
|
|
|
|
|
|
|
const newUserChoreHistory = {}
|
|
|
|
historyData.res.forEach(choreHistory => {
|
|
|
|
const userId = choreHistory.completedBy
|
|
|
|
newUserChoreHistory[userId] = (newUserChoreHistory[userId] || 0) + 1
|
|
|
|
})
|
|
|
|
setUserHistory(newUserChoreHistory)
|
|
|
|
|
|
|
|
setPerformers(usersData.res)
|
|
|
|
updateHistoryInfo(historyData.res, newUserChoreHistory, usersData.res)
|
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
console.error('Error fetching data:', error)
|
|
|
|
// Handle errors, e.g., show an error message to the user
|
|
|
|
})
|
|
|
|
.finally(() => {
|
|
|
|
setIsLoading(false) // Finish loading
|
|
|
|
})
|
|
|
|
}, [choreId])
|
|
|
|
|
|
|
|
const updateHistoryInfo = (histories, userHistories, performers) => {
|
|
|
|
// average delay for task completaion from due date:
|
|
|
|
|
|
|
|
const averageDelay =
|
|
|
|
histories.reduce((acc, chore) => {
|
|
|
|
if (chore.dueDate) {
|
|
|
|
// Only consider chores with a due date
|
|
|
|
return acc + moment(chore.completedAt).diff(chore.dueDate, 'hours')
|
|
|
|
}
|
|
|
|
return acc
|
|
|
|
}, 0) / histories.length
|
|
|
|
const averageDelayMoment = moment.duration(averageDelay, 'hours')
|
|
|
|
const maximumDelay = histories.reduce((acc, chore) => {
|
|
|
|
if (chore.dueDate) {
|
|
|
|
// Only consider chores with a due date
|
|
|
|
const delay = moment(chore.completedAt).diff(chore.dueDate, 'hours')
|
|
|
|
return delay > acc ? delay : acc
|
|
|
|
}
|
|
|
|
return acc
|
|
|
|
}, 0)
|
|
|
|
|
|
|
|
const maxDelayMoment = moment.duration(maximumDelay, 'hours')
|
|
|
|
|
|
|
|
// find max value in userHistories:
|
|
|
|
const userCompletedByMost = Object.keys(userHistories).reduce((a, b) =>
|
|
|
|
userHistories[a] > userHistories[b] ? a : b,
|
|
|
|
)
|
|
|
|
const userCompletedByLeast = Object.keys(userHistories).reduce((a, b) =>
|
|
|
|
userHistories[a] < userHistories[b] ? a : b,
|
|
|
|
)
|
|
|
|
|
|
|
|
const historyInfo = [
|
|
|
|
{
|
|
|
|
icon: (
|
|
|
|
<Avatar>
|
|
|
|
<Checklist />
|
|
|
|
</Avatar>
|
|
|
|
),
|
|
|
|
text: `${histories.length} completed`,
|
|
|
|
subtext: `${Object.keys(userHistories).length} users contributed`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: (
|
|
|
|
<Avatar>
|
|
|
|
<Timelapse />
|
|
|
|
</Avatar>
|
|
|
|
),
|
|
|
|
text: `Completed within ${moment
|
|
|
|
.duration(averageDelayMoment)
|
|
|
|
.humanize()}`,
|
|
|
|
subtext: `Maximum delay was ${moment
|
|
|
|
.duration(maxDelayMoment)
|
|
|
|
.humanize()}`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: <Avatar></Avatar>,
|
|
|
|
text: `${
|
|
|
|
performers.find(p => p.userId === Number(userCompletedByMost))
|
|
|
|
?.displayName
|
|
|
|
} completed most`,
|
|
|
|
subtext: `${userHistories[userCompletedByMost]} time/s`,
|
|
|
|
},
|
|
|
|
]
|
|
|
|
if (userCompletedByLeast !== userCompletedByMost) {
|
|
|
|
historyInfo.push({
|
|
|
|
icon: (
|
|
|
|
<Avatar>
|
|
|
|
{
|
|
|
|
performers.find(p => p.userId === userCompletedByLeast)
|
|
|
|
?.displayName
|
|
|
|
}
|
|
|
|
</Avatar>
|
|
|
|
),
|
|
|
|
text: `${
|
|
|
|
performers.find(p => p.userId === Number(userCompletedByLeast))
|
|
|
|
.displayName
|
|
|
|
} completed least`,
|
|
|
|
subtext: `${userHistories[userCompletedByLeast]} time/s`,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
setHistoryInfo(historyInfo)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isLoading) {
|
|
|
|
return <CircularProgress /> // Show loading indicator
|
|
|
|
}
|
|
|
|
if (!choreHistory.length) {
|
|
|
|
return (
|
|
|
|
<Container
|
|
|
|
maxWidth='md'
|
|
|
|
sx={{
|
|
|
|
textAlign: 'center',
|
|
|
|
display: 'flex',
|
|
|
|
// make sure the content is centered vertically:
|
|
|
|
alignItems: 'center',
|
|
|
|
justifyContent: 'center',
|
|
|
|
flexDirection: 'column',
|
|
|
|
height: '50vh',
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<EventBusy
|
|
|
|
sx={{
|
|
|
|
fontSize: '6rem',
|
|
|
|
// color: 'text.disabled',
|
|
|
|
mb: 1,
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
|
|
|
|
<Typography level='h3' gutterBottom>
|
|
|
|
No History Yet
|
|
|
|
</Typography>
|
|
|
|
<Typography level='body1'>
|
|
|
|
You haven't completed any tasks. Once you start finishing tasks,
|
|
|
|
they'll show up here.
|
|
|
|
</Typography>
|
|
|
|
<Button variant='soft' sx={{ mt: 2 }}>
|
|
|
|
<Link to='/my/chores'>Go back to chores</Link>
|
|
|
|
</Button>
|
|
|
|
</Container>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Container maxWidth='md'>
|
|
|
|
<Typography level='h3' mb={1.5}>
|
|
|
|
Summary:
|
|
|
|
</Typography>
|
|
|
|
{/* <Sheet sx={{ mb: 1, borderRadius: 'sm', p: 2, boxShadow: 'md' }}>
|
|
|
|
<ListItem sx={{ gap: 1.5 }}>
|
|
|
|
<ListItemDecorator>
|
|
|
|
<Avatar>
|
|
|
|
<AccountCircle />
|
|
|
|
</Avatar>
|
|
|
|
</ListItemDecorator>
|
|
|
|
<ListItemContent>
|
|
|
|
<Typography level='body1' sx={{ fontWeight: 'md' }}>
|
|
|
|
{choreHistory.length} completed
|
|
|
|
</Typography>
|
|
|
|
<Typography level='body2' color='text.tertiary'>
|
|
|
|
{Object.keys(userHistory).length} users contributed
|
|
|
|
</Typography>
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem>
|
|
|
|
</Sheet> */}
|
|
|
|
<Grid container>
|
|
|
|
{historyInfo.map((info, index) => (
|
|
|
|
<Grid key={index} item xs={12} sm={6}>
|
|
|
|
<Sheet sx={{ mb: 1, borderRadius: 'sm', p: 2, boxShadow: 'md' }}>
|
|
|
|
<ListItem sx={{ gap: 1.5 }}>
|
|
|
|
<ListItemDecorator>{info.icon}</ListItemDecorator>
|
|
|
|
<ListItemContent>
|
|
|
|
<Typography level='body1' sx={{ fontWeight: 'md' }}>
|
|
|
|
{info.text}
|
|
|
|
</Typography>
|
|
|
|
<Typography level='body1' color='text.tertiary'>
|
|
|
|
{info.subtext}
|
|
|
|
</Typography>
|
|
|
|
</ListItemContent>
|
|
|
|
</ListItem>
|
|
|
|
</Sheet>
|
|
|
|
</Grid>
|
|
|
|
))}
|
|
|
|
</Grid>
|
|
|
|
{/* User History Cards */}
|
|
|
|
<Typography level='h3' my={1.5}>
|
|
|
|
History:
|
|
|
|
</Typography>
|
|
|
|
<Box sx={{ borderRadius: 'sm', p: 2, boxShadow: 'md' }}>
|
|
|
|
{/* Chore History List (Updated Style) */}
|
|
|
|
|
|
|
|
<List sx={{ p: 0 }}>
|
2024-07-09 17:39:16 -04:00
|
|
|
{choreHistory.map((historyEntry, index) => (
|
|
|
|
<HistoryCard
|
|
|
|
historyEntry={historyEntry}
|
|
|
|
performers={performers}
|
|
|
|
allHistory={choreHistory}
|
|
|
|
key={index}
|
|
|
|
index={index}
|
|
|
|
/>
|
2024-06-30 18:55:39 -04:00
|
|
|
))}
|
|
|
|
</List>
|
|
|
|
</Box>
|
|
|
|
</Container>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default ChoreHistory
|