mirror of
https://github.com/juherr/kill-the-news.git
synced 2026-06-20 22:03:48 +00:00
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:
+3
-651
@@ -10,6 +10,7 @@ import { Layout, clampText } from "./admin/ui";
|
|||||||
import { listAllFeeds, updateFeedInList } from "./admin/helpers";
|
import { listAllFeeds, updateFeedInList } from "./admin/helpers";
|
||||||
import { feedsRouter } from "./admin/feeds";
|
import { feedsRouter } from "./admin/feeds";
|
||||||
import { emailsRouter } from "./admin/emails";
|
import { emailsRouter } from "./admin/emails";
|
||||||
|
import { dashboardScript } from "../scripts/generated/dashboard";
|
||||||
|
|
||||||
type AppEnv = { Bindings: Env };
|
type AppEnv = { Bindings: Env };
|
||||||
|
|
||||||
@@ -220,657 +221,8 @@ app.get("/logout", (c) => {
|
|||||||
return c.redirect("/admin/login");
|
return c.redirect("/admin/login");
|
||||||
});
|
});
|
||||||
|
|
||||||
// The large inline script for the dashboard page
|
// dashboardScript is compiled from src/scripts/client/dashboard.ts via `npm run build:client`.
|
||||||
const dashboardScript = `
|
// It is imported from src/scripts/generated/dashboard.ts above.
|
||||||
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();
|
|
||||||
});
|
|
||||||
`;
|
|
||||||
|
|
||||||
// ── Shared SVG icons ──────────────────────────────────────────────────────────
|
// ── Shared SVG icons ──────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
|||||||
@@ -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
Reference in New Issue
Block a user