refactor(admin): extract dashboardScript to src/scripts/client/dashboard.ts

Moves the 650-line inline JS template literal from admin.tsx into a
proper TypeScript source file with full type annotations. esbuild
compiles it to a minified IIFE committed in src/scripts/generated/,
which is imported and inlined into the HTML response as before.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Julien Herr
2026-05-22 13:50:47 +02:00
parent 0663861471
commit 40f4b42cd5
3 changed files with 840 additions and 651 deletions
+3 -651
View File
@@ -10,6 +10,7 @@ import { Layout, clampText } from "./admin/ui";
import { listAllFeeds, updateFeedInList } from "./admin/helpers";
import { feedsRouter } from "./admin/feeds";
import { emailsRouter } from "./admin/emails";
import { dashboardScript } from "../scripts/generated/dashboard";
type AppEnv = { Bindings: Env };
@@ -220,657 +221,8 @@ app.get("/logout", (c) => {
return c.redirect("/admin/login");
});
// The large inline script for the dashboard page
const dashboardScript = `
let FEED_ROWS = [];
let FEED_CHECKBOXES = [];
let FEED_SELECTED_COUNT_EL = null;
let FEED_MATCH_COUNT_EL = null;
let FEED_TOTAL_COUNT_EL = null;
let FEED_BULK_DELETE_BUTTON_EL = null;
let FEED_SELECT_ALL_EL = null;
let FEED_FILTER_TIMER = null;
let FEED_BULK_DELETE_IN_PROGRESS = false;
let FEED_SORT_KEY = 'title';
let FEED_SORT_DIR = 'asc';
const FEED_COLLATOR = new Intl.Collator(undefined, { numeric: true, sensitivity: 'base' });
function initFeedUI() {
FEED_ROWS = Array.from(document.querySelectorAll('.feed-row'));
FEED_CHECKBOXES = Array.from(document.querySelectorAll('.feed-select'));
FEED_SELECTED_COUNT_EL = document.getElementById('selected-feed-count');
FEED_MATCH_COUNT_EL = document.getElementById('feed-match-count');
FEED_TOTAL_COUNT_EL = document.getElementById('feed-total-count');
FEED_BULK_DELETE_BUTTON_EL = document.getElementById('bulk-delete-feeds-button');
FEED_SELECT_ALL_EL = document.getElementById('select-all-feeds');
setupFeedTableResizing();
setupFeedTableSorting();
setupFeedDeleteButtons();
updateFeedMatchCount();
updateFeedSelectionState();
}
function updateFeedMatchCount() {
if (!FEED_MATCH_COUNT_EL) return;
const total = FEED_ROWS.length;
const visible = FEED_ROWS.filter((row) => !row.hidden).length;
const query = (document.getElementById('feed-search')?.value || '').trim();
FEED_MATCH_COUNT_EL.textContent = query ? ('Showing ' + visible + ' of ' + total) : ('Showing ' + total);
}
function scheduleFeedFilter() {
if (FEED_FILTER_TIMER) {
clearTimeout(FEED_FILTER_TIMER);
}
FEED_FILTER_TIMER = setTimeout(filterFeedRows, 120);
}
function getSortValue(row, key) {
const prop = 'sort' + key.charAt(0).toUpperCase() + key.slice(1);
return (row.dataset && row.dataset[prop]) ? row.dataset[prop] : '';
}
function updateFeedSortIndicators(table) {
const headerCells = Array.from(table.querySelectorAll('th[data-sort-key]'));
headerCells.forEach((th) => {
const key = th.getAttribute('data-sort-key') || '';
const indicator = th.querySelector('.sort-indicator');
const active = key === FEED_SORT_KEY;
if (indicator) {
indicator.textContent = active ? (FEED_SORT_DIR === 'asc' ? '^' : 'v') : '';
}
th.setAttribute('aria-sort', active ? (FEED_SORT_DIR === 'asc' ? 'ascending' : 'descending') : 'none');
});
}
function sortFeedTableBy(key) {
const table = document.querySelector('table.table-feeds');
const tbody = document.getElementById('feed-table-body');
if (!table || !tbody) return;
if (FEED_SORT_KEY === key) {
FEED_SORT_DIR = FEED_SORT_DIR === 'asc' ? 'desc' : 'asc';
} else {
FEED_SORT_KEY = key;
FEED_SORT_DIR = 'asc';
}
const dirMultiplier = FEED_SORT_DIR === 'asc' ? 1 : -1;
const rows = Array.from(tbody.querySelectorAll('.feed-row'));
rows.sort((a, b) => {
const av = getSortValue(a, FEED_SORT_KEY);
const bv = getSortValue(b, FEED_SORT_KEY);
return dirMultiplier * FEED_COLLATOR.compare(av, bv);
});
const fragment = document.createDocumentFragment();
rows.forEach((row) => fragment.appendChild(row));
tbody.appendChild(fragment);
updateFeedSortIndicators(table);
}
function setupFeedTableSorting() {
const table = document.querySelector('table.table-feeds');
if (!table) return;
table.querySelectorAll('button.th-button[data-sort-key]').forEach((button) => {
button.addEventListener('click', () => {
const key = button.getAttribute('data-sort-key') || '';
if (!key) return;
sortFeedTableBy(key);
});
});
updateFeedSortIndicators(table);
}
function setupFeedTableResizing() {
const table = document.querySelector('table.table-feeds');
if (!table) return;
const storageKey = 'email-to-rss.admin.feedsTable.colWidths';
const minWidths = {
title: 220,
feedId: 120,
email: 160,
rss: 160,
actions: 160,
};
const defaultWidths = {
title: 340,
feedId: 160,
email: 220,
rss: 220,
actions: 200,
};
const cols = Array.from(table.querySelectorAll('colgroup col'));
const colByKey = {};
cols.forEach((col) => {
const key = col.getAttribute('data-col');
if (key) colByKey[key] = col;
});
// Restore widths
try {
const saved = JSON.parse(localStorage.getItem(storageKey) || '{}');
Object.keys(saved || {}).forEach((key) => {
const px = Number(saved[key]);
if (!colByKey[key] || !Number.isFinite(px)) return;
colByKey[key].style.width = px + 'px';
});
} catch {
// Ignore bad localStorage values
}
const persist = () => {
try {
const out = {};
Object.keys(colByKey).forEach((key) => {
if (key === 'select') return;
const px = parseInt(colByKey[key].style.width || '0', 10);
if (Number.isFinite(px) && px > 0) out[key] = px;
});
localStorage.setItem(storageKey, JSON.stringify(out));
} catch {
// localStorage may be unavailable in some modes; ignore
}
};
let active = null;
let rafId = 0;
let pendingWidth = 0;
table.querySelectorAll('.col-resizer').forEach((handle) => {
handle.addEventListener('pointerdown', (event) => {
event.preventDefault();
event.stopPropagation();
const key = handle.getAttribute('data-col');
const col = key ? colByKey[key] : null;
if (!key || !col) return;
const th = handle.closest('th');
const startWidth = th ? th.getBoundingClientRect().width : parseInt(col.style.width || '0', 10) || 120;
active = { key, col, startX: event.clientX, startWidth };
document.body.classList.add('is-resizing');
handle.setPointerCapture(event.pointerId);
});
handle.addEventListener('pointermove', (event) => {
if (!active) return;
const minPx = minWidths[active.key] || 120;
const nextWidth = Math.max(minPx, Math.round(active.startWidth + (event.clientX - active.startX)));
pendingWidth = nextWidth;
if (rafId) return;
rafId = requestAnimationFrame(() => {
active.col.style.width = pendingWidth + 'px';
rafId = 0;
});
});
const finish = () => {
if (!active) return;
active = null;
document.body.classList.remove('is-resizing');
persist();
};
handle.addEventListener('pointerup', finish);
handle.addEventListener('pointercancel', finish);
handle.addEventListener('dblclick', (event) => {
event.preventDefault();
event.stopPropagation();
const key = handle.getAttribute('data-col');
const col = key ? colByKey[key] : null;
const px = key ? defaultWidths[key] : null;
if (!key || !col || !px) return;
col.style.width = px + 'px';
persist();
});
});
}
const DELETE_CONFIRM_LABEL = 'Confirm delete';
const DELETE_LOADING_LABEL = 'Deleting...';
const DELETE_CONFIRM_TIMEOUT_MS = 4000;
function getDeleteView() {
return new URL(window.location.href).searchParams.get('view') || 'list';
}
function resetDeleteButton(buttonEl) {
if (!buttonEl) return;
buttonEl.classList.remove('is-confirming');
buttonEl.removeAttribute('data-confirming');
buttonEl.disabled = false;
const original = buttonEl.dataset.originalLabel || (buttonEl.textContent || '').trim() || 'Delete';
buttonEl.innerHTML = original;
}
function animateRowRemoval(row, onDone) {
if (!row) {
if (onDone) onDone();
return;
}
const isListItem = row.tagName.toLowerCase() === 'li';
if (isListItem) {
row.style.maxHeight = row.getBoundingClientRect().height + 'px';
row.style.overflow = 'hidden';
}
row.classList.add('is-removing');
requestAnimationFrame(() => {
if (isListItem) {
row.style.maxHeight = '0px';
row.style.marginTop = '0px';
row.style.marginBottom = '0px';
row.style.paddingTop = '0px';
row.style.paddingBottom = '0px';
}
});
window.setTimeout(() => {
row.remove();
if (onDone) onDone();
}, 240);
}
async function deleteFeedRequest(feedId, view) {
const res = await fetch('/admin/feeds/' + encodeURIComponent(feedId) + '/delete?view=' + encodeURIComponent(view), {
method: 'POST',
headers: {
'Accept': 'application/json',
},
credentials: 'same-origin',
});
const data = await res.json().catch(() => ({}));
if (!res.ok) {
const message = data && data.error ? String(data.error) : ('Request failed (' + res.status + ')');
throw new Error(message);
}
return data;
}
function refreshFeedRowCache() {
FEED_ROWS = Array.from(document.querySelectorAll('.feed-row'));
FEED_CHECKBOXES = Array.from(document.querySelectorAll('.feed-select'));
if (FEED_TOTAL_COUNT_EL) {
FEED_TOTAL_COUNT_EL.textContent = String(FEED_ROWS.length);
}
updateFeedMatchCount();
updateFeedSelectionState();
}
function setupFeedDeleteButtons() {
const buttons = Array.from(document.querySelectorAll('button[data-delete-kind="feed"]'));
buttons.forEach((button) => {
if (button.dataset.deleteReady === 'true') return;
button.dataset.deleteReady = 'true';
const original = (button.textContent || '').trim() || 'Delete';
button.dataset.originalLabel = original;
let confirming = false;
let confirmTimer = 0;
let inFlight = false;
const startConfirm = () => {
confirming = true;
button.classList.add('is-confirming');
button.setAttribute('data-confirming', 'true');
button.innerHTML = DELETE_CONFIRM_LABEL;
if (confirmTimer) window.clearTimeout(confirmTimer);
confirmTimer = window.setTimeout(() => {
confirming = false;
resetDeleteButton(button);
}, DELETE_CONFIRM_TIMEOUT_MS);
};
button.addEventListener('click', async (event) => {
event.preventDefault();
if (inFlight) return;
if (!confirming) {
startConfirm();
return;
}
if (confirmTimer) window.clearTimeout(confirmTimer);
inFlight = true;
setButtonLoading(button, true, DELETE_LOADING_LABEL);
const toast = window.showToast
? window.showToast('Deleting feed...', { type: 'info', loading: true, duration: 0 })
: null;
const feedId = button.getAttribute('data-feed-id') || '';
const view = button.getAttribute('data-view') || getDeleteView();
const row = button.closest('.feed-row');
try {
await deleteFeedRequest(feedId, view);
if (toast && toast.update) {
toast.update('Feed deleted.', { type: 'success', loading: false, duration: 3200 });
} else if (window.showToast) {
window.showToast('Feed deleted.', { type: 'success' });
}
animateRowRemoval(row, () => {
refreshFeedRowCache();
});
} catch (error) {
if (toast && toast.update) {
toast.update('Delete failed: ' + (error && error.message ? error.message : 'Unknown error'), { type: 'error', loading: false });
} else if (window.showToast) {
window.showToast('Delete failed: ' + (error && error.message ? error.message : 'Unknown error'), { type: 'error' });
}
setButtonLoading(button, false);
confirming = false;
resetDeleteButton(button);
} finally {
inFlight = false;
if (!row) {
setButtonLoading(button, false);
confirming = false;
resetDeleteButton(button);
}
}
});
button.addEventListener('keydown', (event) => {
if (event.key === 'Escape' && confirming && !inFlight) {
confirming = false;
if (confirmTimer) window.clearTimeout(confirmTimer);
resetDeleteButton(button);
}
});
});
}
function updateFeedSelectionState() {
if (!FEED_CHECKBOXES.length) {
return;
}
const selected = FEED_CHECKBOXES.filter((checkbox) => checkbox.checked);
if (FEED_SELECTED_COUNT_EL) {
FEED_SELECTED_COUNT_EL.textContent = selected.length + ' selected';
}
if (FEED_BULK_DELETE_BUTTON_EL) {
FEED_BULK_DELETE_BUTTON_EL.disabled = selected.length === 0;
}
if (FEED_SELECT_ALL_EL) {
const visibleCheckboxes = FEED_CHECKBOXES.filter((checkbox) => !(checkbox.closest('tr')?.hidden));
FEED_SELECT_ALL_EL.checked = visibleCheckboxes.length > 0 && visibleCheckboxes.every((checkbox) => checkbox.checked);
}
}
function toggleAllFeeds(checked) {
FEED_CHECKBOXES.forEach((checkbox) => {
if (!checkbox.closest('tr')?.hidden) {
checkbox.checked = checked;
}
})
updateFeedSelectionState();
}
function setVisibleFeedSelection(checked) {
FEED_CHECKBOXES.forEach((checkbox) => {
if (!checkbox.closest('tr')?.hidden) {
checkbox.checked = checked;
}
})
updateFeedSelectionState();
}
function selectMatchingFeeds() {
setVisibleFeedSelection(true);
}
function clearFeedSelection() {
FEED_CHECKBOXES.forEach((checkbox) => {
checkbox.checked = false;
})
updateFeedSelectionState();
}
function filterFeedRows() {
const query = (document.getElementById('feed-search')?.value || '').toLowerCase().trim();
FEED_ROWS.forEach((row) => {
const haystack = row.getAttribute('data-search') || '';
row.hidden = !!query && !haystack.includes(query);
});
updateFeedMatchCount();
updateFeedSelectionState();
}
function confirmBulkFeedDelete() {
const selected = FEED_CHECKBOXES.filter((checkbox) => checkbox.checked).length;
if (selected === 0) return false;
const query = (document.getElementById('feed-search')?.value || '').trim();
const extra =
selected >= 50 && !query
? '\\n\\nThis is a large delete. Tip: use Search to narrow down spam first.'
: '';
return confirm(
'Delete ' +
selected +
' selected feed(s)? This disables the feeds immediately. Stored emails are cleaned up best-effort and may take a while.' +
extra,
);
}
function setButtonLoading(buttonEl, loading, label) {
if (!buttonEl) return;
if (loading) {
if (!buttonEl.dataset.originalLabel) {
buttonEl.dataset.originalLabel = (buttonEl.textContent || '').trim();
}
const text = label || 'Working...';
buttonEl.classList.add('is-loading');
buttonEl.disabled = true;
buttonEl.innerHTML = '<span class="spinner" aria-hidden="true"></span>' + text;
return;
}
const original = buttonEl.dataset.originalLabel || (buttonEl.textContent || '').trim();
buttonEl.classList.remove('is-loading');
buttonEl.innerHTML = original;
}
function removeFeedRowsById(feedIds) {
const toRemove = new Set((feedIds || []).map((v) => String(v)));
if (toRemove.size === 0) return;
FEED_ROWS.forEach((row) => {
const checkbox = row.querySelector('input.feed-select');
const id = checkbox ? checkbox.value : '';
if (toRemove.has(id)) {
row.remove();
}
});
FEED_ROWS = Array.from(document.querySelectorAll('.feed-row'));
FEED_CHECKBOXES = Array.from(document.querySelectorAll('.feed-select'));
if (FEED_TOTAL_COUNT_EL) {
FEED_TOTAL_COUNT_EL.textContent = String(FEED_ROWS.length);
}
}
function onBulkFeedDeleteSubmit(event) {
if (event && event.preventDefault) event.preventDefault();
void bulkDeleteSelectedFeeds();
return false;
}
async function bulkDeleteSelectedFeeds() {
if (FEED_BULK_DELETE_IN_PROGRESS) return;
const selectedIds = FEED_CHECKBOXES.filter((checkbox) => checkbox.checked).map((checkbox) => checkbox.value);
if (selectedIds.length === 0) {
if (window.showToast) window.showToast('No feeds selected.', { type: 'info' });
return;
}
if (!confirmBulkFeedDelete()) {
return;
}
FEED_BULK_DELETE_IN_PROGRESS = true;
setButtonLoading(FEED_BULK_DELETE_BUTTON_EL, true, 'Deleting...');
const toast = window.showToast
? window.showToast('Deleting ' + selectedIds.length + ' feed(s)...', { type: 'info', loading: true, duration: 0 })
: null;
const batchSize = 10;
let deletedTotal = 0;
const failed = [];
try {
for (let i = 0; i < selectedIds.length; i += batchSize) {
const batch = selectedIds.slice(i, i + batchSize);
const res = await fetch('/admin/feeds/bulk-delete', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'Accept': 'application/json',
},
credentials: 'same-origin',
body: JSON.stringify({ feedIds: batch }),
});
let data = {};
if (window.parseJsonResponseOrThrow) {
data = await window.parseJsonResponseOrThrow(res, { prefix: 'Bulk feed delete failed' });
} else {
data = await res.json().catch(() => ({}));
if (!res.ok) {
const message = data && data.error ? String(data.error) : ('Bulk feed delete failed (HTTP ' + res.status + ')');
throw new Error(message);
}
}
const deletedIds = Array.isArray(data.deletedFeedIds) ? data.deletedFeedIds : batch;
const failedIds = Array.isArray(data.failedFeedIds) ? data.failedFeedIds : [];
const failureDetails = Array.isArray(data.failures) ? data.failures : [];
removeFeedRowsById(deletedIds);
deletedTotal += deletedIds.length;
if (toast && toast.update) {
const done = Math.min(i + batch.length, selectedIds.length);
toast.update('Deleting... (' + done + ' of ' + selectedIds.length + ')', { type: 'info' });
}
// Keep selection state consistent as rows disappear.
updateFeedMatchCount();
updateFeedSelectionState();
// If a batch fails for some feeds, retry those one-by-one using the bulk-delete
// endpoint with a single id (keeps semantics consistent and avoids hiding active feeds).
if (failedIds.length > 0) {
if (toast && toast.update) {
toast.update('Retrying ' + failedIds.length + ' failed feed(s) one-by-one...', { type: 'info' });
}
const stillFailed = [];
for (let j = 0; j < failedIds.length; j++) {
const feedId = String(failedIds[j] || '');
if (!feedId) continue;
try {
const retryRes = await fetch('/admin/feeds/bulk-delete', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'Accept': 'application/json',
},
credentials: 'same-origin',
body: JSON.stringify({ feedIds: [feedId] }),
});
let retryData = {};
if (window.parseJsonResponseOrThrow) {
retryData = await window.parseJsonResponseOrThrow(retryRes, { prefix: 'Retry delete failed' });
} else {
retryData = await retryRes.json().catch(() => ({}));
if (!retryRes.ok) {
const message = retryData && retryData.error ? String(retryData.error) : ('Retry delete failed (HTTP ' + retryRes.status + ')');
throw new Error(message);
}
}
const retryDeleted = Array.isArray(retryData.deletedFeedIds) ? retryData.deletedFeedIds : [];
const retryFailed = Array.isArray(retryData.failedFeedIds) ? retryData.failedFeedIds : [];
if (retryDeleted.includes(feedId)) {
removeFeedRowsById([feedId]);
deletedTotal += 1;
} else if (retryFailed.includes(feedId)) {
stillFailed.push(feedId);
} else {
stillFailed.push(feedId);
}
} catch (e) {
stillFailed.push(feedId);
}
if (toast && toast.update) {
toast.update('Retrying... (' + (j + 1) + ' of ' + failedIds.length + ')', { type: 'info' });
}
}
// Replace failed ids from this batch with only the ones that still failed after retry.
if (stillFailed.length > 0) {
failed.push(...stillFailed);
if (window.showToast && failureDetails.length > 0) {
const first = failureDetails[0] && failureDetails[0].error ? String(failureDetails[0].error) : '';
if (first) {
window.showToast('Some feeds failed to delete: ' + first, { type: 'error' });
}
}
}
updateFeedMatchCount();
updateFeedSelectionState();
}
}
if (toast && toast.dismiss) toast.dismiss();
const uniqueFailed = Array.from(new Set(failed.map((v) => String(v)).filter(Boolean)));
if (uniqueFailed.length > 0) {
if (window.showToast) {
window.showToast(
'Deleted ' + deletedTotal + ' feed(s). ' + uniqueFailed.length + ' failed (still visible).',
{ type: 'error' },
);
}
} else {
if (window.showToast) window.showToast('Deleted ' + deletedTotal + ' feed(s).', { type: 'success' });
}
} catch (error) {
if (toast && toast.dismiss) toast.dismiss();
if (window.showToast) {
window.showToast((error && error.message) ? error.message : 'Bulk feed delete failed.', { type: 'error' });
}
} finally {
FEED_BULK_DELETE_IN_PROGRESS = false;
setButtonLoading(FEED_BULK_DELETE_BUTTON_EL, false);
updateFeedSelectionState();
}
}
document.addEventListener('DOMContentLoaded', () => {
initFeedUI();
});
`;
// dashboardScript is compiled from src/scripts/client/dashboard.ts via `npm run build:client`.
// It is imported from src/scripts/generated/dashboard.ts above.
// ── Shared SVG icons ──────────────────────────────────────────────────────────
+831
View File
@@ -0,0 +1,831 @@
// Client-side script for the admin dashboard (feeds table).
// Compiled by scripts/build-client.mjs — do not import DOM types from here in Worker code.
let FEED_ROWS: HTMLElement[] = [];
let FEED_CHECKBOXES: HTMLInputElement[] = [];
let FEED_SELECTED_COUNT_EL: HTMLElement | null = null;
let FEED_MATCH_COUNT_EL: HTMLElement | null = null;
let FEED_TOTAL_COUNT_EL: HTMLElement | null = null;
let FEED_BULK_DELETE_BUTTON_EL: HTMLButtonElement | null = null;
let FEED_SELECT_ALL_EL: HTMLInputElement | null = null;
let FEED_FILTER_TIMER: ReturnType<typeof setTimeout> | null = null;
let FEED_BULK_DELETE_IN_PROGRESS = false;
let FEED_SORT_KEY = "title";
let FEED_SORT_DIR = "asc";
const FEED_COLLATOR = new Intl.Collator(undefined, {
numeric: true,
sensitivity: "base",
});
function initFeedUI(): void {
FEED_ROWS = Array.from(document.querySelectorAll<HTMLElement>(".feed-row"));
FEED_CHECKBOXES = Array.from(
document.querySelectorAll<HTMLInputElement>(".feed-select"),
);
FEED_SELECTED_COUNT_EL = document.getElementById("selected-feed-count");
FEED_MATCH_COUNT_EL = document.getElementById("feed-match-count");
FEED_TOTAL_COUNT_EL = document.getElementById("feed-total-count");
FEED_BULK_DELETE_BUTTON_EL = document.getElementById(
"bulk-delete-feeds-button",
) as HTMLButtonElement | null;
FEED_SELECT_ALL_EL = document.getElementById(
"select-all-feeds",
) as HTMLInputElement | null;
setupFeedTableResizing();
setupFeedTableSorting();
setupFeedDeleteButtons();
updateFeedMatchCount();
updateFeedSelectionState();
}
function updateFeedMatchCount(): void {
if (!FEED_MATCH_COUNT_EL) return;
const total = FEED_ROWS.length;
const visible = FEED_ROWS.filter((row) => !row.hidden).length;
const query = (
(document.getElementById("feed-search") as HTMLInputElement | null)
?.value || ""
).trim();
FEED_MATCH_COUNT_EL.textContent = query
? "Showing " + visible + " of " + total
: "Showing " + total;
}
function scheduleFeedFilter(): void {
if (FEED_FILTER_TIMER) {
clearTimeout(FEED_FILTER_TIMER);
}
FEED_FILTER_TIMER = setTimeout(filterFeedRows, 120);
}
function getSortValue(row: HTMLElement, key: string): string {
const prop = "sort" + key.charAt(0).toUpperCase() + key.slice(1);
return row.dataset && row.dataset[prop] ? row.dataset[prop]! : "";
}
function updateFeedSortIndicators(table: Element): void {
const headerCells = Array.from(
table.querySelectorAll<HTMLElement>("th[data-sort-key]"),
);
headerCells.forEach((th) => {
const key = th.getAttribute("data-sort-key") || "";
const indicator = th.querySelector(".sort-indicator");
const active = key === FEED_SORT_KEY;
if (indicator) {
indicator.textContent = active
? FEED_SORT_DIR === "asc"
? "^"
: "v"
: "";
}
th.setAttribute(
"aria-sort",
active ? (FEED_SORT_DIR === "asc" ? "ascending" : "descending") : "none",
);
});
}
function sortFeedTableBy(key: string): void {
const table = document.querySelector("table.table-feeds");
const tbody = document.getElementById("feed-table-body");
if (!table || !tbody) return;
if (FEED_SORT_KEY === key) {
FEED_SORT_DIR = FEED_SORT_DIR === "asc" ? "desc" : "asc";
} else {
FEED_SORT_KEY = key;
FEED_SORT_DIR = "asc";
}
const dirMultiplier = FEED_SORT_DIR === "asc" ? 1 : -1;
const rows = Array.from(tbody.querySelectorAll<HTMLElement>(".feed-row"));
rows.sort((a, b) => {
const av = getSortValue(a, FEED_SORT_KEY);
const bv = getSortValue(b, FEED_SORT_KEY);
return dirMultiplier * FEED_COLLATOR.compare(av, bv);
});
const fragment = document.createDocumentFragment();
rows.forEach((row) => fragment.appendChild(row));
tbody.appendChild(fragment);
updateFeedSortIndicators(table);
}
function setupFeedTableSorting(): void {
const table = document.querySelector("table.table-feeds");
if (!table) return;
table
.querySelectorAll<HTMLButtonElement>("button.th-button[data-sort-key]")
.forEach((button) => {
button.addEventListener("click", () => {
const key = button.getAttribute("data-sort-key") || "";
if (!key) return;
sortFeedTableBy(key);
});
});
updateFeedSortIndicators(table);
}
function setupFeedTableResizing(): void {
const table = document.querySelector("table.table-feeds");
if (!table) return;
const storageKey = "email-to-rss.admin.feedsTable.colWidths";
const minWidths: Record<string, number> = {
title: 220,
feedId: 120,
email: 160,
rss: 160,
actions: 160,
};
const defaultWidths: Record<string, number> = {
title: 340,
feedId: 160,
email: 220,
rss: 220,
actions: 200,
};
const cols = Array.from(table.querySelectorAll<HTMLElement>("colgroup col"));
const colByKey: Record<string, HTMLElement> = {};
cols.forEach((col) => {
const key = col.getAttribute("data-col");
if (key) colByKey[key] = col;
});
// Restore widths
try {
const saved = JSON.parse(
localStorage.getItem(storageKey) || "{}",
) as Record<string, unknown>;
Object.keys(saved || {}).forEach((key) => {
const px = Number(saved[key]);
if (!colByKey[key] || !Number.isFinite(px)) return;
colByKey[key].style.width = px + "px";
});
} catch {
// Ignore bad localStorage values
}
const persist = () => {
try {
const out: Record<string, number> = {};
Object.keys(colByKey).forEach((key) => {
if (key === "select") return;
const px = parseInt(colByKey[key].style.width || "0", 10);
if (Number.isFinite(px) && px > 0) out[key] = px;
});
localStorage.setItem(storageKey, JSON.stringify(out));
} catch {
// localStorage may be unavailable in some modes; ignore
}
};
type ActiveResize = {
key: string;
col: HTMLElement;
startX: number;
startWidth: number;
};
let active: ActiveResize | null = null;
let rafId = 0;
let pendingWidth = 0;
table.querySelectorAll<HTMLElement>(".col-resizer").forEach((handle) => {
handle.addEventListener("pointerdown", (event: PointerEvent) => {
event.preventDefault();
event.stopPropagation();
const key = handle.getAttribute("data-col");
const col = key ? colByKey[key] : null;
if (!key || !col) return;
const th = handle.closest("th");
const startWidth = th
? th.getBoundingClientRect().width
: parseInt(col.style.width || "0", 10) || 120;
active = { key, col, startX: event.clientX, startWidth };
document.body.classList.add("is-resizing");
handle.setPointerCapture(event.pointerId);
});
handle.addEventListener("pointermove", (event: PointerEvent) => {
if (!active) return;
const minPx = minWidths[active.key] || 120;
const nextWidth = Math.max(
minPx,
Math.round(active.startWidth + (event.clientX - active.startX)),
);
pendingWidth = nextWidth;
if (rafId) return;
rafId = requestAnimationFrame(() => {
active!.col.style.width = pendingWidth + "px";
rafId = 0;
});
});
const finish = () => {
if (!active) return;
active = null;
document.body.classList.remove("is-resizing");
persist();
};
handle.addEventListener("pointerup", finish);
handle.addEventListener("pointercancel", finish);
handle.addEventListener("dblclick", (event: MouseEvent) => {
event.preventDefault();
event.stopPropagation();
const key = handle.getAttribute("data-col");
const col = key ? colByKey[key] : null;
const px = key ? defaultWidths[key] : null;
if (!key || !col || !px) return;
col.style.width = px + "px";
persist();
});
});
}
const DELETE_CONFIRM_LABEL = "Confirm delete";
const DELETE_LOADING_LABEL = "Deleting...";
const DELETE_CONFIRM_TIMEOUT_MS = 4000;
function getDeleteView(): string {
return new URL(window.location.href).searchParams.get("view") || "list";
}
function resetDeleteButton(buttonEl: HTMLButtonElement): void {
if (!buttonEl) return;
buttonEl.classList.remove("is-confirming");
buttonEl.removeAttribute("data-confirming");
buttonEl.disabled = false;
const original =
buttonEl.dataset.originalLabel ||
(buttonEl.textContent || "").trim() ||
"Delete";
buttonEl.innerHTML = original;
}
function animateRowRemoval(row: HTMLElement | null, onDone?: () => void): void {
if (!row) {
if (onDone) onDone();
return;
}
const isListItem = row.tagName.toLowerCase() === "li";
if (isListItem) {
row.style.maxHeight = row.getBoundingClientRect().height + "px";
row.style.overflow = "hidden";
}
row.classList.add("is-removing");
requestAnimationFrame(() => {
if (isListItem) {
row.style.maxHeight = "0px";
row.style.marginTop = "0px";
row.style.marginBottom = "0px";
row.style.paddingTop = "0px";
row.style.paddingBottom = "0px";
}
});
window.setTimeout(() => {
row.remove();
if (onDone) onDone();
}, 240);
}
async function deleteFeedRequest(
feedId: string,
view: string,
): Promise<unknown> {
const res = await fetch(
"/admin/feeds/" +
encodeURIComponent(feedId) +
"/delete?view=" +
encodeURIComponent(view),
{
method: "POST",
headers: {
Accept: "application/json",
},
credentials: "same-origin",
},
);
const data = (await res.json().catch(() => ({}))) as Record<string, unknown>;
if (!res.ok) {
const message =
data && data.error
? String(data.error)
: "Request failed (" + res.status + ")";
throw new Error(message);
}
return data;
}
function refreshFeedRowCache(): void {
FEED_ROWS = Array.from(document.querySelectorAll<HTMLElement>(".feed-row"));
FEED_CHECKBOXES = Array.from(
document.querySelectorAll<HTMLInputElement>(".feed-select"),
);
if (FEED_TOTAL_COUNT_EL) {
FEED_TOTAL_COUNT_EL.textContent = String(FEED_ROWS.length);
}
updateFeedMatchCount();
updateFeedSelectionState();
}
interface ToastHandle {
update?: (msg: string, opts?: Record<string, unknown>) => void;
dismiss?: () => void;
}
declare global {
interface Window {
showToast?: (msg: string, opts?: Record<string, unknown>) => ToastHandle;
parseJsonResponseOrThrow?: (
res: Response,
opts?: Record<string, unknown>,
) => Promise<Record<string, unknown>>;
}
}
function setupFeedDeleteButtons(): void {
const buttons = Array.from(
document.querySelectorAll<HTMLButtonElement>(
'button[data-delete-kind="feed"]',
),
);
buttons.forEach((button) => {
if (button.dataset.deleteReady === "true") return;
button.dataset.deleteReady = "true";
const original = (button.textContent || "").trim() || "Delete";
button.dataset.originalLabel = original;
let confirming = false;
let confirmTimer = 0;
let inFlight = false;
const startConfirm = () => {
confirming = true;
button.classList.add("is-confirming");
button.setAttribute("data-confirming", "true");
button.innerHTML = DELETE_CONFIRM_LABEL;
if (confirmTimer) window.clearTimeout(confirmTimer);
confirmTimer = window.setTimeout(() => {
confirming = false;
resetDeleteButton(button);
}, DELETE_CONFIRM_TIMEOUT_MS);
};
button.addEventListener("click", async (event: MouseEvent) => {
event.preventDefault();
if (inFlight) return;
if (!confirming) {
startConfirm();
return;
}
if (confirmTimer) window.clearTimeout(confirmTimer);
inFlight = true;
setButtonLoading(button, true, DELETE_LOADING_LABEL);
const toast = window.showToast
? window.showToast("Deleting feed...", {
type: "info",
loading: true,
duration: 0,
})
: null;
const feedId = button.getAttribute("data-feed-id") || "";
const view = button.getAttribute("data-view") || getDeleteView();
const row = button.closest<HTMLElement>(".feed-row");
try {
await deleteFeedRequest(feedId, view);
if (toast && toast.update) {
toast.update("Feed deleted.", {
type: "success",
loading: false,
duration: 3200,
});
} else if (window.showToast) {
window.showToast("Feed deleted.", { type: "success" });
}
animateRowRemoval(row, () => {
refreshFeedRowCache();
});
} catch (error) {
const errMsg =
error instanceof Error && error.message
? error.message
: "Unknown error";
if (toast && toast.update) {
toast.update("Delete failed: " + errMsg, {
type: "error",
loading: false,
});
} else if (window.showToast) {
window.showToast("Delete failed: " + errMsg, { type: "error" });
}
setButtonLoading(button, false);
confirming = false;
resetDeleteButton(button);
} finally {
inFlight = false;
if (!row) {
setButtonLoading(button, false);
confirming = false;
resetDeleteButton(button);
}
}
});
button.addEventListener("keydown", (event: KeyboardEvent) => {
if (event.key === "Escape" && confirming && !inFlight) {
confirming = false;
if (confirmTimer) window.clearTimeout(confirmTimer);
resetDeleteButton(button);
}
});
});
}
function updateFeedSelectionState(): void {
if (!FEED_CHECKBOXES.length) {
return;
}
const selected = FEED_CHECKBOXES.filter((checkbox) => checkbox.checked);
if (FEED_SELECTED_COUNT_EL) {
FEED_SELECTED_COUNT_EL.textContent = selected.length + " selected";
}
if (FEED_BULK_DELETE_BUTTON_EL) {
FEED_BULK_DELETE_BUTTON_EL.disabled = selected.length === 0;
}
if (FEED_SELECT_ALL_EL) {
const visibleCheckboxes = FEED_CHECKBOXES.filter(
(checkbox) => !(checkbox.closest("tr") as HTMLElement | null)?.hidden,
);
FEED_SELECT_ALL_EL.checked =
visibleCheckboxes.length > 0 &&
visibleCheckboxes.every((checkbox) => checkbox.checked);
}
}
function toggleAllFeeds(checked: boolean): void {
FEED_CHECKBOXES.forEach((checkbox) => {
if (!(checkbox.closest("tr") as HTMLElement | null)?.hidden) {
checkbox.checked = checked;
}
});
updateFeedSelectionState();
}
function setVisibleFeedSelection(checked: boolean): void {
FEED_CHECKBOXES.forEach((checkbox) => {
if (!(checkbox.closest("tr") as HTMLElement | null)?.hidden) {
checkbox.checked = checked;
}
});
updateFeedSelectionState();
}
function selectMatchingFeeds(): void {
setVisibleFeedSelection(true);
}
function clearFeedSelection(): void {
FEED_CHECKBOXES.forEach((checkbox) => {
checkbox.checked = false;
});
updateFeedSelectionState();
}
function filterFeedRows(): void {
const query = (
(document.getElementById("feed-search") as HTMLInputElement | null)
?.value || ""
)
.toLowerCase()
.trim();
FEED_ROWS.forEach((row) => {
const haystack = row.getAttribute("data-search") || "";
row.hidden = !!query && !haystack.includes(query);
});
updateFeedMatchCount();
updateFeedSelectionState();
}
function confirmBulkFeedDelete(): boolean {
const selected = FEED_CHECKBOXES.filter(
(checkbox) => checkbox.checked,
).length;
if (selected === 0) return false;
const query = (
(document.getElementById("feed-search") as HTMLInputElement | null)
?.value || ""
).trim();
const extra =
selected >= 50 && !query
? "\n\nThis is a large delete. Tip: use Search to narrow down spam first."
: "";
return confirm(
"Delete " +
selected +
" selected feed(s)? This disables the feeds immediately. Stored emails are cleaned up best-effort and may take a while." +
extra,
);
}
function setButtonLoading(
buttonEl: HTMLButtonElement | null,
loading: boolean,
label?: string,
): void {
if (!buttonEl) return;
if (loading) {
if (!buttonEl.dataset.originalLabel) {
buttonEl.dataset.originalLabel = (buttonEl.textContent || "").trim();
}
const text = label || "Working...";
buttonEl.classList.add("is-loading");
buttonEl.disabled = true;
buttonEl.innerHTML =
'<span class="spinner" aria-hidden="true"></span>' + text;
return;
}
const original =
buttonEl.dataset.originalLabel || (buttonEl.textContent || "").trim();
buttonEl.classList.remove("is-loading");
buttonEl.innerHTML = original;
}
function removeFeedRowsById(feedIds: string[]): void {
const toRemove = new Set((feedIds || []).map((v) => String(v)));
if (toRemove.size === 0) return;
FEED_ROWS.forEach((row) => {
const checkbox = row.querySelector<HTMLInputElement>("input.feed-select");
const id = checkbox ? checkbox.value : "";
if (toRemove.has(id)) {
row.remove();
}
});
FEED_ROWS = Array.from(document.querySelectorAll<HTMLElement>(".feed-row"));
FEED_CHECKBOXES = Array.from(
document.querySelectorAll<HTMLInputElement>(".feed-select"),
);
if (FEED_TOTAL_COUNT_EL) {
FEED_TOTAL_COUNT_EL.textContent = String(FEED_ROWS.length);
}
}
function onBulkFeedDeleteSubmit(event: Event): boolean {
if (event && event.preventDefault) event.preventDefault();
void bulkDeleteSelectedFeeds();
return false;
}
async function bulkDeleteSelectedFeeds(): Promise<void> {
if (FEED_BULK_DELETE_IN_PROGRESS) return;
const selectedIds = FEED_CHECKBOXES.filter(
(checkbox) => checkbox.checked,
).map((checkbox) => checkbox.value);
if (selectedIds.length === 0) {
if (window.showToast)
window.showToast("No feeds selected.", { type: "info" });
return;
}
if (!confirmBulkFeedDelete()) {
return;
}
FEED_BULK_DELETE_IN_PROGRESS = true;
setButtonLoading(FEED_BULK_DELETE_BUTTON_EL, true, "Deleting...");
const toast = window.showToast
? window.showToast("Deleting " + selectedIds.length + " feed(s)...", {
type: "info",
loading: true,
duration: 0,
})
: null;
const batchSize = 10;
let deletedTotal = 0;
const failed: string[] = [];
try {
for (let i = 0; i < selectedIds.length; i += batchSize) {
const batch = selectedIds.slice(i, i + batchSize);
const res = await fetch("/admin/feeds/bulk-delete", {
method: "POST",
headers: {
"Content-Type": "application/json",
Accept: "application/json",
},
credentials: "same-origin",
body: JSON.stringify({ feedIds: batch }),
});
let data: Record<string, unknown> = {};
if (window.parseJsonResponseOrThrow) {
data = await window.parseJsonResponseOrThrow(res, {
prefix: "Bulk feed delete failed",
});
} else {
data = (await res.json().catch(() => ({}))) as Record<string, unknown>;
if (!res.ok) {
const message =
data && data.error
? String(data.error)
: "Bulk feed delete failed (HTTP " + res.status + ")";
throw new Error(message);
}
}
const deletedIds = Array.isArray(data.deletedFeedIds)
? (data.deletedFeedIds as string[])
: batch;
const failedIds = Array.isArray(data.failedFeedIds)
? (data.failedFeedIds as string[])
: [];
const failureDetails = Array.isArray(data.failures)
? (data.failures as Array<Record<string, unknown>>)
: [];
removeFeedRowsById(deletedIds);
deletedTotal += deletedIds.length;
if (toast && toast.update) {
const done = Math.min(i + batch.length, selectedIds.length);
toast.update(
"Deleting... (" + done + " of " + selectedIds.length + ")",
{ type: "info" },
);
}
// Keep selection state consistent as rows disappear.
updateFeedMatchCount();
updateFeedSelectionState();
// If a batch fails for some feeds, retry those one-by-one using the bulk-delete
// endpoint with a single id (keeps semantics consistent and avoids hiding active feeds).
if (failedIds.length > 0) {
if (toast && toast.update) {
toast.update(
"Retrying " + failedIds.length + " failed feed(s) one-by-one...",
{ type: "info" },
);
}
const stillFailed: string[] = [];
for (let j = 0; j < failedIds.length; j++) {
const feedId = String(failedIds[j] || "");
if (!feedId) continue;
try {
const retryRes = await fetch("/admin/feeds/bulk-delete", {
method: "POST",
headers: {
"Content-Type": "application/json",
Accept: "application/json",
},
credentials: "same-origin",
body: JSON.stringify({ feedIds: [feedId] }),
});
let retryData: Record<string, unknown> = {};
if (window.parseJsonResponseOrThrow) {
retryData = await window.parseJsonResponseOrThrow(retryRes, {
prefix: "Retry delete failed",
});
} else {
retryData = (await retryRes.json().catch(() => ({}))) as Record<
string,
unknown
>;
if (!retryRes.ok) {
const message =
retryData && retryData.error
? String(retryData.error)
: "Retry delete failed (HTTP " + retryRes.status + ")";
throw new Error(message);
}
}
const retryDeleted = Array.isArray(retryData.deletedFeedIds)
? (retryData.deletedFeedIds as string[])
: [];
const retryFailed = Array.isArray(retryData.failedFeedIds)
? (retryData.failedFeedIds as string[])
: [];
if (retryDeleted.includes(feedId)) {
removeFeedRowsById([feedId]);
deletedTotal += 1;
} else if (retryFailed.includes(feedId)) {
stillFailed.push(feedId);
} else {
stillFailed.push(feedId);
}
} catch {
stillFailed.push(feedId);
}
if (toast && toast.update) {
toast.update(
"Retrying... (" + (j + 1) + " of " + failedIds.length + ")",
{ type: "info" },
);
}
}
// Replace failed ids from this batch with only the ones that still failed after retry.
if (stillFailed.length > 0) {
failed.push(...stillFailed);
if (window.showToast && failureDetails.length > 0) {
const first =
failureDetails[0] && failureDetails[0].error
? String(failureDetails[0].error)
: "";
if (first) {
window.showToast("Some feeds failed to delete: " + first, {
type: "error",
});
}
}
}
updateFeedMatchCount();
updateFeedSelectionState();
}
}
if (toast && toast.dismiss) toast.dismiss();
const uniqueFailed = Array.from(
new Set(failed.map((v) => String(v)).filter(Boolean)),
);
if (uniqueFailed.length > 0) {
if (window.showToast) {
window.showToast(
"Deleted " +
deletedTotal +
" feed(s). " +
uniqueFailed.length +
" failed (still visible).",
{ type: "error" },
);
}
} else {
if (window.showToast)
window.showToast("Deleted " + deletedTotal + " feed(s).", {
type: "success",
});
}
} catch (error) {
if (toast && toast.dismiss) toast.dismiss();
if (window.showToast) {
window.showToast(
error instanceof Error && error.message
? error.message
: "Bulk feed delete failed.",
{ type: "error" },
);
}
} finally {
FEED_BULK_DELETE_IN_PROGRESS = false;
setButtonLoading(FEED_BULK_DELETE_BUTTON_EL, false);
updateFeedSelectionState();
}
}
// Expose functions needed by inline HTML event handlers
(window as unknown as Record<string, unknown>).scheduleFeedFilter =
scheduleFeedFilter;
(window as unknown as Record<string, unknown>).toggleAllFeeds = toggleAllFeeds;
(window as unknown as Record<string, unknown>).selectMatchingFeeds =
selectMatchingFeeds;
(window as unknown as Record<string, unknown>).clearFeedSelection =
clearFeedSelection;
(window as unknown as Record<string, unknown>).onBulkFeedDeleteSubmit =
onBulkFeedDeleteSubmit;
document.addEventListener("DOMContentLoaded", () => {
initFeedUI();
});
File diff suppressed because one or more lines are too long