plantslens / index.html
pavan1221's picture
Upload index.html
f7c4d7a verified
<!DOCTYPE html>
<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>