mirror of
https://github.com/InvoiceShelf/InvoiceShelf.git
synced 2026-04-15 17:24:10 +00:00
Now that the legacy v1 frontend (commit 064bdf53) is gone, the v2 directory is the only frontend and the v2 suffix is just noise. Renames resources/scripts-v2 to resources/scripts via git mv (so git records the move as renames, preserving blame and log --follow), then bulk-rewrites the 152 files that imported via @v2/... to use @/scripts/... instead. The existing @ alias (resources/) covers the new path with no extra config needed.
Drops the now-unused @v2 alias from vite.config.js and points the laravel-vite-plugin entry at resources/scripts/main.ts. Updates the only blade reference (resources/views/app.blade.php) to match. The package.json test script (eslint ./resources/scripts) automatically targets the right place after the rename without any edit.
Verified: npm run build exits clean and the Vite warning lines now reference resources/scripts/plugins/i18n.ts, confirming every import resolved through the new path. git log --follow on any moved file walks back through its scripts-v2 history.
385 lines
10 KiB
Vue
385 lines
10 KiB
Vue
<script setup lang="ts">
|
|
import { computed, ref } from 'vue'
|
|
import { useI18n } from 'vue-i18n'
|
|
import { required, minLength, helpers } from '@vuelidate/validators'
|
|
import useVuelidate from '@vuelidate/core'
|
|
import { useModalStore } from '@/scripts/stores/modal.store'
|
|
import { useNotificationStore } from '@/scripts/stores/notification.store'
|
|
import { roleService } from '@/scripts/api/services/role.service'
|
|
import type { CreateRolePayload } from '@/scripts/api/services/role.service'
|
|
import type { Ability } from '@/scripts/types/domain/role'
|
|
|
|
interface AbilityItem {
|
|
name: string
|
|
ability: string
|
|
disabled: boolean
|
|
depends_on?: string[]
|
|
model?: string
|
|
}
|
|
|
|
interface AbilitiesList {
|
|
[group: string]: AbilityItem[]
|
|
}
|
|
|
|
interface RoleForm {
|
|
id: number | null
|
|
name: string
|
|
abilities: AbilityItem[]
|
|
}
|
|
|
|
const modalStore = useModalStore()
|
|
const notificationStore = useNotificationStore()
|
|
const { t } = useI18n()
|
|
|
|
const isSaving = ref<boolean>(false)
|
|
const isFetchingInitialData = ref<boolean>(false)
|
|
const isEdit = ref<boolean>(false)
|
|
|
|
const currentRole = ref<RoleForm>({
|
|
id: null,
|
|
name: '',
|
|
abilities: [],
|
|
})
|
|
|
|
const abilitiesList = ref<AbilitiesList>({})
|
|
|
|
const modalActive = computed<boolean>(
|
|
() => modalStore.active && modalStore.componentName === 'RolesModal'
|
|
)
|
|
|
|
const rules = computed(() => ({
|
|
name: {
|
|
required: helpers.withMessage(t('validation.required'), required),
|
|
minLength: helpers.withMessage(
|
|
t('validation.name_min_length', { count: 3 }),
|
|
minLength(3)
|
|
),
|
|
},
|
|
abilities: {
|
|
required: helpers.withMessage(
|
|
t('validation.at_least_one_ability'),
|
|
required
|
|
),
|
|
},
|
|
}))
|
|
|
|
const v$ = useVuelidate(rules, currentRole)
|
|
|
|
async function setInitialData(): Promise<void> {
|
|
isFetchingInitialData.value = true
|
|
|
|
const abilitiesRes = await roleService.getAbilities()
|
|
if (abilitiesRes.abilities) {
|
|
const grouped: AbilitiesList = {}
|
|
abilitiesRes.abilities.forEach((a: Record<string, unknown>) => {
|
|
// Extract model name from PHP class path (e.g., "App\Models\Customer" → "Customer")
|
|
const modelPath = (a.model as string) ?? ''
|
|
const modelName = modelPath
|
|
? modelPath.substring(modelPath.lastIndexOf('\\') + 1)
|
|
: 'Common'
|
|
|
|
if (!grouped[modelName]) grouped[modelName] = []
|
|
grouped[modelName].push({
|
|
name: a.name as string,
|
|
ability: a.ability as string,
|
|
disabled: false,
|
|
depends_on: (a.depends_on as string[]) ?? [],
|
|
} as AbilityItem)
|
|
})
|
|
abilitiesList.value = grouped
|
|
}
|
|
|
|
if (modalStore.data && typeof modalStore.data === 'number') {
|
|
isEdit.value = true
|
|
const response = await roleService.get(modalStore.data)
|
|
if (response.data) {
|
|
currentRole.value = {
|
|
id: response.data.id,
|
|
name: response.data.name,
|
|
abilities: [],
|
|
}
|
|
|
|
// Match role's abilities with the full ability objects from abilitiesList
|
|
const roleAbilities = (response.data.abilities ?? []) as Array<Record<string, unknown>>
|
|
roleAbilities.forEach((ra) => {
|
|
Object.keys(abilitiesList.value).forEach((group) => {
|
|
abilitiesList.value[group].forEach((_p) => {
|
|
if (_p.ability === ra.name) {
|
|
currentRole.value.abilities.push(_p)
|
|
}
|
|
})
|
|
})
|
|
})
|
|
|
|
// Set disabled state for dependent abilities
|
|
currentRole.value.abilities.forEach((ab) => {
|
|
ab.depends_on?.forEach((_d) => {
|
|
Object.keys(abilitiesList.value).forEach((group) => {
|
|
abilitiesList.value[group].forEach((_a) => {
|
|
if (_d === _a.ability) {
|
|
_a.disabled = true
|
|
}
|
|
})
|
|
})
|
|
})
|
|
})
|
|
}
|
|
} else {
|
|
isEdit.value = false
|
|
currentRole.value = { id: null, name: '', abilities: [] }
|
|
}
|
|
|
|
isFetchingInitialData.value = false
|
|
}
|
|
|
|
async function submitRoleData(): Promise<void> {
|
|
v$.value.$touch()
|
|
|
|
if (v$.value.$invalid) {
|
|
return
|
|
}
|
|
|
|
isSaving.value = true
|
|
try {
|
|
const payload: CreateRolePayload = {
|
|
name: currentRole.value.name,
|
|
abilities: currentRole.value.abilities.map((a) => ({
|
|
ability: a.ability,
|
|
})),
|
|
}
|
|
|
|
if (isEdit.value && currentRole.value.id) {
|
|
await roleService.update(currentRole.value.id, payload)
|
|
notificationStore.showNotification({
|
|
type: 'success',
|
|
message: 'settings.roles.updated_message',
|
|
})
|
|
} else {
|
|
await roleService.create(payload)
|
|
notificationStore.showNotification({
|
|
type: 'success',
|
|
message: 'settings.roles.created_message',
|
|
})
|
|
}
|
|
|
|
isSaving.value = false
|
|
if (modalStore.refreshData) {
|
|
modalStore.refreshData()
|
|
}
|
|
closeRolesModal()
|
|
} catch {
|
|
isSaving.value = false
|
|
}
|
|
}
|
|
|
|
function onUpdateAbility(currentAbility: AbilityItem): void {
|
|
const fd = currentRole.value.abilities.find(
|
|
(_abl) => _abl.ability === currentAbility.ability
|
|
)
|
|
|
|
if (!fd && currentAbility.depends_on?.length) {
|
|
enableAbilities(currentAbility)
|
|
return
|
|
}
|
|
|
|
currentAbility.depends_on?.forEach((_d) => {
|
|
Object.keys(abilitiesList.value).forEach((group) => {
|
|
abilitiesList.value[group].forEach((_a) => {
|
|
if (_d === _a.ability) {
|
|
_a.disabled = true
|
|
const found = currentRole.value.abilities.find(
|
|
(_af) => _af.ability === _d
|
|
)
|
|
if (!found) {
|
|
currentRole.value.abilities.push(_a)
|
|
}
|
|
}
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function setSelectAll(checked: boolean): void {
|
|
const dependList: string[] = []
|
|
Object.keys(abilitiesList.value).forEach((group) => {
|
|
abilitiesList.value[group].forEach((_a) => {
|
|
if (_a.depends_on) {
|
|
dependList.push(..._a.depends_on)
|
|
}
|
|
})
|
|
})
|
|
|
|
Object.keys(abilitiesList.value).forEach((group) => {
|
|
abilitiesList.value[group].forEach((_a) => {
|
|
if (dependList.includes(_a.ability)) {
|
|
_a.disabled = checked
|
|
}
|
|
currentRole.value.abilities.push(_a)
|
|
})
|
|
})
|
|
|
|
if (!checked) {
|
|
currentRole.value.abilities = []
|
|
}
|
|
}
|
|
|
|
function enableAbilities(ability: AbilityItem): void {
|
|
ability.depends_on?.forEach((_d) => {
|
|
Object.keys(abilitiesList.value).forEach((group) => {
|
|
abilitiesList.value[group].forEach((_a) => {
|
|
const found = currentRole.value.abilities.find((_r) =>
|
|
_r.depends_on?.includes(_a.ability)
|
|
)
|
|
if (_d === _a.ability && !found) {
|
|
_a.disabled = false
|
|
}
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function closeRolesModal(): void {
|
|
modalStore.closeModal()
|
|
setTimeout(() => {
|
|
currentRole.value = { id: null, name: '', abilities: [] }
|
|
isEdit.value = false
|
|
|
|
Object.keys(abilitiesList.value).forEach((group) => {
|
|
abilitiesList.value[group].forEach((_a) => {
|
|
_a.disabled = false
|
|
})
|
|
})
|
|
|
|
v$.value.$reset()
|
|
}, 300)
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<BaseModal
|
|
:show="modalActive"
|
|
@close="closeRolesModal"
|
|
@open="setInitialData"
|
|
>
|
|
<template #header>
|
|
<div class="flex justify-between w-full">
|
|
{{ modalStore.title }}
|
|
<BaseIcon
|
|
name="XMarkIcon"
|
|
class="w-6 h-6 text-muted cursor-pointer"
|
|
@click="closeRolesModal"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<form @submit.prevent="submitRoleData">
|
|
<div class="px-4 md:px-8 py-4 md:py-6">
|
|
<BaseInputGroup
|
|
:label="$t('settings.roles.name')"
|
|
class="mt-3"
|
|
:error="v$.name.$error && v$.name.$errors[0].$message"
|
|
required
|
|
:content-loading="isFetchingInitialData"
|
|
>
|
|
<BaseInput
|
|
v-model="currentRole.name"
|
|
:invalid="v$.name.$error"
|
|
type="text"
|
|
:content-loading="isFetchingInitialData"
|
|
@input="v$.name.$touch()"
|
|
/>
|
|
</BaseInputGroup>
|
|
</div>
|
|
|
|
<div class="flex justify-between">
|
|
<h6
|
|
class="text-sm not-italic font-medium text-heading px-4 md:px-8 py-1.5"
|
|
>
|
|
{{ $t('settings.roles.permission', 2) }}
|
|
<span class="text-sm text-red-500"> *</span>
|
|
</h6>
|
|
<div
|
|
class="text-sm not-italic font-medium text-subtle px-4 md:px-8 py-1.5"
|
|
>
|
|
<a
|
|
class="cursor-pointer text-primary-400"
|
|
@click="setSelectAll(true)"
|
|
>
|
|
{{ $t('settings.roles.select_all') }}
|
|
</a>
|
|
/
|
|
<a
|
|
class="cursor-pointer text-primary-400"
|
|
@click="setSelectAll(false)"
|
|
>
|
|
{{ $t('settings.roles.none') }}
|
|
</a>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="border-t border-line-default py-3">
|
|
<div
|
|
class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4 px-8 sm:px-8"
|
|
>
|
|
<div
|
|
v-for="(abilityGroup, gIndex) in abilitiesList"
|
|
:key="gIndex"
|
|
class="flex flex-col space-y-1"
|
|
>
|
|
<p
|
|
class="text-sm text-muted border-b border-line-default pb-1 mb-2"
|
|
>
|
|
{{ gIndex }}
|
|
</p>
|
|
<div
|
|
v-for="(ability, index) in abilityGroup"
|
|
:key="index"
|
|
class="flex"
|
|
>
|
|
<BaseCheckbox
|
|
v-model="currentRole.abilities"
|
|
:set-initial-value="true"
|
|
variant="primary"
|
|
:disabled="ability.disabled"
|
|
:label="ability.name"
|
|
:value="ability"
|
|
@update:model-value="onUpdateAbility(ability)"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<span
|
|
v-if="v$.abilities.$error"
|
|
class="block mt-0.5 text-sm text-red-500"
|
|
>
|
|
{{ v$.abilities.$errors[0].$message }}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
<div
|
|
class="z-0 flex justify-end p-4 border-t border-solid border-line-default"
|
|
>
|
|
<BaseButton
|
|
class="mr-3 text-sm"
|
|
variant="primary-outline"
|
|
type="button"
|
|
@click="closeRolesModal"
|
|
>
|
|
{{ $t('general.cancel') }}
|
|
</BaseButton>
|
|
<BaseButton
|
|
:loading="isSaving"
|
|
:disabled="isSaving"
|
|
variant="primary"
|
|
type="submit"
|
|
>
|
|
<template #left="slotProps">
|
|
<BaseIcon name="ArrowDownOnSquareIcon" :class="slotProps.class" />
|
|
</template>
|
|
{{ isEdit ? $t('general.update') : $t('general.save') }}
|
|
</BaseButton>
|
|
</div>
|
|
</form>
|
|
</BaseModal>
|
|
</template>
|