ad20d1f4c9
Auto Tag Develop / tag (push) Successful in 8s
## Contexte Ticket Lesstime : #73 (id 492) — volet front de la pagination (groupe Transversal). Dépend du back ERP-72 (déjà mergé sur develop). Pas de spec docs/specs ; référence = description #73 + .claude/rules/frontend.md. ## Implémentation - Composable réutilisable `usePaginatedList` (`frontend/shared/composables/`) générique, branché directement sur `MalioDataTable` (props page/perPage/totalItems + events update:page/update:per-page). - Force `Accept: application/ld+json` (sans Accept, API Platform renvoie un tableau plat sans pagination). - Migration des pages admin existantes (M0 catégories, Sites, Utilisateurs, Rôles) vers le composable. - Refactor de `useCategoriesAdmin` : ne porte plus la liste paginée (déplacée vers `usePaginatedList<Category>` dans la page) et concentre son rôle sur le référentiel `CategoryType` (chargé en une fois via `?pagination=false`, échappatoire prévue par `pagination_client_enabled: true` côté back). - Cas limites couverts : liste vide (pas de contrôle pagination affiché), page hors borne après filtre (retombe sur la dernière page valide), items/page 10/25/50, reset filtres/tri, swallow erreur réseau. - Pattern « liste paginée » documenté dans `.claude/rules/frontend.md` (section dédiée + exemple). ## Décision URL Le ticket suggérait « idéalement page/tri/filtre dans l'URL » — arbitré explicitement par Tristan en faveur de la règle ABSOLUE n°6 du CLAUDE.md (state local uniquement, jamais persisté dans l'URL). Aucun reflet URL implémenté ; comportement homogène entre toutes les listes migrées. ## Tests - `make nuxt-test` : 101/101 OK (22 nouveaux tests sur `usePaginatedList`, 6 anciens tests `useCategoriesAdmin.fetchAll` retirés en cohérence avec la refacto). - Vérification manuelle dans le navigateur (`make dev-nuxt`) : Sites, Utilisateurs, Rôles, Catégories affichent le sélecteur `Lignes : 10` et les boutons Prev/Next ; audit-log (non migré, composable spécifique) intact avec ses 3 pages. - Aucun test E2E ajouté (règle d'or projet). - Pre-commit hook : ESLint + PHPUnit 322/322 OK. ## Hors périmètre - `audit-log.vue` non migré : composable `useAuditLog` spécifique (cache partagé page/timeline, filtres complexes, persistance URL préexistante). Refactor risqué et net-zéro pour ERP-73. - M1 répertoire clients : pas encore livré sur develop (seules les specs sont mergées via #23). Le futur écran consommera `usePaginatedList` dès sa création. Reviewed-on: #30 Co-authored-by: tristan <tristan@yuno.malio.fr> Co-committed-by: tristan <tristan@yuno.malio.fr>
159 lines
4.4 KiB
Vue
159 lines
4.4 KiB
Vue
<template>
|
|
<div>
|
|
<PageHeader>
|
|
{{ t('admin.roles.title') }}
|
|
<template #actions>
|
|
<MalioButton
|
|
v-if="can('core.roles.manage')"
|
|
:label="t('admin.roles.newRole')"
|
|
icon-name="mdi:add-bold"
|
|
icon-position="left"
|
|
@click="openCreateDrawer"
|
|
/>
|
|
</template>
|
|
</PageHeader>
|
|
|
|
<!-- Table des roles — pagination serveur via usePaginatedList (#73). -->
|
|
<MalioDataTable
|
|
:columns="columns"
|
|
:items="roleItems"
|
|
:total-items="totalItems"
|
|
:page="currentPage"
|
|
:per-page="itemsPerPage"
|
|
:per-page-options="itemsPerPageOptions"
|
|
:row-clickable="canManage"
|
|
:empty-message="t('admin.roles.noRoles')"
|
|
@row-click="onRowClick"
|
|
@update:page="goToPage"
|
|
@update:per-page="setItemsPerPage"
|
|
>
|
|
<template #cell-code="{ item }">
|
|
<span class="font-mono text-xs">{{ item.code }}</span>
|
|
</template>
|
|
<template #cell-permissions="{ item }">
|
|
{{ item.permissions }}
|
|
</template>
|
|
<template #cell-system="{ item }">
|
|
<span
|
|
v-if="item.isSystem"
|
|
class="inline-flex items-center rounded-full bg-blue-100 px-2.5 py-0.5 text-xs font-medium text-blue-800"
|
|
>
|
|
{{ t('admin.roles.table.system') }}
|
|
</span>
|
|
</template>
|
|
</MalioDataTable>
|
|
|
|
<!-- Drawer creation/edition -->
|
|
<RoleDrawer
|
|
v-model="drawerOpen"
|
|
:role="selectedRole"
|
|
@saved="onRoleSaved"
|
|
@delete="onDeleteRequest"
|
|
/>
|
|
|
|
<!-- Modale de suppression -->
|
|
<RoleDeleteModal
|
|
v-model="deleteModalOpen"
|
|
:role-label="roleToDelete?.label ?? ''"
|
|
:loading="deleting"
|
|
@confirm="handleDelete"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import type { Role } from '~/shared/types/rbac'
|
|
|
|
const { t } = useI18n()
|
|
const api = useApi()
|
|
const { can } = usePermissions()
|
|
const canManage = computed(() => can('core.roles.manage'))
|
|
|
|
useHead({ title: t('admin.roles.title') })
|
|
|
|
// Pagination serveur via le composable partage (#73).
|
|
const {
|
|
items: roles,
|
|
totalItems,
|
|
currentPage,
|
|
itemsPerPage,
|
|
itemsPerPageOptions,
|
|
fetch: loadRoles,
|
|
goToPage,
|
|
setItemsPerPage,
|
|
} = usePaginatedList<Role>({ url: '/roles' })
|
|
|
|
const columns = [
|
|
{ key: 'label', label: t('admin.roles.table.label') },
|
|
{ key: 'code', label: t('admin.roles.table.code') },
|
|
{ key: 'permissions', label: t('admin.roles.table.permissions') },
|
|
{ key: 'system', label: t('admin.roles.table.system') },
|
|
]
|
|
|
|
// Transformer les roles en items compatibles MalioDataTable
|
|
const roleItems = computed(() =>
|
|
roles.value.map(role => ({
|
|
id: role.id,
|
|
label: role.label,
|
|
code: role.code,
|
|
permissions: role.permissions.length,
|
|
isSystem: role.isSystem,
|
|
system: '', // colonne geree par le slot
|
|
}))
|
|
)
|
|
|
|
function getRoleById(id: number): Role | undefined {
|
|
return roles.value.find(r => r.id === id)
|
|
}
|
|
|
|
function onRowClick(item: Record<string, unknown>) {
|
|
const role = getRoleById(item.id as number)
|
|
if (role) openEditDrawer(role)
|
|
}
|
|
const drawerOpen = ref(false)
|
|
const selectedRole = ref<Role | null>(null)
|
|
const deleteModalOpen = ref(false)
|
|
const roleToDelete = ref<Role | null>(null)
|
|
const deleting = ref(false)
|
|
|
|
function openCreateDrawer() {
|
|
selectedRole.value = null
|
|
drawerOpen.value = true
|
|
}
|
|
|
|
function openEditDrawer(role: Role) {
|
|
selectedRole.value = role
|
|
drawerOpen.value = true
|
|
}
|
|
|
|
function onDeleteRequest() {
|
|
if (!selectedRole.value || selectedRole.value.isSystem) return
|
|
roleToDelete.value = selectedRole.value
|
|
deleteModalOpen.value = true
|
|
}
|
|
|
|
async function handleDelete() {
|
|
if (!roleToDelete.value) return
|
|
deleting.value = true
|
|
try {
|
|
await api.delete(`/roles/${roleToDelete.value.id}`, {}, {
|
|
toastSuccessMessage: t('admin.roles.toast.deleted'),
|
|
})
|
|
deleteModalOpen.value = false
|
|
roleToDelete.value = null
|
|
drawerOpen.value = false
|
|
await loadRoles()
|
|
} finally {
|
|
deleting.value = false
|
|
}
|
|
}
|
|
|
|
function onRoleSaved() {
|
|
loadRoles()
|
|
}
|
|
|
|
onMounted(() => {
|
|
loadRoles()
|
|
})
|
|
</script>
|