// models.js — "Modelle"-Ansicht: Download + Einpflegen, Schnelltest-Chat, Modell-Tabelle. import { api } from "../core/api.js"; import { $, badge, esc, toast } from "../core/ui.js"; function refreshSoon() { document.dispatchEvent(new Event("mc:refresh")); } function mount() { $("#m-chat").innerHTML = `

Schnelltest

`; $("#m-table").innerHTML = `

Modelle & Ports

ModellFähigkeitenDetailsStatusPortAktion
`; $("#chat-btn").addEventListener("click", sendChat); } function onStatus(s) { const models = s?.models || []; const tb = $("#models"); if (!tb) return; tb.innerHTML = ""; $("#models-empty").style.display = models.length ? "none" : "block"; $("#m-count").textContent = models.length ? models.length + " konfiguriert" : ""; const sel = $("#chat-model"); const cur = sel.value; sel.innerHTML = ""; for (const m of models) { const tr = document.createElement("tr"); let capsHtml = "–"; if (m.meta && m.meta.caps) { capsHtml = m.meta.caps.map(c => { if (c === "Text") return `T`; if (c === "Code") return `{ }`; if (c === "Bild") return `👁`; return ""; }).join(" "); } let detailsHtml = "–"; if (m.meta) { const q = m.meta.quant || "?"; const c = m.meta.ctx ? (m.meta.ctx / 1024).toFixed(0) + "K" : "?"; const s = m.meta.size_bytes ? (m.meta.size_bytes / 1024 / 1024 / 1024).toFixed(1) + " GB" : "?"; detailsHtml = `${q} · ${c} · ${s}`; } const perfHtml = m.state === "running" ? `
n/a t/s` : ""; tr.innerHTML = `${esc(m.name)} ${capsHtml} ${detailsHtml} ${badge(m.state)}${perfHtml} ${m.port ?? "auto"} `; tb.appendChild(tr); sel.insertAdjacentHTML("beforeend", ``); } if (cur) sel.value = cur; tb.querySelectorAll("[data-unload]").forEach(b => b.addEventListener("click", () => unloadOne(b.getAttribute("data-unload"))) ); } async function unloadOne(m) { try { await api("/api/unload?model=" + encodeURIComponent(m), { method: "POST" }); toast("Entladen: " + m); setTimeout(refreshSoon, 600); } catch (e) { toast(e.message, true); } } async function sendChat() { const model = $("#chat-model").value, message = $("#chat-msg").value.trim(); if (!model) return toast("Kein Modell vorhanden.", true); if (!message) return; const btn = $("#chat-btn"); btn.disabled = true; btn.textContent = "…"; const box = $("#chat-reply"); box.style.display = "block"; box.textContent = "(wecke Modell, kann beim Swap kurz dauern…)"; try { const r = await api("/api/chat", { method: "POST", body: JSON.stringify({ model, message }) }); box.textContent = r.reply; } catch (e) { box.textContent = "Fehler: " + e.message; } btn.disabled = false; btn.textContent = "Senden"; refreshSoon(); } export default { id: "models", mount, onStatus };