c85a9c9bc4
- IX-2: Add confirm() dialogs before all destructive actions (10 history clear buttons, scheduler reset, background decode reset) - IX-6: Add Select All / Deselect All buttons for background decode bookmark checklist - IX-1: Add dirty-state indicator (pulsing dot) on Save buttons when unsaved changes exist in scheduler and background decode panels - A-4: Add role="alert" and aria-live="polite" to toast notification elements for screen reader accessibility - A-3: Add Unicode symbol prefixes to background decode state labels (checkmark/triangle/cross) so state is distinguishable without color https://claude.ai/code/session_01ShfPMW9hPLD3czp9YovkbJ Signed-off-by: Claude <noreply@anthropic.com>
208 lines
8.2 KiB
JavaScript
208 lines
8.2 KiB
JavaScript
// --- FT2 Decoder Plugin (server-side decode) ---
|
|
// SPDX-FileCopyrightText: 2026 Stan Grams <sjg@haxx.space>
|
|
// SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
function ft8RenderMessageFt2(message) {
|
|
if (typeof renderFt8Message === "function") return renderFt8Message(message);
|
|
if (typeof ft8EscapeHtml === "function") return ft8EscapeHtml(message);
|
|
return message;
|
|
}
|
|
|
|
const ft2Status = document.getElementById("ft2-status");
|
|
const ft2PeriodEl = document.getElementById("ft2-period");
|
|
const ft2MessagesEl = document.getElementById("ft2-messages");
|
|
const ft2FilterInput = document.getElementById("ft2-filter");
|
|
const FT2_PERIOD_MS = 3750;
|
|
const FT2_MAX_DOM_ROWS = 200;
|
|
let ft2FilterText = "";
|
|
let ft2MessageHistory = [];
|
|
|
|
function currentFt2HistoryRetentionMs() {
|
|
return typeof window.getDecodeHistoryRetentionMs === "function"
|
|
? window.getDecodeHistoryRetentionMs()
|
|
: 24 * 60 * 60 * 1000;
|
|
}
|
|
|
|
function pruneFt2MessageHistory() {
|
|
const cutoffMs = Date.now() - currentFt2HistoryRetentionMs();
|
|
ft2MessageHistory = ft2MessageHistory.filter((msg) => Number(msg?._tsMs ?? msg?.ts_ms) >= cutoffMs);
|
|
}
|
|
|
|
function scheduleFt2Ui(key, job) {
|
|
if (typeof window.trxScheduleUiFrameJob === "function") {
|
|
window.trxScheduleUiFrameJob(key, job);
|
|
return;
|
|
}
|
|
job();
|
|
}
|
|
|
|
function scheduleFt2HistoryRender() { scheduleFt2Ui("ft2-history", () => renderFt2History()); }
|
|
|
|
function normalizeFt2DisplayFreqHz(freqHz) {
|
|
const rawHz = Number(freqHz);
|
|
if (!Number.isFinite(rawHz)) return null;
|
|
const baseHz = Number.isFinite(window.ft8BaseHz) ? Number(window.ft8BaseHz) : null;
|
|
if (Number.isFinite(baseHz) && baseHz > 0 && rawHz >= 0 && rawHz < 100000) {
|
|
return baseHz + rawHz;
|
|
}
|
|
return rawHz;
|
|
}
|
|
|
|
function updateFt2PeriodTimer() {
|
|
if (!ft2PeriodEl) return;
|
|
const nowMs = Date.now();
|
|
const remaining = (FT2_PERIOD_MS - nowMs % FT2_PERIOD_MS) / 1000;
|
|
ft2PeriodEl.textContent = `Next slot ${remaining.toFixed(1)}s`;
|
|
}
|
|
|
|
updateFt2PeriodTimer();
|
|
setInterval(updateFt2PeriodTimer, 250);
|
|
|
|
function renderFt2Row(msg) {
|
|
const row = document.createElement("div");
|
|
row.className = "ft8-row";
|
|
const rawMessage = (msg.message || "").toString();
|
|
row.dataset.message = rawMessage.toUpperCase();
|
|
row.dataset.decoder = "ft2";
|
|
row.dataset.storedFreqHz = Number.isFinite(msg.freq_hz) ? String(msg.freq_hz) : "";
|
|
const snr = Number.isFinite(msg.snr_db) ? msg.snr_db.toFixed(1) : "--";
|
|
const dt = Number.isFinite(msg.dt_s) ? msg.dt_s.toFixed(2) : "--";
|
|
const displayFreqHz = normalizeFt2DisplayFreqHz(msg.freq_hz);
|
|
const freq = Number.isFinite(displayFreqHz) ? displayFreqHz.toFixed(0) : "--";
|
|
const renderedMessage = ft8RenderMessageFt2(rawMessage);
|
|
const tsMs = msg._tsMs ?? msg.ts_ms;
|
|
const timeStr = tsMs ? new Date(tsMs).toLocaleTimeString([], { hour: "2-digit", minute: "2-digit", second: "2-digit" }) : "--:--:--";
|
|
row.innerHTML = `<span class="ft8-time">${timeStr}</span><span class="ft8-snr">${snr}</span><span class="ft8-dt">${dt}</span><span class="ft8-freq">${freq}</span><span class="ft8-msg">${renderedMessage}</span>`;
|
|
return row;
|
|
}
|
|
|
|
function renderFt2History() {
|
|
pruneFt2MessageHistory();
|
|
if (!ft2MessagesEl) return;
|
|
const filter = ft2FilterText;
|
|
const fragment = document.createDocumentFragment();
|
|
let rendered = 0;
|
|
for (let i = 0; i < ft2MessageHistory.length && rendered < FT2_MAX_DOM_ROWS; i++) {
|
|
const msg = ft2MessageHistory[i];
|
|
if (filter && !(msg.message || "").toString().toUpperCase().includes(filter)) continue;
|
|
fragment.appendChild(renderFt2Row(msg));
|
|
rendered++;
|
|
}
|
|
ft2MessagesEl.replaceChildren(fragment);
|
|
}
|
|
|
|
function addFt2Message(msg) {
|
|
msg._tsMs = Number.isFinite(msg?.ts_ms) ? Number(msg.ts_ms) : Date.now();
|
|
ft2MessageHistory.unshift(msg);
|
|
pruneFt2MessageHistory();
|
|
window.setFt8FamilyBarDecoder?.("ft2");
|
|
window.updateFt8Bar?.();
|
|
scheduleFt2HistoryRender();
|
|
}
|
|
|
|
function normalizeServerFt2Message(msg) {
|
|
const raw = (msg.message || "").toString();
|
|
const locatorDetails = typeof ft8ExtractLocatorDetails === "function" ? ft8ExtractLocatorDetails(raw) : [];
|
|
const grids = locatorDetails.length > 0
|
|
? locatorDetails.map((d) => d.grid)
|
|
: (typeof ft8ExtractAllGrids === "function" ? ft8ExtractAllGrids(raw) : []);
|
|
const station = typeof ft8ExtractLikelyCallsign === "function" ? ft8ExtractLikelyCallsign(raw) : null;
|
|
const rfHz = normalizeFt2DisplayFreqHz(msg.freq_hz);
|
|
return {
|
|
raw, grids, station, rfHz, locatorDetails,
|
|
history: {
|
|
receiver: window.getDecodeRigMeta ? window.getDecodeRigMeta() : null,
|
|
ts_ms: msg.ts_ms, snr_db: msg.snr_db, dt_s: msg.dt_s,
|
|
freq_hz: Number.isFinite(rfHz) ? rfHz : msg.freq_hz,
|
|
message: msg.message,
|
|
},
|
|
};
|
|
}
|
|
|
|
window.onServerFt2Batch = function(messages) {
|
|
if (!Array.isArray(messages) || messages.length === 0) return;
|
|
if (ft2Status) ft2Status.textContent = "Receiving";
|
|
const normalized = [];
|
|
for (const msg of messages) {
|
|
const next = normalizeServerFt2Message(msg);
|
|
if (next.grids.length > 0 && window.mapAddLocator) {
|
|
window.mapAddLocator(next.raw, next.grids, "ft2", next.station, { ...msg, freq_hz: next.rfHz, locator_details: next.locatorDetails });
|
|
}
|
|
next.history._tsMs = Number.isFinite(next.history?.ts_ms) ? Number(next.history.ts_ms) : Date.now();
|
|
normalized.push(next.history);
|
|
}
|
|
normalized.reverse();
|
|
ft2MessageHistory = normalized.concat(ft2MessageHistory);
|
|
pruneFt2MessageHistory();
|
|
window.setFt8FamilyBarDecoder?.("ft2");
|
|
window.updateFt8Bar?.();
|
|
scheduleFt2HistoryRender();
|
|
};
|
|
|
|
window.restoreFt2History = function(messages) { window.onServerFt2Batch(messages); };
|
|
window.pruneFt2HistoryView = function() { pruneFt2MessageHistory(); renderFt2History(); };
|
|
|
|
window.resetFt2HistoryView = function() {
|
|
if (ft2MessagesEl) ft2MessagesEl.innerHTML = "";
|
|
ft2MessageHistory = [];
|
|
window.updateFt8Bar?.();
|
|
renderFt2History();
|
|
};
|
|
|
|
function buildFt2BarFrames() {
|
|
const cutoffMs = Date.now() - 15 * 60 * 1000;
|
|
const messages = ft2MessageHistory.filter((msg) => Number(msg._tsMs ?? msg.ts_ms) >= cutoffMs).slice(0, 8);
|
|
const newestTsMs = messages.reduce((latest, msg) => Math.max(latest, Number(msg._tsMs ?? msg.ts_ms) || 0), 0);
|
|
if (messages.length === 0) {
|
|
return { count: 0, newestTsMs: 0, html: "" };
|
|
}
|
|
let html = "";
|
|
for (const msg of messages) {
|
|
const tsMs = msg._tsMs ?? msg.ts_ms;
|
|
const ts = tsMs ? `<span class="aprs-bar-time">${fmtTime(tsMs)}</span>` : "";
|
|
const snr = Number.isFinite(msg.snr_db) ? `${msg.snr_db.toFixed(1)} dB` : "-- dB";
|
|
const dt = Number.isFinite(msg.dt_s) ? `dt ${msg.dt_s.toFixed(2)}` : null;
|
|
const displayFreqHz = normalizeFt2DisplayFreqHz(msg.freq_hz);
|
|
const rf = Number.isFinite(displayFreqHz) ? `${displayFreqHz.toFixed(0)} Hz` : null;
|
|
const detail = [snr, dt, rf].filter(Boolean).join(" · ");
|
|
const text = ft8RenderMessageFt2((msg.message || "").toString());
|
|
html += `<div class="aprs-bar-frame"><div class="aprs-bar-frame-main">${ts}<span class="aprs-bar-call">${text}</span>${detail ? ` · ${detail}` : ""}</div></div>`;
|
|
}
|
|
return { count: messages.length, newestTsMs, html };
|
|
}
|
|
window.registerFt8FamilyBarRenderer?.("ft2", buildFt2BarFrames);
|
|
|
|
if (ft2FilterInput) {
|
|
ft2FilterInput.addEventListener("input", () => {
|
|
ft2FilterText = ft2FilterInput.value.trim().toUpperCase();
|
|
renderFt2History();
|
|
});
|
|
}
|
|
|
|
const ft2DecodeToggleBtn = document.getElementById("ft2-decode-toggle-btn");
|
|
ft2DecodeToggleBtn?.addEventListener("click", async () => {
|
|
try {
|
|
await window.takeSchedulerControlForDecoderDisable?.(ft2DecodeToggleBtn);
|
|
await postPath("/toggle_ft2_decode");
|
|
} catch (e) {
|
|
console.error("FT2 toggle failed", e);
|
|
}
|
|
});
|
|
|
|
document.getElementById("settings-clear-ft2-history")?.addEventListener("click", async () => {
|
|
if (!confirm("Clear all FT2 decode history? This cannot be undone.")) return;
|
|
try {
|
|
await postPath("/clear_ft2_decode");
|
|
window.resetFt2HistoryView();
|
|
} catch (e) { console.error("FT2 history clear failed", e); }
|
|
});
|
|
|
|
window.onServerFt2 = function(msg) {
|
|
if (ft2Status) ft2Status.textContent = "Receiving";
|
|
const next = normalizeServerFt2Message(msg);
|
|
if (next.grids.length > 0 && window.mapAddLocator) {
|
|
window.mapAddLocator(next.raw, next.grids, "ft2", next.station, { ...msg, freq_hz: next.rfHz, locator_details: next.locatorDetails });
|
|
}
|
|
addFt2Message(next.history);
|
|
};
|