nxtgauge-admin-solid/src/routes/admin/roles/create.tsx
Ashwin Kumar 3b2c09cd4b style: unify all primary buttons to use shared .btn-primary class
Replaced 37 files worth of inconsistent inline Tailwind button classes
(mixed font-medium/semibold, px-4/px-6, with/without shadow-sm, inline-flex
variants) with the shared .btn-primary CSS class. Added :disabled state to
.btn-primary in app.css so disabled buttons visually dim consistently.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-24 08:10:29 +01:00

224 lines
10 KiB
TypeScript

import { A, useNavigate } from '@solidjs/router';
import { createMemo, createResource, createSignal, Show } from 'solid-js';
import AdminShell from '~/components/AdminShell';
import { STATIC_PERMISSIONS, type Permission } from '~/lib/admin-modules';
const API = '/api/gateway';
async function loadPermissions(): Promise<Permission[]> {
try {
const res = await fetch(`${API}/api/admin/permissions`);
if (!res.ok) return STATIC_PERMISSIONS;
const data = await res.json();
const rows: Permission[] = Array.isArray(data) ? data : (data.permissions || []);
return rows.length > 0 ? rows : STATIC_PERMISSIONS;
} catch {
return STATIC_PERMISSIONS;
}
}
export default function CreateInternalRolePage() {
const navigate = useNavigate();
const [permissions] = createResource(loadPermissions);
const [roleName, setRoleName] = createSignal('');
const [description, setDescription] = createSignal('');
const [assignedModules, setAssignedModules] = createSignal<string[]>([]);
const [permissionIds, setPermissionIds] = createSignal<string[]>([]);
const [saving, setSaving] = createSignal(false);
const [error, setError] = createSignal('');
const permsByModule = createMemo(() => {
const map: Record<string, Permission[]> = {};
(permissions() || []).forEach((p) => {
if (!map[p.module]) map[p.module] = [];
map[p.module].push(p);
});
return map;
});
const allModules = createMemo(() => Object.keys(permsByModule()).sort());
const toggleModule = (mod: string) => {
const current = assignedModules();
if (current.includes(mod)) {
setAssignedModules(current.filter((m) => m !== mod));
// remove permissions for this module
const idsToRemove = (permsByModule()[mod] || []).map((p) => p.id);
setPermissionIds(permissionIds().filter((id) => !idsToRemove.includes(id)));
} else {
setAssignedModules([...current, mod]);
}
};
const togglePermission = (id: string) => {
const current = permissionIds();
if (current.includes(id)) {
setPermissionIds(current.filter((p) => p !== id));
} else {
setPermissionIds([...current, id]);
}
};
const handleSave = async () => {
if (!roleName().trim()) { setError('Role name is required'); return; }
if (assignedModules().length === 0) { setError('Select at least one module'); return; }
if (permissionIds().length === 0) { setError('Assign at least one permission'); return; }
try {
setSaving(true);
setError('');
const res = await fetch(`${API}/api/admin/roles`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
name: roleName().trim(),
description: description().trim(),
audience: 'INTERNAL',
modules: assignedModules(),
permissionIds: permissionIds(),
}),
});
if (!res.ok) {
const body = await res.json().catch(() => ({}));
throw new Error(body.message || 'Failed to create role');
}
navigate('/admin/roles');
} catch (err: any) {
setError(err.message || 'Failed to create role');
} finally {
setSaving(false);
}
};
return (
<AdminShell>
<div class="flex flex-col -mx-6 -mt-6 min-h-full">
<div class="bg-white border-b border-gray-200 px-6 py-4">
<h1 class="text-xl font-semibold text-gray-900">Create Internal Role</h1>
<p class="text-sm text-gray-500 mt-0.5">Create a new internal role and choose what it can access.</p>
</div>
<div class="flex-1 p-6">
<nav class="hidden" aria-label="Role Management Navigation">
<A class="hidden" href="/admin/roles">Internal Roles</A>
<A class="hidden" href="/admin/runtime-roles">External Runtime Roles</A>
<A class="hidden" href="/admin/onboarding-schemas">Onboarding Schemas</A>
</nav>
<Show when={error()}>
<div class="mb-4 rounded-lg border border-red-200 bg-red-50 px-4 py-3 text-sm text-red-700">{error()}</div>
</Show>
{/* Role Details */}
<div class="mb-6 rounded-xl border border-gray-200 bg-white p-6 shadow-sm">
<h3>Role Basics</h3>
<p>Start by giving this role a clear name.</p>
<div class="mt-4 grid grid-cols-1 gap-4 sm:grid-cols-2">
<div class="field">
<label class="mb-1.5 block text-sm font-medium text-gray-700">Role Name <span class="text-red-500">*</span></label>
<input
class="w-full rounded-lg border border-gray-200 px-3 py-2 text-sm outline-none focus:border-[#0a1d37] focus:ring-1 focus:ring-[#0a1d37]"
value={roleName()}
onInput={(e) => setRoleName(e.currentTarget.value)}
placeholder="e.g. Customer Support Rep"
/>
</div>
<div class="field">
<label class="mb-1.5 block text-sm font-medium text-gray-700">Description</label>
<input
class="w-full rounded-lg border border-gray-200 px-3 py-2 text-sm outline-none focus:border-[#0a1d37] focus:ring-1 focus:ring-[#0a1d37]"
value={description()}
onInput={(e) => setDescription(e.currentTarget.value)}
placeholder="Short description of this role"
/>
</div>
</div>
</div>
{/* Module Access */}
<div class="mb-6 rounded-xl border border-gray-200 bg-white p-6 shadow-sm">
<h3>Area Access</h3>
<p>Select which areas this role can access. You can set permissions for selected areas below.</p>
<Show when={permissions.loading}>
<p class="notice">Loading available areas...</p>
</Show>
<Show when={!permissions.loading && allModules().length > 0}>
<div class="mt-3 flex flex-wrap gap-2">
{allModules().map((mod) => (
<button
type="button"
class={`flex cursor-pointer items-center gap-1.5 rounded-full border border-gray-200 bg-white px-3 py-1.5 text-xs font-medium text-gray-600 hover:border-gray-300 ${assignedModules().includes(mod) ? 'selected' : ''}`}
onClick={() => toggleModule(mod)}
>
<span class={`inline-block w-3.5 h-3.5 rounded-[3px] border-2 flex-shrink-0 ${assignedModules().includes(mod) ? 'border-[#0a1d37] bg-[#0a1d37]' : 'border-slate-300 bg-white'}`} />
{mod}
</button>
))}
</div>
</Show>
<Show when={!permissions.loading && allModules().length === 0}>
<p class="notice">No areas available.</p>
</Show>
</div>
{/* Permission Table */}
<Show when={assignedModules().length > 0}>
<div class="mb-6 rounded-xl border border-gray-200 bg-white p-6 shadow-sm">
<h3>Permissions</h3>
<p>Choose what this role can do in each selected area.</p>
<div class="table-card overflow-x-auto">
<table data-table class="w-full text-sm">
<thead>
<tr>
<th style="width:45%">Area</th>
<th style="width:11%">No Access</th>
<th style="width:11%">Read</th>
<th style="width:11%">Create</th>
<th style="width:11%">Update</th>
<th style="width:11%">Delete</th>
</tr>
</thead>
<tbody>
{assignedModules().sort().map((mod) => {
const perms = permsByModule()[mod] || [];
const actionMap: Record<string, string> = {};
perms.forEach((p) => { actionMap[p.action] = p.id; });
const hasRead = !!actionMap['Read'] && permissionIds().includes(actionMap['Read']);
const hasCreate = !!actionMap['Create'] && permissionIds().includes(actionMap['Create']);
const hasUpdate = !!actionMap['Update'] && permissionIds().includes(actionMap['Update']);
const hasDelete = !!actionMap['Delete'] && permissionIds().includes(actionMap['Delete']);
const noAccess = !hasRead && !hasCreate && !hasUpdate && !hasDelete;
return (
<tr class="hover:bg-slate-50">
<td class="font-semibold text-slate-900">{mod}</td>
<td><input type="checkbox" checked={noAccess} disabled aria-label={`${mod} no access`} /></td>
<td>{actionMap['Read'] ? <input type="checkbox" checked={hasRead} onChange={() => togglePermission(actionMap['Read'])} aria-label={`${mod} read`} /> : <span class="text-slate-300"></span>}</td>
<td>{actionMap['Create'] ? <input type="checkbox" checked={hasCreate} onChange={() => togglePermission(actionMap['Create'])} aria-label={`${mod} create`} /> : <span class="text-slate-300"></span>}</td>
<td>{actionMap['Update'] ? <input type="checkbox" checked={hasUpdate} onChange={() => togglePermission(actionMap['Update'])} aria-label={`${mod} update`} /> : <span class="text-slate-300"></span>}</td>
<td>{actionMap['Delete'] ? <input type="checkbox" checked={hasDelete} onChange={() => togglePermission(actionMap['Delete'])} aria-label={`${mod} delete`} /> : <span class="text-slate-300"></span>}</td>
</tr>
);
})}
</tbody>
</table>
</div>
</div>
</Show>
{/* Save */}
<div class="flex justify-end gap-3 mt-2">
<A class="rounded-lg border border-gray-200 px-4 py-2 text-sm font-medium text-gray-700 hover:bg-gray-50 transition-colors" href="/admin/roles">Cancel</A>
<button
class="btn-primary"
onClick={handleSave}
disabled={saving() || !roleName().trim()}
>
{saving() ? 'Creating...' : 'Create Role'}
</button>
</div>
</div>
</div>
</AdminShell>
);
}