💄 style(time-slot): Add chart display

This commit is contained in:
web@ppanel 2024-12-30 13:02:03 +07:00
parent 837157cc42
commit c44ad47f3c
26 changed files with 155 additions and 91 deletions

View File

@ -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>
</> </>
); );

View File

@ -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ů",

View File

@ -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",

View File

@ -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",

View File

@ -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",

View File

@ -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",

View File

@ -61,7 +61,8 @@
"reset": "بازنشانی", "reset": "بازنشانی",
"save": "ذخیره", "save": "ذخیره",
"saveSuccess": "ذخیره با موفقیت انجام شد", "saveSuccess": "ذخیره با موفقیت انجام شد",
"startTime": "زمان شروع" "startTime": "زمان شروع",
"timeSlot": "بازه زمانی"
}, },
"register": { "register": {
"emailSuffixWhitelist": "لیست سفید پسوند ایمیل", "emailSuffixWhitelist": "لیست سفید پسوند ایمیل",

View File

@ -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",

View File

@ -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",

View File

@ -61,7 +61,8 @@
"reset": "रीसेट", "reset": "रीसेट",
"save": "सहेजें", "save": "सहेजें",
"saveSuccess": "सफलतापूर्वक सहेजा गया", "saveSuccess": "सफलतापूर्वक सहेजा गया",
"startTime": "प्रारंभ समय" "startTime": "प्रारंभ समय",
"timeSlot": "समय स्लॉट"
}, },
"register": { "register": {
"emailSuffixWhitelist": "ईमेल प्रत्यय श्वेतसूची", "emailSuffixWhitelist": "ईमेल प्रत्यय श्वेतसूची",

View File

@ -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",

View File

@ -61,7 +61,8 @@
"reset": "リセット", "reset": "リセット",
"save": "保存", "save": "保存",
"saveSuccess": "保存成功", "saveSuccess": "保存成功",
"startTime": "開始時間" "startTime": "開始時間",
"timeSlot": "時間枠"
}, },
"register": { "register": {
"emailSuffixWhitelist": "メールサフィックスホワイトリスト", "emailSuffixWhitelist": "メールサフィックスホワイトリスト",

View File

@ -61,7 +61,8 @@
"reset": "재설정", "reset": "재설정",
"save": "저장", "save": "저장",
"saveSuccess": "저장 성공", "saveSuccess": "저장 성공",
"startTime": "시작 시간" "startTime": "시작 시간",
"timeSlot": "시간대"
}, },
"register": { "register": {
"emailSuffixWhitelist": "이메일 접미사 화이트리스트", "emailSuffixWhitelist": "이메일 접미사 화이트리스트",

View File

@ -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",

View File

@ -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",

View File

@ -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",

View File

@ -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",

View File

@ -61,7 +61,8 @@
"reset": "Сбросить", "reset": "Сбросить",
"save": "Сохранить", "save": "Сохранить",
"saveSuccess": "Успешно сохранено", "saveSuccess": "Успешно сохранено",
"startTime": "Время начала" "startTime": "Время начала",
"timeSlot": "Временной интервал"
}, },
"register": { "register": {
"emailSuffixWhitelist": "Белый список суффиксов электронной почты", "emailSuffixWhitelist": "Белый список суффиксов электронной почты",

View File

@ -61,7 +61,8 @@
"reset": "รีเซ็ต", "reset": "รีเซ็ต",
"save": "บันทึก", "save": "บันทึก",
"saveSuccess": "บันทึกสำเร็จ", "saveSuccess": "บันทึกสำเร็จ",
"startTime": "เวลาเริ่มต้น" "startTime": "เวลาเริ่มต้น",
"timeSlot": "ช่วงเวลา"
}, },
"register": { "register": {
"emailSuffixWhitelist": "รายการอนุญาตโดเมนอีเมล", "emailSuffixWhitelist": "รายการอนุญาตโดเมนอีเมล",

View File

@ -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",

View File

@ -61,7 +61,8 @@
"reset": "Скинути", "reset": "Скинути",
"save": "Зберегти", "save": "Зберегти",
"saveSuccess": "Збережено успішно", "saveSuccess": "Збережено успішно",
"startTime": "Час Початку" "startTime": "Час Початку",
"timeSlot": "Часовий проміжок"
}, },
"register": { "register": {
"emailSuffixWhitelist": "Білий список суфіксів електронної пошти", "emailSuffixWhitelist": "Білий список суфіксів електронної пошти",

View File

@ -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",

View File

@ -61,7 +61,8 @@
"reset": "重置", "reset": "重置",
"save": "保存", "save": "保存",
"saveSuccess": "保存成功", "saveSuccess": "保存成功",
"startTime": "开始时间" "startTime": "开始时间",
"timeSlot": "时间段"
}, },
"register": { "register": {
"emailSuffixWhitelist": "电子邮件后缀白名单", "emailSuffixWhitelist": "电子邮件后缀白名单",

View File

@ -61,7 +61,8 @@
"reset": "重設", "reset": "重設",
"save": "儲存", "save": "儲存",
"saveSuccess": "保存成功", "saveSuccess": "保存成功",
"startTime": "開始時間" "startTime": "開始時間",
"timeSlot": "時間段"
}, },
"register": { "register": {
"emailSuffixWhitelist": "電子郵件後綴白名單", "emailSuffixWhitelist": "電子郵件後綴白名單",

View File

@ -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) => (

View File

@ -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>
); );
} }