import React, {useCallback, useEffect, useMemo, useState} from "react"; import {classes, formatTime, Modify} from "../../Utils"; export function TimepickerInput( { children, className, value, onChange, // Properties to pass down. onKeyUp, onBlur, // Already set properties. type, placeholder, ...props }: React.PropsWithChildren, { value?: Date|null; onChange: (newDateTime: Date) => void; // Already set properties. type?: never; placeholder?: never; }>>): React.ReactElement { // Time text state. const [timeText, setTimeText] = useState(""); // Update time text when datetime value has changed. useEffect(() => { if (value && value instanceof Date && !isNaN(value.getTime())) setTimeText(formatTime(value)); }, [value]); // Check if time is valid. const invalidTime = useMemo(() => !(value && value instanceof Date && !isNaN(value.getTime())) && timeText.length > 0, [value, timeText]); const timeValue = useMemo(() => invalidTime ? new Date() : (value ?? new Date()), [invalidTime, value]); /** * Submit a new time from its raw text. */ const submitTime = useCallback((customTimeText?: string) => { // Get the current date text status. const timeTextMatch = (customTimeText ?? timeText).match(/^([0-2]?[0-9]):([0-5]?[0-9])$/); if (timeTextMatch) { // Parse hours. let rawHours = timeTextMatch[1]; if (rawHours.length < 2) rawHours = "0" + rawHours; let hours = parseInt(rawHours); if (isNaN(hours)) hours = 0; // Parse minutes. let rawMinutes = timeTextMatch[2]; if (rawMinutes.length < 2) rawMinutes = "0" + rawMinutes; let minutes = parseInt(rawMinutes); if (isNaN(minutes)) minutes = 0; //TODO // Parse seconds? // Try to build the structurally valid date with this time. //TODO const datetime = new Date(timeValue); datetime.setHours(hours, minutes); // Put the date back, if it changed to another day. datetime.setFullYear(timeValue.getFullYear(), timeValue.getMonth(), timeValue.getDate()); // Set the structurally valid date. onChange?.(datetime); } else // No structurally valid date, removing it. onChange?.(null); }, [timeText, timeValue, onChange]); return ( ); }