Spaces:
Sleeping
Sleeping
| <html lang="en"> | |
| <head> | |
| <meta charset="UTF-8"> | |
| <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> | |
| <meta name="theme-color" content="#f7f3ee"> | |
| <title>EucalyptusLens</title> | |
| <link href="https://fonts.googleapis.com/css2?family=Cormorant+Garamond:ital,wght@0,300;0,400;0,600;1,300;1,400&family=DM+Mono:wght@300;400;500&family=Outfit:wght@200;300;400;500&display=swap" rel="stylesheet"> | |
| <style> | |
| *,*::before,*::after{box-sizing:border-box;margin:0;padding:0} | |
| :root{ | |
| --bg:#f7f3ee;--bg2:#f0eae0;--surface:#ffffff;--surface2:#faf7f2; | |
| --border:#e2d9cc;--border2:#d4c8b8; | |
| --rose:#c97ba8;--mauve:#9b6fa0;--periwinkle:#7b8fd4; | |
| --teal:#4aabb5;--amber:#d4955a;--sage:#8aaa6a;--lilac:#c4b8e8; | |
| --ink:#2a2035;--text:#3a2f48;--text2:#7a6d88;--text3:#b0a4bc | |
| } | |
| html{scroll-behavior:smooth;-webkit-tap-highlight-color:transparent} | |
| body{background:var(--bg);color:var(--text);font-family:'Outfit',sans-serif;font-weight:300;min-height:100dvh;overflow-x:hidden} | |
| body::before{content:'';position:fixed;inset:0; | |
| background: | |
| radial-gradient(ellipse 70% 50% at 0% 0%,rgba(123,143,212,.12) 0%,transparent 55%), | |
| radial-gradient(ellipse 60% 60% at 100% 0%,rgba(201,123,168,.10) 0%,transparent 50%), | |
| radial-gradient(ellipse 50% 40% at 50% 100%,rgba(74,171,181,.08) 0%,transparent 50%); | |
| pointer-events:none;z-index:0} | |
| *{position:relative;z-index:1} | |
| /* NAV */ | |
| nav{display:flex;align-items:center;justify-content:space-between;padding:18px 32px; | |
| background:rgba(247,243,238,.88);backdrop-filter:blur(24px);-webkit-backdrop-filter:blur(24px); | |
| border-bottom:1px solid var(--border);position:sticky;top:0;z-index:200} | |
| .logo{font-family:'Cormorant Garamond',serif;font-size:22px;font-weight:600; | |
| background:linear-gradient(135deg,var(--mauve),var(--periwinkle)); | |
| -webkit-background-clip:text;-webkit-text-fill-color:transparent;background-clip:text} | |
| .logo span{font-style:italic;background:linear-gradient(135deg,var(--teal),var(--sage)); | |
| -webkit-background-clip:text;-webkit-text-fill-color:transparent;background-clip:text} | |
| .nav-badge{font-family:'DM Mono',monospace;font-size:10px;letter-spacing:.1em;padding:5px 12px; | |
| border-radius:20px;background:linear-gradient(135deg,rgba(123,143,212,.12),rgba(201,123,168,.12)); | |
| border:1px solid var(--lilac);color:var(--mauve)} | |
| /* HERO */ | |
| .hero{text-align:center;padding:56px 24px 44px;animation:fadeUp .7s ease both} | |
| .hero-tag{font-family:'DM Mono',monospace;font-size:10px;letter-spacing:.25em;text-transform:uppercase; | |
| color:var(--teal);margin-bottom:16px;display:flex;align-items:center;justify-content:center;gap:12px} | |
| .hero-tag::before,.hero-tag::after{content:'';width:28px;height:1px;background:var(--border2)} | |
| .hero h1{font-family:'Cormorant Garamond',serif;font-size:clamp(40px,8vw,80px);font-weight:300; | |
| line-height:1.05;color:var(--ink);margin-bottom:14px;letter-spacing:-.02em} | |
| .hero h1 em{font-style:italic;background:linear-gradient(135deg,var(--rose),var(--periwinkle)); | |
| -webkit-background-clip:text;-webkit-text-fill-color:transparent;background-clip:text} | |
| .hero p{font-size:15px;color:var(--text2);max-width:420px;margin:0 auto;line-height:1.75} | |
| .tagline{text-align:center;padding:28px 24px 20px; | |
| font-family:'DM Mono',monospace;font-size:13px; | |
| letter-spacing:.2em;color:var(--text3);text-transform:uppercase} | |
| /* MAIN */ | |
| .main{max-width:960px;margin:0 auto;padding:0 24px 80px} | |
| /* UPLOAD GRID */ | |
| .upload-grid{display:grid;grid-template-columns:1fr 1fr;gap:14px;margin-bottom:14px} | |
| /* FILE UPLOAD BOX */ | |
| .upload-box{border:2px dashed var(--border2);border-radius:16px;padding:44px 20px; | |
| text-align:center;cursor:pointer;background:var(--surface); | |
| transition:all .3s;overflow:hidden;display:block; | |
| text-decoration:none;color:inherit} | |
| .upload-box:hover{border-color:var(--periwinkle);transform:translateY(-2px); | |
| box-shadow:0 8px 28px rgba(123,143,212,.12)} | |
| .upload-box:active{transform:scale(.98)} | |
| .upload-box input[type=file]{position:absolute;inset:0;width:100%;height:100%; | |
| opacity:0;cursor:pointer;font-size:0} | |
| /* CAMERA BOX */ | |
| .camera-box{border:2px dashed var(--border2);border-radius:16px;overflow:hidden; | |
| background:var(--surface);display:flex;flex-direction:column;min-height:200px} | |
| #cameraFeed{width:100%;flex:1;object-fit:cover;display:none} | |
| #cameraCanvas{display:none} | |
| .camera-placeholder{flex:1;display:flex;flex-direction:column;align-items:center; | |
| justify-content:center;padding:28px 20px;text-align:center} | |
| .camera-controls{display:flex;gap:8px;padding:12px;background:var(--bg2); | |
| border-top:1px solid var(--border);justify-content:center} | |
| /* ZONE ICON */ | |
| .zone-icon{width:52px;height:52px;margin:0 auto 14px;border-radius:14px; | |
| display:flex;align-items:center;justify-content:center} | |
| .upload-icon{background:linear-gradient(135deg,rgba(123,143,212,.15),rgba(201,123,168,.15))} | |
| .camera-icon-bg{background:linear-gradient(135deg,rgba(74,171,181,.15),rgba(138,170,106,.15))} | |
| .zone-h3{font-family:'Cormorant Garamond',serif;font-size:18px;font-weight:500;color:var(--ink);margin-bottom:5px} | |
| .zone-p{font-size:12px;color:var(--text3);line-height:1.5} | |
| .formats{font-family:'DM Mono',monospace;font-size:9px;color:var(--text3);margin-top:10px;letter-spacing:.1em} | |
| /* PREVIEW */ | |
| .preview-container{display:none;margin-bottom:14px;border:1px solid var(--border); | |
| border-radius:16px;overflow:hidden;background:var(--surface); | |
| box-shadow:0 4px 20px rgba(123,143,212,.08);animation:fadeUp .3s ease both} | |
| .preview-header{display:flex;align-items:center;justify-content:space-between; | |
| padding:10px 18px;border-bottom:1px solid var(--border);background:var(--bg2)} | |
| .preview-header span{font-family:'DM Mono',monospace;font-size:10px;color:var(--text3); | |
| letter-spacing:.12em;text-transform:uppercase} | |
| #previewImg{width:100%;max-height:260px;object-fit:contain;padding:16px;display:block} | |
| /* BUTTONS */ | |
| .btn{padding:13px 24px;border:none;border-radius:10px;cursor:pointer; | |
| font-family:'DM Mono',monospace;font-size:11px;letter-spacing:.12em; | |
| text-transform:uppercase;transition:all .2s;touch-action:manipulation} | |
| .btn-primary{background:linear-gradient(135deg,var(--periwinkle),var(--mauve));color:#fff; | |
| width:100%;font-size:13px;padding:16px; | |
| box-shadow:0 4px 16px rgba(123,143,212,.25);border-radius:12px} | |
| .btn-primary:hover{transform:translateY(-2px);box-shadow:0 8px 24px rgba(123,143,212,.35)} | |
| .btn-primary:active{transform:scale(.98)} | |
| .btn-primary:disabled{opacity:.4;cursor:not-allowed;transform:none;box-shadow:none} | |
| .btn-sm{background:var(--surface);color:var(--text2);border:1px solid var(--border); | |
| font-size:10px;padding:9px 16px;border-radius:8px} | |
| .btn-sm:hover{background:var(--bg2);border-color:var(--periwinkle);color:var(--periwinkle)} | |
| .btn-danger{background:transparent;color:var(--rose);border:1px solid rgba(201,123,168,.3); | |
| font-size:10px;padding:6px 14px;border-radius:8px} | |
| .btn-danger:hover{background:rgba(201,123,168,.08)} | |
| /* LOADING */ | |
| .loading{display:none;text-align:center;padding:48px 24px;border:1px solid var(--border); | |
| border-radius:16px;background:var(--surface);margin-top:14px} | |
| .spinner{width:44px;height:44px;border-radius:50%;border:2px solid var(--border); | |
| border-top-color:var(--periwinkle);border-right-color:var(--rose); | |
| animation:spin 1s linear infinite;margin:0 auto 18px} | |
| .loading p{font-family:'DM Mono',monospace;font-size:11px;color:var(--text3);letter-spacing:.15em} | |
| .loading-steps{margin-top:14px;display:flex;flex-direction:column;gap:7px;align-items:center} | |
| .loading-step{font-size:11px;color:var(--text3);font-family:'DM Mono',monospace; | |
| opacity:0;animation:fadeIn .5s ease forwards} | |
| /* TOOLTIP */ | |
| .img-wrapper{position:relative;display:inline-block;width:100%} | |
| .img-wrapper img{width:100%;display:block} | |
| .tooltip-dot{position:absolute;width:0;height:0;cursor:pointer;transform:translate(-50%,-50%)} | |
| .plant-tooltip{ | |
| position:absolute; | |
| background:rgba(30,24,40,.85); | |
| color:#fff; | |
| font-family:'DM Mono',monospace; | |
| font-size:11px; | |
| letter-spacing:.05em; | |
| padding:6px 12px; | |
| border-radius:8px; | |
| pointer-events:none; | |
| white-space:nowrap; | |
| opacity:0; | |
| transition:opacity .15s; | |
| z-index:10; | |
| transform:translate(-50%, -110%); | |
| } | |
| .plant-tooltip.visible{opacity:1} | |
| /* RESULT */ | |
| .result-area{display:none;margin-top:14px;animation:fadeUp .5s ease both} | |
| .result-annotated{border:1px solid var(--border);border-radius:16px;overflow:hidden; | |
| background:var(--surface);margin-bottom:20px; | |
| box-shadow:0 8px 32px rgba(123,143,212,.1)} | |
| .result-annotated-header{display:flex;align-items:center;justify-content:space-between; | |
| padding:12px 20px;border-bottom:1px solid var(--border);background:var(--bg2)} | |
| .result-annotated-header span{font-family:'DM Mono',monospace;font-size:10px; | |
| color:var(--text3);letter-spacing:.12em;text-transform:uppercase} | |
| .count-badge{font-family:'DM Mono',monospace;font-size:10px;padding:4px 12px;border-radius:20px; | |
| background:linear-gradient(135deg,rgba(74,171,181,.15),rgba(138,170,106,.15)); | |
| border:1px solid rgba(74,171,181,.3);color:var(--teal)} | |
| #annotatedImg{width:100%;display:block;max-height:520px;object-fit:contain;padding:12px} | |
| .plants-label{font-family:'DM Mono',monospace;font-size:10px;color:var(--text3); | |
| letter-spacing:.2em;text-transform:uppercase;margin-bottom:12px} | |
| .plants-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(280px,1fr));gap:14px} | |
| /* PLANT CARD */ | |
| .plant-card{border-radius:16px;overflow:hidden;background:var(--surface); | |
| border:1px solid var(--border);box-shadow:0 4px 18px rgba(0,0,0,.05)} | |
| .card-header{padding:16px 18px 14px;border-bottom:1px solid var(--border)} | |
| .card-top{display:flex;align-items:flex-start;justify-content:space-between;gap:8px;margin-bottom:4px} | |
| .card-name{font-family:'Cormorant Garamond',serif;font-size:22px;font-weight:500;color:var(--ink);line-height:1.1} | |
| .card-sci{font-family:'Cormorant Garamond',serif;font-size:13px;font-style:italic;color:var(--mauve)} | |
| .conf-badge{font-family:'DM Mono',monospace;font-size:9px;padding:4px 10px;border-radius:20px; | |
| letter-spacing:.1em;text-transform:uppercase;white-space:nowrap;flex-shrink:0} | |
| .conf-high{background:linear-gradient(135deg,rgba(74,171,181,.15),rgba(138,170,106,.15)); | |
| color:var(--teal);border:1px solid rgba(74,171,181,.3)} | |
| .conf-medium{background:rgba(212,149,90,.12);color:var(--amber);border:1px solid rgba(212,149,90,.3)} | |
| .conf-low{background:rgba(201,123,168,.12);color:var(--rose);border:1px solid rgba(201,123,168,.3)} | |
| .card-body{padding:14px 18px} | |
| .card-field{margin-bottom:12px} | |
| .card-field h5{font-family:'DM Mono',monospace;font-size:9px;color:var(--text3); | |
| letter-spacing:.2em;text-transform:uppercase;margin-bottom:5px} | |
| .card-field p{font-size:13px;color:var(--text);line-height:1.65} | |
| .features{display:flex;flex-direction:column;gap:5px} | |
| .feature{display:flex;align-items:flex-start;gap:8px;font-size:12px;color:var(--text2);line-height:1.4} | |
| .dot{width:6px;height:6px;border-radius:50%;flex-shrink:0;margin-top:4px} | |
| .wiki-text{font-size:12px;color:var(--text2);line-height:1.8; | |
| display:-webkit-box;-webkit-line-clamp:4;-webkit-box-orient:vertical;overflow:hidden} | |
| .wiki-link{display:inline-flex;align-items:center;gap:4px;margin-top:8px; | |
| font-family:'DM Mono',monospace;font-size:10px;color:var(--periwinkle); | |
| text-decoration:none;transition:color .2s} | |
| .wiki-link:hover{color:var(--mauve)} | |
| .no-plants{text-align:center;padding:44px 24px;border:1px dashed var(--border2); | |
| border-radius:16px;color:var(--text3)} | |
| .no-plants p{font-family:'Cormorant Garamond',serif;font-size:20px;font-style:italic} | |
| /* FOOTER */ | |
| footer{border-top:1px solid var(--border);padding:24px 32px; | |
| display:flex;align-items:center;justify-content:space-between; | |
| background:var(--surface2);flex-wrap:wrap;gap:12px} | |
| footer p{font-family:'DM Mono',monospace;font-size:10px;color:var(--text3);letter-spacing:.08em} | |
| .stack{display:flex;gap:6px;flex-wrap:wrap} | |
| .stack-badge{font-family:'DM Mono',monospace;font-size:9px;color:var(--text3); | |
| border:1px solid var(--border);padding:4px 10px;border-radius:20px;background:var(--surface)} | |
| /* ANIMATIONS */ | |
| @keyframes fadeUp{from{opacity:0;transform:translateY(16px)}to{opacity:1;transform:translateY(0)}} | |
| @keyframes fadeIn{from{opacity:0}to{opacity:.7}} | |
| @keyframes spin{to{transform:rotate(360deg)}} | |
| /* RESPONSIVE */ | |
| @media(max-width:540px){ | |
| nav{padding:12px 16px} | |
| .hero{padding:32px 16px 28px} | |
| .main{padding:0 12px 60px} | |
| .upload-grid{grid-template-columns:1fr} | |
| .plants-grid{grid-template-columns:1fr} | |
| footer{padding:20px 16px} | |
| } | |
| </style> | |
| </head> | |
| <body> | |
| <nav> | |
| <div class="logo">Eucalyptus<span>Lens</span></div> | |
| <div class="nav-badge">LLaMA 4 Scout Β· Multi-Plant</div> | |
| </nav> | |
| <p class="tagline">Photo. Detect. Identify.</p> | |
| <div class="main"> | |
| <div class="upload-grid"> | |
| <!-- Upload Box β label wraps the input for native browser file picking --> | |
| <label class="upload-box" style="position:relative"> | |
| <input type="file" id="fileInput" accept="image/*"> | |
| <div class="zone-icon upload-icon" style="pointer-events:none"> | |
| <svg width="24" height="24" viewBox="0 0 24 24" fill="none"> | |
| <path d="M12 15V5M8 9l4-4 4 4" stroke="url(#g1)" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/> | |
| <path d="M4 17v1a2 2 0 002 2h12a2 2 0 002-2v-1" stroke="url(#g1)" stroke-width="1.5" stroke-linecap="round"/> | |
| <defs><linearGradient id="g1" x1="4" y1="4" x2="20" y2="20"> | |
| <stop stop-color="#7b8fd4"/><stop offset="1" stop-color="#c97ba8"/> | |
| </linearGradient></defs> | |
| </svg> | |
| </div> | |
| <h3 class="zone-h3">Upload Image</h3> | |
| <p class="zone-p">Click to browse or drop a photo</p> | |
| <div class="formats">JPG Β· PNG Β· WEBP Β· HEIC</div> | |
| </label> | |
| <!-- Camera Box --> | |
| <div class="camera-box"> | |
| <video id="cameraFeed" autoplay playsinline></video> | |
| <canvas id="cameraCanvas"></canvas> | |
| <div class="camera-placeholder" id="cameraPlaceholder"> | |
| <div class="zone-icon camera-icon-bg"> | |
| <svg width="24" height="24" viewBox="0 0 24 24" fill="none"> | |
| <rect x="2" y="7" width="20" height="13" rx="3" stroke="url(#g2)" stroke-width="1.5" fill="none"/> | |
| <circle cx="12" cy="13" r="3.5" stroke="url(#g2)" stroke-width="1.5" fill="none"/> | |
| <path d="M8 7l1.5-2.5h5L16 7" stroke="url(#g2)" stroke-width="1.5" stroke-linecap="round"/> | |
| <defs><linearGradient id="g2" x1="2" y1="7" x2="22" y2="20"> | |
| <stop stop-color="#4aabb5"/><stop offset="1" stop-color="#8aaa6a"/> | |
| </linearGradient></defs> | |
| </svg> | |
| </div> | |
| <h3 class="zone-h3">Use Camera</h3> | |
| <p class="zone-p">Take a live photo</p> | |
| </div> | |
| <div class="camera-controls"> | |
| <button class="btn btn-sm" id="btnStart" onclick="startCamera()">Start</button> | |
| <button class="btn btn-sm" id="btnCapture" onclick="capturePhoto()" style="display:none">Capture</button> | |
| <button class="btn btn-sm" id="btnStop" onclick="stopCamera()" style="display:none">Stop</button> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- Preview --> | |
| <div class="preview-container" id="previewBox"> | |
| <div class="preview-header"> | |
| <span>Preview</span> | |
| <button class="btn btn-danger" onclick="clearImage()">Clear</button> | |
| </div> | |
| <img id="previewImg" src="" alt="Preview"> | |
| </div> | |
| <!-- Analyze button --> | |
| <button class="btn btn-primary" id="btnAnalyze" onclick="analyze()" disabled> | |
| Detect All Plants | |
| </button> | |
| <!-- Loading --> | |
| <div class="loading" id="loadingBox"> | |
| <div class="spinner"></div> | |
| <p>Scanning for plants...</p> | |
| <div class="loading-steps"> | |
| <div class="loading-step" style="animation-delay:.5s">Processing image with LLaMA 4 Scout</div> | |
| <div class="loading-step" style="animation-delay:2s">Identifying all plants in frame</div> | |
| <div class="loading-step" style="animation-delay:3.5s">Fetching Wikipedia botanical data</div> | |
| <div class="loading-step" style="animation-delay:5s">Annotating image with labels</div> | |
| </div> | |
| </div> | |
| <!-- Results --> | |
| <div class="result-area" id="resultArea"> | |
| <div class="result-annotated"> | |
| <div class="result-annotated-header"> | |
| <span>Annotated Result</span> | |
| <div class="count-badge" id="countBadge">0 plants</div> | |
| </div> | |
| <div class="img-wrapper" id="imgWrapper"> | |
| <img id="annotatedImg" src="" alt="Annotated"> | |
| <div id="tooltipDots"></div> | |
| <div class="plant-tooltip" id="plantTooltip"></div> | |
| </div> | |
| </div> | |
| <div class="plants-label" id="plantsLabel"></div> | |
| <div class="plants-grid" id="plantsGrid"></div> | |
| </div> | |
| </div> | |
| <footer> | |
| <p>EucalyptusLens β AI Botanical Intelligence</p> | |
| <div class="stack"> | |
| <span class="stack-badge">LLaMA 4 Scout</span> | |
| <span class="stack-badge">Groq</span> | |
| <span class="stack-badge">Wikipedia</span> | |
| <span class="stack-badge">Flask</span> | |
| </div> | |
| </footer> | |
| <script> | |
| let currentFile = null; | |
| let stream = null; | |
| const COLORS = [ | |
| 'rgb(99,143,212)','rgb(201,123,168)','rgb(74,171,181)', | |
| 'rgb(212,149,90)','rgb(138,170,106)','rgb(220,120,100)' | |
| ]; | |
| // ββ File Input ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ | |
| document.getElementById('fileInput').onchange = function() { | |
| if (this.files && this.files[0]) showPreview(this.files[0]); | |
| }; | |
| function showPreview(file) { | |
| currentFile = file; | |
| const reader = new FileReader(); | |
| reader.onload = function(e) { | |
| document.getElementById('previewImg').src = e.target.result; | |
| document.getElementById('previewBox').style.display = 'block'; | |
| document.getElementById('btnAnalyze').disabled = false; | |
| document.getElementById('resultArea').style.display = 'none'; | |
| }; | |
| reader.readAsDataURL(file); | |
| } | |
| function clearImage() { | |
| currentFile = null; | |
| document.getElementById('previewBox').style.display = 'none'; | |
| document.getElementById('btnAnalyze').disabled = true; | |
| document.getElementById('resultArea').style.display = 'none'; | |
| document.getElementById('fileInput').value = ''; | |
| } | |
| // ββ Camera ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ | |
| async function startCamera() { | |
| try { | |
| stream = await navigator.mediaDevices.getUserMedia({ video: { facingMode: 'environment' } }); | |
| const v = document.getElementById('cameraFeed'); | |
| v.srcObject = stream; | |
| v.style.display = 'block'; | |
| document.getElementById('cameraPlaceholder').style.display = 'none'; | |
| document.getElementById('btnStart').style.display = 'none'; | |
| document.getElementById('btnCapture').style.display = 'inline-block'; | |
| document.getElementById('btnStop').style.display = 'inline-block'; | |
| } catch(e) { | |
| alert('Camera access denied: ' + e.message); | |
| } | |
| } | |
| function capturePhoto() { | |
| const v = document.getElementById('cameraFeed'); | |
| const c = document.getElementById('cameraCanvas'); | |
| c.width = v.videoWidth; | |
| c.height = v.videoHeight; | |
| c.getContext('2d').drawImage(v, 0, 0); | |
| c.toBlob(blob => { | |
| showPreview(new File([blob], 'camera.jpg', { type: 'image/jpeg' })); | |
| stopCamera(); | |
| }, 'image/jpeg', 0.92); | |
| } | |
| function stopCamera() { | |
| if (stream) { stream.getTracks().forEach(t => t.stop()); stream = null; } | |
| document.getElementById('cameraFeed').style.display = 'none'; | |
| document.getElementById('cameraPlaceholder').style.display = 'flex'; | |
| document.getElementById('btnStart').style.display = 'inline-block'; | |
| document.getElementById('btnCapture').style.display = 'none'; | |
| document.getElementById('btnStop').style.display = 'none'; | |
| } | |
| // ββ Analyze βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ | |
| async function analyze() { | |
| if (!currentFile) return; | |
| document.getElementById('loadingBox').style.display = 'block'; | |
| document.getElementById('resultArea').style.display = 'none'; | |
| document.getElementById('btnAnalyze').disabled = true; | |
| const fd = new FormData(); | |
| fd.append('file', currentFile); | |
| try { | |
| const res = await fetch('/analyze', { method: 'POST', body: fd }); | |
| const data = await res.json(); | |
| if (!res.ok || data.error) { | |
| alert('Error: ' + (data.error || res.statusText)); | |
| return; | |
| } | |
| showResults(data); | |
| } catch(e) { | |
| alert('Connection error: ' + e.message); | |
| } finally { | |
| document.getElementById('loadingBox').style.display = 'none'; | |
| document.getElementById('btnAnalyze').disabled = false; | |
| } | |
| } | |
| // ββ Show Results ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ | |
| function showResults(data) { | |
| const plants = data.plants || []; | |
| const count = plants.length; | |
| const dots = data.dot_positions || []; | |
| document.getElementById('annotatedImg').src = data.annotated_image || ''; | |
| document.getElementById('countBadge').textContent = count === 1 ? '1 plant found' : count + ' plants found'; | |
| document.getElementById('plantsLabel').textContent = count > 0 ? 'Identified Plants' : ''; | |
| // Build tooltip dots | |
| const dotsEl = document.getElementById('tooltipDots'); | |
| const tooltip = document.getElementById('plantTooltip'); | |
| dotsEl.innerHTML = ''; | |
| dots.forEach(d => { | |
| const dot = document.createElement('div'); | |
| dot.className = 'tooltip-dot'; | |
| dot.style.left = d.x_pct + '%'; | |
| dot.style.top = d.y_pct + '%'; | |
| dot.style.width = '40px'; | |
| dot.style.height = '40px'; | |
| dot.style.transform = 'translate(-50%,-50%)'; | |
| dot.style.position = 'absolute'; | |
| dot.addEventListener('mouseenter', function() { | |
| tooltip.textContent = d.name; | |
| tooltip.style.left = d.x_pct + '%'; | |
| tooltip.style.top = d.y_pct + '%'; | |
| tooltip.classList.add('visible'); | |
| }); | |
| dot.addEventListener('mouseleave', function() { | |
| tooltip.classList.remove('visible'); | |
| }); | |
| // Mobile tap support | |
| dot.addEventListener('touchstart', function(e) { | |
| e.preventDefault(); | |
| tooltip.textContent = d.name; | |
| tooltip.style.left = d.x_pct + '%'; | |
| tooltip.style.top = d.y_pct + '%'; | |
| tooltip.classList.add('visible'); | |
| setTimeout(() => tooltip.classList.remove('visible'), 2000); | |
| }); | |
| dotsEl.appendChild(dot); | |
| }); | |
| const grid = document.getElementById('plantsGrid'); | |
| grid.innerHTML = ''; | |
| if (count === 0) { | |
| grid.innerHTML = '<div class="no-plants"><p>No plants detected in this image.</p></div>'; | |
| } else { | |
| plants.forEach((p, i) => grid.appendChild(makeCard(p, i))); | |
| } | |
| document.getElementById('resultArea').style.display = 'block'; | |
| document.getElementById('resultArea').scrollIntoView({ behavior: 'smooth', block: 'start' }); | |
| } | |
| function makeCard(p, i) { | |
| const color = COLORS[i % COLORS.length]; | |
| const conf = (p.confidence || 'low').toLowerCase(); | |
| const div = document.createElement('div'); | |
| div.className = 'plant-card'; | |
| div.style.borderTop = '3px solid ' + color; | |
| const features = (p.key_features || []).slice(0, 4) | |
| .map(f => `<div class="feature"><div class="dot" style="background:${color}"></div><span>${f}</span></div>`) | |
| .join(''); | |
| div.innerHTML = ` | |
| <div class="card-header" style="background:linear-gradient(135deg,${color}18,${color}08)"> | |
| <div class="card-top"> | |
| <div> | |
| <div class="card-name">${p.common_name || 'Unknown'}</div> | |
| <div class="card-sci">${p.scientific_name || ''}</div> | |
| </div> | |
| <div class="conf-badge conf-${conf}">${conf}</div> | |
| </div> | |
| </div> | |
| <div class="card-body"> | |
| ${p.family ? `<div class="card-field"><h5>Family</h5><p>${p.family}</p></div>` : ''} | |
| ${features ? `<div class="card-field"><h5>Key Features</h5><div class="features">${features}</div></div>` : ''} | |
| ${p.wiki_summary ? ` | |
| <div class="card-field"> | |
| <h5>Botanical Overview</h5> | |
| <p class="wiki-text">${p.wiki_summary}</p> | |
| ${p.wiki_url ? `<a class="wiki-link" href="${p.wiki_url}" target="_blank" rel="noopener">Read on Wikipedia β</a>` : ''} | |
| </div>` : ''} | |
| </div>`; | |
| return div; | |
| } | |
| </script> | |
| </body> | |
| </html> | |