mirror of
https://github.com/multipleof4/sune.git
synced 2026-02-04 01:57:55 +00:00
This build was committed by a bot.
This commit is contained in:
@@ -615,21 +615,26 @@ const titleFrom = (t) => (t || "").replace(/\s+/g, " ").trim().slice(0, 60) || "
|
|||||||
const TKEY = "threads_v1", THREAD = window.THREAD = { list: [], load: async function() {
|
const TKEY = "threads_v1", THREAD = window.THREAD = { list: [], load: async function() {
|
||||||
this.list = await localforage.getItem(TKEY).then((v) => Array.isArray(v) ? v : []) || [];
|
this.list = await localforage.getItem(TKEY).then((v) => Array.isArray(v) ? v : []) || [];
|
||||||
}, save: async function() {
|
}, save: async function() {
|
||||||
await localforage.setItem(TKEY, this.list);
|
await localforage.setItem(TKEY, this.list.map((t) => {
|
||||||
|
const n = { ...t };
|
||||||
|
delete n.messages;
|
||||||
|
return n;
|
||||||
|
}));
|
||||||
}, get: function(id) {
|
}, get: function(id) {
|
||||||
return this.list.find((t) => t.id === id);
|
return this.list.find((t) => t.id === id);
|
||||||
}, get active() {
|
}, get active() {
|
||||||
return this.get(state.currentThreadId);
|
return this.get(state.currentThreadId);
|
||||||
}, persist: async function(full = true) {
|
}, persist: async function(full = true) {
|
||||||
if (!state.currentThreadId) return;
|
const id = state.currentThreadId;
|
||||||
const th = this.active;
|
if (!id) return;
|
||||||
if (!th) return;
|
const meta = this.get(id);
|
||||||
th.messages = [...state.messages];
|
if (!meta) return;
|
||||||
|
await localforage.setItem("t_" + id, [...state.messages]);
|
||||||
if (full) {
|
if (full) {
|
||||||
th.updatedAt = Date.now();
|
meta.updatedAt = Date.now();
|
||||||
|
await this.save();
|
||||||
|
await renderThreads();
|
||||||
}
|
}
|
||||||
await this.save();
|
|
||||||
if (full) await renderThreads();
|
|
||||||
}, setTitle: async function(id, title) {
|
}, setTitle: async function(id, title) {
|
||||||
const th = this.get(id);
|
const th = this.get(id);
|
||||||
if (!th || !title) return;
|
if (!th || !title) return;
|
||||||
@@ -645,6 +650,18 @@ const TKEY = "threads_v1", THREAD = window.THREAD = { list: [], load: async func
|
|||||||
if (!/^\s*You\b/.test(h.textContent || "")) return b.dataset.mid || null;
|
if (!/^\s*You\b/.test(h.textContent || "")) return b.dataset.mid || null;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
}, migrate: async function() {
|
||||||
|
const old = await localforage.getItem(TKEY);
|
||||||
|
if (Array.isArray(old) && old.length > 0 && old[0].messages) {
|
||||||
|
for (const t of old) {
|
||||||
|
if (t.messages) {
|
||||||
|
await localforage.setItem("t_" + t.id, t.messages);
|
||||||
|
delete t.messages;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
await localforage.setItem(TKEY, old);
|
||||||
|
this.list = old;
|
||||||
|
}
|
||||||
} };
|
} };
|
||||||
const cacheStore = localforage.createInstance({ name: "threads_cache", storeName: "streams_status" });
|
const cacheStore = localforage.createInstance({ name: "threads_cache", storeName: "streams_status" });
|
||||||
async function ensureThreadOnFirstUser(text) {
|
async function ensureThreadOnFirstUser(text) {
|
||||||
@@ -652,10 +669,11 @@ async function ensureThreadOnFirstUser(text) {
|
|||||||
if (state.messages.length === 0) state.currentThreadId = null;
|
if (state.messages.length === 0) state.currentThreadId = null;
|
||||||
if (state.currentThreadId && !THREAD.get(state.currentThreadId)) needNew = true;
|
if (state.currentThreadId && !THREAD.get(state.currentThreadId)) needNew = true;
|
||||||
if (!needNew) return;
|
if (!needNew) return;
|
||||||
const id = gid(), now = Date.now(), th = { id, title: "", pinned: false, updatedAt: now, messages: [] };
|
const id = gid(), now = Date.now(), th = { id, title: "", pinned: false, updatedAt: now };
|
||||||
state.currentThreadId = id;
|
state.currentThreadId = id;
|
||||||
THREAD.list.unshift(th);
|
THREAD.list.unshift(th);
|
||||||
await THREAD.save();
|
await THREAD.save();
|
||||||
|
await localforage.setItem("t_" + id, []);
|
||||||
await renderThreads();
|
await renderThreads();
|
||||||
}
|
}
|
||||||
const generateTitleWithAI = async (messages) => {
|
const generateTitleWithAI = async (messages) => {
|
||||||
@@ -730,7 +748,8 @@ $(el.threadList).on("click", async (e) => {
|
|||||||
}
|
}
|
||||||
state.currentThreadId = id;
|
state.currentThreadId = id;
|
||||||
clearChat();
|
clearChat();
|
||||||
state.messages = Array.isArray(th.messages) ? [...th.messages] : [];
|
const msgs = await localforage.getItem("t_" + id);
|
||||||
|
state.messages = Array.isArray(msgs) ? [...msgs] : [];
|
||||||
for (const m of state.messages) {
|
for (const m of state.messages) {
|
||||||
const b = msgRow(m);
|
const b = msgRow(m);
|
||||||
b.dataset.mid = m.id || "";
|
b.dataset.mid = m.id || "";
|
||||||
@@ -777,13 +796,14 @@ $(el.threadPopover).on("click", async (e) => {
|
|||||||
} else if (act === "delete") {
|
} else if (act === "delete") {
|
||||||
if (confirm("Delete this chat?")) {
|
if (confirm("Delete this chat?")) {
|
||||||
THREAD.list = THREAD.list.filter((x) => x.id !== th.id);
|
THREAD.list = THREAD.list.filter((x) => x.id !== th.id);
|
||||||
|
await localforage.removeItem("t_" + th.id);
|
||||||
if (state.currentThreadId === th.id) {
|
if (state.currentThreadId === th.id) {
|
||||||
state.currentThreadId = null;
|
state.currentThreadId = null;
|
||||||
clearChat();
|
clearChat();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (act === "count_tokens") {
|
} else if (act === "count_tokens") {
|
||||||
const msgs = Array.isArray(th.messages) ? th.messages : [];
|
const msgs = await localforage.getItem("t_" + th.id) || [];
|
||||||
let totalChars = 0;
|
let totalChars = 0;
|
||||||
for (const m of msgs) {
|
for (const m of msgs) {
|
||||||
if (!m || !m.role || m.role === "system") continue;
|
if (!m || !m.role || m.role === "system") continue;
|
||||||
@@ -793,7 +813,8 @@ $(el.threadPopover).on("click", async (e) => {
|
|||||||
const k = tokens >= 1e3 ? Math.round(tokens / 1e3) + "k" : String(tokens);
|
const k = tokens >= 1e3 ? Math.round(tokens / 1e3) + "k" : String(tokens);
|
||||||
alert(tokens + " tokens (" + k + ")");
|
alert(tokens + " tokens (" + k + ")");
|
||||||
} else if (act === "export") {
|
} else if (act === "export") {
|
||||||
dl(`thread-${(th.title || "thread").replace(/\W/g, "_")}-${ts()}.json`, { version: 1, threads: [th] });
|
const msgs = await localforage.getItem("t_" + th.id) || [];
|
||||||
|
dl(`thread-${(th.title || "thread").replace(/\W/g, "_")}-${ts()}.json`, { version: 1, threads: [{ ...th, messages: msgs }] });
|
||||||
}
|
}
|
||||||
hideThreadPopover();
|
hideThreadPopover();
|
||||||
await THREAD.save();
|
await THREAD.save();
|
||||||
@@ -1106,8 +1127,13 @@ $(el.sunesImportOption).on("click", () => {
|
|||||||
el.importInput.value = "";
|
el.importInput.value = "";
|
||||||
el.importInput.click();
|
el.importInput.click();
|
||||||
});
|
});
|
||||||
$(el.threadsExportOption).on("click", () => {
|
$(el.threadsExportOption).on("click", async () => {
|
||||||
dl(`threads-${ts()}.json`, { version: 1, threads: THREAD.list });
|
const all = [];
|
||||||
|
for (const t of THREAD.list) {
|
||||||
|
const msgs = await localforage.getItem("t_" + t.id) || [];
|
||||||
|
all.push({ ...t, messages: msgs });
|
||||||
|
}
|
||||||
|
dl(`threads-${ts()}.json`, { version: 1, threads: all });
|
||||||
el.userMenu.classList.add("hidden");
|
el.userMenu.classList.add("hidden");
|
||||||
});
|
});
|
||||||
$(el.threadsImportOption).on("click", () => {
|
$(el.threadsImportOption).on("click", () => {
|
||||||
@@ -1167,8 +1193,11 @@ $(el.importInput).on("change", async () => {
|
|||||||
skipped++;
|
skipped++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
const msgs = th.messages;
|
||||||
|
delete th.messages;
|
||||||
if (!ex) THREAD.list.push(th);
|
if (!ex) THREAD.list.push(th);
|
||||||
else Object.assign(ex, th);
|
else Object.assign(ex, th);
|
||||||
|
await localforage.setItem("t_" + th.id, msgs);
|
||||||
kept++;
|
kept++;
|
||||||
}
|
}
|
||||||
await THREAD.save();
|
await THREAD.save();
|
||||||
@@ -1307,6 +1336,7 @@ const USER = window.USER = { log: async (s) => {
|
|||||||
async function init() {
|
async function init() {
|
||||||
await SUNE.fetchDotSune("sune-org/store@main/marketplace.sune");
|
await SUNE.fetchDotSune("sune-org/store@main/marketplace.sune");
|
||||||
await THREAD.load();
|
await THREAD.load();
|
||||||
|
await THREAD.migrate();
|
||||||
await renderThreads();
|
await renderThreads();
|
||||||
renderSidebar();
|
renderSidebar();
|
||||||
await reflectActiveSune();
|
await reflectActiveSune();
|
||||||
2
dist/index.html
vendored
2
dist/index.html
vendored
@@ -12,7 +12,7 @@
|
|||||||
<script defer src="https://cdn.jsdelivr.net/npm/cash-dom/dist/cash.min.js"></script>
|
<script defer src="https://cdn.jsdelivr.net/npm/cash-dom/dist/cash.min.js"></script>
|
||||||
<script defer src="//unpkg.com/alpinejs"></script>
|
<script defer src="//unpkg.com/alpinejs"></script>
|
||||||
|
|
||||||
<script type="module" crossorigin src="/assets/index-Cv7JfxES.js"></script>
|
<script type="module" crossorigin src="/assets/index-u8WIj8uY.js"></script>
|
||||||
<link rel="stylesheet" crossorigin href="/assets/index-CZ8Js0gk.css">
|
<link rel="stylesheet" crossorigin href="/assets/index-CZ8Js0gk.css">
|
||||||
<link rel="manifest" href="/manifest.webmanifest"><script id="vite-plugin-pwa:register-sw" src="/registerSW.js"></script></head>
|
<link rel="manifest" href="/manifest.webmanifest"><script id="vite-plugin-pwa:register-sw" src="/registerSW.js"></script></head>
|
||||||
<body class="bg-white text-gray-900 selection:bg-black/10" x-data @click.window="if($event.target.closest('button')) haptic(); if(!document.getElementById('threadPopover').contains($event.target)&&!$event.target.closest('[data-thread-menu]')) hideThreadPopover(); if(!document.getElementById('sunePopover').contains($event.target)&&!$event.target.closest('[data-sune-menu]')) hideSunePopover(); if(!document.getElementById('userMenu').contains($event.target)&&!document.getElementById('userMenuBtn').contains($event.target)) document.getElementById('userMenu').classList.add('hidden')">
|
<body class="bg-white text-gray-900 selection:bg-black/10" x-data @click.window="if($event.target.closest('button')) haptic(); if(!document.getElementById('threadPopover').contains($event.target)&&!$event.target.closest('[data-thread-menu]')) hideThreadPopover(); if(!document.getElementById('sunePopover').contains($event.target)&&!$event.target.closest('[data-sune-menu]')) hideSunePopover(); if(!document.getElementById('userMenu').contains($event.target)&&!document.getElementById('userMenuBtn').contains($event.target)) document.getElementById('userMenu').classList.add('hidden')">
|
||||||
|
|||||||
2
dist/sw.js
vendored
2
dist/sw.js
vendored
@@ -1 +1 @@
|
|||||||
if(!self.define){let e,s={};const i=(i,n)=>(i=new URL(i+".js",n).href,s[i]||new Promise(s=>{if("document"in self){const e=document.createElement("script");e.src=i,e.onload=s,document.head.appendChild(e)}else e=i,importScripts(i),s()}).then(()=>{let e=s[i];if(!e)throw new Error(`Module ${i} didn’t register its module`);return e}));self.define=(n,r)=>{const t=e||("document"in self?document.currentScript.src:"")||location.href;if(s[t])return;let o={};const d=e=>i(e,t),c={module:{uri:t},exports:o,require:d};s[t]=Promise.all(n.map(e=>c[e]||d(e))).then(e=>(r(...e),o))}}define(["./workbox-8c29f6e4"],function(e){"use strict";self.skipWaiting(),e.clientsClaim(),e.precacheAndRoute([{url:"registerSW.js",revision:"1872c500de691dce40960bb85481de07"},{url:"index.html",revision:"8b290d06fdea7a854064c5dad25c9259"},{url:"assets/index-Cv7JfxES.js",revision:null},{url:"assets/index-CZ8Js0gk.css",revision:null},{url:"manifest.webmanifest",revision:"7a6c5c6ab9cb5d3605d21df44c6b17a2"}],{}),e.cleanupOutdatedCaches(),e.registerRoute(new e.NavigationRoute(e.createHandlerBoundToURL("index.html")))});
|
if(!self.define){let e,s={};const i=(i,n)=>(i=new URL(i+".js",n).href,s[i]||new Promise(s=>{if("document"in self){const e=document.createElement("script");e.src=i,e.onload=s,document.head.appendChild(e)}else e=i,importScripts(i),s()}).then(()=>{let e=s[i];if(!e)throw new Error(`Module ${i} didn’t register its module`);return e}));self.define=(n,r)=>{const t=e||("document"in self?document.currentScript.src:"")||location.href;if(s[t])return;let o={};const c=e=>i(e,t),l={module:{uri:t},exports:o,require:c};s[t]=Promise.all(n.map(e=>l[e]||c(e))).then(e=>(r(...e),o))}}define(["./workbox-8c29f6e4"],function(e){"use strict";self.skipWaiting(),e.clientsClaim(),e.precacheAndRoute([{url:"registerSW.js",revision:"1872c500de691dce40960bb85481de07"},{url:"index.html",revision:"0e74136a980230d8c5f1388db4cc0fca"},{url:"assets/index-u8WIj8uY.js",revision:null},{url:"assets/index-CZ8Js0gk.css",revision:null},{url:"manifest.webmanifest",revision:"7a6c5c6ab9cb5d3605d21df44c6b17a2"}],{}),e.cleanupOutdatedCaches(),e.registerRoute(new e.NavigationRoute(e.createHandlerBoundToURL("index.html")))});
|
||||||
|
|||||||
Reference in New Issue
Block a user