254 lines
13 KiB
JavaScript
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);
|
|
})();
|