Browse Source

Mobile UI sprint 1: actions-bar, version cards, meta line, safe area

- Add docs/СПРИНТЫ_МОБИЛЬНЫЙ_ДИЗАЙН.md and DOC/ШАГИ/ШАГ_2026-04-27_002.md
- cabinet-theme: .actions-bar, .version-card-list, .list-row__meta-tail,
  .inline-actions--block-mobile, btn--sm/ghost tweaks, safe-area main padding
- TestDetail: replace version table with cards; command panel uses actions-bar;
  assign list only when there are people; publication full-width on narrow
- TestsList: version suffix in non-breaking tail span

Made-with: Cursor
dev
Константин Лебединский 2 weeks ago
parent
commit
5db12c2348
  1. 5
      DOC/ШАГИ/ШАГ_2026-04-27_002.md
  2. 40
      docs/СПРИНТЫ_МОБИЛЬНЫЙ_ДИЗАЙН.md
  3. 71
      frontend/src/pages/TestDetail.jsx
  4. 12
      frontend/src/pages/TestsList.jsx
  5. 125
      frontend/src/styles/cabinet-theme.css

5
DOC/ШАГИ/ШАГ_2026-04-27_002.md

@ -0,0 +1,5 @@
# Шаг 2026-04-27 — спринты мобильного UI и правки
- Документ спринтов: [`docs/СПРИНТЫ_МОБИЛЬНЫЙ_ДИЗАЙН.md`](../../docs/СПРИНТЫ_МОБИЛЬНЫЙ_ДИЗАЙН.md) (спринт 1 выполнен в коде).
- Стили: `actions-bar`, `version-card-list`, `list-row__meta-tail`, `inline-actions--block-mobile`, safe-area у `.cabinet-main`, `.btn--sm` / `.btn-ghost`, `assign-list` без пустой «коробки`.
- Страницы: `TestDetail.jsx` (карточки версий, панель команд, назначение), `TestsList.jsx` (мета-строка).

40
docs/СПРИНТЫ_МОБИЛЬНЫЙ_ДИЗАЙН.md

@ -0,0 +1,40 @@
# Спринты: мобильный UI кабинета тестов
Рядом с: [`ПРЕДЛОЖЕНИЕ_ДИЗАЙН_СОЗДАНИЕ_ТЕСТА.md`](./ПРЕДЛОЖЕНИЕ_ДИЗАЙН_СОЗДАНИЕ_ТЕСТА.md).
---
## Спринт 1 — быстрые исправления (текущий)
**Цель:** выровнять кнопки, мета-строку списка, историю версий, назначение и safe-area; без смены контентной модели страниц.
| # | Задача | Статус |
|---|--------|--------|
| 1.1 | Панель «Сохранить черновик / К списку»: убрать конфликт `inline-actions .btn { width: auto }` с `btn-primary` — колонка на всю ширину (`.actions-bar`) | done |
| 1.2 | Touch: `min-height` у `.btn--sm` (убрать, удалить вопрос, сделать активной…) | done |
| 1.3 | Список тестов: не разбивать «· v1» — хвост в `list-row__meta-tail` + `white-space: nowrap` | done |
| 1.4 | «История версий»: вместо `<table>` — карточки (`surface-card` + flex) | done |
| 1.5 | «Назначение»: не рендерить пустой `.assign-list` (убрать «коробку» без людей) | done |
| 1.6 | Сильнее рамка `.btn-ghost` (согласование с полями) | done |
| 1.7 | `padding-bottom` у `.cabinet-main` + `env(safe-area-inset-bottom)` | done |
| 1.8 | «Публикация»: на узком экране — кнопка на всю ширину (`.inline-actions--block-mobile`) | done |
**Файлы:** `frontend/src/styles/cabinet-theme.css`, `frontend/src/pages/TestDetail.jsx`, `frontend/src/pages/TestsList.jsx`.
---
## Спринт 2 — бэклог (следующий)
| # | Задача |
|---|--------|
| 2.1 | «Прогоны и разбор»: на мобилке заменить таблицу на карточки или гориз. скролл с фиксированными колонками |
| 2.2 | «Импорт из файла»: кастомная кнопка (скрытый `input` + стилизованный `label` под `.btn`) |
| 2.3 | «Вопрос 1» + «Сгенерировать вопрос (ИИ)» — не в одной строке на узком экране; явная иерархия primary/secondary |
| 2.4 | Радио vs чекбокс у вариантов ответа при «несколько верных» — визуальная метафора (квадраты vs круги) |
| 2.5 | Закреплённый футер с действиями «Сохранить» (опционально) |
---
## Спринт 3 — дизайн-токены (по желанию)
Единая шкала: `--control-height`, `--control-padding-x`, `--button-gap` — рефакторинг всех `inline-actions` и форм.

71
frontend/src/pages/TestDetail.jsx

@ -938,8 +938,8 @@ export default function TestDetail() {
<div className="cabinet-brick" style={{ marginTop: 0 }}> <div className="cabinet-brick" style={{ marginTop: 0 }}>
<div <div
className="inline-actions" className="actions-bar"
style={{ flexWrap: 'wrap', gap: '0.5rem', marginBottom: draftStatus ? '0.35rem' : 0 }} style={{ marginBottom: draftStatus ? '0.35rem' : 0 }}
> >
<button type="button" className="btn btn-primary" onClick={saveDraft}> <button type="button" className="btn btn-primary" onClick={saveDraft}>
Сохранить черновик Сохранить черновик
@ -955,48 +955,41 @@ export default function TestDetail() {
)} )}
</div> </div>
<AccSection title="История версий" defaultOpen={false}> <AccSection title="История версий" defaultOpen={false}>
<div className="surface-card" style={{ padding: 0, overflow: 'hidden' }}> <ul className="version-card-list" aria-label="Список версий теста">
<table className="table-cabinet">
<thead>
<tr>
<th>Версия</th>
<th>Активна</th>
<th>Создана</th>
<th />
</tr>
</thead>
<tbody>
{versions.map((r) => ( {versions.map((r) => (
<tr key={r.id}> <li key={r.id} className="surface-card version-card-list__item">
<td> <div className="version-card-list__row">
<div className="version-card-list__main">
<div className="version-card-list__title-line">
<span className="font-headline" style={{ fontSize: '1rem' }}>
v{r.version} v{r.version}
</span>
{r.is_active && ( {r.is_active && (
<span <span className="code-inline" style={{ fontSize: '0.7rem' }}>
className="code-inline"
style={{ marginLeft: '0.5rem', fontSize: '0.7rem' }}
>
текущая текущая
</span> </span>
)} )}
</td> </div>
<td>{r.is_active ? 'да' : 'нет'}</td> <p className="muted mono" style={{ margin: '0.4rem 0 0', fontSize: '0.8rem' }}>
<td className="mono">{fmtDt(r.created_at)}</td> {fmtDt(r.created_at)}
<td> </p>
<p className="muted" style={{ margin: '0.2rem 0 0', fontSize: '0.8rem' }}>
Активна: {r.is_active ? 'да' : 'нет'}
</p>
</div>
{!r.is_active && ( {!r.is_active && (
<button <button
type="button" type="button"
className="btn btn-ghost btn--sm" className="btn btn-ghost btn--sm version-card-list__action"
onClick={() => activateVersion(r.id)} onClick={() => activateVersion(r.id)}
> >
сделать активной Сделать активной
</button> </button>
)} )}
</td>
</tr>
))}
</tbody>
</table>
</div> </div>
</li>
))}
</ul>
</AccSection> </AccSection>
{attemptsList != null && attemptsList.length > 0 && ( {attemptsList != null && attemptsList.length > 0 && (
@ -1065,7 +1058,10 @@ export default function TestDetail() {
)} )}
<AccSection title="Публикация (видимость в списке)" defaultOpen={false}> <AccSection title="Публикация (видимость в списке)" defaultOpen={false}>
<div className="inline-actions" style={{ marginTop: '0.5rem' }}> <div
className="inline-actions inline-actions--block-mobile"
style={{ marginTop: '0.5rem' }}
>
{test?.chainActive !== false ? ( {test?.chainActive !== false ? (
<button <button
type="button" type="button"
@ -1191,12 +1187,8 @@ export default function TestDetail() {
</select> </select>
{assignLoadBusy && <span className="text-muted">Загрузка</span>} {assignLoadBusy && <span className="text-muted">Загрузка</span>}
</div> </div>
{assignPeople.length > 0 ? (
<div className="assign-list" role="group" aria-label="Список сотрудников"> <div className="assign-list" role="group" aria-label="Список сотрудников">
{assignPeople.length === 0 && !assignLoadBusy && (
<p className="text-muted" style={{ margin: '0.5rem 0' }}>
Нет подходящих записей. Смените фильтры или поиск.
</p>
)}
{assignPeople.map((p) => { {assignPeople.map((p) => {
const k = assignPersonKey(p); const k = assignPersonKey(p);
const picked = assignSelected.has(k); const picked = assignSelected.has(k);
@ -1226,6 +1218,13 @@ export default function TestDetail() {
); );
})} })}
</div> </div>
) : (
!assignLoadBusy && (
<p className="text-muted" style={{ margin: '0.5rem 0' }}>
Нет подходящих записей. Смените фильтры или поиск.
</p>
)
)}
<div className="inline-actions" style={{ marginTop: '0.75rem' }}> <div className="inline-actions" style={{ marginTop: '0.75rem' }}>
<button <button
type="button" type="button"

12
frontend/src/pages/TestsList.jsx

@ -106,11 +106,7 @@ export default function TestsList() {
<span className="list-row__title">{t.title}</span> <span className="list-row__title">{t.title}</span>
<span className="list-row__meta"> <span className="list-row__meta">
{formatTestAuthorLabel(user, t.created_by, t.author_full_name)} {formatTestAuthorLabel(user, t.created_by, t.author_full_name)}
<span className="list-row__meta-sep" aria-hidden> <span className="list-row__meta-tail">{' · '}v{t.version}</span>
{' '}
·{' '}
</span>
v{t.version}
</span> </span>
</Link> </Link>
</div> </div>
@ -151,11 +147,9 @@ export default function TestsList() {
<span className="list-row__title">{t.title}</span> <span className="list-row__title">{t.title}</span>
<span className="list-row__meta"> <span className="list-row__meta">
{formatTestAuthorLabel(user, t.created_by, t.author_full_name)} {formatTestAuthorLabel(user, t.created_by, t.author_full_name)}
<span className="list-row__meta-sep" aria-hidden> <span className="list-row__meta-tail">
{' '} {' · '}v{t.version} · скрыт
·{' '}
</span> </span>
v{t.version} · скрыт
</span> </span>
</Link> </Link>
</div> </div>

125
frontend/src/styles/cabinet-theme.css

@ -248,7 +248,7 @@ code,
.btn-ghost { .btn-ghost {
background: transparent; background: transparent;
color: var(--primary); color: var(--primary);
border-color: color-mix(in srgb, var(--outline-variant) 50%, transparent); border-color: color-mix(in srgb, var(--outline-variant) 70%, transparent);
} }
.btn-ghost:hover { .btn-ghost:hover {
@ -264,8 +264,11 @@ code,
.btn--sm { .btn--sm {
font-size: 0.8rem; font-size: 0.8rem;
padding: 0.35rem 0.6rem; padding: 0.5rem 0.75rem;
border-radius: 0.5rem; border-radius: 0.5rem;
min-height: 2.75rem;
min-width: 2.75rem;
box-sizing: border-box;
} }
/* --- App shell (cabinet/base) --- */ /* --- App shell (cabinet/base) --- */
@ -378,7 +381,7 @@ code,
max-width: var(--max-content); max-width: var(--max-content);
width: 100%; width: 100%;
margin: 0 auto; margin: 0 auto;
padding: 1.25rem 1.25rem 2.5rem; padding: 1.25rem 1.25rem calc(2.5rem + env(safe-area-inset-bottom, 0px));
} }
/* Cards & lists */ /* Cards & lists */
@ -431,6 +434,11 @@ code,
margin-top: 0.25rem; margin-top: 0.25rem;
} }
/* «· v1» и хвост мета — не рвём посередине (мобайл) */
.list-row__meta-tail {
white-space: nowrap;
}
/* Вся плитка — одна ссылка */ /* Вся плитка — одна ссылка */
.list-row--action { .list-row--action {
padding: 0; padding: 0;
@ -609,7 +617,8 @@ code,
} }
.assign-list { .assign-list {
max-height: min(50vh, 22rem); max-height: min(40vh, 18rem);
min-height: 0;
overflow: auto; overflow: auto;
border: 1px solid color-mix(in srgb, var(--outline-variant) 30%, transparent); border: 1px solid color-mix(in srgb, var(--outline-variant) 30%, transparent);
border-radius: 0.75rem; border-radius: 0.75rem;
@ -755,6 +764,57 @@ code,
color: var(--on-surface-variant); color: var(--on-surface-variant);
} }
/* История версий: карточки вместо таблицы (мобайл) */
.version-card-list {
list-style: none;
margin: 0;
padding: 0;
display: flex;
flex-direction: column;
gap: 0.5rem;
}
.version-card-list__item {
margin: 0;
}
.version-card-list__row {
display: flex;
flex-wrap: wrap;
align-items: flex-start;
justify-content: space-between;
gap: 0.75rem;
}
.version-card-list__main {
min-width: 0;
flex: 1 1 12rem;
}
.version-card-list__title-line {
display: flex;
flex-wrap: wrap;
align-items: center;
gap: 0.5rem;
}
.version-card-list__action {
flex: 0 0 auto;
align-self: center;
}
@media (max-width: 520px) {
.version-card-list__row {
flex-direction: column;
align-items: stretch;
}
.version-card-list__action {
width: 100%;
align-self: stretch;
}
}
.draft-block { .draft-block {
margin-top: 1.25rem; margin-top: 1.25rem;
padding: 1rem; padding: 1rem;
@ -783,3 +843,60 @@ code,
.inline-actions .btn { .inline-actions .btn {
width: auto; width: auto;
} }
/* Нижняя панель: полноширинные primary + secondary (без перебития .inline-actions .btn) */
.actions-bar {
display: flex;
flex-direction: column;
align-items: stretch;
gap: 0.5rem;
width: 100%;
box-sizing: border-box;
}
.actions-bar .btn-primary {
width: 100%;
margin-top: 0;
box-sizing: border-box;
}
.actions-bar a.btn,
.actions-bar .btn.btn-ghost {
display: block;
width: 100%;
text-align: center;
box-sizing: border-box;
}
@media (min-width: 480px) {
.actions-bar {
flex-direction: row;
flex-wrap: wrap;
align-items: center;
}
.actions-bar .btn-primary {
width: auto;
min-width: 12rem;
flex: 1 1 auto;
}
.actions-bar a.btn,
.actions-bar .btn.btn-ghost {
display: inline-block;
width: auto;
flex: 0 0 auto;
}
}
@media (max-width: 520px) {
.inline-actions--block-mobile {
flex-direction: column;
align-items: stretch;
}
.inline-actions--block-mobile .btn {
width: 100%;
box-sizing: border-box;
}
}

Loading…
Cancel
Save