💄 style(time-slot): Add chart display
This commit is contained in:
parent
837157cc42
commit
c44ad47f3c
@ -8,15 +8,26 @@ import {
|
|||||||
} from '@/services/admin/system';
|
} from '@/services/admin/system';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useQuery } from '@tanstack/react-query';
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
|
import { ChartContainer, ChartTooltip } from '@workspace/ui/components/chart';
|
||||||
import { Label } from '@workspace/ui/components/label';
|
import { Label } from '@workspace/ui/components/label';
|
||||||
import { Table, TableBody, TableCell, TableRow } from '@workspace/ui/components/table';
|
import { Table, TableBody, TableCell, TableRow } from '@workspace/ui/components/table';
|
||||||
|
import { ArrayInput } from '@workspace/ui/custom-components/dynamic-Inputs';
|
||||||
import { EnhancedInput } from '@workspace/ui/custom-components/enhanced-input';
|
import { EnhancedInput } from '@workspace/ui/custom-components/enhanced-input';
|
||||||
import { DicesIcon } from 'lucide-react';
|
import { DicesIcon } from 'lucide-react';
|
||||||
import { nanoid } from 'nanoid';
|
import { nanoid } from 'nanoid';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { useState } from 'react';
|
import { useMemo, useState } from 'react';
|
||||||
|
import { Cell, Legend, Pie, PieChart } from 'recharts';
|
||||||
import { toast } from 'sonner';
|
import { toast } from 'sonner';
|
||||||
|
|
||||||
|
const COLORS = [
|
||||||
|
'hsl(var(--chart-1))',
|
||||||
|
'hsl(var(--chart-2))',
|
||||||
|
'hsl(var(--chart-3))',
|
||||||
|
'hsl(var(--chart-4))',
|
||||||
|
'hsl(var(--chart-5))',
|
||||||
|
];
|
||||||
|
|
||||||
export default function Node() {
|
export default function Node() {
|
||||||
const t = useTranslations('system.node');
|
const t = useTranslations('system.node');
|
||||||
|
|
||||||
@ -56,23 +67,25 @@ export default function Node() {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const addTimeSlot = () => {
|
const chartTimeSlots = useMemo(() => {
|
||||||
setTimeSlots([...timeSlots, { start_time: '', end_time: '', multiplier: 1 }]);
|
return timeSlots.map((slot) => ({
|
||||||
};
|
name: `${slot.start_time} - ${slot.end_time}`,
|
||||||
|
value: slot.multiplier,
|
||||||
|
}));
|
||||||
|
}, [timeSlots]);
|
||||||
|
|
||||||
const removeTimeSlot = (index: number) => {
|
const chartConfig = useMemo(() => {
|
||||||
setTimeSlots(timeSlots.filter((_, i) => i !== index));
|
return chartTimeSlots.reduce(
|
||||||
};
|
(acc, item, index) => {
|
||||||
|
acc[item.name] = {
|
||||||
const updateTimeSlot = (index: number, field: keyof API.TimePeriod, value: string | number) => {
|
label: item.name,
|
||||||
const updatedSlots = timeSlots.map((slot, i) => {
|
color: COLORS[index % COLORS.length] || 'hsl(var(--default-chart-color))',
|
||||||
if (i === index) {
|
};
|
||||||
return { ...slot, [field]: value };
|
return acc;
|
||||||
}
|
},
|
||||||
return slot;
|
{} as Record<string, { label: string; color: string }>,
|
||||||
});
|
);
|
||||||
setTimeSlots(updatedSlots);
|
}, [data]);
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@ -162,55 +175,73 @@ export default function Node() {
|
|||||||
</TableRow>
|
</TableRow>
|
||||||
</TableBody>
|
</TableBody>
|
||||||
</Table>
|
</Table>
|
||||||
<div className='px-2'>
|
<div className='flex flex-col-reverse gap-8 px-4 pt-6 md:flex-row md:items-start'>
|
||||||
<div className='mt-4 grid gap-4'>
|
<div className='w-full md:w-1/2'>
|
||||||
{timeSlots.map((slot, index) => (
|
<ChartContainer config={chartConfig} className='mx-auto aspect-square max-w-[400px]'>
|
||||||
<div key={index} className='flex flex-col items-end gap-2 lg:flex-row'>
|
<PieChart>
|
||||||
<div>
|
<Pie
|
||||||
<Label>{t('startTime')}</Label>
|
data={chartTimeSlots}
|
||||||
<EnhancedInput
|
cx='50%'
|
||||||
key={`${index}-start-time`}
|
cy='50%'
|
||||||
type='time'
|
labelLine={false}
|
||||||
value={slot.start_time}
|
outerRadius='80%'
|
||||||
onValueChange={(value) => updateTimeSlot(index, 'start_time', value as string)}
|
fill='#8884d8'
|
||||||
/>
|
dataKey='value'
|
||||||
</div>
|
label={({ name, percent, value }) =>
|
||||||
|
`${value?.toFixed(2)}x (${(percent * 100).toFixed(0)}%)`
|
||||||
<div>
|
}
|
||||||
<Label>{t('endTime')}</Label>
|
|
||||||
<EnhancedInput
|
|
||||||
key={`${index}-end-time`}
|
|
||||||
type='time'
|
|
||||||
value={slot.end_time}
|
|
||||||
onValueChange={(value) => updateTimeSlot(index, 'end_time', value as string)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<Label>{t('multiplier')}</Label>
|
|
||||||
<EnhancedInput
|
|
||||||
key={`${index}-multiplier`}
|
|
||||||
type='number'
|
|
||||||
value={slot.multiplier}
|
|
||||||
onValueChange={(value) => updateTimeSlot(index, 'multiplier', value as number)}
|
|
||||||
min={1}
|
|
||||||
step='0.1'
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<Button
|
|
||||||
variant='destructive'
|
|
||||||
onClick={() => {
|
|
||||||
removeTimeSlot(index);
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{t('delete')}
|
{chartTimeSlots.map((entry, index) => (
|
||||||
</Button>
|
<Cell key={`cell-${index}`} fill={COLORS[index % COLORS.length]} />
|
||||||
</div>
|
))}
|
||||||
))}
|
</Pie>
|
||||||
|
<ChartTooltip
|
||||||
|
content={({ payload }) => {
|
||||||
|
if (payload && payload.length) {
|
||||||
|
const data = payload[0]?.payload;
|
||||||
|
return (
|
||||||
|
<div className='bg-background rounded-lg border p-2 shadow-sm'>
|
||||||
|
<div className='grid grid-cols-2 gap-2'>
|
||||||
|
<div className='flex flex-col'>
|
||||||
|
<span className='text-muted-foreground text-[0.70rem] uppercase'>
|
||||||
|
{t('timeSlot')}
|
||||||
|
</span>
|
||||||
|
<span className='text-muted-foreground font-bold'>
|
||||||
|
{data.name || '其他'}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className='flex flex-col'>
|
||||||
|
<span className='text-muted-foreground text-[0.70rem] uppercase'>
|
||||||
|
{t('multiplier')}
|
||||||
|
</span>
|
||||||
|
<span className='font-bold'>{data.value.toFixed(2)}x</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Legend />
|
||||||
|
</PieChart>
|
||||||
|
</ChartContainer>
|
||||||
|
</div>
|
||||||
|
<div className='w-full md:w-1/2'>
|
||||||
|
<ArrayInput<API.TimePeriod>
|
||||||
|
fields={[
|
||||||
|
{
|
||||||
|
name: 'start_time',
|
||||||
|
prefix: t('startTime'),
|
||||||
|
type: 'time',
|
||||||
|
},
|
||||||
|
{ name: 'end_time', prefix: t('endTime'), type: 'time' },
|
||||||
|
{ name: 'multiplier', prefix: t('multiplier'), type: 'number' },
|
||||||
|
]}
|
||||||
|
value={timeSlots}
|
||||||
|
onChange={setTimeSlots}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<Button onClick={addTimeSlot} variant='outline' className='mt-4 w-full'>
|
|
||||||
{t('addTimeSlot')}
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Resetovat",
|
"reset": "Resetovat",
|
||||||
"save": "Uložit",
|
"save": "Uložit",
|
||||||
"saveSuccess": "Úspěšně uloženo",
|
"saveSuccess": "Úspěšně uloženo",
|
||||||
"startTime": "Čas začátku"
|
"startTime": "Čas začátku",
|
||||||
|
"timeSlot": "Časový úsek"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Seznam povolených přípon e-mailů",
|
"emailSuffixWhitelist": "Seznam povolených přípon e-mailů",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Zurücksetzen",
|
"reset": "Zurücksetzen",
|
||||||
"save": "Speichern",
|
"save": "Speichern",
|
||||||
"saveSuccess": "Erfolgreich gespeichert",
|
"saveSuccess": "Erfolgreich gespeichert",
|
||||||
"startTime": "Startzeit"
|
"startTime": "Startzeit",
|
||||||
|
"timeSlot": "Zeitfenster"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "E-Mail-Suffix-Whitelist",
|
"emailSuffixWhitelist": "E-Mail-Suffix-Whitelist",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Reset",
|
"reset": "Reset",
|
||||||
"save": "Save",
|
"save": "Save",
|
||||||
"saveSuccess": "Save Successful",
|
"saveSuccess": "Save Successful",
|
||||||
"startTime": "Start Time"
|
"startTime": "Start Time",
|
||||||
|
"timeSlot": "Time Slot"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Email Suffix Whitelist",
|
"emailSuffixWhitelist": "Email Suffix Whitelist",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Restablecer",
|
"reset": "Restablecer",
|
||||||
"save": "Guardar",
|
"save": "Guardar",
|
||||||
"saveSuccess": "Guardado exitosamente",
|
"saveSuccess": "Guardado exitosamente",
|
||||||
"startTime": "Hora de Inicio"
|
"startTime": "Hora de Inicio",
|
||||||
|
"timeSlot": "Franja Horaria"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Lista blanca de sufijos de correo electrónico",
|
"emailSuffixWhitelist": "Lista blanca de sufijos de correo electrónico",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Restablecer",
|
"reset": "Restablecer",
|
||||||
"save": "Guardar",
|
"save": "Guardar",
|
||||||
"saveSuccess": "Guardado exitosamente",
|
"saveSuccess": "Guardado exitosamente",
|
||||||
"startTime": "Hora de Inicio"
|
"startTime": "Hora de Inicio",
|
||||||
|
"timeSlot": "Franja Horaria"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Lista blanca de sufijos de correo electrónico",
|
"emailSuffixWhitelist": "Lista blanca de sufijos de correo electrónico",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "بازنشانی",
|
"reset": "بازنشانی",
|
||||||
"save": "ذخیره",
|
"save": "ذخیره",
|
||||||
"saveSuccess": "ذخیره با موفقیت انجام شد",
|
"saveSuccess": "ذخیره با موفقیت انجام شد",
|
||||||
"startTime": "زمان شروع"
|
"startTime": "زمان شروع",
|
||||||
|
"timeSlot": "بازه زمانی"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "لیست سفید پسوند ایمیل",
|
"emailSuffixWhitelist": "لیست سفید پسوند ایمیل",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Nollaa",
|
"reset": "Nollaa",
|
||||||
"save": "Tallenna",
|
"save": "Tallenna",
|
||||||
"saveSuccess": "Tallennus onnistui",
|
"saveSuccess": "Tallennus onnistui",
|
||||||
"startTime": "Aloitusaika"
|
"startTime": "Aloitusaika",
|
||||||
|
"timeSlot": "Aikaväli"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Sähköpostin jälkiliitteen sallittu luettelo",
|
"emailSuffixWhitelist": "Sähköpostin jälkiliitteen sallittu luettelo",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Réinitialiser",
|
"reset": "Réinitialiser",
|
||||||
"save": "Enregistrer",
|
"save": "Enregistrer",
|
||||||
"saveSuccess": "Enregistrement réussi",
|
"saveSuccess": "Enregistrement réussi",
|
||||||
"startTime": "Heure de Début"
|
"startTime": "Heure de Début",
|
||||||
|
"timeSlot": "Créneau horaire"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Liste blanche des suffixes d'email",
|
"emailSuffixWhitelist": "Liste blanche des suffixes d'email",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "रीसेट",
|
"reset": "रीसेट",
|
||||||
"save": "सहेजें",
|
"save": "सहेजें",
|
||||||
"saveSuccess": "सफलतापूर्वक सहेजा गया",
|
"saveSuccess": "सफलतापूर्वक सहेजा गया",
|
||||||
"startTime": "प्रारंभ समय"
|
"startTime": "प्रारंभ समय",
|
||||||
|
"timeSlot": "समय स्लॉट"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "ईमेल प्रत्यय श्वेतसूची",
|
"emailSuffixWhitelist": "ईमेल प्रत्यय श्वेतसूची",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Visszaállítás",
|
"reset": "Visszaállítás",
|
||||||
"save": "Mentés",
|
"save": "Mentés",
|
||||||
"saveSuccess": "Sikeres mentés",
|
"saveSuccess": "Sikeres mentés",
|
||||||
"startTime": "Kezdési Idő"
|
"startTime": "Kezdési Idő",
|
||||||
|
"timeSlot": "Időpont"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "E-mail végződés fehérlista",
|
"emailSuffixWhitelist": "E-mail végződés fehérlista",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "リセット",
|
"reset": "リセット",
|
||||||
"save": "保存",
|
"save": "保存",
|
||||||
"saveSuccess": "保存成功",
|
"saveSuccess": "保存成功",
|
||||||
"startTime": "開始時間"
|
"startTime": "開始時間",
|
||||||
|
"timeSlot": "時間枠"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "メールサフィックスホワイトリスト",
|
"emailSuffixWhitelist": "メールサフィックスホワイトリスト",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "재설정",
|
"reset": "재설정",
|
||||||
"save": "저장",
|
"save": "저장",
|
||||||
"saveSuccess": "저장 성공",
|
"saveSuccess": "저장 성공",
|
||||||
"startTime": "시작 시간"
|
"startTime": "시작 시간",
|
||||||
|
"timeSlot": "시간대"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "이메일 접미사 화이트리스트",
|
"emailSuffixWhitelist": "이메일 접미사 화이트리스트",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Tilbakestill",
|
"reset": "Tilbakestill",
|
||||||
"save": "Lagre",
|
"save": "Lagre",
|
||||||
"saveSuccess": "Lagring vellykket",
|
"saveSuccess": "Lagring vellykket",
|
||||||
"startTime": "Starttid"
|
"startTime": "Starttid",
|
||||||
|
"timeSlot": "Tidsluke"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "E-post suffiks hviteliste",
|
"emailSuffixWhitelist": "E-post suffiks hviteliste",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Resetuj",
|
"reset": "Resetuj",
|
||||||
"save": "Zapisz",
|
"save": "Zapisz",
|
||||||
"saveSuccess": "Zapisano pomyślnie",
|
"saveSuccess": "Zapisano pomyślnie",
|
||||||
"startTime": "Czas rozpoczęcia"
|
"startTime": "Czas rozpoczęcia",
|
||||||
|
"timeSlot": "Przedział czasowy"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Biała lista sufiksów e-mail",
|
"emailSuffixWhitelist": "Biała lista sufiksów e-mail",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Redefinir",
|
"reset": "Redefinir",
|
||||||
"save": "Salvar",
|
"save": "Salvar",
|
||||||
"saveSuccess": "Salvo com sucesso",
|
"saveSuccess": "Salvo com sucesso",
|
||||||
"startTime": "Hora de Início"
|
"startTime": "Hora de Início",
|
||||||
|
"timeSlot": "Horário"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Lista branca de sufixos de e-mail",
|
"emailSuffixWhitelist": "Lista branca de sufixos de e-mail",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Resetare",
|
"reset": "Resetare",
|
||||||
"save": "Salvare",
|
"save": "Salvare",
|
||||||
"saveSuccess": "Salvare reușită",
|
"saveSuccess": "Salvare reușită",
|
||||||
"startTime": "Ora de Începere"
|
"startTime": "Ora de Începere",
|
||||||
|
"timeSlot": "Interval de timp"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Lista albă a sufixelor de e-mail",
|
"emailSuffixWhitelist": "Lista albă a sufixelor de e-mail",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Сбросить",
|
"reset": "Сбросить",
|
||||||
"save": "Сохранить",
|
"save": "Сохранить",
|
||||||
"saveSuccess": "Успешно сохранено",
|
"saveSuccess": "Успешно сохранено",
|
||||||
"startTime": "Время начала"
|
"startTime": "Время начала",
|
||||||
|
"timeSlot": "Временной интервал"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Белый список суффиксов электронной почты",
|
"emailSuffixWhitelist": "Белый список суффиксов электронной почты",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "รีเซ็ต",
|
"reset": "รีเซ็ต",
|
||||||
"save": "บันทึก",
|
"save": "บันทึก",
|
||||||
"saveSuccess": "บันทึกสำเร็จ",
|
"saveSuccess": "บันทึกสำเร็จ",
|
||||||
"startTime": "เวลาเริ่มต้น"
|
"startTime": "เวลาเริ่มต้น",
|
||||||
|
"timeSlot": "ช่วงเวลา"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "รายการอนุญาตโดเมนอีเมล",
|
"emailSuffixWhitelist": "รายการอนุญาตโดเมนอีเมล",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Sıfırla",
|
"reset": "Sıfırla",
|
||||||
"save": "Kaydet",
|
"save": "Kaydet",
|
||||||
"saveSuccess": "Başarıyla kaydedildi",
|
"saveSuccess": "Başarıyla kaydedildi",
|
||||||
"startTime": "Başlangıç Zamanı"
|
"startTime": "Başlangıç Zamanı",
|
||||||
|
"timeSlot": "Zaman Dilimi"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "E-posta Soneki Beyaz Listesi",
|
"emailSuffixWhitelist": "E-posta Soneki Beyaz Listesi",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Скинути",
|
"reset": "Скинути",
|
||||||
"save": "Зберегти",
|
"save": "Зберегти",
|
||||||
"saveSuccess": "Збережено успішно",
|
"saveSuccess": "Збережено успішно",
|
||||||
"startTime": "Час Початку"
|
"startTime": "Час Початку",
|
||||||
|
"timeSlot": "Часовий проміжок"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Білий список суфіксів електронної пошти",
|
"emailSuffixWhitelist": "Білий список суфіксів електронної пошти",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "Đặt lại",
|
"reset": "Đặt lại",
|
||||||
"save": "Lưu",
|
"save": "Lưu",
|
||||||
"saveSuccess": "Lưu thành công",
|
"saveSuccess": "Lưu thành công",
|
||||||
"startTime": "Thời gian bắt đầu"
|
"startTime": "Thời gian bắt đầu",
|
||||||
|
"timeSlot": "Khung giờ"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "Danh sách trắng hậu tố email",
|
"emailSuffixWhitelist": "Danh sách trắng hậu tố email",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "重置",
|
"reset": "重置",
|
||||||
"save": "保存",
|
"save": "保存",
|
||||||
"saveSuccess": "保存成功",
|
"saveSuccess": "保存成功",
|
||||||
"startTime": "开始时间"
|
"startTime": "开始时间",
|
||||||
|
"timeSlot": "时间段"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "电子邮件后缀白名单",
|
"emailSuffixWhitelist": "电子邮件后缀白名单",
|
||||||
|
|||||||
@ -61,7 +61,8 @@
|
|||||||
"reset": "重設",
|
"reset": "重設",
|
||||||
"save": "儲存",
|
"save": "儲存",
|
||||||
"saveSuccess": "保存成功",
|
"saveSuccess": "保存成功",
|
||||||
"startTime": "開始時間"
|
"startTime": "開始時間",
|
||||||
|
"timeSlot": "時間段"
|
||||||
},
|
},
|
||||||
"register": {
|
"register": {
|
||||||
"emailSuffixWhitelist": "電子郵件後綴白名單",
|
"emailSuffixWhitelist": "電子郵件後綴白名單",
|
||||||
|
|||||||
@ -6,7 +6,7 @@ import { useEffect, useState } from 'react';
|
|||||||
|
|
||||||
interface FieldConfig extends Omit<EnhancedInputProps, 'type'> {
|
interface FieldConfig extends Omit<EnhancedInputProps, 'type'> {
|
||||||
name: string;
|
name: string;
|
||||||
type: 'text' | 'number' | 'select';
|
type: 'text' | 'number' | 'select' | 'time';
|
||||||
options?: { label: string; value: string }[];
|
options?: { label: string; value: string }[];
|
||||||
internal?: boolean;
|
internal?: boolean;
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
@ -53,7 +53,7 @@ export function ObjectInput<T extends Record<string, any>>({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='flex flex-1 gap-4'>
|
<div className='flex flex-1 flex-wrap gap-4'>
|
||||||
{fields.map(({ name, type, options, ...fieldProps }) => (
|
{fields.map(({ name, type, options, ...fieldProps }) => (
|
||||||
<div key={name} className='flex-1'>
|
<div key={name} className='flex-1'>
|
||||||
{type === 'select' && options ? (
|
{type === 'select' && options ? (
|
||||||
@ -127,6 +127,12 @@ export function ArrayInput<T extends Record<string, any>>({
|
|||||||
onChange(modifiedItems);
|
onChange(modifiedItems);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (value.length > 0) {
|
||||||
|
setDisplayItems(value);
|
||||||
|
}
|
||||||
|
}, [value]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='flex flex-col gap-4'>
|
<div className='flex flex-col gap-4'>
|
||||||
{displayItems.map((item, index) => (
|
{displayItems.map((item, index) => (
|
||||||
|
|||||||
@ -76,7 +76,9 @@ export function EnhancedInput({
|
|||||||
className={cn('border-input flex w-full items-center rounded-md border', className)}
|
className={cn('border-input flex w-full items-center rounded-md border', className)}
|
||||||
suppressHydrationWarning
|
suppressHydrationWarning
|
||||||
>
|
>
|
||||||
{prefix && <div className='bg-muted mr-px flex h-9 items-center px-3'>{prefix}</div>}
|
{prefix && (
|
||||||
|
<div className='bg-muted mr-px flex h-9 items-center text-nowrap px-3'>{prefix}</div>
|
||||||
|
)}
|
||||||
<Input
|
<Input
|
||||||
{...props}
|
{...props}
|
||||||
value={value}
|
value={value}
|
||||||
@ -84,7 +86,9 @@ export function EnhancedInput({
|
|||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
onBlur={handleBlur}
|
onBlur={handleBlur}
|
||||||
/>
|
/>
|
||||||
{suffix && <div className='bg-muted ml-px flex h-9 items-center px-3'>{suffix}</div>}
|
{suffix && (
|
||||||
|
<div className='bg-muted ml-px flex h-9 items-center text-nowrap px-3'>{suffix}</div>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user