Files
InvoiceShelf/resources/scripts-v2/features/company/settings/components/RolesModal.vue
Darko Gjorgjijoski a46cca5cd8 Complete scripts-v2 TypeScript migration — all imports resolved,
build passes

Create all missing components (modals, dropdowns, icons, tabs, mail
drivers, customer partials), fix all @/scripts/ imports to @v2/,
wire up vite entry point and blade template. 382 files, 48883 lines.

- 27 settings components: modals (tax, payment, custom field, note,
  category, role, exchange rate, unit, mail test), dropdowns (6),
  customization tabs (4), mail driver forms (4)
- 22 icon components: 5 utility icons, 4 dashboard icons, 13 editor
  toolbar icons with typed barrel export
- 3 customer components: info, chart placeholder, custom fields single
- Fixed usePopper composable, client/format-money import patterns
- Zero remaining @/scripts/ imports in scripts-v2/

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-04 09:30:00 +02:00

348 lines
8.9 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 '@v2/stores/modal.store'
import { roleService } from '@v2/api/services/role.service'
import type { CreateRolePayload } from '@v2/api/services/role.service'
import type { Ability } from '@v2/types/domain/role'
interface AbilityItem extends Ability {
ability: string
disabled: boolean
depends_on?: string[]
}
interface AbilitiesList {
[group: string]: AbilityItem[]
}
interface RoleForm {
id: number | null
name: string
abilities: AbilityItem[]
}
const modalStore = useModalStore()
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) => {
const group = a.title || 'General'
if (!grouped[group]) grouped[group] = []
grouped[group].push({
...a,
ability: a.name,
disabled: false,
depends_on: [],
})
})
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: response.data.abilities.map((a) => ({
...a,
ability: a.name,
disabled: false,
depends_on: [],
})),
}
}
} 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)
} else {
await roleService.create(payload)
}
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>