Initial commit MKD fixes
This commit is contained in:
365
components/applications/CreateOutageCard.tsx
Executable file
365
components/applications/CreateOutageCard.tsx
Executable file
@@ -0,0 +1,365 @@
|
||||
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>
|
||||
);
|
||||
};
|
||||
Reference in New Issue
Block a user