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="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">
|
{versions.map((r) => (
|
||||||
<thead>
|
<li key={r.id} className="surface-card version-card-list__item">
|
||||||
<tr>
|
<div className="version-card-list__row">
|
||||||
<th>Версия</th>
|
<div className="version-card-list__main">
|
||||||
<th>Активна</th>
|
<div className="version-card-list__title-line">
|
||||||
<th>Создана</th>
|
<span className="font-headline" style={{ fontSize: '1rem' }}>
|
||||||
<th />
|
v{r.version}
|
||||||
</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' }}
|
|
||||||
>
|
|
||||||
текущая
|
|
||||||
</span>
|
</span>
|
||||||
)}
|
{r.is_active && (
|
||||||
</td>
|
<span className="code-inline" style={{ fontSize: '0.7rem' }}>
|
||||||
<td>{r.is_active ? 'да' : 'нет'}</td>
|
текущая
|
||||||
<td className="mono">{fmtDt(r.created_at)}</td>
|
</span>
|
||||||
<td>
|
)}
|
||||||
{!r.is_active && (
|
</div>
|
||||||
<button
|
<p className="muted mono" style={{ margin: '0.4rem 0 0', fontSize: '0.8rem' }}>
|
||||||
type="button"
|
{fmtDt(r.created_at)}
|
||||||
className="btn btn-ghost btn--sm"
|
</p>
|
||||||
onClick={() => activateVersion(r.id)}
|
<p className="muted" style={{ margin: '0.2rem 0 0', fontSize: '0.8rem' }}>
|
||||||
>
|
Активна: {r.is_active ? 'да' : 'нет'}
|
||||||
сделать активной
|
</p>
|
||||||
</button>
|
</div>
|
||||||
)}
|
{!r.is_active && (
|
||||||
</td>
|
<button
|
||||||
</tr>
|
type="button"
|
||||||
))}
|
className="btn btn-ghost btn--sm version-card-list__action"
|
||||||
</tbody>
|
onClick={() => activateVersion(r.id)}
|
||||||
</table>
|
>
|
||||||
</div>
|
Сделать активной
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</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,41 +1187,44 @@ export default function TestDetail() {
|
|||||||
</select>
|
</select>
|
||||||
{assignLoadBusy && <span className="text-muted">Загрузка…</span>}
|
{assignLoadBusy && <span className="text-muted">Загрузка…</span>}
|
||||||
</div>
|
</div>
|
||||||
<div className="assign-list" role="group" aria-label="Список сотрудников">
|
{assignPeople.length > 0 ? (
|
||||||
{assignPeople.length === 0 && !assignLoadBusy && (
|
<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 className="text-muted" style={{ margin: '0.5rem 0' }}>
|
||||||
Нет подходящих записей. Смените фильтры или поиск.
|
Нет подходящих записей. Смените фильтры или поиск.
|
||||||
</p>
|
</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' }}>
|
<div className="inline-actions" style={{ marginTop: '0.75rem' }}>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user