likwid/frontend/src/pages/admin/roles.astro
Marco Allegretti 554bbf5334 frontend: modify 1 file
Verified changes:
- modify frontend/src/pages/admin/roles.astro

Diffstat:
- 1 file changed, 1 insertion(+), 1 deletion(-)
2026-01-30 13:54:27 +01:00

263 lines
6.3 KiB
Text

---
export const prerender = false;
import Layout from '../../layouts/Layout.astro';
import AdminNav from '../../components/AdminNav.astro';
import { API_BASE } from '../../lib/api';
---
<Layout title="Role Management - Admin">
<div class="admin-container">
<AdminNav currentPage="/admin/roles" />
<main class="admin-content">
<header class="ui-page-header">
<div class="ui-page-title">
<h1>Role Management</h1>
<p class="ui-subtitle">Manage platform roles and permissions</p>
</div>
</header>
<section class="roles-section">
<h2>Platform Roles</h2>
<div class="roles-list" id="platform-roles">
<p class="loading">Loading roles...</p>
</div>
</section>
<section class="permissions-section">
<h2>Available Permissions</h2>
<div class="permissions-grid" id="permissions-list">
<p class="loading">Loading permissions...</p>
</div>
</section>
</main>
</div>
</Layout>
<style>
.admin-container {
display: flex;
min-height: calc(100vh - 60px);
}
.admin-content {
flex: 1;
padding: 2rem;
max-width: 1000px;
}
.roles-section, .permissions-section {
margin-bottom: 3rem;
}
.roles-section h2, .permissions-section h2 {
margin: 0 0 1rem 0;
font-size: 1.25rem;
}
.roles-list {
display: grid;
gap: 1rem;
}
.role-card {
}
.role-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 0.75rem;
}
.role-name {
display: flex;
align-items: center;
gap: 0.75rem;
}
.role-name h3 {
margin: 0;
font-size: 1.1rem;
}
.role-color {
width: 12px;
height: 12px;
border-radius: 50%;
}
.role-badge {
font-size: 0.7rem;
padding: 0.15rem 0.5rem;
border-radius: 1rem;
background: var(--color-bg);
border: 1px solid var(--color-border);
color: var(--color-text-muted);
}
.role-badge.system {
background: var(--color-primary);
border-color: transparent;
color: var(--color-on-primary);
}
.role-description {
color: var(--color-text-muted);
font-size: 0.9rem;
margin: 0 0 1rem 0;
}
.role-permissions {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
}
.perm-tag {
font-size: 0.75rem;
padding: 0.25rem 0.5rem;
background: var(--color-bg);
border: 1px solid var(--color-border);
border-radius: 0.25rem;
color: var(--color-text-muted);
}
.permissions-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
gap: 1rem;
}
.permission-category {
}
.permission-category h4 {
margin: 0 0 0.75rem 0;
font-size: 0.9rem;
text-transform: capitalize;
color: var(--color-primary);
}
.permission-list {
display: flex;
flex-direction: column;
gap: 0.5rem;
}
.permission-item {
font-size: 0.85rem;
}
.permission-item code {
font-size: 0.75rem;
color: var(--color-text-muted);
}
.loading {
text-align: center;
padding: 2rem;
color: var(--color-text-muted);
}
</style>
<script define:vars={{ API_BASE }}>
const token = localStorage.getItem('token');
if (!token) {
window.location.href = '/login?redirect=/admin/roles';
}
async function loadData() {
await Promise.all([loadRoles(), loadPermissions()]);
}
async function loadRoles() {
const container = document.getElementById('platform-roles');
try {
const res = await fetch(`${API_BASE}/api/roles`, {
headers: { 'Authorization': `Bearer ${token}` },
});
if (res.status === 401) {
window.location.href = '/login?redirect=/admin/roles';
return;
}
if (res.status === 403) {
container.innerHTML = '<p class="loading">Admin access required</p>';
return;
}
const roles = await res.json();
container.innerHTML = roles.map(r => `
<div class="role-card ui-card ui-card-soft ui-card-pad-md">
<div class="role-header">
<div class="role-name">
${r.color ? `<span class="role-color" style="background: ${r.color}"></span>` : ''}
<h3>${r.display_name}</h3>
${r.is_system ? '<span class="role-badge system">System</span>' : ''}
${r.is_default ? '<span class="role-badge">Default</span>' : ''}
</div>
<span style="color: var(--color-text-muted); font-size: 0.85rem">Priority: ${r.priority}</span>
</div>
<p class="role-description">${r.description || 'No description'}</p>
<div class="role-permissions">
${(r.permissions || []).slice(0, 10).map(p => `<span class="perm-tag">${p}</span>`).join('')}
${(r.permissions || []).length > 10 ? `<span class="perm-tag">+${r.permissions.length - 10} more</span>` : ''}
</div>
</div>
`).join('');
} catch (e) {
container.innerHTML = '<p class="loading">Failed to load roles</p>';
}
}
async function loadPermissions() {
const container = document.getElementById('permissions-list');
try {
const res = await fetch(`${API_BASE}/api/permissions`, {
headers: { 'Authorization': `Bearer ${token}` },
});
if (res.status === 401) {
window.location.href = '/login?redirect=/admin/roles';
return;
}
if (res.status === 403) {
container.innerHTML = '<p class="loading">Admin access required</p>';
return;
}
const permissions = await res.json();
const byCategory = {};
permissions.forEach(p => {
if (!byCategory[p.category]) byCategory[p.category] = [];
byCategory[p.category].push(p);
});
container.innerHTML = Object.entries(byCategory).map(([cat, perms]) => `
<div class="permission-category ui-card ui-card-soft ui-card-pad-md">
<h4>${cat}</h4>
<div class="permission-list">
${perms.map(p => `
<div class="permission-item">
<code>${p.name}</code>
${p.description ? `<br><small>${p.description}</small>` : ''}
</div>
`).join('')}
</div>
</div>
`).join('');
} catch (e) {
container.innerHTML = '<p class="loading">Failed to load permissions</p>';
}
}
loadData();
</script>