Final modularization layout (user-chosen): every page is a self-contained folder under components/<id>/ (controllers + CSS + its html fragment), and all shared/framework code folds into core/: core/kernel (feature-registry, lifecycle, services, spa) core/boot (auth, system-loader/orchestrator, setup, loaders) core/lib (data-loader, router, helpers, the task kernel, shared modules) core/ui (topbar, modal, notifications, … + topbar.html) core/css (all shared stylesheets) core/icons Top level is now just: components/, core/, themes/, index.html (+ runtime data/). Every path reference rewritten (index.html, scripts arrays, fetch()/ loadFragment()/loadScript() literals, system-loader + config-manager controller paths, kernel manifest URL, feature.json, backend FEATURES_DIR). The /api/features/list endpoint NAME is unchanged (it now scans components/). Deleted 3 dead files (app-content.html, apps-content.html, html-cache.js). Verified: 0 stale prefixes, 0 double-rewrites, all JS/JSON valid. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com> Signed-off-by: librelad <librelad@digitalangels.vip>
131 lines
6.3 KiB
CSS
131 lines
6.3 KiB
CSS
/* App Updater feature styles. Self-contained (does not borrow other features'
|
|
classes); uses the shared palette tokens + the --page-updater identity hue
|
|
set on #updater-page. Eager-linked from index.html. */
|
|
|
|
.updater-page { padding: 4px 2px 40px; }
|
|
|
|
.updater-layout {
|
|
display: grid;
|
|
grid-template-columns: 210px 1fr;
|
|
gap: 22px;
|
|
align-items: start;
|
|
}
|
|
|
|
/* Sidebar */
|
|
.updater-sidebar {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 4px;
|
|
padding: 10px;
|
|
border-radius: 14px;
|
|
background: rgba(var(--text-rgb), 0.03);
|
|
border: 1px solid rgba(var(--text-rgb), 0.07);
|
|
}
|
|
.updater-sidebar .sidebar-heading {
|
|
font-size: 0.7rem; font-weight: 700; letter-spacing: 0.07em; text-transform: uppercase;
|
|
color: rgba(var(--text-rgb), 0.45);
|
|
padding: 6px 10px 8px;
|
|
}
|
|
.updater-sidebar .category {
|
|
display: flex; align-items: center; gap: 9px;
|
|
width: 100%; text-align: left;
|
|
padding: 9px 11px; border: 0; border-radius: 9px;
|
|
background: transparent; color: rgba(var(--text-rgb), 0.7);
|
|
font-size: 0.86rem; font-weight: 500; cursor: pointer;
|
|
transition: background 0.15s, color 0.15s;
|
|
}
|
|
.updater-sidebar .category .cat-ico { opacity: 0.8; font-size: 0.95em; width: 1.1em; text-align: center; }
|
|
.updater-sidebar .category:hover { background: rgba(var(--text-rgb), 0.05); color: rgba(var(--text-rgb), 0.95); }
|
|
.updater-sidebar .category.active {
|
|
background: rgba(var(--page-rgb, var(--accent-rgb)), 0.16);
|
|
color: rgb(var(--page-rgb, var(--accent-rgb)));
|
|
}
|
|
|
|
/* Header */
|
|
.updater-header { display: flex; align-items: center; gap: 14px; margin-bottom: 18px; }
|
|
.updater-header-icon {
|
|
display: grid; place-items: center; width: 42px; height: 42px; border-radius: 11px;
|
|
color: rgb(var(--page-rgb, var(--accent-rgb)));
|
|
background: rgba(var(--page-rgb, var(--accent-rgb)), 0.14);
|
|
border: 1px solid rgba(var(--page-rgb, var(--accent-rgb)), 0.3);
|
|
}
|
|
.updater-header h1 { margin: 0; font-size: 1.5rem; }
|
|
.updater-header p { margin: 2px 0 0; color: rgba(var(--text-rgb), 0.55); font-size: 0.86rem; }
|
|
|
|
/* Stat grid (overview) */
|
|
.updater-stat-grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(190px, 1fr)); gap: 14px; }
|
|
.updater-stat {
|
|
position: relative; padding: 18px; border-radius: 14px;
|
|
background: rgba(var(--text-rgb), 0.035);
|
|
border: 1px solid rgba(var(--page-rgb, var(--accent-rgb)), 0.22);
|
|
}
|
|
.updater-stat-big { font-size: 2rem; font-weight: 700; line-height: 1; color: rgb(var(--page-rgb, var(--accent-rgb))); }
|
|
.updater-stat-label { margin-top: 6px; font-weight: 600; font-size: 0.9rem; }
|
|
.updater-stat-sub { color: rgba(var(--text-rgb), 0.5); font-size: 0.78rem; margin-top: 2px; }
|
|
.updater-stat .updater-btn { margin-top: 12px; }
|
|
|
|
/* Lists / rows */
|
|
.updater-toolbar { display: flex; gap: 10px; margin-bottom: 14px; }
|
|
.updater-list { display: flex; flex-direction: column; gap: 8px; }
|
|
.updater-row {
|
|
display: grid; grid-template-columns: 1fr auto auto; align-items: center; gap: 14px;
|
|
padding: 12px 15px; border-radius: 11px;
|
|
background: rgba(var(--text-rgb), 0.035);
|
|
border: 1px solid rgba(var(--text-rgb), 0.07);
|
|
}
|
|
.updater-row-name { font-weight: 600; }
|
|
.updater-row-ver { color: rgba(var(--text-rgb), 0.6); font-family: var(--font-mono); font-size: 0.8rem; }
|
|
.updater-arrow { color: rgb(var(--page-rgb, var(--accent-rgb))); }
|
|
|
|
/* Badges */
|
|
.updater-badge {
|
|
display: inline-block; padding: 1px 8px; border-radius: 999px;
|
|
font-size: 0.68rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.04em;
|
|
vertical-align: middle; margin-left: 6px;
|
|
}
|
|
.updater-badge-update { background: rgba(var(--page-updater-rgb), 0.18); color: rgb(var(--page-updater-rgb)); }
|
|
.updater-badge-ok { background: rgba(var(--page-verify-rgb), 0.16); color: rgb(var(--page-verify-rgb)); }
|
|
.updater-badge-unknown { background: rgba(var(--text-rgb), 0.1); color: rgba(var(--text-rgb), 0.55); }
|
|
.sev-critical { background: rgba(220, 38, 38, 0.18); color: #f87171; }
|
|
.sev-high { background: rgba(234, 88, 12, 0.18); color: #fb923c; }
|
|
.sev-medium { background: rgba(234, 179, 8, 0.16); color: #fcd34d; }
|
|
.sev-low { background: rgba(var(--text-rgb), 0.1); color: rgba(var(--text-rgb), 0.6); }
|
|
|
|
/* CVE blocks */
|
|
.updater-cve-app { padding: 12px 15px; border-radius: 11px; background: rgba(var(--text-rgb), 0.035); border: 1px solid rgba(var(--text-rgb), 0.07); }
|
|
.updater-cve-app-name { font-weight: 600; margin-bottom: 8px; }
|
|
.updater-cve { display: flex; align-items: center; gap: 10px; padding: 6px 0; border-top: 1px solid rgba(var(--text-rgb), 0.06); font-size: 0.82rem; }
|
|
.updater-cve-sev { font-weight: 700; font-size: 0.66rem; min-width: 58px; }
|
|
.updater-cve.sev-critical .updater-cve-sev { color: #f87171; }
|
|
.updater-cve.sev-high .updater-cve-sev { color: #fb923c; }
|
|
.updater-cve.sev-medium .updater-cve-sev { color: #fcd34d; }
|
|
.updater-cve-id { font-family: var(--font-mono); color: rgb(var(--page-rgb, var(--accent-rgb))); text-decoration: none; }
|
|
.updater-cve-id:hover { text-decoration: underline; }
|
|
.updater-cve-pkg { color: rgba(var(--text-rgb), 0.6); }
|
|
.updater-cve-fix { margin-left: auto; color: rgba(var(--page-verify-rgb), 0.9); font-size: 0.76rem; }
|
|
|
|
/* Buttons */
|
|
.updater-btn {
|
|
padding: 7px 14px; border-radius: 9px; cursor: pointer; font-size: 0.82rem; font-weight: 600;
|
|
background: rgba(var(--text-rgb), 0.07); color: rgba(var(--text-rgb), 0.9);
|
|
border: 1px solid rgba(var(--text-rgb), 0.12); transition: background 0.15s, border-color 0.15s;
|
|
}
|
|
.updater-btn:hover { background: rgba(var(--text-rgb), 0.12); }
|
|
.updater-btn-primary {
|
|
background: rgba(var(--page-rgb, var(--accent-rgb)), 0.2);
|
|
border-color: rgba(var(--page-rgb, var(--accent-rgb)), 0.45);
|
|
color: rgb(var(--page-rgb, var(--accent-rgb)));
|
|
}
|
|
.updater-btn-primary:hover { background: rgba(var(--page-rgb, var(--accent-rgb)), 0.3); }
|
|
|
|
/* Hints / empty states */
|
|
.updater-hint { padding: 12px 15px; border-radius: 11px; margin-bottom: 14px; font-size: 0.84rem;
|
|
background: rgba(var(--page-rgb, var(--accent-rgb)), 0.08); border: 1px solid rgba(var(--page-rgb, var(--accent-rgb)), 0.2); color: rgba(var(--text-rgb), 0.75); }
|
|
.updater-empty { padding: 40px 20px; text-align: center; color: rgba(var(--text-rgb), 0.55); display: flex; flex-direction: column; gap: 14px; align-items: center; }
|
|
|
|
@media (max-width: 820px) {
|
|
.updater-layout { grid-template-columns: 1fr; }
|
|
.updater-sidebar { flex-direction: row; flex-wrap: wrap; }
|
|
.updater-row { grid-template-columns: 1fr; gap: 6px; }
|
|
}
|