- Add Activepieces fork with SmoothSchedule custom piece - Create integrations app with Activepieces service layer - Add embed token endpoint for iframe integration - Create Automations page with embedded workflow builder - Add sidebar visibility fix for embed mode - Add list inactive customers endpoint to Public API - Include SmoothSchedule triggers: event created/updated/cancelled - Include SmoothSchedule actions: create/update/cancel events, list resources/services/customers 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
128 lines
3.6 KiB
TypeScript
128 lines
3.6 KiB
TypeScript
'use client';
|
|
|
|
import * as React from 'react';
|
|
|
|
import { cn } from '@/lib/utils';
|
|
import { isNil } from '@activepieces/shared';
|
|
|
|
import { TimePeriodSelect } from './time-period-select';
|
|
import { Period } from './time-picker-utils';
|
|
import { TimeUnitPickerInput } from './time-unit-input';
|
|
|
|
interface TimePickerProps {
|
|
date: Date | undefined;
|
|
setDate: (date: Date) => void;
|
|
showSeconds?: boolean;
|
|
name?: string;
|
|
}
|
|
|
|
const minutesItems = new Array(60).fill(0).map((_, index) => ({
|
|
value: index.toString(),
|
|
label: index < 10 ? `0${index}` : index.toString(),
|
|
}));
|
|
|
|
const hoursItems = new Array(12).fill(0).map((_, index) => ({
|
|
value: (index + 1).toString(),
|
|
label: index + 1 < 10 ? `0${index + 1}` : (index + 1).toString(),
|
|
}));
|
|
|
|
export function TimePicker({
|
|
date,
|
|
setDate,
|
|
showSeconds,
|
|
name = 'from',
|
|
}: TimePickerProps) {
|
|
const [period, setPeriod] = React.useState<Period>(() => {
|
|
if (date) {
|
|
return date.getHours() >= 12 ? 'PM' : 'AM';
|
|
}
|
|
return name === 'from' ? 'AM' : 'PM';
|
|
});
|
|
React.useEffect(() => {
|
|
if (date && date.getHours() >= 12) {
|
|
setPeriod('PM');
|
|
} else if (!date) {
|
|
setPeriod(name === 'from' ? 'AM' : 'PM');
|
|
}
|
|
}, [date]);
|
|
const hasValueChanged =
|
|
name === 'from'
|
|
? date?.getHours() !== 0 || date?.getMinutes() !== 0 || period !== 'AM'
|
|
: date?.getHours() !== 23 || date?.getMinutes() !== 59 || period !== 'PM';
|
|
const isActive = !isNil(date) && hasValueChanged;
|
|
const minuteRef = React.useRef<HTMLInputElement>(null);
|
|
const hourRef = React.useRef<HTMLInputElement>(null);
|
|
const secondRef = React.useRef<HTMLInputElement>(null);
|
|
const periodRef = React.useRef<HTMLButtonElement>(null);
|
|
|
|
return (
|
|
<div
|
|
className={cn(
|
|
'flex items-center transition-all gap-2 w-full text-muted-foreground justify-center bg-accent/50 py-1 px-2 rounded-sm h-[43px] border border-solid border-border',
|
|
{
|
|
'text-foreground': isActive,
|
|
},
|
|
)}
|
|
>
|
|
<div className="grid gap-1 text-center">
|
|
<TimeUnitPickerInput
|
|
picker="12hours"
|
|
isActive={isActive}
|
|
period={period}
|
|
date={date}
|
|
setDate={setDate}
|
|
name={name}
|
|
ref={hourRef}
|
|
onRightFocus={() => minuteRef.current?.focus()}
|
|
autoCompleteList={hoursItems}
|
|
/>
|
|
</div>
|
|
:
|
|
<div className="grid gap-1 text-center">
|
|
<TimeUnitPickerInput
|
|
picker="minutes"
|
|
id="minutes12"
|
|
isActive={isActive}
|
|
name={name}
|
|
date={date}
|
|
period={period}
|
|
setDate={setDate}
|
|
ref={minuteRef}
|
|
onLeftFocus={() => hourRef.current?.focus()}
|
|
onRightFocus={() => secondRef.current?.focus()}
|
|
autoCompleteList={minutesItems}
|
|
/>
|
|
</div>
|
|
{showSeconds && (
|
|
<>
|
|
:
|
|
<div className="grid gap-1 text-center">
|
|
<TimeUnitPickerInput
|
|
picker="seconds"
|
|
id="seconds12"
|
|
name={name}
|
|
isActive={isActive}
|
|
date={date}
|
|
setDate={setDate}
|
|
ref={secondRef}
|
|
onLeftFocus={() => minuteRef.current?.focus()}
|
|
onRightFocus={() => periodRef.current?.focus()}
|
|
/>
|
|
</div>
|
|
</>
|
|
)}
|
|
<div className="grid gap-1 text-center">
|
|
<TimePeriodSelect
|
|
period={period}
|
|
isActive={isActive}
|
|
setPeriod={setPeriod}
|
|
date={date}
|
|
setDate={setDate}
|
|
ref={periodRef}
|
|
onLeftFocus={() => secondRef.current?.focus()}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|