366 lines
17 KiB
TypeScript
366 lines
17 KiB
TypeScript
|
|
import React, { useState, useEffect } from 'react';
|
|||
|
|
import { X, Calendar, HelpCircle, Sparkles } from 'lucide-react';
|
|||
|
|
import { CreateOutagePayload, OutageWorkType } from '../../types';
|
|||
|
|
import { backendApi } from '../../services/apiClient';
|
|||
|
|
import { Building } from '../../types';
|
|||
|
|
|
|||
|
|
const CATEGORIES = [
|
|||
|
|
{ value: '', label: 'Выбрать' },
|
|||
|
|
{ value: 'water', label: 'Вода' },
|
|||
|
|
{ value: 'electric', label: 'Электрика' },
|
|||
|
|
{ value: 'heating', label: 'Отопление' },
|
|||
|
|
{ value: 'elevator', label: 'Лифт' },
|
|||
|
|
{ value: 'common', label: 'Общедомовое' },
|
|||
|
|
{ value: 'other', label: 'Другое' },
|
|||
|
|
];
|
|||
|
|
|
|||
|
|
const PROBLEMS_BY_CATEGORY: Record<string, { value: string; label: string }[]> = {
|
|||
|
|
water: [
|
|||
|
|
{ value: 'leak', label: 'Течь сантехнических приборов' },
|
|||
|
|
{ value: 'no_water', label: 'Нет воды' },
|
|||
|
|
{ value: 'blockage', label: 'Засор' },
|
|||
|
|
{ value: 'other', label: 'Другое' },
|
|||
|
|
],
|
|||
|
|
electric: [
|
|||
|
|
{ value: 'no_power', label: 'Нет электричества' },
|
|||
|
|
{ value: 'repair', label: 'Ремонт сети' },
|
|||
|
|
{ value: 'other', label: 'Другое' },
|
|||
|
|
],
|
|||
|
|
heating: [
|
|||
|
|
{ value: 'no_heat', label: 'Нет отопления' },
|
|||
|
|
{ value: 'repair', label: 'Ремонт системы' },
|
|||
|
|
{ value: 'other', label: 'Другое' },
|
|||
|
|
],
|
|||
|
|
elevator: [
|
|||
|
|
{ value: 'repair', label: 'Ремонт лифта' },
|
|||
|
|
{ value: 'inspection', label: 'Техобслуживание' },
|
|||
|
|
{ value: 'other', label: 'Другое' },
|
|||
|
|
],
|
|||
|
|
common: [
|
|||
|
|
{ value: 'repair', label: 'Ремонт' },
|
|||
|
|
{ value: 'other', label: 'Другое' },
|
|||
|
|
],
|
|||
|
|
other: [{ value: 'other', label: 'Другое' }],
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
const WORK_TYPES: { value: OutageWorkType; label: string }[] = [
|
|||
|
|
{ value: 'absent', label: 'Отсутствует' },
|
|||
|
|
{ value: 'planned', label: 'Плановый' },
|
|||
|
|
{ value: 'emergency', label: 'Аварийный' },
|
|||
|
|
];
|
|||
|
|
|
|||
|
|
const MAX_WHAT_HAPPENED = 1500;
|
|||
|
|
const MAX_RESIDENT_MESSAGE = 1000;
|
|||
|
|
|
|||
|
|
interface Props {
|
|||
|
|
isOpen: boolean;
|
|||
|
|
onClose: () => void;
|
|||
|
|
onSuccess: () => void;
|
|||
|
|
/** Если передан — форма «из дома», дом уже выбран; иначе — диспетчерская, выбор адресов */
|
|||
|
|
buildingId?: string;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
export const CreateOutageCard: React.FC<Props> = ({ isOpen, onClose, onSuccess, buildingId }) => {
|
|||
|
|
const [buildingIds, setBuildingIds] = useState<string[]>([]);
|
|||
|
|
const [selectAll, setSelectAll] = useState(false);
|
|||
|
|
const [startAt, setStartAt] = useState('');
|
|||
|
|
const [endAt, setEndAt] = useState('');
|
|||
|
|
const [category, setCategory] = useState('');
|
|||
|
|
const [problemDetail, setProblemDetail] = useState('');
|
|||
|
|
const [workType, setWorkType] = useState<OutageWorkType>('absent');
|
|||
|
|
const [whatHappened, setWhatHappened] = useState('');
|
|||
|
|
const [residentMessage, setResidentMessage] = useState('');
|
|||
|
|
const [generateNews, setGenerateNews] = useState(false);
|
|||
|
|
const [buildings, setBuildings] = useState<Building[]>([]);
|
|||
|
|
const [loading, setLoading] = useState(false);
|
|||
|
|
const [error, setError] = useState<string | null>(null);
|
|||
|
|
|
|||
|
|
const fromHouse = !!buildingId;
|
|||
|
|
|
|||
|
|
useEffect(() => {
|
|||
|
|
if (isOpen) {
|
|||
|
|
backendApi.getBuildings().then(setBuildings).catch(() => setBuildings([]));
|
|||
|
|
const now = new Date();
|
|||
|
|
setStartAt(now.toISOString().slice(0, 16));
|
|||
|
|
setEndAt('');
|
|||
|
|
if (fromHouse) {
|
|||
|
|
setBuildingIds(buildingId ? [buildingId] : []);
|
|||
|
|
} else {
|
|||
|
|
setBuildingIds([]);
|
|||
|
|
setSelectAll(false);
|
|||
|
|
}
|
|||
|
|
setCategory('');
|
|||
|
|
setProblemDetail('');
|
|||
|
|
setWorkType('absent');
|
|||
|
|
setWhatHappened('');
|
|||
|
|
setResidentMessage('');
|
|||
|
|
setGenerateNews(false);
|
|||
|
|
setError(null);
|
|||
|
|
}
|
|||
|
|
}, [isOpen, buildingId, fromHouse]);
|
|||
|
|
|
|||
|
|
useEffect(() => {
|
|||
|
|
if (selectAll && !fromHouse) {
|
|||
|
|
setBuildingIds(buildings.map(b => b.id));
|
|||
|
|
} else if (!selectAll && !fromHouse) {
|
|||
|
|
setBuildingIds([]);
|
|||
|
|
}
|
|||
|
|
}, [selectAll, fromHouse, buildings]);
|
|||
|
|
|
|||
|
|
if (!isOpen) return null;
|
|||
|
|
|
|||
|
|
const problemOptions = category ? (PROBLEMS_BY_CATEGORY[category] || [{ value: 'other', label: 'Другое' }]) : [];
|
|||
|
|
const selectedBuildingAddresses = fromHouse
|
|||
|
|
? buildings.filter(b => b.id === buildingId).map(b => (b.passport as any)?.address || b.id)
|
|||
|
|
: buildings.filter(b => buildingIds.includes(b.id)).map(b => (b.passport as any)?.address || b.id);
|
|||
|
|
|
|||
|
|
const handleSubmit = async (e: React.FormEvent) => {
|
|||
|
|
e.preventDefault();
|
|||
|
|
setError(null);
|
|||
|
|
const ids = fromHouse ? (buildingId ? [buildingId] : []) : buildingIds;
|
|||
|
|
if (ids.length === 0) {
|
|||
|
|
setError('Выберите хотя бы один адрес');
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
if (!startAt.trim()) {
|
|||
|
|
setError('Укажите начало работ');
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
if (!category) {
|
|||
|
|
setError('Выберите категорию');
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
if (!whatHappened.trim()) {
|
|||
|
|
setError('Заполните поле «Что случилось»');
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
if (whatHappened.length > MAX_WHAT_HAPPENED) {
|
|||
|
|
setError(`«Что случилось» — не более ${MAX_WHAT_HAPPENED} символов`);
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
setLoading(true);
|
|||
|
|
try {
|
|||
|
|
const categoryLabel = CATEGORIES.find(c => c.value === category)?.label || category;
|
|||
|
|
const payload: CreateOutagePayload = {
|
|||
|
|
buildingId: ids.length === 1 ? ids[0] : undefined,
|
|||
|
|
buildingIds: ids.length > 1 ? ids : undefined,
|
|||
|
|
startAt: new Date(startAt).toISOString(),
|
|||
|
|
endAt: endAt ? new Date(endAt).toISOString() : undefined,
|
|||
|
|
type: categoryLabel,
|
|||
|
|
description: whatHappened.trim(),
|
|||
|
|
authorName: 'Администратор',
|
|||
|
|
category: categoryLabel,
|
|||
|
|
problemDetail: problemOptions.find(p => p.value === problemDetail)?.label || problemDetail || undefined,
|
|||
|
|
workType,
|
|||
|
|
residentMessage: residentMessage.trim() || undefined,
|
|||
|
|
generateNews,
|
|||
|
|
};
|
|||
|
|
await backendApi.createOutage(payload);
|
|||
|
|
onSuccess();
|
|||
|
|
onClose();
|
|||
|
|
} catch (err: any) {
|
|||
|
|
setError(err.message || 'Ошибка при создании записи');
|
|||
|
|
} finally {
|
|||
|
|
setLoading(false);
|
|||
|
|
}
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
const handleClose = () => {
|
|||
|
|
if (!loading) onClose();
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
return (
|
|||
|
|
<div className="fixed inset-0 z-[100] flex items-center justify-center p-4 bg-black/50 backdrop-blur-sm animate-fade-in" onClick={handleClose}>
|
|||
|
|
<div className="bg-white rounded-3xl w-full max-w-2xl max-h-[90vh] overflow-y-auto shadow-2xl animate-scale-in" onClick={e => e.stopPropagation()}>
|
|||
|
|
<div className="sticky top-0 bg-white border-b border-slate-200 px-6 py-4 flex justify-between items-center rounded-t-3xl z-10">
|
|||
|
|
<h3 className="text-lg font-black text-slate-800">Новая запись</h3>
|
|||
|
|
<button type="button" onClick={handleClose} className="p-2 text-slate-400 hover:bg-slate-100 hover:text-slate-600 rounded-xl transition-colors disabled:opacity-50">
|
|||
|
|
<X className="w-5 h-5" />
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<form onSubmit={handleSubmit} className="p-6 space-y-5">
|
|||
|
|
{error && (
|
|||
|
|
<div className="bg-red-50 border border-red-200 text-red-600 text-sm px-4 py-3 rounded-xl">
|
|||
|
|
{error}
|
|||
|
|
</div>
|
|||
|
|
)}
|
|||
|
|
|
|||
|
|
{/* Адреса */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">
|
|||
|
|
Адреса <span className="text-red-500">*</span>
|
|||
|
|
</label>
|
|||
|
|
{fromHouse ? (
|
|||
|
|
<div className="px-4 py-3 border border-slate-200 rounded-xl text-sm text-slate-800 bg-slate-50">
|
|||
|
|
{selectedBuildingAddresses[0] || buildingId || '—'}
|
|||
|
|
</div>
|
|||
|
|
) : (
|
|||
|
|
<>
|
|||
|
|
<select
|
|||
|
|
multiple
|
|||
|
|
value={buildingIds}
|
|||
|
|
onChange={e => {
|
|||
|
|
const opts = Array.from(e.target.selectedOptions, o => o.value);
|
|||
|
|
setBuildingIds(opts);
|
|||
|
|
}}
|
|||
|
|
className="w-full px-4 py-3 border border-slate-200 rounded-xl text-sm outline-none focus:ring-2 focus:ring-primary-500 focus:border-primary-500 min-h-[100px]"
|
|||
|
|
>
|
|||
|
|
{buildings.map(b => (
|
|||
|
|
<option key={b.id} value={b.id}>{(b.passport as any)?.address || b.id}</option>
|
|||
|
|
))}
|
|||
|
|
</select>
|
|||
|
|
<label className="inline-flex items-center gap-2 cursor-pointer mt-2">
|
|||
|
|
<input type="checkbox" checked={selectAll} onChange={e => setSelectAll(e.target.checked)} className="rounded border-slate-300 text-primary-600" />
|
|||
|
|
<span className="text-xs font-bold text-slate-700">Выбрать все</span>
|
|||
|
|
</label>
|
|||
|
|
</>
|
|||
|
|
)}
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Начало работ */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">
|
|||
|
|
Начало работ <span className="text-red-500">*</span>
|
|||
|
|
</label>
|
|||
|
|
<div className="flex items-center gap-2">
|
|||
|
|
<input
|
|||
|
|
type="datetime-local"
|
|||
|
|
value={startAt}
|
|||
|
|
onChange={e => setStartAt(e.target.value)}
|
|||
|
|
className="flex-1 px-4 py-3 border border-slate-200 rounded-xl text-sm outline-none focus:ring-2 focus:ring-primary-500 focus:border-primary-500"
|
|||
|
|
/>
|
|||
|
|
<Calendar className="w-5 h-5 text-slate-400 shrink-0" />
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Завершение работ */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">Завершение работ</label>
|
|||
|
|
<div className="flex items-center gap-2">
|
|||
|
|
<input
|
|||
|
|
type="datetime-local"
|
|||
|
|
value={endAt}
|
|||
|
|
onChange={e => setEndAt(e.target.value)}
|
|||
|
|
min={startAt || undefined}
|
|||
|
|
className="flex-1 px-4 py-3 border border-slate-200 rounded-xl text-sm outline-none focus:ring-2 focus:ring-primary-500 focus:border-primary-500"
|
|||
|
|
/>
|
|||
|
|
<Calendar className="w-5 h-5 text-slate-400 shrink-0" />
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Категория */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">
|
|||
|
|
Категория <span className="text-red-500">*</span>
|
|||
|
|
</label>
|
|||
|
|
<select
|
|||
|
|
value={category}
|
|||
|
|
onChange={e => {
|
|||
|
|
setCategory(e.target.value);
|
|||
|
|
setProblemDetail('');
|
|||
|
|
}}
|
|||
|
|
className="w-full px-4 py-3 border border-slate-200 rounded-xl text-sm outline-none focus:ring-2 focus:ring-primary-500 focus:border-primary-500"
|
|||
|
|
>
|
|||
|
|
{CATEGORIES.map(opt => (
|
|||
|
|
<option key={opt.value || 'empty'} value={opt.value}>{opt.label}</option>
|
|||
|
|
))}
|
|||
|
|
</select>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Проблема (подкатегория) */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">Проблема</label>
|
|||
|
|
<select
|
|||
|
|
value={problemDetail}
|
|||
|
|
onChange={e => setProblemDetail(e.target.value)}
|
|||
|
|
className="w-full px-4 py-3 border border-slate-200 rounded-xl text-sm outline-none focus:ring-2 focus:ring-primary-500 focus:border-primary-500"
|
|||
|
|
>
|
|||
|
|
<option value="">{category ? 'Выбрать' : 'Выберите категорию'}</option>
|
|||
|
|
{problemOptions.map(opt => (
|
|||
|
|
<option key={opt.value} value={opt.value}>{opt.label}</option>
|
|||
|
|
))}
|
|||
|
|
</select>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Тип работ */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">Тип работ</label>
|
|||
|
|
<div className="flex p-1 bg-slate-200/50 rounded-2xl gap-1">
|
|||
|
|
{WORK_TYPES.map(opt => (
|
|||
|
|
<button
|
|||
|
|
key={opt.value}
|
|||
|
|
type="button"
|
|||
|
|
onClick={() => setWorkType(opt.value)}
|
|||
|
|
className={`flex-1 py-2 rounded-xl text-[10px] font-black uppercase tracking-wider transition-all ${workType === opt.value ? 'bg-white text-primary-600 shadow-sm' : 'text-slate-500 hover:text-slate-700'}`}
|
|||
|
|
>
|
|||
|
|
{opt.label}
|
|||
|
|
</button>
|
|||
|
|
))}
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Что случилось */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">
|
|||
|
|
Что случилось <span className="text-red-500">*</span>
|
|||
|
|
</label>
|
|||
|
|
<textarea
|
|||
|
|
value={whatHappened}
|
|||
|
|
onChange={e => setWhatHappened(e.target.value.slice(0, MAX_WHAT_HAPPENED))}
|
|||
|
|
placeholder="Опишите подробности"
|
|||
|
|
rows={4}
|
|||
|
|
className="w-full px-4 py-3 border border-slate-200 rounded-xl text-sm outline-none focus:ring-2 focus:ring-primary-500 focus:border-primary-500 resize-y"
|
|||
|
|
/>
|
|||
|
|
<p className="text-right text-[10px] font-bold text-slate-500 mt-1 uppercase">{whatHappened.length}/{MAX_WHAT_HAPPENED}</p>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Что сказать жителю */}
|
|||
|
|
<div>
|
|||
|
|
<label className="block text-xs font-black text-slate-700 mb-2 uppercase tracking-wider">Что сказать жителю</label>
|
|||
|
|
<textarea
|
|||
|
|
value={residentMessage}
|
|||
|
|
onChange={e => setResidentMessage(e.target.value.slice(0, MAX_RESIDENT_MESSAGE))}
|
|||
|
|
placeholder="Напишите, что скажет жителю сотрудник"
|
|||
|
|
rows={3}
|
|||
|
|
className="w-full px-4 py-3 border border-slate-200 rounded-xl text-sm outline-none focus:ring-2 focus:ring-primary-500 focus:border-primary-500 resize-y"
|
|||
|
|
/>
|
|||
|
|
<div className="flex items-center justify-between mt-1">
|
|||
|
|
<button type="button" className="inline-flex items-center gap-1.5 text-xs font-bold text-slate-500 hover:text-primary-600 transition-colors">
|
|||
|
|
<Sparkles className="w-3.5 h-3.5" /> Улучшить текст
|
|||
|
|
</button>
|
|||
|
|
<span className="text-[10px] font-bold text-slate-500 uppercase">{residentMessage.length}/{MAX_RESIDENT_MESSAGE}</span>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
{/* Сгенерировать новость */}
|
|||
|
|
<div className="flex items-center justify-between p-3 bg-slate-50 rounded-xl border border-slate-100">
|
|||
|
|
<div className="flex items-center gap-2">
|
|||
|
|
<span className="text-xs font-bold text-slate-700">Сгенерировать новость для жителей</span>
|
|||
|
|
<HelpCircle className="w-3.5 h-3.5 text-slate-400" />
|
|||
|
|
</div>
|
|||
|
|
<button
|
|||
|
|
type="button"
|
|||
|
|
role="switch"
|
|||
|
|
aria-checked={generateNews}
|
|||
|
|
onClick={() => setGenerateNews(!generateNews)}
|
|||
|
|
className={`relative inline-flex h-6 w-11 shrink-0 cursor-pointer rounded-full border-2 transition-colors focus:outline-none focus:ring-2 focus:ring-primary-500 ${generateNews ? 'bg-primary-600 border-primary-600' : 'bg-slate-200 border-slate-200'}`}
|
|||
|
|
>
|
|||
|
|
<span className={`absolute top-0.5 left-0.5 inline-block h-4 w-4 rounded-full bg-white shadow transition-transform ${generateNews ? 'translate-x-5' : 'translate-x-0'}`} />
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<div className="flex gap-3 pt-4 border-t border-slate-200">
|
|||
|
|
<button type="button" onClick={handleClose} disabled={loading} className="flex-1 px-4 py-3 bg-slate-100 text-slate-700 font-black text-sm rounded-xl hover:bg-slate-200 transition-colors disabled:opacity-50">
|
|||
|
|
Отмена
|
|||
|
|
</button>
|
|||
|
|
<button type="submit" disabled={loading} className="flex-1 px-4 py-3 bg-primary-600 text-white font-black text-sm rounded-xl hover:bg-primary-700 disabled:opacity-50 transition-colors">
|
|||
|
|
{loading ? 'Создание…' : 'Создать'}
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
</form>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
);
|
|||
|
|
};
|