Files
html/wevia-claude.js
2026-04-12 22:57:03 +02:00

254 lines
13 KiB
JavaScript

/* WEVIA Claude UX v3 — Deep wire from Claude Code source patterns */
(function(){
if(document.getElementById('wcux'))return;
var css=document.createElement('style');
css.textContent=`
/* HINT TEXT CENTERING FIX */
.input-hint,.hint,.footer-hint{text-align:center!important;display:block!important;width:100%!important;margin:0 auto!important}
.input-wrap+*,.input-box~div,.input-box~span,.input-box~p{text-align:center!important}
/* Footer area below input */
.footer,.input-footer,.bottom-bar{text-align:center!important;display:flex!important;justify-content:center!important}
/* FIX: invisible spans (empty icon containers) */
span:empty{display:none!important}
/* FIX: font unification - Inter everywhere */
*{font-family:Inter,system-ui,-apple-system,sans-serif!important}
pre,code,.meta,.speed-ind,.prov-indicator,.ctx-label,.artifact-title{font-family:"JetBrains Mono",monospace!important}
h1.logo,h1[style*="letter-spacing"]{font-family:Inter,system-ui,sans-serif!important}
/* FIX: sidebar items no overlap (proper spacing) */
.sb-item{margin-bottom:1px!important;position:relative!important;z-index:1!important}
.sb-item+.sb-item{margin-top:0!important}
/* ACTION BADGE */
.action-badge{margin:8px 12px;padding:6px 10px;background:rgba(239,68,68,.1);border:1px solid rgba(239,68,68,.2);border-radius:8px;font-size:11px;color:#fca5a5;cursor:pointer;text-align:center;transition:.15s}
.action-badge:hover{background:rgba(239,68,68,.15);transform:translateY(-1px)}
/* 1. CONTEXT WINDOW BAR (Claude Code: context window management) */
.ctx-bar{position:fixed;top:0;left:0;right:0;height:2px;z-index:9999;background:rgba(255,255,255,.03)}
.ctx-fill{height:100%;background:linear-gradient(90deg,#22c55e,#eab308,#ef4444);border-radius:0 2px 2px 0;transition:width .5s ease}
.ctx-label{position:fixed;top:4px;right:80px;font-size:9px;color:rgba(255,255,255,.15);font-family:'JetBrains Mono',monospace;font-variant-numeric:tabular-nums;z-index:9999;opacity:0;transition:.2s}
.ctx-bar:hover+.ctx-label,.ctx-label:hover{opacity:1}
/* 2. SLASH COMMANDS (Claude Code: /review /mcp style) */
.slash-menu{position:absolute;bottom:100%;left:0;right:0;background:#18181b;border:1px solid rgba(255,255,255,.08);border-radius:10px;max-height:240px;overflow-y:auto;display:none;z-index:100;box-shadow:0 -8px 30px rgba(0,0,0,.4)}
.slash-menu.show{display:block}
.slash-item{display:flex;align-items:center;gap:10px;padding:8px 14px;cursor:pointer;transition:.1s}
.slash-item:hover,.slash-item.active{background:rgba(255,255,255,.05)}
.slash-item .si-cmd{font-family:'JetBrains Mono',monospace;font-size:12px;color:#fafafa;min-width:100px}
.slash-item .si-desc{font-size:11px;color:#71717a}
/* 3. PROVIDER STATUS (Claude Code: model indicator) */
.prov-indicator{display:inline-flex;align-items:center;gap:4px;padding:2px 8px;border-radius:4px;font-size:9px;font-family:'JetBrains Mono',monospace;color:rgba(255,255,255,.25);background:rgba(255,255,255,.02);border:1px solid rgba(255,255,255,.04)}
.prov-dot{width:5px;height:5px;border-radius:50%;background:#22c55e}
/* 4. THINKING SECTION (Claude Code: extended thinking) */
.think-section{margin:6px 0;border-radius:10px;border:1px solid rgba(255,255,255,.04);overflow:hidden}
.think-header{display:flex;align-items:center;gap:8px;padding:7px 12px;cursor:pointer;font-size:11px;color:#71717a;background:rgba(255,255,255,.02)}
.think-header:hover{background:rgba(255,255,255,.04)}
.think-header svg{width:12px;height:12px;transition:transform .2s}
.think-header.open svg{transform:rotate(90deg)}
.think-content{display:none;padding:8px 14px;font-size:12px;color:#a1a1aa;line-height:1.6;border-top:1px solid rgba(255,255,255,.04)}
.think-content.show{display:block}
/* 5. RAG RECALL BADGE (memory indicator) */
.rag-badge{display:inline-flex;align-items:center;gap:4px;padding:2px 8px;border-radius:12px;font-size:9px;color:#a1a1aa;background:rgba(139,92,246,.08);border:1px solid rgba(139,92,246,.12)}
.rag-badge .rb-dot{width:4px;height:4px;border-radius:50%;background:#8b5cf6}
/* 6. RETRY BUTTON (error recovery) */
.retry-btn{display:inline-flex;align-items:center;gap:4px;padding:4px 12px;border-radius:6px;font-size:11px;color:#fafafa;background:rgba(239,68,68,.12);border:1px solid rgba(239,68,68,.2);cursor:pointer;transition:.15s;font-family:Inter,system-ui,sans-serif}
.retry-btn:hover{background:rgba(239,68,68,.2)}
/* 7. FILE ATTACHMENT CHIP */
.file-chip{display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:8px;font-size:11px;color:#a1a1aa;background:rgba(255,255,255,.04);border:1px solid rgba(255,255,255,.06)}
.file-chip .fc-ico{font-size:13px}
.file-chip .fc-name{max-width:120px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
.file-chip .fc-size{font-size:9px;color:#71717a}
/* 8. STREAMING SPEED INDICATOR */
.speed-ind{font-size:9px;color:rgba(255,255,255,.15);font-family:'JetBrains Mono',monospace;font-variant-numeric:tabular-nums}
/* 9. EMPTY STATE (Claude-style welcome) */
.welcome h1{font-family:Inter,system-ui,sans-serif!important;letter-spacing:-.04em!important}
/* FILE ARTIFACT */
.file-artifact{margin:8px 0;display:inline-block}
.fa-link{display:flex;align-items:center;gap:10px;padding:10px 16px;background:#18181b;border:1px solid rgba(255,255,255,.08);border-radius:10px;text-decoration:none;color:#fafafa;transition:.15s;font-size:13px}
.fa-link:hover{background:#27272a;border-color:rgba(34,197,94,.2);transform:translateY(-1px);box-shadow:0 4px 12px rgba(0,0,0,.3)}
.fa-name{font-family:"JetBrains Mono",monospace;font-size:12px;color:#a1a1aa}
.fa-dl{font-size:10px;color:#22c55e;margin-left:auto;padding:2px 8px;border-radius:4px;background:rgba(34,197,94,.08);border:1px solid rgba(34,197,94,.15)}
/* 10. INPUT ENHANCEMENTS */
.input-wrap textarea{font-family:Inter,system-ui,sans-serif!important;font-size:14px!important;line-height:1.5!important;color:#fafafa!important}
.input-wrap textarea::placeholder{color:#52525b!important;font-size:13px!important}
`;
document.head.appendChild(css);
// === CONTEXT BAR ===
var ctxBar=document.createElement('div');ctxBar.className='ctx-bar';
ctxBar.innerHTML='<div class="ctx-fill" style="width:0%"></div>';
document.body.appendChild(ctxBar);
var ctxLabel=document.createElement('div');ctxLabel.className='ctx-label';ctxLabel.textContent='0 tokens';
document.body.appendChild(ctxLabel);
// Track tokens from streaming
var totalTokens=0;
var _origFinish=window.finishStream;
if(_origFinish){
window.finishStream=function(el,meta){
_origFinish(el,meta);
if(meta&&meta.tokens){
totalTokens+=meta.tokens;
var pct=Math.min(100,totalTokens/128000*100);
ctxBar.querySelector('.ctx-fill').style.width=pct+'%';
ctxLabel.textContent=totalTokens.toLocaleString()+' tokens';
}
// Add RAG badge if memory was used
if(meta&&meta.rag_score){
var badge=document.createElement('span');
badge.className='rag-badge';
badge.innerHTML='<span class="rb-dot"></span>RAG '+meta.rag_score;
el.closest('.body').querySelector('.meta').prepend(badge);
}
// Mark stream done (remove cursor)
el.closest('.msg').classList.add('stream-done');
};
}
// === SLASH COMMANDS ===
var slashCmds=[
{cmd:'/scan',desc:'Scan infrastructure S204'},
{cmd:'/docker',desc:'Status Docker containers'},
{cmd:'/ssl',desc:'Scan SSL certificats'},
{cmd:'/providers',desc:'Liste providers IA'},
{cmd:'/ethica',desc:'Stats Ethica HCPs'},
{cmd:'/consensus',desc:'Multi-IA consensus'},
{cmd:'/dream',desc:'autoDream consolidation'},
{cmd:'/kairos',desc:'KAIROS daemon proactif'},
{cmd:'/dark',desc:'Dark tools (nuclei/gitleaks)'},
{cmd:'/enterprise',desc:'Dashboard entreprise'},
{cmd:'/compare',desc:'WEVIA vs Opus 4.6'},
{cmd:'/architect',desc:'Architecture complète'},
];
var menu=document.createElement('div');menu.className='slash-menu';
slashCmds.forEach(function(c){
var item=document.createElement('div');item.className='slash-item';
item.innerHTML='<span class="si-cmd">'+c.cmd+'</span><span class="si-desc">'+c.desc+'</span>';
item.onclick=function(){
var input=document.getElementById('input');
if(input){
// Map slash to actual command
var cmdMap={'/scan':'audit infra S204 complet','/docker':'docker containers status','/ssl':'scan SSL weval-consulting.com','/providers':'liste tous les providers IA','/ethica':'status ethica medecins validated','/consensus':'consensus expert meilleur LLM pour WEVAL','/dream':'dream consolider memoires Qdrant','/kairos':'kairos daemon proactif alertes','/dark':'scan nuclei vulnerabilites','/enterprise':'dashboard entreprise WEVAL stats','/compare':'compare WEVIA Master vs Opus 4.6','/architect':'architecture WEVIA complete'};
input.value=cmdMap[c.cmd]||c.cmd.slice(1);
menu.classList.remove('show');
if(typeof send==='function')send();
}
};
menu.appendChild(item);
});
var inputWrap=document.querySelector('.input-wrap,.input-box');
if(inputWrap){inputWrap.style.position='relative';inputWrap.appendChild(menu);}
// Show slash menu on /
var input=document.getElementById('input');
if(input){
input.addEventListener('input',function(){
var v=this.value;
if(v.startsWith('/')){
var filter=v.toLowerCase();
var items=menu.querySelectorAll('.slash-item');
var any=false;
items.forEach(function(it){
var match=it.querySelector('.si-cmd').textContent.toLowerCase().startsWith(filter);
it.style.display=match?'flex':'none';
if(match)any=true;
});
if(any)menu.classList.add('show');
else menu.classList.remove('show');
}else{
menu.classList.remove('show');
}
});
input.addEventListener('keydown',function(e){
if(e.key==='Escape')menu.classList.remove('show');
});
}
// === THINKING TOGGLE ===
document.addEventListener('click',function(e){
var h=e.target.closest('.think-header');
if(h){h.classList.toggle('open');var c=h.nextElementSibling;if(c)c.classList.toggle('show');}
});
// === FILE ARTIFACT DETECTION ===
// Override md() to detect file links and render download cards
var _md2 = window.md;
if (_md2) {
window.md = function(text) {
var h = _md2(text);
// Detect file links: weval-doc-*.html, *.pdf, /exports/*, /api/exports/*
h = h.replace(/(https?:\/\/[^\s"<]+\.(html|pdf|csv|json|xlsx)[^\s"<]*)/gi, function(url) {
var ext = url.split(".").pop().split("?")[0].toLowerCase();
var icon = ext==="pdf"?"📄":ext==="html"?"🌐":ext==="csv"?"📊":ext==="json"?"📋":"📎";
var name = url.split("/").pop().split("?")[0];
return '<div class="file-artifact"><a href="'+url+'" target="_blank" class="fa-link">'+icon+' <span class="fa-name">'+name+'</span><span class="fa-dl">Télécharger ↗</span></a></div>';
});
// Also detect quoted filenames like "weval-doc-*.html"
h = h.replace(/[""](weval-doc-[^""]+\.(html|pdf))[""]/gi, function(m, fname) {
var url = "/api/exports/" + fname;
var ext = fname.split(".").pop();
var icon = ext==="pdf"?"📄":"🌐";
return '<div class="file-artifact"><a href="'+url+'" target="_blank" class="fa-link">'+icon+' <span class="fa-name">'+fname+'</span><span class="fa-dl">Ouvrir ↗</span></a></div>';
});
return h;
};
}
// === ACTION BADGE (fixed) ===
setTimeout(function(){
fetch("/api/wevia-actions.php").then(function(r){return r.json()}).then(function(d){
var u = d.summary ? d.summary.urgent : 0;
var c = document.getElementById("action-badge-container");
if (u > 0 && c) {
c.innerHTML = '<div style="padding:6px 10px;background:rgba(239,68,68,.12);border:1px solid rgba(239,68,68,.2);border-radius:8px;font-size:11px;color:#fca5a5;cursor:pointer;text-align:center" onclick="document.getElementById(\'input\').value=\'actions urgentes Yacine\';send()">⚠️ ' + u + ' actions urgentes</div>';
}
}).catch(function(){});
}, 3000);
// === ACTIONS SIDEBAR + BADGE ===
setTimeout(function(){
var sb = document.querySelector(".sidebar");
if (!sb) return;
// Add ACTIONS section
var raccourcis = null;
sb.querySelectorAll(".sb-section").forEach(function(s){if(s.textContent.trim()==="RACCOURCIS")raccourcis=s;});
if (raccourcis && !document.getElementById("actions-sidebar")) {
var sec = document.createElement("div");sec.className="sb-section";sec.id="actions-sidebar";sec.style.color="#ef4444";sec.textContent="ACTIONS";
var item = document.createElement("div");item.className="sb-item";
item.innerHTML="<span>⚠️</span> Actions Yacine";
item.onclick=function(){document.getElementById("input").value="actions urgentes Yacine";if(typeof send==="function")send();};
raccourcis.parentNode.insertBefore(item, raccourcis);
raccourcis.parentNode.insertBefore(sec, item);
}
// Fetch badge count
fetch("/api/wevia-actions.php").then(function(r){return r.json()}).then(function(d){
var u = d.summary ? d.summary.urgent : 0;
var item = document.querySelector("#actions-sidebar + .sb-item");
if (item && u > 0) item.innerHTML = "<span>🔴</span> " + u + " Actions Yacine";
}).catch(function(){});
}, 2000);
var d=document.createElement('div');d.id='wcux';document.body.appendChild(d);
})();