2024-01-11 12:52:41 +02:00
|
|
|
import { Box, Typography } from "@mui/material";
|
|
|
|
import classNames from "classnames";
|
|
|
|
import utils from "../utils";
|
|
|
|
import "./css/AvailabilityPicker.css";
|
|
|
|
import dayjs, { Dayjs } from "dayjs";
|
|
|
|
import utc from 'dayjs/plugin/utc';
|
|
|
|
import timezone from 'dayjs/plugin/timezone';
|
|
|
|
import localizedFormat from 'dayjs/plugin/localizedFormat';
|
|
|
|
import DisableableTooltip from "./DisableableTooltip";
|
|
|
|
|
|
|
|
dayjs.extend(utc)
|
|
|
|
dayjs.extend(timezone)
|
|
|
|
dayjs.extend(localizedFormat)
|
|
|
|
|
|
|
|
const AvailabilityPickerHour = (props: {
|
|
|
|
dateTime: Dayjs,
|
|
|
|
isFullHourSelected: boolean,
|
|
|
|
isHalfHourSelected: boolean,
|
|
|
|
halfHourDisplayHeight: number,
|
2024-01-12 11:53:39 +02:00
|
|
|
currentTotalRespondents: number,
|
2024-01-11 12:52:41 +02:00
|
|
|
namesMarkedFullHourAsAvailable: String[],
|
|
|
|
namesMarkedHalfHourAsAvailable: String[],
|
|
|
|
onMouseEnterHalfhour: (e: React.MouseEvent<HTMLDivElement, globalThis.MouseEvent>, time: Dayjs) => void,
|
|
|
|
onMouseClickOnHalfhour: (time: Dayjs, isDelete: boolean) => void
|
|
|
|
}) => {
|
|
|
|
const generateTooltipText = (names: String[]): String => {
|
|
|
|
return `${names.length} ${names.length > 1 ? "people have" : "person has"} marked this time as available: ${names.join(", ")}`;
|
|
|
|
}
|
|
|
|
|
2024-01-12 11:53:39 +02:00
|
|
|
const heatMapColorforValue = (value: number) => {
|
|
|
|
if (value === 0 || props.currentTotalRespondents === 0) {
|
|
|
|
return 'inherit';
|
|
|
|
}
|
|
|
|
|
|
|
|
var h = (1.0 - (value / props.currentTotalRespondents)) * 240
|
|
|
|
return "hsl(" + h + ", 75%, 35%)";
|
|
|
|
}
|
|
|
|
|
2024-01-11 12:52:41 +02:00
|
|
|
return (
|
|
|
|
<Box
|
|
|
|
//className={classNames({ "hour-light": isEvenHour, "hour-dark": !isEvenHour })}
|
|
|
|
className={"hour-light"}
|
|
|
|
>
|
|
|
|
<DisableableTooltip
|
|
|
|
disabled={props.namesMarkedFullHourAsAvailable.length < 1}
|
|
|
|
title={generateTooltipText(props.namesMarkedFullHourAsAvailable)}
|
|
|
|
placement="top"
|
|
|
|
disableInteractive
|
|
|
|
followCursor={true}
|
|
|
|
arrow
|
|
|
|
enterDelay={500}
|
|
|
|
>
|
|
|
|
<Box
|
2024-01-12 11:53:39 +02:00
|
|
|
className={classNames("full-hour", { "selected-availability": props.isFullHourSelected })}
|
|
|
|
bgcolor={heatMapColorforValue(props.namesMarkedFullHourAsAvailable.length)}
|
2024-01-11 12:52:41 +02:00
|
|
|
height={props.halfHourDisplayHeight}
|
|
|
|
onMouseEnter={(e) => props.onMouseEnterHalfhour(e, props.dateTime)}
|
|
|
|
onMouseDown={(e) => {
|
|
|
|
if (e.button !== 0 && e.button !== 2) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
props.onMouseClickOnHalfhour(props.dateTime, e.button === 2);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Typography className={"noselect time-text"}>
|
|
|
|
{ utils.formatTimeFromHourOfDay(props.dateTime.hour(), 0) }
|
|
|
|
</Typography>
|
|
|
|
</Box>
|
|
|
|
</DisableableTooltip>
|
|
|
|
<DisableableTooltip
|
|
|
|
disabled={props.namesMarkedHalfHourAsAvailable.length < 1}
|
|
|
|
title={generateTooltipText(props.namesMarkedHalfHourAsAvailable)}
|
|
|
|
placement="top"
|
|
|
|
disableInteractive
|
|
|
|
followCursor={true}
|
|
|
|
arrow
|
|
|
|
enterDelay={500}
|
|
|
|
>
|
|
|
|
<Box
|
2024-01-12 11:53:39 +02:00
|
|
|
className={classNames("half-hour", { "selected-availability": props.isHalfHourSelected })}
|
|
|
|
bgcolor={heatMapColorforValue(props.namesMarkedHalfHourAsAvailable.length)}
|
2024-01-11 12:52:41 +02:00
|
|
|
height={props.halfHourDisplayHeight}
|
|
|
|
onMouseEnter={(e) => props.onMouseEnterHalfhour(e, props.dateTime.add(30, "minutes"))}
|
|
|
|
onMouseDown={(e) => {
|
|
|
|
if (e.button !== 0 && e.button !== 2) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
props.onMouseClickOnHalfhour(props.dateTime.add(30, "minutes"), e.button === 2);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</DisableableTooltip>
|
|
|
|
</Box>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default AvailabilityPickerHour;
|