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
This commit is contained in:
@@ -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` (мета-строка).
|
||||
@@ -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` и форм.
|
||||
@@ -938,8 +938,8 @@ export default function TestDetail() {
|
||||
|
||||
<div className="cabinet-brick" style={{ marginTop: 0 }}>
|
||||
<div
|
||||
className="inline-actions"
|
||||
style={{ flexWrap: 'wrap', gap: '0.5rem', marginBottom: draftStatus ? '0.35rem' : 0 }}
|
||||
className="actions-bar"
|
||||
style={{ marginBottom: draftStatus ? '0.35rem' : 0 }}
|
||||
>
|
||||
<button type="button" className="btn btn-primary" onClick={saveDraft}>
|
||||
Сохранить черновик
|
||||
@@ -955,48 +955,41 @@ export default function TestDetail() {
|
||||
)}
|
||||
</div>
|
||||
<AccSection title="История версий" defaultOpen={false}>
|
||||
<div className="surface-card" style={{ padding: 0, overflow: 'hidden' }}>
|
||||
<table className="table-cabinet">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Версия</th>
|
||||
<th>Активна</th>
|
||||
<th>Создана</th>
|
||||
<th />
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{versions.map((r) => (
|
||||
<tr key={r.id}>
|
||||
<td>
|
||||
v{r.version}
|
||||
{r.is_active && (
|
||||
<span
|
||||
className="code-inline"
|
||||
style={{ marginLeft: '0.5rem', fontSize: '0.7rem' }}
|
||||
>
|
||||
текущая
|
||||
<ul className="version-card-list" aria-label="Список версий теста">
|
||||
{versions.map((r) => (
|
||||
<li key={r.id} className="surface-card version-card-list__item">
|
||||
<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}
|
||||
</span>
|
||||
)}
|
||||
</td>
|
||||
<td>{r.is_active ? 'да' : 'нет'}</td>
|
||||
<td className="mono">{fmtDt(r.created_at)}</td>
|
||||
<td>
|
||||
{!r.is_active && (
|
||||
<button
|
||||
type="button"
|
||||
className="btn btn-ghost btn--sm"
|
||||
onClick={() => activateVersion(r.id)}
|
||||
>
|
||||
сделать активной
|
||||
</button>
|
||||
)}
|
||||
</td>
|
||||
</tr>
|
||||
))}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
{r.is_active && (
|
||||
<span className="code-inline" style={{ fontSize: '0.7rem' }}>
|
||||
текущая
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
<p className="muted mono" style={{ margin: '0.4rem 0 0', fontSize: '0.8rem' }}>
|
||||
{fmtDt(r.created_at)}
|
||||
</p>
|
||||
<p className="muted" style={{ margin: '0.2rem 0 0', fontSize: '0.8rem' }}>
|
||||
Активна: {r.is_active ? 'да' : 'нет'}
|
||||
</p>
|
||||
</div>
|
||||
{!r.is_active && (
|
||||
<button
|
||||
type="button"
|
||||
className="btn btn-ghost btn--sm version-card-list__action"
|
||||
onClick={() => activateVersion(r.id)}
|
||||
>
|
||||
Сделать активной
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
</li>
|
||||
))}
|
||||
</ul>
|
||||
</AccSection>
|
||||
|
||||
{attemptsList != null && attemptsList.length > 0 && (
|
||||
@@ -1065,7 +1058,10 @@ export default function TestDetail() {
|
||||
)}
|
||||
|
||||
<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 ? (
|
||||
<button
|
||||
type="button"
|
||||
@@ -1191,41 +1187,44 @@ export default function TestDetail() {
|
||||
</select>
|
||||
{assignLoadBusy && <span className="text-muted">Загрузка…</span>}
|
||||
</div>
|
||||
<div className="assign-list" role="group" aria-label="Список сотрудников">
|
||||
{assignPeople.length === 0 && !assignLoadBusy && (
|
||||
{assignPeople.length > 0 ? (
|
||||
<div className="assign-list" role="group" aria-label="Список сотрудников">
|
||||
{assignPeople.map((p) => {
|
||||
const k = assignPersonKey(p);
|
||||
const picked = assignSelected.has(k);
|
||||
return (
|
||||
<label
|
||||
key={k}
|
||||
className={`assign-row${picked ? ' assign-row--selected' : ''}`}
|
||||
>
|
||||
<input
|
||||
type="checkbox"
|
||||
checked={picked}
|
||||
onChange={() => toggleAssignPerson(p)}
|
||||
/>
|
||||
<span className="assign-row__text">
|
||||
<span className="assign-row__fio">{p.fio}</span>
|
||||
{p.webLogin && (
|
||||
<span className="assign-row__login">{p.webLogin}</span>
|
||||
)}
|
||||
<span className="assign-row__meta">
|
||||
{p.departments || '—'}
|
||||
{p.clinicUserId
|
||||
? ' · есть учётка в модуле'
|
||||
: ' · нет учётки (создадим при назначении)'}
|
||||
</span>
|
||||
</span>
|
||||
</label>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
) : (
|
||||
!assignLoadBusy && (
|
||||
<p className="text-muted" style={{ margin: '0.5rem 0' }}>
|
||||
Нет подходящих записей. Смените фильтры или поиск.
|
||||
</p>
|
||||
)}
|
||||
{assignPeople.map((p) => {
|
||||
const k = assignPersonKey(p);
|
||||
const picked = assignSelected.has(k);
|
||||
return (
|
||||
<label
|
||||
key={k}
|
||||
className={`assign-row${picked ? ' assign-row--selected' : ''}`}
|
||||
>
|
||||
<input
|
||||
type="checkbox"
|
||||
checked={picked}
|
||||
onChange={() => toggleAssignPerson(p)}
|
||||
/>
|
||||
<span className="assign-row__text">
|
||||
<span className="assign-row__fio">{p.fio}</span>
|
||||
{p.webLogin && (
|
||||
<span className="assign-row__login">{p.webLogin}</span>
|
||||
)}
|
||||
<span className="assign-row__meta">
|
||||
{p.departments || '—'}
|
||||
{p.clinicUserId
|
||||
? ' · есть учётка в модуле'
|
||||
: ' · нет учётки (создадим при назначении)'}
|
||||
</span>
|
||||
</span>
|
||||
</label>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
)
|
||||
)}
|
||||
<div className="inline-actions" style={{ marginTop: '0.75rem' }}>
|
||||
<button
|
||||
type="button"
|
||||
|
||||
@@ -106,11 +106,7 @@ export default function TestsList() {
|
||||
<span className="list-row__title">{t.title}</span>
|
||||
<span className="list-row__meta">
|
||||
{formatTestAuthorLabel(user, t.created_by, t.author_full_name)}
|
||||
<span className="list-row__meta-sep" aria-hidden>
|
||||
{' '}
|
||||
·{' '}
|
||||
</span>
|
||||
v{t.version}
|
||||
<span className="list-row__meta-tail">{' · '}v{t.version}</span>
|
||||
</span>
|
||||
</Link>
|
||||
</div>
|
||||
@@ -151,11 +147,9 @@ export default function TestsList() {
|
||||
<span className="list-row__title">{t.title}</span>
|
||||
<span className="list-row__meta">
|
||||
{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>
|
||||
v{t.version} · скрыт
|
||||
</span>
|
||||
</Link>
|
||||
</div>
|
||||
|
||||
@@ -248,7 +248,7 @@ code,
|
||||
.btn-ghost {
|
||||
background: transparent;
|
||||
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 {
|
||||
@@ -264,8 +264,11 @@ code,
|
||||
|
||||
.btn--sm {
|
||||
font-size: 0.8rem;
|
||||
padding: 0.35rem 0.6rem;
|
||||
padding: 0.5rem 0.75rem;
|
||||
border-radius: 0.5rem;
|
||||
min-height: 2.75rem;
|
||||
min-width: 2.75rem;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
/* --- App shell (cabinet/base) --- */
|
||||
@@ -378,7 +381,7 @@ code,
|
||||
max-width: var(--max-content);
|
||||
width: 100%;
|
||||
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 */
|
||||
@@ -431,6 +434,11 @@ code,
|
||||
margin-top: 0.25rem;
|
||||
}
|
||||
|
||||
/* «· v1» и хвост мета — не рвём посередине (мобайл) */
|
||||
.list-row__meta-tail {
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
/* Вся плитка — одна ссылка */
|
||||
.list-row--action {
|
||||
padding: 0;
|
||||
@@ -609,7 +617,8 @@ code,
|
||||
}
|
||||
|
||||
.assign-list {
|
||||
max-height: min(50vh, 22rem);
|
||||
max-height: min(40vh, 18rem);
|
||||
min-height: 0;
|
||||
overflow: auto;
|
||||
border: 1px solid color-mix(in srgb, var(--outline-variant) 30%, transparent);
|
||||
border-radius: 0.75rem;
|
||||
@@ -755,6 +764,57 @@ code,
|
||||
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 {
|
||||
margin-top: 1.25rem;
|
||||
padding: 1rem;
|
||||
@@ -783,3 +843,60 @@ code,
|
||||
.inline-actions .btn {
|
||||
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;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user