Files
dezky/apps/portal/components/partner/InviteTeammateModal.vue
T
Ronni Baslund 0bd4e5498e feat: portal redesign, pricing catalog, partner-staff invites
- portal: new admin/ and partner/ surfaces with full component library
  (AppLauncher, Avatar, Badge, Card, Modal, Tabs, etc.), composables,
  layouts, partner-routing middleware, and supporting server APIs
- pricing: Price schema/module with operator CRUD, pricing.vue catalog UI,
  Subscription extended with cycle/currency/perSeatAmount/seats snapshots
  for stable MRR aggregation
- partner staff: User.partnerId, invite-partner-user DTO and flow,
  /partners/:slug/users endpoints, InvitePartnerUserModal, shared
  dezky-partner-staff Authentik group
- /me: partner-aware endpoint returning user + partner context so portal
  can route between end-user and partner-admin surfaces
- tenant: seats field for portfolio displays and future MRR calculations
- operator: pricing page, signed-out page, useMe/useToast composables,
  ToastStack
2026-05-28 20:00:33 +02:00

296 lines
8.6 KiB
Vue

<script setup lang="ts">
// Invite a teammate to the partner organization. Role + customer-access
// scoping + require-MFA toggle + optional personal note. Invitations expire
// after 7 days — the design surfaces that explicitly.
import { customers } from '~/data/customers'
defineProps<{ open: boolean }>()
const emit = defineEmits<{ close: []; sent: [payload: { email: string; role: string }] }>()
const name = ref('')
const email = ref('')
const role = ref<'Partner admin' | 'Sales' | 'Support' | 'Billing'>('Sales')
const access = ref<'all' | 'specific' | 'none'>('all')
const specific = ref<string[]>([])
const requireMfa = ref(true)
const message = ref('')
const ROLE_OPTS = [
{ v: 'Partner admin', d: 'Full access · billing · settings · all customers' },
{ v: 'Sales', d: 'Customer orgs · provisioning · plan changes' },
{ v: 'Support', d: 'Enter customers · view tickets · no billing' },
{ v: 'Billing', d: 'Invoices · payouts · cannot enter customers' },
] as const
const ACCESS_OPTS = [
{ v: 'all', l: 'All customers', d: 'Including new ones added later' },
{ v: 'specific', l: 'Specific customers', d: 'Pick from the list below' },
{ v: 'none', l: 'No customer access', d: 'Partner-only console (for Billing role)' },
] as const
function toggleCustomer(id: string) {
if (specific.value.includes(id)) specific.value = specific.value.filter((x) => x !== id)
else specific.value = [...specific.value, id]
}
function planBadgeTone(p: string) {
return p === 'enterprise' ? 'invert' : 'neutral'
}
</script>
<template>
<Modal
:open="open"
eyebrow="Partner team · invite"
title="Invite teammate"
size="md"
@close="emit('close')"
>
<div class="form">
<div class="row-2">
<label class="field">
<Eyebrow>Full name</Eyebrow>
<input v-model="name" placeholder="Anne Baslund" />
</label>
<label class="field">
<Eyebrow>Email</Eyebrow>
<input v-model="email" placeholder="name@nordicmsp.dk" />
</label>
</div>
<div>
<Eyebrow>Role</Eyebrow>
<div class="role-grid">
<button
v-for="o in ROLE_OPTS"
:key="o.v"
type="button"
class="role-card"
:class="{ selected: role === o.v }"
@click="role = o.v as any"
>
<div class="rc-top">
<span class="rc-name">{{ o.v }}</span>
<Badge v-if="o.v === 'Partner admin'" tone="invert">all access</Badge>
</div>
<Mono dim>{{ o.d }}</Mono>
</button>
</div>
</div>
<div>
<Eyebrow>Customer access</Eyebrow>
<div class="access-list">
<button
v-for="o in ACCESS_OPTS"
:key="o.v"
type="button"
class="access-row"
:class="{ selected: access === o.v }"
@click="access = o.v as any"
>
<span class="radio" :class="{ on: access === o.v }">
<span v-if="access === o.v" class="radio-inner" />
</span>
<div class="ar-meta">
<div class="ar-label">{{ o.l }}</div>
<Mono dim>{{ o.d }}</Mono>
</div>
</button>
</div>
<div v-if="access === 'specific'" class="picker">
<div class="picker-head">
<Mono dim>{{ specific.length }} of {{ customers.length }} selected</Mono>
</div>
<div class="picker-list">
<label v-for="c in customers" :key="c.id" class="picker-row">
<input
type="checkbox"
:checked="specific.includes(c.id)"
@change="toggleCustomer(c.id)"
/>
<div class="cust-swatch" :style="{ background: c.brandColor }" />
<span class="cust-name">{{ c.name }}</span>
<Badge :tone="planBadgeTone(c.plan)">{{ c.planLabel }}</Badge>
</label>
</div>
</div>
</div>
<div class="mfa-row">
<div>
<div class="mfa-label">Require MFA on first sign-in</div>
<Mono dim>recommended for any partner role with customer access</Mono>
</div>
<button class="switch" :class="{ on: requireMfa }" @click="requireMfa = !requireMfa">
<span class="thumb" />
</button>
</div>
<label class="field">
<Eyebrow>Personal note · optional</Eyebrow>
<textarea
v-model="message"
rows="3"
placeholder="Welcome to the team — looking forward to working together."
/>
</label>
<div class="warn">
<UiIcon name="shield" :size="14" />
<p>
Invitations expire after <b>7 days</b>. The teammate will create their own password and
complete MFA enrolment before getting access.
</p>
</div>
</div>
<template #footer>
<UiButton variant="ghost" @click="emit('close')">Cancel</UiButton>
<UiButton
variant="primary"
:disabled="!email"
@click="emit('sent', { email, role }); emit('close')"
>
<template #leading><UiIcon name="mail" :size="14" /></template>
Send invitation
</UiButton>
</template>
</Modal>
</template>
<style scoped>
.form { display: flex; flex-direction: column; gap: 16px; }
.row-2 { display: grid; grid-template-columns: 1fr 1fr; gap: 12px; }
.field { display: flex; flex-direction: column; gap: 6px; }
.field input, .field textarea {
padding: 9px 12px;
background: var(--surface);
border: 1px solid var(--border);
border-radius: 6px;
font-family: inherit;
font-size: 13px;
color: var(--text);
}
.field textarea { resize: vertical; line-height: 1.55; }
.field input:focus, .field textarea:focus { outline: none; border-color: var(--border-hi); }
.role-grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 8px;
margin-top: 8px;
}
.role-card {
padding: 12px;
border-radius: 6px;
border: 1px solid var(--border);
background: var(--surface);
text-align: left;
cursor: pointer;
font-family: inherit;
}
.role-card.selected { border-color: var(--text); background: var(--bg); }
.rc-top { display: flex; justify-content: space-between; align-items: center; margin-bottom: 4px; }
.rc-name { font-size: 13px; font-weight: 500; }
.access-list { display: flex; flex-direction: column; gap: 8px; margin-top: 8px; }
.access-row {
display: flex;
align-items: center;
gap: 10px;
padding: 10px;
border-radius: 6px;
border: 1px solid var(--border);
background: var(--surface);
font-family: inherit;
cursor: pointer;
text-align: left;
}
.access-row.selected { border-color: var(--text); background: var(--bg); }
.radio {
width: 14px;
height: 14px;
border-radius: 999px;
border: 1.5px solid var(--border-hi);
background: var(--bg);
flex-shrink: 0;
display: inline-flex;
align-items: center;
justify-content: center;
}
.radio.on { border: 4px solid var(--text); }
.ar-meta { flex: 1; }
.ar-label { font-size: 13px; font-weight: 500; }
.picker {
margin-top: 10px;
padding: 12px;
background: var(--surface);
border: 1px solid var(--border);
border-radius: 6px;
max-height: 240px;
overflow-y: auto;
}
.picker-head { margin-bottom: 8px; }
.picker-list { display: flex; flex-direction: column; gap: 6px; }
.picker-row {
display: flex;
align-items: center;
gap: 10px;
cursor: pointer;
font-size: 13px;
}
.picker-row input[type='checkbox'] { width: 14px; height: 14px; accent-color: var(--text); }
.cust-swatch { width: 16px; height: 16px; border-radius: 3px; flex-shrink: 0; }
.cust-name { flex: 1; }
.mfa-row {
padding: 12px 14px;
background: var(--surface);
border: 1px solid var(--border);
border-radius: 6px;
display: flex;
align-items: center;
justify-content: space-between;
gap: 12px;
}
.mfa-label { font-size: 13px; font-weight: 500; }
.switch {
width: 36px;
height: 20px;
border-radius: 999px;
background: var(--border);
border: none;
padding: 2px;
display: inline-flex;
align-items: center;
cursor: pointer;
transition: background 150ms;
flex-shrink: 0;
}
.switch.on { background: var(--text); }
.thumb {
width: 16px;
height: 16px;
border-radius: 999px;
background: var(--bg);
transition: transform 150ms;
}
.switch.on .thumb { transform: translateX(16px); }
.warn {
padding: 12px;
background: rgba(232, 154, 31, 0.08);
border: 1px solid rgba(232, 154, 31, 0.24);
border-radius: 6px;
display: flex;
gap: 10px;
}
.warn :deep(svg) { color: var(--warn); margin-top: 2px; flex-shrink: 0; }
.warn p { font-size: 12px; color: var(--text-dim); line-height: 1.55; margin: 0; }
</style>