Add invitation frontend: invite modal, pending invitations, no-company view

Members Index:
- "Invite Member" button opens InviteMemberModal (email + role dropdown)
- Pending invitations section shows below members table with cancel buttons
- Members store gains inviteMember, fetchPendingInvitations, cancelInvitation

CompanySwitcher:
- Shows pending invitations greyed out below active companies
- Each with Accept/Decline mini-buttons
- Accepting refreshes bootstrap and switches to new company

NoCompanyView:
- Standalone page for users with zero accepted companies
- Shows pending invitations with Accept/Decline or "no companies" message
- Route: /admin/no-company

Invitation Pinia store:
- Manages user's own pending invitations (fetchPending, accept, decline)
- Bootstrap populates invitations from API response

Global store:
- Bootstrap action stores pending_invitations from response
This commit is contained in:
Darko Gjorgjijoski
2026-04-03 23:20:41 +02:00
parent 8a6c085288
commit 6343b4a17f
9 changed files with 491 additions and 2 deletions

View File

@@ -0,0 +1,132 @@
<template>
<BaseModal :show="show" @close="$emit('close')">
<template #header>
<div class="flex justify-between w-full">
{{ $t('members.invite_member') }}
<BaseIcon
name="XMarkIcon"
class="w-6 h-6 text-gray-500 cursor-pointer"
@click="$emit('close')"
/>
</div>
</template>
<form @submit.prevent="submitInvitation">
<div class="p-4 space-y-4">
<BaseInputGroup
:label="$t('members.email')"
:error="v$.email.$error && v$.email.$errors[0].$message"
required
>
<BaseInput
v-model="form.email"
type="email"
:invalid="v$.email.$error"
@input="v$.email.$touch()"
/>
</BaseInputGroup>
<BaseInputGroup
:label="$t('members.role')"
:error="v$.role_id.$error && v$.role_id.$errors[0].$message"
required
>
<BaseMultiselect
v-model="form.role_id"
:options="roles"
label="title"
value-prop="id"
track-by="title"
:searchable="true"
/>
</BaseInputGroup>
</div>
<div class="flex justify-end p-4 border-t border-gray-200">
<BaseButton
variant="primary-outline"
class="mr-3"
@click="$emit('close')"
>
{{ $t('general.cancel') }}
</BaseButton>
<BaseButton
:loading="isSending"
:disabled="isSending"
type="submit"
>
{{ $t('members.invite_member') }}
</BaseButton>
</div>
</form>
</BaseModal>
</template>
<script setup>
import { ref, reactive, computed, onMounted } from 'vue'
import { useMembersStore } from '@/scripts/admin/stores/members'
import { useI18n } from 'vue-i18n'
import { helpers, required, email } from '@vuelidate/validators'
import { useVuelidate } from '@vuelidate/core'
import http from '@/scripts/http'
const props = defineProps({
show: {
type: Boolean,
default: false,
},
})
const emit = defineEmits(['close'])
const membersStore = useMembersStore()
const { t } = useI18n()
const isSending = ref(false)
const roles = ref([])
const form = reactive({
email: '',
role_id: null,
})
const rules = computed(() => ({
email: {
required: helpers.withMessage(t('validation.required'), required),
email: helpers.withMessage(t('validation.email_incorrect'), email),
},
role_id: {
required: helpers.withMessage(t('validation.required'), required),
},
}))
const v$ = useVuelidate(
rules,
computed(() => form)
)
onMounted(async () => {
const response = await http.get('/api/v1/roles')
roles.value = response.data.data
})
async function submitInvitation() {
v$.value.$touch()
if (v$.value.$invalid) return
isSending.value = true
try {
await membersStore.inviteMember({
email: form.email,
role_id: form.role_id,
})
form.email = ''
form.role_id = null
v$.value.$reset()
emit('close')
} catch (e) {
// Error handled by store
} finally {
isSending.value = false
}
}
</script>