335 lines
23 KiB
HTML
335 lines
23 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Project — Dealspace</title>
|
|
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap" rel="stylesheet">
|
|
<script src="https://cdn.tailwindcss.com"></script>
|
|
<style>* { font-family: 'Inter', sans-serif; } body { background: #0a1628; }
|
|
.sidebar-link.active { background: rgba(201,168,76,0.1); color: #c9a84c; border-left: 3px solid #c9a84c; }
|
|
.sidebar-link:hover:not(.active) { background: rgba(255,255,255,0.04); }
|
|
.tab.active { color: #c9a84c; border-bottom: 2px solid #c9a84c; }
|
|
.tab { border-bottom: 2px solid transparent; }
|
|
.req-row:hover { background: rgba(255,255,255,0.03); }
|
|
.section-header { cursor: pointer; user-select: none; }
|
|
.section-header:hover { background: rgba(255,255,255,0.02); }
|
|
</style>
|
|
<style id="ds-themes">
|
|
:root,html[data-theme="midnight"]{--ds-bg:#0a1628;--ds-sf:#0d1f3c;--ds-tx:#fff;--ds-tx2:#94a3b8;--ds-tx3:#475569;--ds-ac:#c9a84c;--ds-ac2:#b8973f;--ds-act:#0a1628;--ds-bd:rgba(255,255,255,.08);--ds-hv:rgba(255,255,255,.04);--ds-inp:#0a1628;--ds-hf:'Inter',sans-serif}
|
|
html[data-theme="light"]{--ds-bg:#f0f2f5;--ds-sf:#fff;--ds-tx:#1a202c;--ds-tx2:#64748b;--ds-tx3:#94a3b8;--ds-ac:#2563eb;--ds-ac2:#1d4ed8;--ds-act:#fff;--ds-bd:rgba(0,0,0,.08);--ds-hv:rgba(0,0,0,.05);--ds-inp:#f8fafc;--ds-hf:'Inter',sans-serif}
|
|
html[data-theme="slate"]{--ds-bg:#1e293b;--ds-sf:#334155;--ds-tx:#f1f5f9;--ds-tx2:#94a3b8;--ds-tx3:#64748b;--ds-ac:#14b8a6;--ds-ac2:#0d9488;--ds-act:#0f172a;--ds-bd:rgba(148,163,184,.12);--ds-hv:rgba(255,255,255,.05);--ds-inp:#1e293b;--ds-hf:'Inter',sans-serif}
|
|
html[data-theme="compact"]{font-size:13.5px;--ds-bg:#0a1628;--ds-sf:#0d1f3c;--ds-tx:#fff;--ds-tx2:#94a3b8;--ds-tx3:#475569;--ds-ac:#c9a84c;--ds-ac2:#b8973f;--ds-act:#0a1628;--ds-bd:rgba(255,255,255,.08);--ds-hv:rgba(255,255,255,.04);--ds-inp:#0a1628;--ds-hf:'Inter',sans-serif}
|
|
html[data-theme="executive"]{font-size:17px;--ds-bg:#1a1410;--ds-sf:#2a2218;--ds-tx:#f5f0e8;--ds-tx2:#c4b8a4;--ds-tx3:#8a7e6e;--ds-ac:#d4a843;--ds-ac2:#c49835;--ds-act:#1a1410;--ds-bd:rgba(196,184,164,.12);--ds-hv:rgba(212,168,67,.06);--ds-inp:#1a1410;--ds-hf:'Georgia','Times New Roman',serif}
|
|
body{background:var(--ds-bg)!important}
|
|
h1,h2,h3{font-family:var(--ds-hf)!important}
|
|
html[data-theme="executive"] *{font-family:'Georgia','Times New Roman',serif!important}
|
|
header{background:var(--ds-sf)!important;border-color:var(--ds-bd)!important}
|
|
nav{background:var(--ds-sf)!important;border-color:var(--ds-bd)!important}
|
|
.text-white{color:var(--ds-tx)!important}
|
|
[class*="text-white/"]{color:var(--ds-tx3)!important}
|
|
.sidebar-link.active{background:color-mix(in srgb,var(--ds-ac) 10%,transparent)!important;color:var(--ds-ac)!important;border-left-color:var(--ds-ac)!important}
|
|
.sidebar-link:hover:not(.active){background:var(--ds-hv)!important}
|
|
[class*="bg-[#0d1f3c]"]{background-color:var(--ds-sf)!important}
|
|
[class*="bg-[#0a1628]"]{background-color:var(--ds-bg)!important}
|
|
[class*="text-[#c9a84c]"]{color:var(--ds-ac)!important}
|
|
[class*="text-[#94a3b8]"]{color:var(--ds-tx2)!important}
|
|
[class*="text-[#475569]"]{color:var(--ds-tx3)!important}
|
|
[class*="text-[#0a1628]"]{color:var(--ds-act)!important}
|
|
[class*="bg-[#c9a84c]"]{background-color:var(--ds-ac)!important}
|
|
[class*="hover:bg-[#b8973f]"]:hover{background-color:var(--ds-ac2)!important}
|
|
[class*="border-white"]{border-color:var(--ds-bd)!important}
|
|
[class*="bg-white/"]{background-color:var(--ds-hv)!important}
|
|
[class*="hover:bg-white/"]:hover{background-color:var(--ds-hv)!important}
|
|
input,textarea,select{background-color:var(--ds-inp)!important;color:var(--ds-tx)!important;border-color:var(--ds-bd)!important}
|
|
input:focus,textarea:focus,select:focus{border-color:var(--ds-ac)!important}
|
|
[class*="bg-black/"]{background-color:rgba(0,0,0,.6)!important}
|
|
.tab.active{color:var(--ds-ac)!important;border-bottom-color:var(--ds-ac)!important}
|
|
#ds-theme-bar{position:fixed;bottom:20px;right:20px;z-index:9999;display:flex;gap:3px;padding:4px;background:var(--ds-sf);border:1px solid var(--ds-bd);border-radius:12px;box-shadow:0 4px 20px rgba(0,0,0,.3)}
|
|
#ds-theme-bar button{padding:5px 10px;border-radius:8px;border:none;cursor:pointer;background:transparent;color:var(--ds-tx2);white-space:nowrap;transition:all .15s;font-size:12px}
|
|
#ds-theme-bar button:hover{background:var(--ds-hv)}
|
|
#ds-theme-bar button.active{background:var(--ds-ac);color:var(--ds-act);font-weight:600}
|
|
#ds-theme-bar,#ds-theme-bar *{font-family:'Inter',sans-serif!important}
|
|
</style>
|
|
<script>document.documentElement.setAttribute('data-theme',localStorage.getItem('ds_theme')||'midnight')</script>
|
|
</head>
|
|
<body>
|
|
<header class="bg-[#0d1f3c] border-b border-white/[0.08] px-6 py-3 flex items-center justify-between sticky top-0 z-50">
|
|
<div class="flex items-center gap-3">
|
|
<a href="/app/projects" class="text-xl font-bold text-white tracking-tight"><span class="text-[#c9a84c]">Deal</span>space</a>
|
|
<span class="text-white/20">/</span>
|
|
<a href="/app/projects" class="text-sm text-[#94a3b8] hover:text-white transition">Projects</a>
|
|
<span class="text-white/20">/</span>
|
|
<span id="projectName" class="text-sm text-white font-medium">Loading...</span>
|
|
</div>
|
|
<div class="flex items-center gap-4">
|
|
<span id="userName" class="text-sm text-[#94a3b8]"></span>
|
|
<button onclick="logout()" class="text-sm text-[#94a3b8] hover:text-white transition">Logout</button>
|
|
</div>
|
|
</header>
|
|
<div class="flex">
|
|
<nav class="w-56 bg-[#0d1f3c] border-r border-white/[0.08] min-h-[calc(100vh-49px)] sticky top-[49px] shrink-0">
|
|
<div class="p-3 space-y-0.5">
|
|
<a href="/app/tasks" class="sidebar-link flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium text-[#94a3b8] transition">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"/></svg>
|
|
My Tasks</a>
|
|
<a href="/app/projects" class="sidebar-link active flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium text-white transition">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 7v10a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-6l-2-2H5a2 2 0 00-2 2z"/></svg>
|
|
Projects</a>
|
|
<a href="/app/orgs" class="sidebar-link flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium text-[#94a3b8] transition">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 21V5a2 2 0 00-2-2H7a2 2 0 00-2 2v16m14 0h2m-2 0h-5m-9 0H3m2 0h5M9 7h1m-1 4h1m4-4h1m-1 4h1m-5 10v-5a1 1 0 011-1h2a1 1 0 011 1v5m-4 0h4"/></svg>
|
|
Organizations</a>
|
|
<div id="adminLinks" class="hidden"><div class="border-t border-white/[0.08] my-3"></div>
|
|
<a href="/admin" class="sidebar-link flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium text-[#94a3b8] transition">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.066 2.573c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.573 1.066c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.066-2.573c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z"/><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"/></svg>
|
|
Admin</a></div>
|
|
</div>
|
|
</nav>
|
|
<main class="flex-1 p-8 max-w-6xl">
|
|
<!-- Header -->
|
|
<div class="flex items-start justify-between mb-6">
|
|
<div>
|
|
<div class="flex items-center gap-3 mb-1">
|
|
<h1 id="projectTitle" class="text-2xl font-bold text-white">Loading...</h1>
|
|
<span id="projectStatus" class="px-2.5 py-0.5 rounded-full text-xs font-medium"></span>
|
|
</div>
|
|
<p id="projectDesc" class="text-[#94a3b8] text-sm"></p>
|
|
</div>
|
|
<div class="flex gap-2">
|
|
<button onclick="openImportModal()" class="px-4 py-2 bg-white/[0.05] hover:bg-white/[0.08] text-white rounded-lg text-sm transition flex items-center gap-2">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-8l-4-4m0 0L8 8m4-4v12"/></svg>
|
|
Import
|
|
</button>
|
|
<button id="newRequestBtn" class="px-4 py-2 bg-[#c9a84c] hover:bg-[#b8973f] text-[#0a1628] font-semibold rounded-lg text-sm transition">+ New Request</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Tabs -->
|
|
<div class="flex gap-6 border-b border-white/[0.08] mb-6">
|
|
<button class="tab active pb-3 text-sm font-medium transition" onclick="switchTab('requests', this)">Requests</button>
|
|
<button class="tab pb-3 text-sm font-medium text-[#94a3b8] transition" onclick="switchTab('orgs', this)">Organizations</button>
|
|
<button class="tab pb-3 text-sm font-medium text-[#94a3b8] transition" onclick="switchTab('team', this)">Team</button>
|
|
</div>
|
|
|
|
<!-- Requests Tab -->
|
|
<div id="tab-requests">
|
|
<div id="requestList" class="space-y-2"><div class="text-[#94a3b8] text-sm">Loading requests...</div></div>
|
|
<div id="requestEmpty" class="hidden text-center py-16">
|
|
<div class="text-4xl mb-3">📋</div>
|
|
<h3 class="text-lg font-semibold text-white mb-1">No requests yet</h3>
|
|
<p class="text-[#94a3b8] text-sm">Create the first data request for this deal.</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Orgs Tab -->
|
|
<div id="tab-orgs" class="hidden">
|
|
<div class="flex justify-between items-center mb-4">
|
|
<p class="text-[#94a3b8] text-sm">Organizations participating in this deal.</p>
|
|
<button id="addOrgBtn" class="px-3 py-1.5 bg-white/[0.05] hover:bg-white/[0.08] text-white rounded-lg text-sm transition">+ Add Org</button>
|
|
</div>
|
|
<div id="orgList" class="space-y-3"><div class="text-[#94a3b8] text-sm">Loading...</div></div>
|
|
</div>
|
|
|
|
<!-- Team Tab -->
|
|
<div id="tab-team" class="hidden">
|
|
<div class="flex justify-between items-center mb-4">
|
|
<p class="text-[#94a3b8] text-sm">People with access to this deal.</p>
|
|
<button id="inviteBtn" class="px-3 py-1.5 bg-[#c9a84c] hover:bg-[#b8973f] text-[#0a1628] font-semibold rounded-lg text-sm transition">+ Invite</button>
|
|
</div>
|
|
<div id="teamList" class="space-y-2"><div class="text-[#94a3b8] text-sm">Loading...</div></div>
|
|
</div>
|
|
</main>
|
|
</div>
|
|
|
|
|
|
<!-- Import Modal -->
|
|
<div id="importModal" style="display:none" class="fixed inset-0 bg-black/60 z-50 flex items-center justify-center p-4">
|
|
<div class="bg-[#0d1f3c] border border-white/[0.08] rounded-xl p-8 w-full max-w-md">
|
|
<h2 class="text-xl font-semibold text-white mb-6">Import Diligence Checklist</h2>
|
|
<div id="importError" class="hidden mb-4 p-3 bg-red-500/10 border border-red-500/20 rounded-lg text-red-400 text-sm"></div>
|
|
<div class="space-y-4">
|
|
<div>
|
|
<label class="block text-sm font-medium text-[#94a3b8] mb-1.5">File (CSV or XLSX)</label>
|
|
<input type="file" id="importFile" accept=".csv,.xlsx,.xls" required
|
|
class="w-full px-4 py-2.5 bg-[#0a1628] border border-white/[0.08] rounded-lg text-white text-sm focus:outline-none focus:border-[#c9a84c] focus:ring-1 focus:ring-[#c9a84c]">
|
|
</div>
|
|
<div>
|
|
<label class="block text-sm font-medium text-[#94a3b8] mb-1.5">Import Mode</label>
|
|
<select id="importMode" class="w-full px-4 py-2.5 bg-[#0a1628] border border-white/[0.08] rounded-lg text-white focus:outline-none focus:border-[#c9a84c]">
|
|
<option value="add">Add to existing requests</option>
|
|
<option value="replace">Replace all requests</option>
|
|
</select>
|
|
</div>
|
|
<div>
|
|
<label class="flex items-center gap-2 cursor-pointer">
|
|
<input type="checkbox" id="createWorkstreams" class="rounded border-white/20">
|
|
<span class="text-sm text-[#94a3b8]">Create workstreams from sections</span>
|
|
</label>
|
|
</div>
|
|
</div>
|
|
<div class="flex gap-3 mt-6">
|
|
<button onclick="closeImportModal()" class="flex-1 py-2.5 bg-white/[0.05] hover:bg-white/[0.08] text-white rounded-lg text-sm font-medium transition">Cancel</button>
|
|
<button id="importBtn" onclick="submitImport()" class="flex-1 py-2.5 bg-[#c9a84c] hover:bg-[#b8973f] text-[#0a1628] font-semibold rounded-lg text-sm transition">Import</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<script>
|
|
const token = localStorage.getItem('ds_token');
|
|
if (!token) window.location.href = '/app/login';
|
|
const user = JSON.parse(localStorage.getItem('ds_user') || '{}');
|
|
const projectID = location.pathname.split('/').pop();
|
|
document.getElementById('userName').textContent = user.name || user.email || '';
|
|
if (user.is_super_admin) document.getElementById('adminLinks').classList.remove('hidden');
|
|
|
|
function fetchAPI(path, opts = {}) {
|
|
opts.headers = { ...opts.headers, 'Authorization': 'Bearer ' + token, 'Content-Type': 'application/json' };
|
|
return fetch(path, opts).then(r => { if (r.status === 401) { localStorage.removeItem('ds_token'); window.location.href = '/app/login'; } return r; });
|
|
}
|
|
function logout() { fetchAPI('/api/auth/logout', { method: 'POST' }).finally(() => { localStorage.clear(); window.location.href = '/app/login'; }); }
|
|
function parseData(t) { try { return JSON.parse(t); } catch { return {}; } }
|
|
function escHtml(s) { const d = document.createElement('div'); d.textContent = s; return d.innerHTML; }
|
|
|
|
const statusColors = { active: 'bg-green-500/20 text-green-300', draft: 'bg-gray-500/20 text-gray-300', closed: 'bg-red-500/20 text-red-300' };
|
|
const roleColors = { seller: 'bg-blue-500/20 text-blue-300', buyer: 'bg-green-500/20 text-green-300', ib: 'bg-[#c9a84c]/20 text-[#c9a84c]', advisor: 'bg-purple-500/20 text-purple-300' };
|
|
|
|
async function loadProject() {
|
|
try {
|
|
const res = await fetchAPI('/api/projects/' + projectID);
|
|
if (!res.ok) { document.getElementById('projectTitle').textContent = 'Not found'; return; }
|
|
const raw = await res.json();
|
|
const p = raw.project || raw;
|
|
const d = parseData(p.data_text);
|
|
const name = d.name || p.summary_text || p.summary || 'Untitled';
|
|
document.title = name + ' — Dealspace';
|
|
document.getElementById('projectName').textContent = name;
|
|
document.getElementById('projectTitle').textContent = name;
|
|
document.getElementById('projectDesc').textContent = d.description || '';
|
|
const status = d.status || 'active';
|
|
const sc = statusColors[status] || 'bg-gray-500/20 text-gray-300';
|
|
document.getElementById('projectStatus').className = 'px-2.5 py-0.5 rounded-full text-xs font-medium capitalize ' + sc;
|
|
document.getElementById('projectStatus').textContent = status;
|
|
} catch(e) {}
|
|
}
|
|
|
|
async function loadRequests() {
|
|
try {
|
|
const res = await fetchAPI('/api/projects/' + projectID + '/requests');
|
|
const items = await res.json();
|
|
const list = document.getElementById('requestList');
|
|
if (!items || items.length === 0) { list.classList.add('hidden'); document.getElementById('requestEmpty').classList.remove('hidden'); return; }
|
|
list.innerHTML = items.map(r => {
|
|
const d = r.data || parseData(r.data_text) || {};
|
|
return `<a href="/app/requests/${r.entry_id}" class="req-row flex items-center gap-4 px-5 py-4 rounded-xl border border-white/[0.08] transition cursor-pointer">
|
|
<div class="flex-1 min-w-0">
|
|
<div class="flex items-center gap-2 mb-0.5">
|
|
${d.ref ? `<span class="text-xs font-mono text-[#94a3b8]">${escHtml(d.ref)}</span>` : ''}
|
|
<span class="text-white font-medium truncate">${escHtml(d.title || r.summary || 'Untitled')}</span>
|
|
</div>
|
|
${d.description ? `<p class="text-[#94a3b8] text-xs truncate">${escHtml(d.description)}</p>` : ''}
|
|
</div>
|
|
<span class="shrink-0 px-2.5 py-0.5 rounded-full text-xs font-medium capitalize ${d.status === 'open' ? 'bg-yellow-500/20 text-yellow-300' : d.status === 'answered' ? 'bg-green-500/20 text-green-300' : 'bg-gray-500/20 text-gray-300'}">${d.status || 'open'}</span>
|
|
</a>`;
|
|
}).join('');
|
|
} catch(e) { document.getElementById('requestList').innerHTML = '<div class="text-red-400 text-sm">Failed to load.</div>'; }
|
|
}
|
|
|
|
async function loadOrgs() {
|
|
try {
|
|
const res = await fetchAPI('/api/projects/' + projectID + '/orgs');
|
|
const orgs = await res.json();
|
|
const list = document.getElementById('orgList');
|
|
if (!orgs || orgs.length === 0) { list.innerHTML = '<div class="text-[#94a3b8] text-sm">No organizations added yet.</div>'; return; }
|
|
list.innerHTML = orgs.map(o => {
|
|
const d = parseData(o.data_text);
|
|
const rc = roleColors[d.role] || 'bg-gray-500/20 text-gray-300';
|
|
return `<div class="flex items-center gap-4 px-5 py-4 rounded-xl bg-[#0d1f3c] border border-white/[0.08]">
|
|
<div class="flex-1">
|
|
<div class="flex items-center gap-2 mb-1">
|
|
<span class="text-white font-medium">${escHtml(d.org_name || d.name || 'Unknown')}</span>
|
|
<span class="px-2 py-0.5 rounded-full text-xs font-medium capitalize ${rc}">${d.role || '?'}</span>
|
|
${d.domain_lock ? '<span class="px-2 py-0.5 rounded-full text-xs bg-white/[0.05] text-[#94a3b8]">🔒 domain locked</span>' : ''}
|
|
</div>
|
|
${d.domains ? `<div class="flex gap-1.5 flex-wrap">${(Array.isArray(d.domains)?d.domains:[d.domains]).map(dm=>`<span class="text-xs text-[#94a3b8] font-mono bg-white/[0.05] px-2 py-0.5 rounded">@${dm}</span>`).join('')}</div>` : ''}
|
|
</div>
|
|
</div>`;
|
|
}).join('');
|
|
} catch(e) {}
|
|
}
|
|
|
|
async function loadTeam() {
|
|
try {
|
|
const res = await fetchAPI('/api/projects/' + projectID + '/members');
|
|
const members = await res.json();
|
|
const list = document.getElementById('teamList');
|
|
if (!members || members.length === 0) { list.innerHTML = '<div class="text-[#94a3b8] text-sm">No team members yet.</div>'; return; }
|
|
list.innerHTML = members.map(m => `
|
|
<div class="flex items-center gap-4 px-5 py-3 rounded-xl bg-[#0d1f3c] border border-white/[0.08]">
|
|
<div class="w-8 h-8 rounded-full bg-[#c9a84c]/20 flex items-center justify-center text-[#c9a84c] font-semibold text-sm">${(m.name||m.email||'?')[0].toUpperCase()}</div>
|
|
<div class="flex-1">
|
|
<div class="text-white text-sm font-medium">${escHtml(m.name || m.email)}</div>
|
|
${m.name ? `<div class="text-[#94a3b8] text-xs">${escHtml(m.email)}</div>` : ''}
|
|
</div>
|
|
<span class="text-xs text-[#94a3b8] capitalize">${m.role || 'member'}</span>
|
|
</div>`).join('');
|
|
} catch(e) {}
|
|
}
|
|
|
|
function switchTab(name, el) {
|
|
document.querySelectorAll('.tab').forEach(t => { t.classList.remove('active','text-white'); t.classList.add('text-[#94a3b8]'); });
|
|
el.classList.add('active','text-white'); el.classList.remove('text-[#94a3b8]');
|
|
document.getElementById('tab-requests').classList.toggle('hidden', name !== 'requests');
|
|
document.getElementById('tab-orgs').classList.toggle('hidden', name !== 'orgs');
|
|
document.getElementById('tab-team').classList.toggle('hidden', name !== 'team');
|
|
if (name === 'orgs') loadOrgs();
|
|
if (name === 'team') loadTeam();
|
|
}
|
|
|
|
document.getElementById('newRequestBtn').onclick = () => {
|
|
const title = prompt('Request title:');
|
|
if (!title) return;
|
|
fetchAPI('/api/projects/' + projectID + '/entries', { method: 'POST', body: JSON.stringify({ type: 'request', data: { title, status: 'open' } }) })
|
|
.then(r => r.json()).then(d => { if (d.entry_id) window.location.href = '/app/requests/' + d.entry_id; else loadRequests(); });
|
|
};
|
|
|
|
|
|
function openImportModal() { document.getElementById('importModal').style.display='flex'; document.getElementById('importFile').value=''; document.getElementById('importError').classList.add('hidden'); }
|
|
function closeImportModal() { document.getElementById('importModal').style.display='none'; }
|
|
|
|
async function submitImport() {
|
|
const file = document.getElementById('importFile').files[0];
|
|
const errEl = document.getElementById('importError');
|
|
const btn = document.getElementById('importBtn');
|
|
if (!file) { errEl.textContent = 'Please select a file'; errEl.classList.remove('hidden'); return; }
|
|
btn.disabled = true; btn.textContent = 'Importing...'; errEl.classList.add('hidden');
|
|
const fd = new FormData();
|
|
fd.append('file', file);
|
|
fd.append('mode', document.getElementById('importMode').value);
|
|
fd.append('create_workstreams', document.getElementById('createWorkstreams').checked ? 'true' : 'false');
|
|
try {
|
|
const res = await fetch('/api/projects/' + projectID + '/requests/import', { method: 'POST', headers: { 'Authorization': 'Bearer ' + token }, body: fd });
|
|
const data = await res.json();
|
|
if (!res.ok) throw new Error(data.error || 'Import failed');
|
|
closeImportModal();
|
|
loadRequests();
|
|
alert('Imported ' + data.imported + ' requests from ' + (data.sections?.length || 0) + ' sections.');
|
|
} catch(e) { errEl.textContent = e.message; errEl.classList.remove('hidden'); }
|
|
finally { btn.disabled = false; btn.textContent = 'Import'; }
|
|
}
|
|
|
|
loadProject();
|
|
loadRequests();
|
|
</script>
|
|
<div id="ds-theme-bar">
|
|
<button data-t="midnight" onclick="setTheme('midnight')">Midnight</button>
|
|
<button data-t="light" onclick="setTheme('light')">Light</button>
|
|
<button data-t="slate" onclick="setTheme('slate')">Slate</button>
|
|
<button data-t="compact" onclick="setTheme('compact')">Compact</button>
|
|
<button data-t="executive" onclick="setTheme('executive')">Executive</button>
|
|
</div>
|
|
<script>
|
|
function setTheme(t){document.documentElement.setAttribute('data-theme',t);localStorage.setItem('ds_theme',t);document.querySelectorAll('#ds-theme-bar button').forEach(b=>b.classList.toggle('active',b.getAttribute('data-t')===t))}
|
|
setTheme(localStorage.getItem('ds_theme')||'midnight');
|
|
</script>
|
|
</body>
|
|
</html>
|