feat(story-2.2+2.3): device setup page, account linking, naming & configuration
Story 2.2 — /setup/{mac} Twig page (no Vue, works JS-disabled):
- Register tab: creates account + logs in + links device → /setup/{mac}/configure
- Login tab: manual credential check via UserPasswordHasherInterface + Security::login()
+ links device → /setup/{mac}/configure
- Re-provisioning: DeviceService.linkToUser() atomically transfers ownership + stubs
purgeDeviceHistory() (completed in Epic 3 when Image/Approval entities exist)
Story 2.3 — /setup/{mac}/configure (requires auth):
- GET: device name, orientation (landscape/portrait), rotation interval (6/12/24/48/168h),
uniqueness window (5/10/20/50 cycles)
- POST: validates name, saves to Device entity, redirects to Vue SPA
- Device entity: mac, name, orientation (Orientation enum), rotationIntervalHours,
uniquenessWindow, user (ManyToOne), linkedAt
- PATCH /api/devices/{id}: Vue SPA can edit any device field (Story 2.3 "edit from app")
- GET /api/devices: list authenticated user's devices
- Migration: create device table with Orientation enum column
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,81 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>Name your frame — pictureFrame</title>
|
||||
<style>
|
||||
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
|
||||
body { font-family: system-ui, sans-serif; min-height: 100dvh; background: #fdf6ee; color: #3a2e22;
|
||||
display: flex; align-items: flex-start; justify-content: center; padding: 2rem 1rem; }
|
||||
.card { width: 100%; max-width: 400px; }
|
||||
h1 { font-size: 1.4rem; font-weight: 700; margin-bottom: .25rem; }
|
||||
.subtitle { font-size: .875rem; color: #8a7060; margin-bottom: 1.5rem; }
|
||||
.field { margin-bottom: 1.25rem; }
|
||||
label { display: block; font-size: .8125rem; font-weight: 600; color: #8a7060; margin-bottom: .375rem; }
|
||||
input[type="text"], select {
|
||||
width: 100%; min-height: 44px; padding: 0 .875rem; border: 1px solid #e8d9c4;
|
||||
border-radius: 10px; background: #fff; font-size: 1rem; color: #3a2e22; }
|
||||
select { padding-right: 2rem; appearance: none;
|
||||
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='8' viewBox='0 0 12 8'%3E%3Cpath d='M1 1l5 5 5-5' stroke='%238a7060' stroke-width='1.5' fill='none'/%3E%3C/svg%3E");
|
||||
background-repeat: no-repeat; background-position: right .875rem center; }
|
||||
.field-error { margin-top: .375rem; font-size: .8125rem; color: #c0392b; }
|
||||
.hint { margin-top: .375rem; font-size: .8125rem; color: #8a7060; }
|
||||
.btn { display: flex; align-items: center; justify-content: center; width: 100%; min-height: 44px;
|
||||
margin-top: 1.5rem; background: #c97c3a; color: #fff; border: none; border-radius: 9999px;
|
||||
font-size: 1rem; font-weight: 700; cursor: pointer; }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="card">
|
||||
<h1>Name your frame</h1>
|
||||
<p class="subtitle">You can always change these settings later.</p>
|
||||
|
||||
{% if error %}
|
||||
<p class="field-error" role="alert" style="margin-bottom:1rem">{{ error }}</p>
|
||||
{% endif %}
|
||||
|
||||
<form method="post" novalidate>
|
||||
<div class="field">
|
||||
<label for="name">Frame name</label>
|
||||
<input type="text" id="name" name="name"
|
||||
value="{{ device.name }}"
|
||||
placeholder="e.g. Living room frame"
|
||||
maxlength="100" required autofocus>
|
||||
</div>
|
||||
|
||||
<div class="field">
|
||||
<label for="orientation">Display orientation</label>
|
||||
<select id="orientation" name="orientation">
|
||||
<option value="landscape" {% if device.orientation.value == 'landscape' %}selected{% endif %}>Landscape</option>
|
||||
<option value="portrait" {% if device.orientation.value == 'portrait' %}selected{% endif %}>Portrait</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="field">
|
||||
<label for="rotation_interval_hours">Rotation frequency</label>
|
||||
<select id="rotation_interval_hours" name="rotation_interval_hours">
|
||||
<option value="6" {% if device.rotationIntervalHours == 6 %}selected{% endif %}>Every 6 hours</option>
|
||||
<option value="12" {% if device.rotationIntervalHours == 12 %}selected{% endif %}>Every 12 hours</option>
|
||||
<option value="24" {% if device.rotationIntervalHours == 24 %}selected{% endif %}>Daily (every 24 hours)</option>
|
||||
<option value="48" {% if device.rotationIntervalHours == 48 %}selected{% endif %}>Every 2 days</option>
|
||||
<option value="168" {% if device.rotationIntervalHours == 168 %}selected{% endif %}>Weekly</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="field">
|
||||
<label for="uniqueness_window">Uniqueness window</label>
|
||||
<select id="uniqueness_window" name="uniqueness_window">
|
||||
<option value="5" {% if device.uniquenessWindow == 5 %}selected{% endif %}>5 cycles</option>
|
||||
<option value="10" {% if device.uniquenessWindow == 10 %}selected{% endif %}>10 cycles (default)</option>
|
||||
<option value="20" {% if device.uniquenessWindow == 20 %}selected{% endif %}>20 cycles</option>
|
||||
<option value="50" {% if device.uniquenessWindow == 50 %}selected{% endif %}>50 cycles</option>
|
||||
</select>
|
||||
<p class="hint">Images won't repeat until this many other photos have been shown.</p>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn">Save & finish setup</button>
|
||||
</form>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
@@ -0,0 +1,102 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>Set up your frame — pictureFrame</title>
|
||||
<style>
|
||||
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
|
||||
body { font-family: system-ui, sans-serif; min-height: 100dvh; background: #fdf6ee; color: #3a2e22;
|
||||
display: flex; align-items: flex-start; justify-content: center; padding: 2rem 1rem; }
|
||||
.card { width: 100%; max-width: 400px; }
|
||||
h1 { font-size: 1.4rem; font-weight: 700; margin-bottom: .25rem; }
|
||||
.subtitle { font-size: .875rem; color: #8a7060; margin-bottom: 1.5rem; }
|
||||
.tabs { display: flex; border-bottom: 1px solid #e8d9c4; margin-bottom: 1.5rem; }
|
||||
.tab { flex: 1; padding: .75rem; text-align: center; font-weight: 700; font-size: .9rem;
|
||||
color: #8a7060; text-decoration: none; border-bottom: 2px solid transparent; transition: color .15s; }
|
||||
.tab.active { color: #c97c3a; border-bottom-color: #c97c3a; }
|
||||
.panel { display: none; } .panel.active { display: block; }
|
||||
.field { margin-bottom: 1rem; }
|
||||
label { display: block; font-size: .8125rem; font-weight: 600; color: #8a7060; margin-bottom: .375rem; }
|
||||
input[type="email"], input[type="password"], input[type="text"] {
|
||||
width: 100%; min-height: 44px; padding: 0 .875rem; border: 1px solid #e8d9c4;
|
||||
border-radius: 10px; background: #fff; font-size: 1rem; color: #3a2e22; }
|
||||
input[aria-invalid="true"] { border-color: #c0392b; }
|
||||
.field-error { margin-top: .25rem; font-size: .8125rem; color: #c0392b; }
|
||||
.btn { display: flex; align-items: center; justify-content: center; width: 100%; min-height: 44px;
|
||||
margin-top: 1.25rem; background: #c97c3a; color: #fff; border: none; border-radius: 9999px;
|
||||
font-size: 1rem; font-weight: 700; cursor: pointer; }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="card">
|
||||
<h1>Set up your frame</h1>
|
||||
<p class="subtitle">Create an account or sign in to link this frame.</p>
|
||||
|
||||
<div class="tabs">
|
||||
<a href="#register" class="tab {% if not login_error %}active{% endif %}" data-tab="register">Create account</a>
|
||||
<a href="#login" class="tab {% if login_error %}active{% endif %}" data-tab="login">Sign in</a>
|
||||
</div>
|
||||
|
||||
{# ── Register panel ────────────────────────────────────────────────────── #}
|
||||
<div id="register" class="panel {% if not login_error %}active{% endif %}">
|
||||
{{ form_start(reg_form, {action: path('setup_register', {mac: mac}), attr: {novalidate: 'novalidate'}}) }}
|
||||
<div class="field">
|
||||
{{ form_label(reg_form.email) }}
|
||||
{{ form_widget(reg_form.email, {attr: {
|
||||
id: 'reg-email',
|
||||
'aria-invalid': reg_form.email.vars.errors|length > 0 ? 'true' : 'false'
|
||||
}}) }}
|
||||
{% for error in reg_form.email.vars.errors %}
|
||||
<p class="field-error" role="alert">{{ error.message }}</p>
|
||||
{% endfor %}
|
||||
</div>
|
||||
<div class="field">
|
||||
{{ form_label(reg_form.plainPassword) }}
|
||||
{{ form_widget(reg_form.plainPassword, {attr: {
|
||||
id: 'reg-pass',
|
||||
'aria-invalid': reg_form.plainPassword.vars.errors|length > 0 ? 'true' : 'false'
|
||||
}}) }}
|
||||
{% for error in reg_form.plainPassword.vars.errors %}
|
||||
<p class="field-error" role="alert">{{ error.message }}</p>
|
||||
{% endfor %}
|
||||
</div>
|
||||
<button type="submit" class="btn">Create account & link frame</button>
|
||||
{{ form_end(reg_form) }}
|
||||
</div>
|
||||
|
||||
{# ── Login panel ───────────────────────────────────────────────────────── #}
|
||||
<div id="login" class="panel {% if login_error %}active{% endif %}">
|
||||
<form method="post" action="{{ path('setup_login', {mac: mac}) }}" novalidate>
|
||||
{% if login_error %}
|
||||
<p class="field-error" role="alert" style="margin-bottom:1rem">{{ login_error }}</p>
|
||||
{% endif %}
|
||||
<div class="field">
|
||||
<label for="login-email">Email address</label>
|
||||
<input type="email" id="login-email" name="_username" autocomplete="email" min-height="44px">
|
||||
</div>
|
||||
<div class="field">
|
||||
<label for="login-pass">Password</label>
|
||||
<input type="password" id="login-pass" name="_password" autocomplete="current-password">
|
||||
</div>
|
||||
<button type="submit" class="btn">Sign in & link frame</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
(function () {
|
||||
var tabs = document.querySelectorAll('.tab');
|
||||
var panels = document.querySelectorAll('.panel');
|
||||
tabs.forEach(function (tab) {
|
||||
tab.addEventListener('click', function (e) {
|
||||
e.preventDefault();
|
||||
var target = tab.dataset.tab;
|
||||
tabs.forEach(function (t) { t.classList.toggle('active', t.dataset.tab === target); });
|
||||
panels.forEach(function (p) { p.classList.toggle('active', p.id === target); });
|
||||
});
|
||||
});
|
||||
}());
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
Reference in New Issue
Block a user