Files
sfera-new/docs/presentation-layer/CSS_LAYOUT_SCROLL_RULES.md
Veronika Smirnova 121a4dece1 docs: создать правила для синхронизации данных, layout и статистических компонентов
- DATA_SYNCHRONIZATION_RULES.md - правила синхронизации между компонентами
- GRAPHQL_CACHE_RULES.md - настройки кеширования и fetchPolicy
- CSS_LAYOUT_SCROLL_RULES.md - решение проблем с overflow и scroll
- STATISTICAL_COMPONENTS_RULES.md - правила Master-Detail архитектуры

Документация основана на исправлениях в кабинете фулфилмента

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-08-27 12:29:00 +03:00

14 KiB
Raw Blame History

🎨 ПРАВИЛА CSS LAYOUT И СКРОЛЛА

Цель: Предотвратить проблемы с overflow, scroll и позиционированием в Next.js 15 + React 19 приложении

📋 ОСНОВНЫЕ ПРИНЦИПЫ LAYOUT

1. БАЗОВАЯ АРХИТЕКТУРА LAYOUT

// ✅ ПРАВИЛЬНАЯ структура для всех страниц приложения
<div className="h-screen flex overflow-hidden">  {/* Контейнер полной высоты */}
  <Sidebar />                                    {/* Фиксированная боковая панель */}
  <main className={`
    flex-1                                       /* Занимает оставшееся место */
    ${getSidebarMargin()}                        /* Динамический отступ */
    px-4 py-3                                    /* Внутренние отступы */
    flex flex-col                                /* Вертикальная компоновка */
    transition-all duration-300                  /* Плавные анимации */
    overflow-hidden                              /* Предотвращение двойного скролла */
  `}>
    <div className="flex-1 overflow-y-auto space-y-6">  {/* ЕДИНСТВЕННАЯ зона скролла */}
      {/* Весь контент здесь */}
    </div>
  </main>
</div>

2. КРИТИЧЕСКИЕ ПРАВИЛА OVERFLOW

ПРАВИЛЬНО: Один уровень overflow

/* Родительский контейнер */
.container {
  height: 100vh;
  overflow: hidden; /* Запрещаем скролл на уровне экрана */
}

/* Дочерний скроллируемый контейнер */
.scrollable-content {
  flex: 1;
  overflow-y: auto; /* ЕДИНСТВЕННАЯ зона скролла */
}

НЕПРАВИЛЬНО: Множественные overflow зоны

/* Создаёт конфликты скролла */
.parent {
  overflow-y: auto; /* Первый скролл */
}
.child {
  overflow-y: auto; /* Второй скролл - ПРОБЛЕМА! */
}

🚨 ТИПИЧНЫЕ ПРОБЛЕМЫ И РЕШЕНИЯ

ПРОБЛЕМА 1: Контент смещается вправо

Причина:

<div style={{ minHeight: '200vh' }}>
  {' '}
  {/* Принудительная высота */}
  <div className="overflow-hidden">
    {' '}
    {/* Скрывает контент */}
    {/* Контент не помещается и смещается */}
  </div>
</div>

Решение:

<div className="space-y-6">
  {' '}
  {/* Простая вертикальная компоновка */}
  {/* Контент автоматически размещается правильно */}
</div>

ПРОБЛЕМА 2: Скролл не работает

Причина:

<div className="h-screen overflow-hidden">
  {' '}
  {/* Блокирует скролл */}
  <div className="overflow-hidden">
    {' '}
    {/* Дублирует блокировку */}
    <div className="min-h-[200vh]">
      {' '}
      {/* Создаёт контент больше экрана */}
      {/* Контент не скроллится */}
    </div>
  </div>
</div>

Решение:

<div className="h-screen flex flex-col overflow-hidden">
  <div className="flex-1 overflow-y-auto">
    {' '}
    {/* ЕДИНСТВЕННАЯ зона скролла */}
    {/* Контент свободно скроллится */}
  </div>
</div>

ПРОБЛЕМА 3: Двойной sidebar в layout

Причина:

// В page.tsx
<div className="flex">
  <Sidebar /> {/* Первый sidebar */}
  <main>
    <div className="flex">
      <Sidebar /> {/* Второй sidebar - ДУБЛИРОВАНИЕ! */}
      <content />
    </div>
  </main>
</div>

Решение:

// В page.tsx - УБРАТЬ дублирование
<div className="h-screen flex overflow-hidden">
  <Sidebar /> {/* ЕДИНСТВЕННЫЙ sidebar */}
  <main className={`flex-1 ${getSidebarMargin()} overflow-hidden`}>
    <div className="flex-1 overflow-y-auto space-y-6">{/* Весь контент */}</div>
  </main>
</div>

🏗️ АРХИТЕКТУРНЫЕ ПАТТЕРНЫ

ПАТТЕРН 1: Dashboard с статистикой

export function DashboardPage() {
  return (
    <div className="h-screen flex overflow-hidden">
      <Sidebar />
      <main className={`flex-1 ${getSidebarMargin()} px-4 py-3 flex flex-col overflow-hidden`}>
        <div className="flex-1 overflow-y-auto space-y-6">
          {/* Заголовок */}
          <Header />

          {/* Статистические карточки */}
          <StatsCards />

          {/* Основное содержимое */}
          <MainContent />
        </div>
      </main>
    </div>
  )
}

ПАТТЕРН 2: Таблица с фильтрами

export function TablePage() {
  return (
    <div className="h-screen flex overflow-hidden">
      <Sidebar />
      <main className={`flex-1 ${getSidebarMargin()} px-4 py-3 flex flex-col overflow-hidden`}>
        <div className="flex-1 overflow-y-auto space-y-6">
          {/* Фильтры (фиксированные) */}
          <FiltersPanel />

          {/* Таблица (скроллируемая) */}
          <div className="flex-1 overflow-y-auto">
            <DataTable />
          </div>
        </div>
      </main>
    </div>
  )
}

ПАТТЕРН 3: Модальные окна

// ✅ Модалки не должны влиять на основной скролл
<Dialog>
  <div className="max-h-[80vh] overflow-y-auto">
    {' '}
    {/* Скролл только внутри модалки */}
    <ModalContent />
  </div>
</Dialog>

📱 RESPONSIVE DESIGN ПРАВИЛА

АДАПТИВНЫЕ КОНТЕЙНЕРЫ

<div
  className="
  grid 
  grid-cols-1           /* Мобильные: 1 колонка */
  md:grid-cols-2        /* Планшеты: 2 колонки */
  lg:grid-cols-4        /* Десктоп: 4 колонки */
  xl:grid-cols-6        /* Большие экраны: 6 колонок */
  gap-4 
"
>
  {/* Карточки адаптивно размещаются */}
</div>

АДАПТИВНЫЕ ОТСТУПЫ

<main className={`
  flex-1
  ${getSidebarMargin()}         /* Динамический отступ для sidebar */
  px-4 py-3                     /* Базовые отступы */
  lg:px-6 lg:py-4              /* Увеличенные отступы на больших экранах */
  overflow-hidden
`}>

🎯 СПЕЦИФИЧЕСКИЕ ПРАВИЛА ДЛЯ SFERA

ПРАВИЛА ДЛЯ ФУЛФИЛМЕНТ КОМПОНЕНТОВ

1. Главная страница склада

// src/app/fulfillment-warehouse/page.tsx
<div className="h-screen flex overflow-hidden">
  <Sidebar />
  <main className={`flex-1 ${getSidebarMargin()} px-4 py-3 flex flex-col overflow-hidden`}>
    <div className="flex-1 overflow-y-auto space-y-6">
      <WarehouseHeader />
      <WarehouseStats /> {/* Статистика - НЕ скроллится */}
      <WarehouseContent /> {/* Контент - скроллится */}
    </div>
  </main>
</div>

2. Таблицы поставок

// src/components/fulfillment-warehouse/fulfillment-supplies-page.tsx
<div className="h-screen flex overflow-hidden">
  <Sidebar />
  <main className={`flex-1 ${getSidebarMargin()} px-4 py-3 flex flex-col overflow-hidden`}>
    <div className="flex-1 overflow-y-auto space-y-6">
      {' '}
      {/* ← КЛЮЧЕВОЙ ЭЛЕМЕНТ */}
      <SuppliesHeader /> {/* Фильтры */}
      <SuppliesStats /> {/* Статистика */}
      <SuppliesList /> {/* Основная таблица - скроллируется естественно */}
    </div>
  </main>
</div>

3. Раскрывающиеся детали (Master-Detail)

// Детали поставок в раскрывающихся строках
<tr className="border-t border-white/10">
  <td colSpan="100%" className="p-0">
    <div className="max-h-96 overflow-y-auto bg-white/5">
      {' '}
      {/* Локальный скролл */}
      <DeliveryDetails />
    </div>
  </td>
</tr>

ПРОИЗВОДИТЕЛЬНОСТЬ И ОПТИМИЗАЦИЯ

ВИРТУАЛИЗАЦИЯ ДЛЯ БОЛЬШИХ СПИСКОВ

// Для таблиц с >100 строк
import { FixedSizeList as List } from 'react-window'

;<div className="h-96 overflow-hidden">
  {' '}
  {/* Контейнер фиксированной высоты */}
  <List
    height={384} // Высота контейнера
    itemCount={items.length} // Количество элементов
    itemSize={64} // Высота каждого элемента
    className="scrollbar-thin" // Кастомный скроллбар
  >
    {({ index, style }) => (
      <div style={style}>
        <TableRow item={items[index]} />
      </div>
    )}
  </List>
</div>

LAZY LOADING ДЛЯ КОНТЕНТА

const LazyTableSection = lazy(() => import('./TableSection'))

// В компоненте
<div className="flex-1 overflow-y-auto space-y-6">
  <StatsSection />  {/* Загружается сразу */}
  <Suspense fallback={<div>Загрузка...</div>}>
    <LazyTableSection />  {/* Загружается по требованию */}
  </Suspense>
</div>

🚨 АНТИ-ПАТТЕРНЫ И ЗАПРЕТЫ

НИКОГДА НЕ ДЕЛАЙТЕ:

1. Принудительные размеры

// ❌ Создаёт проблемы с layout
<div style={{ minHeight: '200vh' }}>
<div className="h-[2000px]">

2. Множественные overflow зоны

// ❌ Конфликты скролла
<div className="overflow-y-auto">
  <div className="overflow-y-auto">
    <div className="overflow-y-auto">

3. Смешивание fixed и sticky позиционирования

// ❌ Непредсказуемое поведение
<div className="fixed top-0">
  <div className="sticky top-0">

4. Игнорирование responsive дизайна

// ❌ Не адаптируется к мобильным устройствам
<div className="grid grid-cols-6 gap-4">  {/* Сломается на мобильных */}

🔧 ИНСТРУМЕНТЫ ОТЛАДКИ

CSS DEBUG КЛАССЫ

/* Добавьте для визуализации проблем */
.debug-borders * {
  border: 1px solid red !important;
}

.debug-overflow {
  overflow: visible !important;
  background: rgba(255, 0, 0, 0.1) !important;
}

.debug-scroll {
  scrollbar-color: red transparent !important;
  scrollbar-width: thick !important;
}

REACT DevTools

// Добавьте data-атрибуты для отладки
<div
  className="overflow-y-auto"
  data-scroll-zone="main-content"
  data-debug="scroll-container"
>

КОНСОЛЬНЫЕ ЛОГИ ДЛЯ РАЗМЕРОВ

useEffect(() => {
  const element = ref.current
  if (element) {
    console.log('LAYOUT DEBUG:', {
      scrollHeight: element.scrollHeight,
      clientHeight: element.clientHeight,
      offsetHeight: element.offsetHeight,
      hasOverflow: element.scrollHeight > element.clientHeight,
    })
  }
}, [])

📊 ЧЕКЛИСТ ПРОВЕРКИ LAYOUT

Перед релизом:

  • Единственная зона скролла на странице
  • Нет принудительных высот (minHeight: '200vh')
  • Правильная структура с h-screen и overflow-hidden
  • Responsive дизайн для мобильных устройств
  • Sidebar не дублируется в компонентах
  • Модальные окна не влияют на основной скролл

При проблемах со скроллом:

  • Проверить количество overflow-y-auto в иерархии
  • Убедиться в отсутствии overflow-hidden на скроллируемом контейнере
  • Проверить наличие flex-1 у родительского контейнера
  • Убрать принудительные размеры (min-height, height: 200vh)

🎨 КАСТОМНЫЕ СКРОЛЛБАРЫ

TAILWIND CLASSES

<div className="
  overflow-y-auto
  scrollbar-thin          /* Тонкий скроллбар */
  scrollbar-track-transparent
  scrollbar-thumb-white/20
  hover:scrollbar-thumb-white/30
">

CUSTOM CSS

/* Кастомные скроллбары для SFERA */
.custom-scrollbar::-webkit-scrollbar {
  width: 8px;
}

.custom-scrollbar::-webkit-scrollbar-track {
  background: transparent;
}

.custom-scrollbar::-webkit-scrollbar-thumb {
  background: rgba(255, 255, 255, 0.2);
  border-radius: 4px;
}

.custom-scrollbar::-webkit-scrollbar-thumb:hover {
  background: rgba(255, 255, 255, 0.3);
}

Следование этим правилам обеспечит корректную работу layout и скролла во всех компонентах! 🚀