diff --git a/frontend/src/ui/Input.tsx b/frontend/src/ui/Input.tsx index 532b0953b..342364147 100644 --- a/frontend/src/ui/Input.tsx +++ b/frontend/src/ui/Input.tsx @@ -1,4 +1,4 @@ -import React, { Fragment, ReactNode, useId, useState } from "react"; +import React, { Fragment, ReactNode, useId, useRef, useState } from "react"; import { FiCheck, FiCopy } from "react-icons/fi"; import { WithTooltip } from "@opencast/appkit"; @@ -88,12 +88,14 @@ export type TimeUnit = "h" | "m" | "s"; /** A custom three-part input for time inputs split into hours, minutes and seconds */ export const TimeInput: React.FC = ({ timestamp, setTimestamp, disabled }) => { + const inputRefs = useRef>([]); + const hours = Math.floor(timestamp / 3600); const minutes = Math.floor((timestamp % 3600) / 60); const seconds = Math.floor(timestamp % 60); - const handleTimeChange = (newValue: number, type: TimeUnit) => { - if (isNaN(newValue) || newValue < 0 || newValue > 99) { + const handleInput = (newValue: number, type: TimeUnit, index: number) => { + if (isNaN(newValue) || newValue < 0) { return; } @@ -103,6 +105,30 @@ export const TimeInput: React.FC = ({ timestamp, setTimestamp, d + `${type === "s" ? cappedValue : seconds}s`; setTimestamp(timeStringToSeconds(timeString)); + + if (index < 2 && newValue > 9) { + inputRefs.current[index + 1]?.focus(); + } + }; + + const handleArrowNavigation = ( + e: React.KeyboardEvent, + index: number, + ) => { + const inputElement = inputRefs.current[index]; + + if (e.key === "ArrowLeft" && index > 0 && inputElement?.selectionStart === 0) { + e.preventDefault(); + inputRefs.current[index - 1]?.focus(); + } + + if ( + e.key === "ArrowRight" && index < inputRefs.current.length - 1 + && inputElement?.selectionStart === inputElement?.value.length + ) { + e.preventDefault(); + inputRefs.current[index + 1]?.focus(); + } }; const entries: [number, TimeUnit][] = [ @@ -124,13 +150,15 @@ export const TimeInput: React.FC = ({ timestamp, setTimestamp, d outline: `1px solid ${COLORS.neutral20}`, }, }}> - {entries.map(([time, unit]) => + {entries.map(([time, unit], index) => (inputRefs.current[index] = ref)} value={time} inputMode="numeric" - onChange={e => handleTimeChange(Number(e.target.value), unit)} + onChange={e => handleInput(Number(e.target.value), unit, index)} onFocus={e => e.target.select()} + onKeyDown={e => handleArrowNavigation(e, index)} css={{ width: time > 9 ? "2ch" : "1ch", lineHeight: 1,