feat: photo-search, system settings, ROADMAP

- PhotoSearchModal: Yandex photo-search с профилями доменов
- SystemSettings: управление app_settings (admin-only, /system)
- ROADMAP.md: актуальный план фич P1-P10
- Header, ChannelView, session: поддержка is_admin
This commit is contained in:
Nik (Claude)
2026-06-07 14:04:14 +03:00
parent 76eb519018
commit 2e550d2993
14 changed files with 931 additions and 38 deletions
+142 -32
View File
@@ -4,8 +4,9 @@ import Link from 'next/link';
import {
ArrowLeft, Sparkles, Wand2, Copy, Check, Loader2, Settings,
Image as ImageIcon, RefreshCw, Scissors, Maximize2, Zap, Heart,
MessageSquare, Pencil, X, ChevronDown, Send, Clock, Trash2
MessageSquare, Pencil, X, Send, Clock, Search, Camera, ExternalLink
} from 'lucide-react';
import PhotoSearchModal from './PhotoSearchModal';
const GOAL_LABELS = {
educational: 'Обучение', news: 'Новости',
@@ -22,6 +23,15 @@ const TRANSFORMS = [
{ action: 'forVk', label: 'Для ВК', icon: RefreshCw, desc: 'Адаптировать под ВКонтакте' },
];
// Хвостовая подпись «📷 Фото: domain» — добавляется к посту, можно убрать
function buildCaption(domain) {
return domain ? `\n\n📷 Фото: ${domain}` : '';
}
// Удаляет существующую подпись из текста поста (по любому домену)
function stripCaption(text) {
return (text || '').replace(/\n{1,2}📷\s*Фото:\s*[^\n]+\s*$/u, '').trimEnd();
}
export default function ChannelView({ channel }) {
const [topic, setTopic] = useState('');
const [generating, setGenerating] = useState(false);
@@ -36,8 +46,12 @@ export default function ChannelView({ channel }) {
// Картинка
const [image, setImage] = useState(null);
const [imageCredit, setImageCredit] = useState(null); // { domain, sourceUrl, title } | null
const [genImage, setGenImage] = useState(false);
// Photo search modal
const [showPhotoSearch, setShowPhotoSearch] = useState(false);
// Трансформации
const [transforming, setTransforming] = useState(false);
@@ -74,7 +88,6 @@ export default function ChannelView({ channel }) {
const [history, setHistory] = useState([]);
const [loadingHistory, setLoadingHistory] = useState(false);
// Подгрузка истории при монтировании
useEffect(() => { loadHistory(); }, []);
async function loadHistory() {
@@ -86,6 +99,23 @@ export default function ChannelView({ channel }) {
} catch {} finally { setLoadingHistory(false); }
}
function clearImage() {
setImage(null);
setImageCredit(null);
// Если в посте была подпись «📷 Фото: …» — убираем её при удалении фото
if (post) setPost(p => stripCaption(p));
}
function applyPhotoPick({ imageUrl, credit }) {
setImage(imageUrl);
setImageCredit(credit || null);
// Подменяем (или добавляем) caption
if (post && credit?.domain) {
setPost(p => stripCaption(p) + buildCaption(credit.domain));
}
setShowPhotoSearch(false);
}
async function savePost(status = 'draft', scheduledAt = null) {
if (!post) return;
setPublishing(true);
@@ -93,12 +123,12 @@ export default function ChannelView({ channel }) {
try {
let id = savedPostId;
if (!id) {
// Создаём
const res = await fetch('/api/user-posts', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
channel_id: channel.id, content: post, image_url: image,
channel_id: channel.id, content: post,
image_url: image, image_credit: imageCredit,
topic: topic.trim(), status, scheduled_at: scheduledAt,
}),
});
@@ -107,11 +137,15 @@ export default function ChannelView({ channel }) {
id = data.id;
setSavedPostId(id);
} else {
// Обновляем
const res = await fetch(`/api/user-posts/${id}`, {
method: 'PATCH',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ content: post, image_url: image, status, scheduled_at: scheduledAt }),
body: JSON.stringify({
content: post,
image_url: image,
image_credit: imageCredit,
status, scheduled_at: scheduledAt,
}),
});
if (!res.ok) throw new Error((await res.json()).error || 'Ошибка');
}
@@ -137,6 +171,7 @@ export default function ChannelView({ channel }) {
setPost(null);
setSavedPostId(null);
setImage(null);
setImageCredit(null);
setTopic('');
} catch (err) { setError(err.message); }
finally { setPublishing(false); }
@@ -151,6 +186,7 @@ export default function ChannelView({ channel }) {
setPost(null);
setSavedPostId(null);
setImage(null);
setImageCredit(null);
setTopic('');
}
@@ -183,14 +219,14 @@ export default function ChannelView({ channel }) {
if (!final) throw new Error('Таймаут — попробуй ещё раз');
if (final.status === 'failed') throw new Error(final.error || 'Генерация упала');
// Сохраняем предыдущий вариант в variants
if (asVariant && post) {
setVariants(v => [...v, { content: post, tokens, image }]);
setVariants(v => [...v, { content: post, tokens, image, imageCredit }]);
}
setPost(final.result);
setTokens({ in: final.tokens_in, out: final.tokens_out });
setImage(null); // сбрасываем картинку при новом посте
setImage(null);
setImageCredit(null);
} catch (err) {
setError(err.message);
} finally {
@@ -210,10 +246,10 @@ export default function ChannelView({ channel }) {
});
const data = await res.json();
if (!res.ok) throw new Error(data.error || 'Ошибка');
// Сохраняем текущий в варианты
setVariants(v => [...v, { content: post, tokens, image }]);
setVariants(v => [...v, { content: post, tokens, image, imageCredit }]);
setPost(data.content);
setImage(null);
setImageCredit(null);
} catch (err) {
setError(err.message);
} finally {
@@ -234,6 +270,7 @@ export default function ChannelView({ channel }) {
const data = await res.json();
if (!res.ok) throw new Error(data.error || 'Ошибка генерации картинки');
setImage(data.url);
setImageCredit(null); // сгенерированная — без credit'а
} catch (err) {
setError(err.message);
} finally {
@@ -245,12 +282,13 @@ export default function ChannelView({ channel }) {
const v = variants[idx];
setVariants(arr => {
const next = arr.filter((_, i) => i !== idx);
next.push({ content: post, tokens, image });
next.push({ content: post, tokens, image, imageCredit });
return next;
});
setPost(v.content);
setTokens(v.tokens);
setImage(v.image);
setImageCredit(v.imageCredit || null);
}
async function copy() {
@@ -299,7 +337,6 @@ export default function ChannelView({ channel }) {
</button>
</div>
{/* Список идей */}
{showIdeas && ideas.length > 0 && (
<div className="mb-3 p-3 rounded-lg bg-accent/5 border border-accent/20">
<div className="flex items-center justify-between mb-2">
@@ -384,7 +421,6 @@ export default function ChannelView({ channel }) {
</div>
</div>
{/* Сам пост — редактируемый или нет */}
{editing ? (
<textarea
value={post}
@@ -400,30 +436,86 @@ export default function ChannelView({ channel }) {
{/* Картинка к посту */}
{image && (
<div className="mt-4 relative">
<img src={image} alt="" className="w-full rounded-lg" />
<div className="mt-4">
<div className="relative">
<img src={image} alt="" className="w-full rounded-lg" referrerPolicy="no-referrer" />
<button
onClick={clearImage}
className="absolute top-2 right-2 p-1.5 rounded-lg bg-black/60 hover:bg-black/80 text-white"
title="Убрать картинку"
>
<X className="w-4 h-4" />
</button>
</div>
{imageCredit?.domain && (
<div className="mt-2 flex items-center justify-between flex-wrap gap-2 text-xs">
<div className="text-gray-500">
📷 Фото:{' '}
{imageCredit.sourceUrl ? (
<a
href={imageCredit.sourceUrl}
target="_blank"
rel="noopener noreferrer"
className="text-accent hover:underline inline-flex items-center gap-1"
>
{imageCredit.domain}
<ExternalLink className="w-3 h-3" />
</a>
) : (
<span className="text-accent">{imageCredit.domain}</span>
)}
</div>
<button
onClick={() => setPost(p => stripCaption(p))}
className="text-gray-500 hover:text-gray-300"
title="Убрать подпись «📷 Фото: …» из текста поста (саму картинку оставить)"
>
убрать подпись из поста
</button>
</div>
)}
</div>
)}
{/* Кнопки получения картинки */}
{!image && (
<div className="mt-4 grid sm:grid-cols-2 gap-2">
<button
onClick={() => setImage(null)}
className="absolute top-2 right-2 p-1.5 rounded-lg bg-black/60 hover:bg-black/80 text-white"
onClick={() => setShowPhotoSearch(true)}
className="inline-flex items-center justify-center gap-2 py-2.5 rounded-lg border border-dashed border-border hover:border-accent hover:bg-accent/5 text-sm text-gray-400 hover:text-accent transition-colors"
>
<X className="w-4 h-4" />
<Search className="w-4 h-4" /> Найти фото
</button>
<button
onClick={generateImage}
disabled={genImage}
className="inline-flex items-center justify-center gap-2 py-2.5 rounded-lg border border-dashed border-border hover:border-accent hover:bg-accent/5 text-sm text-gray-400 hover:text-accent transition-colors disabled:opacity-50"
>
{genImage ? (
<><Loader2 className="w-4 h-4 animate-spin" /> Генерирую (~30 сек)</>
) : (
<><Camera className="w-4 h-4" /> Сгенерировать (AI)</>
)}
</button>
</div>
)}
{/* Кнопка генерации картинки */}
{!image && (
<div className="mt-4">
{/* Если уже есть картинка — даём ещё раз поменять */}
{image && (
<div className="mt-2 flex flex-wrap gap-2 text-xs">
<button
onClick={() => setShowPhotoSearch(true)}
className="btn-ghost text-xs py-1"
>
<Search className="w-3.5 h-3.5" /> Другое фото
</button>
<button
onClick={generateImage}
disabled={genImage}
className="w-full inline-flex items-center justify-center gap-2 py-2.5 rounded-lg border border-dashed border-border hover:border-accent hover:bg-accent/5 text-sm text-gray-400 hover:text-accent transition-colors disabled:opacity-50"
className="btn-ghost text-xs py-1"
>
{genImage ? (
<><Loader2 className="w-4 h-4 animate-spin" /> Генерирую картинку... (~30 сек)</>
) : (
<><ImageIcon className="w-4 h-4" /> Сгенерировать картинку</>
)}
{genImage ? <Loader2 className="w-3.5 h-3.5 animate-spin" /> : <Camera className="w-3.5 h-3.5" />}
Заменить AI-картинкой
</button>
</div>
)}
@@ -457,7 +549,6 @@ export default function ChannelView({ channel }) {
</button>
</div>
{/* Планировщик */}
{showScheduler && (
<div className="mt-3 p-3 rounded-lg bg-surface2 border border-border">
<label className="label text-xs">Время публикации (МСК)</label>
@@ -498,6 +589,15 @@ export default function ChannelView({ channel }) {
</div>
)}
{/* Photo search modal */}
<PhotoSearchModal
open={showPhotoSearch}
onClose={() => setShowPhotoSearch(false)}
topic={topic}
post={post}
onPick={applyPhotoPick}
/>
{/* История вариантов */}
{variants.length > 0 && (
<div className="card p-5">
@@ -541,11 +641,11 @@ export default function ChannelView({ channel }) {
return (
<div key={p.id} className="flex items-start gap-3 p-3 rounded-lg bg-surface2 border border-border">
{p.image_url && (
<img src={p.image_url} alt="" className="w-14 h-14 rounded-lg object-cover shrink-0" />
<img src={p.image_url} alt="" className="w-14 h-14 rounded-lg object-cover shrink-0" referrerPolicy="no-referrer" />
)}
<div className="flex-1 min-w-0">
<div className="text-xs text-gray-400 line-clamp-2 whitespace-pre-wrap mb-1">{p.content.slice(0, 200)}</div>
<div className="flex items-center gap-2 text-xs">
<div className="flex items-center gap-2 text-xs flex-wrap">
<span className={`px-1.5 py-0.5 rounded font-medium ${statusColors[p.status] || statusColors.draft}`}>
{statusLabels[p.status] || p.status}
</span>
@@ -558,13 +658,23 @@ export default function ChannelView({ channel }) {
{!p.scheduled_at && !p.published_at && (
<span className="text-gray-500">{new Date(p.created_at).toLocaleString('ru-RU', { day:'numeric', month:'short', hour:'2-digit', minute:'2-digit' })}</span>
)}
{p.image_credit?.domain && (
<span className="text-gray-500">📷 {p.image_credit.domain}</span>
)}
{p.error && (
<span className="text-red-400 truncate">{p.error}</span>
)}
</div>
</div>
<button
onClick={() => { setPost(p.content); setImage(p.image_url); setSavedPostId(p.id); setTopic(p.topic || ''); window.scrollTo({top: 0, behavior: 'smooth'}); }}
onClick={() => {
setPost(p.content);
setImage(p.image_url);
setImageCredit(p.image_credit || null);
setSavedPostId(p.id);
setTopic(p.topic || '');
window.scrollTo({top: 0, behavior: 'smooth'});
}}
className="text-xs text-accent hover:underline shrink-0"
>
открыть
+11 -1
View File
@@ -1,7 +1,7 @@
'use client';
import Link from 'next/link';
import { useRouter } from 'next/navigation';
import { Sparkles, LogOut } from 'lucide-react';
import { Sparkles, LogOut, Settings2 } from 'lucide-react';
import ThemeToggle from './ThemeToggle';
export default function Header({ user }) {
@@ -18,6 +18,16 @@ export default function Header({ user }) {
<span className="font-bold">ZeroPost</span>
</Link>
<div className="flex items-center gap-2">
{user?.isAdmin && (
<Link
href="/system"
className="btn-ghost p-2 text-sm hidden sm:inline-flex"
title="Системные настройки"
>
<Settings2 className="w-4 h-4" />
<span className="hidden md:inline">Система</span>
</Link>
)}
<span className="text-sm text-gray-500 hidden sm:inline mr-2">{user?.email}</span>
<ThemeToggle />
<button onClick={logout} className="btn-ghost p-2" title="Выйти">
+287
View File
@@ -0,0 +1,287 @@
'use client';
import { useState, useEffect } from 'react';
import { Loader2, Search, X, AlertCircle, ExternalLink, Check } from 'lucide-react';
// Простая эвристика: если в тексте поста есть капитализованные ФИО (двусловные) —
// предлагаем их как стартовый query. Иначе берём первые слова темы.
function suggestQuery({ topic, post }) {
const text = `${topic || ''}\n${post || ''}`.trim();
if (!text) return '';
// Имена вида «Имя Фамилия» (две заглавные кириллицей или латиницей подряд)
const reName = /\b([А-ЯЁA-Z][а-яёa-z]{2,})\s+([А-ЯЁA-Z][а-яёa-z]{2,})/g;
const matches = [];
let m;
while ((m = reName.exec(text)) !== null) {
matches.push(`${m[1]} ${m[2]}`);
if (matches.length >= 3) break;
}
if (matches.length) return matches[0];
// Иначе — первые ~6 слов темы
return (topic || '').split(/\s+/).slice(0, 6).join(' ');
}
export default function PhotoSearchModal({ open, onClose, topic, post, onPick }) {
const [profiles, setProfiles] = useState([]);
const [profile, setProfile] = useState('general');
const [quota, setQuota] = useState(null);
const [query, setQuery] = useState('');
const [items, setItems] = useState([]);
const [searching, setSearching] = useState(false);
const [error, setError] = useState('');
const [meta, setMeta] = useState(null);
const [pickedIdx, setPickedIdx] = useState(null);
// Загружаем профили + квоту при первом открытии
useEffect(() => {
if (!open) return;
let cancelled = false;
(async () => {
try {
const [profRes, quotaRes] = await Promise.all([
fetch('/api/photo-search/profiles'),
fetch('/api/photo-search/quota'),
]);
const profData = profRes.ok ? await profRes.json() : [];
const quotaData = quotaRes.ok ? await quotaRes.json() : null;
if (cancelled) return;
setProfiles(profData);
setQuota(quotaData);
// Дефолт-профиль: general (если есть), иначе первый
if (profData.length && !profData.find(p => p.slug === profile)) {
setProfile(profData[0].slug);
}
} catch (e) {
if (!cancelled) setError(e.message);
}
})();
return () => { cancelled = true; };
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [open]);
// Подсказываем стартовый запрос
useEffect(() => {
if (!open) return;
if (!query) setQuery(suggestQuery({ topic, post }));
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [open]);
async function search() {
if (!query.trim() || searching) return;
setSearching(true);
setError('');
setItems([]);
setPickedIdx(null);
try {
const res = await fetch('/api/photo-search/by-query', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ query: query.trim(), profile, num: 6 }),
});
const data = await res.json();
if (!res.ok) {
if (data.code === 'DAILY_LIMIT_EXCEEDED') {
throw new Error('Дневной лимит поиска фото исчерпан. Попробуй завтра или подними лимит в /system.');
}
throw new Error(data.error || `HTTP ${res.status}`);
}
setItems(data.items || []);
setMeta({
total: data.total,
raw: data.raw_count,
filtered: data.filtered_count,
elapsedMs: data.elapsed_ms,
domains: data.domains || [],
});
if (data.quota) setQuota(data.quota);
if (!data.items || data.items.length === 0) {
setError('Ничего не нашлось в whitelisted доменах. Попробуй другой профиль или уточни запрос.');
}
} catch (e) {
setError(e.message);
} finally {
setSearching(false);
}
}
function pick(idx) {
const item = items[idx];
if (!item) return;
setPickedIdx(idx);
onPick?.({
imageUrl: item.imageUrl,
thumbUrl: item.thumbUrl,
credit: {
domain: item.sourceDomain || null,
sourceUrl: item.sourceUrl || null,
title: item.title || null,
},
});
}
if (!open) return null;
return (
<div
className="fixed inset-0 z-50 bg-black/60 backdrop-blur-sm flex items-start sm:items-center justify-center p-3 sm:p-6 overflow-y-auto"
onClick={onClose}
>
<div
className="card w-full max-w-3xl my-4"
onClick={e => e.stopPropagation()}
>
{/* Header */}
<div className="flex items-start justify-between p-5 border-b border-border gap-3">
<div>
<h3 className="font-semibold flex items-center gap-2">
<Search className="w-4 h-4 text-accent" />
Найти фото
</h3>
<p className="text-xs text-gray-500 mt-1">
Поиск по доменам из whitelist'а профиля. Используется Yandex Search API.
</p>
</div>
<button onClick={onClose} className="btn-ghost p-2" title="Закрыть">
<X className="w-4 h-4" />
</button>
</div>
{/* Controls */}
<div className="p-5 space-y-3 border-b border-border">
<div className="grid sm:grid-cols-[1fr_auto] gap-2">
<div>
<label className="label text-xs">Запрос</label>
<input
className="input text-sm"
value={query}
onChange={e => setQuery(e.target.value)}
onKeyDown={e => { if (e.key === 'Enter') search(); }}
placeholder="Имя, событие, объект…"
autoFocus
/>
</div>
<div>
<label className="label text-xs">Профиль</label>
<select
className="input text-sm"
value={profile}
onChange={e => setProfile(e.target.value)}
>
{profiles.length === 0 && <option value="general">general</option>}
{profiles.map(p => (
<option key={p.slug} value={p.slug}>
{p.name || p.slug} {p.domains?.length ? `(${p.domains.length})` : ''}
</option>
))}
</select>
</div>
</div>
<div className="flex items-center justify-between gap-3 flex-wrap">
<div className="text-xs text-gray-500">
{quota && (
<>Квота сегодня: <b>{quota.used}</b> / {quota.limit}{' '}
{quota.remaining === 0 && <span className="text-amber-500">(исчерпана)</span>}
</>
)}
</div>
<button
onClick={search}
disabled={searching || !query.trim() || (quota?.remaining === 0)}
className="btn-primary text-sm"
>
{searching ? <Loader2 className="w-4 h-4 animate-spin" /> : <Search className="w-4 h-4" />}
{searching ? 'Ищу' : 'Найти'}
</button>
</div>
{profiles.find(p => p.slug === profile)?.domains?.length > 0 && (
<div className="text-[11px] text-gray-500">
Whitelist:{' '}
{profiles.find(p => p.slug === profile).domains.slice(0, 8).join(', ')}
{profiles.find(p => p.slug === profile).domains.length > 8 && ''}
</div>
)}
</div>
{/* Body */}
<div className="p-5">
{error && (
<div className="flex items-start gap-2 text-sm text-amber-500 bg-amber-500/10 border border-amber-500/30 rounded-lg p-3 mb-3">
<AlertCircle className="w-4 h-4 mt-0.5 shrink-0" />
<div>{error}</div>
</div>
)}
{meta && !error && (
<div className="text-[11px] text-gray-500 mb-3">
Найдено всего: {meta.total} · после фильтра: {meta.filtered} · показано: {items.length} · {meta.elapsedMs} мс
</div>
)}
{items.length > 0 && (
<div className="grid grid-cols-2 sm:grid-cols-3 gap-3">
{items.map((it, i) => (
<button
key={i}
onClick={() => pick(i)}
className={`group relative rounded-lg overflow-hidden border transition-all text-left ${
pickedIdx === i
? 'border-accent ring-2 ring-accent/40'
: 'border-border hover:border-accent/60'
}`}
title={it.title || it.sourceUrl}
>
<div className="aspect-square bg-surface2 overflow-hidden">
<img
src={it.thumbUrl || it.imageUrl}
alt=""
className="w-full h-full object-cover group-hover:scale-105 transition-transform"
loading="lazy"
referrerPolicy="no-referrer"
onError={(e) => { e.currentTarget.style.opacity = '0.3'; }}
/>
</div>
<div className="p-2">
<div className="text-[10px] text-accent font-mono truncate">{it.sourceDomain}</div>
{it.title && (
<div className="text-[11px] text-gray-500 line-clamp-2 mt-0.5">{it.title}</div>
)}
<div className="text-[10px] text-gray-500 mt-1">{it.width}×{it.height}</div>
</div>
{pickedIdx === i && (
<div className="absolute top-1.5 right-1.5 bg-accent text-white rounded-full p-1">
<Check className="w-3 h-3" />
</div>
)}
{it.sourceUrl && (
<a
href={it.sourceUrl}
target="_blank"
rel="noopener noreferrer"
onClick={e => e.stopPropagation()}
className="absolute top-1.5 left-1.5 bg-black/60 hover:bg-black/80 text-white rounded-full p-1"
title="Открыть источник"
>
<ExternalLink className="w-3 h-3" />
</a>
)}
</button>
))}
</div>
)}
{items.length === 0 && !error && !searching && (
<div className="text-sm text-gray-500 text-center py-10">
Введи запрос и нажми «Найти»
</div>
)}
</div>
{/* Footer */}
<div className="p-4 border-t border-border flex items-center justify-end gap-2">
<button onClick={onClose} className="btn-ghost text-sm">Закрыть</button>
</div>
</div>
</div>
);
}
+192
View File
@@ -0,0 +1,192 @@
'use client';
import { useState, useEffect } from 'react';
import { Loader2, Save, Eye, EyeOff, RefreshCw, Check, AlertCircle } from 'lucide-react';
// Категории, которые управляются здесь (в админке tool, а не в админке блога).
// Категория `engine` (TELEGRAM_API_BASE и т.п.) намеренно живёт в zeropost.ru/admin.
const CATEGORIES = [
{ slug: 'photo_search', title: 'Поиск фото',
hint: 'Yandex Search API: provider, ключ, folder, лимиты.' },
// Сюда позже: { slug: 'billing', ... }, { slug: 'serpapi', ... }
];
export default function SystemSettings() {
const [byCategory, setByCategory] = useState({});
const [loading, setLoading] = useState(true);
const [error, setError] = useState('');
async function load() {
setLoading(true);
setError('');
try {
const result = {};
for (const cat of CATEGORIES) {
const res = await fetch(`/api/admin/settings?category=${cat.slug}`);
if (!res.ok) throw new Error((await res.json()).error || `HTTP ${res.status}`);
result[cat.slug] = await res.json();
}
setByCategory(result);
} catch (err) {
setError(err.message);
} finally {
setLoading(false);
}
}
useEffect(() => { load(); }, []);
if (loading) {
return (
<div className="card p-12 text-center">
<Loader2 className="w-6 h-6 animate-spin mx-auto text-accent" />
</div>
);
}
if (error) {
return (
<div className="card p-5 border-red-500/40">
<div className="flex items-center gap-2 text-red-400">
<AlertCircle className="w-4 h-4" />
{error}
</div>
<button onClick={load} className="btn-ghost mt-3 text-sm">
<RefreshCw className="w-4 h-4" /> Повторить
</button>
</div>
);
}
return (
<div className="space-y-6">
{CATEGORIES.map(cat => (
<CategoryBlock
key={cat.slug}
category={cat}
rows={byCategory[cat.slug] || []}
onSaved={() => load()}
/>
))}
</div>
);
}
function CategoryBlock({ category, rows, onSaved }) {
return (
<section className="card p-5">
<div className="mb-4">
<h2 className="font-semibold">{category.title}</h2>
{category.hint && <p className="text-xs text-gray-500 mt-1">{category.hint}</p>}
</div>
{rows.length === 0 ? (
<p className="text-sm text-gray-500">Нет настроек в этой категории.</p>
) : (
<div className="space-y-3">
{rows.map(r => (
<SettingRow key={r.key} row={r} onSaved={onSaved} />
))}
</div>
)}
</section>
);
}
function SettingRow({ row, onSaved }) {
const [value, setValue] = useState(row.value ?? '');
const [saving, setSaving] = useState(false);
const [saved, setSaved] = useState(false);
const [reveal, setReveal] = useState(false);
const [err, setErr] = useState('');
// если row.value меняется снаружи — синхронизируем
useEffect(() => { setValue(row.value ?? ''); }, [row.value]);
const isSecret = row.is_secret;
const dirty = value !== (row.value ?? '');
async function save() {
setSaving(true);
setErr('');
try {
const res = await fetch(`/api/admin/settings/${encodeURIComponent(row.key)}`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ value: value === '' ? null : value }),
});
if (!res.ok) throw new Error((await res.json()).error || `HTTP ${res.status}`);
setSaved(true);
setTimeout(() => setSaved(false), 1500);
onSaved?.();
} catch (e) {
setErr(e.message);
} finally {
setSaving(false);
}
}
// Маскировка отображения секретов: показываем хвост ***last4 пока не reveal=true
const masked = isSecret && value && !reveal
? '•'.repeat(Math.max(value.length - 4, 4)) + value.slice(-4)
: value;
return (
<div className="rounded-lg border border-border bg-surface2/50 p-3">
<div className="flex items-start justify-between flex-wrap gap-2 mb-2">
<div className="min-w-0">
<div className="flex items-center gap-2">
<code className="text-sm font-mono">{row.key}</code>
{isSecret && (
<span className="text-[10px] uppercase tracking-wide px-1.5 py-0.5 rounded bg-amber-500/15 text-amber-500">
secret
</span>
)}
</div>
{row.description && (
<p className="text-xs text-gray-500 mt-1">{row.description}</p>
)}
</div>
</div>
<div className="flex items-center gap-2">
<input
type={isSecret && !reveal ? 'password' : 'text'}
className="input text-sm font-mono"
value={reveal || !isSecret ? value : masked}
onChange={e => {
// При маскированном просмотре редактирование запрещаем — пусть сначала откроют
if (isSecret && !reveal) return;
setValue(e.target.value);
}}
placeholder={isSecret ? '(скрыто)' : '(пусто)'}
spellCheck={false}
/>
{isSecret && (
<button
type="button"
onClick={() => setReveal(v => !v)}
className="btn-ghost p-2"
title={reveal ? 'Скрыть' : 'Показать'}
>
{reveal ? <EyeOff className="w-4 h-4" /> : <Eye className="w-4 h-4" />}
</button>
)}
<button
onClick={save}
disabled={saving || !dirty}
className="btn-primary text-sm"
>
{saving ? <Loader2 className="w-4 h-4 animate-spin" />
: saved ? <Check className="w-4 h-4" />
: <Save className="w-4 h-4" />}
{saved ? 'Сохранено' : 'Сохранить'}
</button>
</div>
{err && (
<div className="text-xs text-red-400 mt-2">{err}</div>
)}
<div className="text-[11px] text-gray-500 mt-2">
Категория: <code>{row.category}</code> · обновлено{' '}
{row.updated_at ? new Date(row.updated_at).toLocaleString('ru-RU') : '—'}
</div>
</div>
);
}