0bd4e5498e
- 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
296 lines
8.6 KiB
Vue
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>
|