Update web/templates/stats.html

This commit is contained in:
2025-08-16 16:01:38 +00:00
parent 46c0e4cc18
commit 38220d0784
+90 -89
View File
@@ -5,6 +5,7 @@
<div class="container"> <div class="container">
<h1 class="page-title">Statistics</h1> <h1 class="page-title">Statistics</h1>
<!-- Totals -->
<div class="card" style="padding:16px; margin-bottom:16px;"> <div class="card" style="padding:16px; margin-bottom:16px;">
<div class="meta-grid"> <div class="meta-grid">
<div> <div>
@@ -36,15 +37,15 @@
</div> </div>
</div> </div>
<!-- Top subjects --> <!-- Top subjects (clickable) -->
<div class="card" style="padding:16px; margin-bottom:16px;"> <div class="card" style="padding:16px; margin-bottom:16px;">
<div class="meta-label">Top Subjects</div> <div class="meta-label">Top Subjects</div>
<div style="display:flex; flex-wrap:wrap; gap:8px; margin-top:8px;"> <div style="display:flex; flex-wrap:wrap; gap:8px; margin-top:8px;">
{% for s in top_subjects %} {% for s in top_subjects %}
<a <a
class="chip chip-link" class="chip chip-muted"
href="{% url 'search' %}?q={{ s.name|urlencode }}&subject=on" href="{% url 'search' %}?q={{ s.name|urlencode }}&subject=on"
title="Search subject: {{ s.name }}" title="Search by subject: {{ s.name }}"
> >
{{ s.name }} ({{ s.count }}) {{ s.name }} ({{ s.count }})
</a> </a>
@@ -52,7 +53,7 @@
<span class="muted">No subjects.</span> <span class="muted">No subjects.</span>
{% endfor %} {% endfor %}
</div> </div>
</div> </div>
<!-- Scripture usage --> <!-- Scripture usage -->
<div class="card" style="padding:16px; margin-bottom:16px;"> <div class="card" style="padding:16px; margin-bottom:16px;">
@@ -64,7 +65,15 @@
<h4 style="margin:0 0 6px;">Top 10 Books</h4> <h4 style="margin:0 0 6px;">Top 10 Books</h4>
<ol class="small" style="margin:0; padding-left:18px;"> <ol class="small" style="margin:0; padding-left:18px;">
{% for book, cnt in top_books %} {% for book, cnt in top_books %}
<li><strong>{{ book }}</strong> — {{ cnt }}</li> <li>
<a
href="{% url 'search' %}?q={{ book|urlencode }}&scripture_raw=on"
title="Search scripture field for book: {{ book }}"
>
<strong>{{ book }}</strong>
</a>
— {{ cnt }}
</li>
{% empty %} {% empty %}
<li class="muted">No data yet.</li> <li class="muted">No data yet.</li>
{% endfor %} {% endfor %}
@@ -76,56 +85,47 @@
{% for ref, cnt in top_refs %} {% for ref, cnt in top_refs %}
<li> <li>
<a <a
class="chip chip-link"
href="{% url 'search' %}?q={{ ref|urlencode }}&scripture_raw=on" href="{% url 'search' %}?q={{ ref|urlencode }}&scripture_raw=on"
title="Search in Scripture field: {{ ref }}" title="Search scripture field for: {{ ref }}"
> >
{{ ref }} <strong>{{ ref }}</strong>
</a> — {{ cnt }} </a>
— {{ cnt }}
</li> </li>
{% empty %} {% empty %}
<li class="muted">No data yet.</li> <li class="muted">No data yet.</li>
{% endfor %} {% endfor %}
</ol> </ol>
</div> </div>
</div> </div>
</div> </div>
<!-- Tools: collapsible panel --> <!-- Maintenance tools (collapsible like the search help) -->
<div class="card" style="padding:16px; margin-bottom:24px;"> <div class="card" style="padding:16px; margin-bottom:24px;">
<div style="display:flex; align-items:center; justify-content:space-between; gap:10px;"> <details class="tools-details">
<div class="meta-label">Maintenance Tools</div> <summary class="tools-summary">
<button 🛠️ Maintenance Tools
class="btn btn-secondary" </summary>
type="button"
id="tools-toggle"
aria-expanded="false"
aria-controls="tools-panel"
>
▾ Show tools
</button>
</div>
<div id="tools-panel" class="tools-panel" hidden> <div class="tools-grid">
<!-- Scripture Normalizer --> <!-- Scripture Normalizer -->
<div class="tool-block"> <div class="tool">
<h4>Scripture Normalizer</h4> <h4>Scripture Normalizer</h4>
<p class="small muted"> <p class="small muted">
Standardizes Bible references (book abbreviations, separators, and repeats book names as needed). Normalize scripture references (abbreviations, separators, and repeat book names where needed).
Use preview first; Apply will modify all <code>scripture_raw</code> fields.
</p> </p>
<div class="tool-actions"> <div class="tool-actions">
<form method="get" action="{% url 'normalize_scripture' %}" class="inline"> <form method="get" action="{% url 'normalize_scripture' %}">
<input type="number" min="0" name="limit" placeholder="Preview limit (optional)" <input type="number" min="0" name="limit" placeholder="Preview limit (optional)"
class="tool-input"> class="tool-input">
<button class="btn btn-secondary">Preview (dryrun)</button> <button class="btn btn-secondary">Preview (dry-run)</button>
</form> </form>
{% if user.is_authenticated and user.is_staff %} {% if user.is_authenticated and user.is_staff %}
<form method="post" action="{% url 'normalize_scripture' %}" class="inline"> <form method="post" action="{% url 'normalize_scripture' %}">
{% csrf_token %} {% csrf_token %}
<button class="btn btn-danger" <button class="btn btn-danger"
onclick="return confirm('Apply Scripture Normalizer to ALL entries? This will modify scripture_raw fields.');"> onclick="return confirm('Apply normalization to ALL entries? This will modify scripture_raw fields.');">
Apply to all Apply to all
</button> </button>
</form> </form>
@@ -133,28 +133,51 @@
</div> </div>
</div> </div>
<hr class="tool-sep">
<!-- Source Normalizer --> <!-- Source Normalizer -->
<div class="tool-block"> <div class="tool">
<h4>Source Normalizer</h4> <h4>Source Normalizer</h4>
<p class="small muted"> <p class="small muted">
Converts Watchtower/Awake!/Yearbook/KM references (e.g., “March 15, 2013 WT page 14”) into WOL shortcodes Convert Watchtower/Awake!/Yearbook/KM references into WOL short-codes (e.g., <code>w13 3/15 p.14</code>).
like <code>w13 3/15 p.14</code>. NonJW sources are left unchanged. Use preview first; Apply will modify all Non-JW sources are left unchanged.
<code>source</code> fields.
</p> </p>
<div class="tool-actions"> <div class="tool-actions">
<form method="get" action="{% url 'normalize_source' %}" class="inline"> <form method="get" action="{% url 'normalize_source' %}">
<input type="number" min="0" name="limit" placeholder="Preview limit (optional)" <input type="number" min="0" name="limit" placeholder="Preview limit (optional)"
class="tool-input"> class="tool-input">
<button class="btn btn-secondary">Preview (dryrun)</button> <button class="btn btn-secondary">Preview (dry-run)</button>
</form> </form>
{% if user.is_authenticated and user.is_staff %} {% if user.is_authenticated and user.is_staff %}
<form method="post" action="{% url 'normalize_source' %}" class="inline"> <form method="post" action="{% url 'normalize_source' %}">
{% csrf_token %} {% csrf_token %}
<button class="btn btn-danger" <button class="btn btn-danger"
onclick="return confirm('Apply Source Normalizer to ALL entries? This cannot be undone. Make a backup first.');"> onclick="return confirm('Apply normalization to all entries? This cannot be undone. Make a backup first.')">
Apply to all
</button>
</form>
{% endif %}
</div>
</div>
<!-- Subject Normalizer (NEW) -->
<div class="tool">
<h4>Subject Normalizer</h4>
<p class="small muted">
Unify delimiters (semicolons, dashes, slashes, etc.) into commas and trim spacing.
Preserves multi-word subjects. Most entries are already correct—this avoids over-splitting.
</p>
<div class="tool-actions">
<form method="get" action="{% url 'normalize_subjects' %}">
<input type="number" min="0" name="limit" placeholder="Preview limit (optional)"
class="tool-input">
<button class="btn btn-secondary">Preview (dry-run)</button>
</form>
{% if user.is_authenticated and user.is_staff %}
<form method="post" action="{% url 'normalize_subjects' %}">
{% csrf_token %}
<button class="btn btn-danger"
onclick="return confirm('Apply subject normalization to all entries? This will modify subject fields. Make a backup first.')">
Apply to all Apply to all
</button> </button>
</form> </form>
@@ -162,59 +185,37 @@
</div> </div>
</div> </div>
</div> </div>
<div class="small muted" style="margin-top:10px;">
Tip: add <code>?limit=1000</code> on the preview to test against a subset before applying globally.
</div>
</details>
</div> </div>
</div> </div>
<!-- Tiny, scoped styles + JS for the collapsible panel --> <!-- Minimal styles for the collapsible tools, matching your existing look -->
<style> <style>
.tools-panel{ .tools-details { margin-top: 4px; }
.tools-summary {
display:inline-flex; align-items:center; gap:8px;
padding:8px 12px; border-radius:10px;
background: var(--btn-bg); border:1px solid var(--btn-border);
color: var(--nav-ink); cursor:pointer; user-select:none;
}
.tools-summary:hover { background: var(--btn-hover); }
.tools-details[open] .tools-summary { background:#fff; }
.tools-grid {
margin-top:12px; margin-top:12px;
border:1px solid var(--border); display:grid; gap:16px;
border-radius:12px; grid-template-columns: repeat( auto-fit, minmax(260px, 1fr) );
background:#fff;
padding:14px;
} }
.tool-block h4{ .tool { border:1px solid var(--nav-border); border-radius:12px; padding:12px; background:#fff; }
margin:0 0 6px; .tool h4 { margin:0 0 6px; }
font-size:16px; .tool-actions { display:flex; flex-wrap:wrap; gap:8px; margin-top:8px; }
color:#0f172a; .tool-input {
} border:1px solid var(--nav-border);
.tool-actions{ border-radius:8px; padding:6px 8px; min-width: 190px;
display:flex;
flex-wrap:wrap;
gap:10px;
margin-top:8px;
}
.tool-input{
border:1px solid var(--border);
border-radius:8px;
padding:6px 8px;
font-size:14px;
margin-right:6px;
}
.tool-sep{
border:none;
border-top:1px solid var(--border);
margin:14px 0;
} }
</style> </style>
<script>
(function(){
var btn = document.getElementById('tools-toggle');
var panel = document.getElementById('tools-panel');
if(!btn || !panel) return;
btn.addEventListener('click', function(){
var open = panel.hasAttribute('hidden') ? false : true;
if(open){
panel.setAttribute('hidden', '');
btn.textContent = '▾ Show tools';
btn.setAttribute('aria-expanded', 'false');
}else{
panel.removeAttribute('hidden');
btn.textContent = '▴ Hide tools';
btn.setAttribute('aria-expanded', 'true');
}
});
})();
</script>
{% endblock %} {% endblock %}