Add Activepieces integration for workflow automation
- 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>
This commit is contained in:
@@ -0,0 +1,127 @@
|
||||
'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>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user