feat (ui): netwerken en files verfraaid

This commit is contained in:
kodi
2026-02-25 10:07:35 +01:00
parent ec13059437
commit ebb6d755a0
3 changed files with 318 additions and 47 deletions
+172 -1
View File
@@ -436,6 +436,7 @@ pre{
.sidebar .navLabel { display: none; }
.sidebar .tab { justify-content: center; }
}
/* Files tree (Portainer-ish) */
.file-folder-row{
display:flex;
align-items:center;
@@ -443,16 +444,64 @@ pre{
gap:10px;
cursor:pointer;
user-select:none;
padding: 10px 12px;
border: 1px solid rgba(36,52,95,.75);
border-radius: 12px;
background: rgba(17,26,46,.35);
transition: background .12s ease, border-color .12s ease, transform .06s ease;
}
.file-folder-row:hover{
background: rgba(96,165,250,.08);
border-color: rgba(96,165,250,.35);
}
.file-folder-row:active{
transform: translateY(1px);
}
.file-folder-left{
display:flex;
align-items:center;
gap:10px;
min-width: 0;
}
.folder-toggle{
display:inline-flex;
align-items:center;
justify-content:center;
width: 18px;
height: 18px;
border-radius: 6px;
border: 1px solid rgba(36,52,95,.8);
background: rgba(8,12,25,.35);
font-size: 12px;
opacity: .9;
flex: 0 0 auto;
}
.file-folder-left span:last-child{
overflow:hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.file-folder-files{
margin-left: 18px;
margin-left: 0;
margin-top: 6px;
padding-left: 12px;
border-left: 1px dashed rgba(36,52,95,.55);
}
/* File rows inside folders (werkt ook met inline styles die je nu al hebt) */
.file-folder-files > div{
border-radius: 10px;
}
.file-folder-files > div:hover{
background: rgba(96,165,250,.06);
}
.data-table {
width: 100%;
@@ -637,3 +686,125 @@ pre{
.mapDetailList{ margin: 8px 0 0 0; padding-left: 18px; }
.mapDetailList li{ margin: 4px 0; }
.mapDetailLink{ cursor: pointer; text-decoration: underline; }
/* ===== Netwerken kaart split layout ===== */
.mapSplit{
display:grid;
grid-template-columns: 1fr 360px;
gap: 12px;
align-items: start;
}
.mapMain{ min-width: 0; }
.mapSide{ min-width: 0; }
@media (max-width: 1100px){
.mapSplit{ grid-template-columns: 1fr; }
}
/* ===== Legenda horizontaal ===== */
.mapLegend{
display:flex;
align-items:center;
gap:18px;
flex-wrap:wrap;
}
.mapLegend .legendTitle{
font-weight:600;
margin-right:10px;
}
.mapLegend .legendRow{
display:flex;
align-items:center;
gap:6px;
}
/* ===== Legenda horizontaal + compact (override) ===== */
.mapLegend{
display:flex !important;
flex-direction: row;
align-items: center;
gap: 14px;
flex-wrap: wrap;
max-width: none; /* was 360px */
padding: 10px 12px;
}
.mapLegend .legendTitle{
margin: 0;
font-weight: 800;
}
.mapLegend .legendRow{
margin: 0; /* haalt verticale spacing weg */
display:flex;
align-items:center;
gap: 8px;
white-space: nowrap; /* voorkomt rare afbreking */
}
/* ===== Legenda badge stijl ===== */
.mapLegend{
display:flex;
align-items:center;
gap:12px;
flex-wrap:wrap;
}
.mapLegend .legendTitle{
font-weight:800;
margin-right:6px;
}
.mapLegend .legendRow{
display:inline-flex;
align-items:center;
gap:8px;
padding:6px 10px;
border:1px solid rgba(255,255,255,0.12);
border-radius:20px;
background:rgba(255,255,255,0.03);
font-size:13px;
white-space:nowrap;
transition:all .15s ease;
}
.mapLegend .legendRow:hover{
border-color:rgba(255,255,255,0.25);
background:rgba(255,255,255,0.06);
}
/* Files tree: actions only on hover */
.file-folder-row .file-folder-actions{
opacity: 0;
pointer-events: none;
transition: opacity .12s ease;
}
.file-folder-row:hover .file-folder-actions{
opacity: 1;
pointer-events: auto;
}
.file-folder-meta{
display:inline-flex;
align-items:center;
gap:8px;
flex: 0 0 auto;
}
.file-badge{
display:inline-flex;
align-items:center;
gap:6px;
padding: 3px 8px;
border-radius: 999px;
border: 1px solid rgba(36,52,95,.8);
background: rgba(8,12,25,.35);
font-size: 12px;
opacity: .92;
}
+2 -2
View File
@@ -653,11 +653,11 @@
.attr('class', d => `graphNode ${d.type}`);
node.append('circle')
.attr('r', d => d.type === 'network' ? 10 : 8);
.attr('r', d => d.type === 'network' ? 14 : 9);
node.append('text')
.attr('class', 'graphLabel')
.attr('x', 12)
.attr('x', d => d.type === 'network' ? 18 : 12)
.attr('y', 4)
.text(d => d.label || d.key);
+143 -43
View File
@@ -174,16 +174,23 @@
</div>
</div>
<div id="networksMapHost" class="mapHost">
</div>
<div id="networksDetailPanel" class="mapDetail" style="display:none;">
<div class="mapDetailHeader">
<button class="btn small ghost" type="button" id="networksMapBackBtn">← Terug</button>
<div class="mapDetailTitle" id="networksDetailTitle">Netwerk</div>
<div class="mapSplit">
<div class="mapMain">
<div id="networksMapHost" class="mapHost">
</div>
</div>
<div class="mapDetailBody" id="networksDetailBody"></div>
<div class="mapSide">
<div id="networksDetailPanel" class="mapDetail" style="display:none;">
<div class="mapDetailHeader">
<button class="btn small ghost" type="button" id="networksMapBackBtn">← Terug</button>
<div class="mapDetailTitle" id="networksDetailTitle">Netwerk</div>
</div>
<div class="mapDetailBody" id="networksDetailBody"></div>
</div>
</div>
</div>
<div id="networksMapLegend" class="mapLegend" style="display:none;">
<div class="legendTitle">Legenda</div>
<div class="legendRow"><span class="legendSwatch net"></span> Netwerk</div>
@@ -1022,49 +1029,142 @@
return;
}
// Render simpel: folders met files eronder
const parts = [];
for (const folder of scoped) {
const apiFolderPath = (folder.path || '').replace(/^\/+/, '');
const uiFolderPath = filesToUiPath(apiFolderPath); // zonder systemd/
const folderLabel = uiFolderPath || 'root';
const folderKey = apiFolderPath; // unieke key (met systemd/..)
const collapsed = _isFolderCollapsed(folderKey);
// Bouw een geneste folder-tree uit de "platte" API response.
const folderByPath = new Map();
for (const f of scoped) {
const apiPath = (f.path || '').replace(/^\/+/, '');
folderByPath.set(apiPath, f);
}
parts.push(`
<div class="mono file-folder-row" data-folder="${esc(folderKey)}" style="margin:8px 0 6px 0; font-weight:600;">
<span class="file-folder-left">
<span class="folder-toggle">${collapsed ? '▶' : '▼'}</span>
<span>📂 ${esc(folderLabel)}</span>
</span>
<span class="flex" onclick="event.stopPropagation();">
<button class="btn small ok" title="Nieuw bestand in ${esc(folderLabel)}" onclick="filesNewFileInFolder(decodeURIComponent('${encodeURIComponent(uiFolderPath)}'))">+</button>
<button class="btn small bad" title="Verwijder map (alleen als leeg)" onclick="filesDeleteFolder(decodeURIComponent('${encodeURIComponent(uiFolderPath)}'))">🗑️</button>
</span>
</div>
`);
function getOrCreateChild(parent, name) {
if (!parent.children.has(name)) {
const apiPath = parent.apiPath ? `${parent.apiPath}/${name}` : name;
parent.children.set(name, {
name,
apiPath,
uiPath: filesToUiPath(apiPath),
children: new Map(),
});
}
return parent.children.get(name);
}
const files = folder.files || [];
if (!files.length) {
parts.push(`<div class="muted" style="margin-left:18px;">(leeg)</div>`);
continue;
}
const root = { name: FILES_ROOT, apiPath: FILES_ROOT, uiPath: '', children: new Map() };
parts.push(`<div class="file-folder-files" data-folder-files="${esc(folderKey)}" style="${collapsed ? 'display:none;' : ''}">`);
// 1) Nodes aanmaken op basis van bekende folder paths
for (const apiPath of folderByPath.keys()) {
if (apiPath === FILES_ROOT) continue;
if (!apiPath.startsWith(FILES_ROOT + '/')) continue;
const rel = apiPath.slice((FILES_ROOT + '/').length);
const segs = rel.split('/').filter(Boolean);
let cur = root;
for (const s of segs) cur = getOrCreateChild(cur, s);
}
for (const f of files) {
const fullUi = uiFolderPath ? `${uiFolderPath}/${f}` : f;
parts.push(`
<div style="display:flex;align-items:center;justify-content:space-between;gap:10px;padding:4px 0;border-bottom:1px dashed rgba(36,52,95,.35)">
// 2) Nodes aanvullen op basis van dirs-lijsten (zodat lege tussenfolders ook verschijnen)
for (const [apiPath, folder] of folderByPath.entries()) {
if (apiPath !== FILES_ROOT && !apiPath.startsWith(FILES_ROOT + '/')) continue;
let base = root;
if (apiPath !== FILES_ROOT) {
const rel = apiPath.slice((FILES_ROOT + '/').length);
const segs = rel.split('/').filter(Boolean);
for (const s of segs) base = getOrCreateChild(base, s);
}
for (const d of (folder.dirs || [])) getOrCreateChild(base, d);
}
function renderNode(node, level) {
const folderKey = node.apiPath;
const collapsed = _isFolderCollapsed(folderKey);
const label = node.uiPath || 'root';
const indent = Math.max(0, level) * 14;
const folder = folderByPath.get(folderKey);
const files = (folder && folder.files) ? folder.files : [];
// subfolders (NU AL beschikbaar voor de badges)
const childNames = Array.from(node.children.keys()).sort((a,b) => a.localeCompare(b));
// files (NU AL beschikbaar voor de badges)
const sortedFiles = (files || []).slice().sort((a,b) => a.localeCompare(b));
const out = [];
out.push(`
<div class="mono file-folder-row" data-folder="${esc(folderKey)}" style="margin:${level === 0 ? '8px' : '6px'} 0 6px 0; padding-left:${indent}px; font-weight:600;">
<span class="file-folder-left">
<span class="folder-toggle">${collapsed ? '▶' : '▼'}</span>
<span>📂 ${esc(label)}</span>
</span>
<span class="file-folder-meta" onclick="event.stopPropagation();">
<span class="file-badge" title="Subfolders in deze map">📁 ${childNames.length}</span>
<span class="file-badge" title="Bestanden in deze map">📄 ${sortedFiles.length}</span>
<span class="flex file-folder-actions">
<button class="btn small ok" title="Nieuw bestand in ${esc(label)}" onclick="filesNewFileInFolder(decodeURIComponent('${encodeURIComponent(node.uiPath)}'))">+</button>
<button class="btn small bad" title="Verwijder map (alleen als leeg)" onclick="filesDeleteFolder(decodeURIComponent('${encodeURIComponent(node.uiPath)}'))">🗑️</button>
</span>
</span>
</div>
`);
out.push(`<div class="file-folder-files" data-folder-files="${esc(folderKey)}" style="${collapsed ? 'display:none;' : ''}">`);
for (const name of childNames) {
out.push(renderNode(node.children.get(name), level + 1));
}
for (const f of sortedFiles) {
const fullUi = node.uiPath ? `${node.uiPath}/${f}` : f;
out.push(`
<div style="display:flex;align-items:center;justify-content:space-between;gap:10px;padding:4px 0 4px ${indent + 18}px;border-bottom:1px dashed rgba(36,52,95,.35)">
<span class="mono" style="cursor:pointer" onclick="filesOpen(decodeURIComponent('${encodeURIComponent(fullUi)}'))">📄 ${esc(f)}</span>
</div>
`);
}
if (!childNames.length && !sortedFiles.length) {
out.push(`<div class="muted" style="padding-left:${indent + 18}px;">(leeg)</div>`);
}
out.push(`</div>`);
return out.join('');
}
const parts = [];
const topNames = Array.from(root.children.keys()).sort((a,b) => a.localeCompare(b));
for (const n of topNames) parts.push(renderNode(root.children.get(n), 0));
// Files direct onder "systemd/" (root) tonen bovenaan
const rootFolder = folderByPath.get(FILES_ROOT);
if (rootFolder && (rootFolder.files || []).length) {
const folderKey = FILES_ROOT;
const collapsed = _isFolderCollapsed(folderKey);
parts.unshift(`
<div class="mono file-folder-row" data-folder="${esc(folderKey)}" style="margin:8px 0 6px 0; font-weight:600;">
<span class="file-folder-left">
<span class="folder-toggle">${collapsed ? '▶' : '▼'}</span>
<span>📂 root</span>
</span>
<span class="flex" onclick="event.stopPropagation();">
<button class="btn small ok" title="Nieuw bestand in root" onclick="filesNewFileInFolder('')">+</button>
</span>
</div>
<div class="file-folder-files" data-folder-files="${esc(folderKey)}" style="${collapsed ? 'display:none;' : ''}">
${(rootFolder.files || []).slice().sort((a,b)=>a.localeCompare(b)).map(f => {
const fullUi = f;
return `
<div style="display:flex;align-items:center;justify-content:space-between;gap:10px;padding:4px 0 4px 18px;border-bottom:1px dashed rgba(36,52,95,.35)">
<span class="mono" style="cursor:pointer" onclick="filesOpen(decodeURIComponent('${encodeURIComponent(fullUi)}'))">📄 ${esc(f)}</span>
</div>
`);
}
`;
}).join('')}
</div>
`);
}
parts.push(`</div>`);
}
treeEl.innerHTML = parts.join('');
treeEl.innerHTML = parts.join('');
treeEl.onclick = (ev) => {
const row = ev.target.closest('.file-folder-row');
if (!row) return;