mirror of
https://github.com/sune-org/store.git
synced 2026-01-13 16:17:58 +00:00
1 line
12 KiB
JSON
1 line
12 KiB
JSON
[{"id":"4awhhn5","name":"LIVE CHAT (alpha)","pinned":false,"avatar":"","url":"gh://sune-org/store/chatroom.sune","updatedAt":1757304752767,"settings":{"model":"","temperature":"","top_p":"","top_k":"","frequency_penalty":"","repetition_penalty":"","min_p":"","top_a":"","verbosity":"","reasoning_effort":"default","system_prompt":"","html":"<!-- Name generator library (Hard Dependency) -->\n<script src=\"https://cdn.jsdelivr.net/npm/unique-names-generator/dist/index.umd.js\"></script>\n\n<div class=\"max-w-md 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.4.5</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 & Rate-Limited 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 initChatroom = () => {\n const URL = 'wss://chatsune.awww.workers.dev/ws';\n const logIsVisible = false;\n const USERNAME_KEY = 'sune-chatroom-username';\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 elements.');\n if (logIsVisible) ui.log.classList.remove('hidden');\n\n let ws = null;\n let currentUsername = localStorage.getItem(USERNAME_KEY);\n if (!currentUsername) {\n const { uniqueNamesGenerator, adjectives, animals } = window.uniqueNamesGenerator;\n currentUsername = uniqueNamesGenerator({ dictionaries: [adjectives, animals], separator: '', style: 'capital' });\n localStorage.setItem(USERNAME_KEY, currentUsername);\n }\n ui.username.textContent = currentUsername;\n\n const log = m => {\n if (!ui.log || !logIsVisible) return;\n ui.log.textContent = `[${new Date().toLocaleTimeString()}] ${m}\\n${ui.log.textContent}`.substring(0, 3000);\n };\n \n const renderSystemMessage = text => {\n const el = document.createElement('div');\n el.className = 'px-4 py-2 text-xs text-center text-gray-500 italic';\n el.textContent = text;\n ui.messages.appendChild(el);\n 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?.readyState < 2) {\n ws.onclose = null; ws.close(); ws = null;\n updateStatus('Offline', 'bg-gray-400', false);\n log('Disconnected.');\n renderSystemMessage('You have left the chatroom.');\n }\n };\n\n const connect = () => {\n if (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') renderSystemMessage(msg.text);\n else if (msg.author.name === currentUsername) window.addMessage({ role: 'user', content: msg.text }, false);\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(handleMsg); 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 }\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 renderSystemMessage('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.length > 80 ? txt.slice(0, 80) + '...' : txt}`);\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\n // Wait for the uniqueNamesGenerator library to be loaded before initializing\n let attempts = 0;\n const maxAttempts = 50; // 50 * 100ms = 5 seconds\n const interval = setInterval(() => {\n if (window.uniqueNamesGenerator) {\n clearInterval(interval);\n initChatroom();\n } else if (++attempts > maxAttempts) {\n clearInterval(interval);\n console.error('Sune Error: Name generator library failed to load in time.');\n }\n }, 100);\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":{}}] |