mirror of
https://github.com/sune-org/store.git
synced 2026-01-13 16:17:58 +00:00
1 line
18 KiB
JSON
1 line
18 KiB
JSON
[{"id":"4awhhn5","name":"LIVE CHAT (alpha)","pinned":false,"avatar":"","url":"gh://sune-org/store/chatroom.sune","updatedAt":1757309136344,"settings":{"model":"","temperature":"","top_p":"","top_k":"","frequency_penalty":"","repetition_penalty":"","min_p":"","top_a":"","verbosity":"","reasoning_effort":"default","system_prompt":"","html":"<div class=\"max-w-md w-full m-1 p-3 font-sans bg-white rounded-2xl border border-gray-200 shadow-lg text-sm\" id=\"sune-chatroom-root\">\n <div class=\"flex justify-between items-center mb-2 px-1\">\n <h3 class=\"font-semibold text-gray-800\">Sune LIVE Chatroom</h3>\n <div class=\"flex items-center gap-3\">\n <div class=\"flex items-center gap-1.5 text-gray-500\" title=\"Users online\">\n <i data-lucide=\"users\" class=\"h-4 w-4\"></i>\n <span id=\"wsUserCount\" class=\"text-sm font-medium\">0</span>\n </div>\n <span class=\"text-xs text-gray-400\">v0.5.0</span>\n </div>\n </div>\n\n <div class=\"flex items-center justify-between gap-3 p-2 mb-2 bg-gray-50 rounded-lg\">\n <div class=\"flex items-center gap-2 truncate\">\n <span id=\"wsStatusDot\" class=\"h-3 w-3 rounded-full bg-gray-400 flex-shrink-0\"></span>\n <div class=\"text-xs text-gray-500 truncate\">\n <span id=\"wsStatusText\" class=\"font-medium text-gray-700\">Offline</span>: <span id=\"wsUsernameDisplay\" class=\"font-medium text-gray-800\"></span>\n </div>\n </div>\n <button id=\"wsConnectionButton\" class=\"px-3 py-1 text-xs text-white font-semibold bg-gray-800 rounded-md hover:bg-gray-700 active:scale-[.98] transition-transform focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500 whitespace-nowrap\">Connect</button>\n </div>\n\n <div class=\"mt-3 pt-2 border-t border-gray-100 flex items-center justify-center gap-2 text-xs text-gray-500\">\n <i data-lucide=\"shield-check\" class=\"h-3.5 w-3.5 text-gray-400\"></i>\n <span>Moderated by Gemini 2.5 Flash-Lite</span>\n </div>\n\n <pre id=\"wsLog\" class=\"w-full h-24 p-2 bg-gray-900 text-white text-xs rounded-lg overflow-y-auto font-mono whitespace-pre-wrap break-all hidden mt-2\">Initializing...</pre>\n</div>\n\n<script>\n(() => {\n const URL = 'wss://chatsune.awww.workers.dev/ws';\n const LOG_IS_VISIBLE = false;\n const USERNAME_KEY = `sune-chatroom-username-${window.SUNE?.id || 'default'}`;\n \n const ui = {\n root: document.getElementById('sune-chatroom-root'),\n dot: document.getElementById('wsStatusDot'),\n text: document.getElementById('wsStatusText'),\n log: document.getElementById('wsLog'),\n count: document.getElementById('wsUserCount'),\n username: document.getElementById('wsUsernameDisplay'),\n button: document.getElementById('wsConnectionButton'),\n composer: document.getElementById('composer'),\n messages: document.getElementById('messages'),\n suneHtmlContainer: document.getElementById('suneHtml')\n };\n\n if (Object.values(ui).some(el => !el)) return console.error('Sune Error: Missing required UI elements (#composer, #messages, #suneHtml).');\n if (LOG_IS_VISIBLE) ui.log.classList.remove('hidden');\n\n let ws = null;\n let currentUsername = localStorage.getItem(USERNAME_KEY);\n if (!currentUsername) {\n const ADJ = ['Omnipotent','Omniscient','Divine','Celestial','Cosmic','Mystic','Sacred','Radiant','Sovereign','Supreme','Exalted','Immortal','Majestic','Mythic','Sublime','Solar','Stellar','Prophetic','Visionary'];\n const NOUN = ['Pantokrator','Thearch','Kosmokrator','Aion','Numen','Daemon','Anima','Archon','Seraph','Cherub','Kyrios','Sophos','Magus','Mystes','Oracle','Dominus','Imperator','Pontifex','Primarch','Allfather'];\n const randEl = arr => arr[Math.floor(Math.random() * arr.length)];\n currentUsername = `${randEl(ADJ)}${randEl(NOUN)}`;\n localStorage.setItem(USERNAME_KEY, currentUsername);\n }\n ui.username.textContent = currentUsername;\n\n const log = m => LOG_IS_VISIBLE && (ui.log.textContent = `[${new Date().toLocaleTimeString()}] ${m}\\n${ui.log.textContent}`.substring(0, 3000));\n \n const renderNotification = (text, type = 'system') => {\n const el = document.createElement('div');\n const styles = { system: 'text-gray-500', error: 'text-red-600 font-semibold' };\n el.className = `px-4 py-2 text-xs text-center italic ${styles[type] || styles.system}`;\n el.textContent = text;\n ui.messages.appendChild(el);\n if(ui.messages.parentElement) ui.messages.parentElement.scrollTo({ top: ui.messages.parentElement.scrollHeight, behavior: 'smooth' });\n };\n\n const updateStatus = (status, color, isConnected) => {\n ui.text.textContent = status;\n ui.dot.className = `h-3 w-3 rounded-full flex-shrink-0 ${color}`;\n ui.dot.classList.toggle('animate-pulse', status === 'Connecting');\n ui.button.textContent = isConnected ? 'Disconnect' : 'Connect';\n };\n\n const disconnect = () => {\n if (!ws || ws.readyState > 1) return;\n ws.onclose = null;\n ws.close();\n ws = null;\n updateStatus('Offline', 'bg-gray-400', false);\n log('Disconnected.');\n renderNotification('You have left the chatroom.');\n };\n\n const connect = () => {\n if (ws && ws.readyState < 2) return;\n updateStatus('Connecting', 'bg-yellow-400', true);\n ws = new WebSocket(URL);\n \n ws.onopen = () => {\n updateStatus('Connected', 'bg-green-500', true);\n log('Connection established.');\n ui.messages.innerHTML = '';\n ws.send(JSON.stringify({ type: 'USER_JOINED', payload: { name: currentUsername } }));\n };\n\n ws.onmessage = e => {\n log(`RECV: ${e.data.substring(0, 100)}`);\n try {\n const data = JSON.parse(e.data);\n const handleMsg = msg => {\n if (msg.author.name === 'system') renderNotification(msg.text);\n else window.addMessage({ role: 'assistant', content: msg.text, sune_name: msg.author.name }, false);\n };\n switch (data.type) {\n case 'HISTORY': ui.messages.innerHTML = ''; data.payload.forEach(msg => (msg.author.name === currentUsername) ? window.addMessage({ role: 'user', content: msg.text }, false) : handleMsg(msg)); break;\n case 'NEW_MESSAGE': if(data.payload.author.name !== currentUsername) handleMsg(data.payload); break;\n case 'CONNECTION_COUNT': ui.count.textContent = data.payload.count || 0; break;\n case 'ERROR': renderNotification(data.payload, 'error'); break;\n }\n } catch (err) { log(`Error parsing message: ${err}`); }\n };\n\n ws.onerror = () => log('Connection error.');\n ws.onclose = () => {\n updateStatus('Offline', 'bg-red-500', false);\n log('Connection closed by server.');\n ws = null;\n renderNotification('You have been disconnected.');\n };\n };\n\n const handleComposerSend = e => {\n const txt = (e.detail?.message?.content?.find(p => p.type === 'text')?.text || '').trim();\n if (!txt || ws?.readyState !== 1) return;\n ws.send(JSON.stringify({ type: 'NEW_MESSAGE', payload: { text: txt } }));\n log(`SENT: ${txt.substring(0, 80)}`);\n };\n \n const cleanup = () => {\n log('Sune unloaded. Cleaning up...');\n disconnect();\n ui.composer.removeEventListener('sune:send', handleComposerSend);\n observer.disconnect();\n };\n\n const observer = new MutationObserver(m => m.forEach(r => r.removedNodes.forEach(n => n === ui.root && cleanup())));\n observer.observe(ui.suneHtmlContainer, { childList: true });\n ui.button.addEventListener('click', () => ws?.readyState === 1 ? disconnect() : connect());\n ui.composer.addEventListener('sune:send', handleComposerSend);\n \n connect();\n window.lucide?.createIcons();\n})();\n</script>\n","extension_html":"<sune src='https://raw.githubusercontent.com/sune-org/store/refs/heads/main/sync.sune' private />","hide_composer":false},"storage":{}}] |