chore: update lockfile and commit all pending release fixes
Some checks failed
Monorepo Pipeline / ⚡ Prioritize Release (push) Successful in 1s
Monorepo Pipeline / 🧪 Test (push) Successful in 2m2s
Monorepo Pipeline / 🧹 Lint (push) Successful in 2m14s
Monorepo Pipeline / 🚀 Release (push) Has been cancelled
Monorepo Pipeline / 🐳 Build Directus (Base) (push) Has been cancelled
Monorepo Pipeline / 🐳 Build Gatekeeper (Product) (push) Has been cancelled
Monorepo Pipeline / 🐳 Build Build-Base (push) Has been cancelled
Monorepo Pipeline / 🐳 Build Production Runtime (push) Has been cancelled
Monorepo Pipeline / 🏗️ Build (push) Has been cancelled

This commit is contained in:
2026-02-14 13:57:46 +01:00
parent ceaf3ae3ea
commit d69ade6268
25 changed files with 941 additions and 598 deletions

2
.env
View File

@@ -1,5 +1,5 @@
# Project # Project
IMAGE_TAG=v1.8.2 IMAGE_TAG=1.8.4
PROJECT_NAME=at-mintel PROJECT_NAME=at-mintel
PROJECT_COLOR=#82ed20 PROJECT_COLOR=#82ed20
GITEA_TOKEN=ccce002e30fe16a31a6c9d5a414740af2f72a582 GITEA_TOKEN=ccce002e30fe16a31a6c9d5a414740af2f72a582

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

View File

@@ -1,8 +1,16 @@
<template> <template>
<private-view title="Acquisition Manager"> <MintelManagerLayout
title="Acquisition Manager"
:item-title="getCompanyName(selectedLead) || 'Lead wählen'"
:is-empty="!selectedLead"
empty-title="Lead auswählen"
empty-icon="auto_awesome"
:notice="notice"
@close-notice="notice = null"
>
<template #navigation> <template #navigation>
<v-list nav> <v-list nav>
<v-list-item @click="showAddLead = true" clickable> <v-list-item @click="openCreateDrawer" clickable>
<v-list-item-icon><v-icon name="add" color="var(--theme--primary)" /></v-list-item-icon> <v-list-item-icon><v-icon name="add" color="var(--theme--primary)" /></v-list-item-icon>
<v-list-item-content> <v-list-item-content>
<v-text-overflow text="Neuen Lead anlegen" /> <v-text-overflow text="Neuen Lead anlegen" />
@@ -15,7 +23,7 @@
v-for="lead in leads" v-for="lead in leads"
:key="lead.id" :key="lead.id"
:active="selectedLeadId === lead.id" :active="selectedLeadId === lead.id"
class="lead-item" class="nav-item"
clickable clickable
@click="selectLead(lead.id)" @click="selectLead(lead.id)"
> >
@@ -29,131 +37,118 @@
</v-list> </v-list>
</template> </template>
<template #title-outer:after> <template #subtitle>
<v-notice v-if="notice" :type="notice.type" @close="notice = null" dismissible> <template v-if="selectedLead">
{{ notice.message }} <v-icon name="language" x-small />
</v-notice> <a :href="selectedLead.website_url" target="_blank" class="url-link">
{{ selectedLead.website_url.replace(/^https?:\/\//, '') }}
</a>
&middot; Status: {{ selectedLead.status.toUpperCase() }}
</template>
</template> </template>
<div class="content-wrapper"> <template #actions>
<v-button
v-if="selectedLead?.status === 'new'"
secondary
:loading="loadingAudit"
@click="runAudit"
>
<v-icon name="settings_suggest" left />
Audit starten
</v-button>
<div v-if="!selectedLead" class="empty-state"> <template v-if="selectedLead?.status === 'audit_ready'">
<v-info title="Lead auswählen" icon="auto_awesome" center> <v-button secondary :loading="loadingEmail" @click="sendAuditEmail">
Wähle einen Lead in der Navigation aus oder <v-icon name="mail" left />
<v-button x-small @click="showAddLead = true">registriere einen neuen Lead</v-button>. Audit E-Mail
</v-info> </v-button>
</div> <v-button :loading="loadingPdf" @click="generatePdf">
<v-icon name="picture_as_pdf" left />
PDF Erstellen
</v-button>
</template>
<template v-else> <v-button v-if="selectedLead?.audit_pdf_path" secondary icon v-tooltip.bottom="'PDF öffnen'" @click="openPdf">
<header class="header"> <v-icon name="open_in_new" />
<div class="header-left"> </v-button>
<h1 class="title">{{ getCompanyName(selectedLead) }}</h1>
<p class="subtitle"> <v-button
<v-icon name="language" x-small /> v-if="selectedLead?.audit_pdf_path"
<a :href="selectedLead.website_url" target="_blank" class="url-link"> primary
{{ selectedLead.website_url.replace(/^https?:\/\//, '') }} :loading="loadingEmail"
</a> @click="sendEstimateEmail"
&middot; Status: {{ selectedLead.status.toUpperCase() }} >
</p> <v-icon name="send" left />
</div> Angebot senden
<div class="header-right"> </v-button>
<v-button </template>
v-if="selectedLead.status === 'new'"
secondary
:loading="loadingAudit"
@click="runAudit"
>
<v-icon name="settings_suggest" left />
Audit starten
</v-button>
<template v-if="selectedLead.status === 'audit_ready'"> <template #empty-state>
<v-button secondary :loading="loadingEmail" @click="sendAuditEmail"> Wähle einen Lead in der Navigation aus oder
<v-icon name="mail" left /> <v-button x-small @click="openCreateDrawer">registriere einen neuen Lead</v-button>.
Audit E-Mail </template>
</v-button>
<v-button :loading="loadingPdf" @click="generatePdf">
<v-icon name="picture_as_pdf" left />
PDF Erstellen
</v-button>
</template>
<v-button v-if="selectedLead.audit_pdf_path" secondary icon v-tooltip.bottom="'PDF öffnen'" @click="openPdf"> <div v-if="selectedLead" class="sections">
<v-icon name="open_in_new" /> <div class="main-info">
</v-button> <div class="form-grid">
<div class="field">
<v-button <span class="label">Kontaktperson</span>
v-if="selectedLead.audit_pdf_path" <div v-if="selectedLead.contact_person" class="value person-link" @click="goToPerson(selectedLead.contact_person)">
primary {{ getPersonName(selectedLead.contact_person) }}
:loading="loadingEmail"
@click="sendEstimateEmail"
>
<v-icon name="send" left />
Angebot senden
</v-button>
</div>
</header>
<div class="sections">
<div class="main-info">
<div class="form-grid">
<div class="field">
<span class="label">Kontaktperson</span>
<div v-if="selectedLead.contact_person" class="value person-link" @click="goToPerson(selectedLead.contact_person)">
{{ getPersonName(selectedLead.contact_person) }}
</div>
<div v-else class="value text-subdued">Keine Person verknüpft</div>
</div>
<div class="field full">
<span class="label">Briefing / Fokus</span>
<div class="value text-block">{{ selectedLead.briefing || 'Kein Briefing hinterlegt.' }}</div>
</div>
</div> </div>
<div v-else class="value text-subdued">Keine Person verknüpft</div>
</div> </div>
<div class="field full">
<v-divider /> <span class="label">Briefing / Fokus</span>
<div class="value text-block">{{ selectedLead.briefing || 'Kein Briefing hinterlegt.' }}</div>
<div v-if="selectedLead.ai_state" class="ai-observations">
<h3 class="section-title">AI Observations & Estimation</h3>
<div class="metrics">
<v-info label="Projekt-Modus" :value="selectedLead.ai_state.projectType || 'Unbekannt'" />
<v-info label="Seitenanzahl" :value="selectedLead.ai_state.sitemap?.length || '0'" />
</div>
<v-table
v-if="selectedLead.ai_state.sitemap"
:headers="[ { text: 'Seite', value: 'title' }, { text: 'URL', value: 'url' } ]"
:items="selectedLead.ai_state.sitemap"
class="observation-table"
>
<template #[`item.title`]="{ item }">
<span class="page-title">{{ item.title }}</span>
</template>
<template #[`item.url`]="{ item }">
<span class="page-url">{{ item.url }}</span>
</template>
</v-table>
</div> </div>
</div> </div>
</template> </div>
<v-divider />
<div v-if="selectedLead.ai_state" class="ai-observations">
<h3 class="section-title">AI Observations & Estimation</h3>
<div class="metrics">
<MintelStatCard label="Projekt-Modus" :value="selectedLead.ai_state.projectType || 'Unbekannt'" icon="category" />
<MintelStatCard label="Seitenanzahl" :value="selectedLead.ai_state.sitemap?.length || '0'" icon="description" />
</div>
<v-table
v-if="selectedLead.ai_state.sitemap"
:headers="[ { text: 'Seite', value: 'title' }, { text: 'URL', value: 'url' } ]"
:items="selectedLead.ai_state.sitemap"
class="observation-table"
>
<template #[`item.title`]="{ item }">
<span class="page-title">{{ item.title }}</span>
</template>
<template #[`item.url`]="{ item }">
<span class="page-url">{{ item.url }}</span>
</template>
</v-table>
</div>
</div> </div>
<!-- Drawer: New Lead --> <!-- Drawer: New Lead -->
<v-drawer <v-drawer
v-model="showAddLead" v-model="drawerActive"
title="Neuen Lead registrieren" title="Neuen Lead registrieren"
icon="person_add" icon="person_add"
@cancel="showAddLead = false" @cancel="drawerActive = false"
> >
<div class="drawer-content"> <div class="drawer-content">
<div class="form-section"> <div class="form-section">
<div class="field"> <div class="field">
<span class="label">Organisation / Firma (Zentral)</span> <span class="label">Organisation / Firma (Zentral)</span>
<v-select <MintelSelect
v-model="newLead.company" v-model="newLead.company"
:items="companyOptions" :items="companyOptions"
placeholder="Bestehende Firma auswählen..." placeholder="Bestehende Firma auswählen..."
allow-add
@add="openQuickAdd('company')"
/> />
</div> </div>
<div class="field"> <div class="field">
@@ -166,10 +161,12 @@
</div> </div>
<div class="field"> <div class="field">
<span class="label">Kontaktperson (Optional)</span> <span class="label">Kontaktperson (Optional)</span>
<v-select <MintelSelect
v-model="newLead.contact_person" v-model="newLead.contact_person"
:items="peopleOptions" :items="peopleOptions"
placeholder="Person auswählen..." placeholder="Person auswählen..."
allow-add
@add="openQuickAdd('person')"
/> />
</div> </div>
</div> </div>
@@ -179,12 +176,13 @@
</div> </div>
</div> </div>
</v-drawer> </v-drawer>
</private-view> </MintelManagerLayout>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted, computed } from 'vue'; import { ref, onMounted, computed } from 'vue';
import { useApi } from '@directus/extensions-sdk'; import { useApi } from '@directus/extensions-sdk';
import { MintelManagerLayout, MintelSelect, MintelStatCard } from '@mintel/directus-extension-toolkit';
const api = useApi(); const api = useApi();
const leads = ref<any[]>([]); const leads = ref<any[]>([]);
@@ -192,7 +190,7 @@ const selectedLeadId = ref<string | null>(null);
const loadingAudit = ref(false); const loadingAudit = ref(false);
const loadingPdf = ref(false); const loadingPdf = ref(false);
const loadingEmail = ref(false); const loadingEmail = ref(false);
const showAddLead = ref(false); const drawerActive = ref(false);
const savingLead = ref(false); const savingLead = ref(false);
const notice = ref<{ type: string; message: string } | null>(null); const notice = ref<{ type: string; message: string } | null>(null);
@@ -222,10 +220,11 @@ const peopleOptions = computed(() =>
); );
function getCompanyName(lead: any) { function getCompanyName(lead: any) {
if (!lead) return '';
if (lead.company) { if (lead.company) {
return typeof lead.company === 'object' ? lead.company.name : (companies.value.find(c => c.id === lead.company)?.name || lead.company_name); return typeof lead.company === 'object' ? lead.company.name : (companies.value.find(c => c.id === lead.company)?.name || 'Unbekannte Firma');
} }
return lead.company_name; return 'Unbekannte Organisation';
} }
function getPersonName(id: string | any) { function getPersonName(id: string | any) {
@@ -236,31 +235,32 @@ function getPersonName(id: string | any) {
} }
function goToPerson(id: string) { function goToPerson(id: string) {
// Logic to navigate to people manager or open details
notice.value = { type: 'info', message: `Navigiere zu Person: ${id}` }; notice.value = { type: 'info', message: `Navigiere zu Person: ${id}` };
} }
const selectedLead = computed(() => leads.value.find(l => l.id === selectedLeadId.value)); const selectedLead = computed(() => leads.value.find(l => l.id === selectedLeadId.value));
onMounted(fetchData);
async function fetchData() { async function fetchData() {
const [leadsResp, peopleResp, companiesResp] = await Promise.all([ try {
api.get('/items/leads', { const [leadsResp, peopleResp, companiesResp] = await Promise.all([
params: { api.get('/items/leads', {
sort: '-date_created', params: {
fields: '*.*' sort: '-date_created',
} fields: '*.*'
}), }
api.get('/items/people', { params: { sort: 'last_name' } }), }),
api.get('/items/companies', { params: { sort: 'name' } }) api.get('/items/people', { params: { sort: 'last_name' } }),
]); api.get('/items/companies', { params: { sort: 'name' } })
leads.value = leadsResp.data.data; ]);
people.value = peopleResp.data.data; leads.value = leadsResp.data.data;
companies.value = companiesResp.data.data; people.value = peopleResp.data.data;
companies.value = companiesResp.data.data;
if (!selectedLeadId.value && leads.value.length > 0) {
selectedLeadId.value = leads.value[0].id; if (!selectedLeadId.value && leads.value.length > 0) {
selectedLeadId.value = leads.value[0].id;
}
} catch (e: any) {
console.error('Fetch error:', e);
} }
} }
@@ -272,6 +272,17 @@ function selectLead(id: string) {
selectedLeadId.value = id; selectedLeadId.value = id;
} }
function openCreateDrawer() {
newLead.value = {
company: null,
website_url: '',
contact_person: null,
briefing: '',
status: 'new'
};
drawerActive.value = true;
}
async function runAudit() { async function runAudit() {
if (!selectedLeadId.value) return; if (!selectedLeadId.value) return;
loadingAudit.value = true; loadingAudit.value = true;
@@ -346,16 +357,9 @@ async function saveLead() {
}; };
await api.post('/items/leads', payload); await api.post('/items/leads', payload);
notice.value = { type: 'success', message: 'Lead erfolgreich registriert!' }; notice.value = { type: 'success', message: 'Lead erfolgreich registriert!' };
showAddLead.value = false; drawerActive.value = false;
await fetchLeads(); await fetchLeads();
selectedLeadId.value = payload.id; selectedLeadId.value = payload.id;
newLead.value = {
company: null,
website_url: '',
contact_person: null,
briefing: '',
status: 'new'
};
} catch (e: any) { } catch (e: any) {
notice.value = { type: 'danger', message: `Fehler beim Speichern: ${e.message}` }; notice.value = { type: 'danger', message: `Fehler beim Speichern: ${e.message}` };
} finally { } finally {
@@ -363,6 +367,10 @@ async function saveLead() {
} }
} }
function openQuickAdd(type: string) {
notice.value = { type: 'info', message: `${type === 'company' ? 'Firma' : 'Person'} im jeweiligen Manager anlegen.` };
}
function getStatusIcon(status: string) { function getStatusIcon(status: string) {
switch(status) { switch(status) {
case 'new': return 'fiber_new'; case 'new': return 'fiber_new';
@@ -382,18 +390,13 @@ function getStatusColor(status: string) {
default: return 'var(--theme--foreground-subdued)'; default: return 'var(--theme--foreground-subdued)';
} }
} }
onMounted(fetchData);
</script> </script>
<style scoped> <style scoped>
.content-wrapper { padding: 32px; height: 100%; display: flex; flex-direction: column; overflow-y: auto; }
.lead-item { cursor: pointer; }
.header { margin-bottom: 24px; display: flex; justify-content: space-between; align-items: flex-end; }
.header-right { display: flex; gap: 12px; }
.title { font-size: 24px; font-weight: 800; margin-bottom: 4px; color: var(--theme--foreground); }
.subtitle { color: var(--theme--foreground-subdued); font-size: 14px; display: flex; align-items: center; gap: 8px; }
.url-link { color: inherit; text-decoration: none; border-bottom: 1px solid transparent; } .url-link { color: inherit; text-decoration: none; border-bottom: 1px solid transparent; }
.url-link:hover { border-bottom-color: currentColor; } .url-link:hover { border-bottom-color: currentColor; }
.empty-state { height: 100%; display: flex; align-items: center; justify-content: center; }
.sections { display: flex; flex-direction: column; gap: 32px; } .sections { display: flex; flex-direction: column; gap: 32px; }
@@ -406,7 +409,7 @@ function getStatusColor(status: string) {
.ai-observations { display: flex; flex-direction: column; gap: 16px; } .ai-observations { display: flex; flex-direction: column; gap: 16px; }
.section-title { font-size: 16px; font-weight: 700; color: var(--theme--foreground); margin-bottom: 8px; } .section-title { font-size: 16px; font-weight: 700; color: var(--theme--foreground); margin-bottom: 8px; }
.metrics { display: flex; gap: 32px; margin-bottom: 16px; } .metrics { display: flex; gap: 24px; margin-bottom: 16px; }
.observation-table { border: 1px solid var(--theme--border); border-radius: 8px; overflow: hidden; } .observation-table { border: 1px solid var(--theme--border); border-radius: 8px; overflow: hidden; }
.page-title { font-weight: 600; } .page-title { font-weight: 600; }
@@ -415,6 +418,4 @@ function getStatusColor(status: string) {
.drawer-content { padding: 24px; display: flex; flex-direction: column; gap: 32px; } .drawer-content { padding: 24px; display: flex; flex-direction: column; gap: 32px; }
.form-section { display: flex; flex-direction: column; gap: 20px; } .form-section { display: flex; flex-direction: column; gap: 20px; }
.drawer-actions { margin-top: 24px; display: flex; flex-direction: column; gap: 12px; } .drawer-actions { margin-top: 24px; display: flex; flex-direction: column; gap: 12px; }
:deep(.v-list-item) { cursor: pointer !important; }
</style> </style>

Binary file not shown.

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

File diff suppressed because one or more lines are too long

View File

@@ -22,6 +22,7 @@
}, },
"devDependencies": { "devDependencies": {
"@directus/extensions-sdk": "11.0.2", "@directus/extensions-sdk": "11.0.2",
"@mintel/directus-extension-toolkit": "workspace:*",
"vue": "^3.4.0" "vue": "^3.4.0"
} }
} }

View File

@@ -146,6 +146,30 @@ collections:
versioning: false versioning: false
schema: schema:
name: visual_feedback_comments name: visual_feedback_comments
- collection: customers
meta:
accountability: all
archive_app_filter: true
archive_field: null
archive_value: null
collapse: open
collection: customers
color: null
display_template: '{{company.name}}'
group: null
hidden: false
icon: handshake
item_duplication_fields: null
note: null
preview_url: null
singleton: false
sort: null
sort_field: null
translations: null
unarchive_value: null
versioning: false
schema:
name: customers
fields: fields:
- collection: client_users - collection: client_users
field: id field: id
@@ -1959,13 +1983,204 @@ fields:
validation_message: null validation_message: null
width: half width: half
schema: schema:
name: person foreign_key_column: null
table: visual_feedback_comments - collection: customers
field: id
type: uuid
meta:
collection: customers
conditions: null
display: null
display_options: null
field: id
group: null
hidden: true
interface: null
note: null
options: null
readonly: false
required: false
searchable: true
sort: 1
special:
- uuid
translations: null
validation: null
validation_message: null
width: full
schema:
name: id
table: customers
data_type: char data_type: char
default_value: null default_value: null
max_length: 36 max_length: 36
numeric_precision: null numeric_precision: null
numeric_scale: null numeric_scale: null
is_nullable: false
is_unique: true
is_indexed: false
is_primary_key: true
is_generated: false
generation_expression: null
has_auto_increment: false
foreign_key_table: null
foreign_key_column: null
- collection: customers
field: company
type: uuid
meta:
collection: customers
conditions: null
display: null
display_options: null
field: company
group: null
hidden: false
interface: select-dropdown-m2o
note: null
options: null
readonly: false
required: true
searchable: true
sort: 2
special: null
translations: null
validation: null
validation_message: null
width: half
schema:
name: company
table: customers
data_type: char
default_value: null
max_length: 36
numeric_precision: null
numeric_scale: null
is_nullable: true
is_unique: false
is_indexed: false
is_primary_key: false
is_generated: false
generation_expression: null
has_auto_increment: false
foreign_key_table: companies
foreign_key_column: id
- collection: customers
field: contact_person
type: uuid
meta:
collection: customers
conditions: null
display: null
display_options: null
field: contact_person
group: null
hidden: false
interface: select-dropdown-m2o
note: null
options: null
readonly: false
required: false
searchable: true
sort: 3
special: null
translations: null
validation: null
validation_message: null
width: half
schema:
name: contact_person
table: customers
data_type: char
default_value: null
max_length: 36
numeric_precision: null
numeric_scale: null
is_nullable: true
is_unique: false
is_indexed: false
is_primary_key: false
is_generated: false
generation_expression: null
has_auto_increment: false
foreign_key_table: people
foreign_key_column: id
- collection: customers
field: status
type: string
meta:
collection: customers
conditions: null
display: null
display_options: null
field: status
group: null
hidden: false
interface: select-dropdown
note: null
options:
choices:
- text: Active
value: active
- text: Inactive
value: inactive
readonly: false
required: false
searchable: true
sort: 4
special: null
translations: null
validation: null
validation_message: null
width: half
schema:
name: status
table: customers
data_type: varchar
default_value: active
max_length: 255
numeric_precision: null
numeric_scale: null
is_nullable: true
is_unique: false
is_indexed: false
is_primary_key: false
is_generated: false
generation_expression: null
has_auto_increment: false
foreign_key_table: null
foreign_key_column: null
- collection: customers
field: notes
type: text
meta:
collection: customers
conditions: null
display: null
display_options: null
field: notes
group: null
hidden: false
interface: input-multiline
note: null
options: null
readonly: false
required: false
searchable: true
sort: 5
special: null
translations: null
validation: null
validation_message: null
width: full
schema:
name: notes
table: customers
data_type: text
default_value: null
max_length: null
numeric_precision: null
numeric_scale: null
is_nullable: true is_nullable: true
is_unique: false is_unique: false
is_indexed: false is_indexed: false
@@ -1989,6 +2204,28 @@ systemFields:
schema: schema:
is_indexed: true is_indexed: true
relations: relations:
- collection: customers
field: company
related_collection: companies
schema:
on_update: null
on_delete: SET NULL
constraint_name: customers_company_foreign
table: customers
column: company
foreign_key_table: companies
foreign_key_column: id
- collection: customers
field: contact_person
related_collection: people
schema:
on_update: null
on_delete: SET NULL
constraint_name: customers_contact_person_foreign
table: customers
column: contact_person
foreign_key_table: people
foreign_key_column: id
- collection: client_users - collection: client_users
field: company field: company
related_collection: companies related_collection: companies

View File

@@ -1 +1 @@
xEDE- --tVj

View File

@@ -1,5 +1,13 @@
<template> <template>
<private-view title="Company Manager"> <MintelManagerLayout
title="Company Manager"
:item-title="selectedCompany?.name || 'Firma wählen'"
:is-empty="!selectedCompany"
empty-title="Firma auswählen"
empty-icon="business"
:notice="feedback"
@close-notice="feedback = null"
>
<template #navigation> <template #navigation>
<v-list nav> <v-list nav>
<v-list-item @click="openCreateDrawer" clickable> <v-list-item @click="openCreateDrawer" clickable>
@@ -17,7 +25,7 @@
v-for="company in companies" v-for="company in companies"
:key="company.id" :key="company.id"
:active="selectedCompany?.id === company.id" :active="selectedCompany?.id === company.id"
class="company-item" class="nav-item"
clickable clickable
@click="selectCompany(company)" @click="selectCompany(company)"
> >
@@ -31,41 +39,30 @@
</v-list> </v-list>
</template> </template>
<div class="content-wrapper"> <template #subtitle>
<v-notice v-if="feedback" :type="feedback.type" @close="feedback = null" dismissible> <template v-if="selectedCompany">
{{ feedback.message }} {{ selectedCompany.domain || 'Keine Domain angegeben' }}
</v-notice> </template>
</template>
<div v-if="!selectedCompany" class="empty-state"> <template #actions>
<v-info title="Firma auswählen" icon="business" center> <v-button secondary rounded icon v-tooltip.bottom="'Firma bearbeiten'" @click="openEditDrawer">
Wähle eine Firma in der Navigation aus oder <v-icon name="edit" />
<v-button x-small @click="openCreateDrawer">erstelle eine neue Firma</v-button>. </v-button>
</v-info> <v-button danger rounded icon v-tooltip.bottom="'Firma löschen'" @click="deleteCompany">
</div> <v-icon name="delete" />
</v-button>
</template>
<div v-else> <template #empty-state>
<header class="header"> Wähle eine Firma in der Navigation aus oder
<div class="header-left"> <v-button x-small @click="openCreateDrawer">erstelle eine neue Firma</v-button>.
<h1 class="title">{{ selectedCompany.name }}</h1> </template>
<p class="subtitle">{{ selectedCompany.domain || 'Keine Domain angegeben' }}</p>
</div>
<div class="header-right"> <div v-if="selectedCompany" class="details-grid">
<v-button secondary rounded icon v-tooltip="'Firma bearbeiten'" @click="openEditDrawer"> <div class="detail-item full">
<v-icon name="edit" /> <span class="label">Notizen / Adresse</span>
</v-button> <p class="value">{{ selectedCompany.notes || '---' }}</p>
<v-button danger rounded icon v-tooltip="'Firma löschen'" @click="deleteCompany">
<v-icon name="delete" />
</v-button>
</div>
</header>
<v-divider />
<div class="detail-item full">
<span class="label">Notizen / Adresse</span>
<p class="value">{{ selectedCompany.notes || '---' }}</p>
</div>
</div> </div>
</div> </div>
@@ -101,12 +98,13 @@
</div> </div>
</template> </template>
</v-drawer> </v-drawer>
</private-view> </MintelManagerLayout>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted } from 'vue'; import { ref, onMounted } from 'vue';
import { useApi } from '@directus/extensions-sdk'; import { useApi } from '@directus/extensions-sdk';
import { MintelManagerLayout } from '@mintel/directus-extension-toolkit';
const api = useApi(); const api = useApi();
const companies = ref([]); const companies = ref([]);
@@ -144,14 +142,19 @@ function openCreateDrawer() {
id: null, id: null,
name: '', name: '',
domain: '', domain: '',
notes: '' notes: ''
}; };
drawerActive.value = true; drawerActive.value = true;
} }
function openEditDrawer() { function openEditDrawer() {
isEditing.value = true; isEditing.value = true;
form.value = { ...selectedCompany.value }; form.value = {
id: selectedCompany.value.id,
name: selectedCompany.value.name,
domain: selectedCompany.value.domain,
notes: selectedCompany.value.notes
};
drawerActive.value = true; drawerActive.value = true;
} }
@@ -163,17 +166,20 @@ async function saveCompany() {
saving.value = true; saving.value = true;
try { try {
let updatedItem;
if (isEditing.value) { if (isEditing.value) {
await api.patch(`/items/companies/${form.value.id}`, form.value); const res = await api.patch(`/items/companies/${form.value.id}`, form.value);
updatedItem = res.data.data;
feedback.value = { type: 'success', message: 'Firma aktualisiert!' }; feedback.value = { type: 'success', message: 'Firma aktualisiert!' };
} else { } else {
await api.post('/items/companies', form.value); const res = await api.post('/items/companies', form.value);
updatedItem = res.data.data;
feedback.value = { type: 'success', message: 'Firma angelegt!' }; feedback.value = { type: 'success', message: 'Firma angelegt!' };
} }
drawerActive.value = false; drawerActive.value = false;
await fetchData(); await fetchData();
if (isEditing.value) { if (updatedItem) {
selectedCompany.value = companies.value.find(c => c.id === form.value.id); selectedCompany.value = companies.value.find(c => c.id === updatedItem.id) || updatedItem;
} }
} catch (error) { } catch (error) {
feedback.value = { type: 'danger', message: error.message }; feedback.value = { type: 'danger', message: error.message };
@@ -199,13 +205,7 @@ onMounted(fetchData);
</script> </script>
<style scoped> <style scoped>
.content-wrapper { padding: 32px; height: 100%; } .details-grid { display: flex; flex-direction: column; gap: 24px; }
.header { margin-bottom: 24px; display: flex; justify-content: space-between; align-items: flex-end; }
.title { font-size: 24px; font-weight: 800; margin-bottom: 4px; }
.subtitle { color: var(--theme--foreground-subdued); font-size: 14px; }
.header-right { display: flex; gap: 12px; }
.empty-state { height: 100%; display: flex; align-items: center; justify-content: center; }
.details-grid { display: flex; flex-direction: column; gap: 24px; margin-top: 32px; }
.detail-item { display: flex; flex-direction: column; gap: 8px; } .detail-item { display: flex; flex-direction: column; gap: 8px; }
.detail-item.full { width: 100%; } .detail-item.full { width: 100%; }
.label { font-size: 12px; font-weight: 700; text-transform: uppercase; color: var(--theme--foreground-subdued); letter-spacing: 0.5px; } .label { font-size: 12px; font-weight: 700; text-transform: uppercase; color: var(--theme--foreground-subdued); letter-spacing: 0.5px; }

View File

@@ -1,161 +1,191 @@
<template> <template>
<private-view title="Customer Manager"> <MintelManagerLayout
title="Customer Manager"
:item-title="selectedItem?.company?.name || 'Kunde wählen'"
:is-empty="!selectedItem"
empty-title="Kunde auswählen"
empty-icon="handshake"
:notice="notice"
@close-notice="notice = null"
>
<template #navigation> <template #navigation>
<v-list nav> <v-list nav>
<v-list-item @click="openCreateCompany" clickable> <v-list-item @click="openCreateDrawer" clickable>
<v-list-item-icon><v-icon name="add" color="var(--theme--primary)" /></v-list-item-icon> <v-list-item-icon><v-icon name="add" color="var(--theme--primary)" /></v-list-item-icon>
<v-list-item-content> <v-list-item-content>
<v-text-overflow text="Neue Firma anlegen" /> <v-text-overflow text="Neuen Kunden verlinken" />
</v-list-item-content> </v-list-item-content>
</v-list-item> </v-list-item>
<v-divider /> <v-divider />
<v-list-item <v-list-item
v-for="company in companies" v-for="item in items"
:key="company.id" :key="item.id"
:active="selectedCompany?.id === company.id" :active="selectedItem?.id === item.id"
class="company-item" class="nav-item"
clickable clickable
@click="selectCompany(company)" @click="selectItem(item)"
> >
<v-list-item-icon><v-icon name="business" /></v-list-item-icon> <v-list-item-icon><v-icon name="business" /></v-list-item-icon>
<v-list-item-content> <v-list-item-content>
<v-text-overflow :text="company.name" /> <v-text-overflow :text="item.company?.name" />
</v-list-item-content> </v-list-item-content>
</v-list-item> </v-list-item>
</v-list> </v-list>
</template> </template>
<template #title-outer:after> <template #subtitle>
<v-notice v-if="notice" :type="notice.type" @close="notice = null" dismissible> <template v-if="selectedItem">
{{ notice.message }} {{ clientUsers.length }} Portal-Nutzer &middot; {{ selectedItem.company?.domain }}
</v-notice> </template>
</template> </template>
<div class="content-wrapper"> <template #actions>
<div v-if="!selectedCompany" class="empty-state"> <v-button secondary rounded icon v-tooltip.bottom="'Kunden-Verlinkung bearbeiten'" @click="openEditDrawer">
<v-info title="Firmen auswählen" icon="business" center> <v-icon name="edit" />
Wähle eine Firma in der Navigation aus oder </v-button>
<v-button x-small @click="openCreateCompany">erstelle eine neue Firma</v-button>. <v-button primary @click="openCreateClientUser">
</v-info> Portal-Nutzer hinzufügen
</div> </v-button>
</template>
<template v-else> <template #empty-state>
<header class="header"> Wähle einen Kunden aus der Liste oder
<div class="header-left"> <v-button x-small @click="openCreateDrawer">verlinke eine neue Firma</v-button>.
<h1 class="title">{{ selectedCompany.name }}</h1> </template>
<p class="subtitle">{{ employees.length }} Kunden-Mitarbeiter</p>
</div>
<div class="header-right">
<v-button secondary rounded icon v-tooltip.bottom="'Firma bearbeiten'" @click="openEditCompany">
<v-icon name="edit" />
</v-button>
<v-button primary @click="openCreateEmployee">
Mitarbeiter hinzufügen
</v-button>
</div>
</header>
<v-table <!-- Main Content: Client Users Table -->
:headers="tableHeaders" <v-table
:items="employees" :headers="tableHeaders"
:loading="loading" :items="clientUsers"
class="clickable-table" :loading="loading"
fixed-header class="clickable-table"
@click:row="onRowClick" fixed-header
> @click:row="onRowClick"
<template #[`item.name`]="{ item }">
<div class="user-cell">
<v-avatar :name="item.first_name" x-small />
<span class="user-name">{{ item.first_name }} {{ item.last_name }}</span>
</div>
</template>
<template #[`item.last_invited`]="{ item }">
<span v-if="item.last_invited" class="status-date">
{{ formatDate(item.last_invited) }}
</span>
<v-chip v-else x-small>Noch nie</v-chip>
</template>
</v-table>
</template>
</div>
<!-- Drawer: Company Form -->
<v-drawer
v-model="drawerCompanyActive"
:title="isEditingCompany ? 'Firma bearbeiten' : 'Neue Firma anlegen'"
icon="business"
@cancel="drawerCompanyActive = false"
> >
<div v-if="drawerCompanyActive" class="drawer-content"> <template #[`item.name`]="{ item }">
<div class="form-section"> <div class="user-cell">
<div class="field"> <v-avatar :name="item.first_name" x-small />
<span class="label">Firmenname</span> <span class="user-name">{{ item.first_name }} {{ item.last_name }}</span>
<v-input v-model="companyForm.name" placeholder="z.B. KLZ Cables" autofocus /> </div>
</div> </template>
</div>
<template #[`item.last_invited`]="{ item }">
<span v-if="item.last_invited" class="status-date">
{{ formatDate(item.last_invited) }}
</span>
<v-chip v-else x-small>Noch nie</v-chip>
</template>
</v-table>
<!-- Drawer: Customer (Link) Form -->
<v-drawer
v-model="drawerActive"
:title="isEditing ? 'Kunden-Verlinkung bearbeiten' : 'Kunden verlinken'"
icon="handshake"
@cancel="drawerActive = false"
>
<div v-if="drawerActive" class="drawer-content">
<div class="form-section">
<div class="field">
<span class="label">Organisation / Firma</span>
<MintelSelect
v-model="form.company"
:items="companyOptions"
placeholder="Firma auswählen..."
allow-add
@add="openQuickAdd('company')"
/>
</div>
<div class="field">
<span class="label">Haupt-Ansprechpartner (optional)</span>
<MintelSelect
v-model="form.contact_person"
:items="peopleOptions"
placeholder="Person auswählen..."
allow-add
@add="openQuickAdd('person')"
/>
</div>
<div class="field">
<span class="label">Status</span>
<v-select
v-model="form.status"
:items="[
{ text: 'Aktiv', value: 'active' },
{ text: 'Inaktiv', value: 'inactive' }
]"
/>
</div>
<div class="field">
<span class="label">Notizen</span>
<v-textarea v-model="form.notes" placeholder="Besonderheiten zu diesem Kunden..." />
</div>
</div>
<div class="drawer-actions"> <div class="drawer-actions">
<v-button primary block :loading="saving" @click="saveCompany">Speichern</v-button> <v-button primary block :loading="saving" @click="saveItem">Speichern</v-button>
</div> </div>
</div> </div>
</v-drawer> </v-drawer>
<!-- Drawer: Employee Form --> <!-- Drawer: Client User Form -->
<v-drawer <v-drawer
v-model="drawerEmployeeActive" v-model="drawerUserActive"
:title="isEditingEmployee ? 'Mitarbeiter bearbeiten' : 'Neuen Mitarbeiter anlegen'" :title="isEditingUser ? 'Portal-Nutzer bearbeiten' : 'Neuen Portal-Nutzer anlegen'"
icon="person" icon="person"
@cancel="drawerEmployeeActive = false" @cancel="drawerUserActive = false"
> >
<div v-if="drawerEmployeeActive" class="drawer-content"> <div v-if="drawerUserActive" class="drawer-content">
<div class="form-section"> <div class="form-section">
<div class="field"> <div class="field">
<span class="label">Vorname</span> <span class="label">Vorname</span>
<v-input v-model="employeeForm.first_name" placeholder="Vorname" autofocus /> <v-input v-model="userForm.first_name" placeholder="Vorname" autofocus />
</div> </div>
<div class="field"> <div class="field">
<span class="label">Nachname</span> <span class="label">Nachname</span>
<v-input v-model="employeeForm.last_name" placeholder="Nachname" /> <v-input v-model="userForm.last_name" placeholder="Nachname" />
</div> </div>
<div class="field"> <div class="field">
<span class="label">E-Mail</span> <span class="label">E-Mail</span>
<v-input v-model="employeeForm.email" placeholder="E-Mail Adresse" type="email" /> <v-input v-model="userForm.email" placeholder="E-Mail Adresse" type="email" />
</div> </div>
<div class="field"> <div class="field">
<span class="label">Zentrale Person (Verknüpfung)</span> <span class="label">Zentrale Person (Verknüpfung)</span>
<v-select <v-select
v-model="employeeForm.person" v-model="userForm.person"
:items="peopleOptions" :items="peopleOptions"
placeholder="Person aus dem People Manager auswählen..." placeholder="Master-Person auswählen..."
/> />
</div> </div>
<v-divider v-if="isEditingEmployee" /> <v-divider v-if="isEditingUser" />
<div v-if="isEditingEmployee" class="field"> <div v-if="isEditingUser" class="field">
<span class="label">Temporäres Passwort</span> <span class="label">Temporäres Passwort</span>
<v-input v-model="employeeForm.temporary_password" readonly class="password-input" /> <v-input v-model="userForm.temporary_password" readonly class="password-input" />
<p class="field-note">Wird beim Senden der Zugangsdaten automatisch generiert.</p> <p class="field-note">Wird beim Senden der Zugangsdaten automatisch generiert.</p>
</div> </div>
</div> </div>
<div class="drawer-actions"> <div class="drawer-actions">
<v-button primary block :loading="saving" @click="saveEmployee">Daten speichern</v-button> <v-button primary block :loading="saving" @click="saveClientUser">Daten speichern</v-button>
<template v-if="isEditingEmployee"> <template v-if="isEditingUser">
<v-divider /> <v-divider />
<v-button <v-button
v-tooltip.bottom="'Generiert PW, speichert es und sendet E-Mail'" v-tooltip.bottom="'Generiert PW, speichert es und sendet E-Mail'"
secondary secondary
block block
:loading="invitingId === employeeForm.id" :loading="invitingId === userForm.id"
@click="inviteUser(employeeForm)" @click="inviteUser(userForm)"
> >
<v-icon name="send" left /> Zugangsdaten senden <v-icon name="send" left /> Zugangsdaten senden
</v-button> </v-button>
@@ -163,38 +193,34 @@
</div> </div>
</div> </div>
</v-drawer> </v-drawer>
</private-view> </MintelManagerLayout>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted, nextTick, computed } from 'vue'; import { ref, onMounted, nextTick, computed } from 'vue';
import { useApi } from '@directus/extensions-sdk'; import { useApi } from '@directus/extensions-sdk';
import { MintelManagerLayout, MintelSelect } from '@mintel/directus-extension-toolkit';
const api = useApi(); const api = useApi();
const companies = ref<any[]>([]); const items = ref<any[]>([]);
const selectedCompany = ref<any>(null); const selectedItem = ref<any>(null);
const employees = ref<any[]>([]); const clientUsers = ref<any[]>([]);
const loading = ref(false); const loading = ref(false);
const saving = ref(false); const saving = ref(false);
const invitingId = ref<string | null>(null); const invitingId = ref<string | null>(null);
const notice = ref<{ type: string; message: string } | null>(null); const notice = ref<{ type: string; message: string } | null>(null);
// Forms State const companies = ref<any[]>([]);
const drawerCompanyActive = ref(false); const people = ref<any[]>([]);
const isEditingCompany = ref(false);
const companyForm = ref({ id: '', name: '' });
const drawerEmployeeActive = ref(false); const drawerActive = ref(false);
const isEditingEmployee = ref(false); const isEditing = ref(false);
const employeeForm = ref({ const form = ref({ id: null, company: null, contact_person: null, status: 'active', notes: '' });
id: '',
first_name: '', const drawerUserActive = ref(false);
last_name: '', const isEditingUser = ref(false);
email: '', const userForm = ref({ id: '', first_name: '', last_name: '', email: '', person: null, temporary_password: '' });
person: null,
temporary_password: ''
});
const tableHeaders = [ const tableHeaders = [
{ text: 'Name', value: 'name', sortable: true }, { text: 'Name', value: 'name', sortable: true },
@@ -202,180 +228,159 @@ const tableHeaders = [
{ text: 'Zuletzt eingeladen', value: 'last_invited', sortable: true } { text: 'Zuletzt eingeladen', value: 'last_invited', sortable: true }
]; ];
const people = ref<any[]>([]); const companyOptions = computed(() => companies.value.map(c => ({ text: c.name, value: c.id })));
const peopleOptions = computed(() => people.value.map(p => ({ text: `${p.first_name} ${p.last_name} (${p.email})`, value: p.id })));
const peopleOptions = computed(() =>
people.value.map(p => ({
text: `${p.first_name} ${p.last_name} (${p.email})`,
value: p.id
}))
);
async function fetchData() { async function fetchData() {
const [companiesResp, peopleResp] = await Promise.all([ loading.value = true;
api.get('/items/companies', { params: { sort: 'name', fields: ['id', 'name'] } }), try {
api.get('/items/people', { params: { sort: 'last_name' } }) const [custResp, compResp, peopleResp] = await Promise.all([
]); api.get('/items/customers', { params: { fields: ['*', 'company.*', 'contact_person.*'], sort: 'company.name' } }),
companies.value = companiesResp.data.data; api.get('/items/companies', { params: { sort: 'name' } }),
people.value = peopleResp.data.data; api.get('/items/people', { params: { sort: 'last_name' } })
]);
items.value = custResp.data.data;
companies.value = compResp.data.data;
people.value = peopleResp.data.data;
} finally {
loading.value = false;
}
} }
async function selectCompany(company: any) { async function selectItem(item: any) {
selectedCompany.value = company; selectedItem.value = item;
loading.value = true; loading.value = true;
try { try {
const res = await api.get('/items/client_users', { const res = await api.get('/items/client_users', {
params: { params: {
filter: { company: { _eq: company.id } }, filter: { company: { _eq: item.company.id } },
fields: ['*', 'person.*'], fields: ['*', 'person.*'],
sort: 'first_name', sort: 'first_name',
}, },
}); });
employees.value = res.data.data; clientUsers.value = res.data.data;
} finally { } finally {
loading.value = false; loading.value = false;
} }
} }
// Company Actions function openCreateDrawer() {
function openCreateCompany() { isEditing.value = false;
isEditingCompany.value = false; form.value = { id: null, company: null, contact_person: null, status: 'active', notes: '' };
companyForm.value = { id: '', name: '' }; drawerActive.value = true;
drawerCompanyActive.value = true;
} }
async function openEditCompany() { function openEditDrawer() {
if (!selectedCompany.value) return; if (!selectedItem.value) return;
companyForm.value = { isEditing.value = true;
id: selectedCompany.value.id, form.value = {
name: selectedCompany.value.name id: selectedItem.value.id,
company: selectedItem.value.company?.id || selectedItem.value.company,
contact_person: selectedItem.value.contact_person?.id || selectedItem.value.contact_person,
status: selectedItem.value.status,
notes: selectedItem.value.notes
}; };
isEditingCompany.value = true; drawerActive.value = true;
await nextTick();
drawerCompanyActive.value = true;
} }
async function saveCompany() { async function saveItem() {
if (!companyForm.value.name) return; if (!form.value.company) return;
saving.value = true; saving.value = true;
try { try {
if (isEditingCompany.value) { if (isEditing.value) {
await api.patch(`/items/companies/${companyForm.value.id}`, { name: companyForm.value.name }); await api.patch(`/items/customers/${form.value.id}`, form.value);
notice.value = { type: 'success', message: 'Firma aktualisiert!' }; notice.value = { type: 'success', message: 'Kunde aktualisiert!' };
} else { } else {
await api.post('/items/companies', { name: companyForm.value.name }); await api.post('/items/customers', form.value);
notice.value = { type: 'success', message: 'Firma angelegt!' }; notice.value = { type: 'success', message: 'Neuer Kunde verlinkt!' };
} }
drawerCompanyActive.value = false; drawerActive.value = false;
await fetchCompanies(); await fetchData();
if (selectedCompany.value?.id === companyForm.value.id) { if (form.value.id) {
selectedCompany.value.name = companyForm.value.name; const updated = items.value.find(i => i.id === form.value.id);
} if (updated) selectItem(updated);
} catch (e: any) { }
notice.value = { type: 'danger', message: e.message }; } catch (e: any) {
} finally { notice.value = { type: 'danger', message: e.message };
saving.value = false; } finally {
} saving.value = false;
}
} }
// Employee Actions // Client User Actions
function openCreateEmployee() { function openCreateClientUser() {
isEditingEmployee.value = false; isEditingUser.value = false;
employeeForm.value = { id: '', first_name: '', last_name: '', email: '', temporary_password: '' }; userForm.value = { id: '', first_name: '', last_name: '', email: '', person: null, temporary_password: '' };
drawerEmployeeActive.value = true; drawerUserActive.value = true;
}
async function openEditEmployee(item: any) {
employeeForm.value = {
id: item.id || '',
first_name: item.first_name || '',
last_name: item.last_name || '',
email: item.email || '',
person: item.person?.id || item.person || null,
temporary_password: item.temporary_password || ''
};
isEditingEmployee.value = true;
await nextTick();
drawerEmployeeActive.value = true;
}
async function saveEmployee() {
if (!employeeForm.value.email || !selectedCompany.value) return;
saving.value = true;
try {
if (isEditingEmployee.value) {
await api.patch(`/items/client_users/${employeeForm.value.id}`, {
first_name: employeeForm.value.first_name,
last_name: employeeForm.value.last_name,
email: employeeForm.value.email,
person: employeeForm.value.person
});
notice.value = { type: 'success', message: 'Mitarbeiter aktualisiert!' };
} else {
await api.post('/items/client_users', {
first_name: employeeForm.value.first_name,
last_name: employeeForm.value.last_name,
email: employeeForm.value.email,
company: selectedCompany.value.id,
person: employeeForm.value.person
});
notice.value = { type: 'success', message: 'Mitarbeiter angelegt!' };
}
drawerEmployeeActive.value = false;
await selectCompany(selectedCompany.value);
} catch (e: any) {
notice.value = { type: 'danger', message: e.message };
} finally {
saving.value = false;
}
}
async function inviteUser(user: any) {
invitingId.value = user.id;
try {
await api.post(`/flows/trigger/33443f6b-cec7-4668-9607-f33ea674d501`, [user.id]);
notice.value = { type: 'success', message: `Zugangsdaten für ${user.first_name} versendet. 📧` };
await selectCompany(selectedCompany.value);
if (drawerEmployeeActive.value && employeeForm.value.id === user.id) {
const updated = employees.value.find(e => e.id === user.id);
if (updated) {
employeeForm.value.temporary_password = updated.temporary_password;
}
}
} catch (e: any) {
notice.value = { type: 'danger', message: `Fehler: ${e.message}` };
} finally {
invitingId.value = null;
}
} }
function onRowClick(event: any) { function onRowClick(event: any) {
const item = event?.item || event; const item = event?.item || event;
if (item && item.id) { if (item && item.id) {
openEditEmployee(item); userForm.value = {
} id: item.id,
first_name: item.first_name,
last_name: item.last_name,
email: item.email,
person: item.person?.id || item.person,
temporary_password: item.temporary_password
};
isEditingUser.value = true;
drawerUserActive.value = true;
}
}
async function saveClientUser() {
if (!userForm.value.email || !selectedItem.value) return;
saving.value = true;
try {
const payload = {
first_name: userForm.value.first_name,
last_name: userForm.value.last_name,
email: userForm.value.email,
person: userForm.value.person,
company: selectedItem.value.company.id
};
if (isEditingUser.value) {
await api.patch(`/items/client_users/${userForm.value.id}`, payload);
} else {
await api.post('/items/client_users', payload);
}
drawerUserActive.value = false;
await selectItem(selectedItem.value);
} finally {
saving.value = false;
}
}
async function inviteUser(user: any) {
invitingId.value = user.id;
try {
await api.post(`/flows/trigger/33443f6b-cec7-4668-9607-f33ea674d501`, [user.id]);
notice.value = { type: 'success', message: `Zugangsdaten versendet. 📧` };
await selectItem(selectedItem.value);
} finally {
invitingId.value = null;
}
}
function openQuickAdd(type: 'company' | 'person') {
// Quick add logic can involve opening another drawer or navigating
// For now, we'll just show a notice
notice.value = { type: 'info', message: `${type === 'company' ? 'Firma' : 'Person'} im jeweiligen Manager anlegen.` };
} }
function formatDate(dateStr: string) { function formatDate(dateStr: string) {
return new Date(dateStr).toLocaleString('de-DE', { return new Date(dateStr).toLocaleString('de-DE', {
day: '2-digit', month: '2-digit', year: 'numeric', day: '2-digit', month: '2-digit', year: 'numeric',
hour: '2-digit', minute: '2-digit' hour: '2-digit', minute: '2-digit'
}); });
} }
onMounted(() => { onMounted(fetchData);
fetchData();
});
</script> </script>
<style scoped> <style scoped>
.content-wrapper { padding: 32px; height: 100%; display: flex; flex-direction: column; }
.company-item { cursor: pointer; }
.header { margin-bottom: 24px; display: flex; justify-content: space-between; align-items: flex-end; }
.header-right { display: flex; gap: 12px; }
.title { font-size: 24px; font-weight: 800; margin-bottom: 4px; }
.subtitle { color: var(--theme--foreground-subdued); font-size: 14px; }
.empty-state { height: 100%; display: flex; align-items: center; justify-content: center; }
.user-cell { display: flex; align-items: center; gap: 12px; } .user-cell { display: flex; align-items: center; gap: 12px; }
.user-name { font-weight: 600; } .user-name { font-weight: 600; }
.status-date { font-size: 12px; color: var(--theme--foreground-subdued); } .status-date { font-size: 12px; color: var(--theme--foreground-subdued); }
@@ -385,6 +390,7 @@ onMounted(() => {
.label { font-size: 12px; font-weight: 700; text-transform: uppercase; color: var(--theme--foreground-subdued); letter-spacing: 0.5px; } .label { font-size: 12px; font-weight: 700; text-transform: uppercase; color: var(--theme--foreground-subdued); letter-spacing: 0.5px; }
.field-note { font-size: 11px; color: var(--theme--foreground-subdued); margin-top: 4px; } .field-note { font-size: 11px; color: var(--theme--foreground-subdued); margin-top: 4px; }
.drawer-actions { margin-top: 24px; display: flex; flex-direction: column; gap: 12px; } .drawer-actions { margin-top: 24px; display: flex; flex-direction: column; gap: 12px; }
.password-input :deep(textarea) { .password-input :deep(textarea) {
font-family: var(--family-monospace); font-family: var(--family-monospace);
font-weight: 800; font-weight: 800;
@@ -394,5 +400,4 @@ onMounted(() => {
.clickable-table :deep(tbody tr) { cursor: pointer; transition: background-color 0.2s ease; } .clickable-table :deep(tbody tr) { cursor: pointer; transition: background-color 0.2s ease; }
.clickable-table :deep(tbody tr:hover) { background-color: var(--theme--background-subdued) !important; } .clickable-table :deep(tbody tr:hover) { background-color: var(--theme--background-subdued) !important; }
:deep(.v-list-item) { cursor: pointer !important; }
</style> </style>

View File

@@ -1,5 +1,13 @@
<template> <template>
<private-view title="People Manager"> <MintelManagerLayout
title="People Manager"
:item-title="`${selectedPerson?.first_name} ${selectedPerson?.last_name}` || 'Person wählen'"
:is-empty="!selectedPerson"
empty-title="Person auswählen"
empty-icon="person"
:notice="feedback"
@close-notice="feedback = null"
>
<template #navigation> <template #navigation>
<v-list nav> <v-list nav>
<v-list-item @click="openCreateDrawer" clickable> <v-list-item @click="openCreateDrawer" clickable>
@@ -17,7 +25,7 @@
v-for="person in people" v-for="person in people"
:key="person.id" :key="person.id"
:active="selectedPerson?.id === person.id" :active="selectedPerson?.id === person.id"
class="person-item" class="nav-item"
clickable clickable
@click="selectPerson(person)" @click="selectPerson(person)"
> >
@@ -31,43 +39,42 @@
</v-list> </v-list>
</template> </template>
<div class="content-wrapper"> <template #subtitle>
<v-notice v-if="feedback" :type="feedback.type" @close="feedback = null" dismissible> <template v-if="selectedPerson">
{{ feedback.message }} {{ getCompanyName(selectedPerson) }}
</v-notice> </template>
</template>
<div v-if="!selectedPerson" class="empty-state"> <template #actions>
<v-info title="Person auswählen" icon="person" center> <v-button secondary rounded icon v-tooltip.bottom="'Person bearbeiten'" @click="openEditDrawer">
Wähle eine Person in der Navigation aus oder <v-icon name="edit" />
<v-button x-small @click="openCreateDrawer">erstelle eine neue Person</v-button>. </v-button>
</v-info> <v-button danger rounded icon v-tooltip.bottom="'Person löschen'" @click="deletePerson">
<v-icon name="delete" />
</v-button>
</template>
<template #empty-state>
Wähle eine Person in der Navigation aus oder
<v-button x-small @click="openCreateDrawer">erstelle eine neue Person</v-button>.
</template>
<div v-if="selectedPerson" class="details-grid">
<div class="detail-item">
<span class="label">Vorname</span>
<p class="value">{{ selectedPerson.first_name }}</p>
</div> </div>
<div class="detail-item">
<div v-else> <span class="label">Nachname</span>
<header class="header"> <p class="value">{{ selectedPerson.last_name }}</p>
<div class="header-left"> </div>
<h1 class="title">{{ selectedPerson.first_name }} {{ selectedPerson.last_name }}</h1> <div class="detail-item">
<p class="subtitle">{{ getCompanyName(selectedPerson) }}</p> <span class="label">E-Mail</span>
</div> <p class="value">{{ selectedPerson.email || '---' }}</p>
</div>
<div class="header-right"> <div class="detail-item">
<v-button secondary rounded icon v-tooltip="'Person bearbeiten'" @click="openEditDrawer"> <span class="label">Organisation</span>
<v-icon name="edit" /> <p class="value">{{ getCompanyName(selectedPerson) }}</p>
</v-button>
<v-button danger rounded icon v-tooltip="'Person löschen'" @click="deletePerson">
<v-icon name="delete" />
</v-button>
</div>
</header>
<v-divider />
<div class="details-grid">
<div class="detail-item">
<span class="label">Name</span>
<p class="value">{{ selectedPerson.first_name }} {{ selectedPerson.last_name }}</p>
</div>
</div>
</div> </div>
</div> </div>
@@ -89,18 +96,20 @@
<span class="label">Nachname</span> <span class="label">Nachname</span>
<v-input v-model="form.last_name" placeholder="Nachname" /> <v-input v-model="form.last_name" placeholder="Nachname" />
</div> </div>
<div class="field">
<span class="label">E-Mail</span>
<v-input v-model="form.email" placeholder="E-Mail Adresse" type="email" />
</div>
<div class="field"> <div class="field">
<span class="label">Zentrale Firma</span> <span class="label">Zentrale Firma</span>
<v-select <MintelSelect
v-model="form.company" v-model="form.company"
:items="companyOptions" :items="companyOptions"
placeholder="Bestehende Firma auswählen..." placeholder="Bestehende Firma auswählen..."
allow-add
@add="openQuickAdd('company')"
/> />
</div> </div>
<div class="field">
<span class="label">Firma (Legacy / Neu)</span>
<v-input v-model="form.company_name" placeholder="z.B. Mintel" />
</div>
</div> </div>
<div class="drawer-actions"> <div class="drawer-actions">
@@ -111,12 +120,13 @@
</div> </div>
</template> </template>
</v-drawer> </v-drawer>
</private-view> </MintelManagerLayout>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted, computed } from 'vue'; import { ref, onMounted, computed, nextTick } from 'vue';
import { useApi } from '@directus/extensions-sdk'; import { useApi } from '@directus/extensions-sdk';
import { MintelManagerLayout, MintelSelect } from '@mintel/directus-extension-toolkit';
const api = useApi(); const api = useApi();
const people = ref([]); const people = ref([]);
@@ -131,8 +141,8 @@ const form = ref({
id: null, id: null,
first_name: '', first_name: '',
last_name: '', last_name: '',
company: null, email: '',
company_name: '' company: null
}); });
const companyOptions = computed(() => const companyOptions = computed(() =>
@@ -145,9 +155,9 @@ const companyOptions = computed(() =>
function getCompanyName(person: any) { function getCompanyName(person: any) {
if (!person) return '---'; if (!person) return '---';
if (person.company) { if (person.company) {
return typeof person.company === 'object' ? person.company.name : (companies.value.find(c => c.id === person.company)?.name || person.company_name); return typeof person.company === 'object' ? person.company.name : (companies.value.find(c => c.id === person.company)?.name || 'Unbekannte Firma');
} }
return person.company_name || '---'; return '---';
} }
async function fetchData() { async function fetchData() {
@@ -180,8 +190,8 @@ function openCreateDrawer() {
id: null, id: null,
first_name: '', first_name: '',
last_name: '', last_name: '',
company: null, email: '',
company_name: '' company: null
}; };
drawerActive.value = true; drawerActive.value = true;
} }
@@ -189,23 +199,12 @@ function openCreateDrawer() {
function openEditDrawer() { function openEditDrawer() {
isEditing.value = true; isEditing.value = true;
const person = selectedPerson.value; const person = selectedPerson.value;
let companyId = null;
let companyName = person.company_name || '';
if (person.company) {
if (typeof person.company === 'object') {
companyId = person.company.id;
} else if (person.company.length === 36) { // Assume UUID
companyId = person.company;
} else {
companyName = person.company;
}
}
form.value = { form.value = {
...person, id: person.id,
company: companyId, first_name: person.first_name,
company_name: companyName last_name: person.last_name,
email: person.email,
company: person.company?.id || person.company
}; };
drawerActive.value = true; drawerActive.value = true;
} }
@@ -218,17 +217,20 @@ async function savePerson() {
saving.value = true; saving.value = true;
try { try {
let updatedItem;
if (isEditing.value) { if (isEditing.value) {
await api.patch(`/items/people/${form.value.id}`, form.value); const res = await api.patch(`/items/people/${form.value.id}`, form.value);
updatedItem = res.data.data;
feedback.value = { type: 'success', message: 'Person aktualisiert!' }; feedback.value = { type: 'success', message: 'Person aktualisiert!' };
} else { } else {
await api.post('/items/people', form.value); const res = await api.post('/items/people', form.value);
updatedItem = res.data.data;
feedback.value = { type: 'success', message: 'Person angelegt!' }; feedback.value = { type: 'success', message: 'Person angelegt!' };
} }
drawerActive.value = false; drawerActive.value = false;
await fetchData(); await fetchData();
if (isEditing.value) { if (updatedItem) {
selectedPerson.value = people.value.find(p => p.id === form.value.id); selectedPerson.value = people.value.find(p => p.id === updatedItem.id) || updatedItem;
} }
} catch (error) { } catch (error) {
feedback.value = { type: 'danger', message: error.message }; feedback.value = { type: 'danger', message: error.message };
@@ -250,50 +252,18 @@ async function deletePerson() {
} }
} }
function openQuickAdd(type: string) {
feedback.value = { type: 'info', message: `Firma im Company Manager anlegen.` };
}
onMounted(fetchData); onMounted(fetchData);
</script> </script>
<style scoped> <style scoped>
.content-wrapper {
padding: 32px;
height: 100%;
}
.header {
margin-bottom: 24px;
display: flex;
justify-content: space-between;
align-items: flex-end;
}
.title {
font-size: 24px;
font-weight: 800;
margin-bottom: 4px;
}
.subtitle {
color: var(--theme--foreground-subdued);
font-size: 14px;
}
.header-right {
display: flex;
gap: 12px;
}
.empty-state {
height: 100%;
display: flex;
align-items: center;
justify-content: center;
}
.details-grid { .details-grid {
display: grid; display: grid;
grid-template-columns: repeat(2, 1fr); grid-template-columns: repeat(2, 1fr);
gap: 32px; gap: 32px;
margin-top: 32px;
} }
.detail-item { .detail-item {

132
pnpm-lock.yaml generated
View File

@@ -161,7 +161,7 @@ importers:
devDependencies: devDependencies:
'@directus/extensions-sdk': '@directus/extensions-sdk':
specifier: 11.0.2 specifier: 11.0.2
version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3) version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)
'@mintel/mail': '@mintel/mail':
specifier: workspace:* specifier: workspace:*
version: link:../mail version: link:../mail
@@ -180,6 +180,9 @@ importers:
'@directus/extensions-sdk': '@directus/extensions-sdk':
specifier: 11.0.2 specifier: 11.0.2
version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3) version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)
'@mintel/directus-extension-toolkit':
specifier: workspace:*
version: link:../directus-extension-toolkit
vue: vue:
specifier: ^3.4.0 specifier: ^3.4.0
version: 3.5.28(typescript@5.9.3) version: 3.5.28(typescript@5.9.3)
@@ -247,6 +250,9 @@ importers:
'@directus/extensions-sdk': '@directus/extensions-sdk':
specifier: 11.0.2 specifier: 11.0.2
version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3) version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)
'@mintel/directus-extension-toolkit':
specifier: workspace:*
version: link:../directus-extension-toolkit
vue: vue:
specifier: ^3.4.0 specifier: ^3.4.0
version: 3.5.28(typescript@5.9.3) version: 3.5.28(typescript@5.9.3)
@@ -256,6 +262,9 @@ importers:
'@directus/extensions-sdk': '@directus/extensions-sdk':
specifier: 11.0.2 specifier: 11.0.2
version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3) version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)
'@mintel/directus-extension-toolkit':
specifier: workspace:*
version: link:../directus-extension-toolkit
vue: vue:
specifier: ^3.4.0 specifier: ^3.4.0
version: 3.5.28(typescript@5.9.3) version: 3.5.28(typescript@5.9.3)
@@ -428,7 +437,7 @@ importers:
version: 5.9.3 version: 5.9.3
vitest: vitest:
specifier: ^3.0.4 specifier: ^3.0.4
version: 3.2.4(@types/debug@4.1.12)(@types/node@22.19.10)(@vitest/ui@4.0.18(vitest@4.0.18))(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0) version: 3.2.4(@types/debug@4.1.12)(@types/node@22.19.10)(@vitest/ui@4.0.18)(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0)
packages/next-config: packages/next-config:
dependencies: dependencies:
@@ -590,7 +599,7 @@ importers:
version: 5.9.3 version: 5.9.3
vitest: vitest:
specifier: ^2.0.0 specifier: ^2.0.0
version: 2.1.9(@types/node@22.19.10)(@vitest/ui@4.0.18(vitest@4.0.18))(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0) version: 2.1.9(@types/node@22.19.10)(@vitest/ui@4.0.18)(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0)
packages/pdf-library: packages/pdf-library:
dependencies: dependencies:
@@ -631,6 +640,9 @@ importers:
'@directus/extensions-sdk': '@directus/extensions-sdk':
specifier: 11.0.2 specifier: 11.0.2
version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3) version: 11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)
'@mintel/directus-extension-toolkit':
specifier: workspace:*
version: link:../directus-extension-toolkit
vue: vue:
specifier: ^3.4.0 specifier: ^3.4.0
version: 3.5.28(typescript@5.9.3) version: 3.5.28(typescript@5.9.3)
@@ -8803,6 +8815,57 @@ snapshots:
'@directus/constants@11.0.3': {} '@directus/constants@11.0.3': {}
'@directus/extensions-sdk@11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)':
dependencies:
'@directus/composables': 10.1.12(vue@3.4.21(typescript@5.9.3))
'@directus/constants': 11.0.3
'@directus/extensions': 1.0.2(@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3)))(knex@3.1.0)(pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)))(pino@10.3.1)(vue@3.4.21(typescript@5.9.3))
'@directus/themes': 0.3.6(@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3)))(pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)))(vue@3.4.21(typescript@5.9.3))
'@directus/types': 11.0.8(knex@3.1.0)(vue@3.4.21(typescript@5.9.3))
'@directus/utils': 11.0.7(vue@3.4.21(typescript@5.9.3))
'@rollup/plugin-commonjs': 25.0.7(rollup@3.29.4)
'@rollup/plugin-json': 6.1.0(rollup@3.29.4)
'@rollup/plugin-node-resolve': 15.2.3(rollup@3.29.4)
'@rollup/plugin-replace': 5.0.5(rollup@3.29.4)
'@rollup/plugin-terser': 0.4.4(rollup@3.29.4)
'@rollup/plugin-virtual': 3.0.2(rollup@3.29.4)
'@vitejs/plugin-vue': 4.6.2(vite@4.5.2(@types/node@22.19.10)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0))(vue@3.4.21(typescript@5.9.3))
chalk: 5.3.0
commander: 10.0.1
esbuild: 0.17.19
execa: 7.2.0
fs-extra: 11.2.0
inquirer: 9.2.16
ora: 6.3.1
rollup: 3.29.4
rollup-plugin-esbuild: 5.0.0(esbuild@0.17.19)(rollup@3.29.4)
rollup-plugin-styles: 4.0.0(rollup@3.29.4)
vite: 4.5.2(@types/node@22.19.10)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0)
vue: 3.4.21(typescript@5.9.3)
transitivePeerDependencies:
- '@types/node'
- '@unhead/vue'
- better-sqlite3
- debug
- knex
- less
- lightningcss
- mysql
- mysql2
- pg
- pg-native
- pinia
- pino
- sass
- sqlite3
- stylus
- sugarss
- supports-color
- tedious
- terser
- typescript
- vue-router
'@directus/extensions-sdk@11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)': '@directus/extensions-sdk@11.0.2(@types/node@22.19.10)(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(lightningcss@1.30.2)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(sass@1.97.3)(terser@5.46.0)(typescript@5.9.3)':
dependencies: dependencies:
'@directus/composables': 10.1.12(vue@3.4.21(typescript@5.9.3)) '@directus/composables': 10.1.12(vue@3.4.21(typescript@5.9.3))
@@ -8854,6 +8917,32 @@ snapshots:
- typescript - typescript
- vue-router - vue-router
'@directus/extensions@1.0.2(@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3)))(knex@3.1.0)(pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)))(pino@10.3.1)(vue@3.4.21(typescript@5.9.3))':
dependencies:
'@directus/constants': 11.0.3
'@directus/themes': 0.3.6(@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3)))(pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)))(vue@3.4.21(typescript@5.9.3))
'@directus/types': 11.0.8(knex@3.1.0)(vue@3.4.21(typescript@5.9.3))
'@directus/utils': 11.0.7(vue@3.4.21(typescript@5.9.3))
'@types/express': 4.17.21
fs-extra: 11.2.0
lodash-es: 4.17.21
zod: 3.22.4
optionalDependencies:
knex: 3.1.0
pino: 10.3.1
vue: 3.4.21(typescript@5.9.3)
transitivePeerDependencies:
- '@unhead/vue'
- better-sqlite3
- mysql
- mysql2
- pg
- pg-native
- pinia
- sqlite3
- supports-color
- tedious
'@directus/extensions@1.0.2(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(vue@3.4.21(typescript@5.9.3))': '@directus/extensions@1.0.2(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(knex@3.1.0)(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(pino@10.3.1)(vue@3.4.21(typescript@5.9.3))':
dependencies: dependencies:
'@directus/constants': 11.0.3 '@directus/constants': 11.0.3
@@ -8897,6 +8986,17 @@ snapshots:
'@directus/system-data@1.0.2': {} '@directus/system-data@1.0.2': {}
'@directus/themes@0.3.6(@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3)))(pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)))(vue@3.4.21(typescript@5.9.3))':
dependencies:
'@directus/utils': 11.0.7(vue@3.4.21(typescript@5.9.3))
'@sinclair/typebox': 0.32.15
'@unhead/vue': 1.11.20(vue@3.4.21(typescript@5.9.3))
decamelize: 6.0.0
flat: 6.0.1
lodash-es: 4.17.21
pinia: 2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3))
vue: 3.4.21(typescript@5.9.3)
'@directus/themes@0.3.6(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(vue@3.4.21(typescript@5.9.3))': '@directus/themes@0.3.6(@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3)))(pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)))(vue@3.4.21(typescript@5.9.3))':
dependencies: dependencies:
'@directus/utils': 11.0.7(vue@3.4.21(typescript@5.9.3)) '@directus/utils': 11.0.7(vue@3.4.21(typescript@5.9.3))
@@ -10988,6 +11088,14 @@ snapshots:
'@unhead/schema': 1.11.20 '@unhead/schema': 1.11.20
packrup: 0.1.2 packrup: 0.1.2
'@unhead/vue@1.11.20(vue@3.4.21(typescript@5.9.3))':
dependencies:
'@unhead/schema': 1.11.20
'@unhead/shared': 1.11.20
hookable: 5.5.3
unhead: 1.11.20
vue: 3.4.21(typescript@5.9.3)
'@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3))': '@unhead/vue@1.11.20(vue@3.5.28(typescript@5.9.3))':
dependencies: dependencies:
'@unhead/schema': 1.11.20 '@unhead/schema': 1.11.20
@@ -14418,6 +14526,16 @@ snapshots:
pify@4.0.1: {} pify@4.0.1: {}
pinia@2.3.1(typescript@5.9.3)(vue@3.4.21(typescript@5.9.3)):
dependencies:
'@vue/devtools-api': 6.6.4
vue: 3.4.21(typescript@5.9.3)
vue-demi: 0.14.10(vue@3.4.21(typescript@5.9.3))
optionalDependencies:
typescript: 5.9.3
transitivePeerDependencies:
- '@vue/composition-api'
pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)): pinia@2.3.1(typescript@5.9.3)(vue@3.5.28(typescript@5.9.3)):
dependencies: dependencies:
'@vue/devtools-api': 6.6.4 '@vue/devtools-api': 6.6.4
@@ -15898,7 +16016,7 @@ snapshots:
tsx: 4.21.0 tsx: 4.21.0
yaml: 2.8.2 yaml: 2.8.2
vitest@2.1.9(@types/node@22.19.10)(@vitest/ui@4.0.18(vitest@4.0.18))(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0): vitest@2.1.9(@types/node@22.19.10)(@vitest/ui@4.0.18)(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0):
dependencies: dependencies:
'@vitest/expect': 2.1.9 '@vitest/expect': 2.1.9
'@vitest/mocker': 2.1.9(vite@5.4.21(@types/node@22.19.10)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0)) '@vitest/mocker': 2.1.9(vite@5.4.21(@types/node@22.19.10)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0))
@@ -15936,7 +16054,7 @@ snapshots:
- supports-color - supports-color
- terser - terser
vitest@3.2.4(@types/debug@4.1.12)(@types/node@22.19.10)(@vitest/ui@4.0.18(vitest@4.0.18))(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0): vitest@3.2.4(@types/debug@4.1.12)(@types/node@22.19.10)(@vitest/ui@4.0.18)(happy-dom@20.5.3)(jsdom@27.4.0)(lightningcss@1.30.2)(sass@1.97.3)(terser@5.46.0):
dependencies: dependencies:
'@types/chai': 5.2.3 '@types/chai': 5.2.3
'@vitest/expect': 3.2.4 '@vitest/expect': 3.2.4
@@ -16019,6 +16137,10 @@ snapshots:
- tsx - tsx
- yaml - yaml
vue-demi@0.14.10(vue@3.4.21(typescript@5.9.3)):
dependencies:
vue: 3.4.21(typescript@5.9.3)
vue-demi@0.14.10(vue@3.5.28(typescript@5.9.3)): vue-demi@0.14.10(vue@3.5.28(typescript@5.9.3)):
dependencies: dependencies:
vue: 3.5.28(typescript@5.9.3) vue: 3.5.28(typescript@5.9.3)