| <!DOCTYPE html> |
| <html lang="en"> |
| <head> |
| <meta charset="UTF-8"> |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> |
| <title>Chat</title> |
| <link rel="manifest" href="/manifest.json"> |
| <style> |
| * { |
| margin: 0; |
| padding: 0; |
| box-sizing: border-box; |
| } |
| |
| body { |
| font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; |
| background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
| min-height: 100vh; |
| display: flex; |
| justify-content: center; |
| align-items: center; |
| padding: 20px; |
| } |
| |
| .container { |
| background: rgba(255, 255, 255, 0.95); |
| backdrop-filter: blur(10px); |
| border-radius: 20px; |
| box-shadow: 0 20px 60px rgba(0, 0, 0, 0.3); |
| width: 100%; |
| max-width: 800px; |
| height: 600px; |
| display: flex; |
| flex-direction: column; |
| overflow: hidden; |
| animation: slideUp 0.5s ease-out; |
| } |
| |
| @keyframes slideUp { |
| from { |
| opacity: 0; |
| transform: translateY(30px); |
| } |
| to { |
| opacity: 1; |
| transform: translateY(0); |
| } |
| } |
| |
| .header { |
| background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
| color: white; |
| padding: 15px 20px; |
| display: flex; |
| align-items: center; |
| justify-content: space-between; |
| box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1); |
| } |
| |
| .header .avatar { |
| width: 35px; |
| height: 35px; |
| border-radius: 50%; |
| background: #ffffff33; |
| display: flex; |
| align-items: center; |
| justify-content: center; |
| font-weight: bold; |
| margin-right: 10px; |
| } |
| |
| .header h1 { |
| font-size: 20px; |
| font-weight: 500; |
| flex: 1; |
| } |
| |
| .message-counter { |
| background: #ef4444; |
| color: white; |
| font-size: 12px; |
| padding: 4px 8px; |
| border-radius: 12px; |
| line-height: 1; |
| } |
| |
| .status { |
| padding: 10px 20px; |
| background: #f7f9fc; |
| border-bottom: 1px solid #e1e8ed; |
| display: flex; |
| align-items: center; |
| gap: 10px; |
| } |
| |
| .status-indicator { |
| width: 10px; |
| height: 10px; |
| border-radius: 50%; |
| background: #fbbf24; |
| animation: pulse 2s infinite; |
| } |
| |
| .status-indicator.ready { |
| background: #10b981; |
| animation: none; |
| } |
| |
| .status-indicator.error { |
| background: #ef4444; |
| animation: none; |
| } |
| |
| @keyframes pulse { |
| 0%, 100% { |
| opacity: 1; |
| } |
| 50% { |
| opacity: 0.5; |
| } |
| } |
| |
| .chat-container { |
| flex: 1; |
| overflow-y: auto; |
| padding: 20px; |
| display: flex; |
| flex-direction: column; |
| gap: 15px; |
| } |
| |
| .message { |
| display: flex; |
| gap: 10px; |
| animation: fadeIn 0.3s ease-in; |
| } |
| |
| @keyframes fadeIn { |
| from { |
| opacity: 0; |
| transform: translateY(10px); |
| } |
| to { |
| opacity: 1; |
| transform: translateY(0); |
| } |
| } |
| |
| .message.user { |
| flex-direction: row-reverse; |
| } |
| |
| .message-bubble { |
| max-width: 70%; |
| padding: 12px 16px; |
| border-radius: 18px; |
| word-wrap: break-word; |
| box-shadow: 0 2px 5px rgba(0, 0, 0, 0.05); |
| } |
| |
| .message.user .message-bubble { |
| background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
| color: white; |
| border-bottom-right-radius: 4px; |
| } |
| |
| .message.ai .message-bubble { |
| background: #f1f3f5; |
| color: #1a1a1a; |
| border-bottom-left-radius: 4px; |
| } |
| |
| .avatar { |
| width: 35px; |
| height: 35px; |
| border-radius: 50%; |
| display: flex; |
| align-items: center; |
| justify-content: center; |
| font-weight: bold; |
| color: white; |
| flex-shrink: 0; |
| } |
| |
| .message.user .avatar { |
| background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
| } |
| |
| .message.ai .avatar { |
| background: linear-gradient(135deg, #10b981 0%, #059669 100%); |
| } |
| |
| .typing-indicator { |
| display: none; |
| padding: 12px 16px; |
| background: #f1f3f5; |
| border-radius: 18px; |
| border-bottom-left-radius: 4px; |
| width: fit-content; |
| } |
| |
| .typing-indicator.active { |
| display: block; |
| } |
| |
| .typing-indicator span { |
| display: inline-block; |
| width: 8px; |
| height: 8px; |
| border-radius: 50%; |
| background: #667eea; |
| margin: 0 2px; |
| animation: typing 1.4s infinite; |
| } |
| |
| .typing-indicator span:nth-child(2) { |
| animation-delay: 0.2s; |
| } |
| |
| .typing-indicator span:nth-child(3) { |
| animation-delay: 0.4s; |
| } |
| |
| @keyframes typing { |
| 0%, 60%, 100% { |
| transform: translateY(0); |
| } |
| 30% { |
| transform: translateY(-10px); |
| } |
| } |
| |
| .input-container { |
| padding: 20px; |
| background: #f7f9fc; |
| border-top: 1px solid #e1e8ed; |
| } |
| |
| .input-wrapper { |
| display: flex; |
| gap: 10px; |
| } |
| |
| .message-input { |
| flex: 1; |
| padding: 12px 16px; |
| border: 2px solid #e1e8ed; |
| border-radius: 25px; |
| font-size: 14px; |
| outline: none; |
| transition: all 0.3s ease; |
| } |
| |
| .message-input:focus { |
| border-color: #667eea; |
| box-shadow: 0 0 0 3px rgba(102, 126, 234, 0.1); |
| } |
| |
| .send-button { |
| padding: 12px 24px; |
| background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
| color: white; |
| border: none; |
| border-radius: 25px; |
| font-size: 14px; |
| font-weight: 600; |
| cursor: pointer; |
| transition: all 0.3s ease; |
| display: flex; |
| align-items: center; |
| gap: 5px; |
| } |
| |
| .send-button:hover:not(:disabled) { |
| transform: translateY(-2px); |
| box-shadow: 0 5px 15px rgba(102, 126, 234, 0.3); |
| } |
| |
| .send-button:disabled { |
| opacity: 0.5; |
| cursor: not-allowed; |
| } |
| |
| .error-message { |
| background: #fee; |
| color: #c00; |
| padding: 10px; |
| border-radius: 8px; |
| margin: 10px 20px; |
| display: none; |
| } |
| |
| .error-message.show { |
| display: block; |
| } |
| |
| .footer { |
| text-align: center; |
| padding: 10px; |
| font-size: 12px; |
| color: #666; |
| } |
| |
| .footer a { |
| color: #667eea; |
| text-decoration: none; |
| } |
| |
| @media (max-width: 600px) { |
| .container { |
| height: 100vh; |
| max-width: 100%; |
| border-radius: 0; |
| } |
| |
| .message-bubble { |
| max-width: 85%; |
| } |
| } |
| </style> |
| </head> |
| <body> |
| <div class="container"> |
| <div class="header"> |
| <div class="message-counter">3</div> |
| <div class="avatar" id="friendAvatar"></div> |
| <h1 id="friendName">Loading...</h1> |
| </div> |
| |
| <div class="status"> |
| <div class="status-indicator" id="statusIndicator"></div> |
| <span id="statusText">Connecting...</span> |
| </div> |
|
|
| <div class="error-message" id="errorMessage"></div> |
| |
| <div class="chat-container" id="chatContainer"> |
| <div class="message ai"> |
| <div class="avatar" id="aiAvatar">AI</div> |
| <div class="message-bubble"> |
| Hey! What's up? |
| </div> |
| </div> |
| </div> |
| |
| <div class="input-container"> |
| <div class="input-wrapper"> |
| <input |
| type="text" |
| class="message-input" |
| id="messageInput" |
| placeholder="Message..." |
| disabled |
| > |
| <button class="send-button" id="sendButton" disabled> |
| <span>Send</span> |
| <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> |
| <path d="M22 2L11 13M22 2l-7 20-4-9-9-4 20-7z"/> |
| </svg> |
| </button> |
| </div> |
| </div> |
| |
| <div class="footer"> |
| Powered by <a href="https://huggingface.co/HuggingFaceTB/SmolLM-135M-Instruct" target="_blank">SmolLM</a> from Hugging Face (Apache 2.0). |
| </div> |
| </div> |
|
|
| <script type="module"> |
| |
| if ('serviceWorker' in navigator) { |
| window.addEventListener('load', () => { |
| navigator.serviceWorker.register('/sw.js') |
| .then(reg => console.log('Service Worker registered')) |
| .catch(err => console.error('Service Worker registration failed:', err)); |
| }); |
| } |
| |
| import { pipeline, env } from 'https://cdn.jsdelivr.net/npm/@huggingface/transformers@3.0.0'; |
| |
| |
| env.allowLocalModels = false; |
| env.useBrowserCache = true; |
| |
| let generator = null; |
| let isProcessing = false; |
| let conversationHistory = []; |
| let modelId = /mobile|android|iphone|ipad/.test(navigator.userAgent.toLowerCase()) |
| ? 'HuggingFaceTB/SmolLM-135M-Instruct' |
| : 'HuggingFaceTB/SmolLM2-360M-Instruct'; |
| |
| const chatContainer = document.getElementById('chatContainer'); |
| const messageInput = document.getElementById('messageInput'); |
| const sendButton = document.getElementById('sendButton'); |
| const statusIndicator = document.getElementById('statusIndicator'); |
| const statusText = document.getElementById('statusText'); |
| const errorMessage = document.getElementById('errorMessage'); |
| const friendNameElement = document.getElementById('friendName'); |
| const friendAvatar = document.getElementById('friendAvatar'); |
| const aiAvatar = document.getElementById('aiAvatar'); |
| |
| |
| let friendName = prompt('Who are you chatting with?', 'Alex'); |
| friendName = friendName ? friendName.trim() : 'Alex'; |
| friendNameElement.textContent = friendName; |
| friendAvatar.textContent = friendName[0].toUpperCase(); |
| aiAvatar.textContent = friendName[0].toUpperCase(); |
| |
| |
| function checkBrowserCompatibility() { |
| const ua = navigator.userAgent.toLowerCase(); |
| const isMobile = /mobile|android|iphone|ipad/.test(ua); |
| const isChrome = ua.includes('chrome') && !ua.includes('edge'); |
| const isEdge = ua.includes('edg/'); |
| const isSafari = ua.includes('safari') && !ua.includes('chrome'); |
| return { isMobile, isChrome, isEdge, isSafari }; |
| } |
| |
| |
| async function checkWebGPU() { |
| if (!navigator.gpu) return false; |
| try { |
| const adapter = await navigator.gpu.requestAdapter(); |
| return !!adapter; |
| } catch (e) { |
| console.error('WebGPU check failed:', e); |
| return false; |
| } |
| } |
| |
| |
| async function initializeModel(attemptSmallerModel = false) { |
| try { |
| statusText.textContent = `Connecting...`; |
| |
| generator = await pipeline( |
| 'text-generation', |
| attemptSmallerModel ? 'HuggingFaceTB/SmolLM-135M-Instruct' : modelId |
| ); |
| |
| statusIndicator.classList.add('ready'); |
| statusText.textContent = 'Connected'; |
| messageInput.disabled = false; |
| sendButton.disabled = false; |
| messageInput.focus(); |
| |
| } catch (error) { |
| console.error('Error loading model:', error, error.stack); |
| if (!attemptSmallerModel && error.message.includes('memory')) { |
| console.warn('Memory error detected, trying smaller model...'); |
| modelId = 'HuggingFaceTB/SmolLM-135M-Instruct'; |
| initializeModel(true); |
| } else { |
| statusIndicator.classList.add('error'); |
| statusText.textContent = 'Offline'; |
| showError(`Oops, can't connect right now. Try refreshing?`); |
| } |
| } |
| } |
| |
| |
| const browser = checkBrowserCompatibility(); |
| if (browser.isMobile && (browser.isSafari || (!browser.isChrome && !browser.isEdge))) { |
| statusText.textContent = 'Connecting...'; |
| showError('Hey, this works best on Chrome or Edge. Give one of those a try?'); |
| initializeModel(); |
| } else { |
| checkWebGPU().then(supported => { |
| if (!supported) { |
| statusText.textContent = 'Connecting...'; |
| showError('Running a bit slow, but we’re good! Try a short message.'); |
| initializeModel(); |
| } else { |
| initializeModel(); |
| } |
| }); |
| } |
| |
| function showError(message) { |
| errorMessage.textContent = message; |
| errorMessage.classList.add('show'); |
| setTimeout(() => { |
| errorMessage.classList.remove('show'); |
| }, 5000); |
| } |
| |
| function addMessage(content, isUser = false) { |
| const messageDiv = document.createElement('div'); |
| messageDiv.className = `message ${isUser ? 'user' : 'ai'}`; |
| |
| const avatar = document.createElement('div'); |
| avatar.className = 'avatar'; |
| avatar.textContent = isUser ? 'You' : friendName[0].toUpperCase(); |
| |
| const bubble = document.createElement('div'); |
| bubble.className = 'message-bubble'; |
| bubble.textContent = content; |
| |
| messageDiv.appendChild(avatar); |
| messageDiv.appendChild(bubble); |
| |
| chatContainer.appendChild(messageDiv); |
| chatContainer.scrollTop = chatContainer.scrollHeight; |
| |
| |
| if (isUser) { |
| conversationHistory.push(`user: ${content}`); |
| } else { |
| conversationHistory.push(`assistant: ${content}`); |
| } |
| if (conversationHistory.length > 2) { |
| conversationHistory = conversationHistory.slice(-2); |
| } |
| } |
| |
| function showTypingIndicator() { |
| const typingDiv = document.createElement('div'); |
| typingDiv.className = 'message ai'; |
| typingDiv.id = 'typingIndicator'; |
| |
| const avatar = document.createElement('div'); |
| avatar.className = 'avatar'; |
| avatar.textContent = friendName[0].toUpperCase(); |
| |
| const indicator = document.createElement('div'); |
| indicator.className = 'typing-indicator active'; |
| indicator.innerHTML = '<span></span><span></span><span></span>'; |
| |
| typingDiv.appendChild(avatar); |
| typingDiv.appendChild(indicator); |
| |
| chatContainer.appendChild(typingDiv); |
| chatContainer.scrollTop = chatContainer.scrollHeight; |
| } |
| |
| function removeTypingIndicator() { |
| const indicator = document.getElementById('typingIndicator'); |
| if (indicator) { |
| indicator.remove(); |
| } |
| } |
| |
| async function generateResponse(userMessage) { |
| if (!generator || isProcessing) return; |
| |
| isProcessing = true; |
| sendButton.disabled = true; |
| messageInput.disabled = true; |
| |
| showTypingIndicator(); |
| |
| try { |
| const history = conversationHistory.length > 0 |
| ? conversationHistory.join('\n') + '\n' |
| : ''; |
| const prompt = `<|im_start|>user\n${history}${userMessage}\n<|im_end|>\n<|im_start|>assistant\n`; |
| |
| const output = await generator(prompt, { |
| max_new_tokens: 100, |
| temperature: 0.8, |
| top_p: 0.9, |
| return_full_text: false |
| }); |
| |
| removeTypingIndicator(); |
| |
| let response = output[0].generated_text.trim(); |
| response = response.replace(/<\|im_end\|>|<\|im_start\|>.*$/g, '').trim(); |
| |
| if (response) { |
| addMessage(response); |
| } else { |
| addMessage("Not sure what to say... Wanna try that again?"); |
| console.warn('Empty or invalid response received from model'); |
| showError('Hmm, I got nothing. Try a different question?'); |
| } |
| |
| } catch (error) { |
| console.error('Error generating response:', error, error.stack); |
| removeTypingIndicator(); |
| let errorMsg = error.message || 'Unknown error'; |
| if (error.message.includes('memory')) { |
| errorMsg = 'Phone’s a bit overloaded. Close some apps?'; |
| } else if (error.message.includes('WebGPU')) { |
| errorMsg = 'Need a better connection or browser. Try Chrome?'; |
| } |
| addMessage("Oops, something’s up! Try again?"); |
| showError(`Can’t reply right now: ${errorMsg}`); |
| } finally { |
| isProcessing = false; |
| sendButton.disabled = false; |
| messageInput.disabled = false; |
| messageInput.focus(); |
| } |
| } |
| |
| async function handleSend() { |
| const inputMessage = messageInput.value.trim(); |
| if (!inputMessage || !generator || isProcessing) return; |
| |
| addMessage(inputMessage, true); |
| messageInput.value = ''; |
| |
| await generateResponse(inputMessage); |
| } |
| |
| |
| sendButton.addEventListener('click', handleSend); |
| |
| messageInput.addEventListener('keypress', (e) => { |
| if (e.key === 'Enter' && !e.shiftKey) { |
| e.preventDefault(); |
| handleSend(); |
| } |
| }); |
| </script> |
| </body> |
| </html> |