Files
html/mission-billing.html

280 lines
14 KiB
HTML
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<!DOCTYPE html><html lang="fr"><head><meta charset="UTF-8"><title>Mission Billing — WEVAL</title>
<style>
body{font-family:system-ui;background:#0a0e1a;color:#e2e8f0;margin:0;padding:18px}
h1{color:#a5b4fc;font-size:1.3rem;margin:0 0 4px}
.sub{color:#64748b;font-size:.82rem;margin-bottom:14px}
.breadcrumb{color:#64748b;font-size:.78rem;margin-bottom:8px}
.breadcrumb a{color:#a5b4fc;text-decoration:none}
.sel-row{display:flex;gap:10px;align-items:center;margin-bottom:16px;background:#151e33;padding:12px;border-radius:8px;flex-wrap:wrap}
select, input{background:#0a0e1a;border:1px solid #1e293b;color:#e2e8f0;padding:8px 10px;border-radius:6px;font:.85rem system-ui}
select:focus, input:focus{border-color:#6366f1;outline:none}
.btn{padding:8px 14px;background:#6366f1;color:#fff;border:none;border-radius:6px;cursor:pointer;font-weight:500;font-size:.85rem}
.btn:hover{background:#4f46e5}
.mission-info{background:#151e33;padding:14px;border-radius:8px;margin-bottom:14px}
.kpis{display:grid;grid-template-columns:repeat(auto-fit,minmax(150px,1fr));gap:10px;margin-bottom:16px}
.kpi{background:#151e33;border:1px solid #1e293b;border-radius:8px;padding:12px}
.kpi-val{font-size:1.4rem;color:#22d3ee;font-weight:700}
.kpi-lbl{color:#94a3b8;font-size:.7rem;text-transform:uppercase}
table{width:100%;border-collapse:collapse;font-size:.83rem;background:#151e33;border-radius:8px;overflow:hidden}
th{background:#1e1b4b;color:#a5b4fc;padding:10px 8px;text-align:left;font-size:.72rem;text-transform:uppercase;letter-spacing:.3px}
td{padding:10px 8px;border-bottom:1px solid #1e293b}
td.num{text-align:right;font-family:'JetBrains Mono',monospace}
tr:hover td{background:#1e293b}
.tot-row td{background:#1e1b4b;font-weight:700;color:#a5b4fc;font-size:.95rem;border-top:2px solid #6366f1}
.add-form{background:#151e33;padding:14px;border-radius:8px;margin-top:14px;display:flex;gap:10px;align-items:center;flex-wrap:wrap}
.tag{background:#0a0e1a;border:1px solid #334155;color:#94a3b8;padding:2px 8px;border-radius:3px;font-size:.72rem}
</style><!-- DOCTRINE-60-UX-ENRICH direct-inject-20260424-144053 -->
<style id="doctrine60-ux-direct">
/* DOCTRINE-60-UX-ENRICH injected-direct */
body::before {
content: '';
position: fixed;
top: 0; left: 0; width: 100vw; height: 100vh;
background: radial-gradient(circle at 50% 50%, rgba(100,180,255,0.08), transparent 60%);
pointer-events: none;
z-index: -1;
}
.card, .kpi, .panel, .btn {
transition: all 0.3s cubic-bezier(0.2,0,0.1,1);
}
.card:hover, .kpi:hover, .panel:hover {
box-shadow: 0 4px 20px rgba(100,180,255,0.2);
border-color: rgba(100,180,255,0.5);
}
@keyframes pulseD60 {
0%,100% { opacity: 1; transform: scale(1); }
50% { opacity: 0.7; transform: scale(1.05); }
}
.pulse, .live-indicator, .active, .online {
animation: pulseD60 3s ease-in-out infinite;
}
.modal, .chat, .speech, .overlay {
backdrop-filter: blur(12px);
-webkit-backdrop-filter: blur(12px);
}
.enter-stagger {
animation: enterStagD60 0.5s cubic-bezier(0.2,0,0.1,1) forwards;
}
@keyframes enterStagD60 {
from { opacity: 0; transform: translateY(20px); }
to { opacity: 1; transform: translateY(0); }
}
</style>
</head>
<body>
<div class="breadcrumb"><a href="/">Home</a> / <a href="/candidates-pool.html">Candidates</a> / Mission Billing</div>
<h1>💰 Mission Billing — Simulation Facturation</h1>
<div class="sub">Style BASELINE — TJM × Jours × Mois — Commissions apporteur/manager</div>
<div class="sel-row">
<label>Mission:</label>
<select id="sel-mission" onchange="loadMission()"></select>
<button class="btn" onclick="loadMission()">↻ Refresh</button>
</div>
<div id="content">Loading missions...</div>
<script>
const TENANT = new URLSearchParams(location.search).get('tenant') || 'weval';
let currentMission = null;
async function init() {
const r = await fetch(`/api/em/missions?tenant=${TENANT}`);
const d = await r.json();
const sel = document.getElementById('sel-mission');
sel.innerHTML = (d.missions || []).map(m =>
`<option value="${m.id}">${m.mission_code}${m.client_name || m.client_code} (${m.consultant_name || '—'})</option>`
).join('') || '<option value="">Aucune mission</option>';
if (d.missions && d.missions.length) { sel.value = d.missions[0].id; loadMission(); }
}
async function loadMission() {
const id = document.getElementById('sel-mission').value;
if (!id) return;
const r = await fetch(`/api/em/missions/${id}?tenant=${TENANT}`);
currentMission = await r.json();
render();
}
function render() {
const m = currentMission;
if (!m) return;
const billing = m.billing || [];
const totalDays = billing.reduce((s,b) => s + parseFloat(b.days_worked || 0), 0);
const totalGross = billing.reduce((s,b) => s + parseFloat(b.gross_amount || 0), 0);
const totalCash = billing.reduce((s,b) => s + parseFloat(b.cash_consultant || 0), 0);
const totalChafik = billing.reduce((s,b) => s + parseFloat(b.gross_amount||0) * parseFloat(b.commission_chafik||0), 0);
const totalYoussef = billing.reduce((s,b) => s + parseFloat(b.gross_amount||0) * parseFloat(b.commission_youssef||0), 0);
let html = `
<div class="mission-info">
<strong style="font-size:1rem;color:#a5b4fc">${m.mission_code}</strong> —
<span class="tag">${m.client_name || m.client_code}</span>
<span class="tag">Consultant: ${m.consultant_name || '—'}</span>
<span class="tag">${m.role || '—'}</span>
<span class="tag">TJM: ${parseFloat(m.tjm).toLocaleString('fr-FR')} MAD</span>
<span class="tag">Statut: ${m.status}</span>
<span class="tag">Start: ${m.start_date || '—'}</span>
</div>
<div class="kpis">
<div class="kpi"><div class="kpi-val">${totalDays.toFixed(1)}</div><div class="kpi-lbl">Jours total</div></div>
<div class="kpi"><div class="kpi-val">${Math.round(totalGross).toLocaleString('fr-FR')}</div><div class="kpi-lbl">Brut (MAD)</div></div>
<div class="kpi"><div class="kpi-val" style="color:#10b981">${Math.round(totalCash).toLocaleString('fr-FR')}</div><div class="kpi-lbl">Cash consultant</div></div>
<div class="kpi"><div class="kpi-val" style="color:#f59e0b">${Math.round(totalChafik).toLocaleString('fr-FR')}</div><div class="kpi-lbl">Com. Chafik (5%)</div></div>
<div class="kpi"><div class="kpi-val" style="color:#a855f7">${Math.round(totalYoussef).toLocaleString('fr-FR')}</div><div class="kpi-lbl">Com. Youssef (25%)</div></div>
<div class="kpi"><div class="kpi-val">${billing.length}</div><div class="kpi-lbl">Périodes</div></div>
</div>
<table><thead><tr>
<th>Mois</th><th>J/Homme</th><th>TJM</th><th>Brut</th>
<th>Com. Chafik</th><th>Com. Youssef</th><th>Cash Consultant</th><th>Facturé</th><th>Payé</th>
</tr></thead><tbody>`;
billing.forEach(b => {
const gross = parseFloat(b.gross_amount || 0);
const cChafik = gross * parseFloat(b.commission_chafik || 0);
const cYoussef = gross * parseFloat(b.commission_youssef || 0);
const cash = parseFloat(b.cash_consultant || 0);
const month = new Date(b.period_month).toLocaleDateString('fr-FR', {month:'long', year:'numeric'});
html += `<tr>
<td><strong>${month}</strong></td>
<td class="num">${parseFloat(b.days_worked).toFixed(1)}</td>
<td class="num">${parseFloat(b.tjm_applied).toLocaleString('fr-FR')}</td>
<td class="num">${Math.round(gross).toLocaleString('fr-FR')}</td>
<td class="num" style="color:#f59e0b">${Math.round(cChafik).toLocaleString('fr-FR')}</td>
<td class="num" style="color:#a855f7">${Math.round(cYoussef).toLocaleString('fr-FR')}</td>
<td class="num" style="color:#10b981"><strong>${Math.round(cash).toLocaleString('fr-FR')}</strong></td>
<td>${b.invoiced ? '✅' : '—'}</td>
<td>${b.paid ? '✅' : '—'}</td>
</tr>`;
});
html += `<tr class="tot-row">
<td>TOTAL</td>
<td class="num">${totalDays.toFixed(1)}</td><td>—</td>
<td class="num">${Math.round(totalGross).toLocaleString('fr-FR')}</td>
<td class="num">${Math.round(totalChafik).toLocaleString('fr-FR')}</td>
<td class="num">${Math.round(totalYoussef).toLocaleString('fr-FR')}</td>
<td class="num">${Math.round(totalCash).toLocaleString('fr-FR')}</td>
<td>—</td><td>—</td>
</tr></tbody></table>
<div class="add-form">
<strong style="color:#a5b4fc">+ Ajouter période</strong>
<input id="new-period" type="month" value="${new Date().toISOString().substring(0,7)}">
<input id="new-days" type="number" step="0.5" placeholder="Jours" style="width:90px">
<input id="new-tjm" type="number" placeholder="TJM" value="${m.tjm || 2470}" style="width:100px">
<button class="btn" onclick="addPeriod()">Ajouter</button>
</div>
`;
document.getElementById('content').innerHTML = html;
}
async function addPeriod() {
const period = document.getElementById('new-period').value + '-01';
const days = parseFloat(document.getElementById('new-days').value);
const tjm = parseFloat(document.getElementById('new-tjm').value);
if (!days || !tjm) { alert('Jours + TJM requis'); return; }
const r = await fetch(`/api/em/missions/${currentMission.id}/billing`, {
method:'POST', headers:{'Content-Type':'application/json'},
body: JSON.stringify({period_month: period, days_worked: days, tjm_applied: tjm})
});
const d = await r.json();
if (d.ok) { alert(`✅ Facturé: ${Math.round(d.gross).toLocaleString('fr-FR')} MAD (cash ${Math.round(d.cash_consultant).toLocaleString('fr-FR')})`); loadMission(); }
else alert('❌ '+d.error);
}
init();
</script>
<!-- === OPUS UNIVERSAL DRILL-DOWN v1 19avr — append-only, doctrine #14 === -->
<script>
(function(){
if (window.__opusUniversalDrill) return; window.__opusUniversalDrill = true;
var d = document;
var m = d.createElement('div');
m.id = 'opus-udrill';
m.style.cssText = 'position:fixed;inset:0;background:rgba(0,0,0,0.82);backdrop-filter:blur(6px);display:none;align-items:center;justify-content:center;z-index:99995;padding:20px;cursor:pointer';
var inner = d.createElement('div');
inner.id = 'opus-udrill-in';
inner.style.cssText = 'max-width:900px;width:100%;max-height:90vh;overflow:auto;background:#0b0d15;border:1px solid rgba(99,102,241,0.35);border-radius:14px;padding:28px;cursor:default;box-shadow:0 20px 60px rgba(0,0,0,0.6);color:#e2e8f0;font:14px/1.55 Inter,system-ui,sans-serif';
inner.addEventListener('click', function(e){ e.stopPropagation(); });
m.appendChild(inner);
m.addEventListener('click', function(){ m.style.display='none'; });
d.addEventListener('keydown', function(e){ if(e.key==='Escape') m.style.display='none'; });
(d.body || d.documentElement).appendChild(m);
function openCard(card) {
// Clone card content + show close btn + increase font-size
var html = '<div style="display:flex;justify-content:flex-end;margin-bottom:14px"><button id="opus-udrill-close" style="padding:6px 14px;background:#171b2a;border:1px solid rgba(99,102,241,0.25);color:#e2e8f0;border-radius:8px;cursor:pointer;font-size:12px">✕ Fermer (Esc)</button></div>';
html += '<div style="transform-origin:top left;font-size:1.05em">' + card.outerHTML + '</div>';
inner.innerHTML = html;
d.getElementById('opus-udrill-close').onclick = function(){ m.style.display='none'; };
m.style.display = 'flex';
}
function wire(root) {
var sels = '.card,[class*="card"],.kpi,[class*="kpi"],.stat,[class*="stat"],.tile,[class*="tile"],.metric,[class*="metric"],.widget,[class*="widget"]';
var cards = root.querySelectorAll(sels);
for (var i = 0; i < cards.length; i++) {
var c = cards[i];
if (c.__opusWired) continue;
if (c.closest('button, a, input, select, textarea, #opus-udrill')) continue;
var r = c.getBoundingClientRect();
if (r.width < 60 || r.height < 40) continue;
c.__opusWired = true;
c.style.cursor = 'pointer';
c.setAttribute('role','button');
c.setAttribute('tabindex','0');
c.addEventListener('click', function(ev){
// If a more-specific drill is already active (e.g. pp-card custom), let it handle
if (ev.target.closest('[data-pp-id]') && window.__opusDrillInit) return;
if (ev.target.closest('a,button,input,select')) return;
ev.preventDefault(); ev.stopPropagation();
openCard(this);
});
c.addEventListener('keydown', function(ev){ if(ev.key==='Enter'||ev.key===' '){ev.preventDefault();openCard(this);} });
}
}
// Initial + mutation observer
var initRun = function(){ wire(d.body || d.documentElement); };
if (d.readyState === 'loading') d.addEventListener('DOMContentLoaded', initRun);
else initRun();
var mo = new MutationObserver(function(muts){
var newCard = false;
for (var i=0;i<muts.length;i++) if (muts[i].addedNodes.length) { newCard = true; break; }
if (newCard) initRun();
});
mo.observe(d.body || d.documentElement, {childList:true, subtree:true});
})();
</script>
<!-- === OPUS UNIVERSAL DRILL-DOWN END === -->
<script src="/api/a11y-auto-enhancer.js" defer></script>
<!-- WTP_UDOCK_V1 (Opus 21-avr t34final) --><script src="/wtp-unified-dock.js" defer></script>
<script src="/opus-antioverlap-doctrine.js?v=1776776094" defer></script>
<!-- DOCTRINE-60-UX-JS --><script id="doctrine60-ux-js-direct">
// DOCTRINE-60-UX-JS staggered entrance
(function(){
if (!('IntersectionObserver' in window)) return;
const obs = new IntersectionObserver((entries) => {
entries.forEach((e, i) => {
if (e.isIntersecting) {
setTimeout(() => e.target.classList.add('enter-stagger'), i * 80);
obs.unobserve(e.target);
}
});
});
document.querySelectorAll('.card, .kpi, .panel').forEach(el => obs.observe(el));
})();
</script>
</body></html>