Files
battery-tracker-app/templates/device_list.html
T
iterminate 39b52a3fa4 Replace browser confirm() dialogs with custom modal; add live label preview on battery add form
- base.html: add CSS/HTML/JS for styled in-app confirmation modal (dark-mode compatible via CSS vars)
- device_list, battery_detail: convert onsubmit confirm() to declarative data-confirm attributes
- dashboard: convert bulk Delete/Install buttons to use modal helpers (submitWithAction pattern)
- app.py: pass brand_counts dict to battery_add template
- battery_add.html: show live "Will create: Brand 001 → Brand 003" preview as brand/quantity change
- tests: add two tests covering brand_counts server-side rendering
2026-04-13 09:53:21 -05:00

111 lines
4.7 KiB
HTML

{% extends "base.html" %}
{% block title %}Devices — Battery Tracker{% endblock %}
{% block content %}
<h1>Devices</h1>
<div class="card">
<div id="device-filter-bar" style="display:flex;gap:0.5rem;flex-wrap:wrap;align-items:center;margin-bottom:0.75rem;">
<select id="filter-type" onchange="applyDeviceFilters()"
style="padding:0.25rem 0.5rem;font-size:0.85rem;border:1px solid #cbd5e1;border-radius:4px;">
<option value="">All Types</option>
{% for t in device_types|default([]) %}
<option value="{{ t }}">{{ t }}</option>
{% endfor %}
</select>
<input type="text" id="filter-device-text" oninput="applyDeviceFilters()" placeholder="Search…"
style="padding:0.25rem 0.5rem;font-size:0.85rem;border:1px solid #cbd5e1;border-radius:4px;width:140px;">
<button type="button" onclick="resetDeviceFilters()" class="btn btn-sm btn-secondary"
id="device-filter-reset" style="display:none;">✕ Reset</button>
<span id="device-filter-count" style="font-size:0.8rem;color:#64748b;"></span>
</div>
<div class="table-wrap">
<table class="responsive-table">
<thead>
<tr>
<th>Name</th>
<th>Type</th>
<th>Slots</th>
<th>Installed</th>
<th>Brands</th>
<th>Actions</th>
</tr>
</thead>
<tbody>
{% for d in devices %}
<tr data-type="{{ d.device_type or '' }}" data-name="{{ d.name|lower }}">
<td data-label="Device"><a href="{{ url_for('device_detail', device_id=d.id) }}"><strong>{{ d.name }}</strong></a></td>
<td data-label="Type">{{ d.device_type or '—' }}</td>
<td data-label="Slots">{{ d.battery_slots }}</td>
<td data-label="Installed">
{{ d.installed_count() }} / {{ d.battery_slots }}
{% if d.installed_count() >= d.battery_slots %}
<span class="badge badge-retired">Full</span>
{% endif %}
</td>
<td data-label="Brands">
{% set brands = d.installed_brands() %}
{% if brands %}
{{ brands|join(', ') }}
{% if d.has_mixed_brands() %}
<span class="badge badge-warning">⚠ mixed</span>
{% endif %}
{% else %}
<span class="text-muted"></span>
{% endif %}
</td>
<td data-label="Actions" style="white-space:nowrap;">
<a class="btn btn-sm btn-secondary" href="{{ url_for('device_detail', device_id=d.id) }}">View</a>
{% if d.installed_count() > 0 %}
<form class="inline" method="post" action="{{ url_for('device_unassign_all', device_id=d.id) }}"
data-confirm="Unassign all batteries from {{ d.name }}?"
data-confirm-ok="Unassign" data-confirm-class="btn-warning">
<button class="btn btn-sm btn-warning" type="submit">Unassign All</button>
</form>
{% endif %}
<form class="inline" method="post" action="{{ url_for('device_delete', device_id=d.id) }}"
data-confirm="Delete {{ d.name }}? All installed batteries will be unassigned."
data-confirm-ok="Delete" data-confirm-class="btn-danger">
<button class="btn btn-sm btn-danger" type="submit">Delete</button>
</form>
</td>
</tr>
{% else %}
<tr><td colspan="6" class="text-muted" style="text-align:center;padding:1rem;">No devices yet. <a href="{{ url_for('device_add') }}">Add one.</a></td></tr>
{% endfor %}
</tbody>
</table>
</div>
</div>
<a class="btn btn-primary" href="{{ url_for('device_add') }}">+ Add Device</a>
<script>
function applyDeviceFilters() {
var typeVal = document.getElementById('filter-type').value.toLowerCase();
var textVal = document.getElementById('filter-device-text').value.toLowerCase();
var rows = document.querySelectorAll('tbody tr[data-name]');
var visible = 0;
rows.forEach(function(row) {
var rowType = (row.dataset.type || '').toLowerCase();
var rowName = (row.dataset.name || '').toLowerCase();
var show = (!typeVal || rowType === typeVal) &&
(!textVal || rowName.includes(textVal) || rowType.includes(textVal));
row.style.display = show ? '' : 'none';
if (show) visible++;
});
var active = typeVal || textVal;
document.getElementById('device-filter-reset').style.display = active ? '' : 'none';
document.getElementById('device-filter-count').textContent =
active ? (visible + ' of ' + rows.length + ' shown') : '';
}
function resetDeviceFilters() {
document.getElementById('filter-type').value = '';
document.getElementById('filter-device-text').value = '';
applyDeviceFilters();
}
</script>
{% endblock %}