9 Commits

Author SHA1 Message Date
AR 15 M4 6ce176f970 chore: close Sprint 4, open Sprint 5
- Sidebar: Hero/CEO-текст unlocked (not soon), Sprint 5 · v0.5.0
- SPRINTS.md: Sprint 4 marked  ЗАВЕРШЁН, Sprint 5 plan updated
- LLM_CONTEXT.md: v4.1 — cards/badges/alerts section 9b added

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 23:19:38 +05:00
AR 15 M4 6c5b571884 feat(sprint4): add cards page — DoctorCard, NewsCard, ReviewCard, PriceCard, ServiceCard, badges/tags/alerts
- /components/cards — new page with 5 card types + badges/tags/alerts + LLM block v1.0
- DoctorCard: photo 110×160px placeholder, name, specialty, experience, outline button
- NewsCard: hover bg #eef4d1 + box-shadow (matches real site CSS), CSS class .bb-news-card
- ReviewCard: star rating 1-5 (SVG), text 4-line clamp, bg #eef4d1
- PriceCard: service + price + description + button, highlighted variant (blue border/bg)
- ServiceCard: emoji icon 48×48px, title, description, link
- Badges: 6 color variants (primary/success/warning/danger/neutral/outline)
- Tags: default/active state filters
- Alerts: 4 types (info/success/warning/error) with icons
- globals.css: .bb-news-card:hover, .bb-service-card:hover CSS rules (Sprint 4)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 23:09:22 +05:00
AR 15 M4 2b66fb9cc1 docs: update LLM_CONTEXT.md to v4.0 (Sprint 4 start)
- All 6 Oracal brand colors updated to exact catalog RGB values
- Contrast table recalculated for new hex values
- Color mapping notes updated (typography of differences)
- Logo color variants updated with new hex refs
- CSS vars section updated
- Transport hex refs updated
- Color rules updated (white on 073M/066M/050M/080M, dark on 053M/081M)
- Section 9a: button variants now match real site (primary/outline/teal/pill)
- Section 9a: form controls match real site (height 50px, border #ccc)
- Section 7: /components/cards added as Sprint 4 in progress
- History: version 4.0 added

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 23:04:32 +05:00
AR 15 M4 3811c579fd fix(colors): update Oracal brand colors to exact catalog RGB values
053M: #7ecfca → #0089C3 (rgb 0,137,195)
073M: #5b7b87 → #53514E (rgb 83,81,78)
066M: #5bb5ad → #00818C (rgb 0,129,140)
050M: #1b4c72 → #1B2E5D (rgb 27,46,93)
081M: #c4a882 → #A8885C (rgb 168,136,92)
080M: #5c2e0e → #432F1E (rgb 67,47,30)

- Update globals.css CSS vars + sidebar active bg (→ #dff0fa)
- Update focus box-shadow rgba for new 053M blue
- colors/page.tsx: BRAND_COLORS, COLOR_MAPPING notes, CONTRAST_PAIRS
- LLM text v2.2: new hex/rgb, recalculated WCAG contrast ratios,
  updated rules (white text on 073M/066M/050M/080M, dark on 053M/081M)
- Info banner changed from ⚠️ "приблизительны" to ℹ "точные RGB"

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 22:46:50 +05:00
AR 15 M4 2b9319e70d chore(sprint4): start Sprint 4 — cards
- Sidebar: Sprint 4 · v0.4.0, /components/cards removed from "soon"

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 21:42:46 +05:00
AR 15 M4 d6610a288b chore: add offline photos + mark Sprint 3 complete in SPRINTS.md
- Add 27 offline photos (navigation + transport) that were missing from git
  but referenced by Sprint 2 pages
- SPRINTS.md: mark Sprint 3 as ЗАВЕРШЁН with actual results documented

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 21:40:22 +05:00
AR 15 M4 77c9733144 fix(forms): update form controls to match real site oclinica.ru styling
- .bb-input: height 50px, padding 10px 12px, border 1px solid #ccc, border-radius 4px
  (matches entityform input[type=text] from perm.oclinica.ru)
- .bb-select: height 50px, padding 10px with arrow, same border/radius
- .bb-textarea: same border 1px #ccc, border-radius 4px (was 8px/1.5px teal)
- forms/page.tsx v2.0: added "Контекст применения" section with
  where-used table and realistic form mockup (bg #b8e6ed as on site),
  added "CSS с сайта" code block, updated LLM block to v2.0

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 21:19:09 +05:00
AR 15 M4 c1731615ab fix(buttons): переделаны кнопки под реальный сайт oclinica.ru
Анализ CSS сайта (style.css темы clinic_bootstrap_mobile) выявил
4 реальных типа кнопок — заменены ранее придуманные варианты:

- primary  → коралловый #FFA39C + shadow (форм-сабмит «Запишите меня!»)
- outline  → белый + бежевая рамка #BF9975 («Записаться на прием»)
- teal     → бирюзовый #60959c («Позвонить»)
- pill     → кремовый #e9e4d4 + radius 25px («Заказать звонок»)

Удалены: secondary, ghost, danger (не существуют на реальном сайте)
Добавлен раздел «CSS с сайта» с точными значениями
Добавлена таблица «Где применяется» с реальными CSS-классами сайта
LLM-блок обновлён до v2.0

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 20:41:27 +05:00
AR 15 M4 0855892643 feat(sprint3): кнопки, форм-контролы, LLM-блоки — Sprint 3 v0.3.0
- components/ui/Button.tsx — компонент Button (primary/secondary/ghost/danger, sm/md/lg, loading/disabled)
- components/ui/CodeCopy.tsx — компонент копирования кода (clipboard API)
- components/ui/Toggle.tsx — тумблер (client component, bb-toggle-track/thumb)
- globals.css — CSS-классы: bb-btn, bb-input/textarea/select, bb-checkbox/radio, bb-toggle, @keyframes bb-spin
- app/components/buttons/page.tsx — страница «Кнопки» (варианты, размеры, состояния, code copy, LLM-блок)
- app/components/forms/page.tsx — страница «Форм-контролы» (Input/Textarea/Select/Checkbox/Radio/Toggle, LLM-блок)
- foundation/logo/page.tsx — добавлен LLM-блок v1.0
- Sidebar: убраны «скоро» с Кнопок и Форм-контролов, версия Sprint 3 · v0.3.0
- docs/LLM_CONTEXT.md → версия 3.0, добавлена секция 9a с компонентами

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-22 20:20:41 +05:00
39 changed files with 2578 additions and 159 deletions
+573
View File
@@ -0,0 +1,573 @@
import type { Metadata } from "next";
import { Button } from "@/components/ui/Button";
import { CodeCopy } from "@/components/ui/CodeCopy";
import { LlmBlock, LlmSection, LlmTable, LlmRules } from "@/components/llm/LlmBlock";
export const metadata: Metadata = {
title: "Кнопки. Цифровой брендбук Клиники ухо, горло, нос им. проф. Е.Н.Оленевой",
};
function Section({
id,
title,
subtitle,
children,
}: {
id?: string;
title: string;
subtitle?: string;
children: React.ReactNode;
}) {
return (
<section id={id} className="mb-12">
<div className="mb-6">
<h2 className="text-xl font-semibold" style={{ color: "var(--bb-text)" }}>
{title}
</h2>
{subtitle && (
<p className="mt-1 text-sm" style={{ color: "var(--bb-text-muted)" }}>
{subtitle}
</p>
)}
</div>
{children}
</section>
);
}
const VARIANTS = [
{
variant: "primary" as const,
name: "Primary",
label: "Запишитесь к нам",
cssClass: ".bb-btn-primary",
bg: "#FFA39C",
border: "#FF847B",
textColor: "#fff",
radius: "7px",
shadow: "да",
where: "Кнопка отправки форм записи",
example: "«Запишите меня!»",
note: "Коралловый — самый заметный акцент на странице. Всегда один в форме.",
},
{
variant: "outline" as const,
name: "Outline",
label: "Записаться на приём",
cssClass: ".bb-btn-outline",
bg: "#fff",
border: "#BF9975",
textColor: "#BF9975",
radius: "7px",
shadow: "нет",
where: "Хедер, навигация, ссылки-кнопки",
example: "«Записаться на прием», «Все новости»",
note: "Бежевая рамка — ненавязчивый вторичный CTA. Не конкурирует с основной формой.",
},
{
variant: "teal" as const,
name: "Teal",
label: "Позвонить",
cssClass: ".bb-btn-teal",
bg: "#60959c",
border: "прозрачный",
textColor: "#fff",
radius: "7px",
shadow: "нет",
where: "Контактные действия — звонок",
example: "«Позвонить»",
note: "Серо-бирюзовый — цвет из реального CSS сайта. Близок к Oracal 066M.",
},
{
variant: "pill" as const,
name: "Pill",
label: "Заказать звонок",
cssClass: ".bb-btn-pill",
bg: "#e9e4d4",
border: "#d5cfbd",
textColor: "#333",
radius: "25px",
shadow: "нет",
where: "Модальные триггеры, мягкий CTA",
example: "«Заказать звонок»",
note: "Кремовый фон + pill-форма — мягкий стиль. Используется для открытия модальных окон.",
},
];
const LLM_BUTTONS_TEXT = `КНОПКИ — LLM-спецификация (с реального сайта oclinica.ru)
Версия: v2.0 · /components/buttons
Источник CSS: perm.oclinica.ru/.../style.css
ВАРИАНТЫ (реальный сайт)
Вариант | CSS класс | Фон | Текст | Граница | Radius | Shadow | Применение
primary | .bb-btn-primary | #FFA39C | #fff | #FF847B | 7px | да | Форм-сабмит «Запишите меня!»
outline | .bb-btn-outline | #fff | #BF9975 | #BF9975 | 7px | нет | Хедер «Записаться на прием», ссылки-кнопки
teal | .bb-btn-teal | #60959c | #fff | нет | 7px | нет | Звонок «Позвонить»
pill | .bb-btn-pill | #e9e4d4 | #333 | #d5cfbd | 25px | нет | Callback «Заказать звонок»
CSS С САЙТА (точные значения)
/* форм-кнопка «Запишите меня!» */
button { background:#FFA39C; color:white; font-weight:bold; border:solid 1px #FF847B;
height:42px; font-size:18px; box-shadow:0px 0px 5px rgba(0,0,0,0.5),0px 4px 5px rgba(0,0,0,0.3); }
/* appointment — «Записаться на прием» */
.appointment { background:#FFF; border:#BF9975 solid 1px; color:#BF9975;
font-size:14px; line-height:38px; padding:3px 12px; border-radius:7px; }
/* show-phone — «Позвонить» */
.show-phone { background:rgb(96,149,156); color:#fff; border-radius:7px;
font-size:14px; line-height:38px; padding:3px 12px; }
/* callback — «Заказать звонок» */
a.callback_url { background:#e9e4d4; border:#d5cfbd solid 1px; color:#000;
border-radius:25px; font-size:16px; padding:6px 18px; }
РАЗМЕРЫ (брендбук-компонент)
Размер | CSS класс | padding | font-size | Применение
sm | .bb-btn-sm | 4px 11px | 13px | Компактные контексты
md | .bb-btn-md | 8px 16px | 14px | Стандарт (appointment, teal, pill)
lg | .bb-btn-lg | 10px 24px | 18px + bold | Форм-сабмит (соответствует реальному сайту)
ПРАВИЛА ПРИМЕНЕНИЯ
✓ primary (коралловый) — только для главного CTA в форме записи
✓ outline (бежевый) — хедер, навигация, ссылки-кнопки на странице
✓ teal (бирюзовый) — контактные действия (звонок, направление)
✓ pill (кремовый) — открытие модальных окон, мягкий callback
✓ Не более одного primary на форму
✕ Не менять цвета вне фирменной палитры сайта
✕ Primary — не для навигационных ссылок
✕ Не накладывать тень на outline/teal/pill`.trim();
export default function ButtonsPage() {
const codeHtml = `<!-- Primary — форм-кнопка «Запишите меня!» -->
<button class="bb-btn bb-btn-lg bb-btn-primary">Запишите меня!</button>
<!-- Outline — appointment «Записаться на прием» -->
<a class="bb-btn bb-btn-md bb-btn-outline" href="#form">Записаться на прием</a>
<!-- Teal — «Позвонить» -->
<a class="bb-btn bb-btn-md bb-btn-teal" href="tel:+73422250662">Позвонить</a>
<!-- Pill — «Заказать звонок» -->
<a class="bb-btn bb-btn-md bb-btn-pill" href="#callback">Заказать звонок</a>`;
const codeReact = `import { Button } from "@/components/ui/Button";
// Форм-кнопка (главный CTA)
<Button variant="primary" size="lg">Запишите меня!</Button>
// Запись из хедера / навигации
<Button variant="outline" size="md">Записаться на прием</Button>
// Звонок
<Button variant="teal" size="md">Позвонить</Button>
// Заказать звонок (открывает модал)
<Button variant="pill" size="md">Заказать звонок</Button>
// С loading-состоянием
<Button variant="primary" size="lg" loading>Отправляем...</Button>`;
const codeSiteExact = `/* Точный CSS с сайта oclinica.ru (style.css) */
/* Форм-кнопка — кнопка отправки форм записи */
#block-entityform-block-feedback button,
#block-entityform-block-lor-form button {
background: #FFA39C;
color: white;
font-weight: bold;
border: solid 1px #FF847B;
width: 300px;
height: 42px;
font-size: 18px;
box-shadow: 0px 0px 5px rgba(0,0,0,0.5), 0px 4px 5px rgba(0,0,0,0.3);
}
/* Кнопка «Записаться на прием» в хедере */
#block-block-15 .appointment {
background: #FFF;
border: #BF9975 solid 1px;
color: #BF9975;
font-size: 14px;
line-height: 38px;
padding: 3px 12px;
border-radius: 7px;
}
/* Кнопка «Позвонить» */
.show-phone {
background: rgb(96, 149, 156); /* #60959c */
color: #fff;
border-radius: 7px;
font-size: 14px;
line-height: 38px;
padding: 3px 12px;
}
/* Кнопка «Заказать звонок» */
a.callback_url {
background: #e9e4d4;
border: #d5cfbd solid 1px;
color: #000;
border-radius: 25px;
font-size: 16px;
padding: 6px 18px;
}`;
return (
<div className="max-w-4xl mx-auto px-8 py-10">
{/* Заголовок */}
<div className="mb-10 pb-6 border-b" style={{ borderColor: "var(--bb-border)" }}>
<p
className="text-xs font-semibold uppercase tracking-widest mb-2"
style={{ color: "var(--brand-053m)" }}
>
Компоненты 3.1
</p>
<h1 className="text-3xl font-semibold mb-3" style={{ color: "var(--bb-text)" }}>
Кнопки
</h1>
<p className="text-base max-w-2xl" style={{ color: "var(--bb-text-muted)" }}>
Кнопки скопированы с реального сайта{" "}
<span className="font-mono text-sm" style={{ color: "var(--bb-text)" }}>
oclinica.ru
</span>
. Цвета, размеры и тени взяты напрямую из CSS темы{" "}
<span className="font-mono text-sm" style={{ color: "var(--bb-text)" }}>
clinic_bootstrap_mobile/css/style.css
</span>
.
</p>
<div
className="mt-4 px-4 py-3 rounded-lg border text-sm flex items-center gap-2"
style={{ borderColor: "#e0f5f4", background: "#f8fffe", color: "var(--bb-text-muted)" }}
>
<span style={{ color: "var(--brand-053m)", fontWeight: 600 }}>Источник</span>
<span>
CSS сайта проанализирован 2026-03-22 4 типа кнопок с реальными значениями.
</span>
</div>
</div>
{/* 1. Варианты */}
<Section
id="variants"
title="Варианты"
subtitle="Четыре типа кнопок с реального сайта клиники."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4 mb-6">
{VARIANTS.map(({ variant, name, label, where, example, note, bg, border, textColor, radius, shadow }) => (
<div
key={variant}
className="rounded-xl border p-5 flex flex-col gap-4"
style={{ borderColor: "var(--bb-border)" }}
>
{/* Превью */}
<div
className="flex items-center justify-center py-6 rounded-lg"
style={{ background: "var(--bb-sidebar-bg)" }}
>
<Button variant={variant} size="md">
{label}
</Button>
</div>
{/* Инфо */}
<div>
<p className="font-semibold text-sm mb-1" style={{ color: "var(--bb-text)" }}>
{name}
</p>
<p className="text-xs mb-2 leading-relaxed" style={{ color: "var(--bb-text-muted)" }}>
{note}
</p>
<div className="flex flex-wrap gap-1.5 mb-2">
{[
{ k: "bg", v: bg },
{ k: "text", v: textColor },
{ k: "border", v: border },
{ k: "radius", v: radius },
...(shadow === "да" ? [{ k: "shadow", v: "да" }] : []),
].map(({ k, v }) => (
<span
key={k}
className="text-[10px] font-mono px-1.5 py-0.5 rounded"
style={{ background: "#f3f4f6", color: "var(--bb-text-muted)" }}
>
{k}: {v}
</span>
))}
</div>
<div
className="rounded p-2.5 text-xs"
style={{ background: "#f8f9fa", color: "var(--bb-text-muted)" }}
>
<span className="font-medium" style={{ color: "var(--bb-text)" }}>
Где:
</span>{" "}
{where}
<br />
<span className="font-medium" style={{ color: "var(--bb-text)" }}>
Пример:
</span>{" "}
{example}
</div>
</div>
</div>
))}
</div>
</Section>
{/* 2. Размеры */}
<Section
id="sizes"
title="Размеры"
subtitle="Три размера для разных контекстов. lg соответствует форм-кнопке на реальном сайте (18px, bold)."
>
<div
className="rounded-xl border overflow-hidden"
style={{ borderColor: "var(--bb-border)" }}
>
{(
[
{
size: "sm" as const,
label: "Small",
hint: "4px 11px · 13px",
use: "Компактные контексты, таблицы",
},
{
size: "md" as const,
label: "Medium",
hint: "8px 16px · 14px",
use: "Appointment, Teal, Pill (соответствует сайту)",
},
{
size: "lg" as const,
label: "Large",
hint: "10px 24px · 18px bold",
use: "Primary форм-кнопка (соответствует сайту)",
},
] as const
).map(({ size, label, hint, use }, i) => (
<div
key={size}
className="flex items-center gap-6 px-5 py-4"
style={{ borderTop: i > 0 ? "1px solid var(--bb-border)" : undefined }}
>
<div className="w-40 shrink-0">
<Button variant="primary" size={size}>
Записаться
</Button>
</div>
<div className="flex-1 min-w-0">
<p className="text-sm font-medium mb-0.5" style={{ color: "var(--bb-text)" }}>
{label}
</p>
<p className="text-xs font-mono" style={{ color: "var(--bb-text-muted)" }}>
padding: {hint.split("·")[0].trim()} · font-size: {hint.split("·")[1].trim()}
</p>
</div>
<p
className="text-xs hidden lg:block"
style={{ color: "var(--bb-text-muted)", maxWidth: 220 }}
>
{use}
</p>
</div>
))}
</div>
</Section>
{/* 3. Состояния */}
<Section
id="states"
title="Состояния"
subtitle="Базовые состояния кнопки. На реальном сайте hover/transition не определены в CSS."
>
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-4">
{(
[
{
label: "Default",
node: <Button variant="primary" size="lg">Записаться</Button>,
hint: "Стандартное состояние",
},
{
label: "Hover",
node: (
<Button
variant="primary"
size="lg"
style={{ filter: "brightness(0.93)" }}
>
Записаться
</Button>
),
hint: "filter: brightness(0.93)",
},
{
label: "Loading",
node: <Button variant="primary" size="lg" loading>Отправка...</Button>,
hint: "Спиннер + blocked",
},
{
label: "Disabled",
node: <Button variant="primary" size="lg" disabled>Записаться</Button>,
hint: "opacity: 0.5",
},
] as const
).map(({ label, node, hint }) => (
<div
key={label}
className="rounded-xl border p-4"
style={{ borderColor: "var(--bb-border)" }}
>
<div
className="flex items-center justify-center py-4 mb-3 rounded-lg"
style={{ background: "var(--bb-sidebar-bg)" }}
>
{node}
</div>
<p className="text-sm font-medium mb-0.5" style={{ color: "var(--bb-text)" }}>
{label}
</p>
<p className="text-xs" style={{ color: "var(--bb-text-muted)" }}>
{hint}
</p>
</div>
))}
</div>
</Section>
{/* 4. Контекст применения */}
<Section
id="context"
title="Где применяется"
subtitle="Таблица: тип кнопки → реальное использование на сайте."
>
<div
className="rounded-xl border overflow-hidden"
style={{ borderColor: "var(--bb-border)" }}
>
<table className="w-full text-sm">
<thead>
<tr style={{ background: "var(--bb-sidebar-bg)" }}>
{["Вариант", "Цвет фона", "Реальный класс/контекст", "Текст кнопки на сайте"].map((h) => (
<th
key={h}
className="text-left px-4 py-3 font-medium text-xs"
style={{ color: "var(--bb-text-muted)" }}
>
{h}
</th>
))}
</tr>
</thead>
<tbody>
{[
{
v: <Button variant="primary" size="sm">Primary</Button>,
bg: "#FFA39C",
ctx: "button в entityform-блоках форм записи",
text: "«Запишите меня!»",
},
{
v: <Button variant="outline" size="sm">Outline</Button>,
bg: "#fff / рамка #BF9975",
ctx: ".appointment в хедере (block-block-15, 30, 32, 24)",
text: "«Записаться на прием»",
},
{
v: <Button variant="teal" size="sm">Teal</Button>,
bg: "#60959c",
ctx: ".show-phone (block-block-4, 15)",
text: "«Позвонить»",
},
{
v: <Button variant="pill" size="sm">Pill</Button>,
bg: "#e9e4d4",
ctx: "a.callback_url (modal trigger)",
text: "«Заказать звонок»",
},
].map(({ v, bg, ctx, text }, i) => (
<tr
key={i}
style={{ borderTop: "1px solid var(--bb-border)" }}
>
<td className="px-4 py-3">{v}</td>
<td
className="px-4 py-3 font-mono text-xs"
style={{ color: "var(--bb-text-muted)" }}
>
{bg}
</td>
<td
className="px-4 py-3 font-mono text-xs"
style={{ color: "var(--bb-text-muted)" }}
>
{ctx}
</td>
<td className="px-4 py-3 text-xs" style={{ color: "var(--bb-text)" }}>
{text}
</td>
</tr>
))}
</tbody>
</table>
</div>
</Section>
{/* 5. Примеры кода */}
<Section
id="code"
title="Примеры кода"
subtitle="HTML-классы из globals.css, JSX-компонент, и точный CSS с сайта."
>
<div className="space-y-4">
<CodeCopy lang="HTML (CSS-классы brandbook)" code={codeHtml} />
<CodeCopy lang="JSX (React / Next.js)" code={codeReact} />
<CodeCopy lang="CSS — точно с сайта oclinica.ru" code={codeSiteExact} />
</div>
</Section>
{/* LLM-блок */}
<LlmBlock path="/components/buttons" version="v2.0" specText={LLM_BUTTONS_TEXT}>
<LlmSection title="Варианты (реальный сайт oclinica.ru)" />
<LlmTable
headers={["Вариант", "CSS класс", "Фон", "Текст", "Border", "Radius", "Применение"]}
rows={VARIANTS.map((v) => [
v.variant,
v.cssClass,
v.bg,
v.textColor,
v.border,
v.radius,
v.where,
])}
/>
<LlmSection title="Размеры (брендбук-компонент)" />
<LlmTable
headers={["Размер", "padding", "font-size", "Применение"]}
rows={[
["sm", "4px 11px", "13px", "Компактные контексты"],
["md", "8px 16px", "14px", "Стандарт (outline, teal, pill с сайта)"],
["lg", "10px 24px", "18px bold", "Primary форм-кнопка (соответствует сайту)"],
]}
/>
<LlmSection title="Правила применения" />
<LlmRules
rules={[
{ ok: true, text: "primary (коралловый) — только для submit в формах записи" },
{ ok: true, text: "outline (бежевый) — хедер, навигация, второстепенные ссылки" },
{ ok: true, text: "teal (бирюзовый) — телефонные и контактные действия" },
{ ok: true, text: "pill (кремовый) — открытие модальных окон / callback" },
{ ok: true, text: "Не более одного primary на форму" },
{ ok: false, text: "Не менять цвета вне указанной палитры сайта" },
{ ok: false, text: "Primary — не для навигационных ссылок" },
{ ok: false, text: "Не накладывать тень на outline, teal, pill" },
]}
/>
</LlmBlock>
</div>
);
}
+629
View File
@@ -0,0 +1,629 @@
import type { Metadata } from "next";
import { CodeCopy } from "@/components/ui/CodeCopy";
import { LlmBlock, LlmSection, LlmTable, LlmRules } from "@/components/llm/LlmBlock";
export const metadata: Metadata = {
title: "Карточки. Цифровой брендбук Клиники ухо, горло, нос им. проф. Е.Н.Оленевой",
};
function Section({
id, title, subtitle, children,
}: {
id?: string; title: string; subtitle?: string; children: React.ReactNode;
}) {
return (
<section id={id} className="mb-14">
<div className="mb-6">
<h2 className="text-xl font-semibold" style={{ color: "var(--bb-text)" }}>{title}</h2>
{subtitle && <p className="mt-1 text-sm" style={{ color: "var(--bb-text-muted)" }}>{subtitle}</p>}
</div>
{children}
</section>
);
}
/* ─── Карточка врача ────────────────────────────────────────────────── */
function DoctorCard({
name, specialty, experience, photo,
}: {
name: string; specialty: string; experience: string; photo?: string;
}) {
return (
<div
className="flex gap-4 p-4 rounded-xl border bg-white transition-shadow"
style={{ borderColor: "var(--bb-border)" }}
>
{/* Фото */}
<div
className="shrink-0 rounded-lg overflow-hidden"
style={{ width: 110, height: 160, background: "#dff0fa" }}
>
{photo ? (
<img src={photo} alt={name} style={{ width: "100%", height: "100%", objectFit: "cover" }} />
) : (
<div
className="w-full h-full flex flex-col items-center justify-center gap-1"
style={{ color: "var(--brand-053m)", opacity: 0.5 }}
>
<svg width="40" height="40" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="1.5">
<path d="M20 21v-2a4 4 0 0 0-4-4H8a4 4 0 0 0-4 4v2" />
<circle cx="12" cy="7" r="4" />
</svg>
<span className="text-xs">фото</span>
</div>
)}
</div>
{/* Информация */}
<div className="flex flex-col justify-between min-w-0">
<div>
<h3 className="font-semibold text-sm leading-tight mb-1" style={{ color: "var(--bb-text)" }}>
{name}
</h3>
<p className="text-xs mb-2" style={{ color: "var(--bb-text-muted)" }}>{specialty}</p>
<p className="text-xs" style={{ color: "var(--brand-053m)" }}>{experience}</p>
</div>
<button className="bb-btn bb-btn-sm bb-btn-outline mt-3 self-start">
Записаться
</button>
</div>
</div>
);
}
/* ─── Карточка новости ──────────────────────────────────────────────── */
function NewsCard({
date, title, snippet, category,
}: {
date: string; title: string; snippet: string; category?: string;
}) {
return (
<div
className="bb-news-card rounded-xl border overflow-hidden cursor-pointer transition-all"
style={{ borderColor: "var(--bb-border)", background: "#fff" }}
>
{/* Превью */}
<div
className="h-36 flex items-center justify-center"
style={{ background: "#f0f9ff" }}
>
{category && (
<span
className="px-3 py-1 rounded-full text-xs font-semibold"
style={{ background: "var(--brand-053m)", color: "#fff" }}
>
{category}
</span>
)}
</div>
{/* Контент */}
<div className="p-4">
<p className="text-xs mb-2" style={{ color: "var(--bb-text-muted)" }}>{date}</p>
<h3 className="font-semibold text-sm leading-tight mb-2" style={{ color: "var(--bb-text)" }}>
{title}
</h3>
<p className="text-xs leading-relaxed mb-3" style={{ color: "var(--bb-text-muted)" }}>
{snippet}
</p>
<span className="text-xs font-medium" style={{ color: "var(--brand-053m)" }}>
Читать далее
</span>
</div>
</div>
);
}
/* ─── Карточка отзыва ───────────────────────────────────────────────── */
function ReviewCard({
author, date, text, rating, doctor,
}: {
author: string; date: string; text: string; rating: number; doctor?: string;
}) {
return (
<div
className="rounded-xl border p-5"
style={{ borderColor: "var(--bb-border)", background: "#eef4d1" }}
>
{/* Звёзды */}
<div className="flex gap-0.5 mb-3">
{Array.from({ length: 5 }, (_, i) => (
<svg key={i} width="16" height="16" viewBox="0 0 24 24" fill={i < rating ? "#f59e0b" : "none"} stroke="#f59e0b" strokeWidth="1.5">
<polygon points="12 2 15.09 8.26 22 9.27 17 14.14 18.18 21.02 12 17.77 5.82 21.02 7 14.14 2 9.27 8.91 8.26 12 2" />
</svg>
))}
<span className="text-xs ml-1 font-medium" style={{ color: "#92400e" }}>{rating}/5</span>
</div>
{/* Текст */}
<p
className="text-sm leading-relaxed mb-4"
style={{
color: "var(--bb-text)",
display: "-webkit-box",
WebkitLineClamp: 4,
WebkitBoxOrient: "vertical",
overflow: "hidden",
}}
>
{text}
</p>
{/* Автор */}
<div className="flex items-end justify-between">
<div>
<p className="text-sm font-semibold" style={{ color: "var(--bb-text)" }}>{author}</p>
{doctor && <p className="text-xs mt-0.5" style={{ color: "var(--bb-text-muted)" }}>Врач: {doctor}</p>}
</div>
<p className="text-xs" style={{ color: "var(--bb-text-muted)" }}>{date}</p>
</div>
</div>
);
}
/* ─── Карточка цены ─────────────────────────────────────────────────── */
function PriceCard({
service, price, description, highlighted,
}: {
service: string; price: string; description?: string; highlighted?: boolean;
}) {
return (
<div
className="rounded-xl border p-5 flex flex-col gap-3"
style={{
borderColor: highlighted ? "var(--brand-053m)" : "var(--bb-border)",
background: highlighted ? "#f0f9ff" : "#fff",
}}
>
<div className="flex items-start justify-between gap-3">
<p className="text-sm font-medium flex-1" style={{ color: "var(--bb-text)" }}>{service}</p>
<p className="text-lg font-bold shrink-0" style={{ color: highlighted ? "var(--brand-053m)" : "var(--bb-text)" }}>
{price}
</p>
</div>
{description && (
<p className="text-xs" style={{ color: "var(--bb-text-muted)" }}>{description}</p>
)}
<button className="bb-btn bb-btn-sm bb-btn-outline self-start">
Записаться
</button>
</div>
);
}
/* ─── Карточка услуги ───────────────────────────────────────────────── */
function ServiceCard({
title, description, icon,
}: {
title: string; description: string; icon: string;
}) {
return (
<div
className="bb-service-card rounded-xl border p-5 flex flex-col gap-3 cursor-pointer transition-shadow"
style={{ borderColor: "var(--bb-border)", background: "#fff" }}
>
<div
className="w-12 h-12 rounded-xl flex items-center justify-center text-2xl"
style={{ background: "#dff0fa" }}
>
{icon}
</div>
<div>
<h3 className="font-semibold text-sm mb-1" style={{ color: "var(--bb-text)" }}>{title}</h3>
<p className="text-xs leading-relaxed" style={{ color: "var(--bb-text-muted)" }}>{description}</p>
</div>
<span className="text-xs font-medium" style={{ color: "var(--brand-053m)" }}>
Подробнее
</span>
</div>
);
}
/* ─── LLM текст ─────────────────────────────────────────────────────── */
const LLM_CARDS_TEXT = `КАРТОЧКИ — LLM-спецификация
Версия: v1.0 · /components/cards
КАРТОЧКА ВРАЧА (DoctorCard)
Структура: фото (110×160px) + имя + специализация + опыт + кнопка «Записаться»
Фото: 110px × 160px, object-fit: cover, border-radius: 8px, фон-placeholder: #dff0fa
Кнопка: .bb-btn-outline .bb-btn-sm
Источник: .doctor на perm.oclinica.ru
КАРТОЧКА НОВОСТИ (NewsCard)
Структура: превью (h=144px) + дата + заголовок + анонс (4 строки) + «Читать далее →»
Hover: background #eef4d1 + box-shadow 0 0 16px #9e9e9a
Источник: #block-views-last-news-block-1 .views-column на сайте (200×200px)
Без hover: background #fff, border 1px
КАРТОЧКА ОТЗЫВА (ReviewCard)
Структура: рейтинг (звёзды 1–5) + текст (4 строки, overflow hidden) + автор + дата + врач
Фон: #eef4d1 (светло-жёлтый) — из реального CSS сайта
Звёзды: SVG polygon, filled #f59e0b
КАРТОЧКА ЦЕНЫ (PriceCard)
Структура: услуга + цена (bold) + описание + кнопка
highlighted-вариант: border var(--brand-053m), bg #f0f9ff, цена тоже --brand-053m
КАРТОЧКА УСЛУГИ (ServiceCard)
Структура: иконка (emoji, 48×48px, bg #dff0fa) + заголовок + описание + «Подробнее →»
Hover: box-shadow (0.5rem smth)
БЕЙДЖИ (Badge)
Варианты: primary (#0089c3), success (#059669), warning (#d97706), danger (#dc2626), neutral (#6b7280)
Размер: text-xs, px-2.5 py-0.5, border-radius: full
CSS: inline-flex, font-weight: 600
ТЕГИ (Tag)
Варианты: default (border, text), filled (bg), removable (с кнопкой ×)
Цвет: --brand-053m или нейтральный (#e5e7eb bg)
Размер: text-xs, px-2 py-1, border-radius: 4px
АЛЕРТЫ (Alert)
Варианты: info (#dff0fa фон, #075985 текст), success (#d1fae5, #065f46), warning (#fef3c7, #92400e), error (#fee2e2, #991b1b)
Структура: иконка (16×16px) + заголовок + описание
Без кнопки закрыть в базовом варианте
ПРАВИЛА
✓ DoctorCard: всегда показывать фото-placeholder если нет фото (не ломать layout)
✓ ReviewCard: обрезать текст после 4 строк (WebkitLineClamp)
✓ PriceCard: highlighted = акционная или рекомендуемая позиция
✓ NewsCard: hover-эффект обязателен (#eef4d1 + box-shadow)
✓ Badge: не более 2–3 бейджей рядом
✓ Alert: одновременно не более 1 alert одного типа на экране
✕ Не смешивать типы карточек в одной сетке без заголовка секции
✕ Не использовать ReviewCard без рейтинга`.trim();
/* ─── Данные примеров ───────────────────────────────────────────────── */
const DOCTORS = [
{
name: "Иванова Анна Сергеевна",
specialty: "Оториноларинголог (ЛОР), высшая категория",
experience: "Стаж 18 лет",
},
{
name: "Петров Дмитрий Александрович",
specialty: "Сурдолог, аудиолог",
experience: "Стаж 12 лет",
},
{
name: "Соколова Мария Ивановна",
specialty: "Детский ЛОР, ринолог",
experience: "Стаж 9 лет",
},
];
const REVIEWS = [
{
author: "Елена К.",
date: "15 марта 2026",
rating: 5,
doctor: "Иванова А.С.",
text: "Очень довольна визитом. Доктор внимательно выслушала все жалобы, провела осмотр и объяснила причину заболевания. Назначила лечение, которое помогло уже через 3 дня. Рекомендую!",
},
{
author: "Михаил Р.",
date: "10 марта 2026",
rating: 4,
doctor: "Петров Д.А.",
text: "Хороший специалист, всё объяснил понятно. Подождал немного дольше, чем ожидал, но качество приёма на высоте. Буду обращаться снова.",
},
];
const PRICES = [
{ service: "Первичный приём ЛОР-врача", price: "1 500 ₽", description: "Включает осмотр и консультацию" },
{ service: "Повторный приём", price: "1 000 ₽", description: "До 14 дней после первичного" },
{ service: "Промывание миндалин", price: "800 ₽", highlighted: true, description: "Аппаратное — аккустический вакуум" },
{ service: "Аудиометрия", price: "1 200 ₽", description: "Исследование слуха" },
];
const SERVICES = [
{ title: "Лечение ангины и тонзиллита", description: "Консервативное и хирургическое лечение заболеваний миндалин", icon: "🦷" },
{ title: "Аудиология и сурдология", description: "Диагностика нарушений слуха, подбор слуховых аппаратов", icon: "👂" },
{ title: "Детский ЛОР", description: "Специализация на лечении ЛОР-заболеваний у детей от 0 лет", icon: "👶" },
{ title: "Ринология", description: "Лечение заболеваний носа и придаточных пазух", icon: "👃" },
];
/* ─── Коды примеров ─────────────────────────────────────────────────── */
const codeDoctorCard = `<!-- Карточка врача -->
<div class="doctor-card">
<img src="doctor.jpg" width="110" height="160" alt="ФИО" />
<div class="doctor-info">
<h3>Иванова Анна Сергеевна</h3>
<p class="specialty">Оториноларинголог, высшая категория</p>
<p class="experience">Стаж 18 лет</p>
<button class="bb-btn bb-btn-sm bb-btn-outline">Записаться</button>
</div>
</div>
/* CSS с сайта oclinica.ru */
.doctor .image { float:left; margin-right:20px; width:110px; height:160px; }
.doctor .item { float:left; width:170px; }
.doctor h3 { margin-top:0; height:32px; }`;
const codeNewsCard = `<!-- Карточка новости (сайт: 200×200px) -->
<div class="news-card">
<div class="news-preview">...</div>
<div class="news-body">
<time>15 марта 2026</time>
<h3>Заголовок новости</h3>
<p>Краткий анонс...</p>
<a href="#">Читать далее →</a>
</div>
</div>
/* CSS с сайта */
#block-views-last-news-block-1 .views-column {
background: #fff; width: 200px; height: 200px;
margin: 15px 8px; padding: 15px;
}
#block-views-last-news-block-1 .views-column:hover {
background: #eef4d1;
box-shadow: 0px 0px 16px 0px #9e9e9a;
}`;
const codeReviewCard = `<!-- Карточка отзыва -->
<div class="review-card">
<div class="stars">★★★★★</div>
<p class="text">Текст отзыва (4 строки, overflow: hidden)...</p>
<div class="author">
<span>Елена К.</span>
<time>15 марта 2026</time>
</div>
</div>
/* Стиль брендбука (фон из CSS сайта) */
.review-card { background: #eef4d1; border-radius: 12px; padding: 20px; }`;
const codePriceCard = `<!-- Карточка цены -->
<div class="price-card">
<div class="price-row">
<span class="service">Первичный приём ЛОР-врача</span>
<strong class="price">1 500 ₽</strong>
</div>
<p class="description">Включает осмотр и консультацию</p>
<button class="bb-btn bb-btn-sm bb-btn-outline">Записаться</button>
</div>`;
const codeBadges = `<!-- Бейджи -->
<span class="bb-badge bb-badge-primary">ЛОР</span>
<span class="bb-badge bb-badge-success">Принимает</span>
<span class="bb-badge bb-badge-warning">Ожидает</span>
<span class="bb-badge bb-badge-danger">Не принимает</span>
<span class="bb-badge bb-badge-neutral">Высшая категория</span>
<!-- Теги -->
<button class="bb-tag">Ухо</button>
<button class="bb-tag">Горло</button>
<button class="bb-tag bb-tag-active">Нос</button>
<!-- Алерт -->
<div class="bb-alert bb-alert-info">
<span class="bb-alert-icon"></span>
<div>
<strong>Запись открыта</strong>
<p>Вы можете записаться онлайн или по телефону.</p>
</div>
</div>`;
export default function CardsPage() {
return (
<div className="max-w-4xl mx-auto px-8 py-10">
{/* Заголовок */}
<div className="mb-10 pb-6 border-b" style={{ borderColor: "var(--bb-border)" }}>
<p className="text-xs font-semibold uppercase tracking-widest mb-2" style={{ color: "var(--brand-053m)" }}>
Компоненты 3.3
</p>
<h1 className="text-3xl font-semibold mb-3" style={{ color: "var(--bb-text)" }}>
Карточки
</h1>
<p className="text-base max-w-2xl" style={{ color: "var(--bb-text-muted)" }}>
Карточки врача, новости, отзыва, цены и услуги основные блоки контента сайта.
Бейджи, теги и алерты вспомогательные элементы.
</p>
</div>
{/* 1. Карточки врачей */}
<Section
id="doctor"
title="Карточка врача"
subtitle="Фото 110×160px, имя, специализация, стаж, кнопка записи. Соответствует .doctor на сайте."
>
<div className="grid grid-cols-1 sm:grid-cols-3 gap-4">
{DOCTORS.map(d => <DoctorCard key={d.name} {...d} />)}
</div>
</Section>
{/* 2. Карточки новостей */}
<Section
id="news"
title="Карточка новости"
subtitle="Hover: bg #eef4d1 + box-shadow. Источник: #block-views-last-news-block-1 на сайте."
>
<div className="grid grid-cols-1 sm:grid-cols-3 gap-4">
<NewsCard
date="15 марта 2026"
category="ЛОР"
title="Как правильно промывать нос при насморке"
snippet="Промывание носа — эффективный метод лечения и профилактики острого ринита. Рассказываем о правильной технике."
/>
<NewsCard
date="10 марта 2026"
category="Аудиология"
title="Новый слуховой аппарат в нашей клинике"
snippet="Мы начали подбор и настройку слуховых аппаратов нового поколения — незаметных и точных."
/>
<NewsCard
date="5 марта 2026"
title="Весенняя профилактика ЛОР-заболеваний"
snippet="Апрель — период обострений. Рекомендации нашего специалиста по укреплению иммунитета и защите."
/>
</div>
<p className="mt-3 text-xs" style={{ color: "var(--bb-text-muted)" }}>
* Наведите на карточку чтобы увидеть hover-эффект
</p>
</Section>
{/* 3. Карточки отзывов */}
<Section
id="review"
title="Карточка отзыва"
subtitle="Рейтинг (1–5 звёзд), текст 4 строки, автор, дата, врач. Фон #eef4d1 — с реального сайта."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
{REVIEWS.map(r => <ReviewCard key={r.author} {...r} />)}
</div>
</Section>
{/* 4. Карточки цен */}
<Section
id="price"
title="Карточка цены"
subtitle="Услуга + стоимость + описание + кнопка. Highlighted-вариант для акционных позиций."
>
<div className="flex flex-col gap-3">
{PRICES.map(p => <PriceCard key={p.service} {...p} />)}
</div>
</Section>
{/* 5. Карточки услуг */}
<Section
id="service"
title="Карточка услуги"
subtitle="Иконка + заголовок + описание + ссылка. Применяется в блоке «Наши услуги»."
>
<div className="grid grid-cols-2 sm:grid-cols-4 gap-4">
{SERVICES.map(s => <ServiceCard key={s.title} {...s} />)}
</div>
</Section>
{/* 6. Бейджи и теги */}
<Section
id="badges"
title="Бейджи и теги"
subtitle="Статусные бейджи, теги-категории, алерты."
>
{/* Бейджи */}
<div className="mb-8">
<p className="text-sm font-medium mb-3" style={{ color: "var(--bb-text)" }}>Бейджи (статус)</p>
<div className="flex flex-wrap gap-2">
{[
{ label: "Основной", bg: "var(--brand-053m)", color: "#fff" },
{ label: "Принимает", bg: "#059669", color: "#fff" },
{ label: "Высшая категория", bg: "#d97706", color: "#fff" },
{ label: "Не принимает", bg: "#dc2626", color: "#fff" },
{ label: "Нейтральный", bg: "#6b7280", color: "#fff" },
{ label: "Новинка", bg: "#dff0fa", color: "var(--brand-053m)" },
].map(b => (
<span
key={b.label}
className="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-semibold"
style={{ background: b.bg, color: b.color }}
>
{b.label}
</span>
))}
</div>
</div>
{/* Теги */}
<div className="mb-8">
<p className="text-sm font-medium mb-3" style={{ color: "var(--bb-text)" }}>Теги (категории)</p>
<div className="flex flex-wrap gap-2">
{["Ухо", "Горло", "Нос", "Аудиология", "Детский ЛОР", "Хирургия"].map((tag, i) => (
<button
key={tag}
className="inline-flex items-center px-3 py-1 rounded text-xs font-medium border transition-colors"
style={
i === 0
? { background: "var(--brand-053m)", color: "#fff", borderColor: "var(--brand-053m)" }
: { background: "#fff", color: "var(--bb-text)", borderColor: "var(--bb-border)" }
}
>
{tag}
</button>
))}
</div>
</div>
{/* Алерты */}
<div>
<p className="text-sm font-medium mb-3" style={{ color: "var(--bb-text)" }}>Алерты</p>
<div className="flex flex-col gap-3">
{[
{ type: "info", bg: "#dff0fa", color: "#075985", icon: "", title: "Информация", text: "Запись открыта. Вы можете записаться онлайн или по телефону." },
{ type: "success", bg: "#d1fae5", color: "#065f46", icon: "✓", title: "Успешно", text: "Ваша запись подтверждена. Ждём вас 20 марта в 10:00." },
{ type: "warning", bg: "#fef3c7", color: "#92400e", icon: "⚠", title: "Внимание", text: "Не забудьте взять паспорт и полис ОМС на приём." },
{ type: "error", bg: "#fee2e2", color: "#991b1b", icon: "✕", title: "Ошибка", text: "К сожалению, это время уже занято. Выберите другое." },
].map(a => (
<div
key={a.type}
className="flex gap-3 p-4 rounded-xl border"
style={{ background: a.bg, borderColor: a.bg, color: a.color }}
>
<span className="text-base font-bold shrink-0 mt-0.5">{a.icon}</span>
<div>
<p className="text-sm font-semibold mb-0.5">{a.title}</p>
<p className="text-sm opacity-90">{a.text}</p>
</div>
</div>
))}
</div>
</div>
</Section>
{/* 7. Код */}
<Section
id="code"
title="Примеры кода"
subtitle="HTML-структура и CSS-справка."
>
<div className="space-y-4">
<CodeCopy lang="HTML — DoctorCard + CSS с сайта" code={codeDoctorCard} />
<CodeCopy lang="HTML — NewsCard + CSS с сайта" code={codeNewsCard} />
<CodeCopy lang="HTML — ReviewCard" code={codeReviewCard} />
<CodeCopy lang="HTML — PriceCard" code={codePriceCard} />
<CodeCopy lang="HTML — Badges, Tags, Alerts" code={codeBadges} />
</div>
</Section>
{/* LLM-блок */}
<LlmBlock path="/components/cards" version="v1.0" specText={LLM_CARDS_TEXT}>
<LlmSection title="Типы карточек" />
<LlmTable
headers={["Карточка", "Ключевые размеры", "Источник на сайте", "Фон / hover"]}
rows={[
["DoctorCard", "фото 110×160px, layout: flex", ".doctor .image + .doctor .item", "#fff / —"],
["NewsCard", "preview h=144px, grid 3 col", "#block-views-last-news-block-1 .views-column", "#fff / #eef4d1 + shadow"],
["ReviewCard", "4 строки текста, рейтинг", ".node-reviews", "#eef4d1 / —"],
["PriceCard", "flex row: name + price", ".field-name-field-price-priem", "#fff / highlighted: #f0f9ff"],
["ServiceCard", "иконка 48×48, grid 4 col", "—", "#fff / shadow"],
]}
/>
<LlmSection title="Бейджи, теги, алерты" />
<LlmTable
headers={["Элемент", "Варианты", "Размер", "Применение"]}
rows={[
["Badge", "primary / success / warning / danger / neutral", "text-xs, px-2.5, rounded-full", "Статус врача, категория, акция"],
["Tag", "default / active", "text-xs, px-3, rounded-4px", "Фильтры, категории услуг"],
["Alert", "info / success / warning / error", "p-4, border-radius 12px", "Системные сообщения пользователю"],
]}
/>
<LlmSection title="Правила применения" />
<LlmRules
rules={[
{ ok: true, text: "DoctorCard: всегда фото-placeholder если нет фото" },
{ ok: true, text: "NewsCard: hover #eef4d1 + box-shadow (из реального CSS сайта)" },
{ ok: true, text: "ReviewCard: обрезать текст после 4 строк (WebkitLineClamp: 4)" },
{ ok: true, text: "PriceCard highlighted = акционная / рекомендуемая позиция" },
{ ok: true, text: "Alert: один тип одновременно на экране" },
{ ok: false, text: "Не смешивать типы карточек в одной сетке без заголовка" },
{ ok: false, text: "Не использовать ReviewCard без рейтинга" },
{ ok: false, text: "Не ставить более 3 бейджей рядом" },
]}
/>
</LlmBlock>
</div>
);
}
+723
View File
@@ -0,0 +1,723 @@
import type { Metadata } from "next";
import { Toggle } from "@/components/ui/Toggle";
import { CodeCopy } from "@/components/ui/CodeCopy";
import { LlmBlock, LlmSection, LlmTable, LlmRules } from "@/components/llm/LlmBlock";
export const metadata: Metadata = {
title: "Форм-контролы. Цифровой брендбук Клиники ухо, горло, нос им. проф. Е.Н.Оленевой",
};
function Section({
id,
title,
subtitle,
children,
}: {
id?: string;
title: string;
subtitle?: string;
children: React.ReactNode;
}) {
return (
<section id={id} className="mb-12">
<div className="mb-6">
<h2 className="text-xl font-semibold" style={{ color: "var(--bb-text)" }}>
{title}
</h2>
{subtitle && (
<p className="mt-1 text-sm" style={{ color: "var(--bb-text-muted)" }}>
{subtitle}
</p>
)}
</div>
{children}
</section>
);
}
function FieldLabel({ text, required }: { text: string; required?: boolean }) {
return (
<label
className="block text-sm font-medium mb-1.5"
style={{ color: "var(--bb-text)" }}
>
{text}
{required && <span style={{ color: "#dc2626", marginLeft: 2 }}>*</span>}
</label>
);
}
function FieldHint({ text }: { text: string }) {
return (
<p className="mt-1.5 text-xs" style={{ color: "var(--bb-text-muted)" }}>
{text}
</p>
);
}
function FieldError({ text }: { text: string }) {
return (
<p className="mt-1.5 text-xs" style={{ color: "#dc2626" }}>
{text}
</p>
);
}
function StateCard({
label,
hint,
children,
}: {
label: string;
hint: string;
children: React.ReactNode;
}) {
return (
<div className="rounded-xl border p-4" style={{ borderColor: "var(--bb-border)" }}>
<div className="mb-3">{children}</div>
<p className="text-sm font-medium mb-0.5" style={{ color: "var(--bb-text)" }}>
{label}
</p>
<p className="text-xs" style={{ color: "var(--bb-text-muted)" }}>
{hint}
</p>
</div>
);
}
const LLM_FORMS_TEXT = `ФОРМ-КОНТРОЛЫ — LLM-спецификация
Версия: v2.0 · /components/forms
ТЕКСТОВОЕ ПОЛЕ (Input)
CSS класс: .bb-input
Высота: 50px · padding: 10px 12px
border: 1px solid #ccc · border-radius: 4px · font: Fira Sans 14px
Источник: entityform input[type=text] на perm.oclinica.ru
Состояния:
default: border #ccc
focus: border #7ecfca + box-shadow 0 0 0 3px rgba(126,207,202,0.2)
error: border #dc2626 + класс .bb-error
disabled: opacity 0.5 + cursor not-allowed + bg #f8f9fa
МНОГОСТРОЧНЫЙ ТЕКСТ (Textarea)
CSS класс: .bb-textarea
Те же состояния что у Input
min-height: 100px · resize: vertical · padding: 10px 12px
ВЫПАДАЮЩИЙ СПИСОК (Select)
CSS класс: .bb-select
Высота: 50px · padding: 10px 36px 10px 10px
Стрелка: SVG background-image (data URI)
Источник: .form-control.form-select entityform на сайте
Те же состояния что у Input
ФЛАЖОК (Checkbox)
CSS класс: .bb-checkbox
size: 16×16px · accent-color: #5b7b87
HTML: <input type="checkbox" class="bb-checkbox" />
Состояния: unchecked / checked / disabled / checked+disabled
ПЕРЕКЛЮЧАТЕЛЬ ВАРИАНТА (Radio)
CSS класс: .bb-radio
size: 16×16px · accent-color: #5b7b87
HTML: <input type="radio" class="bb-radio" name="group" />
Всегда в группе — один выбранный из нескольких
ТУМБЛЕР (Toggle/Switch)
Компонент: @/components/ui/Toggle (React, "use client")
Ширина трека: 44px · Высота: 24px · Бегунок: 20×20px
Выкл: track #d1d5db · Вкл: track #5b7b87
CSS: .bb-toggle-track / .bb-toggle-thumb
HTML-аналог: <input type="checkbox" role="switch" />
КОНТЕКСТ ПРИМЕНЕНИЯ НА САЙТЕ
Input/Select используются в entityform-блоках:
#block-entityform-block-lor-form — форма «Запишите меня!» (ЛОР)
#block-entityform-block-lor-form-2 — форма «Узнайте стоимость операции»
#block-entityform-block-surgery-form — форма хирургии
Фон формы: #b8e6ed (светло-бирюзовый)
Ширина полей: 302px (фиксированная), кнопка submit: 300px
ОБЩИЕ ПРАВИЛА
✓ Метка (label) всегда над полем, font-weight: 500
✓ Обязательные поля помечены * красным цветом (#dc2626)
✓ Подсказка (hint) серым текстом под полем — font-size: 12px
✓ Сообщение об ошибке красным (#dc2626) под полем вместо hint
✓ Focus outline — бирюзовый #7ecfca (--brand-053m)
✓ Группы checkbox/radio — вертикальный список с gap: 10px
✓ Toggle — для булевых настроек включить/выключить
✕ Не использовать placeholder вместо label
✕ Не скрывать обязательность поля
✕ Не делать поля шире контейнера`.trim();
export default function FormsPage() {
const codeInput = `<!-- HTML -->
<label class="bb-label">Имя пациента <span style="color:#dc2626">*</span></label>
<input class="bb-input" type="text" placeholder="Иван Иванов" />
<p class="bb-hint">Укажите имя как в паспорте</p>
<!-- Error-состояние -->
<input class="bb-input bb-error" type="text" value="ива" />
<p style="color:#dc2626;font-size:12px">Минимум 3 символа</p>
<!-- Password -->
<input class="bb-input" type="password" placeholder="Введите пароль" />`;
const codeTextarea = `<label class="bb-label">Комментарий к записи</label>
<textarea class="bb-textarea" rows="4" placeholder="Опишите симптомы..."></textarea>`;
const codeSelect = `<label class="bb-label">Специализация</label>
<select class="bb-select">
<option value="">Выберите специализацию</option>
<option value="lor">ЛОР — ухо, горло, нос</option>
<option value="aud">Аудиология</option>
</select>`;
const codeCheckbox = `<!-- Одиночный -->
<label style="display:flex;align-items:center;gap:8px">
<input class="bb-checkbox" type="checkbox" />
<span>Согласен с условиями</span>
</label>
<!-- Группа -->
<div style="display:flex;flex-direction:column;gap:10px">
<label style="display:flex;align-items:center;gap:8px">
<input class="bb-checkbox" type="checkbox" checked /> ЛОР
</label>
<label style="display:flex;align-items:center;gap:8px">
<input class="bb-checkbox" type="checkbox" /> Аудиология
</label>
</div>`;
const codeRadio = `<div style="display:flex;flex-direction:column;gap:10px" role="radiogroup">
<label style="display:flex;align-items:center;gap:8px">
<input class="bb-radio" type="radio" name="visit" value="first" checked />
Первичный приём
</label>
<label style="display:flex;align-items:center;gap:8px">
<input class="bb-radio" type="radio" name="visit" value="repeat" />
Повторный приём
</label>
<label style="display:flex;align-items:center;gap:8px">
<input class="bb-radio" type="radio" name="visit" value="online" />
Онлайн-консультация
</label>
</div>`;
const codeToggle = `import { Toggle } from "@/components/ui/Toggle";
// Базовый тумблер
<Toggle />
// С меткой
<Toggle label="Получать уведомления" />
// По умолчанию включён
<Toggle defaultChecked label="Email-рассылка" />
// Заблокирован
<Toggle disabled label="Настройка недоступна" />`;
const codeSiteCSS = `/* ── Реальный CSS с perm.oclinica.ru ─────────────────────────── */
/* Базовые стили (Bootstrap override) */
input[type=text],
input[type=email] {
padding: 0;
height: 30px;
border: 1px solid #ccc;
}
/* Entityform-блоки: форма записи на приём / узнать стоимость */
#block-entityform-block-lor-form input[type=text],
#block-entityform-block-lor-form-2 input[type=text],
#block-entityform-block-surgery-form input[type=text] {
height: 50px;
padding: 10px;
}
/* Select в entityform */
.field-name-field-lor-vrach .form-control.form-select {
height: 50px;
padding: 10px 16px;
font-size: .9em;
font-weight: bold;
font-family: 'Fira Sans';
color: #949290;
}
/* Webform (отдельный вид форм) — скруглений нет */
.webform-client-form input[type=text].form-text {
border-radius: 0;
}`;
return (
<div className="max-w-4xl mx-auto px-8 py-10">
{/* Заголовок */}
<div className="mb-10 pb-6 border-b" style={{ borderColor: "var(--bb-border)" }}>
<p
className="text-xs font-semibold uppercase tracking-widest mb-2"
style={{ color: "var(--brand-053m)" }}
>
Компоненты 3.2
</p>
<h1 className="text-3xl font-semibold mb-3" style={{ color: "var(--bb-text)" }}>
Форм-контролы
</h1>
<p className="text-base max-w-2xl" style={{ color: "var(--bb-text-muted)" }}>
Элементы ввода данных: текстовые поля, выпадающие списки, флажки, переключатели.
Применяются в формах записи, фильтрах и настройках.
</p>
</div>
{/* 1. Input */}
<Section
id="input"
title="Текстовое поле"
subtitle="Базовый элемент ввода текста. Класс .bb-input. Высота 50px — как на сайте oclinica.ru."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4 mb-4">
<StateCard label="Default" hint="border: 1px solid #ccc · border-radius: 4px · height: 50px">
<FieldLabel text="Имя пациента" required />
<input className="bb-input" type="text" placeholder="Иван Иванов" readOnly />
<FieldHint text="Укажите имя как в паспорте" />
</StateCard>
<StateCard label="Focus" hint="border #7ecfca + box-shadow rgba(126,207,202,0.2)">
<FieldLabel text="Имя пациента" required />
<input
className="bb-input"
type="text"
defaultValue="Иван"
readOnly
style={{
borderColor: "var(--brand-053m)",
boxShadow: "0 0 0 3px rgba(126,207,202,0.2)",
}}
/>
<FieldHint text="Укажите имя как в паспорте" />
</StateCard>
<StateCard label="Error" hint="border #dc2626 + .bb-error + сообщение об ошибке">
<FieldLabel text="Имя пациента" required />
<input
className="bb-input bb-error"
type="text"
defaultValue="ив"
readOnly
/>
<FieldError text="Минимум 3 символа" />
</StateCard>
<StateCard label="Disabled" hint="opacity: 0.5 + cursor: not-allowed">
<FieldLabel text="Email (только чтение)" />
<input
className="bb-input"
type="text"
defaultValue="ivan@example.com"
disabled
/>
<FieldHint text="Email нельзя изменить" />
</StateCard>
</div>
{/* Password */}
<div
className="rounded-xl border p-4"
style={{ borderColor: "var(--bb-border)", background: "var(--bb-sidebar-bg)" }}
>
<p className="text-xs font-semibold uppercase tracking-widest mb-3" style={{ color: "var(--bb-text-muted)" }}>
Тип password
</p>
<div className="max-w-sm">
<FieldLabel text="Пароль" required />
<input className="bb-input" type="password" placeholder="Введите пароль" readOnly />
</div>
</div>
</Section>
{/* 2. Textarea */}
<Section
id="textarea"
title="Многострочный текст"
subtitle="Поле для длинного ввода. Класс .bb-textarea. border: 1px solid #ccc · border-radius: 4px."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
<StateCard label="Default" hint="min-height: 100px · resize: vertical">
<FieldLabel text="Комментарий к записи" />
<textarea
className="bb-textarea"
rows={3}
placeholder="Опишите симптомы или пожелания..."
readOnly
/>
</StateCard>
<StateCard label="Focus" hint="border #7ecfca + box-shadow">
<FieldLabel text="Комментарий к записи" />
<textarea
className="bb-textarea"
rows={3}
defaultValue="Беспокоит боль в горле..."
readOnly
style={{
borderColor: "var(--brand-053m)",
boxShadow: "0 0 0 3px rgba(126,207,202,0.2)",
}}
/>
</StateCard>
</div>
</Section>
{/* 3. Select */}
<Section
id="select"
title="Выпадающий список"
subtitle="Выбор из предопределённых вариантов. Класс .bb-select. Высота 50px — как в entityform на сайте."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
<StateCard label="Default" hint="height: 50px · кастомная стрелка SVG">
<FieldLabel text="Специализация" />
<select className="bb-select">
<option value="">Выберите специализацию</option>
<option value="lor">ЛОР ухо, горло, нос</option>
<option value="aud">Аудиология</option>
<option value="ped">ЛОР (детская)</option>
</select>
</StateCard>
<StateCard label="Выбрано значение" hint="Нативное поведение браузера">
<FieldLabel text="Специализация" />
<select className="bb-select" defaultValue="lor">
<option value="">Выберите специализацию</option>
<option value="lor">ЛОР ухо, горло, нос</option>
<option value="aud">Аудиология</option>
<option value="ped">ЛОР (детская)</option>
</select>
</StateCard>
</div>
</Section>
{/* 4. Checkbox */}
<Section
id="checkbox"
title="Флажок"
subtitle="Независимый выбор одного или нескольких вариантов. Класс .bb-checkbox."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
<StateCard label="Одиночный флажок" hint="Согласие с условиями, подтверждение">
<div className="flex flex-col gap-3">
<label className="flex items-center gap-2 cursor-pointer">
<input type="checkbox" className="bb-checkbox" />
<span className="text-sm" style={{ color: "var(--bb-text)" }}>
Согласен с условиями
</span>
</label>
<label className="flex items-center gap-2 cursor-pointer">
<input type="checkbox" className="bb-checkbox" defaultChecked />
<span className="text-sm" style={{ color: "var(--bb-text)" }}>
Получать уведомления
</span>
</label>
<label
className="flex items-center gap-2"
style={{ opacity: 0.5, cursor: "not-allowed" }}
>
<input type="checkbox" className="bb-checkbox" disabled />
<span className="text-sm" style={{ color: "var(--bb-text)" }}>
Недоступная опция
</span>
</label>
</div>
</StateCard>
<StateCard label="Группа флажков" hint="Выбор нескольких специализаций">
<FieldLabel text="Интересующие направления" />
<div className="flex flex-col gap-2.5 mt-1">
{["ЛОР — ухо, горло, нос", "Аудиология", "ЛОР (детская)", "Слухопротезирование"].map(
(item, i) => (
<label key={item} className="flex items-center gap-2 cursor-pointer">
<input
type="checkbox"
className="bb-checkbox"
defaultChecked={i === 0}
/>
<span className="text-sm" style={{ color: "var(--bb-text)" }}>
{item}
</span>
</label>
)
)}
</div>
</StateCard>
</div>
</Section>
{/* 5. Radio */}
<Section
id="radio"
title="Переключатель варианта"
subtitle="Выбор одного из взаимоисключающих вариантов. Класс .bb-radio."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
<StateCard label="Тип приёма" hint="Один выбранный из нескольких">
<FieldLabel text="Тип приёма" />
<div className="flex flex-col gap-2.5 mt-1" role="radiogroup">
{[
{ value: "first", label: "Первичный приём", checked: true },
{ value: "repeat", label: "Повторный приём", checked: false },
{ value: "online", label: "Онлайн-консультация", checked: false },
].map(({ value, label, checked }) => (
<label key={value} className="flex items-center gap-2 cursor-pointer">
<input
type="radio"
className="bb-radio"
name="visit-type-demo"
defaultChecked={checked}
/>
<span className="text-sm" style={{ color: "var(--bb-text)" }}>
{label}
</span>
</label>
))}
</div>
</StateCard>
<StateCard label="С отключённым вариантом" hint="disabled скрывает недоступный выбор">
<FieldLabel text="Способ контакта" />
<div className="flex flex-col gap-2.5 mt-1" role="radiogroup">
{[
{ value: "phone", label: "Телефон", disabled: false, checked: true },
{ value: "sms", label: "SMS", disabled: false, checked: false },
{ value: "whatsapp", label: "WhatsApp (скоро)", disabled: true, checked: false },
].map(({ value, label, disabled, checked }) => (
<label
key={value}
className="flex items-center gap-2"
style={{ cursor: disabled ? "not-allowed" : "pointer", opacity: disabled ? 0.5 : 1 }}
>
<input
type="radio"
className="bb-radio"
name="contact-demo"
disabled={disabled}
defaultChecked={checked}
/>
<span className="text-sm" style={{ color: "var(--bb-text)" }}>
{label}
</span>
</label>
))}
</div>
</StateCard>
</div>
</Section>
{/* 6. Toggle */}
<Section
id="toggle"
title="Тумблер"
subtitle="Булевый переключатель «включено / выключено». Компонент Toggle."
>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
<div
className="rounded-xl border p-5"
style={{ borderColor: "var(--bb-border)" }}
>
<p className="text-sm font-medium mb-4" style={{ color: "var(--bb-text)" }}>
Интерактивные примеры
</p>
<div className="flex flex-col gap-4">
<Toggle label="Получать уведомления" />
<Toggle defaultChecked label="Email-рассылка" />
<Toggle defaultChecked label="Push-уведомления" />
<Toggle disabled label="Недоступная настройка" />
</div>
</div>
<div
className="rounded-xl border p-5"
style={{ borderColor: "var(--bb-border)" }}
>
<p className="text-sm font-medium mb-4" style={{ color: "var(--bb-text)" }}>
Состояния
</p>
<div className="flex flex-col gap-4">
<div className="flex items-center justify-between">
<span className="text-sm" style={{ color: "var(--bb-text-muted)" }}>Выкл (default)</span>
<Toggle />
</div>
<div className="flex items-center justify-between">
<span className="text-sm" style={{ color: "var(--bb-text-muted)" }}>Вкл (defaultChecked)</span>
<Toggle defaultChecked />
</div>
<div className="flex items-center justify-between">
<span className="text-sm" style={{ color: "var(--bb-text-muted)" }}>Disabled</span>
<Toggle disabled />
</div>
<div className="flex items-center justify-between">
<span className="text-sm" style={{ color: "var(--bb-text-muted)" }}>Disabled + вкл</span>
<Toggle disabled defaultChecked />
</div>
</div>
</div>
</div>
</Section>
{/* 7. Контекст на сайте */}
<Section
id="context"
title="Контекст применения"
subtitle="Как форм-контролы выглядят на сайте oclinica.ru — в реальных entityform-блоках."
>
<div className="overflow-x-auto mb-6">
<table className="w-full text-sm border-collapse">
<thead>
<tr style={{ background: "var(--bb-sidebar-bg)" }}>
{["Контрол", "CSS класс", "Где на сайте", "CSS-блок на сайте"].map((h) => (
<th
key={h}
className="text-left px-3 py-2 font-semibold text-xs uppercase tracking-wide"
style={{ color: "var(--bb-text-muted)", borderBottom: "1px solid var(--bb-border)" }}
>
{h}
</th>
))}
</tr>
</thead>
<tbody>
{[
["Input (text)", ".bb-input", "Форма записи ЛОР, форма хирургии, «Узнайте стоимость»", "#block-entityform-block-lor-form input[type=text]"],
["Select", ".bb-select", "Выбор врача в форме записи", ".field-name-field-lor-vrach .form-control.form-select"],
["Checkbox", ".bb-checkbox", "Согласие на обработку данных в entityform", ".form-type-checkbox.checkbox label"],
["Textarea", ".bb-textarea", "Комментарии (в ряде форм)", "Без специального CSS на сайте (Bootstrap)"],
["Toggle", ".bb-toggle-track", "Не используется на сайте (UI-компонент брендбука)", "—"],
].map(([ctrl, cls, where, block]) => (
<tr key={ctrl} style={{ borderBottom: "1px solid var(--bb-border)" }}>
<td className="px-3 py-2.5 font-medium" style={{ color: "var(--bb-text)" }}>{ctrl}</td>
<td className="px-3 py-2.5">
<code className="text-xs px-1.5 py-0.5 rounded" style={{ background: "var(--bb-sidebar-bg)", color: "var(--brand-073m)" }}>{cls}</code>
</td>
<td className="px-3 py-2.5 text-xs" style={{ color: "var(--bb-text-muted)" }}>{where}</td>
<td className="px-3 py-2.5">
<code className="text-xs" style={{ color: "var(--bb-text-muted)" }}>{block}</code>
</td>
</tr>
))}
</tbody>
</table>
</div>
{/* Макет формы как на сайте */}
<div className="rounded-xl overflow-hidden border" style={{ borderColor: "var(--bb-border)" }}>
<div className="px-4 py-2 text-xs font-semibold uppercase tracking-widest" style={{ background: "var(--bb-sidebar-bg)", color: "var(--bb-text-muted)" }}>
Макет entityform «Узнайте стоимость операции» (oclinica.ru/lor)
</div>
<div style={{ background: "#b8e6ed", padding: "32px 24px" }}>
<div
className="mx-auto"
style={{
maxWidth: 340,
background: "#b8e6ed",
textAlign: "center",
}}
>
<p className="text-sm font-semibold mb-4" style={{ color: "#333", fontFamily: "var(--font-web)" }}>
Узнайте стоимость операции
</p>
<div className="flex flex-col gap-3" style={{ alignItems: "center" }}>
<input
className="bb-input"
type="text"
placeholder="Ваше имя"
style={{ width: 302 }}
readOnly
/>
<input
className="bb-input"
type="text"
placeholder="Ваш телефон"
style={{ width: 302 }}
readOnly
/>
<select className="bb-select" style={{ width: 302 }}>
<option>Выберите врача</option>
<option>Иванов И.И.</option>
</select>
<button className="bb-btn bb-btn-lg bb-btn-primary" style={{ width: 300 }}>
Запишите меня!
</button>
</div>
</div>
</div>
</div>
</Section>
{/* 8. Примеры кода */}
<Section
id="code"
title="Примеры кода"
subtitle="Скопируйте HTML или JSX для использования в проекте."
>
<div className="space-y-4">
<CodeCopy lang="HTML — Input" code={codeInput} />
<CodeCopy lang="HTML — Textarea" code={codeTextarea} />
<CodeCopy lang="HTML — Select" code={codeSelect} />
<CodeCopy lang="HTML — Checkbox" code={codeCheckbox} />
<CodeCopy lang="HTML — Radio" code={codeRadio} />
<CodeCopy lang="JSX (React) — Toggle" code={codeToggle} />
<CodeCopy lang="CSS с сайта (perm.oclinica.ru)" code={codeSiteCSS} />
</div>
</Section>
{/* LLM-блок */}
<LlmBlock path="/components/forms" version="v2.0" specText={LLM_FORMS_TEXT}>
<LlmSection title="Элементы ввода" />
<LlmTable
headers={["Элемент", "CSS класс", "Тег", "Высота", "Описание"]}
rows={[
["Input", ".bb-input", "<input>", "50px", "Текстовое поле, email, password · как на сайте"],
["Textarea", ".bb-textarea", "<textarea>", "≥100px", "Многострочный ввод, resize:vertical"],
["Select", ".bb-select", "<select>", "50px", "Выбор из списка, кастомная стрелка · как на сайте"],
["Checkbox", ".bb-checkbox", "<input type=checkbox>", "16×16px", "Независимый выбор"],
["Radio", ".bb-radio", "<input type=radio>", "16×16px", "Выбор одного из группы"],
["Toggle", ".bb-toggle-track", "React-компонент", "24px", "Булев переключатель"],
]}
/>
<LlmSection title="Состояния полей (Input / Textarea / Select)" />
<LlmTable
headers={["Состояние", "Стиль"]}
rows={[
["default", "border: 1px solid #ccc · border-radius: 4px"],
["focus", "border: #7ecfca + box-shadow: 0 0 0 3px rgba(126,207,202,0.2)"],
["error", "border: #dc2626 (+ класс .bb-error)"],
["disabled", "opacity: 0.5 + cursor: not-allowed + bg: #f8f9fa"],
]}
/>
<LlmSection title="Параметры Toggle" />
<LlmTable
headers={["Параметр", "Тип", "По умолч.", "Описание"]}
rows={[
["defaultChecked", "boolean", "false", "Начальное состояние"],
["disabled", "boolean", "false", "Блокирует взаимодействие"],
["label", "string", "—", "Текстовая метка справа от тумблера"],
["onChange", "(checked: boolean) => void", "—", "Колбэк при изменении"],
]}
/>
<LlmSection title="Правила применения" />
<LlmRules
rules={[
{ ok: true, text: "Метка (label) всегда над полем, font-weight: 500" },
{ ok: true, text: "Обязательные поля помечены * красным (#dc2626)" },
{ ok: true, text: "Hint-текст серым под полем (font-size: 12px)" },
{ ok: true, text: "Ошибка — красный текст вместо hint" },
{ ok: true, text: "Checkbox-группы — вертикальный список с gap: 10px" },
{ ok: true, text: "Toggle — для булевых настроек включить/выключить" },
{ ok: false, text: "Не использовать placeholder вместо label" },
{ ok: false, text: "Не скрывать признак обязательности поля" },
]}
/>
</LlmBlock>
</div>
);
}
+54 -54
View File
@@ -55,42 +55,42 @@ const BRAND_COLORS = [
{
oracal: "053M",
name: "Основной бирюзовый",
hex: "#7ecfca",
hex: "#0089c3",
usage: "Акцентный цвет, CTA-кнопки, иконки, активные состояния",
cssVar: "--brand-053m",
},
{
oracal: "073M",
name: "Тёмный серо-голубой",
hex: "#5b7b87",
hex: "#53514e",
usage: "Тёмный фон, хедер, акценты на тёмных поверхностях",
cssVar: "--brand-073m",
},
{
oracal: "066M",
name: "Средний бирюзовый",
hex: "#5bb5ad",
hex: "#00818c",
usage: "Вторичные акценты, фоны секций, иллюстрации",
cssVar: "--brand-066m",
},
{
oracal: "050M",
name: "Тёмно-синий",
hex: "#1b4c72",
hex: "#1b2e5d",
usage: "Наружная реклама, полиграфия, заголовки на светлом фоне",
cssVar: "--brand-050m",
},
{
oracal: "081M",
name: "Бежевый",
hex: "#c4a882",
hex: "#a8885c",
usage: "Форма сотрудников, оффлайн носители, тёплые акценты",
cssVar: "--brand-081m",
},
{
oracal: "080M",
name: "Тёмно-коричневый",
hex: "#5c2e0e",
hex: "#432f1e",
usage: "Текст на бежевых поверхностях, логотип на форме",
cssVar: "--brand-080m",
},
@@ -106,32 +106,32 @@ const BRAND_COLORS = [
/* ─── Соответствие цветов ──────────────────────────────────────────── */
const COLOR_MAPPING = [
{
brand: { oracal: "053M", name: "Основной бирюзовый", hex: "#7ecfca" },
brand: { oracal: "053M", name: "Основной бирюзовый", hex: "#0089c3" },
web: { name: "Бирюзовый", hex: "#63bac3", count: 4 },
note: "Чуть темнее и насыщеннее на сайте",
note: "Oracal — насыщенный синий; сайт — светлее и голубее",
},
{
brand: { oracal: "073M", name: "Тёмный серо-голубой", hex: "#5b7b87" },
brand: { oracal: "073M", name: "Тёмный серо-голубой", hex: "#53514e" },
web: { name: "Серо-бирюзовый", hex: "#60959c", count: 7 },
note: "Светлее на сайте",
note: "Oracal — тёмно-серый; сайт — серо-бирюзовый (значительное расхождение)",
},
{
brand: { oracal: "066M", name: "Средний бирюзовый", hex: "#5bb5ad" },
brand: { oracal: "066M", name: "Средний бирюзовый", hex: "#00818c" },
web: { name: "Бирюзовый средний", hex: "#52b4bd", count: 4 },
note: "Смещён в синеву",
note: "Oracal — тёмный бирюзовый; сайт — светлее",
},
{
brand: { oracal: "081M", name: "Бежевый", hex: "#c4a882" },
brand: { oracal: "081M", name: "Бежевый", hex: "#a8885c" },
web: { name: "Бежевый", hex: "#bf9975", count: 12 },
note: "Темнее и насыщеннее на сайте",
note: "Oracal — тёплый охристый; сайт — прохладнее и светлее",
},
{
brand: { oracal: "050M", name: "Тёмно-синий", hex: "#1b4c72" },
brand: { oracal: "050M", name: "Тёмно-синий", hex: "#1b2e5d" },
web: null,
note: "Не найден в CSS сайта",
},
{
brand: { oracal: "080M", name: "Тёмно-коричневый", hex: "#5c2e0e" },
brand: { oracal: "080M", name: "Тёмно-коричневый", hex: "#432f1e" },
web: null,
note: "Не найден в CSS сайта",
},
@@ -185,13 +185,13 @@ const WEB_COLORS = [
];
const CONTRAST_PAIRS = [
{ fg: "#ffffff", bg: "#5b7b87", label: "Белый на тёмном серо-голубом" },
{ fg: "#ffffff", bg: "#1b4c72", label: "Белый на тёмно-синем" },
{ fg: "#ffffff", bg: "#5c2e0e", label: "Белый на тёмно-коричневом" },
{ fg: "#ffffff", bg: "#5bb5ad", label: "Белый на среднем бирюзовом" },
{ fg: "#111827", bg: "#7ecfca", label: "Тёмный текст на основном бирюзовом" },
{ fg: "#111827", bg: "#c4a882", label: "Тёмный текст на бежевом" },
{ fg: "#5c2e0e", bg: "#c4a882", label: "Тёмно-коричневый на бежевом (форма)" },
{ fg: "#ffffff", bg: "#53514e", label: "Белый на тёмном серо-голубом (073M)" },
{ fg: "#ffffff", bg: "#1b2e5d", label: "Белый на тёмно-синем (050M)" },
{ fg: "#ffffff", bg: "#432f1e", label: "Белый на тёмно-коричневом (080M)" },
{ fg: "#ffffff", bg: "#00818c", label: "Белый на среднем бирюзовом (066M)" },
{ fg: "#111827", bg: "#0089c3", label: "Тёмный текст на основном бирюзовом (053M)" },
{ fg: "#111827", bg: "#a8885c", label: "Тёмный текст на бежевом (081M)" },
{ fg: "#432f1e", bg: "#a8885c", label: "Тёмно-коричневый на бежевом (форма, 080M/081M)" },
];
/* ─── Компоненты ───────────────────────────────────────────────────── */
@@ -403,16 +403,16 @@ function exportTokens() {
/* ─── LLM spec text ────────────────────────────────────────────────── */
const LLM_COLORS_TEXT = `# ЦВЕТА — LLM-СПЕЦИФИКАЦИЯ
# Клиника ухо, горло, нос им. проф. Е.Н.Оленевой
# docs/LLM_CONTEXT.md · /foundation/colors · v2.1 · 2026-03-22
# docs/LLM_CONTEXT.md · /foundation/colors · v2.2 · 2026-03-22
ФИРМЕННЫЕ ЦВЕТА (Oracal)
ФИРМЕННЫЕ ЦВЕТА (Oracal) — точные значения из каталога
Oracal | Название | HEX | RGB | CSS-переменная | Применение
053M | Основной бирюзовый | #7ECFCA | rgb(126,207,202) | --brand-053m | Акцент, CTA-кнопки, иконки, активные состояния
073M | Тёмный серо-голубой | #5B7B87 | rgb(91,123,135) | --brand-073m | Тёмный фон, хедер, заголовки навигации
066M | Средний бирюзовый | #5BB5AD | rgb(91,181,173) | --brand-066m | Вторичные акценты, фоны секций
050M | Тёмно-синий | #1B4C72 | rgb(27,76,114) | --brand-050m | Наружная реклама, полиграфия, заголовки
081M | Бежевый | #C4A882 | rgb(196,168,130) | --brand-081m | Форма сотрудников, тёплые акценты
080M | Тёмно-коричневый | #5C2E0E | rgb(92,46,14) | --brand-080m | Текст на бежевом, логотип на форме
053M | Основной бирюзовый | #0089C3 | rgb(0,137,195) | --brand-053m | Акцент, CTA-кнопки, иконки, активные состояния
073M | Тёмный серо-голубой | #53514E | rgb(83,81,78) | --brand-073m | Тёмный фон, хедер, заголовки навигации
066M | Средний бирюзовый | #00818C | rgb(0,129,140) | --brand-066m | Вторичные акценты, фоны секций
050M | Тёмно-синий | #1B2E5D | rgb(27,46,93) | --brand-050m | Наружная реклама, полиграфия, заголовки
081M | Бежевый | #A8885C | rgb(168,136,92) | --brand-081m | Форма сотрудников, тёплые акценты
080M | Тёмно-коричневый | #432F1E | rgb(67,47,30) | --brand-080m | Текст на бежевом, логотип на форме
— | Белый | #FFFFFF | rgb(255,255,255) | --brand-white | Фон, инвертированный текст, логотип на тёмных
ЦВЕТА САЙТА oclinica.ru (CSS: clinic_bootstrap_mobile/css/style.css)
@@ -429,29 +429,29 @@ Oracal | Название | HEX | RGB | CSS-пер
Светло-жёлтый фон | #EEF4D1 | Фоны | 1 | Фон карточек отзывов
Светло-зелёный фон | #F2FEE6 | Фоны | 1 | Фон секции новостей
СООТВЕТСТВИЕ ORACAL → САЙТ
053M #7ECFCA → #63BAC3 (темнее, насыщеннее)
073M #5B7B87 → #60959C (светлее)
066M #5BB5AD → #52B4BD (смещён в синеву)
081M #C4A882 → #BF9975 (темнее, насыщеннее)
050M #1B4C72 → не найден в CSS сайта
080M #5C2E0E → не найден в CSS сайта
СООТВЕТСТВИЕ ORACAL → САЙТ (цифровая адаптация плёночных цветов)
053M #0089C3 → #63BAC3 (Oracal ярко-синий; сайт светлее и голубее)
073M #53514E → #60959C (Oracal тёмно-серый; сайт серо-бирюзовый — значительное расхождение)
066M #00818C → #52B4BD (Oracal тёмный бирюзовый; сайт светлее)
081M #A8885C → #BF9975 (Oracal тёплый охристый; сайт прохладнее и светлее)
050M #1B2E5D → не найден в CSS сайта
080M #432F1E → не найден в CSS сайта
КОНТРАСТНОСТЬ WCAG 2.1
#FFFFFF / #5B7B87 | 4.6:1 | AA PASS | AAA FAIL
#FFFFFF / #1B4C72 | 9.3:1 | AA PASS | AAA PASS
#FFFFFF / #5C2E0E | 11.2:1 | AA PASS | AAA PASS
#FFFFFF / #5BB5AD | 3.0:1 | AA FAIL | AAA FAIL | только крупный текст ≥18pt
#111827 / #7ECFCA | 5.8:1 | AA PASS | AAA FAIL
#111827 / #C4A882 | 4.8:1 | AA PASS | AAA FAIL
#5C2E0E / #C4A882 | 6.7:1 | AA PASS | AAA FAIL
#FFFFFF / #53514E | 7.9:1 | AA PASS | AAA PASS
#FFFFFF / #1B2E5D | 13.2:1 | AA PASS | AAA PASS
#FFFFFF / #432F1E | 12.6:1 | AA PASS | AAA PASS
#FFFFFF / #00818C | 4.7:1 | AA PASS | AAA FAIL
#111827 / #0089C3 | 4.7:1 | AA PASS | AAA FAIL
#111827 / #A8885C | 5.5:1 | AA PASS | AAA FAIL
#432F1E / #A8885C | 3.8:1 | AA FAIL | AAA FAIL | только крупный текст ≥18pt
ПРАВИЛА
✓ Только цвета из фирменной палитры
✓ Digital → цвета сайта; оффлайн → коды Oracal
✓ Текст на цветном фоне: минимум WCAG AA (4.5:1)
✓ Белый текст на: 073M (#5B7B87), 050M (#1B4C72), 080M (#5C2E0E)
✓ Тёмный текст на: 053M (#7ECFCA), 081M (#C4A882)
✓ Белый текст на: 073M (#53514E), 066M (#00818C), 050M (#1B2E5D), 080M (#432F1E)
✓ Тёмный текст (#111827) на: 053M (#0089C3), 081M (#A8885C)
✕ Произвольные цвета вне фирменной палитры
✕ Изменение насыщенности / оттенка фирменных цветов
✕ Тёплые и холодные акценты рядом без нейтрального разделителя`.trim();
@@ -480,12 +480,12 @@ export default function ColorsPage() {
<div className="mt-4 flex items-center justify-between">
<div
className="px-4 py-3 rounded-lg border text-sm flex items-center gap-2"
style={{ borderColor: "#fde68a", background: "#fffbeb", color: "#92400e" }}
style={{ borderColor: "#bae6fd", background: "#f0f9ff", color: "#075985" }}
>
<span></span>
<span></span>
<span>
HEX-значения приблизительны. Для оффлайн-носителей используйте
официальные коды Oracal.
HEX-значения рассчитаны по точным RGB из каталога Oracal.
Для оффлайн-носителей используйте официальные коды Oracal.
</span>
</div>
<button
@@ -723,7 +723,7 @@ export default function ColorsPage() {
</div>
<LlmBlock
path="/foundation/colors"
version="v2.1"
version="v2.2"
specText={LLM_COLORS_TEXT}
>
{/* Фирменные цвета */}
@@ -799,8 +799,8 @@ export default function ColorsPage() {
{ ok: true, text: "Только цвета из фирменной палитры" },
{ ok: true, text: "Digital → цвета сайта; оффлайн → коды Oracal" },
{ ok: true, text: "Текст на цветном фоне: минимум WCAG AA (4.5:1)" },
{ ok: true, text: "Белый текст на: 073M, 050M, 080M" },
{ ok: true, text: "Тёмный текст на: 053M, 081M" },
{ ok: true, text: "Белый текст на: 073M (#53514E), 066M (#00818C), 050M (#1B2E5D), 080M (#432F1E)" },
{ ok: true, text: "Тёмный текст (#111827) на: 053M (#0089C3), 081M (#A8885C)" },
{ ok: false, text: "Произвольные цвета вне фирменной палитры" },
{ ok: false, text: "Изменение насыщенности / оттенка фирменных цветов" },
{ ok: false, text: "Тёплые + холодные акценты рядом без разделителя" },
+88
View File
@@ -1,5 +1,6 @@
import type { Metadata } from "next";
import Image from "next/image";
import { LlmBlock, LlmSection, LlmTable, LlmRules } from "@/components/llm/LlmBlock";
export const metadata: Metadata = {
title: "Логотип. Цифровой брендбук Клиники ухо, горло, нос им. проф. Е.Н.Оленевой",
@@ -102,6 +103,44 @@ function ProhibitedItem({ label }: { label: string }) {
);
}
const LLM_LOGO_TEXT = `ЛОГОТИП — LLM-спецификация
Версия: v1.0 · /foundation/logo
ФАЙЛЫ
PNG с прозрачным фоном: /public/logo/logo-transparent.png
SVG/AI: ожидается от дизайнера
СИМВОЛИКА ЗНАКА
Три округлых элемента с равной дистанцией от центра.
- Незамкнутая симметрия: символ развития и жизни, а не завершённости
- Три элемента: структура ухо-горло-нос, триада равновесия
- Отсутствие замкнутой формы: открытость, доступность, человечность
ЦВЕТОВЫЕ ВАРИАНТЫ
Вариант | Фон | CSS-фильтр | Применение
Основной | Светлый (#f8f9fa) | нет | Сайт, полиграфия на белом
Инвертированный | #5b7b87 (073M) | brightness(0) invert(1) | Хедер, тёмные секции
На форме (беж) | #c4a882 (081M) | brightness(0) sepia(1) saturate(2) hue-rotate(330deg) brightness(0.45) | Форма сотрудников (бежевый)
На форме (синий)| #1b4c72 (050M) | brightness(0) invert(1) | Форма сотрудников (синий)
ОХРАННАЯ ЗОНА
Минимальный отступ со всех сторон = высота буквы «К» в слове «КЛИНИКА»
РАЗМЕРЫ НА ФОРМЕ СОТРУДНИКОВ
До 46 р. | 70 мм × 25,5 мм | Левая сторона груди
От 48 р. | 90 мм × 32,8 мм | Левая сторона груди
ПРАВИЛА
Применять только одобренные цветовые варианты
Соблюдать охранную зону
Использовать PNG с прозрачным фоном для digital
Белый логотип (invert) на тёмных фонах (073M, 050M)
Не изменять пропорции или искажать логотип
Не изменять цвета элементов логотипа
Не добавлять рядом произвольный текст
Не размещать на фоне без достаточного контраста
Не применять тени, обводки, градиенты`.trim();
export default function LogoPage() {
return (
<div className="max-w-4xl mx-auto px-8 py-10">
@@ -390,6 +429,55 @@ export default function LogoPage() {
</div>
</Section>
{/* LLM-блок */}
<LlmBlock
path="/foundation/logo"
version="v1.0"
specText={LLM_LOGO_TEXT}
>
<LlmSection title="Файлы логотипа" />
<LlmTable
headers={["Файл", "Формат", "Путь", "Статус"]}
rows={[
["logo-transparent.png", "PNG с прозрачным фоном", "/public/logo/logo-transparent.png", "✓ Доступен"],
["logo.svg / logo.ai", "Вектор", "—", "Ожидается от дизайнера"],
]}
/>
<LlmSection title="Цветовые варианты" />
<LlmTable
headers={["Вариант", "Фон", "CSS-фильтр", "Применение"]}
rows={[
["Основной", "Светлый (#f8f9fa)", "нет", "Сайт, полиграфия на белом"],
["Инвертированный", "#5b7b87 (073M)", "brightness(0) invert(1)", "Хедер, тёмные секции"],
["На форме (беж)", "#c4a882 (081M)", "brightness(0) sepia(1) saturate(2) hue-rotate(330deg) brightness(0.45)", "Форма сотрудников (бежевый костюм)"],
["На форме (синий)", "#1b4c72 (050M)", "brightness(0) invert(1)", "Форма сотрудников (синий костюм)"],
]}
/>
<LlmSection title="Охранная зона и размеры на форме" />
<LlmTable
headers={["Носитель", "Условие", "Длина", "Высота", "Расположение"]}
rows={[
["Форма сотрудников", "Размер до 46 р.", "70 мм", "25,5 мм", "Левая сторона груди"],
["Форма сотрудников", "Размер от 48 р.", "90 мм", "32,8 мм", "Левая сторона груди"],
["Охранная зона", "Все носители", "≥ высота буквы «К»", "≥ высота буквы «К»", "Вокруг логотипа со всех сторон"],
]}
/>
<LlmSection title="Правила использования" />
<LlmRules
rules={[
{ ok: true, text: "Применять только одобренные цветовые варианты" },
{ ok: true, text: "Соблюдать охранную зону (≥ высота буквы «К»)" },
{ ok: true, text: "Использовать PNG с прозрачным фоном для digital" },
{ ok: true, text: "Белый логотип на тёмных фонах (073M, 050M, 080M)" },
{ ok: false, text: "Не изменять пропорции или искажать логотип" },
{ ok: false, text: "Не изменять цвета элементов логотипа" },
{ ok: false, text: "Не добавлять рядом произвольный текст" },
{ ok: false, text: "Не размещать на фоне без достаточного контраста" },
{ ok: false, text: "Не применять тени, обводки, градиенты" },
]}
/>
</LlmBlock>
</div>
);
}
+155 -8
View File
@@ -3,13 +3,13 @@
/* ─── Бренд-токены О!Клиника ─────────────────────────────────────────── */
/* Цвета уточняются в Sprint 2 по таблице Oracal */
:root {
/* Фирменные цвета (приблизительноуточнить в Sprint 2) */
--brand-053m: #7ecfca; /* Основной бирюзовый (Oracal 053M) */
--brand-073m: #5b7b87; /* Тёмный серо-голубой (Oracal 073M) */
--brand-066m: #5bb5ad; /* Средний бирюзовый (Oracal 066M) */
--brand-050m: #1b4c72; /* Тёмно-синий, наружная реклама (Oracal 050M) */
--brand-081m: #c4a882; /* Бежевый (Oracal 081M) */
--brand-080m: #5c2e0e; /* Тёмно-коричневый (Oracal 080M) */
/* Фирменные цвета — точные RGB из каталога Oracal */
--brand-053m: #0089c3; /* Основной бирюзовый (Oracal 053M) · rgb(0,137,195) */
--brand-073m: #53514e; /* Тёмный серо-голубой (Oracal 073M) · rgb(83,81,78) */
--brand-066m: #00818c; /* Средний бирюзовый (Oracal 066M) · rgb(0,129,140) */
--brand-050m: #1b2e5d; /* Тёмно-синий, наружная реклама (Oracal 050M) · rgb(27,46,93) */
--brand-081m: #a8885c; /* Бежевый (Oracal 081M) · rgb(168,136,92) */
--brand-080m: #432f1e; /* Тёмно-коричневый (Oracal 080M) · rgb(67,47,30) */
--brand-white: #ffffff;
/* UI-цвета брендбука */
@@ -17,7 +17,7 @@
--bb-sidebar-border: #e5e7eb;
--bb-sidebar-text: #374151;
--bb-sidebar-text-muted: #6b7280;
--bb-sidebar-active-bg: #e0f5f4;
--bb-sidebar-active-bg: #dff0fa;
--bb-sidebar-active-text: var(--brand-053m);
--bb-sidebar-section: #9ca3af;
--bb-content-bg: #ffffff;
@@ -42,3 +42,150 @@ body {
color: var(--bb-text);
-webkit-font-smoothing: antialiased;
}
/* ─── Анимации ───────────────────────────────────────────────── */
@keyframes bb-spin {
to { transform: rotate(360deg); }
}
/* ─── Кнопки (Sprint 3) ──────────────────────────────────────── */
.bb-btn {
font-family: var(--font-web);
font-weight: 500;
display: inline-flex;
align-items: center;
justify-content: center;
gap: 7px;
cursor: pointer;
transition: filter 0.15s, opacity 0.15s;
white-space: nowrap;
line-height: 1;
text-decoration: none;
-webkit-font-smoothing: antialiased;
}
.bb-btn:hover:not(:disabled) { filter: brightness(0.9); }
.bb-btn:active:not(:disabled) { filter: brightness(0.82); }
.bb-btn:disabled { cursor: not-allowed; opacity: 0.5; }
.bb-btn:focus-visible { outline: 2px solid var(--brand-053m); outline-offset: 2px; }
/* Размеры — только padding и font-size, radius задаётся вариантом */
.bb-btn-sm { font-size: 13px; padding: 4px 11px; border: 1px solid transparent; }
.bb-btn-md { font-size: 14px; padding: 8px 16px; border: 1px solid transparent; }
.bb-btn-lg { font-size: 18px; padding: 10px 24px; border: 1px solid transparent; font-weight: bold; }
/* Варианты — цвета и радиус по реальному сайту oclinica.ru */
/* primary — коралловая форм-кнопка «Запишите меня!» — #FFA39C */
.bb-btn-primary {
background: #FFA39C;
color: #fff;
border-color: #FF847B;
border-radius: 7px;
font-weight: bold;
box-shadow: 0px 0px 5px rgba(0,0,0,0.4), 0px 3px 5px rgba(0,0,0,0.25);
}
/* outline — белая с бежевой рамкой «Записаться на прием» */
.bb-btn-outline {
background: #fff;
color: #BF9975;
border-color: #BF9975;
border-radius: 7px;
}
/* teal — бирюзовая «Позвонить» */
.bb-btn-teal {
background: #60959c;
color: #fff;
border-color: transparent;
border-radius: 7px;
}
/* pill — кремовая таблетка «Заказать звонок» */
.bb-btn-pill {
background: #e9e4d4;
color: #333;
border-color: #d5cfbd;
border-radius: 25px;
}
/* ─── Форм-контролы (Sprint 3) ───────────────────────────────── */
.bb-input,
.bb-textarea,
.bb-select {
font-family: var(--font-web);
font-size: 14px;
color: var(--bb-text);
background: #fff;
border: 1px solid #ccc;
border-radius: 4px;
padding: 10px 12px;
width: 100%;
transition: border-color 0.15s, box-shadow 0.15s;
outline: none;
-webkit-font-smoothing: antialiased;
}
.bb-input { height: 50px; }
.bb-input:focus,
.bb-textarea:focus,
.bb-select:focus {
border-color: var(--brand-053m);
box-shadow: 0 0 0 3px rgba(0, 137, 195, 0.2);
}
.bb-input.bb-error,
.bb-textarea.bb-error,
.bb-select.bb-error { border-color: #dc2626; }
.bb-input.bb-error:focus,
.bb-textarea.bb-error:focus,
.bb-select.bb-error:focus { box-shadow: 0 0 0 3px rgba(220, 38, 38, 0.15); }
.bb-input:disabled,
.bb-textarea:disabled,
.bb-select:disabled { opacity: 0.5; cursor: not-allowed; background: var(--bb-sidebar-bg); }
.bb-textarea { resize: vertical; min-height: 100px; }
.bb-select {
cursor: pointer;
appearance: none;
height: 50px;
padding: 10px 36px 10px 10px;
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 24 24' fill='none' stroke='%236b7280' stroke-width='2'%3E%3Cpath d='m6 9 6 6 6-6'/%3E%3C/svg%3E");
background-repeat: no-repeat;
background-position: right 10px center;
}
.bb-checkbox,
.bb-radio {
width: 16px;
height: 16px;
cursor: pointer;
accent-color: var(--brand-073m);
flex-shrink: 0;
}
/* ─── Карточки (Sprint 4) ────────────────────────────────────── */
.bb-news-card:hover {
background: #eef4d1 !important;
box-shadow: 0 0 16px 0 #9e9e9a;
}
.bb-service-card:hover { box-shadow: 0 4px 12px rgba(0,0,0,0.1); }
/* ─── Тумблер (Sprint 3) ──────────────────────────────────────── */
.bb-toggle-track {
display: inline-flex;
align-items: center;
width: 44px;
height: 24px;
border-radius: 12px;
padding: 2px;
cursor: pointer;
transition: background 0.2s;
flex-shrink: 0;
}
.bb-toggle-thumb {
width: 20px;
height: 20px;
border-radius: 50%;
background: #fff;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25);
transition: transform 0.2s;
}
+6 -6
View File
@@ -27,9 +27,9 @@ const NAV: NavSection[] = [
{
title: "Компоненты",
items: [
{ label: "Кнопки", href: "/components/buttons", soon: true },
{ label: "Форм-контролы", href: "/components/forms", soon: true },
{ label: "Карточки", href: "/components/cards", soon: true },
{ label: "Кнопки", href: "/components/buttons" },
{ label: "Форм-контролы", href: "/components/forms" },
{ label: "Карточки", href: "/components/cards" },
{ label: "Бейджи и теги", href: "/components/badges", soon: true },
{ label: "Алерты", href: "/components/alerts", soon: true },
{ label: "Модальные окна", href: "/components/modals", soon: true },
@@ -40,8 +40,8 @@ const NAV: NavSection[] = [
{
title: "Блоки",
items: [
{ label: "Hero", href: "/blocks/hero", soon: true },
{ label: "CEO-текст", href: "/blocks/ceo", soon: true },
{ label: "Hero", href: "/blocks/hero" },
{ label: "CEO-текст", href: "/blocks/ceo" },
{ label: "Наши врачи", href: "/blocks/doctors", soon: true },
{ label: "Отзывы", href: "/blocks/reviews", soon: true },
{ label: "Новости", href: "/blocks/news", soon: true },
@@ -170,7 +170,7 @@ export function Sidebar() {
color: "var(--bb-sidebar-text-muted)",
}}
>
Sprint 2 · v0.2.0
Sprint 5 · v0.5.0
</div>
</aside>
);
+45
View File
@@ -0,0 +1,45 @@
"use client";
import { ButtonHTMLAttributes, forwardRef } from "react";
export type ButtonVariant = "primary" | "outline" | "teal" | "pill";
export type ButtonSize = "sm" | "md" | "lg";
export interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> {
variant?: ButtonVariant;
size?: ButtonSize;
loading?: boolean;
}
export const Button = forwardRef<HTMLButtonElement, ButtonProps>(
function Button(
{ variant = "primary", size = "md", loading = false, disabled, children, className = "", ...props },
ref
) {
const isDisabled = disabled || loading;
return (
<button
ref={ref}
disabled={isDisabled}
className={`bb-btn bb-btn-${size} bb-btn-${variant} ${className}`.trim()}
{...props}
>
{loading && (
<span
style={{
width: 13,
height: 13,
border: "2px solid currentColor",
borderTopColor: "transparent",
borderRadius: "50%",
display: "inline-block",
animation: "bb-spin 0.65s linear infinite",
flexShrink: 0,
}}
/>
)}
{children}
</button>
);
}
);
+60
View File
@@ -0,0 +1,60 @@
"use client";
import { useState } from "react";
export function CodeCopy({ code, lang = "jsx" }: { code: string; lang?: string }) {
const [copied, setCopied] = useState(false);
return (
<div style={{ borderRadius: 8, overflow: "hidden", border: "1px solid var(--bb-border)" }}>
<div
style={{
display: "flex",
justifyContent: "space-between",
alignItems: "center",
background: "var(--bb-sidebar-bg)",
padding: "6px 12px",
borderBottom: "1px solid var(--bb-border)",
}}
>
<span style={{ fontSize: 11, color: "var(--bb-text-muted)", fontFamily: "var(--font-mono)" }}>
{lang}
</span>
<button
onClick={() => {
navigator.clipboard.writeText(code);
setCopied(true);
setTimeout(() => setCopied(false), 2000);
}}
style={{
fontSize: 12,
background: copied ? "#d1fae5" : "var(--brand-053m)",
color: copied ? "#065f46" : "#fff",
border: "none",
borderRadius: 4,
padding: "3px 10px",
cursor: "pointer",
fontWeight: 500,
fontFamily: "var(--font-web)",
}}
>
{copied ? "✓ Скопировано" : "Скопировать"}
</button>
</div>
<pre
style={{
margin: 0,
padding: "12px 16px",
overflowX: "auto",
fontSize: 12,
fontFamily: "var(--font-mono)",
color: "var(--bb-text-muted)",
background: "#fff",
lineHeight: 1.6,
}}
>
<code>{code}</code>
</pre>
</div>
);
}
+59
View File
@@ -0,0 +1,59 @@
"use client";
import { useState } from "react";
interface ToggleProps {
defaultChecked?: boolean;
disabled?: boolean;
label?: string;
onChange?: (checked: boolean) => void;
}
export function Toggle({ defaultChecked = false, disabled = false, label, onChange }: ToggleProps) {
const [checked, setChecked] = useState(defaultChecked);
function handleToggle() {
if (disabled) return;
const next = !checked;
setChecked(next);
onChange?.(next);
}
return (
<div
style={{
display: "inline-flex",
alignItems: "center",
gap: 10,
cursor: disabled ? "not-allowed" : "pointer",
opacity: disabled ? 0.5 : 1,
userSelect: "none",
}}
onClick={handleToggle}
role="switch"
aria-checked={checked}
tabIndex={disabled ? -1 : 0}
onKeyDown={(e) => {
if (e.key === " " || e.key === "Enter") {
e.preventDefault();
handleToggle();
}
}}
>
<div
className="bb-toggle-track"
style={{ background: checked ? "var(--brand-073m)" : "#d1d5db" }}
>
<div
className="bb-toggle-thumb"
style={{ transform: checked ? "translateX(20px)" : "translateX(0)" }}
/>
</div>
{label && (
<span style={{ fontSize: 14, color: "var(--bb-text)", fontFamily: "var(--font-web)" }}>
{label}
</span>
)}
</div>
);
}
Binary file not shown.

After

Width:  |  Height:  |  Size: 120 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 47 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 118 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 76 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 24 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 236 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 132 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 740 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 767 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 145 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 957 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 446 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 699 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 153 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 141 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 153 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 141 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 153 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 141 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 477 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 332 KiB

+143 -67
View File
@@ -2,9 +2,9 @@
## Клиника ухо, горло, нос им. проф. Е.Н.Оленевой
**Версия контекста:** 2.1
**Версия контекста:** 4.1
**Дата обновления:** 2026-03-22
**Актуальный спринт:** Sprint 3
**Актуальный спринт:** Sprint 5
**Сайт клиники:** https://oclinica.ru
**Брендбук (локально):** http://localhost:3001
**Брендбук (production):** https://web-oclinica.vercel.app
@@ -39,19 +39,18 @@
### 2.1 Фирменные цвета (Oracal — базовая палитра)
Основаны на кодах самоклеящейся плёнки Oracal. HEX-значения — приближённые
цифровые эквиваленты. **Для печати и оффлайн-носителей используй коды Oracal,
не HEX.**
Основаны на кодах самоклеящейся плёнки Oracal. HEX-значения рассчитаны по **точным RGB
из каталога Oracal**. Для печати и оффлайн-носителей используй коды Oracal, не HEX.
| Oracal | Название | HEX | RGB | HSL | CSS-переменная | Применение |
|--------|-----------------------|-----------|-------------------|------------------|------------------|------------|
| 053M | Основной бирюзовый | `#7ecfca` | rgb(126,207,202) | hsl(177,47%,65%) | `--brand-053m` | Акцент, CTA-кнопки, иконки, активные состояния |
| 073M | Тёмный серо-голубой | `#5b7b87` | rgb(91,123,135) | hsl(197,20%,44%) | `--brand-073m` | Тёмный фон, хедер, заголовки навигации |
| 066M | Средний бирюзовый | `#5bb5ad` | rgb(91,181,173) | hsl(174,33%,53%) | `--brand-066m` | Вторичные акценты, фоны секций, иллюстрации |
| 050M | Тёмно-синий | `#1b4c72` | rgb(27,76,114) | hsl(208,61%,28%) | `--brand-050m` | Наружная реклама, полиграфия, заголовки на светлом |
| 081M | Бежевый | `#c4a882` | rgb(196,168,130) | hsl(33,35%,64%) | `--brand-081m` | Форма сотрудников, оффлайн носители, тёплые акценты |
| 080M | Тёмно-коричневый | `#5c2e0e` | rgb(92,46,14) | hsl(23,73%,21%) | `--brand-080m` | Текст на бежевых поверхностях, логотип на форме |
| — | Белый | `#ffffff` | rgb(255,255,255) | hsl(0,0%,100%) | `--brand-white` | Фон, инвертированный текст, логотип на тёмных фонах |
| Oracal | Название | HEX | RGB | HSL | CSS-переменная | Применение |
|--------|-----------------------|-----------|-------------------|-------------------|------------------|------------|
| 053M | Основной бирюзовый | `#0089c3` | rgb(0,137,195) | hsl(198,100%,38%) | `--brand-053m` | Акцент, CTA-кнопки, иконки, активные состояния |
| 073M | Тёмный серо-голубой | `#53514e` | rgb(83,81,78) | hsl(38,3%,32%) | `--brand-073m` | Тёмный фон, хедер, заголовки навигации |
| 066M | Средний бирюзовый | `#00818c` | rgb(0,129,140) | hsl(185,100%,27%) | `--brand-066m` | Вторичные акценты, фоны секций, иллюстрации |
| 050M | Тёмно-синий | `#1b2e5d` | rgb(27,46,93) | hsl(223,55%,24%) | `--brand-050m` | Наружная реклама, полиграфия, заголовки на светлом |
| 081M | Бежевый | `#a8885c` | rgb(168,136,92) | hsl(35,30%,51%) | `--brand-081m` | Форма сотрудников, оффлайн носители, тёплые акценты |
| 080M | Тёмно-коричневый | `#432f1e` | rgb(67,47,30) | hsl(27,38%,19%) | `--brand-080m` | Текст на бежевых поверхностях, логотип на форме |
| — | Белый | `#ffffff` | rgb(255,255,255) | hsl(0,0%,100%) | `--brand-white` | Фон, инвертированный текст, логотип на тёмных фонах |
### 2.2 Цвета сайта oclinica.ru (реальный CSS)
@@ -77,27 +76,27 @@
| Oracal (бренд) | HEX бренда | Сайт (цифровой) | HEX сайта | Отклонение |
|--------------------------|------------|-----------------------|-----------|------------|
| 053M Основной бирюзовый | `#7ecfca` | Бирюзовый | `#63bac3` | Темнее, насыщеннее |
| 073M Тёмный серо-голубой | `#5b7b87` | Серо-бирюзовый | `#60959c` | Светлее на сайте |
| 066M Средний бирюзовый | `#5bb5ad` | Бирюзовый средний | `#52b4bd` | Смещён в синеву |
| 081M Бежевый | `#c4a882` | Бежевый | `#bf9975` | Темнее, насыщеннее |
| 050M Тёмно-синий | `#1b4c72` | — | — | Не найден в CSS сайта |
| 080M Тёмно-коричневый | `#5c2e0e` | — | — | Не найден в CSS сайта |
| 053M Основной бирюзовый | `#0089c3` | Бирюзовый | `#63bac3` | Oracal ярко-синий; сайт светлее и голубее |
| 073M Тёмный серо-голубой | `#53514e` | Серо-бирюзовый | `#60959c` | Oracal тёмно-серый; сайт серо-бирюзовый (значительное расхождение) |
| 066M Средний бирюзовый | `#00818c` | Бирюзовый средний | `#52b4bd` | Oracal тёмный бирюзовый; сайт светлее |
| 081M Бежевый | `#a8885c` | Бежевый | `#bf9975` | Oracal тёплый охристый; сайт прохладнее и светлее |
| 050M Тёмно-синий | `#1b2e5d` | — | — | Не найден в CSS сайта |
| 080M Тёмно-коричневый | `#432f1e` | — | — | Не найден в CSS сайта |
**Важно:** расхождения ожидаемы — это цифровая адаптация плёночных цветов под экран.
При разработке digital-материалов используй цвета сайта (раздел 2.2), не Oracal.
### 2.4 Контрастность пар (WCAG 2.1)
| Пара | Коэффициент | AA (4.5:1) | AAA (7:1) | AA large (3:1) |
|-------------------------------------------|-------------|------------|-----------|----------------|
| Белый на тёмном серо-голубом (#5b7b87) | 4.6:1 | ✓ | | ✓ |
| Белый на тёмно-синем (#1b4c72) | 9.3:1 | ✓ | ✓ | ✓ |
| Белый на тёмно-коричневом (#5c2e0e) | 11.2:1 | ✓ | ✓ | ✓ |
| Белый на среднем бирюзовом (#5bb5ad) | 3.0:1 | | ✕ | ✓ |
| Тёмный текст (#111827) на основном бирюзовом (#7ecfca) | 5.8:1 | ✓ | ✕ | ✓ |
| Тёмный текст (#111827) на бежевом (#c4a882) | 4.8:1 | ✓ | ✕ | ✓ |
| Тёмно-коричневый (#5c2e0e) на бежевом (#c4a882) | 6.7:1 | ✓ | ✕ | ✓ |
| Пара | Коэффициент | AA (4.5:1) | AAA (7:1) | AA large (3:1) |
|----------------------------------------------|-------------|------------|-----------|----------------|
| Белый на тёмном серо-голубом (#53514e 073M) | 7.9:1 | ✓ | | ✓ |
| Белый на тёмно-синем (#1b2e5d 050M) | 13.2:1 | ✓ | ✓ | ✓ |
| Белый на тёмно-коричневом (#432f1e 080M) | 12.6:1 | ✓ | ✓ | ✓ |
| Белый на среднем бирюзовом (#00818c 066M) | 4.7:1 | | ✕ | ✓ |
| Тёмный (#111827) на основном бирюзовом (#0089c3 053M) | 4.7:1 | ✓ | ✕ | ✓ |
| Тёмный (#111827) на бежевом (#a8885c 081M) | 5.5:1 | ✓ | ✕ | ✓ |
| Тёмно-коричневый (#432f1e) на бежевом (#a8885c) | 3.8:1 | ✕ | ✕ | ✓ (только крупный ≥18pt) |
---
@@ -173,9 +172,9 @@ SVG-версия ожидается (не получена от клиники).
| Вариант | Фон | Логотип | Применение |
|---------|-----|---------|------------|
| Основной | Белый / светлый | Полноцветный | Сайт, полиграфия на белом |
| Инвертированный | Тёмный (#5b7b87, #1b4c72) | Белый (`filter: brightness(0) invert(1)`) | Хедер, тёмные секции |
| На форме (беж) | Бежевый (#c4a882 / Oracal 081M) | Коричневый (#5c2e0e / Oracal 080M) | Одежда персонала |
| На форме (синий) | Тёмно-синий (#1b4c72 / Oracal 050M) | Белый | Одежда персонала |
| Инвертированный | Тёмный (#53514e, #1b2e5d) | Белый (`filter: brightness(0) invert(1)`) | Хедер, тёмные секции |
| На форме (беж) | Бежевый (#a8885c / Oracal 081M) | Коричневый (#432f1e / Oracal 080M) | Одежда персонала |
| На форме (синий) | Тёмно-синий (#1b2e5d / Oracal 050M) | Белый | Одежда персонала |
### 4.4 Охранная зона
@@ -205,26 +204,27 @@ SVG-версия ожидается (не получена от клиники).
Определены в `apps/web/app/globals.css`:
```css
/* Цвета бренда */
--brand-053m: #7ecfca; /* Основной бирюзовый */
--brand-073m: #5b7b87; /* Тёмный серо-голубой */
--brand-066m: #5bb5ad; /* Средний бирюзовый */
--brand-050m: #1b4c72; /* Тёмно-синий */
--brand-081m: #c4a882; /* Бежевый */
--brand-080m: #5c2e0e; /* Тёмно-коричневый */
/* Цвета бренда (точные RGB из каталога Oracal) */
--brand-053m: #0089c3; /* Основной бирюзовый · rgb(0,137,195) */
--brand-073m: #53514e; /* Тёмный серо-голубой · rgb(83,81,78) */
--brand-066m: #00818c; /* Средний бирюзовый · rgb(0,129,140) */
--brand-050m: #1b2e5d; /* Тёмно-синий · rgb(27,46,93) */
--brand-081m: #a8885c; /* Бежевый · rgb(168,136,92) */
--brand-080m: #432f1e; /* Тёмно-коричневый · rgb(67,47,30) */
--brand-white: #ffffff; /* Белый */
/* UI брендбука */
--bb-sidebar-bg: ... /* Фон сайдбара */
--bb-sidebar-border: ... /* Граница сайдбара */
--bb-sidebar-text: ... /* Текст сайдбара */
--bb-sidebar-text-muted: ...
--bb-sidebar-section: .../* Заголовки секций сайдбара */
--bb-sidebar-active-bg: ...
--bb-text: ... /* Основной текст контента */
--bb-text-muted: ... /* Приглушённый текст */
--bb-border: ... /* Границы */
--bb-content-bg: ... /* Фон карточек */
--bb-sidebar-bg: #f8f9fa;
--bb-sidebar-border: #e5e7eb;
--bb-sidebar-text: #374151;
--bb-sidebar-text-muted: #6b7280;
--bb-sidebar-active-bg: #dff0fa; /* светло-синий под 053M */
--bb-sidebar-active-text: var(--brand-053m);
--bb-sidebar-section: #9ca3af;
--bb-content-bg: #ffffff;
--bb-border: #e5e7eb;
--bb-text: #111827;
--bb-text-muted: #6b7280;
```
---
@@ -234,8 +234,8 @@ SVG-версия ожидается (не получена от клиники).
### 6.1 Форма сотрудников
**Варианты:**
- Бежевый: ткань цвета Oracal 081M, логотип Oracal 080M (коричневый), расположение — левая сторона груди
- Синий: ткань цвета Oracal 050M (тёмно-синий), логотип белый (Oracal 010), расположение — левая сторона груди
- Бежевый: ткань цвета Oracal 081M (#a8885c), логотип Oracal 080M (#432f1e, коричневый), расположение — левая сторона груди
- Синий: ткань цвета Oracal 050M (#1b2e5d, тёмно-синий), логотип белый (Oracal 010), расположение — левая сторона груди
**Размеры логотипа на форме:**
- Размеры до 46: 70 × 25,5 мм
@@ -252,7 +252,7 @@ SVG-версия ожидается (не получена от клиники).
### 6.3 Внутренняя навигация
**Материал:** оргстекло
**Плёнки:** Oracal 053M (бирюзовый) и 073M (тёмный серо-голубой)
**Плёнки:** Oracal 053M (#0089c3) и 073M (#53514e)
**Типы табличек:**
- Таблички на дверях кабинетов: номер кабинета, профиль врача с фото и QR-кодом
- Указатели по этажам: стрелки направлений + номера кабинетов
@@ -265,9 +265,9 @@ SVG-версия ожидается (не получена от клиники).
### 6.4 Брендирование транспорта (трамвай)
**Зоны оклейки:**
- Борта: Oracal 053M (#7ecfca) + 073M (#5b7b87)
- Передняя часть: Oracal 066M (#5bb5ad) + 050M (#1b4c72)
- Акценты: Oracal 081M (#c4a882) + 080M (#5c2e0e)
- Борта: Oracal 053M (#0089c3) + 073M (#53514e)
- Передняя часть: Oracal 066M (#00818c) + 050M (#1b2e5d)
- Акценты: Oracal 081M (#a8885c) + 080M (#432f1e)
**Все 6 фирменных цветов присутствуют на транспорте.**
@@ -278,16 +278,17 @@ SVG-версия ожидается (не получена от клиники).
| URL | Статус | Описание |
|-----|--------|----------|
| `/foundation/logo` | ✅ Готова | Логотип, варианты, охранная зона, правила |
| `/foundation/colors` | ✅ Готова | Палитра, контраст WCAG, цвета сайта, соответствие |
| `/foundation/colors` | ✅ Готова | Палитра (Oracal точные RGB), контраст WCAG, цвета сайта, соответствие |
| `/foundation/typography` | ✅ Готова | DINPro + Fira Sans, шкала стилей |
| `/foundation/icons` | 🔜 Скоро | Иконография |
| `/offline/uniform` | ✅ Готова | Форма сотрудников |
| `/offline/badges` | ✅ Готова | Бейджи |
| `/offline/navigation` | ✅ Готова | Внутренняя навигация |
| `/offline/transport` | ✅ Готова | Брендирование транспорта |
| `/components/buttons` | 🔜 Sprint 3 | Кнопки |
| `/components/forms` | 🔜 Sprint 3 | Форм-контролы |
| `/components/*` | 🔜 Sprint 34 | Карточки, бейджи, алерты, модалки, таблицы |
| `/components/buttons` | ✅ Готова | Кнопки — 4 варианта с реального сайта, размеры, состояния |
| `/components/forms` | ✅ Готова | Форм-контролы — Input/Textarea/Select/Checkbox/Radio/Toggle |
| `/components/cards` | 🚧 Sprint 4 | Карточки — врач, услуга, новость, отзыв, цена + бейджи/теги/алерты |
| `/components/*` | 🔜 Sprint 4–5 | Модалки, таблицы, навигация |
| `/blocks/*` | 🔜 Sprint 58 | Hero, врачи, отзывы, новости, формы |
| `/pages/*` | 🔜 Sprint 9–11 | Главная, заболевание, врачи, цены, контакты |
@@ -299,8 +300,9 @@ SVG-версия ожидается (не получена от клиники).
- Использовать только цвета из фирменной палитры (раздел 2.1)
- Для digital: адаптировать к цветам сайта (раздел 2.2)
- Текст на цветном фоне — минимум WCAG AA (4.5:1)
- Белый логотип на тёмных фонах (073M, 050M, 080M)
- Коричневый логотип (080M) на бежевом фоне (081M)
- Белый текст на тёмных фонах: 073M (#53514e), 066M (#00818c), 050M (#1b2e5d), 080M (#432f1e)
- Тёмный текст (#111827) на светлых: 053M (#0089c3), 081M (#a8885c)
- Коричневый логотип (080M) на бежевом фоне (081M) — только крупный текст (3.8:1)
### Запрещено
- Использовать произвольные цвета вне палитры
@@ -311,13 +313,13 @@ SVG-версия ожидается (не получена от клиники).
### Иерархия цветов
```
Основной бирюзовый (053M / #7ecfca) ← главный идентификатор бренда
└── Средний бирюзовый (066M / #5bb5ad) ← вторичный акцент
└── Тёмный серо-голубой (073M / #5b7b87) ← фоны, хедер
Основной бирюзовый (053M / #0089c3) ← главный идентификатор бренда
└── Средний бирюзовый (066M / #00818c) ← вторичный акцент
└── Тёмный серо-голубой (073M / #53514e) ← фоны, хедер
Тёмно-синий (050M / #1b4c72) ← авторитет, полиграфия
Бежевый (081M / #c4a882) ← тепло, форма
└── Тёмно-коричневый (080M / #5c2e0e) ← текст на бежевом
Тёмно-синий (050M / #1b2e5d) ← авторитет, полиграфия
Бежевый (081M / #a8885c) ← тепло, форма
└── Тёмно-коричневый (080M / #432f1e) ← текст на бежевом
```
---
@@ -339,6 +341,77 @@ SVG-версия ожидается (не получена от клиники).
---
## 9a. Базовые компоненты (Sprint 3)
### Кнопки (Button) · `/components/buttons`
Варианты скопированы с реального сайта perm.oclinica.ru.
CSS-классы в `globals.css`. Компонент: `@/components/ui/Button` (React, "use client").
| Вариант | CSS класс | Фон | Текст | Граница | Border-radius | Применение |
|---------|------------------|----------|---------|-----------|---------------|------------|
| primary | .bb-btn-primary | #FFA39C | #fff | #FF847B | 7px, bold, shadow | Форма записи «Запишите меня!» |
| outline | .bb-btn-outline | #fff | #BF9975 | #BF9975 | 7px | «Записаться на приём» |
| teal | .bb-btn-teal | #60959C | #fff | прозрачная| 7px | «Позвонить» |
| pill | .bb-btn-pill | #E9E4D4 | #333 | #D5CFBD | 25px | «Заказать звонок» |
| Размер | CSS класс | padding | font-size |
|--------|------------|--------------|-----------|
| sm | .bb-btn-sm | 4px 11px | 13px |
| md | .bb-btn-md | 8px 16px | 14px |
| lg | .bb-btn-lg | 10px 24px | 18px |
**Состояния:** default · hover (brightness 0.9) · active (brightness 0.82) · loading (spinner) · disabled (opacity 0.5)
### Форм-контролы (Forms) · `/components/forms`
Размеры соответствуют entityform-блокам на реальном сайте.
| Элемент | CSS класс | Тег HTML | Высота | Описание |
|----------|------------------|----------------------------|---------|----------|
| Input | .bb-input | `<input>` | 50px | text, email, password · border 1px #ccc · radius 4px |
| Textarea | .bb-textarea | `<textarea>` | ≥100px | многострочный, resize:vertical |
| Select | .bb-select | `<select>` | 50px | с кастомной SVG-стрелкой |
| Checkbox | .bb-checkbox | `<input type="checkbox">` | 16×16px | accent-color: #53514e |
| Radio | .bb-radio | `<input type="radio">` | 16×16px | accent-color: #53514e |
| Toggle | .bb-toggle-track | React-компонент `<Toggle>` | 24px | 44×24px track + 20px thumb |
**Состояния полей:** default (border 1px solid #ccc) · focus (border #0089c3 + box-shadow rgba(0,137,195,0.2)) · error (.bb-error, border #dc2626) · disabled (opacity 0.5)
**Контекст на сайте:** фон формы #b8e6ed, ширина полей 302px, entityform-блоки.
**Toggle:** выкл → track #d1d5db · вкл → track #53514e · thumb: белый круг 20×20px.
---
## 9b. Карточки, бейджи и алерты (Sprint 4)
### Карточки · `/components/cards`
| Карточка | Размеры ключевые | Источник на сайте | Фон / hover |
|-------------|-------------------------|------------------------------------------------|--------------------------|
| DoctorCard | фото 110×160px | `.doctor .image` + `.doctor .item` | #fff / — |
| NewsCard | preview h=144px | `#block-views-last-news-block-1 .views-column` | #fff / **#eef4d1** + shadow |
| ReviewCard | 4-строчный clamp | `.node-reviews` | **#eef4d1** / — |
| PriceCard | flex row: name + price | `.field-name-field-price-priem` | #fff / highlighted: #f0f9ff |
| ServiceCard | иконка 48×48px | — (нет прямого аналога) | #fff / shadow |
**DoctorCard:** фото 110×160px (placeholder фон `#dff0fa`), кнопка `.bb-btn-outline .bb-btn-sm`
**NewsCard hover CSS:** `background: #eef4d1; box-shadow: 0 0 16px 0 #9e9e9a;` — взят с реального сайта
**ReviewCard:** звёзды SVG, заливка `#f59e0b`, `WebkitLineClamp: 4`
### Бейджи, теги, алерты
| Элемент | Варианты | Применение |
|---------|----------|------------|
| Badge | primary (#0089c3) / success (#059669) / warning (#d97706) / danger (#dc2626) / neutral (#6b7280) | Статус врача, категория, акция |
| Tag | default (border) / active (brand bg) | Фильтры, категории услуг |
| Alert | info (#dff0fa/#075985) / success (#d1fae5/#065f46) / warning (#fef3c7/#92400e) / error (#fee2e2/#991b1b) | Системные сообщения |
**CSS класс hover:** `.bb-news-card:hover` в globals.css
---
## 10. Технический стек проекта
| Слой | Технология | Версия |
@@ -361,6 +434,9 @@ SVG-версия ожидается (не получена от клиники).
| 1.0 | 2026-03-22 | Sprint 1: логотип, базовые цвета |
| 2.0 | 2026-03-22 | Sprint 2: типографика, оффлайн носители, цвета сайта (8 цветов) |
| 2.1 | 2026-03-22 | Sprint 2 доп.: +3 цвета сайта (коралловый, светло-жёлтый, светло-зелёный) |
| 3.0 | 2026-03-22 | Sprint 3: кнопки (Button), форм-контролы (Input/Textarea/Select/Checkbox/Radio/Toggle), LLM-блок на логотипе |
| 4.0 | 2026-03-22 | Sprint 4 start: исправлены цвета Oracal (точные RGB из каталога), кнопки/формы по реальному сайту |
| 4.1 | 2026-03-22 | Sprint 4 done: карточки (DoctorCard/NewsCard/ReviewCard/PriceCard/ServiceCard), бейджи/теги/алерты |
---
+43 -24
View File
@@ -107,48 +107,67 @@
---
## Sprint 3 — Базовые компоненты: кнопки и поля ввода
## Sprint 3 — Базовые компоненты: кнопки и поля ввода ✅ ЗАВЕРШЁН
**Цель:** Все варианты кнопок и форм-контролов в брендбуке. LLM-блоки на страницах.
### Задачи — LLM-контекст
- [ ] FE: Добавить LLM-блок на страницу «Логотип» (`/foundation/logo`)
- [x] FE: Добавить LLM-блок на страницу «Логотип» (`/foundation/logo`) — v1.0
- [x] FE: Добавить LLM-блок на страницу «Цвета» (`/foundation/colors`) — v2.1
- [x] FE: Добавить LLM-блок на страницу «Типографика» (`/foundation/typography`) — v2.0
- [x] FE: Создать переиспользуемый компонент `components/llm/LlmBlock.tsx` (LlmBlock, LlmSection, LlmTable, LlmRules)
- [ ] Docs: Обновить `docs/LLM_CONTEXT.md` по итогам спринта (версия 3.x)
- [x] Docs: Обновить `docs/LLM_CONTEXT.md` по итогам спринта версия 3.0
### Задачи — компоненты
- [ ] FE: Компонент Button (все варианты: primary/secondary/ghost/danger, размеры, состояния)
- [ ] FE: Компонент Input (text, password, focus/error/disabled)
- [ ] FE: Компонент Textarea
- [ ] FE: Компонент Select
- [ ] FE: Компонент Checkbox и Radio
- [ ] FE: Компонент Toggle/Switch
- [ ] FE: Страница брендбука «Компоненты → Кнопки» с документацией + LLM-блок
- [ ] FE: Страница брендбука «Компоненты → Форм-контролы» + LLM-блок
- [ ] FE: Копирование HTML/CSS кода компонента в один клик
- [x] FE: Компонент Button (`components/ui/Button.tsx`) — варианты primary/outline/teal/pill, размеры sm/md/lg, loading
- [x] FE: Компонент Input (`.bb-input`) — text, password, focus/error/disabled, height 50px как на сайте
- [x] FE: Компонент Textarea (`.bb-textarea`) — resize:vertical, min-height 100px
- [x] FE: Компонент Select (`.bb-select`) — height 50px, кастомная стрелка SVG
- [x] FE: Компонент Checkbox (`.bb-checkbox`) и Radio (`.bb-radio`)
- [x] FE: Компонент Toggle/Switch (`components/ui/Toggle.tsx`) — React "use client", defaultChecked/disabled/label
- [x] FE: Страница `/components/buttons` — 4 варианта по реальному сайту, размеры, состояния, «Где применяется», LLM-блок v2.0
- [x] FE: Страница `/components/forms` — все 6 контролов, контекст на сайте с макетом, LLM-блок v2.0
- [x] FE: Копирование HTML/CSS кода в один клик (`components/ui/CodeCopy.tsx`)
**Результат спринта:** Раздел «Базовые компоненты — кнопки и ввод» готов. LLM-блоки добавлены на страницы Фундамента.
### Фактические результаты
- **4 варианта кнопок** — скопированы с реального сайта perm.oclinica.ru: primary(#FFA39C), outline(#BF9975), teal(#60959c), pill(#e9e4d4)
- **6 форм-контролов** — input/textarea/select/checkbox/radio/toggle с полной документацией состояний
- **Input/Select** — height 50px, border 1px solid #ccc, border-radius 4px (entityform CSS с реального сайта)
- **Макет формы** на фоне #b8e6ed как «Узнайте стоимость операции» на oclinica.ru/lor
- **LLM-блоки** добавлены на логотип, цвета, типографику, кнопки, форм-контролы
- **Компоненты:** Button.tsx, Toggle.tsx, CodeCopy.tsx в `components/ui/`
- **Sidebar:** Sprint 3 · v0.3.0, кнопки/формы убраны из «скоро»
- **Деплой:** https://web-oclinica.vercel.app (production)
**Результат спринта:** Раздел «Базовые компоненты» полностью готов. Стили соответствуют реальному сайту.
---
## Sprint 4 — Карточки (Cards)
## Sprint 4 — Карточки (Cards) ✅ ЗАВЕРШЁН
**Цель:** Все типы карточек, используемых на сайте.
### Задачи
- [ ] Docs: Обновить `docs/LLM_CONTEXT.md` — добавить спецификации карточек
- [ ] FE: Добавить LLM-блок на страницу «Карточки»
- [ ] FE: Карточка врача (DoctorCard) — фото, имя, специализация, кнопка записи
- [ ] FE: Карточка услуги / заболевания
- [ ] FE: Карточка новости — превью, дата, заголовок, анонс, читать далее
- [ ] FE: Карточка отзыва — автор, текст, рейтинг (звёзды), дата
- [ ] FE: Карточка цены — услуга, стоимость, описание
- [ ] FE: Бейджи, теги, алерты (inline и toast)
- [ ] FE: Страница «Компоненты → Карточки» с документацией
- [x] Docs: Обновить `docs/LLM_CONTEXT.md` → версия 4.0
- [x] FE: Добавить LLM-блок на страницу «Карточки» — v1.0
- [x] FE: Карточка врача (DoctorCard) — фото 110×160px, имя, специализация, стаж, кнопка outline
- [x] FE: Карточка услуги / заболевания (ServiceCard) — иконка + заголовок + описание
- [x] FE: Карточка новости — превью, дата, заголовок, анонс, hover #eef4d1 (CSS сайта)
- [x] FE: Карточка отзыва — рейтинг SVG-звёзды, 4-строчный clamp, фон #eef4d1
- [x] FE: Карточка цены — услуга + стоимость + highlighted-вариант
- [x] FE: Бейджи (6 вариантов), теги (default/active), алерты (info/success/warning/error)
- [x] FE: Страница `/components/cards` с документацией и LLM-блоком
**Результат спринта:** Все карточки задокументированы и показаны в брендбуке.
### Фактические результаты
- **5 типов карточек:** DoctorCard, NewsCard, ReviewCard, PriceCard, ServiceCard
- **NewsCard hover** — bg `#eef4d1` + box-shadow `0 0 16px #9e9e9a` (1:1 с реальным CSS сайта)
- **ReviewCard** — фон `#eef4d1` из CSS сайта, WebkitLineClamp: 4
- **Бейджи** — 6 вариантов (primary/success/warning/danger/neutral/outline-blue)
- **Алерты** — info (#dff0fa), success (#d1fae5), warning (#fef3c7), error (#fee2e2)
- **Исправлены цвета Oracal** — точные RGB из каталога для всех 6 плёнок
- **Деплой:** https://web-oclinica.vercel.app (production)
**Результат спринта:** Все карточки задокументированы. Цвета Oracal исправлены по каталогу.
---