Compare commits
36 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| a4d021c658 | |||
| 269d19bbef | |||
| 30ff08c66d | |||
| 81deaf447f | |||
| a0ebc58d6d | |||
| 7498c24c9a | |||
| efba82337c | |||
| c083b309fb | |||
| eb8bf60408 | |||
| a3819490ac | |||
| 1127954fea | |||
| fa0b133012 | |||
| 1b40baebd4 | |||
| 316c03869a | |||
| 63d2acfab5 | |||
| bdeae0aca6 | |||
| 47c70a16f1 | |||
| b96d44bf6d | |||
| 73b60f14a9 | |||
| b3f43c421f | |||
| a2339f7106 | |||
| e83a76f111 | |||
| 0096c18098 | |||
| 3284931f84 | |||
| 28517a3558 | |||
| 3b9f10ec98 | |||
| 65fd248993 | |||
| ebd9ab132c | |||
| ddaeb2c3ca | |||
| ad1a8c4fbf | |||
| 013b0259b2 | |||
| d5a9a3bce4 | |||
| b9fd583ac4 | |||
| bfdbaba0d0 | |||
| 4ea9cbc551 | |||
| d8c1a38c0d |
7
.changeset/resilient-build-scripts.md
Normal file
7
.changeset/resilient-build-scripts.md
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
---
|
||||||
|
"@mintel/monorepo": patch
|
||||||
|
"acquisition-manager": patch
|
||||||
|
"feedback-commander": patch
|
||||||
|
---
|
||||||
|
|
||||||
|
fix: make directus extension build scripts more resilient
|
||||||
36
.env
Normal file
36
.env
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
# Project
|
||||||
|
IMAGE_TAG=latest
|
||||||
|
PROJECT_NAME=at-mintel
|
||||||
|
PROJECT_COLOR=#82ed20
|
||||||
|
GITEA_TOKEN=ccce002e30fe16a31a6c9d5a414740af2f72a582
|
||||||
|
|
||||||
|
# Authentication
|
||||||
|
GATEKEEPER_PASSWORD=mintel
|
||||||
|
AUTH_COOKIE_NAME=mintel_gatekeeper_session
|
||||||
|
|
||||||
|
# Host Config (Local)
|
||||||
|
TRAEFIK_HOST=at-mintel.localhost
|
||||||
|
DIRECTUS_HOST=cms.localhost
|
||||||
|
|
||||||
|
# Next.js
|
||||||
|
NEXT_PUBLIC_BASE_URL=http://at-mintel.localhost
|
||||||
|
|
||||||
|
# Directus
|
||||||
|
DIRECTUS_URL=http://cms.localhost
|
||||||
|
DIRECTUS_KEY=F9IIfahEjPq6NZhKyRLw516D8GotuFj79EGK7pGfIWg=
|
||||||
|
DIRECTUS_SECRET=OZfxMu8lBxzaEnFGRKreNBoJpRiRu58U+HsVg2yWk4o=
|
||||||
|
CORS_ENABLED=true
|
||||||
|
CORS_ORIGIN=true
|
||||||
|
LOG_LEVEL=debug
|
||||||
|
DIRECTUS_ADMIN_EMAIL=mmintel@mintel.me
|
||||||
|
DIRECTUS_ADMIN_PASSWORD=Tim300493.
|
||||||
|
DIRECTUS_DB_NAME=directus
|
||||||
|
DIRECTUS_DB_USER=directus
|
||||||
|
DIRECTUS_DB_PASSWORD=mintel-db-pass
|
||||||
|
|
||||||
|
# Sentry / Glitchtip
|
||||||
|
SENTRY_DSN=
|
||||||
|
|
||||||
|
# Analytics (Umami)
|
||||||
|
NEXT_PUBLIC_UMAMI_WEBSITE_ID=
|
||||||
|
NEXT_PUBLIC_UMAMI_SCRIPT_URL=https://analytics.infra.mintel.me/script.js
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
# Project
|
# Project
|
||||||
IMAGE_TAG=v1.7.3
|
IMAGE_TAG=v1.7.12
|
||||||
PROJECT_NAME=sample-website
|
PROJECT_NAME=sample-website
|
||||||
PROJECT_COLOR=#82ed20
|
PROJECT_COLOR=#82ed20
|
||||||
|
|
||||||
|
|||||||
2
.eslintignore
Normal file
2
.eslintignore
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
**/index.js
|
||||||
|
**/dist/**
|
||||||
44
.gitea/workflows/maintenance.yml
Normal file
44
.gitea/workflows/maintenance.yml
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
name: 🏥 Server Maintenance
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '0 3 * * *' # Every day at 3:00 AM
|
||||||
|
workflow_dispatch: # Allow manual trigger
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
maintenance:
|
||||||
|
name: 🧹 Prune & Clean
|
||||||
|
runs-on: docker
|
||||||
|
container:
|
||||||
|
image: catthehacker/ubuntu:act-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: 🚀 Execute Maintenance via SSH
|
||||||
|
run: |
|
||||||
|
mkdir -p ~/.ssh
|
||||||
|
echo "${{ secrets.SSH_KEY }}" > ~/.ssh/id_ed25519
|
||||||
|
chmod 600 ~/.ssh/id_ed25519
|
||||||
|
ssh-keyscan -H ${{ secrets.SSH_HOST }} >> ~/.ssh/known_hosts 2>/dev/null
|
||||||
|
|
||||||
|
# Run the prune script on the host
|
||||||
|
# We transfer the script and execute it to ensure it matches the repo version
|
||||||
|
scp packages/infra/scripts/prune-registry.sh root@${{ secrets.SSH_HOST }}:/tmp/prune-registry.sh
|
||||||
|
ssh root@${{ secrets.SSH_HOST }} "bash /tmp/prune-registry.sh && rm /tmp/prune-registry.sh"
|
||||||
|
|
||||||
|
- name: 🔔 Notification - Success
|
||||||
|
if: success()
|
||||||
|
run: |
|
||||||
|
curl -s -k -X POST "${{ secrets.GOTIFY_URL }}/message?token=${{ secrets.GOTIFY_TOKEN }}" \
|
||||||
|
-F "title=🏥 Maintenance Complete" \
|
||||||
|
-F "message=Server-Wartung erfolgreich ausgeführt.\nRegistry & Docker Ressourcen bereinigt." \
|
||||||
|
-F "priority=2" || true
|
||||||
|
|
||||||
|
- name: 🔔 Notification - Failure
|
||||||
|
if: failure()
|
||||||
|
run: |
|
||||||
|
curl -s -k -X POST "${{ secrets.GOTIFY_URL }}/message?token=${{ secrets.GOTIFY_TOKEN }}" \
|
||||||
|
-F "title=❌ Maintenance FAILED" \
|
||||||
|
-F "message=Die automatische Server-Wartung ist fehlgeschlagen!\nBitte Logs prüfen." \
|
||||||
|
-F "priority=8" || true
|
||||||
@@ -12,8 +12,56 @@ concurrency:
|
|||||||
cancel-in-progress: true
|
cancel-in-progress: true
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
qa:
|
prioritize:
|
||||||
name: 🧪 Quality Assurance
|
name: ⚡ Prioritize Release
|
||||||
|
runs-on: docker
|
||||||
|
container:
|
||||||
|
image: catthehacker/ubuntu:act-latest
|
||||||
|
steps:
|
||||||
|
- name: 🛑 Cancel Redundant Runs
|
||||||
|
env:
|
||||||
|
GITEA_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
REPO: ${{ github.repository }}
|
||||||
|
RUN_ID: ${{ github.run_id }}
|
||||||
|
REF: ${{ github.ref }}
|
||||||
|
REF_NAME: ${{ github.ref_name }}
|
||||||
|
EVENT: ${{ github.event_name }}
|
||||||
|
run: |
|
||||||
|
echo "🔎 Debug: Event=$EVENT, Ref=$REF, RefName=$REF_NAME, RunId=$RUN_ID"
|
||||||
|
|
||||||
|
case "$REF" in
|
||||||
|
refs/tags/v*)
|
||||||
|
echo "🚀 Release detected ($REF_NAME). Cancelling non-tag runs..."
|
||||||
|
|
||||||
|
# Fetch all runs
|
||||||
|
RUNS=$(curl -s -H "Authorization: token $GITEA_TOKEN" "https://git.infra.mintel.me/api/v1/repos/$REPO/actions/runs")
|
||||||
|
|
||||||
|
# Identify runs to cancel: in_progress/queued, NOT this run, and NOT a tag run
|
||||||
|
echo "$RUNS" | jq -c '.workflow_runs[] | select(.status == "in_progress" or .status == "queued") | select(.id | tostring != "'$RUN_ID'")' | while read -r run; do
|
||||||
|
ID=$(echo "$run" | jq -r '.id')
|
||||||
|
RUN_REF=$(echo "$run" | jq -r '.ref')
|
||||||
|
TITLE=$(echo "$run" | jq -r '.display_title')
|
||||||
|
|
||||||
|
case "$RUN_REF" in
|
||||||
|
refs/tags/v*)
|
||||||
|
echo "⏭️ Skipping parallel release run $ID ($TITLE) on $RUN_REF"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "🛑 Cancelling redundant branch run $ID ($TITLE) on $RUN_REF..."
|
||||||
|
curl -X POST -s -H "Authorization: token $GITEA_TOKEN" "https://git.infra.mintel.me/api/v1/repos/$REPO/actions/runs/$ID/cancel"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "ℹ️ Regular push. No prioritization needed."
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
lint:
|
||||||
|
name: 🧹 Lint
|
||||||
|
needs: prioritize
|
||||||
|
if: always() && !cancelled() && (needs.prioritize.result == 'success' || needs.prioritize.result == 'skipped')
|
||||||
runs-on: docker
|
runs-on: docker
|
||||||
container:
|
container:
|
||||||
image: catthehacker/ubuntu:act-latest
|
image: catthehacker/ubuntu:act-latest
|
||||||
@@ -22,36 +70,66 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Install pnpm
|
|
||||||
uses: pnpm/action-setup@v4
|
|
||||||
with:
|
|
||||||
version: 10
|
|
||||||
|
|
||||||
- name: Set up Node.js
|
- name: Set up Node.js
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node_version: 20
|
node_version: 20
|
||||||
|
- name: Enable pnpm
|
||||||
|
run: corepack enable && corepack prepare pnpm@10.2.0 --activate
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: pnpm install --frozen-lockfile
|
run: pnpm install --frozen-lockfile --prefer-offline --ignore-scripts --no-color
|
||||||
|
|
||||||
- name: 🏷️ Sync Versions (if Tagged)
|
|
||||||
if: startsWith(github.ref, 'refs/tags/v')
|
|
||||||
run: pnpm sync-versions
|
|
||||||
|
|
||||||
- name: Lint
|
- name: Lint
|
||||||
run: pnpm lint
|
run: pnpm lint
|
||||||
|
|
||||||
|
test:
|
||||||
|
name: 🧪 Test
|
||||||
|
needs: prioritize
|
||||||
|
if: always() && !cancelled() && (needs.prioritize.result == 'success' || needs.prioritize.result == 'skipped')
|
||||||
|
runs-on: docker
|
||||||
|
container:
|
||||||
|
image: catthehacker/ubuntu:act-latest
|
||||||
|
env:
|
||||||
|
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- name: Set up Node.js
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node_version: 20
|
||||||
|
- name: Enable pnpm
|
||||||
|
run: corepack enable && corepack prepare pnpm@10.2.0 --activate
|
||||||
|
- name: Install dependencies
|
||||||
|
run: pnpm install --frozen-lockfile --prefer-offline --ignore-scripts --no-color
|
||||||
- name: Test
|
- name: Test
|
||||||
run: pnpm test
|
run: pnpm test
|
||||||
|
|
||||||
|
build:
|
||||||
|
name: 🏗️ Build
|
||||||
|
needs: prioritize
|
||||||
|
if: always() && !cancelled() && (needs.prioritize.result == 'success' || needs.prioritize.result == 'skipped')
|
||||||
|
runs-on: docker
|
||||||
|
container:
|
||||||
|
image: catthehacker/ubuntu:act-latest
|
||||||
|
env:
|
||||||
|
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- name: Set up Node.js
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node_version: 20
|
||||||
|
- name: Enable pnpm
|
||||||
|
run: corepack enable && corepack prepare pnpm@10.2.0 --activate
|
||||||
|
- name: Install dependencies
|
||||||
|
run: pnpm install --frozen-lockfile --prefer-offline --ignore-scripts --no-color
|
||||||
- name: Build
|
- name: Build
|
||||||
run: pnpm build
|
run: pnpm build
|
||||||
|
|
||||||
release:
|
release:
|
||||||
name: 🚀 Release
|
name: 🚀 Release
|
||||||
needs: qa
|
needs: [lint, test, build]
|
||||||
if: startsWith(github.ref, 'refs/tags/v')
|
if: startsWith(github.ref, 'refs/tags/v')
|
||||||
runs-on: docker
|
runs-on: docker
|
||||||
container:
|
container:
|
||||||
@@ -64,20 +142,16 @@ jobs:
|
|||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Install pnpm
|
|
||||||
uses: pnpm/action-setup@v4
|
|
||||||
with:
|
|
||||||
version: 10
|
|
||||||
|
|
||||||
- name: Set up Node.js
|
- name: Set up Node.js
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node_version: 20
|
node_version: 20
|
||||||
|
- name: Enable pnpm
|
||||||
|
run: corepack enable && corepack prepare pnpm@10.2.0 --activate
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: pnpm install --frozen-lockfile
|
run: pnpm install --frozen-lockfile --prefer-offline --ignore-scripts --no-color
|
||||||
|
- name: 🏷️ Sync Versions (if Tagged)
|
||||||
|
run: pnpm sync-versions
|
||||||
- name: 🏷️ Release Packages (Tag-Driven)
|
- name: 🏷️ Release Packages (Tag-Driven)
|
||||||
run: |
|
run: |
|
||||||
echo "🏷️ Tag detected [${{ github.ref_name }}], performing sync release..."
|
echo "🏷️ Tag detected [${{ github.ref_name }}], performing sync release..."
|
||||||
@@ -85,7 +159,7 @@ jobs:
|
|||||||
|
|
||||||
build-images:
|
build-images:
|
||||||
name: 🐳 Build ${{ matrix.name }}
|
name: 🐳 Build ${{ matrix.name }}
|
||||||
needs: qa
|
needs: [lint, test, build]
|
||||||
if: startsWith(github.ref, 'refs/tags/v')
|
if: startsWith(github.ref, 'refs/tags/v')
|
||||||
runs-on: docker
|
runs-on: docker
|
||||||
container:
|
container:
|
||||||
|
|||||||
56
Dockerfile.template
Normal file
56
Dockerfile.template
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
# Stage 1: Builder
|
||||||
|
FROM registry.infra.mintel.me/mintel/nextjs:latest AS builder
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
# Clean the workspace in case the base image is dirty
|
||||||
|
RUN rm -rf ./*
|
||||||
|
|
||||||
|
# Arguments for build-time configuration
|
||||||
|
ARG NEXT_PUBLIC_BASE_URL
|
||||||
|
ARG NEXT_PUBLIC_TARGET
|
||||||
|
ARG DIRECTUS_URL
|
||||||
|
ARG NPM_TOKEN
|
||||||
|
|
||||||
|
# Environment variables for Next.js build
|
||||||
|
ENV NEXT_PUBLIC_BASE_URL=$NEXT_PUBLIC_BASE_URL
|
||||||
|
ENV NEXT_PUBLIC_TARGET=$NEXT_PUBLIC_TARGET
|
||||||
|
ENV DIRECTUS_URL=$DIRECTUS_URL
|
||||||
|
ENV SKIP_RUNTIME_ENV_VALIDATION=true
|
||||||
|
ENV CI=true
|
||||||
|
|
||||||
|
# Enable pnpm
|
||||||
|
RUN corepack enable
|
||||||
|
|
||||||
|
# Copy lockfile and manifest for dependency installation caching
|
||||||
|
COPY pnpm-lock.yaml package.json .npmrc* ./
|
||||||
|
|
||||||
|
# Install dependencies with cache mount
|
||||||
|
RUN --mount=type=cache,id=pnpm,target=/pnpm/store \
|
||||||
|
--mount=type=secret,id=NPM_TOKEN \
|
||||||
|
export NPM_TOKEN=$(cat /run/secrets/NPM_TOKEN 2>/dev/null || echo $NPM_TOKEN) && \
|
||||||
|
pnpm install --frozen-lockfile
|
||||||
|
|
||||||
|
# Copy source code
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
# Build application
|
||||||
|
RUN pnpm build
|
||||||
|
|
||||||
|
# Stage 2: Runner
|
||||||
|
FROM registry.infra.mintel.me/mintel/runtime:latest AS runner
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
ENV HOSTNAME="0.0.0.0"
|
||||||
|
ENV PORT=3000
|
||||||
|
ENV NODE_ENV=production
|
||||||
|
|
||||||
|
# Copy standalone output and static files
|
||||||
|
# Adjust paths if using a monorepo structure (e.g., /app/apps/web/public)
|
||||||
|
COPY --from=builder --chown=nextjs:nodejs /app/public ./public
|
||||||
|
COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./
|
||||||
|
COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static
|
||||||
|
COPY --from=builder --chown=nextjs:nodejs /app/.next/cache ./.next/cache
|
||||||
|
|
||||||
|
USER nextjs
|
||||||
|
|
||||||
|
CMD ["node", "server.js"]
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "sample-website",
|
"name": "sample-website",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
1
directus/extensions/acquisition-manager/index.js
Normal file
1
directus/extensions/acquisition-manager/index.js
Normal file
File diff suppressed because one or more lines are too long
30
directus/extensions/acquisition-manager/package.json
Normal file
30
directus/extensions/acquisition-manager/package.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "acquisition-manager",
|
||||||
|
"description": "Custom High-Fidelity Acquisition Management for Directus",
|
||||||
|
"icon": "account_balance_wallet",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "Acquisition Manager"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
27
directus/extensions/acquisition/package.json
Normal file
27
directus/extensions/acquisition/package.json
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
"name": "acquisition",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "endpoint",
|
||||||
|
"path": "dist/index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "^11.0.0"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "node build.mjs",
|
||||||
|
"dev": "node build.mjs --watch"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"@mintel/acquisition": "workspace:*",
|
||||||
|
"@mintel/mail": "workspace:*",
|
||||||
|
"esbuild": "^0.25.0",
|
||||||
|
"typescript": "^5.6.3"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"jquery": "^3.7.1",
|
||||||
|
"react": "^19.2.4",
|
||||||
|
"react-dom": "^19.2.4"
|
||||||
|
}
|
||||||
|
}
|
||||||
1
directus/extensions/customer-manager/index.js
Normal file
1
directus/extensions/customer-manager/index.js
Normal file
File diff suppressed because one or more lines are too long
30
directus/extensions/customer-manager/package.json
Normal file
30
directus/extensions/customer-manager/package.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "customer-manager",
|
||||||
|
"description": "Custom High-Fidelity Customer & Company Management for Directus",
|
||||||
|
"icon": "supervisor_account",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "Customer Manager"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
1
directus/extensions/feedback-commander/index.js
Normal file
1
directus/extensions/feedback-commander/index.js
Normal file
File diff suppressed because one or more lines are too long
30
directus/extensions/feedback-commander/package.json
Normal file
30
directus/extensions/feedback-commander/package.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "feedback-commander",
|
||||||
|
"description": "Custom High-Fidelity Feedback Management Extension for Directus",
|
||||||
|
"icon": "view_kanban",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "Feedback Commander"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
1
directus/extensions/people-manager/index.js
Normal file
1
directus/extensions/people-manager/index.js
Normal file
File diff suppressed because one or more lines are too long
30
directus/extensions/people-manager/package.json
Normal file
30
directus/extensions/people-manager/package.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "people-manager",
|
||||||
|
"description": "Custom High-Fidelity People Management for Directus",
|
||||||
|
"icon": "person",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "People Manager"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
1
directus/uploads/directus-health-file
Normal file
1
directus/uploads/directus-health-file
Normal file
@@ -0,0 +1 @@
|
|||||||
|
Qy-qP
|
||||||
@@ -24,6 +24,12 @@ services:
|
|||||||
|
|
||||||
directus:
|
directus:
|
||||||
image: registry.infra.mintel.me/mintel/directus:${IMAGE_TAG:-latest}
|
image: registry.infra.mintel.me/mintel/directus:${IMAGE_TAG:-latest}
|
||||||
|
healthcheck:
|
||||||
|
test: [ "CMD", "wget", "--no-verbose", "--tries=1", "--spider", "http://127.0.0.1:8055/server/health" ]
|
||||||
|
interval: 30s
|
||||||
|
timeout: 10s
|
||||||
|
retries: 3
|
||||||
|
start_period: 5s
|
||||||
restart: always
|
restart: always
|
||||||
networks:
|
networks:
|
||||||
- infra
|
- infra
|
||||||
@@ -35,7 +41,7 @@ services:
|
|||||||
ADMIN_EMAIL: ${DIRECTUS_ADMIN_EMAIL:-admin@mintel.me}
|
ADMIN_EMAIL: ${DIRECTUS_ADMIN_EMAIL:-admin@mintel.me}
|
||||||
ADMIN_PASSWORD: ${DIRECTUS_ADMIN_PASSWORD:-mintel-admin}
|
ADMIN_PASSWORD: ${DIRECTUS_ADMIN_PASSWORD:-mintel-admin}
|
||||||
DB_CLIENT: 'pg'
|
DB_CLIENT: 'pg'
|
||||||
DB_HOST: 'directus-db'
|
DB_HOST: 'at-mintel-directus-db'
|
||||||
DB_PORT: '5432'
|
DB_PORT: '5432'
|
||||||
DB_DATABASE: ${DIRECTUS_DB_NAME:-directus}
|
DB_DATABASE: ${DIRECTUS_DB_NAME:-directus}
|
||||||
DB_USER: ${DIRECTUS_DB_USER:-directus}
|
DB_USER: ${DIRECTUS_DB_USER:-directus}
|
||||||
@@ -53,7 +59,7 @@ services:
|
|||||||
- "traefik.http.routers.sample-website-directus.rule=Host(`${DIRECTUS_HOST:-cms.sample-website.localhost}`)"
|
- "traefik.http.routers.sample-website-directus.rule=Host(`${DIRECTUS_HOST:-cms.sample-website.localhost}`)"
|
||||||
- "traefik.http.services.sample-website-directus.loadbalancer.server.port=8055"
|
- "traefik.http.services.sample-website-directus.loadbalancer.server.port=8055"
|
||||||
|
|
||||||
directus-db:
|
at-mintel-directus-db:
|
||||||
image: postgres:15-alpine
|
image: postgres:15-alpine
|
||||||
restart: always
|
restart: always
|
||||||
networks:
|
networks:
|
||||||
|
|||||||
@@ -52,7 +52,7 @@
|
|||||||
"pino-pretty": "^13.1.3",
|
"pino-pretty": "^13.1.3",
|
||||||
"require-in-the-middle": "^8.0.1"
|
"require-in-the-middle": "^8.0.1"
|
||||||
},
|
},
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"pnpm": {
|
"pnpm": {
|
||||||
"overrides": {
|
"overrides": {
|
||||||
"next": "16.1.6",
|
"next": "16.1.6",
|
||||||
|
|||||||
1
packages/acquisition-manager/index.js
Normal file
1
packages/acquisition-manager/index.js
Normal file
File diff suppressed because one or more lines are too long
30
packages/acquisition-manager/package.json
Normal file
30
packages/acquisition-manager/package.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "acquisition-manager",
|
||||||
|
"description": "Custom High-Fidelity Acquisition Management for Directus",
|
||||||
|
"icon": "account_balance_wallet",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "Acquisition Manager"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
19
packages/acquisition-manager/src/index.ts
Normal file
19
packages/acquisition-manager/src/index.ts
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import { defineModule } from "@directus/extensions-sdk";
|
||||||
|
import ModuleComponent from "./module.vue";
|
||||||
|
|
||||||
|
export default defineModule({
|
||||||
|
id: "acquisition-manager",
|
||||||
|
name: "Acquisition",
|
||||||
|
icon: "auto_awesome",
|
||||||
|
routes: [
|
||||||
|
{
|
||||||
|
path: "",
|
||||||
|
component: ModuleComponent,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: ":id",
|
||||||
|
component: ModuleComponent,
|
||||||
|
props: true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
});
|
||||||
403
packages/acquisition-manager/src/module.vue
Normal file
403
packages/acquisition-manager/src/module.vue
Normal file
@@ -0,0 +1,403 @@
|
|||||||
|
<template>
|
||||||
|
<private-view title="Acquisition Manager">
|
||||||
|
<template #navigation>
|
||||||
|
<v-list nav>
|
||||||
|
<v-list-item @click="showAddLead = true" clickable>
|
||||||
|
<v-list-item-icon><v-icon name="add" color="var(--theme--primary)" /></v-list-item-icon>
|
||||||
|
<v-list-item-content>
|
||||||
|
<v-text-overflow text="Neuen Lead anlegen" />
|
||||||
|
</v-list-item-content>
|
||||||
|
</v-list-item>
|
||||||
|
|
||||||
|
<v-divider />
|
||||||
|
|
||||||
|
<v-list-item
|
||||||
|
v-for="lead in leads"
|
||||||
|
:key="lead.id"
|
||||||
|
:active="selectedLeadId === lead.id"
|
||||||
|
class="lead-item"
|
||||||
|
clickable
|
||||||
|
@click="selectLead(lead.id)"
|
||||||
|
>
|
||||||
|
<v-list-item-icon>
|
||||||
|
<v-icon :name="getStatusIcon(lead.status)" :color="getStatusColor(lead.status)" />
|
||||||
|
</v-list-item-icon>
|
||||||
|
<v-list-item-content>
|
||||||
|
<v-text-overflow :text="lead.company_name" />
|
||||||
|
</v-list-item-content>
|
||||||
|
</v-list-item>
|
||||||
|
</v-list>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #title-outer:after>
|
||||||
|
<v-notice v-if="notice" :type="notice.type" @close="notice = null" dismissible>
|
||||||
|
{{ notice.message }}
|
||||||
|
</v-notice>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<div class="content-wrapper">
|
||||||
|
<v-notice type="success" style="margin-bottom: 16px;">
|
||||||
|
DEBUG: Module Version 1.1.0 - Native Build - {{ new Date().toISOString() }}
|
||||||
|
</v-notice>
|
||||||
|
|
||||||
|
<div v-if="!selectedLead" class="empty-state">
|
||||||
|
<v-info title="Lead auswählen" icon="auto_awesome" center>
|
||||||
|
Wähle einen Lead in der Navigation aus oder
|
||||||
|
<v-button x-small @click="showAddLead = true">registriere einen neuen Lead</v-button>.
|
||||||
|
</v-info>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<template v-else>
|
||||||
|
<header class="header">
|
||||||
|
<div class="header-left">
|
||||||
|
<h1 class="title">{{ selectedLead.company_name }}</h1>
|
||||||
|
<p class="subtitle">
|
||||||
|
<v-icon name="language" x-small />
|
||||||
|
<a :href="selectedLead.website_url" target="_blank" class="url-link">
|
||||||
|
{{ selectedLead.website_url.replace(/^https?:\/\//, '') }}
|
||||||
|
</a>
|
||||||
|
· Status: {{ selectedLead.status.toUpperCase() }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="header-right">
|
||||||
|
<v-button
|
||||||
|
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'">
|
||||||
|
<v-button secondary :loading="loadingEmail" @click="sendAuditEmail">
|
||||||
|
<v-icon name="mail" left />
|
||||||
|
Audit E-Mail
|
||||||
|
</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">
|
||||||
|
<v-icon name="open_in_new" />
|
||||||
|
</v-button>
|
||||||
|
|
||||||
|
<v-button
|
||||||
|
v-if="selectedLead.audit_pdf_path"
|
||||||
|
primary
|
||||||
|
: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">
|
||||||
|
<span class="label">E-Mail (Legacy)</span>
|
||||||
|
<div class="value">{{ selectedLead.contact_email || '—' }}</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>
|
||||||
|
|
||||||
|
<v-divider />
|
||||||
|
|
||||||
|
<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>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Drawer: New Lead -->
|
||||||
|
<v-drawer
|
||||||
|
v-model="showAddLead"
|
||||||
|
title="Neuen Lead registrieren"
|
||||||
|
icon="person_add"
|
||||||
|
@cancel="showAddLead = false"
|
||||||
|
>
|
||||||
|
<div class="drawer-content">
|
||||||
|
<div class="form-section">
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Firma</span>
|
||||||
|
<v-input v-model="newLead.company_name" placeholder="z.B. Schmidt GmbH" autofocus />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Website URL</span>
|
||||||
|
<v-input v-model="newLead.website_url" placeholder="https://..." />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Ansprechpartner</span>
|
||||||
|
<v-input v-model="newLead.contact_name" placeholder="Vorname Nachname" />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">E-Mail Adresse</span>
|
||||||
|
<v-input v-model="newLead.contact_email" placeholder="email@beispiel.de" type="email" />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Briefing / Fokus</span>
|
||||||
|
<v-textarea v-model="newLead.briefing" placeholder="Besonderheiten für das Audit..." />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Kontaktperson (Optional)</span>
|
||||||
|
<v-select
|
||||||
|
v-model="newLead.contact_person"
|
||||||
|
:items="peopleOptions"
|
||||||
|
placeholder="Person auswählen..."
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="drawer-actions">
|
||||||
|
<v-button primary block :loading="savingLead" @click="saveLead">Lead speichern</v-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</v-drawer>
|
||||||
|
</private-view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, onMounted, computed } from 'vue';
|
||||||
|
import { useApi } from '@directus/extensions-sdk';
|
||||||
|
|
||||||
|
const api = useApi();
|
||||||
|
const leads = ref<any[]>([]);
|
||||||
|
const selectedLeadId = ref<string | null>(null);
|
||||||
|
const loadingAudit = ref(false);
|
||||||
|
const loadingPdf = ref(false);
|
||||||
|
const loadingEmail = ref(false);
|
||||||
|
const showAddLead = ref(false);
|
||||||
|
const savingLead = ref(false);
|
||||||
|
const notice = ref<{ type: string; message: string } | null>(null);
|
||||||
|
|
||||||
|
const newLead = ref({
|
||||||
|
company_name: '',
|
||||||
|
website_url: '',
|
||||||
|
contact_name: '',
|
||||||
|
contact_email: '',
|
||||||
|
contact_person: null,
|
||||||
|
briefing: '',
|
||||||
|
status: 'new'
|
||||||
|
});
|
||||||
|
|
||||||
|
const people = ref<any[]>([]);
|
||||||
|
|
||||||
|
const peopleOptions = computed(() =>
|
||||||
|
people.value.map(p => ({
|
||||||
|
text: `${p.first_name} ${p.last_name}`,
|
||||||
|
value: p.id
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
function getPersonName(id: string) {
|
||||||
|
const person = people.value.find(p => p.id === id);
|
||||||
|
return person ? `${person.first_name} ${person.last_name}` : id;
|
||||||
|
}
|
||||||
|
|
||||||
|
function goToPerson(id: string) {
|
||||||
|
// Logic to navigate to people manager or open details
|
||||||
|
notice.value = { type: 'info', message: `Navigiere zu Person: ${id}` };
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectedLead = computed(() => leads.value.find(l => l.id === selectedLeadId.value));
|
||||||
|
|
||||||
|
onMounted(fetchLeads);
|
||||||
|
|
||||||
|
async function fetchLeads() {
|
||||||
|
const [leadsResp, peopleResp] = await Promise.all([
|
||||||
|
api.get('/items/leads', { params: { sort: '-date_created' } }),
|
||||||
|
api.get('/items/people', { params: { sort: 'last_name' } })
|
||||||
|
]);
|
||||||
|
leads.value = leadsResp.data.data;
|
||||||
|
people.value = peopleResp.data.data;
|
||||||
|
if (!selectedLeadId.value && leads.value.length > 0) {
|
||||||
|
selectedLeadId.value = leads.value[0].id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectLead(id: string) {
|
||||||
|
selectedLeadId.value = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function runAudit() {
|
||||||
|
if (!selectedLeadId.value) return;
|
||||||
|
loadingAudit.value = true;
|
||||||
|
try {
|
||||||
|
await api.post(`/acquisition/audit/${selectedLeadId.value}`);
|
||||||
|
notice.value = { type: 'success', message: 'Audit erfolgreich gestartet!' };
|
||||||
|
await fetchLeads();
|
||||||
|
} catch (e: any) {
|
||||||
|
notice.value = { type: 'danger', message: `Fehler beim Audit: ${e.message}` };
|
||||||
|
} finally {
|
||||||
|
loadingAudit.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function sendAuditEmail() {
|
||||||
|
if (!selectedLeadId.value) return;
|
||||||
|
loadingEmail.value = true;
|
||||||
|
try {
|
||||||
|
await api.post(`/acquisition/audit-email/${selectedLeadId.value}`);
|
||||||
|
notice.value = { type: 'success', message: 'Audit E-Mail versendet!' };
|
||||||
|
await fetchLeads();
|
||||||
|
} catch (e: any) {
|
||||||
|
notice.value = { type: 'danger', message: `Fehler beim Versenden: ${e.message}` };
|
||||||
|
} finally {
|
||||||
|
loadingEmail.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function generatePdf() {
|
||||||
|
if (!selectedLeadId.value) return;
|
||||||
|
loadingPdf.value = true;
|
||||||
|
try {
|
||||||
|
await api.post(`/acquisition/estimate/${selectedLeadId.value}`);
|
||||||
|
notice.value = { type: 'success', message: 'Angebot (PDF) wurde generiert!' };
|
||||||
|
await fetchLeads();
|
||||||
|
} catch (e: any) {
|
||||||
|
notice.value = { type: 'danger', message: `Fehler bei PDF Generierung: ${e.message}` };
|
||||||
|
} finally {
|
||||||
|
loadingPdf.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function sendEstimateEmail() {
|
||||||
|
if (!selectedLeadId.value) return;
|
||||||
|
loadingEmail.value = true;
|
||||||
|
try {
|
||||||
|
await api.post(`/acquisition/estimate-email/${selectedLeadId.value}`);
|
||||||
|
notice.value = { type: 'success', message: 'Angebot erfolgreich versendet!' };
|
||||||
|
await fetchLeads();
|
||||||
|
} catch (e: any) {
|
||||||
|
notice.value = { type: 'danger', message: `Fehler beim Versenden: ${e.message}` };
|
||||||
|
} finally {
|
||||||
|
loadingEmail.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function openPdf() {
|
||||||
|
if (!selectedLead.value?.audit_pdf_path) return;
|
||||||
|
window.open(`${window.location.origin}/assets/${selectedLead.value.audit_pdf_path}`, '_blank');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveLead() {
|
||||||
|
if (!newLead.value.company_name) return;
|
||||||
|
savingLead.value = true;
|
||||||
|
try {
|
||||||
|
const payload = {
|
||||||
|
id: crypto.randomUUID(),
|
||||||
|
...newLead.value
|
||||||
|
};
|
||||||
|
await api.post('/items/leads', payload);
|
||||||
|
notice.value = { type: 'success', message: 'Lead erfolgreich registriert!' };
|
||||||
|
showAddLead.value = false;
|
||||||
|
await fetchLeads();
|
||||||
|
selectedLeadId.value = payload.id;
|
||||||
|
newLead.value = {
|
||||||
|
company_name: '',
|
||||||
|
website_url: '',
|
||||||
|
contact_name: '',
|
||||||
|
contact_email: '',
|
||||||
|
contact_person: null,
|
||||||
|
briefing: '',
|
||||||
|
status: 'new'
|
||||||
|
};
|
||||||
|
} catch (e: any) {
|
||||||
|
notice.value = { type: 'danger', message: `Fehler beim Speichern: ${e.message}` };
|
||||||
|
} finally {
|
||||||
|
savingLead.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getStatusIcon(status: string) {
|
||||||
|
switch(status) {
|
||||||
|
case 'new': return 'fiber_new';
|
||||||
|
case 'auditing': return 'hourglass_empty';
|
||||||
|
case 'audit_ready': return 'check_circle';
|
||||||
|
case 'contacted': return 'mail_outline';
|
||||||
|
default: return 'help_outline';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getStatusColor(status: string) {
|
||||||
|
switch(status) {
|
||||||
|
case 'new': return 'var(--theme--primary)';
|
||||||
|
case 'auditing': return 'var(--theme--warning)';
|
||||||
|
case 'audit_ready': return 'var(--theme--success)';
|
||||||
|
case 'contacted': return 'var(--theme--secondary)';
|
||||||
|
default: return 'var(--theme--foreground-subdued)';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<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: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; }
|
||||||
|
|
||||||
|
.form-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 24px; }
|
||||||
|
.field { display: flex; flex-direction: column; gap: 8px; }
|
||||||
|
.field.full { grid-column: span 2; }
|
||||||
|
.label { font-size: 11px; font-weight: 700; text-transform: uppercase; color: var(--theme--foreground-subdued); letter-spacing: 0.5px; }
|
||||||
|
.value { font-size: 15px; color: var(--theme--foreground); }
|
||||||
|
.text-block { line-height: 1.6; white-space: pre-wrap; background: var(--theme--background-subdued); padding: 16px; border-radius: 8px; }
|
||||||
|
|
||||||
|
.ai-observations { display: flex; flex-direction: column; gap: 16px; }
|
||||||
|
.section-title { font-size: 16px; font-weight: 700; color: var(--theme--foreground); margin-bottom: 8px; }
|
||||||
|
.metrics { display: flex; gap: 32px; margin-bottom: 16px; }
|
||||||
|
|
||||||
|
.observation-table { border: 1px solid var(--theme--border); border-radius: 8px; overflow: hidden; }
|
||||||
|
.page-title { font-weight: 600; }
|
||||||
|
.page-url { font-family: var(--family-monospace); font-size: 12px; color: var(--theme--foreground-subdued); }
|
||||||
|
|
||||||
|
.drawer-content { padding: 24px; display: flex; flex-direction: column; gap: 32px; }
|
||||||
|
.form-section { display: flex; flex-direction: column; gap: 20px; }
|
||||||
|
.drawer-actions { margin-top: 24px; display: flex; flex-direction: column; gap: 12px; }
|
||||||
|
|
||||||
|
:deep(.v-list-item) { cursor: pointer !important; }
|
||||||
|
</style>
|
||||||
49
packages/acquisition/build.mjs
Normal file
49
packages/acquisition/build.mjs
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
import { build } from 'esbuild';
|
||||||
|
import { resolve, dirname } from 'path';
|
||||||
|
import { mkdirSync } from 'fs';
|
||||||
|
import { fileURLToPath } from 'url';
|
||||||
|
|
||||||
|
const __filename = fileURLToPath(import.meta.url);
|
||||||
|
const __dirname = dirname(__filename);
|
||||||
|
|
||||||
|
const entryPoint = resolve(__dirname, 'src/index.ts');
|
||||||
|
const outfile = resolve(__dirname, 'dist/index.js');
|
||||||
|
|
||||||
|
try {
|
||||||
|
mkdirSync(dirname(outfile), { recursive: true });
|
||||||
|
} catch (e) { }
|
||||||
|
|
||||||
|
console.log(`Building from ${entryPoint} to ${outfile}...`);
|
||||||
|
|
||||||
|
build({
|
||||||
|
entryPoints: [entryPoint],
|
||||||
|
bundle: true,
|
||||||
|
platform: 'node',
|
||||||
|
target: 'node18',
|
||||||
|
outfile: outfile,
|
||||||
|
jsx: 'automatic',
|
||||||
|
loader: {
|
||||||
|
'.tsx': 'tsx',
|
||||||
|
'.ts': 'ts',
|
||||||
|
'.js': 'js',
|
||||||
|
},
|
||||||
|
external: ["@react-pdf/renderer", "react", "react-dom", "jsdom", "jsdom/*", "jquery", "jquery/*", "canvas", "fs", "path", "os", "http", "https", "zlib", "stream", "util", "url", "net", "tls", "crypto"],
|
||||||
|
plugins: [{
|
||||||
|
name: 'mock-canvas',
|
||||||
|
setup(build) {
|
||||||
|
build.onResolve({ filter: /^canvas/ }, args => ({ path: args.path, namespace: 'mock-canvas' }));
|
||||||
|
build.onLoad({ filter: /.*/, namespace: 'mock-canvas' }, () => ({ contents: 'export default {};', loader: 'js' }));
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
name: 'mock-jsdom',
|
||||||
|
setup(build) {
|
||||||
|
build.onResolve({ filter: /^jsdom/ }, args => ({ path: args.path, namespace: 'mock-jsdom' }));
|
||||||
|
build.onLoad({ filter: /.*/, namespace: 'mock-jsdom' }, () => ({ contents: 'export default {};', loader: 'js' }));
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
}).then(() => {
|
||||||
|
console.log("Build succeeded!");
|
||||||
|
}).catch((e) => {
|
||||||
|
console.error("Build failed:", e);
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
27
packages/acquisition/package.json
Normal file
27
packages/acquisition/package.json
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
"name": "acquisition",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "endpoint",
|
||||||
|
"path": "dist/index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "^11.0.0"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "node build.mjs",
|
||||||
|
"dev": "node build.mjs --watch"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"@mintel/acquisition": "workspace:*",
|
||||||
|
"@mintel/mail": "workspace:*",
|
||||||
|
"esbuild": "^0.25.0",
|
||||||
|
"typescript": "^5.6.3"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"jquery": "^3.7.1",
|
||||||
|
"react": "^19.2.4",
|
||||||
|
"react-dom": "^19.2.4"
|
||||||
|
}
|
||||||
|
}
|
||||||
171
packages/acquisition/src/index.ts
Normal file
171
packages/acquisition/src/index.ts
Normal file
@@ -0,0 +1,171 @@
|
|||||||
|
import { defineEndpoint } from "@directus/extensions-sdk";
|
||||||
|
import { AcquisitionService, PdfEngine } from "@mintel/acquisition";
|
||||||
|
import { render, SiteAuditTemplate, ProjectEstimateTemplate } from "@mintel/mail";
|
||||||
|
import { createElement } from "react";
|
||||||
|
import * as path from "path";
|
||||||
|
import * as fs from "fs";
|
||||||
|
|
||||||
|
export default defineEndpoint((router, { services, env }) => {
|
||||||
|
const { ItemsService, MailService } = services;
|
||||||
|
|
||||||
|
router.get("/ping", (req, res) => res.send("pong"));
|
||||||
|
|
||||||
|
router.post("/audit/:id", async (req: any, res: any) => {
|
||||||
|
const { id } = req.params;
|
||||||
|
const leadsService = new ItemsService("leads", { schema: req.schema, accountability: req.accountability });
|
||||||
|
|
||||||
|
try {
|
||||||
|
const lead = await leadsService.readOne(id);
|
||||||
|
if (!lead) return res.status(404).send({ error: "Lead not found" });
|
||||||
|
|
||||||
|
await leadsService.updateOne(id, { status: "auditing" });
|
||||||
|
|
||||||
|
const acqService = new AcquisitionService(env.OPENROUTER_API_KEY);
|
||||||
|
const result = await acqService.runFullSequence(lead.website_url, lead.briefing, lead.comments);
|
||||||
|
|
||||||
|
await leadsService.updateOne(id, {
|
||||||
|
status: "audit_ready",
|
||||||
|
ai_state: result.state,
|
||||||
|
audit_context: JSON.stringify(result.usage),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.send({ success: true, result });
|
||||||
|
} catch (error: any) {
|
||||||
|
console.error("Audit failed:", error);
|
||||||
|
await leadsService.updateOne(id, { status: "new", comments: `Audit failed: ${error.message}` });
|
||||||
|
res.status(500).send({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post("/audit-email/:id", async (req: any, res: any) => {
|
||||||
|
const { id } = req.params;
|
||||||
|
const leadsService = new ItemsService("leads", { schema: req.schema, accountability: req.accountability });
|
||||||
|
const peopleService = new ItemsService("people", { schema: req.schema, accountability: req.accountability });
|
||||||
|
const mailService = new MailService({ schema: req.schema, accountability: req.accountability });
|
||||||
|
|
||||||
|
try {
|
||||||
|
const lead = await leadsService.readOne(id);
|
||||||
|
if (!lead || !lead.ai_state) return res.status(400).send({ error: "Lead or Audit not ready" });
|
||||||
|
|
||||||
|
let recipientEmail = lead.contact_email;
|
||||||
|
let companyName = lead.company_name;
|
||||||
|
|
||||||
|
if (lead.contact_person) {
|
||||||
|
const person = await peopleService.readOne(lead.contact_person);
|
||||||
|
if (person && person.email) {
|
||||||
|
recipientEmail = person.email;
|
||||||
|
companyName = person.company || lead.company_name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!recipientEmail) return res.status(400).send({ error: "No recipient email found" });
|
||||||
|
|
||||||
|
const auditHighlights = [
|
||||||
|
`Projekt-Typ: ${lead.ai_state.projectType === "website" ? "Website" : "Web App"}`,
|
||||||
|
...(lead.ai_state.sitemap || []).slice(0, 3).map((item: any) => `Potenzial in: ${item.category}`),
|
||||||
|
];
|
||||||
|
|
||||||
|
const html = await render(createElement(SiteAuditTemplate, {
|
||||||
|
companyName: companyName,
|
||||||
|
websiteUrl: lead.website_url,
|
||||||
|
auditHighlights
|
||||||
|
}));
|
||||||
|
|
||||||
|
await mailService.send({
|
||||||
|
to: recipientEmail,
|
||||||
|
subject: `Analyse Ihrer Webpräsenz: ${companyName}`,
|
||||||
|
html
|
||||||
|
});
|
||||||
|
|
||||||
|
await leadsService.updateOne(id, {
|
||||||
|
status: "contacted",
|
||||||
|
last_contacted_at: new Date().toISOString(),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.send({ success: true });
|
||||||
|
} catch (error: any) {
|
||||||
|
console.error("Audit Email failed:", error);
|
||||||
|
res.status(500).send({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post("/estimate/:id", async (req: any, res: any) => {
|
||||||
|
const { id } = req.params;
|
||||||
|
const leadsService = new ItemsService("leads", { schema: req.schema, accountability: req.accountability });
|
||||||
|
|
||||||
|
try {
|
||||||
|
const lead = await leadsService.readOne(id);
|
||||||
|
if (!lead || !lead.ai_state) return res.status(400).send({ error: "Lead or AI state not found" });
|
||||||
|
|
||||||
|
const pdfEngine = new PdfEngine();
|
||||||
|
const filename = `estimate_${id}_${Date.now()}.pdf`;
|
||||||
|
const storageRoot = env.STORAGE_LOCAL_ROOT || "./storage";
|
||||||
|
const outputPath = path.join(storageRoot, filename);
|
||||||
|
|
||||||
|
await pdfEngine.generateEstimatePdf(lead.ai_state, outputPath);
|
||||||
|
|
||||||
|
await leadsService.updateOne(id, {
|
||||||
|
audit_pdf_path: filename,
|
||||||
|
});
|
||||||
|
|
||||||
|
res.send({ success: true, filename });
|
||||||
|
} catch (error: any) {
|
||||||
|
console.error("PDF Generation failed:", error);
|
||||||
|
res.status(500).send({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post("/estimate-email/:id", async (req: any, res: any) => {
|
||||||
|
const { id } = req.params;
|
||||||
|
const leadsService = new ItemsService("leads", { schema: req.schema, accountability: req.accountability });
|
||||||
|
const peopleService = new ItemsService("people", { schema: req.schema, accountability: req.accountability });
|
||||||
|
const mailService = new MailService({ schema: req.schema, accountability: req.accountability });
|
||||||
|
|
||||||
|
try {
|
||||||
|
const lead = await leadsService.readOne(id);
|
||||||
|
if (!lead || !lead.audit_pdf_path) return res.status(400).send({ error: "PDF not generated" });
|
||||||
|
|
||||||
|
let recipientEmail = lead.contact_email;
|
||||||
|
let companyName = lead.company_name;
|
||||||
|
|
||||||
|
if (lead.contact_person) {
|
||||||
|
const person = await peopleService.readOne(lead.contact_person);
|
||||||
|
if (person && person.email) {
|
||||||
|
recipientEmail = person.email;
|
||||||
|
companyName = person.company || lead.company_name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!recipientEmail) return res.status(400).send({ error: "No recipient email found" });
|
||||||
|
|
||||||
|
const html = await render(createElement(ProjectEstimateTemplate, {
|
||||||
|
companyName: companyName,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const storageRoot = env.STORAGE_LOCAL_ROOT || "./storage";
|
||||||
|
const attachmentPath = path.join(storageRoot, lead.audit_pdf_path);
|
||||||
|
|
||||||
|
await mailService.send({
|
||||||
|
to: recipientEmail,
|
||||||
|
subject: `Ihre Projekt-Schätzung: ${companyName}`,
|
||||||
|
html,
|
||||||
|
attachments: [
|
||||||
|
{
|
||||||
|
filename: `Angebot_${companyName}.pdf`,
|
||||||
|
content: fs.readFileSync(attachmentPath)
|
||||||
|
}
|
||||||
|
]
|
||||||
|
});
|
||||||
|
|
||||||
|
await leadsService.updateOne(id, {
|
||||||
|
status: "contacted",
|
||||||
|
last_contacted_at: new Date().toISOString(),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.send({ success: true });
|
||||||
|
} catch (error: any) {
|
||||||
|
console.error("Estimate Email failed:", error);
|
||||||
|
res.status(500).send({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/cli",
|
"name": "@mintel/cli",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
BIN
packages/cms-infra/database/data.db
Normal file → Executable file
BIN
packages/cms-infra/database/data.db
Normal file → Executable file
Binary file not shown.
File diff suppressed because one or more lines are too long
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "acquisition-manager",
|
||||||
|
"description": "Custom High-Fidelity Acquisition Management for Directus",
|
||||||
|
"icon": "account_balance_wallet",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "Acquisition Manager"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because one or more lines are too long
@@ -1,29 +1,30 @@
|
|||||||
{
|
{
|
||||||
"name": "customer-manager",
|
"name": "customer-manager",
|
||||||
"description": "Custom High-Fidelity Customer & Company Management for Directus",
|
"description": "Custom High-Fidelity Customer & Company Management for Directus",
|
||||||
"icon": "supervisor_account",
|
"icon": "supervisor_account",
|
||||||
"version": "1.0.0",
|
"version": "1.7.12",
|
||||||
"keywords": [
|
"type": "module",
|
||||||
"directus",
|
"keywords": [
|
||||||
"directus-extension",
|
"directus",
|
||||||
"directus-extension-module"
|
"directus-extension",
|
||||||
],
|
"directus-extension-module"
|
||||||
"files": [
|
],
|
||||||
"dist"
|
"files": [
|
||||||
],
|
"dist"
|
||||||
"directus:extension": {
|
],
|
||||||
"type": "module",
|
"directus:extension": {
|
||||||
"path": "index.js",
|
"type": "module",
|
||||||
"source": "src/index.ts",
|
"path": "index.js",
|
||||||
"host": "*",
|
"source": "src/index.ts",
|
||||||
"name": "Customer Manager"
|
"host": "*",
|
||||||
},
|
"name": "Customer Manager"
|
||||||
"scripts": {
|
},
|
||||||
"build": "directus-extension build",
|
"scripts": {
|
||||||
"dev": "directus-extension build -w"
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
},
|
"dev": "directus-extension build -w"
|
||||||
"devDependencies": {
|
},
|
||||||
"@directus/extensions-sdk": "11.0.2",
|
"devDependencies": {
|
||||||
"vue": "^3.4.0"
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
}
|
"vue": "^3.4.0"
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,29 +1,30 @@
|
|||||||
{
|
{
|
||||||
"name": "feedback-commander",
|
"name": "feedback-commander",
|
||||||
"description": "Custom High-Fidelity Feedback Management Extension for Directus",
|
"description": "Custom High-Fidelity Feedback Management Extension for Directus",
|
||||||
"icon": "view_kanban",
|
"icon": "view_kanban",
|
||||||
"version": "1.0.0",
|
"version": "1.7.12",
|
||||||
"keywords": [
|
"type": "module",
|
||||||
"directus",
|
"keywords": [
|
||||||
"directus-extension",
|
"directus",
|
||||||
"directus-extension-module"
|
"directus-extension",
|
||||||
],
|
"directus-extension-module"
|
||||||
"files": [
|
],
|
||||||
"index.js"
|
"files": [
|
||||||
],
|
"dist"
|
||||||
"directus:extension": {
|
],
|
||||||
"type": "module",
|
"directus:extension": {
|
||||||
"path": "index.js",
|
"type": "module",
|
||||||
"source": "src/index.ts",
|
"path": "index.js",
|
||||||
"host": "*",
|
"source": "src/index.ts",
|
||||||
"name": "Feedback Commander"
|
"host": "*",
|
||||||
},
|
"name": "Feedback Commander"
|
||||||
"scripts": {
|
},
|
||||||
"build": "directus-extension build",
|
"scripts": {
|
||||||
"dev": "directus-extension build -w"
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
},
|
"dev": "directus-extension build -w"
|
||||||
"devDependencies": {
|
},
|
||||||
"@directus/extensions-sdk": "11.0.2",
|
"devDependencies": {
|
||||||
"vue": "^3.4.0"
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
}
|
"vue": "^3.4.0"
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|||||||
1
packages/cms-infra/extensions/people-manager/index.js
Normal file
1
packages/cms-infra/extensions/people-manager/index.js
Normal file
File diff suppressed because one or more lines are too long
30
packages/cms-infra/extensions/people-manager/package.json
Normal file
30
packages/cms-infra/extensions/people-manager/package.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "people-manager",
|
||||||
|
"description": "Custom High-Fidelity People Management for Directus",
|
||||||
|
"icon": "person",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "People Manager"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,11 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/cms-infra",
|
"name": "@mintel/cms-infra",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"up": "docker compose up -d",
|
"up": "npm run build:extensions && docker compose up -d",
|
||||||
"down": "docker compose down",
|
"down": "docker compose down",
|
||||||
"logs": "docker compose logs -f"
|
"logs": "docker compose logs -f",
|
||||||
|
"build:extensions": "../../scripts/sync-extensions.sh"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
@@ -2,7 +2,8 @@
|
|||||||
"name": "customer-manager",
|
"name": "customer-manager",
|
||||||
"description": "Custom High-Fidelity Customer & Company Management for Directus",
|
"description": "Custom High-Fidelity Customer & Company Management for Directus",
|
||||||
"icon": "supervisor_account",
|
"icon": "supervisor_account",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"directus",
|
"directus",
|
||||||
"directus-extension",
|
"directus-extension",
|
||||||
@@ -19,7 +20,7 @@
|
|||||||
"name": "Customer Manager"
|
"name": "Customer Manager"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "directus-extension build",
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
"dev": "directus-extension build -w"
|
"dev": "directus-extension build -w"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/eslint-config",
|
"name": "@mintel/eslint-config",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
1
packages/feedback-commander/index.js
Normal file
1
packages/feedback-commander/index.js
Normal file
File diff suppressed because one or more lines are too long
@@ -1,8 +1,9 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/extension-feedback-commander",
|
"name": "feedback-commander",
|
||||||
"description": "Custom High-Fidelity Feedback Management Extension for Directus",
|
"description": "Custom High-Fidelity Feedback Management Extension for Directus",
|
||||||
"icon": "view_kanban",
|
"icon": "view_kanban",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"directus",
|
"directus",
|
||||||
"directus-extension",
|
"directus-extension",
|
||||||
@@ -13,13 +14,13 @@
|
|||||||
],
|
],
|
||||||
"directus:extension": {
|
"directus:extension": {
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"path": "dist/index.js",
|
"path": "index.js",
|
||||||
"source": "src/index.ts",
|
"source": "src/index.ts",
|
||||||
"host": "*",
|
"host": "*",
|
||||||
"name": "Feedback Commander"
|
"name": "Feedback Commander"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "directus-extension build",
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
"dev": "directus-extension build -w"
|
"dev": "directus-extension build -w"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/gatekeeper",
|
"name": "@mintel/gatekeeper",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
@@ -11,6 +11,8 @@ export async function GET(req: NextRequest) {
|
|||||||
|
|
||||||
// 1. URL Parameter Bypass (for automated tests/staging)
|
// 1. URL Parameter Bypass (for automated tests/staging)
|
||||||
const originalUrl = req.headers.get("x-forwarded-uri") || "/";
|
const originalUrl = req.headers.get("x-forwarded-uri") || "/";
|
||||||
|
|
||||||
|
console.log(`[Verify] Check: ${originalUrl} | Cookie: ${session ? "Found" : "Missing"}`);
|
||||||
const host =
|
const host =
|
||||||
req.headers.get("x-forwarded-host") || req.headers.get("host") || "";
|
req.headers.get("x-forwarded-host") || req.headers.get("host") || "";
|
||||||
const proto = req.headers.get("x-forwarded-proto") || "https";
|
const proto = req.headers.get("x-forwarded-proto") || "https";
|
||||||
@@ -54,15 +56,17 @@ export async function GET(req: NextRequest) {
|
|||||||
if (session?.value) {
|
if (session?.value) {
|
||||||
if (session.value === password) {
|
if (session.value === password) {
|
||||||
isAuthenticated = true;
|
isAuthenticated = true;
|
||||||
|
console.log(`[Verify] Legacy password match`);
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
const payload = JSON.parse(session.value);
|
const payload = JSON.parse(session.value);
|
||||||
if (payload.identity) {
|
if (payload.identity) {
|
||||||
isAuthenticated = true;
|
isAuthenticated = true;
|
||||||
identity = payload.identity;
|
identity = payload.identity;
|
||||||
|
console.log(`[Verify] Identity verified: ${identity}`);
|
||||||
}
|
}
|
||||||
} catch (_e) {
|
} catch (_e) {
|
||||||
// Fallback or old format
|
console.log(`[Verify] JSON Parse failed for cookie: ${session.value.substring(0, 10)}...`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,8 +17,8 @@ export default async function LoginPage({ searchParams }: LoginPageProps) {
|
|||||||
async function login(formData: FormData) {
|
async function login(formData: FormData) {
|
||||||
"use server";
|
"use server";
|
||||||
|
|
||||||
const email = formData.get("email") as string;
|
const email = (formData.get("email") as string || "").trim();
|
||||||
const password = formData.get("password") as string;
|
const password = (formData.get("password") as string || "").trim();
|
||||||
|
|
||||||
const expectedCode = process.env.GATEKEEPER_PASSWORD || "mintel";
|
const expectedCode = process.env.GATEKEEPER_PASSWORD || "mintel";
|
||||||
const adminEmail = process.env.DIRECTUS_ADMIN_EMAIL;
|
const adminEmail = process.env.DIRECTUS_ADMIN_EMAIL;
|
||||||
@@ -31,19 +31,19 @@ export default async function LoginPage({ searchParams }: LoginPageProps) {
|
|||||||
let userIdentity = "";
|
let userIdentity = "";
|
||||||
let userCompany: any = null;
|
let userCompany: any = null;
|
||||||
|
|
||||||
// 1. Check Global Admin (from ENV)
|
// 1. Check Generic Code (Guest) - High Priority to prevent autofill traps
|
||||||
if (
|
if (password === expectedCode) {
|
||||||
|
userIdentity = "Guest";
|
||||||
|
}
|
||||||
|
// 2. Check Global Admin (from ENV)
|
||||||
|
else if (
|
||||||
adminEmail &&
|
adminEmail &&
|
||||||
adminPassword &&
|
adminPassword &&
|
||||||
email === adminEmail &&
|
email === adminEmail.trim() &&
|
||||||
password === adminPassword
|
password === adminPassword.trim()
|
||||||
) {
|
) {
|
||||||
userIdentity = "Admin";
|
userIdentity = "Admin";
|
||||||
}
|
}
|
||||||
// 2. Check Generic Code (Guest)
|
|
||||||
else if (!email && password === expectedCode) {
|
|
||||||
userIdentity = "Guest";
|
|
||||||
}
|
|
||||||
// 3. Check Lightweight Client Users (dedicated collection)
|
// 3. Check Lightweight Client Users (dedicated collection)
|
||||||
if (email && password && process.env.INFRA_DIRECTUS_URL) {
|
if (email && password && process.env.INFRA_DIRECTUS_URL) {
|
||||||
try {
|
try {
|
||||||
@@ -116,6 +116,7 @@ export default async function LoginPage({ searchParams }: LoginPageProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (userIdentity) {
|
if (userIdentity) {
|
||||||
|
console.log(`[Login] Success: ${userIdentity} | Redirect: ${targetRedirect}`);
|
||||||
const cookieStore = await cookies();
|
const cookieStore = await cookies();
|
||||||
// Store identity in the cookie (simplified for now, ideally signed)
|
// Store identity in the cookie (simplified for now, ideally signed)
|
||||||
const sessionValue = JSON.stringify({
|
const sessionValue = JSON.stringify({
|
||||||
@@ -126,6 +127,8 @@ export default async function LoginPage({ searchParams }: LoginPageProps) {
|
|||||||
|
|
||||||
const isDev = process.env.NODE_ENV === "development";
|
const isDev = process.env.NODE_ENV === "development";
|
||||||
|
|
||||||
|
console.log(`[Login] Setting Cookie: ${authCookieName} | Domain: ${cookieDomain || "Default"}`);
|
||||||
|
|
||||||
cookieStore.set(authCookieName, sessionValue, {
|
cookieStore.set(authCookieName, sessionValue, {
|
||||||
httpOnly: true,
|
httpOnly: true,
|
||||||
secure: !isDev,
|
secure: !isDev,
|
||||||
@@ -136,6 +139,7 @@ export default async function LoginPage({ searchParams }: LoginPageProps) {
|
|||||||
});
|
});
|
||||||
redirect(targetRedirect);
|
redirect(targetRedirect);
|
||||||
} else {
|
} else {
|
||||||
|
console.log(`[Login] Failed for inputs. Redirecting back with error.`);
|
||||||
redirect(`/login?error=1&redirect=${encodeURIComponent(targetRedirect)}`);
|
redirect(`/login?error=1&redirect=${encodeURIComponent(targetRedirect)}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/husky-config",
|
"name": "@mintel/husky-config",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
@@ -1,38 +1,13 @@
|
|||||||
# Step 1: Builder image
|
# Step 1: Base image for Next.js builds
|
||||||
FROM node:20-alpine AS builder
|
FROM node:20-alpine
|
||||||
RUN apk add --no-cache libc6-compat curl
|
RUN apk add --no-cache libc6-compat curl
|
||||||
|
|
||||||
|
# Enable pnpm
|
||||||
|
RUN corepack enable pnpm && \
|
||||||
|
corepack prepare pnpm@10.2.0 --activate
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
RUN corepack enable pnpm
|
|
||||||
|
|
||||||
# Step 2: Install dependencies
|
# Final environment
|
||||||
ENV NPM_TOKEN=placeholder
|
ENV NODE_ENV=production
|
||||||
# Copy manifest files specifically for better layer caching
|
ENV NEXT_TELEMETRY_DISABLED=1
|
||||||
COPY pnpm-lock.yaml pnpm-workspace.yaml package.json .npmrc ./
|
|
||||||
# Copy package manifest files individually to preserve directory structure
|
|
||||||
COPY packages/cli/package.json ./packages/cli/
|
|
||||||
COPY packages/cms-infra/package.json ./packages/cms-infra/
|
|
||||||
COPY packages/customer-manager/package.json ./packages/customer-manager/
|
|
||||||
COPY packages/eslint-config/package.json ./packages/eslint-config/
|
|
||||||
COPY packages/feedback-commander/package.json ./packages/feedback-commander/
|
|
||||||
COPY packages/gatekeeper/package.json ./packages/gatekeeper/
|
|
||||||
COPY packages/husky-config/package.json ./packages/husky-config/
|
|
||||||
COPY packages/infra/package.json ./packages/infra/
|
|
||||||
COPY packages/mail/package.json ./packages/mail/
|
|
||||||
COPY packages/next-config/package.json ./packages/next-config/
|
|
||||||
COPY packages/next-feedback/package.json ./packages/next-feedback/
|
|
||||||
COPY packages/next-observability/package.json ./packages/next-observability/
|
|
||||||
COPY packages/next-utils/package.json ./packages/next-utils/
|
|
||||||
COPY packages/observability/package.json ./packages/observability/
|
|
||||||
COPY packages/tsconfig/package.json ./packages/tsconfig/
|
|
||||||
# packages/ui does not have a package.json
|
|
||||||
|
|
||||||
# Use a secret for NPM_TOKEN and a standardized cache mount
|
|
||||||
RUN --mount=type=cache,id=pnpm,target=/pnpm/store \
|
|
||||||
--mount=type=secret,id=NPM_TOKEN \
|
|
||||||
export NPM_TOKEN=$(cat /run/secrets/NPM_TOKEN) && \
|
|
||||||
pnpm config set store-dir /pnpm/store && \
|
|
||||||
pnpm i --frozen-lockfile
|
|
||||||
|
|
||||||
# Step 3: Build shared packages
|
|
||||||
COPY . .
|
|
||||||
RUN pnpm --filter "./packages/*" -r build
|
|
||||||
|
|||||||
@@ -275,6 +275,10 @@ jobs:
|
|||||||
docker system prune -f --filter "until=24h"
|
docker system prune -f --filter "until=24h"
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
|
- name: 🧹 Post-Deploy Cleanup (Runner)
|
||||||
|
if: always()
|
||||||
|
run: docker builder prune -f --filter "until=1h"
|
||||||
|
|
||||||
# ──────────────────────────────────────────────────────────────────────────────
|
# ──────────────────────────────────────────────────────────────────────────────
|
||||||
# JOB 5: Notifications
|
# JOB 5: Notifications
|
||||||
# ──────────────────────────────────────────────────────────────────────────────
|
# ──────────────────────────────────────────────────────────────────────────────
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/infra",
|
"name": "@mintel/infra",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
set -e
|
set -e
|
||||||
|
|
||||||
# Configuration
|
# Configuration
|
||||||
REGISTRY_DATA="/opt/infra/registry/data/docker/registry/v2"
|
REGISTRY_DATA="/mnt/HC_Volume_104575103/registry-data/docker/registry/v2"
|
||||||
KEEP_TAGS=3
|
KEEP_TAGS=3
|
||||||
|
|
||||||
echo "🏥 Starting Aggressive Registry & Docker Maintenance..."
|
echo "🏥 Starting Aggressive Registry & Docker Maintenance..."
|
||||||
@@ -15,31 +15,26 @@ for repo_dir in "$REGISTRY_DATA/repositories/mintel/"*; do
|
|||||||
if [ -d "$tags_dir" ]; then
|
if [ -d "$tags_dir" ]; then
|
||||||
echo "🔍 Processing repository: mintel/$repo_name"
|
echo "🔍 Processing repository: mintel/$repo_name"
|
||||||
|
|
||||||
# Prune main-* tags
|
# Prune various tag patterns
|
||||||
echo " 📦 Pruning main tags..."
|
PATTERNS=("main-*" "testing-*" "branch-*" "v*" "rc*" "[0-9a-f]*")
|
||||||
main_tags=$(ls -dt "$tags_dir"/main-* 2>/dev/null || true)
|
|
||||||
count=0
|
for pattern in "${PATTERNS[@]}"; do
|
||||||
for tag_path in $main_tags; do
|
echo " 📦 Pruning $pattern tags..."
|
||||||
((++count))
|
tags=$(ls -dt "$tags_dir"/${pattern} 2>/dev/null || true)
|
||||||
if [ $count -gt $KEEP_TAGS ]; then
|
count=0
|
||||||
echo " 🗑️ Deleting old main tag: $(basename "$tag_path")"
|
for tag_path in $tags; do
|
||||||
rm -rf "$tag_path"
|
tag_name=$(basename "$tag_path")
|
||||||
fi
|
if [[ "$tag_name" == "latest" ]]; then continue; fi
|
||||||
|
|
||||||
|
((++count))
|
||||||
|
if [ $count -gt $KEEP_TAGS ]; then
|
||||||
|
echo " 🗑️ Deleting old tag: $tag_name"
|
||||||
|
rm -rf "$tag_path"
|
||||||
|
fi
|
||||||
|
done
|
||||||
done
|
done
|
||||||
|
|
||||||
# Prune version tags (v* and rc*)
|
# Always prune buildcache
|
||||||
echo " 🏷️ Pruning version tags..."
|
|
||||||
version_tags=$(ls -dt "$tags_dir"/v1* 2>/dev/null || true)
|
|
||||||
count=0
|
|
||||||
for tag_path in $version_tags; do
|
|
||||||
((++count))
|
|
||||||
if [ $count -gt $KEEP_TAGS ]; then
|
|
||||||
echo " 🗑️ Deleting old version tag: $(basename "$tag_path")"
|
|
||||||
rm -rf "$tag_path"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
# Always prune buildcache (as it rebuilds quickly)
|
|
||||||
if [ -d "$tags_dir/buildcache" ]; then
|
if [ -d "$tags_dir/buildcache" ]; then
|
||||||
echo " 🧹 Deleting buildcache tag"
|
echo " 🧹 Deleting buildcache tag"
|
||||||
rm -rf "$tags_dir/buildcache"
|
rm -rf "$tags_dir/buildcache"
|
||||||
@@ -48,8 +43,15 @@ for repo_dir in "$REGISTRY_DATA/repositories/mintel/"*; do
|
|||||||
done
|
done
|
||||||
|
|
||||||
# 2. Run Garbage Collection
|
# 2. Run Garbage Collection
|
||||||
echo "♻️ Running Registry Garbage Collection..."
|
echo "♻️ Detecting Registry Container..."
|
||||||
docker exec registry-registry-1 bin/registry garbage-collect /etc/docker/registry/config.yml
|
REGISTRY_CONTAINER=$(docker ps --format "{{.Names}}" | grep registry | head -1 || true)
|
||||||
|
|
||||||
|
if [ -n "$REGISTRY_CONTAINER" ]; then
|
||||||
|
echo "♻️ Running Registry Garbage Collection on $REGISTRY_CONTAINER..."
|
||||||
|
docker exec "$REGISTRY_CONTAINER" bin/registry garbage-collect /etc/docker/registry/config.yml --delete-untagged
|
||||||
|
else
|
||||||
|
echo "⚠️ Registry container not found. Skipping GC."
|
||||||
|
fi
|
||||||
|
|
||||||
# 3. Prune Host Docker resources (Shorter window: 24h)
|
# 3. Prune Host Docker resources (Shorter window: 24h)
|
||||||
echo "🧹 Pruning Host Docker resources..."
|
echo "🧹 Pruning Host Docker resources..."
|
||||||
|
|||||||
93
packages/infra/scripts/wait-for-upstream.sh
Executable file
93
packages/infra/scripts/wait-for-upstream.sh
Executable file
@@ -0,0 +1,93 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# wait-for-upstream.sh
|
||||||
|
# Usage: ./wait-for-upstream.sh <org/repo> <version_tag> [poll_interval_sec]
|
||||||
|
|
||||||
|
REPO=$1
|
||||||
|
TAG=$2
|
||||||
|
INTERVAL=${3:-30}
|
||||||
|
MAX_RETRIES=40 # ~20 minutes default
|
||||||
|
|
||||||
|
if [[ -z "$REPO" || -z "$TAG" ]]; then
|
||||||
|
echo "❌ Error: REPO and TAG are required."
|
||||||
|
echo "Usage: $0 <org/repo> <version_tag>"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -z "$GITEA_TOKEN" ]]; then
|
||||||
|
echo "❌ Error: GITEA_TOKEN is not set."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
GITEA_API="https://git.infra.mintel.me/api/v1"
|
||||||
|
|
||||||
|
echo "🔎 Searching for upstream release $TAG in $REPO..."
|
||||||
|
|
||||||
|
# 1. Get the SHA of the tag to be more precise
|
||||||
|
TAG_INFO=$(curl -s -H "Authorization: token $GITEA_TOKEN" "$GITEA_API/repos/$REPO/tags/$TAG")
|
||||||
|
TARGET_SHA=$(echo "$TAG_INFO" | jq -r '.commit.sha // empty')
|
||||||
|
|
||||||
|
if [[ -z "$TARGET_SHA" || "$TARGET_SHA" == "null" ]]; then
|
||||||
|
echo "⚠️ Warning: Tag $TAG not found yet. Upstream might be lagging."
|
||||||
|
echo " Waiting 15s for tag to appear..."
|
||||||
|
sleep 15
|
||||||
|
TAG_INFO=$(curl -s -H "Authorization: token $GITEA_TOKEN" "$GITEA_API/repos/$REPO/tags/$TAG")
|
||||||
|
TARGET_SHA=$(echo "$TAG_INFO" | jq -r '.commit.sha // empty')
|
||||||
|
|
||||||
|
if [[ -z "$TARGET_SHA" || "$TARGET_SHA" == "null" ]]; then
|
||||||
|
echo "❌ Error: Tag $TAG does not exist in $REPO."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "✅ Target SHA for $TAG is $TARGET_SHA"
|
||||||
|
|
||||||
|
# 2. Find the run for the specific SHA
|
||||||
|
# We list recent runs and filter by head_sha
|
||||||
|
RUN_QUERY=$(curl -s -H "Authorization: token $GITEA_TOKEN" "$GITEA_API/repos/$REPO/actions/runs?limit=30")
|
||||||
|
RUN_ID=$(echo "$RUN_QUERY" | jq -r ".workflow_runs[] | select(.head_sha == \"$TARGET_SHA\") | .id" | head -n 1)
|
||||||
|
|
||||||
|
if [[ -z "$RUN_ID" || "$RUN_ID" == "null" ]]; then
|
||||||
|
echo "ℹ️ No recent action run found for SHA $TARGET_SHA yet."
|
||||||
|
echo " Checking if we should wait or if it was already successful..."
|
||||||
|
|
||||||
|
# Fallback: wait a bit more for new tags
|
||||||
|
echo "⏳ waiting for run to appear..."
|
||||||
|
sleep 20
|
||||||
|
RUN_QUERY=$(curl -s -H "Authorization: token $GITEA_TOKEN" "$GITEA_API/repos/$REPO/actions/runs?limit=30")
|
||||||
|
RUN_ID=$(echo "$RUN_QUERY" | jq -r ".workflow_runs[] | select(.head_sha == \"$TARGET_SHA\") | .id" | head -n 1)
|
||||||
|
|
||||||
|
if [[ -z "$RUN_ID" || "$RUN_ID" == "null" ]]; then
|
||||||
|
echo "✅ No run found but Tag exists. Assuming manual release or already completed. Proceeding."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "⏳ Waiting for upstream run $RUN_ID status..."
|
||||||
|
|
||||||
|
RETRY_COUNT=0
|
||||||
|
while [ $RETRY_COUNT -lt $MAX_RETRIES ]; do
|
||||||
|
STATUS_QUERY=$(curl -s -H "Authorization: token $GITEA_TOKEN" "$GITEA_API/repos/$REPO/actions/runs/$RUN_ID")
|
||||||
|
STATUS=$(echo "$STATUS_QUERY" | jq -r '.status')
|
||||||
|
CONCLUSION=$(echo "$STATUS_QUERY" | jq -r '.conclusion')
|
||||||
|
|
||||||
|
echo " - Current Status: $STATUS (Conclusion: $CONCLUSION)"
|
||||||
|
|
||||||
|
if [[ "$STATUS" == "success" || "$CONCLUSION" == "success" ]]; then
|
||||||
|
echo "✅ Upstream release $TAG is READY."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$STATUS" == "failure" || "$CONCLUSION" == "failure" || "$CONCLUSION" == "cancelled" ]]; then
|
||||||
|
echo "❌ Error: Upstream release $TAG FAILED or was CANCELLED."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo " - Still working... waiting $INTERVAL seconds (Attempt $((RETRY_COUNT+1))/$MAX_RETRIES)"
|
||||||
|
sleep $INTERVAL
|
||||||
|
RETRY_COUNT=$((RETRY_COUNT+1))
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "❌ Error: Timeout waiting for upstream release $TAG."
|
||||||
|
exit 1
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/mail",
|
"name": "@mintel/mail",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"private": false,
|
"private": false,
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
|
|||||||
@@ -22,3 +22,6 @@ export * from "./layouts/ClientLayout";
|
|||||||
// Export Templates
|
// Export Templates
|
||||||
export * from "./templates/ContactFormNotification";
|
export * from "./templates/ContactFormNotification";
|
||||||
export * from "./templates/ConfirmationMessage";
|
export * from "./templates/ConfirmationMessage";
|
||||||
|
export * from "./templates/FollowUpTemplate";
|
||||||
|
export * from "./templates/ProjectEstimateTemplate";
|
||||||
|
export * from "./templates/SiteAuditTemplate";
|
||||||
|
|||||||
88
packages/mail/src/templates/FollowUpTemplate.tsx
Normal file
88
packages/mail/src/templates/FollowUpTemplate.tsx
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
import * as React from "react";
|
||||||
|
import { Heading, Text, Button } from "@react-email/components";
|
||||||
|
import { MintelLayout } from "../layouts/MintelLayout";
|
||||||
|
|
||||||
|
export interface FollowUpTemplateProps {
|
||||||
|
companyName: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const FollowUpTemplate = ({
|
||||||
|
companyName,
|
||||||
|
}: FollowUpTemplateProps) => {
|
||||||
|
const preview = `Kurzes Follow-up: ${companyName}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MintelLayout preview={preview}>
|
||||||
|
<Heading style={h1}>Kurzes Follow-up</Heading>
|
||||||
|
<Text style={intro}>
|
||||||
|
Hallo noch einmal,<br /><br />
|
||||||
|
ich wollte mich nur kurz erkundigen, ob Sie bereits Zeit hatten,
|
||||||
|
einen Blick auf das Audit Ihrer Website zu werfen, das ich Ihnen
|
||||||
|
vor ein paar Tagen gesendet habe.
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Text style={bodyText}>
|
||||||
|
Vielleicht passt es ja diese Woche für ein kurzes, unverbindliches
|
||||||
|
Telefonat, um die Punkte gemeinsam durchzugehen?
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
style={button}
|
||||||
|
href="https://calendly.com/mintel-me/intro"
|
||||||
|
>
|
||||||
|
Treffen vereinbaren
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<Text style={footerText}>
|
||||||
|
Beste Grüße,<br />
|
||||||
|
<strong>Marc Mintel</strong>
|
||||||
|
</Text>
|
||||||
|
</MintelLayout>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FollowUpTemplate;
|
||||||
|
|
||||||
|
const h1 = {
|
||||||
|
fontSize: "28px",
|
||||||
|
fontWeight: "900",
|
||||||
|
margin: "0 0 24px",
|
||||||
|
color: "#ffffff",
|
||||||
|
letterSpacing: "-0.04em",
|
||||||
|
};
|
||||||
|
|
||||||
|
const intro = {
|
||||||
|
fontSize: "16px",
|
||||||
|
lineHeight: "24px",
|
||||||
|
color: "#cccccc",
|
||||||
|
margin: "0 0 24px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const bodyText = {
|
||||||
|
fontSize: "16px",
|
||||||
|
lineHeight: "24px",
|
||||||
|
color: "#888888",
|
||||||
|
margin: "0 0 32px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const button = {
|
||||||
|
backgroundColor: "#333333",
|
||||||
|
borderRadius: "0",
|
||||||
|
color: "#ffffff",
|
||||||
|
fontSize: "13px",
|
||||||
|
fontWeight: "bold",
|
||||||
|
textDecoration: "none",
|
||||||
|
textAlign: "center" as const,
|
||||||
|
display: "inline-block",
|
||||||
|
padding: "12px 24px",
|
||||||
|
textTransform: "uppercase" as const,
|
||||||
|
letterSpacing: "0.1em",
|
||||||
|
border: "1px solid #444444",
|
||||||
|
};
|
||||||
|
|
||||||
|
const footerText = {
|
||||||
|
fontSize: "14px",
|
||||||
|
color: "#666666",
|
||||||
|
lineHeight: "20px",
|
||||||
|
marginTop: "48px",
|
||||||
|
};
|
||||||
86
packages/mail/src/templates/ProjectEstimateTemplate.tsx
Normal file
86
packages/mail/src/templates/ProjectEstimateTemplate.tsx
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
import * as React from "react";
|
||||||
|
import { Heading, Text, Section } from "@react-email/components";
|
||||||
|
import { MintelLayout } from "../layouts/MintelLayout";
|
||||||
|
|
||||||
|
export interface ProjectEstimateTemplateProps {
|
||||||
|
companyName: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ProjectEstimateTemplate = ({
|
||||||
|
companyName,
|
||||||
|
}: ProjectEstimateTemplateProps) => {
|
||||||
|
const preview = `Ihre personalisierte Projekt-Schätzung: ${companyName}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MintelLayout preview={preview}>
|
||||||
|
<Heading style={h1}>Ihre Projekt-Schätzung</Heading>
|
||||||
|
<Text style={intro}>
|
||||||
|
Hallo {companyName},<br /><br />
|
||||||
|
vielen Dank für unser Gespräch. Wie versprochen sende ich Ihnen hiermit
|
||||||
|
die detaillierte Schätzung für Ihre neue digitale Webpräsenz.
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Section style={infoBox}>
|
||||||
|
<Text style={infoText}>
|
||||||
|
Im Anhang finden Sie das PDF-Dokument mit allen Positionen,
|
||||||
|
Umfängen und dem strategischen Ausblick.
|
||||||
|
</Text>
|
||||||
|
</Section>
|
||||||
|
|
||||||
|
<Text style={bodyText}>
|
||||||
|
Ich freue mich auf Ihr Feedback und stehe für Rückfragen jederzeit
|
||||||
|
zur Verfügung.
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Text style={footerText}>
|
||||||
|
Beste Grüße,<br />
|
||||||
|
<strong>Marc Mintel</strong>
|
||||||
|
</Text>
|
||||||
|
</MintelLayout>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ProjectEstimateTemplate;
|
||||||
|
|
||||||
|
const h1 = {
|
||||||
|
fontSize: "28px",
|
||||||
|
fontWeight: "900",
|
||||||
|
margin: "0 0 24px",
|
||||||
|
color: "#ffffff",
|
||||||
|
letterSpacing: "-0.04em",
|
||||||
|
};
|
||||||
|
|
||||||
|
const intro = {
|
||||||
|
fontSize: "16px",
|
||||||
|
lineHeight: "24px",
|
||||||
|
color: "#cccccc",
|
||||||
|
margin: "0 0 24px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const infoBox = {
|
||||||
|
backgroundColor: "#0f172a",
|
||||||
|
padding: "24px",
|
||||||
|
borderLeft: "4px solid #ffffff",
|
||||||
|
marginBottom: "32px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const infoText = {
|
||||||
|
fontSize: "15px",
|
||||||
|
color: "#ffffff",
|
||||||
|
margin: "0",
|
||||||
|
lineHeight: "1.5",
|
||||||
|
};
|
||||||
|
|
||||||
|
const bodyText = {
|
||||||
|
fontSize: "16px",
|
||||||
|
lineHeight: "24px",
|
||||||
|
color: "#888888",
|
||||||
|
margin: "0 0 32px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const footerText = {
|
||||||
|
fontSize: "14px",
|
||||||
|
color: "#666666",
|
||||||
|
lineHeight: "20px",
|
||||||
|
marginTop: "48px",
|
||||||
|
};
|
||||||
146
packages/mail/src/templates/SiteAuditTemplate.tsx
Normal file
146
packages/mail/src/templates/SiteAuditTemplate.tsx
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
import * as React from "react";
|
||||||
|
import { Heading, Section, Text, Button, Link } from "@react-email/components";
|
||||||
|
import { MintelLayout } from "../layouts/MintelLayout";
|
||||||
|
|
||||||
|
export interface SiteAuditTemplateProps {
|
||||||
|
companyName: string;
|
||||||
|
auditHighlights: string[];
|
||||||
|
websiteUrl: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SiteAuditTemplate = ({
|
||||||
|
companyName,
|
||||||
|
auditHighlights,
|
||||||
|
websiteUrl,
|
||||||
|
}: SiteAuditTemplateProps) => {
|
||||||
|
const preview = `Analyse Ihrer Webpräsenz: ${companyName}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MintelLayout preview={preview}>
|
||||||
|
<Heading style={h1}>Analyse Ihrer Webpräsenz</Heading>
|
||||||
|
<Text style={intro}>
|
||||||
|
Hallo {companyName},<br /><br />
|
||||||
|
ich habe mir Ihre aktuelle Website ({websiteUrl}) im Detail angeschaut und
|
||||||
|
einige technische sowie strategische Potenziale identifiziert.
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Section style={auditContainer}>
|
||||||
|
<Heading as="h2" style={h2}>Audit Highlights</Heading>
|
||||||
|
{auditHighlights.map((highlight, i) => (
|
||||||
|
<div key={i} style={highlightRow}>
|
||||||
|
<div style={bullet} />
|
||||||
|
<Text style={highlightText}>{highlight}</Text>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</Section>
|
||||||
|
|
||||||
|
<Text style={bodyText}>
|
||||||
|
In der heutigen digitalen Landschaft ist eine performante und strategisch
|
||||||
|
ausgerichtete Website kein Luxus mehr, sondern das Fundament für
|
||||||
|
nachhaltiges Wachstum.
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Section style={ctaSection}>
|
||||||
|
<Button
|
||||||
|
style={button}
|
||||||
|
href={`mailto:marc@mintel.me?subject=Feedback zum Audit: ${companyName}`}
|
||||||
|
>
|
||||||
|
Audit gemeinsam besprechen
|
||||||
|
</Button>
|
||||||
|
</Section>
|
||||||
|
|
||||||
|
<Text style={footerText}>
|
||||||
|
Beste Grüße,<br />
|
||||||
|
<strong>Marc Mintel</strong><br />
|
||||||
|
Digitaler Architekt
|
||||||
|
</Text>
|
||||||
|
</MintelLayout>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SiteAuditTemplate;
|
||||||
|
|
||||||
|
const h1 = {
|
||||||
|
fontSize: "28px",
|
||||||
|
fontWeight: "900",
|
||||||
|
margin: "0 0 24px",
|
||||||
|
color: "#ffffff",
|
||||||
|
letterSpacing: "-0.04em",
|
||||||
|
};
|
||||||
|
|
||||||
|
const h2 = {
|
||||||
|
fontSize: "14px",
|
||||||
|
fontWeight: "900",
|
||||||
|
textTransform: "uppercase" as const,
|
||||||
|
color: "#444444",
|
||||||
|
margin: "0 0 16px",
|
||||||
|
letterSpacing: "0.1em",
|
||||||
|
};
|
||||||
|
|
||||||
|
const intro = {
|
||||||
|
fontSize: "16px",
|
||||||
|
lineHeight: "24px",
|
||||||
|
color: "#cccccc",
|
||||||
|
margin: "0 0 32px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const auditContainer = {
|
||||||
|
backgroundColor: "#151515",
|
||||||
|
padding: "32px",
|
||||||
|
borderRadius: "8px",
|
||||||
|
marginBottom: "32px",
|
||||||
|
border: "1px solid #222222",
|
||||||
|
};
|
||||||
|
|
||||||
|
const highlightRow = {
|
||||||
|
display: "flex" as const,
|
||||||
|
alignItems: "flex-start" as const,
|
||||||
|
marginBottom: "12px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const bullet = {
|
||||||
|
width: "6px",
|
||||||
|
height: "6px",
|
||||||
|
backgroundColor: "#4CAF50",
|
||||||
|
marginTop: "8px",
|
||||||
|
marginRight: "12px",
|
||||||
|
flexShrink: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
const highlightText = {
|
||||||
|
fontSize: "15px",
|
||||||
|
color: "#ffffff",
|
||||||
|
margin: "0",
|
||||||
|
};
|
||||||
|
|
||||||
|
const bodyText = {
|
||||||
|
fontSize: "16px",
|
||||||
|
lineHeight: "24px",
|
||||||
|
color: "#888888",
|
||||||
|
margin: "0 0 32px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const ctaSection = {
|
||||||
|
textAlign: "center" as const,
|
||||||
|
marginBottom: "48px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const button = {
|
||||||
|
backgroundColor: "#ffffff",
|
||||||
|
borderRadius: "0",
|
||||||
|
color: "#000000",
|
||||||
|
fontSize: "14px",
|
||||||
|
fontWeight: "bold",
|
||||||
|
textDecoration: "none",
|
||||||
|
textAlign: "center" as const,
|
||||||
|
display: "inline-block",
|
||||||
|
padding: "16px 32px",
|
||||||
|
textTransform: "uppercase" as const,
|
||||||
|
letterSpacing: "0.1em",
|
||||||
|
};
|
||||||
|
|
||||||
|
const footerText = {
|
||||||
|
fontSize: "14px",
|
||||||
|
color: "#666666",
|
||||||
|
lineHeight: "20px",
|
||||||
|
};
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/next-config",
|
"name": "@mintel/next-config",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/next-feedback",
|
"name": "@mintel/next-feedback",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/next-observability",
|
"name": "@mintel/next-observability",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/next-utils",
|
"name": "@mintel/next-utils",
|
||||||
"version": "1.7.4",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
},
|
},
|
||||||
|
"type": "module",
|
||||||
"main": "./dist/index.js",
|
"main": "./dist/index.js",
|
||||||
"module": "./dist/index.mjs",
|
|
||||||
"types": "./dist/index.d.ts",
|
"types": "./dist/index.d.ts",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "tsup src/index.ts --format cjs,esm --dts",
|
"build": "tsup src/index.ts --format esm --dts --clean",
|
||||||
"dev": "tsup src/index.ts --format cjs,esm --watch --dts",
|
"dev": "tsup src/index.ts --format esm --watch --dts",
|
||||||
"lint": "eslint src/",
|
"lint": "eslint src/",
|
||||||
"test": "vitest run"
|
"test": "vitest run"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -7,20 +7,21 @@ import {
|
|||||||
AuthenticationClient,
|
AuthenticationClient,
|
||||||
} from "@directus/sdk";
|
} from "@directus/sdk";
|
||||||
|
|
||||||
export type MintelDirectusClient = DirectusClient<any> &
|
export type MintelDirectusClient<Schema extends object = any> =
|
||||||
RestClient<any> &
|
DirectusClient<Schema> & RestClient<Schema> & AuthenticationClient<Schema>;
|
||||||
AuthenticationClient<any>;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a Directus client configured with Mintel standards.
|
* Creates a Directus client configured with Mintel standards.
|
||||||
* Automatically handles internal vs. external URLs based on environment.
|
* Automatically handles internal vs. external URLs based on environment.
|
||||||
*/
|
*/
|
||||||
export function createMintelDirectusClient(url?: string): MintelDirectusClient {
|
export function createMintelDirectusClient<Schema extends object = any>(
|
||||||
|
url?: string,
|
||||||
|
): MintelDirectusClient<Schema> {
|
||||||
const isServer = typeof window === "undefined";
|
const isServer = typeof window === "undefined";
|
||||||
|
|
||||||
// 1. If an explicit URL is provided, use it.
|
// 1. If an explicit URL is provided, use it.
|
||||||
if (url) {
|
if (url) {
|
||||||
return createDirectus(url).with(rest()).with(authentication());
|
return createDirectus<Schema>(url).with(rest()).with(authentication());
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. On server: Prioritize INTERNAL_DIRECTUS_URL, fallback to DIRECTUS_URL
|
// 2. On server: Prioritize INTERNAL_DIRECTUS_URL, fallback to DIRECTUS_URL
|
||||||
@@ -29,7 +30,9 @@ export function createMintelDirectusClient(url?: string): MintelDirectusClient {
|
|||||||
process.env.INTERNAL_DIRECTUS_URL ||
|
process.env.INTERNAL_DIRECTUS_URL ||
|
||||||
process.env.DIRECTUS_URL ||
|
process.env.DIRECTUS_URL ||
|
||||||
"http://localhost:8055";
|
"http://localhost:8055";
|
||||||
return createDirectus(directusUrl).with(rest()).with(authentication());
|
return createDirectus<Schema>(directusUrl)
|
||||||
|
.with(rest())
|
||||||
|
.with(authentication());
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. In browser: Use a proxy path if we are on a different origin,
|
// 3. In browser: Use a proxy path if we are on a different origin,
|
||||||
@@ -40,7 +43,7 @@ export function createMintelDirectusClient(url?: string): MintelDirectusClient {
|
|||||||
? `${window.location.origin}${proxyPath}`
|
? `${window.location.origin}${proxyPath}`
|
||||||
: proxyPath;
|
: proxyPath;
|
||||||
|
|
||||||
return createDirectus(browserUrl).with(rest()).with(authentication());
|
return createDirectus<Schema>(browserUrl).with(rest()).with(authentication());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -50,16 +50,51 @@ export const mintelEnvSchema = {
|
|||||||
INTERNAL_DIRECTUS_URL: z.string().url().optional(),
|
INTERNAL_DIRECTUS_URL: z.string().url().optional(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Standard Mintel refinements for environment variables.
|
||||||
|
* Enforces mandatory requirements for non-development environments.
|
||||||
|
*/
|
||||||
|
export const withMintelRefinements = <T extends z.ZodTypeAny>(schema: T) => {
|
||||||
|
return schema.superRefine((data: any, ctx) => {
|
||||||
|
const skipValidation =
|
||||||
|
process.env.SKIP_ENV_VALIDATION === "true" ||
|
||||||
|
process.env.SKIP_RUNTIME_ENV_VALIDATION === "true";
|
||||||
|
|
||||||
|
if (skipValidation) return;
|
||||||
|
|
||||||
|
const target = data.TARGET || data.NEXT_PUBLIC_TARGET || "development";
|
||||||
|
|
||||||
|
// Strict validation for non-development environments
|
||||||
|
if (target !== "development") {
|
||||||
|
if (!data.MAIL_HOST) {
|
||||||
|
ctx.addIssue({
|
||||||
|
code: z.ZodIssueCode.custom,
|
||||||
|
message: "MAIL_HOST is required in non-development environments",
|
||||||
|
path: ["MAIL_HOST"],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export type MintelEnv<T extends z.ZodRawShape = Record<string, never>> =
|
||||||
|
z.infer<
|
||||||
|
ReturnType<
|
||||||
|
typeof withMintelRefinements<z.ZodObject<typeof mintelEnvSchema & T>>
|
||||||
|
>
|
||||||
|
>;
|
||||||
|
|
||||||
export function validateMintelEnv<
|
export function validateMintelEnv<
|
||||||
T extends z.ZodRawShape = Record<string, never>,
|
T extends z.ZodRawShape = Record<string, never>,
|
||||||
>(
|
>(schemaExtension: T = {} as T): MintelEnv<T> {
|
||||||
schemaExtension: T = {} as T,
|
const fullSchema = withMintelRefinements(
|
||||||
): z.infer<z.ZodObject<typeof mintelEnvSchema & T>> {
|
z.object(mintelEnvSchema).extend(schemaExtension),
|
||||||
const fullSchema = z.object(mintelEnvSchema).extend(schemaExtension);
|
);
|
||||||
|
|
||||||
const isBuildTime =
|
const isBuildTime =
|
||||||
process.env.NEXT_PHASE === "phase-production-build" ||
|
process.env.NEXT_PHASE === "phase-production-build" ||
|
||||||
process.env.SKIP_ENV_VALIDATION === "true";
|
process.env.SKIP_ENV_VALIDATION === "true" ||
|
||||||
|
process.env.SKIP_RUNTIME_ENV_VALIDATION === "true";
|
||||||
|
|
||||||
const result = fullSchema.safeParse(process.env);
|
const result = fullSchema.safeParse(process.env);
|
||||||
|
|
||||||
@@ -68,7 +103,7 @@ export function validateMintelEnv<
|
|||||||
console.warn(
|
console.warn(
|
||||||
"⚠️ Some environment variables are missing during build, but skipping strict validation.",
|
"⚠️ Some environment variables are missing during build, but skipping strict validation.",
|
||||||
);
|
);
|
||||||
// Return process.env casted to ensure types match for the full schema
|
// Return process.env casted to the full schema type to unblock builds
|
||||||
return process.env as unknown as z.infer<typeof fullSchema>;
|
return process.env as unknown as z.infer<typeof fullSchema>;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -79,5 +114,5 @@ export function validateMintelEnv<
|
|||||||
throw new Error("Invalid environment variables");
|
throw new Error("Invalid environment variables");
|
||||||
}
|
}
|
||||||
|
|
||||||
return result.data as z.infer<typeof fullSchema>;
|
return result.data as MintelEnv<T>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/observability",
|
"name": "@mintel/observability",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
57
packages/pdf-library/build.mjs
Normal file
57
packages/pdf-library/build.mjs
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
import { build } from 'esbuild';
|
||||||
|
import { resolve, dirname } from 'path';
|
||||||
|
import { mkdirSync } from 'fs';
|
||||||
|
import { fileURLToPath } from 'url';
|
||||||
|
|
||||||
|
const __filename = fileURLToPath(import.meta.url);
|
||||||
|
const __dirname = dirname(__filename);
|
||||||
|
|
||||||
|
const entryPoints = [
|
||||||
|
resolve(__dirname, 'src/index.ts'),
|
||||||
|
resolve(__dirname, 'src/server.ts')
|
||||||
|
];
|
||||||
|
|
||||||
|
try {
|
||||||
|
mkdirSync(resolve(__dirname, 'dist'), { recursive: true });
|
||||||
|
} catch (e) { }
|
||||||
|
|
||||||
|
console.log(`Building entry points...`);
|
||||||
|
|
||||||
|
build({
|
||||||
|
entryPoints: entryPoints,
|
||||||
|
bundle: true,
|
||||||
|
platform: 'node',
|
||||||
|
target: 'node18',
|
||||||
|
outdir: resolve(__dirname, 'dist'),
|
||||||
|
format: 'esm',
|
||||||
|
jsx: 'automatic',
|
||||||
|
loader: {
|
||||||
|
'.tsx': 'tsx',
|
||||||
|
'.ts': 'ts',
|
||||||
|
'.js': 'js',
|
||||||
|
},
|
||||||
|
external: ["@react-pdf/renderer", "react", "react-dom", "jsdom", "jsdom/*", "jquery", "jquery/*", "canvas", "fs", "path", "os", "http", "https", "zlib", "stream", "util", "url", "net", "tls", "crypto"],
|
||||||
|
plugins: [{
|
||||||
|
name: 'mock-canvas',
|
||||||
|
setup(build) {
|
||||||
|
build.onResolve({ filter: /^canvas/ }, args => ({ path: args.path, namespace: 'mock-canvas' }));
|
||||||
|
build.onLoad({ filter: /.*/, namespace: 'mock-canvas' }, () => ({ contents: 'export default {};', loader: 'js' }));
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
name: 'mock-jsdom',
|
||||||
|
setup(build) {
|
||||||
|
build.onResolve({ filter: /^jsdom/ }, args => ({ path: args.path, namespace: 'mock-jsdom' }));
|
||||||
|
build.onLoad({ filter: /.*/, namespace: 'mock-jsdom' }, () => ({ contents: 'export default {};', loader: 'js' }));
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
}).then(() => {
|
||||||
|
console.log("Build succeeded!");
|
||||||
|
}).catch((e) => {
|
||||||
|
if (e.errors) {
|
||||||
|
console.error("Build failed with errors:");
|
||||||
|
e.errors.forEach(err => console.error(` ${err.text} at ${err.location?.file}:${err.location?.line}`));
|
||||||
|
} else {
|
||||||
|
console.error("Build failed:", e);
|
||||||
|
}
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
38
packages/pdf-library/package.json
Normal file
38
packages/pdf-library/package.json
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
{
|
||||||
|
"name": "@mintel/pdf",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"main": "dist/index.js",
|
||||||
|
"module": "dist/index.js",
|
||||||
|
"types": "dist/index.d.ts",
|
||||||
|
"exports": {
|
||||||
|
".": {
|
||||||
|
"types": "./dist/index.d.ts",
|
||||||
|
"import": "./dist/index.js",
|
||||||
|
"default": "./dist/index.js"
|
||||||
|
},
|
||||||
|
"./server": {
|
||||||
|
"types": "./dist/server.d.ts",
|
||||||
|
"import": "./dist/server.js",
|
||||||
|
"default": "./dist/server.js"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "node build.mjs",
|
||||||
|
"dev": "node build.mjs --watch"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"esbuild": "^0.25.0",
|
||||||
|
"typescript": "^5.6.3"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@crawlee/cheerio": "^3.16.0",
|
||||||
|
"@mintel/mail": "workspace:*",
|
||||||
|
"@react-pdf/renderer": "^4.3.0",
|
||||||
|
"axios": "^1.7.9",
|
||||||
|
"cheerio": "^1.0.0",
|
||||||
|
"react": "^19.2.4",
|
||||||
|
"react-dom": "^19.2.4"
|
||||||
|
}
|
||||||
|
}
|
||||||
241
packages/pdf-library/src/components/AgbsPDF.tsx
Normal file
241
packages/pdf-library/src/components/AgbsPDF.tsx
Normal file
@@ -0,0 +1,241 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
Page as PDFPage,
|
||||||
|
Text as PDFText,
|
||||||
|
View as PDFView,
|
||||||
|
StyleSheet as PDFStyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import {
|
||||||
|
pdfStyles,
|
||||||
|
Header,
|
||||||
|
Footer,
|
||||||
|
FoldingMarks,
|
||||||
|
DocumentTitle,
|
||||||
|
} from "./pdf/SharedUI.js";
|
||||||
|
import { SimpleLayout } from "./pdf/SimpleLayout.js";
|
||||||
|
|
||||||
|
const localStyles = PDFStyleSheet.create({
|
||||||
|
sectionContainer: {
|
||||||
|
marginTop: 0,
|
||||||
|
},
|
||||||
|
agbSection: {
|
||||||
|
marginBottom: 20,
|
||||||
|
},
|
||||||
|
labelRow: {
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "baseline",
|
||||||
|
marginBottom: 6,
|
||||||
|
},
|
||||||
|
monoNumber: {
|
||||||
|
fontSize: 7,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: "#94a3b8",
|
||||||
|
letterSpacing: 2,
|
||||||
|
width: 25,
|
||||||
|
},
|
||||||
|
sectionTitle: {
|
||||||
|
fontSize: 9,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: "#000000",
|
||||||
|
textTransform: "uppercase",
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
},
|
||||||
|
officialText: {
|
||||||
|
fontSize: 8,
|
||||||
|
lineHeight: 1.5,
|
||||||
|
color: "#334155",
|
||||||
|
textAlign: "justify",
|
||||||
|
paddingLeft: 25,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const AGBSection = ({
|
||||||
|
index,
|
||||||
|
title,
|
||||||
|
children,
|
||||||
|
}: {
|
||||||
|
index: string;
|
||||||
|
title: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
}) => (
|
||||||
|
<PDFView style={localStyles.agbSection} wrap={false}>
|
||||||
|
<PDFView style={localStyles.labelRow}>
|
||||||
|
<PDFText style={localStyles.monoNumber}>{index}</PDFText>
|
||||||
|
<PDFText style={localStyles.sectionTitle}>{title}</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
<PDFText style={localStyles.officialText}>{children}</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
|
||||||
|
interface AgbsPDFProps {
|
||||||
|
headerIcon?: string;
|
||||||
|
footerLogo?: string;
|
||||||
|
mode?: "estimation" | "full";
|
||||||
|
}
|
||||||
|
|
||||||
|
export const AgbsPDF = ({
|
||||||
|
headerIcon,
|
||||||
|
footerLogo,
|
||||||
|
mode = "full",
|
||||||
|
}: AgbsPDFProps) => {
|
||||||
|
const date = new Date().toLocaleDateString("de-DE", {
|
||||||
|
year: "numeric",
|
||||||
|
month: "long",
|
||||||
|
day: "numeric",
|
||||||
|
});
|
||||||
|
|
||||||
|
const companyData = {
|
||||||
|
name: "Marc Mintel",
|
||||||
|
address1: "Georg-Meistermann-Straße 7",
|
||||||
|
address2: "54586 Schüller",
|
||||||
|
ustId: "DE367588065",
|
||||||
|
};
|
||||||
|
|
||||||
|
const bankData = {
|
||||||
|
name: "N26",
|
||||||
|
bic: "NTSBDEB1XXX",
|
||||||
|
iban: "DE50 1001 1001 2620 4328 65",
|
||||||
|
};
|
||||||
|
|
||||||
|
const content = (
|
||||||
|
<>
|
||||||
|
<DocumentTitle
|
||||||
|
title="Allgemeine Geschäftsbedingungen"
|
||||||
|
subLines={[`Stand: ${date}`]}
|
||||||
|
/>
|
||||||
|
<PDFView style={localStyles.sectionContainer}>
|
||||||
|
<AGBSection index="01" title="Geltungsbereich">
|
||||||
|
Diese Allgemeinen Geschäftsbedingungen gelten für alle Verträge
|
||||||
|
zwischen Marc Mintel (nachfolgend „Auftragnehmer“) und dem jeweiligen
|
||||||
|
Kunden (nachfolgend „Auftraggeber“). Abweichende oder ergänzende
|
||||||
|
Bedingungen des Auftraggebers werden nicht Vertragsbestandteil, auch
|
||||||
|
wenn ihrer Geltung nicht ausdrücklich widersprochen wird.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="02" title="Vertragsgegenstand">
|
||||||
|
Der Auftragnehmer erbringt Dienstleistungen im Bereich:
|
||||||
|
Webentwicklung, technische Umsetzung digitaler Systeme, Funktionen,
|
||||||
|
Schnittstellen und Automatisierungen sowie Hosting, Betrieb und
|
||||||
|
Wartung, sofern ausdrücklich vereinbard. Der Auftragnehmer schuldet
|
||||||
|
ausschließlich die vereinbarte technische Leistung, nicht jedoch einen
|
||||||
|
wirtschaftlichen Erfolg, bestimmte Umsätze, Conversions, Reichweiten,
|
||||||
|
Suchmaschinen-Rankings oder rechtliche Ergebnisse.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="03" title="Mitwirkungspflichten des Auftraggebers">
|
||||||
|
Der Auftraggeber verpflichtet sich, alle zur Leistungserbringung
|
||||||
|
erforderlichen Inhalte, Informationen, Zugänge und Entscheidungen
|
||||||
|
rechtzeitig, vollständig und korrekt bereitzustellen. Hierzu zählen
|
||||||
|
insbesondere Texte, Bilder, Videos, Produktdaten, Freigaben, Feedback,
|
||||||
|
Zugangsdaten sowie rechtlich erforderliche Inhalte (z. B. Impressum,
|
||||||
|
DSGVO). Verzögerungen oder Unterlassungen führen zu Verschiebungen
|
||||||
|
aller Termine ohne Schadensersatzanspruch.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="04" title="Ausführungs- und Bearbeitungszeiten">
|
||||||
|
Angegebene Bearbeitungszeiten sind unverbindliche Schätzungen, keine
|
||||||
|
garantierten Fristen. Fixe Termine oder Deadlines gelten nur, wenn sie
|
||||||
|
ausdrücklich schriftlich als verbindlich vereinbart wurden.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="05" title="Abnahme">
|
||||||
|
Die Leistung gilt als abgenommen, wenn der Auftraggeber sie produktiv
|
||||||
|
nutzt oder innerhalb von 7 Tagen nach Bereitstellung keine
|
||||||
|
wesentlichen Mängel angezeigt werden. Optische Abweichungen,
|
||||||
|
Geschmacksfragen oder subjektive Einschätzungen stellen keine Mängel
|
||||||
|
dar.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="06" title="Haftung">
|
||||||
|
Der Auftragnehmer haftet nur für Schäden, die auf vorsätzlicher oder
|
||||||
|
grob fahrlässiger Pflichtverletzung beruhen. Eine Haftung für
|
||||||
|
entgangenen Gewinn, Umsatzausfälle, Datenverlust,
|
||||||
|
Betriebsunterbrechungen, mittelbare oder Folgeschäden ist
|
||||||
|
ausgeschlossen, soweit gesetzlich zulässig.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="07" title="Verfügbarkeit & Betrieb">
|
||||||
|
Bei vereinbartem Hosting oder Betrieb schuldet der Auftragnehmer keine
|
||||||
|
permanente Verfügbarkeit. Wartungsarbeiten, Updates,
|
||||||
|
Sicherheitsmaßnahmen oder externe Störungen können zu zeitweisen
|
||||||
|
Einschränkungen führen und begründen keine Haftungsansprüche.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="07a" title="Betriebs- und Pflegeleistung">
|
||||||
|
Die Betriebs- und Pflegeleistung umfasst ausschließlich die
|
||||||
|
Sicherstellung des technischen Betriebs, Wartung, Updates,
|
||||||
|
Fehlerbehebung der bestehenden Systeme sowie Pflege bestehender
|
||||||
|
Datensätze ohne Strukturänderung. Nicht Bestandteil sind die
|
||||||
|
Erstellung neuer Inhalte (Blogartikel, News, Produkte), redaktionelle
|
||||||
|
Tätigkeiten, strategische Planung oder der Aufbau neuer
|
||||||
|
Features/Datenmodelle. Leistungen darüber hinaus gelten als
|
||||||
|
Neuentwicklung.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="08" title="Drittanbieter & externe Systeme">
|
||||||
|
Der Auftragnehmer übernimmt keine Verantwortung für Leistungen,
|
||||||
|
Ausfälle oder Änderungen externer Dienste, APIs, Schnittstellen oder
|
||||||
|
Plattformen Dritter. Eine Funktionsfähigkeit kann nur im Rahmen der
|
||||||
|
jeweils aktuellen externen Schnittstellen gewährleistet werden.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="09" title="Inhalte & Rechtliches">
|
||||||
|
Der Auftraggeber ist allein verantwortlich für Inhalte, rechtliche
|
||||||
|
Konformität (DSGVO, Urheberrecht etc.) sowie bereitgestellte Daten.
|
||||||
|
Der Auftragnehmer übernimmt keine rechtliche Prüfung.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="10" title="Vergütung & Zahlungsverzug">
|
||||||
|
Alle Preise netto zzgl. MwSt. Rechnungen sind innerhalb von 7 Tagen
|
||||||
|
fällig. Bei Zahlungsverzug ist der Auftragnehmer berechtigt,
|
||||||
|
Leistungen auszusetzen, Systeme offline zu nehmen oder laufende
|
||||||
|
Arbeiten zu stoppen.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="11" title="Kündigung laufender Leistungen">
|
||||||
|
Laufende Leistungen (z. B. Hosting & Betrieb) können mit einer Frist
|
||||||
|
von 4 Wochen zum Monatsende gekündigt werden, sofern nichts anderes
|
||||||
|
vereinbart ist.
|
||||||
|
</AGBSection>
|
||||||
|
|
||||||
|
<AGBSection index="12" title="Schlussbestimmungen">
|
||||||
|
Es gilt das Recht der Bundesrepublik Deutschland. Gerichtsstand ist
|
||||||
|
der Sitz des Auftragnehmers. Sollte eine Bestimmung unwirksam sein,
|
||||||
|
bleibt die Wirksamkeit der übrigen Regelungen unberührt.
|
||||||
|
</AGBSection>
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (mode === "full") {
|
||||||
|
return (
|
||||||
|
<SimpleLayout
|
||||||
|
companyData={companyData}
|
||||||
|
bankData={bankData}
|
||||||
|
footerLogo={footerLogo}
|
||||||
|
icon={headerIcon}
|
||||||
|
pageNumber="10"
|
||||||
|
showPageNumber={false}
|
||||||
|
>
|
||||||
|
{content}
|
||||||
|
</SimpleLayout>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PDFPage size="A4" style={pdfStyles.page}>
|
||||||
|
<FoldingMarks />
|
||||||
|
<Header icon={headerIcon} showAddress={false} />
|
||||||
|
{content}
|
||||||
|
<Footer
|
||||||
|
logo={footerLogo}
|
||||||
|
companyData={companyData}
|
||||||
|
bankData={bankData}
|
||||||
|
showDetails={false}
|
||||||
|
showPageNumber={false}
|
||||||
|
/>
|
||||||
|
</PDFPage>
|
||||||
|
);
|
||||||
|
};
|
||||||
79
packages/pdf-library/src/components/CombinedQuotePDF.tsx
Normal file
79
packages/pdf-library/src/components/CombinedQuotePDF.tsx
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import { Document as PDFDocument } from "@react-pdf/renderer";
|
||||||
|
import { EstimationPDF } from "./EstimationPDF.js";
|
||||||
|
import { AgbsPDF } from "./AgbsPDF.js";
|
||||||
|
import { SimpleLayout } from "./pdf/SimpleLayout.js";
|
||||||
|
import { ClosingModule } from "./pdf/modules/CommonModules.js";
|
||||||
|
|
||||||
|
interface CombinedProps {
|
||||||
|
estimationProps: any;
|
||||||
|
showAgbs?: boolean;
|
||||||
|
techDetails?: any[];
|
||||||
|
principles?: any[];
|
||||||
|
maintenanceDetails?: any[];
|
||||||
|
standardsDetails?: any[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export const CombinedQuotePDF = ({
|
||||||
|
estimationProps,
|
||||||
|
showAgbs = true,
|
||||||
|
techDetails,
|
||||||
|
principles,
|
||||||
|
maintenanceDetails,
|
||||||
|
standardsDetails,
|
||||||
|
mode = "full",
|
||||||
|
}: CombinedProps & { mode?: "estimation" | "full" }) => {
|
||||||
|
const date = new Date().toLocaleDateString("de-DE", {
|
||||||
|
year: "numeric",
|
||||||
|
month: "long",
|
||||||
|
day: "numeric",
|
||||||
|
});
|
||||||
|
|
||||||
|
const companyData = {
|
||||||
|
name: "Marc Mintel",
|
||||||
|
address1: "Georg-Meistermann-Straße 7",
|
||||||
|
address2: "54586 Schüller",
|
||||||
|
ustId: "DE367588065",
|
||||||
|
};
|
||||||
|
|
||||||
|
const bankData = {
|
||||||
|
name: "N26",
|
||||||
|
bic: "NTSBDEB1XXX",
|
||||||
|
iban: "DE50 1001 1001 2620 4328 65",
|
||||||
|
};
|
||||||
|
|
||||||
|
const layoutProps = {
|
||||||
|
date,
|
||||||
|
icon: estimationProps.headerIcon,
|
||||||
|
footerLogo: estimationProps.footerLogo,
|
||||||
|
companyData,
|
||||||
|
bankData,
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PDFDocument
|
||||||
|
title={`Mintel - ${estimationProps.state.companyName || estimationProps.state.name}`}
|
||||||
|
>
|
||||||
|
<EstimationPDF
|
||||||
|
{...estimationProps}
|
||||||
|
mode={mode}
|
||||||
|
techDetails={techDetails}
|
||||||
|
principles={principles}
|
||||||
|
maintenanceDetails={maintenanceDetails}
|
||||||
|
standardsDetails={standardsDetails}
|
||||||
|
/>
|
||||||
|
{showAgbs && (
|
||||||
|
<AgbsPDF
|
||||||
|
mode={mode}
|
||||||
|
headerIcon={estimationProps.headerIcon}
|
||||||
|
footerLogo={estimationProps.footerLogo}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<SimpleLayout {...layoutProps} pageNumber="END" showPageNumber={false}>
|
||||||
|
<ClosingModule />
|
||||||
|
</SimpleLayout>
|
||||||
|
</PDFDocument>
|
||||||
|
);
|
||||||
|
};
|
||||||
95
packages/pdf-library/src/components/EstimationPDF.tsx
Normal file
95
packages/pdf-library/src/components/EstimationPDF.tsx
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import { Page as PDFPage, Document as PDFDocument } from "@react-pdf/renderer";
|
||||||
|
import { pdfStyles } from "./pdf/SharedUI.js";
|
||||||
|
import { SimpleLayout } from "./pdf/SimpleLayout.js";
|
||||||
|
|
||||||
|
// Modules
|
||||||
|
import { FrontPageModule } from "./pdf/modules/FrontPageModule.js";
|
||||||
|
import { BriefingModule } from "./pdf/modules/BriefingModule.js";
|
||||||
|
import { SitemapModule } from "./pdf/modules/SitemapModule.js";
|
||||||
|
import { EstimationModule } from "./pdf/modules/EstimationModule.js";
|
||||||
|
import { TransparenzModule } from "./pdf/modules/TransparenzModule.js";
|
||||||
|
import { ClosingModule } from "./pdf/modules/CommonModules.js";
|
||||||
|
|
||||||
|
import { calculatePositions } from "../logic/pricing/calculator.js";
|
||||||
|
|
||||||
|
interface PDFProps {
|
||||||
|
state: any;
|
||||||
|
totalPrice: number;
|
||||||
|
monthlyPrice?: number;
|
||||||
|
totalPagesCount?: number;
|
||||||
|
pricing: any;
|
||||||
|
headerIcon?: string;
|
||||||
|
footerLogo?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const EstimationPDF = ({
|
||||||
|
state,
|
||||||
|
totalPrice,
|
||||||
|
pricing,
|
||||||
|
headerIcon,
|
||||||
|
footerLogo,
|
||||||
|
}: PDFProps) => {
|
||||||
|
const date = new Date().toLocaleDateString("de-DE", {
|
||||||
|
year: "numeric",
|
||||||
|
month: "long",
|
||||||
|
day: "numeric",
|
||||||
|
});
|
||||||
|
|
||||||
|
const positions = calculatePositions(state, pricing);
|
||||||
|
|
||||||
|
const companyData = {
|
||||||
|
name: "Marc Mintel",
|
||||||
|
address1: "Georg-Meistermann-Straße 7",
|
||||||
|
address2: "54586 Schüller",
|
||||||
|
ustId: "DE367588065",
|
||||||
|
};
|
||||||
|
|
||||||
|
const commonProps = {
|
||||||
|
state,
|
||||||
|
date,
|
||||||
|
icon: headerIcon,
|
||||||
|
footerLogo,
|
||||||
|
companyData,
|
||||||
|
};
|
||||||
|
|
||||||
|
let pageCounter = 1;
|
||||||
|
const getPageNum = () => (pageCounter++).toString().padStart(2, "0");
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PDFDocument title={`Angebot - ${state.companyName || "Projekt"}`}>
|
||||||
|
<PDFPage size="A4" style={pdfStyles.titlePage}>
|
||||||
|
<FrontPageModule state={state} headerIcon={headerIcon} date={date} />
|
||||||
|
</PDFPage>
|
||||||
|
|
||||||
|
<SimpleLayout {...commonProps} pageNumber={getPageNum()}>
|
||||||
|
<BriefingModule state={state} />
|
||||||
|
</SimpleLayout>
|
||||||
|
|
||||||
|
{state.sitemap && state.sitemap.length > 0 && (
|
||||||
|
<SimpleLayout {...commonProps} pageNumber={getPageNum()}>
|
||||||
|
<SitemapModule state={state} />
|
||||||
|
</SimpleLayout>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<SimpleLayout {...commonProps} pageNumber={getPageNum()}>
|
||||||
|
<EstimationModule
|
||||||
|
state={state}
|
||||||
|
positions={positions}
|
||||||
|
totalPrice={totalPrice}
|
||||||
|
date={date}
|
||||||
|
/>
|
||||||
|
</SimpleLayout>
|
||||||
|
|
||||||
|
<SimpleLayout {...commonProps} pageNumber={getPageNum()}>
|
||||||
|
<TransparenzModule pricing={pricing} />
|
||||||
|
</SimpleLayout>
|
||||||
|
|
||||||
|
<SimpleLayout {...commonProps} pageNumber={getPageNum()}>
|
||||||
|
<ClosingModule />
|
||||||
|
</SimpleLayout>
|
||||||
|
</PDFDocument>
|
||||||
|
);
|
||||||
|
};
|
||||||
55
packages/pdf-library/src/components/pdf/DINLayout.tsx
Normal file
55
packages/pdf-library/src/components/pdf/DINLayout.tsx
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Page as PDFPage } from '@react-pdf/renderer';
|
||||||
|
import { FoldingMarks, Header, Footer, pdfStyles } from './SharedUI';
|
||||||
|
|
||||||
|
interface DINLayoutProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
sender?: string;
|
||||||
|
recipient?: {
|
||||||
|
title: string;
|
||||||
|
subtitle?: string;
|
||||||
|
address?: string;
|
||||||
|
phone?: string;
|
||||||
|
email?: string;
|
||||||
|
taxId?: string;
|
||||||
|
};
|
||||||
|
icon?: string;
|
||||||
|
footerLogo?: string;
|
||||||
|
companyData: any;
|
||||||
|
bankData: any;
|
||||||
|
showAddress?: boolean;
|
||||||
|
showFooterDetails?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const DINLayout = ({
|
||||||
|
children,
|
||||||
|
sender,
|
||||||
|
recipient,
|
||||||
|
icon,
|
||||||
|
footerLogo,
|
||||||
|
companyData,
|
||||||
|
bankData,
|
||||||
|
showAddress = true,
|
||||||
|
showFooterDetails = true
|
||||||
|
}: DINLayoutProps) => {
|
||||||
|
return (
|
||||||
|
<PDFPage size="A4" style={pdfStyles.page}>
|
||||||
|
<FoldingMarks />
|
||||||
|
<Header
|
||||||
|
sender={sender}
|
||||||
|
recipient={recipient}
|
||||||
|
icon={icon}
|
||||||
|
showAddress={showAddress}
|
||||||
|
/>
|
||||||
|
{children}
|
||||||
|
<Footer
|
||||||
|
logo={footerLogo}
|
||||||
|
companyData={companyData}
|
||||||
|
bankData={bankData}
|
||||||
|
showDetails={showFooterDetails}
|
||||||
|
/>
|
||||||
|
</PDFPage>
|
||||||
|
);
|
||||||
|
};
|
||||||
728
packages/pdf-library/src/components/pdf/SharedUI.tsx
Normal file
728
packages/pdf-library/src/components/pdf/SharedUI.tsx
Normal file
@@ -0,0 +1,728 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
StyleSheet,
|
||||||
|
Image as PDFImage,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
|
||||||
|
// INDUSTRIAL DESIGN SYSTEM TOKENS
|
||||||
|
export const COLORS = {
|
||||||
|
CHARCOAL: "#0f172a", // Slate 900
|
||||||
|
TEXT_MAIN: "#334155", // Slate 700
|
||||||
|
TEXT_DIM: "#64748b", // Slate 500
|
||||||
|
TEXT_LIGHT: "#94a3b8", // Slate 400
|
||||||
|
DIVIDER: "#cbd5e1", // Slate 300
|
||||||
|
GRID: "#f1f5f9", // Slate 100
|
||||||
|
BLUEPRINT: "#e2e8f0", // Slate 200
|
||||||
|
WHITE: "#ffffff",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const FONT_SIZES = {
|
||||||
|
HERO: 24, // Main Page Titles
|
||||||
|
HEADING: 14, // Section Headers
|
||||||
|
BODY: 11, // Standard Content
|
||||||
|
LABEL: 10, // Bold Labels / Keys
|
||||||
|
SMALL: 9, // Descriptions / Footnotes
|
||||||
|
TINY: 8, // Metadata / Unit prices
|
||||||
|
};
|
||||||
|
|
||||||
|
// Mintel Industrial Glyphs (strictly 1px stroke, 12x12px grid)
|
||||||
|
export const IndustrialGlyph = ({
|
||||||
|
type,
|
||||||
|
color = COLORS.TEXT_LIGHT,
|
||||||
|
size = 12,
|
||||||
|
}: {
|
||||||
|
type: string;
|
||||||
|
color?: string;
|
||||||
|
size?: number;
|
||||||
|
}) => {
|
||||||
|
const stroke = 1;
|
||||||
|
const scale = size / 12;
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case "base": // Skeletal cube base
|
||||||
|
return (
|
||||||
|
<PDFView style={{ width: size, height: size, position: "relative" }}>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: 2 * scale,
|
||||||
|
left: 2 * scale,
|
||||||
|
width: 8 * scale,
|
||||||
|
height: 8 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
width: 4 * scale,
|
||||||
|
height: 4 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
backgroundColor: "white",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
case "pages": // Layered rectangles
|
||||||
|
return (
|
||||||
|
<PDFView style={{ width: size, height: size, position: "relative" }}>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: 3 * scale,
|
||||||
|
left: 3 * scale,
|
||||||
|
width: 6 * scale,
|
||||||
|
height: 8 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
width: 6 * scale,
|
||||||
|
height: 8 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
backgroundColor: "white",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
case "modules": // Four small squares grid
|
||||||
|
return (
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
flexDirection: "row",
|
||||||
|
flexWrap: "wrap",
|
||||||
|
gap: 2 * scale,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 4 * scale,
|
||||||
|
height: 4 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 4 * scale,
|
||||||
|
height: 4 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 4 * scale,
|
||||||
|
height: 4 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 4 * scale,
|
||||||
|
height: 4 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
case "logic": // Diamond with center point
|
||||||
|
return (
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 8 * scale,
|
||||||
|
height: 8 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
transform: "rotate(45deg)",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 2 * scale,
|
||||||
|
height: 2 * scale,
|
||||||
|
backgroundColor: color,
|
||||||
|
position: "absolute",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
case "interface": // Three horizontal lines of varying length
|
||||||
|
return (
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
justifyContent: "center",
|
||||||
|
gap: 2 * scale,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 10 * scale,
|
||||||
|
height: stroke,
|
||||||
|
backgroundColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{ width: 6 * scale, height: stroke, backgroundColor: color }}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 10 * scale,
|
||||||
|
height: stroke,
|
||||||
|
backgroundColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
case "management": // Framed grid
|
||||||
|
return (
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
padding: 1 * scale,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: "100%",
|
||||||
|
height: 2 * scale,
|
||||||
|
backgroundColor: color,
|
||||||
|
marginBottom: 1 * scale,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: "100%",
|
||||||
|
height: 2 * scale,
|
||||||
|
backgroundColor: color,
|
||||||
|
marginBottom: 1 * scale,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{ width: "100%", height: 2 * scale, backgroundColor: color }}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
case "reveal": // Ascending bars
|
||||||
|
return (
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "flex-end",
|
||||||
|
gap: 1 * scale,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 2 * scale,
|
||||||
|
height: 4 * scale,
|
||||||
|
backgroundColor: color,
|
||||||
|
opacity: 0.4,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 2 * scale,
|
||||||
|
height: 7 * scale,
|
||||||
|
backgroundColor: color,
|
||||||
|
opacity: 0.7,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: 2 * scale,
|
||||||
|
height: 10 * scale,
|
||||||
|
backgroundColor: color,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
case "maintenance": // Circle with vertical notch
|
||||||
|
return (
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
borderRadius: 6 * scale,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: color,
|
||||||
|
alignItems: "center",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: stroke,
|
||||||
|
height: 4 * scale,
|
||||||
|
backgroundColor: color,
|
||||||
|
marginTop: 1 * scale,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
return (
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
borderWidth: stroke,
|
||||||
|
borderColor: COLORS.BLUEPRINT,
|
||||||
|
borderStyle: "dashed",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const pdfStyles = StyleSheet.create({
|
||||||
|
page: {
|
||||||
|
paddingTop: 45, // DIN 5008
|
||||||
|
paddingLeft: 70, // ~25mm
|
||||||
|
paddingRight: 57, // ~20mm
|
||||||
|
paddingBottom: 80, // Safe buffer for absolute footer
|
||||||
|
backgroundColor: COLORS.WHITE,
|
||||||
|
fontFamily: "Helvetica",
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
},
|
||||||
|
titlePage: {
|
||||||
|
width: "100%",
|
||||||
|
height: "100%",
|
||||||
|
backgroundColor: COLORS.WHITE,
|
||||||
|
fontFamily: "Helvetica",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
padding: 0,
|
||||||
|
},
|
||||||
|
header: {
|
||||||
|
flexDirection: "row",
|
||||||
|
justifyContent: "space-between",
|
||||||
|
alignItems: "flex-start",
|
||||||
|
marginBottom: 20,
|
||||||
|
minHeight: 120,
|
||||||
|
},
|
||||||
|
addressBlock: {
|
||||||
|
width: "55%",
|
||||||
|
marginTop: 45,
|
||||||
|
},
|
||||||
|
senderLine: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
textDecoration: "underline",
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
marginBottom: 8,
|
||||||
|
},
|
||||||
|
recipientAddress: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
},
|
||||||
|
brandLogoContainer: {
|
||||||
|
width: "40%",
|
||||||
|
alignItems: "flex-end",
|
||||||
|
},
|
||||||
|
brandIconContainer: {
|
||||||
|
width: 40,
|
||||||
|
height: 40,
|
||||||
|
backgroundColor: "#0f172a",
|
||||||
|
borderRadius: 8,
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
marginBottom: 12,
|
||||||
|
},
|
||||||
|
brandIconText: {
|
||||||
|
color: COLORS.WHITE,
|
||||||
|
fontSize: 20,
|
||||||
|
fontWeight: "bold",
|
||||||
|
},
|
||||||
|
titleInfo: {
|
||||||
|
marginBottom: 24,
|
||||||
|
},
|
||||||
|
mainTitle: {
|
||||||
|
fontSize: FONT_SIZES.HEADING,
|
||||||
|
fontWeight: "bold",
|
||||||
|
marginBottom: 4,
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
},
|
||||||
|
subTitle: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
marginTop: 2,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
},
|
||||||
|
section: {
|
||||||
|
marginBottom: 32,
|
||||||
|
},
|
||||||
|
sectionTitle: {
|
||||||
|
fontSize: FONT_SIZES.LABEL,
|
||||||
|
fontWeight: "bold",
|
||||||
|
textTransform: "uppercase",
|
||||||
|
letterSpacing: 1,
|
||||||
|
color: COLORS.TEXT_LIGHT,
|
||||||
|
marginBottom: 8,
|
||||||
|
},
|
||||||
|
footer: {
|
||||||
|
position: "absolute",
|
||||||
|
bottom: 32,
|
||||||
|
left: 70,
|
||||||
|
right: 57,
|
||||||
|
borderTopWidth: 1,
|
||||||
|
borderTopColor: COLORS.GRID,
|
||||||
|
paddingTop: 16,
|
||||||
|
flexDirection: "row",
|
||||||
|
justifyContent: "space-between",
|
||||||
|
alignItems: "flex-start",
|
||||||
|
},
|
||||||
|
footerColumn: {
|
||||||
|
flex: 1,
|
||||||
|
alignItems: "flex-start",
|
||||||
|
},
|
||||||
|
footerLogo: {
|
||||||
|
height: 20,
|
||||||
|
width: "auto",
|
||||||
|
objectFit: "contain",
|
||||||
|
marginBottom: 8,
|
||||||
|
},
|
||||||
|
footerText: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
color: COLORS.TEXT_LIGHT,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
},
|
||||||
|
asymmetryContainer: {
|
||||||
|
flexDirection: "row",
|
||||||
|
gap: 32,
|
||||||
|
},
|
||||||
|
asymmetryLeft: {
|
||||||
|
width: "32%",
|
||||||
|
},
|
||||||
|
asymmetryRight: {
|
||||||
|
width: "63%",
|
||||||
|
},
|
||||||
|
specRow: {
|
||||||
|
flexDirection: "row",
|
||||||
|
justifyContent: "space-between",
|
||||||
|
paddingVertical: 6,
|
||||||
|
borderBottomWidth: 1,
|
||||||
|
borderBottomColor: COLORS.GRID,
|
||||||
|
},
|
||||||
|
specLabel: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.TEXT_LIGHT,
|
||||||
|
textTransform: "uppercase",
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
},
|
||||||
|
specValue: {
|
||||||
|
fontSize: FONT_SIZES.SMALL,
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
fontWeight: "bold",
|
||||||
|
},
|
||||||
|
blueprintBox: {
|
||||||
|
borderWidth: 1,
|
||||||
|
borderColor: COLORS.GRID,
|
||||||
|
padding: 16,
|
||||||
|
backgroundColor: "#fafafa",
|
||||||
|
},
|
||||||
|
footerLabel: {
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
},
|
||||||
|
pageNumber: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
color: COLORS.DIVIDER,
|
||||||
|
fontWeight: "bold",
|
||||||
|
marginTop: 8,
|
||||||
|
textAlign: "right",
|
||||||
|
},
|
||||||
|
foldingMark: {
|
||||||
|
position: "absolute",
|
||||||
|
left: 20,
|
||||||
|
width: 10,
|
||||||
|
borderTopWidth: 0.5,
|
||||||
|
borderTopColor: COLORS.DIVIDER,
|
||||||
|
},
|
||||||
|
divider: {
|
||||||
|
width: "100%",
|
||||||
|
height: 1,
|
||||||
|
backgroundColor: COLORS.DIVIDER,
|
||||||
|
marginVertical: 12,
|
||||||
|
},
|
||||||
|
industrialListItem: {
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "flex-start",
|
||||||
|
marginBottom: 6,
|
||||||
|
},
|
||||||
|
industrialBulletBox: {
|
||||||
|
width: 6,
|
||||||
|
height: 6,
|
||||||
|
backgroundColor: COLORS.DIVIDER,
|
||||||
|
marginRight: 8,
|
||||||
|
marginTop: 5,
|
||||||
|
},
|
||||||
|
industrialTitle: {
|
||||||
|
fontSize: FONT_SIZES.HERO,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
marginBottom: 6,
|
||||||
|
letterSpacing: 0,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const IndustrialListItem = ({
|
||||||
|
children,
|
||||||
|
}: {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}) => (
|
||||||
|
<PDFView style={pdfStyles.industrialListItem}>
|
||||||
|
<PDFView style={pdfStyles.industrialBulletBox} />
|
||||||
|
{children}
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const Divider = ({ style = {} }: { style?: any }) => (
|
||||||
|
<PDFView style={[pdfStyles.divider, style]} />
|
||||||
|
);
|
||||||
|
|
||||||
|
export const FoldingMarks = () => (
|
||||||
|
<>
|
||||||
|
<PDFView style={[pdfStyles.foldingMark, { top: 297.6 }]} fixed />
|
||||||
|
<PDFView style={[pdfStyles.foldingMark, { top: 420.9, width: 15 }]} fixed />
|
||||||
|
<PDFView style={[pdfStyles.foldingMark, { top: 595.3 }]} fixed />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const Footer = ({
|
||||||
|
logo,
|
||||||
|
companyData,
|
||||||
|
bankData,
|
||||||
|
showDetails = true,
|
||||||
|
showPageNumber = true,
|
||||||
|
}: {
|
||||||
|
logo?: string;
|
||||||
|
companyData: any;
|
||||||
|
bankData?: any;
|
||||||
|
showDetails?: boolean;
|
||||||
|
showPageNumber?: boolean;
|
||||||
|
}) => (
|
||||||
|
<PDFView style={pdfStyles.footer}>
|
||||||
|
<PDFView style={pdfStyles.footerColumn}>
|
||||||
|
{logo ? (
|
||||||
|
<PDFImage src={logo} style={pdfStyles.footerLogo} />
|
||||||
|
) : (
|
||||||
|
<PDFText style={{ fontSize: 12, fontWeight: "bold", marginBottom: 8 }}>
|
||||||
|
marc mintel
|
||||||
|
</PDFText>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
{showDetails && (
|
||||||
|
<>
|
||||||
|
<PDFView style={pdfStyles.footerColumn}>
|
||||||
|
<PDFText style={pdfStyles.footerText}>
|
||||||
|
<PDFText style={pdfStyles.footerLabel}>{companyData.name}</PDFText>
|
||||||
|
{"\n"}
|
||||||
|
{companyData.address1}
|
||||||
|
{"\n"}
|
||||||
|
{companyData.address2}
|
||||||
|
{"\n"}UST: {companyData.ustId}
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={[pdfStyles.footerColumn, { alignItems: "flex-end" }]}>
|
||||||
|
{showPageNumber && (
|
||||||
|
<PDFText
|
||||||
|
style={pdfStyles.pageNumber}
|
||||||
|
render={({ pageNumber, totalPages }) =>
|
||||||
|
`${pageNumber} / ${totalPages}`
|
||||||
|
}
|
||||||
|
fixed
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{!showDetails && (
|
||||||
|
<PDFView style={[pdfStyles.footerColumn, { alignItems: "flex-end" }]}>
|
||||||
|
{showPageNumber && (
|
||||||
|
<PDFText
|
||||||
|
style={pdfStyles.pageNumber}
|
||||||
|
render={({ pageNumber, totalPages }) =>
|
||||||
|
`${pageNumber} / ${totalPages}`
|
||||||
|
}
|
||||||
|
fixed
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const Header = ({
|
||||||
|
sender,
|
||||||
|
recipient,
|
||||||
|
icon,
|
||||||
|
showAddress = true,
|
||||||
|
}: {
|
||||||
|
sender?: string;
|
||||||
|
recipient?: {
|
||||||
|
title: string;
|
||||||
|
subtitle?: string;
|
||||||
|
email?: string;
|
||||||
|
address?: string;
|
||||||
|
phone?: string;
|
||||||
|
taxId?: string;
|
||||||
|
};
|
||||||
|
icon?: string;
|
||||||
|
showAddress?: boolean;
|
||||||
|
}) => (
|
||||||
|
<PDFView
|
||||||
|
style={[
|
||||||
|
pdfStyles.header,
|
||||||
|
showAddress ? {} : { minHeight: 40, marginBottom: 0 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<PDFView style={pdfStyles.addressBlock}>
|
||||||
|
{showAddress && sender && (
|
||||||
|
<>
|
||||||
|
<PDFText style={pdfStyles.senderLine}>{sender}</PDFText>
|
||||||
|
{recipient && (
|
||||||
|
<PDFView style={pdfStyles.recipientAddress}>
|
||||||
|
<PDFText style={{ fontWeight: "bold" }}>
|
||||||
|
{recipient.title}
|
||||||
|
</PDFText>
|
||||||
|
{recipient.subtitle && <PDFText>{recipient.subtitle}</PDFText>}
|
||||||
|
{recipient.address && <PDFText>{recipient.address}</PDFText>}
|
||||||
|
{recipient.phone && <PDFText>{recipient.phone}</PDFText>}
|
||||||
|
{recipient.email && <PDFText>{recipient.email}</PDFText>}
|
||||||
|
{recipient.taxId && <PDFText>USt-ID: {recipient.taxId}</PDFText>}
|
||||||
|
</PDFView>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={pdfStyles.brandLogoContainer}>
|
||||||
|
<PDFView style={pdfStyles.brandIconContainer}>
|
||||||
|
{icon ? (
|
||||||
|
<PDFImage src={icon} style={{ width: 24, height: 24 }} />
|
||||||
|
) : (
|
||||||
|
<PDFText style={pdfStyles.brandIconText}>M</PDFText>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const DocumentTitle = ({
|
||||||
|
title,
|
||||||
|
subLines,
|
||||||
|
isHero = false,
|
||||||
|
}: {
|
||||||
|
title: string;
|
||||||
|
subLines?: string[];
|
||||||
|
isHero?: boolean;
|
||||||
|
}) => (
|
||||||
|
<PDFView style={pdfStyles.titleInfo}>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
pdfStyles.mainTitle,
|
||||||
|
{ fontSize: isHero ? FONT_SIZES.HERO : FONT_SIZES.HEADING },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{title}
|
||||||
|
</PDFText>
|
||||||
|
{subLines?.map((line, i) => (
|
||||||
|
<PDFText
|
||||||
|
key={i}
|
||||||
|
style={[
|
||||||
|
pdfStyles.subTitle,
|
||||||
|
i === 1 ? { fontWeight: "bold", color: COLORS.CHARCOAL } : {},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{line}
|
||||||
|
</PDFText>
|
||||||
|
))}
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const TechnicalSpec = ({
|
||||||
|
label,
|
||||||
|
value,
|
||||||
|
}: {
|
||||||
|
label: string;
|
||||||
|
value: string;
|
||||||
|
}) => (
|
||||||
|
<PDFView style={pdfStyles.specRow}>
|
||||||
|
<PDFText style={pdfStyles.specLabel}>{label}</PDFText>
|
||||||
|
<PDFText style={pdfStyles.specValue}>{value}</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const AsymmetryView = ({
|
||||||
|
left,
|
||||||
|
right,
|
||||||
|
style = {},
|
||||||
|
}: {
|
||||||
|
left: React.ReactNode;
|
||||||
|
right: React.ReactNode;
|
||||||
|
style?: any;
|
||||||
|
}) => (
|
||||||
|
<PDFView style={[pdfStyles.asymmetryContainer, style]}>
|
||||||
|
<PDFView style={pdfStyles.asymmetryLeft}>{left}</PDFView>
|
||||||
|
<PDFView style={pdfStyles.asymmetryRight}>{right}</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const IndustrialCard = ({
|
||||||
|
title,
|
||||||
|
children,
|
||||||
|
style = {},
|
||||||
|
}: {
|
||||||
|
title: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
style?: any;
|
||||||
|
}) => (
|
||||||
|
<PDFView style={[pdfStyles.blueprintBox, { marginBottom: 12 }, style]}>
|
||||||
|
<PDFText
|
||||||
|
style={{
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.TEXT_LIGHT,
|
||||||
|
letterSpacing: 1,
|
||||||
|
marginBottom: 6,
|
||||||
|
textTransform: "uppercase",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{title}
|
||||||
|
</PDFText>
|
||||||
|
{children}
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
67
packages/pdf-library/src/components/pdf/SimpleLayout.tsx
Normal file
67
packages/pdf-library/src/components/pdf/SimpleLayout.tsx
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Page as PDFPage, View as PDFView, Text as PDFText, StyleSheet } from '@react-pdf/renderer';
|
||||||
|
import { Header, Footer, pdfStyles } from './SharedUI.js';
|
||||||
|
|
||||||
|
const simpleStyles = StyleSheet.create({
|
||||||
|
industrialPage: {
|
||||||
|
padding: 30,
|
||||||
|
paddingTop: 20,
|
||||||
|
backgroundColor: '#ffffff',
|
||||||
|
},
|
||||||
|
industrialNumber: {
|
||||||
|
fontSize: 60,
|
||||||
|
fontWeight: 'bold',
|
||||||
|
color: '#f1f5f9',
|
||||||
|
position: 'absolute',
|
||||||
|
top: -10,
|
||||||
|
right: 0,
|
||||||
|
zIndex: -1,
|
||||||
|
},
|
||||||
|
industrialSection: {
|
||||||
|
marginTop: 16,
|
||||||
|
paddingTop: 12,
|
||||||
|
flexDirection: 'row',
|
||||||
|
position: 'relative',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
interface SimpleLayoutProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
pageNumber?: string;
|
||||||
|
icon?: string;
|
||||||
|
footerLogo?: string;
|
||||||
|
companyData: any;
|
||||||
|
bankData?: any;
|
||||||
|
showPageNumber?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SimpleLayout = ({
|
||||||
|
children,
|
||||||
|
pageNumber,
|
||||||
|
icon,
|
||||||
|
footerLogo,
|
||||||
|
companyData,
|
||||||
|
bankData,
|
||||||
|
showPageNumber = true
|
||||||
|
}: SimpleLayoutProps) => {
|
||||||
|
return (
|
||||||
|
<PDFPage size="A4" style={[pdfStyles.page, simpleStyles.industrialPage]}>
|
||||||
|
<Header icon={icon} showAddress={false} />
|
||||||
|
{pageNumber && <PDFText style={simpleStyles.industrialNumber}>{pageNumber}</PDFText>}
|
||||||
|
<PDFView style={simpleStyles.industrialSection}>
|
||||||
|
<PDFView style={{ width: '100%' }}>
|
||||||
|
{children}
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
<Footer
|
||||||
|
logo={footerLogo}
|
||||||
|
companyData={companyData}
|
||||||
|
bankData={bankData}
|
||||||
|
showDetails={false}
|
||||||
|
showPageNumber={showPageNumber}
|
||||||
|
/>
|
||||||
|
</PDFPage>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,218 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
StyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import {
|
||||||
|
DocumentTitle,
|
||||||
|
IndustrialListItem,
|
||||||
|
IndustrialCard,
|
||||||
|
Divider,
|
||||||
|
COLORS,
|
||||||
|
FONT_SIZES,
|
||||||
|
} from "../SharedUI";
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
industrialTextLead: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.TEXT_MAIN,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
marginBottom: 16,
|
||||||
|
},
|
||||||
|
industrialText: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
marginBottom: 12,
|
||||||
|
},
|
||||||
|
industrialGrid2: { flexDirection: "row" },
|
||||||
|
industrialCol: { width: "46%" },
|
||||||
|
});
|
||||||
|
|
||||||
|
export const AboutModule = () => (
|
||||||
|
<>
|
||||||
|
<DocumentTitle
|
||||||
|
title="Expertise & Profil"
|
||||||
|
subLines={["Entwicklung & Technischer Partner für den Mittelstand"]}
|
||||||
|
isHero={true}
|
||||||
|
/>
|
||||||
|
<Divider style={{ marginVertical: 16, backgroundColor: COLORS.GRID }} />
|
||||||
|
|
||||||
|
<PDFView style={{ marginTop: 24 }}>
|
||||||
|
<PDFText style={styles.industrialTextLead}>
|
||||||
|
Begleitung mittelständischer Unternehmen und Agenturen bei der
|
||||||
|
Realisierung anspruchsvoller Web-Projekte. Als Senior Software Developer
|
||||||
|
mit over 15 Jahren Erfahrung wird das gesamte technische Spektrum
|
||||||
|
abgedeckt – von der Architektur bis zum fertigen Produkt.
|
||||||
|
</PDFText>
|
||||||
|
|
||||||
|
<PDFView style={[styles.industrialGrid2, { marginTop: 20 }]}>
|
||||||
|
<PDFView style={[styles.industrialCol, { marginRight: "8%" }]}>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
styles.industrialText,
|
||||||
|
{ fontWeight: "bold", color: COLORS.CHARCOAL, marginBottom: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Erfahrung & Substanz
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Der Werdegang umfasst alle Ebenen der Webentwicklung: von der
|
||||||
|
Teamleitung in Kreativagenturen bis zur Softwareentwicklung für
|
||||||
|
internationale Konzerne.
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Die Kenntnis komplexer Enterprise-Systeme wird mit der Agilität
|
||||||
|
kombiniert, die im Mittelstand gefordert ist. Dieses Wissen
|
||||||
|
ermöglicht den Bau von Lösungen, die technologisch auf Augenhöhe mit
|
||||||
|
Konzern-Standards sind, jedoch ohne unnötigen bürokratischen
|
||||||
|
Overhead auskommen.
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
|
||||||
|
<PDFView style={styles.industrialCol}>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
styles.industrialText,
|
||||||
|
{ fontWeight: "bold", color: COLORS.CHARCOAL, marginBottom: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Fokus Einzelentwicklung
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Die Umsetzung erfolgt bewusst als spezialisierter Einzelentwickler.
|
||||||
|
Dies garantiert maximale Geschwindigkeit, direkte Kommunikationswege
|
||||||
|
und volle technologische Verantwortung.
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Als direkter technischer Sparringspartner bleibt die Codebasis von
|
||||||
|
der ersten bis zur letzten Zeile transparent und wartbar. Diese
|
||||||
|
Unmittelbarkeit stellt sicher, dass Ergebnisse sowohl technisch
|
||||||
|
sauber als auch wirtschaftlich sinnvoll realisiert werden.
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
marginTop: 32,
|
||||||
|
paddingVertical: 16,
|
||||||
|
borderTopWidth: 1,
|
||||||
|
borderTopColor: COLORS.GRID,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
styles.industrialText,
|
||||||
|
{ fontWeight: "bold", color: COLORS.CHARCOAL, marginBottom: 4 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Infrastruktur & Souveränität
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Es wird keine instabile Prototyp-Software geliefert, sondern
|
||||||
|
produktionsreife Systeme, die technisch skalierbar bleiben. Die
|
||||||
|
Codebasis folgt modernen Standards – bei wachsenden Ansprüchen oder
|
||||||
|
dem Wechsel zu einer Agentur kann der Quellcode jederzeit nahtlos
|
||||||
|
übernommen und weitergeführt werden.
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const CrossSellModule = ({ state }: any) => {
|
||||||
|
const isWebsite = state.projectType === "website";
|
||||||
|
const title = isWebsite ? "Weitere Potenziale" : "Websites & Ökosysteme";
|
||||||
|
const subtitle = isWebsite
|
||||||
|
? "Automatisierung und Prozessoptimierung"
|
||||||
|
: "Technische Infrastruktur ohne Kompromisse";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<DocumentTitle title={title} subLines={[subtitle]} isHero={true} />
|
||||||
|
<Divider style={{ marginVertical: 16, backgroundColor: COLORS.GRID }} />
|
||||||
|
<PDFView style={[styles.industrialGrid2, { marginTop: 16 }]}>
|
||||||
|
{isWebsite ? (
|
||||||
|
<>
|
||||||
|
<PDFView style={[styles.industrialCol, { marginRight: "8%" }]}>
|
||||||
|
<PDFText style={styles.industrialTextLead}>
|
||||||
|
Über die klassische Webpräsenz hinaus werden maßgeschneiderte
|
||||||
|
Lösungen zur Automatisierung von Routine-Prozessen angeboten.
|
||||||
|
Dies ermöglicht eine signifikante Effizienzsteigerung im
|
||||||
|
Tagesgeschäft.
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={[styles.industrialText, { fontWeight: "bold" }]}>
|
||||||
|
Keine Abos. Keine komplexen neuen Systeme. Gezielte
|
||||||
|
Zeitersparnis.
|
||||||
|
</PDFText>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
marginTop: 24,
|
||||||
|
padding: 16,
|
||||||
|
backgroundColor: "#f8fafc",
|
||||||
|
borderLeftWidth: 2,
|
||||||
|
borderLeftColor: COLORS.GRID,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
styles.industrialText,
|
||||||
|
{
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
marginBottom: 4,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Individuelle Analyse
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Spezifische Prozesse werden auf technisches
|
||||||
|
Automatisierungspotenzial untersucht. Das Ergebnis liefert
|
||||||
|
Klarheit über die wirtschaftliche Sinnhaftigkeit einer
|
||||||
|
Umsetzung.
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={styles.industrialCol}>
|
||||||
|
<IndustrialCard title="DOKUMENT-AUTOMATION">
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Erstellung von PDF-Angeboten, Berichten oder Protokollen in
|
||||||
|
Sekunden statt Stunden.
|
||||||
|
</PDFText>
|
||||||
|
</IndustrialCard>
|
||||||
|
<IndustrialCard title="EXCEL-LOGIK">
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Intelligente Tabellen und automatisierte Auswertungen
|
||||||
|
bestehender Datensätze.
|
||||||
|
</PDFText>
|
||||||
|
</IndustrialCard>
|
||||||
|
<IndustrialCard title="KI-ASSISTENZ">
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Effiziente Verarbeitung von analogen Dokumenten oder
|
||||||
|
handschriftlichen Notizen mittels KI.
|
||||||
|
</PDFText>
|
||||||
|
</IndustrialCard>
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<PDFView style={{ width: "100%" }}>
|
||||||
|
<PDFText style={styles.industrialTextLead}>
|
||||||
|
Bereitstellung einer stabilen technischen Basis ohne
|
||||||
|
Abhängigkeiten von Baukasten-Systemen oder Agenturen.
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.industrialText}>
|
||||||
|
Entwicklung performanter Frontends und skalierbarer Backends. Die
|
||||||
|
Auslieferung erfolgt als kontrollierbarer und nachhaltiger
|
||||||
|
Quellcode.
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,69 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
StyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import { DocumentTitle, COLORS, FONT_SIZES } from "../SharedUI";
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
section: { marginBottom: 24 },
|
||||||
|
sectionTitle: {
|
||||||
|
fontSize: FONT_SIZES.LABEL,
|
||||||
|
fontWeight: "bold",
|
||||||
|
marginBottom: 8,
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
},
|
||||||
|
visionText: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.TEXT_MAIN,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
textAlign: "justify",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const BriefingModule = ({ state }: any) => (
|
||||||
|
<>
|
||||||
|
<DocumentTitle title="Projektdetails" isHero={true} />
|
||||||
|
{state.briefingSummary && (
|
||||||
|
<PDFView style={styles.section}>
|
||||||
|
<PDFText style={styles.sectionTitle}>Briefing Analyse</PDFText>
|
||||||
|
<PDFText
|
||||||
|
style={{
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.TEXT_MAIN,
|
||||||
|
lineHeight: 1.6,
|
||||||
|
textAlign: "justify",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{state.briefingSummary}
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
)}
|
||||||
|
{state.designVision && (
|
||||||
|
<PDFView
|
||||||
|
style={[
|
||||||
|
styles.section,
|
||||||
|
{
|
||||||
|
padding: 12,
|
||||||
|
borderLeftWidth: 2,
|
||||||
|
borderLeftColor: COLORS.DIVIDER,
|
||||||
|
backgroundColor: COLORS.GRID,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
styles.sectionTitle,
|
||||||
|
{ color: COLORS.CHARCOAL, marginBottom: 4 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Strategische Vision
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.visionText}>{state.designVision}</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
@@ -0,0 +1,97 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
StyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import {
|
||||||
|
DocumentTitle,
|
||||||
|
COLORS,
|
||||||
|
FONT_SIZES,
|
||||||
|
} from "../SharedUI.js";
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
section: { marginBottom: 24 },
|
||||||
|
moduleLabel: {
|
||||||
|
fontSize: FONT_SIZES.LABEL,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
marginBottom: 4,
|
||||||
|
},
|
||||||
|
moduleDesc: {
|
||||||
|
fontSize: FONT_SIZES.SMALL,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
lineHeight: 1.5,
|
||||||
|
},
|
||||||
|
ledgerRow: {
|
||||||
|
paddingVertical: 14,
|
||||||
|
borderBottomWidth: 1,
|
||||||
|
borderBottomColor: COLORS.GRID,
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "flex-start",
|
||||||
|
},
|
||||||
|
ledgerPrice: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
},
|
||||||
|
ledgerUnit: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
color: COLORS.TEXT_LIGHT,
|
||||||
|
marginLeft: 2,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ClosingModule = () => (
|
||||||
|
<>
|
||||||
|
<DocumentTitle title="Abschluss & Kontakt" isHero={true} />
|
||||||
|
<PDFView style={styles.section}>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
styles.moduleLabel,
|
||||||
|
{ fontSize: FONT_SIZES.HEADING, marginBottom: 12 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Vielen Dank für Ihr Interesse!
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.moduleDesc}>
|
||||||
|
Die aufgeführten Positionen stellen eine detaillierte Schätzung auf
|
||||||
|
Basis unseres aktuellen Stands dar. Sollten sich Anforderungen ändern
|
||||||
|
oder Sie Fragen zu einzelnen Details haben, lassen Sie uns die
|
||||||
|
Positionen gerne gemeinsam besprechen.
|
||||||
|
</PDFText>
|
||||||
|
<PDFView
|
||||||
|
style={{
|
||||||
|
marginTop: 24,
|
||||||
|
padding: 16,
|
||||||
|
backgroundColor: COLORS.GRID,
|
||||||
|
borderLeftWidth: 2,
|
||||||
|
borderLeftColor: COLORS.DIVIDER,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<PDFText style={[styles.moduleLabel, { marginBottom: 8 }]}>
|
||||||
|
Haben Sie Fragen?
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={styles.moduleDesc}>
|
||||||
|
Ich erkläre Ihnen gerne noch einmal persönlich, was die technische
|
||||||
|
Umsetzung für Ihr Projekt bedeutet und wie wir die nächsten Schritte
|
||||||
|
gemeinsam gehen können.
|
||||||
|
</PDFText>
|
||||||
|
<PDFView style={{ marginTop: 16 }}>
|
||||||
|
<PDFText style={styles.moduleLabel}>Kontakt:</PDFText>
|
||||||
|
<PDFText
|
||||||
|
style={[
|
||||||
|
styles.moduleDesc,
|
||||||
|
{ color: COLORS.CHARCOAL, fontWeight: "bold" },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Marc Mintel – marc@mintel.me
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
);
|
||||||
@@ -0,0 +1,159 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
StyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import { DocumentTitle, COLORS, FONT_SIZES } from "../SharedUI.js";
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
table: { marginTop: 12 },
|
||||||
|
tableHeader: {
|
||||||
|
flexDirection: "row",
|
||||||
|
paddingBottom: 8,
|
||||||
|
borderBottomWidth: 1,
|
||||||
|
borderBottomColor: COLORS.CHARCOAL,
|
||||||
|
marginBottom: 12,
|
||||||
|
},
|
||||||
|
tableRow: {
|
||||||
|
flexDirection: "row",
|
||||||
|
paddingVertical: 10,
|
||||||
|
borderBottomWidth: 1,
|
||||||
|
borderBottomColor: COLORS.GRID,
|
||||||
|
alignItems: "flex-start",
|
||||||
|
},
|
||||||
|
colPos: { width: "8%" },
|
||||||
|
colDesc: { width: "62%" },
|
||||||
|
colQty: { width: "10%", textAlign: "center" },
|
||||||
|
colPrice: { width: "20%", textAlign: "right" },
|
||||||
|
headerText: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
textTransform: "uppercase",
|
||||||
|
letterSpacing: 1,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
},
|
||||||
|
posText: { fontSize: FONT_SIZES.TINY, color: COLORS.TEXT_LIGHT },
|
||||||
|
itemTitle: {
|
||||||
|
fontSize: FONT_SIZES.LABEL,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
marginBottom: 4,
|
||||||
|
},
|
||||||
|
itemDesc: {
|
||||||
|
fontSize: FONT_SIZES.SMALL,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
},
|
||||||
|
priceText: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
},
|
||||||
|
summaryContainer: {
|
||||||
|
borderTopWidth: 1,
|
||||||
|
borderTopColor: COLORS.CHARCOAL,
|
||||||
|
paddingTop: 8,
|
||||||
|
},
|
||||||
|
summaryRow: {
|
||||||
|
flexDirection: "row",
|
||||||
|
justifyContent: "flex-end",
|
||||||
|
paddingVertical: 4,
|
||||||
|
alignItems: "baseline",
|
||||||
|
},
|
||||||
|
summaryLabel: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
textTransform: "uppercase",
|
||||||
|
letterSpacing: 1,
|
||||||
|
fontWeight: "bold",
|
||||||
|
marginRight: 12,
|
||||||
|
},
|
||||||
|
summaryValue: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
width: 100,
|
||||||
|
textAlign: "right",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
},
|
||||||
|
totalRow: {
|
||||||
|
flexDirection: "row",
|
||||||
|
justifyContent: "flex-end",
|
||||||
|
paddingTop: 12,
|
||||||
|
marginTop: 8,
|
||||||
|
borderTopWidth: 2,
|
||||||
|
borderTopColor: COLORS.CHARCOAL,
|
||||||
|
alignItems: "baseline",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const EstimationModule = ({
|
||||||
|
state,
|
||||||
|
positions,
|
||||||
|
totalPrice,
|
||||||
|
date,
|
||||||
|
}: any) => (
|
||||||
|
<>
|
||||||
|
<DocumentTitle
|
||||||
|
title="Kostenschätzung"
|
||||||
|
subLines={[
|
||||||
|
`Datum: ${date}`,
|
||||||
|
`Projekt: ${state.projectType === "website" ? "Website" : "Web App"}`,
|
||||||
|
]}
|
||||||
|
isHero={true}
|
||||||
|
/>
|
||||||
|
<PDFView style={styles.table}>
|
||||||
|
<PDFView style={styles.tableHeader}>
|
||||||
|
<PDFText style={[styles.headerText, styles.colPos]}>Pos</PDFText>
|
||||||
|
<PDFText style={[styles.headerText, styles.colDesc]}>
|
||||||
|
Beschreibung
|
||||||
|
</PDFText>
|
||||||
|
<PDFText style={[styles.headerText, styles.colQty]}>Menge</PDFText>
|
||||||
|
<PDFText style={[styles.headerText, styles.colPrice]}>Betrag</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
{positions.map((item: any, i: number) => (
|
||||||
|
<PDFView key={i} style={styles.tableRow} wrap={false}>
|
||||||
|
<PDFText style={[styles.posText, styles.colPos]}>
|
||||||
|
{item.pos.toString().padStart(2, "0")}
|
||||||
|
</PDFText>
|
||||||
|
<PDFView style={styles.colDesc}>
|
||||||
|
<PDFText style={styles.itemTitle}>{item.title}</PDFText>
|
||||||
|
<PDFText style={styles.itemDesc}>
|
||||||
|
{state.positionDescriptions?.[item.title] || item.desc}
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
<PDFText style={[styles.posText, styles.colQty]}>{item.qty}</PDFText>
|
||||||
|
<PDFText style={[styles.priceText, styles.colPrice]}>
|
||||||
|
{item.price > 0
|
||||||
|
? `${item.price.toLocaleString("de-DE")} €`
|
||||||
|
: "n. A."}
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
))}
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={styles.summaryContainer} wrap={false}>
|
||||||
|
<PDFView style={styles.summaryRow}>
|
||||||
|
<PDFText style={styles.summaryLabel}>Nettobetrag</PDFText>
|
||||||
|
<PDFText style={styles.summaryValue}>
|
||||||
|
{totalPrice.toLocaleString("de-DE")} €
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={styles.summaryRow}>
|
||||||
|
<PDFText style={styles.summaryLabel}>Umsatzsteuer (19%)</PDFText>
|
||||||
|
<PDFText style={styles.summaryValue}>
|
||||||
|
{(totalPrice * 0.19).toLocaleString("de-DE")} €
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={styles.totalRow}>
|
||||||
|
<PDFText style={styles.summaryLabel}>Gesamtbetrag (Brutto)</PDFText>
|
||||||
|
<PDFText
|
||||||
|
style={[styles.summaryValue, { fontSize: FONT_SIZES.HEADING }]}
|
||||||
|
>
|
||||||
|
{(totalPrice * 1.19).toLocaleString("de-DE")} €
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
);
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
Image as PDFImage,
|
||||||
|
StyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import { COLORS, FONT_SIZES } from "../SharedUI.js";
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
titlePage: {
|
||||||
|
flex: 1,
|
||||||
|
padding: 60,
|
||||||
|
justifyContent: "center",
|
||||||
|
alignItems: "center",
|
||||||
|
backgroundColor: COLORS.WHITE,
|
||||||
|
},
|
||||||
|
titleBrandIcon: {
|
||||||
|
width: 80,
|
||||||
|
height: 80,
|
||||||
|
backgroundColor: COLORS.CHARCOAL,
|
||||||
|
borderRadius: 16,
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
marginBottom: 40,
|
||||||
|
},
|
||||||
|
brandIconText: {
|
||||||
|
fontSize: 40,
|
||||||
|
color: COLORS.WHITE,
|
||||||
|
fontWeight: "bold",
|
||||||
|
},
|
||||||
|
titleProjectName: {
|
||||||
|
fontSize: FONT_SIZES.HERO,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
marginBottom: 16,
|
||||||
|
textAlign: "center",
|
||||||
|
maxWidth: "85%",
|
||||||
|
lineHeight: 1.2,
|
||||||
|
},
|
||||||
|
titleDate: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.TEXT_LIGHT,
|
||||||
|
marginTop: 40,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const FrontPageModule = ({ state, headerIcon, date }: any) => {
|
||||||
|
const fullTitle = `Digitale Webpräsenz für\n${state.companyName || "Ihr Projekt"}`;
|
||||||
|
const fontSize = fullTitle.length > 60 ? 14 : fullTitle.length > 40 ? 18 : 22;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PDFView style={styles.titlePage}>
|
||||||
|
<PDFView style={styles.titleBrandIcon}>
|
||||||
|
{headerIcon ? (
|
||||||
|
<PDFImage src={headerIcon} style={{ width: 40, height: 40 }} />
|
||||||
|
) : (
|
||||||
|
<PDFText style={styles.brandIconText}>M</PDFText>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
<PDFText style={[styles.titleProjectName, { fontSize }]}>
|
||||||
|
{fullTitle}
|
||||||
|
</PDFText>
|
||||||
|
<PDFView style={{ marginBottom: 40 }} />
|
||||||
|
<PDFText style={styles.titleDate}>{date} | Marc Mintel</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
StyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import { DocumentTitle, COLORS, FONT_SIZES } from "../SharedUI";
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
section: { marginBottom: 32 },
|
||||||
|
intro: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
lineHeight: 1.4,
|
||||||
|
marginBottom: 24,
|
||||||
|
textAlign: "justify",
|
||||||
|
},
|
||||||
|
sitemapTree: { marginTop: 8 },
|
||||||
|
rootNode: {
|
||||||
|
padding: 12,
|
||||||
|
backgroundColor: COLORS.GRID,
|
||||||
|
marginBottom: 20,
|
||||||
|
borderLeftWidth: 2,
|
||||||
|
borderLeftColor: COLORS.CHARCOAL,
|
||||||
|
},
|
||||||
|
rootTitle: {
|
||||||
|
fontSize: FONT_SIZES.HEADING,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
},
|
||||||
|
categorySection: { marginBottom: 20 },
|
||||||
|
categoryHeader: {
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "center",
|
||||||
|
paddingBottom: 6,
|
||||||
|
borderBottomWidth: 1,
|
||||||
|
borderBottomColor: COLORS.BLUEPRINT,
|
||||||
|
marginBottom: 10,
|
||||||
|
},
|
||||||
|
categoryIcon: {
|
||||||
|
width: 8,
|
||||||
|
height: 8,
|
||||||
|
backgroundColor: COLORS.GRID,
|
||||||
|
borderInlineWidth: 1,
|
||||||
|
borderColor: COLORS.DIVIDER,
|
||||||
|
marginRight: 10,
|
||||||
|
},
|
||||||
|
categoryTitle: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
textTransform: "uppercase",
|
||||||
|
letterSpacing: 1,
|
||||||
|
},
|
||||||
|
pagesGrid: { flexDirection: "row", flexWrap: "wrap" },
|
||||||
|
pageCard: {
|
||||||
|
width: "48%",
|
||||||
|
marginRight: "2%",
|
||||||
|
marginBottom: 12,
|
||||||
|
padding: 10,
|
||||||
|
borderWidth: 1,
|
||||||
|
borderColor: COLORS.GRID,
|
||||||
|
backgroundColor: "#fafafa",
|
||||||
|
},
|
||||||
|
pageTitle: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.TEXT_MAIN,
|
||||||
|
marginBottom: 4,
|
||||||
|
},
|
||||||
|
pageDesc: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
lineHeight: 1.3,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const SitemapModule = ({ state }: any) => (
|
||||||
|
<>
|
||||||
|
<DocumentTitle title="Informationsarchitektur" isHero={true} />
|
||||||
|
<PDFView style={styles.section}>
|
||||||
|
<PDFText style={styles.intro}>
|
||||||
|
Die folgende Struktur definiert die logische Hierarchie und
|
||||||
|
Benutzerführung. Sie dient als Bauplan für die technische Umsetzung und
|
||||||
|
stellt sicher, dass alle relevanten Geschäftsbereiche intuitiv
|
||||||
|
auffindbar sind.
|
||||||
|
</PDFText>
|
||||||
|
|
||||||
|
<PDFView style={styles.sitemapTree}>
|
||||||
|
<PDFView style={styles.rootNode}>
|
||||||
|
<PDFText style={styles.rootTitle}>Seitenstruktur</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
|
||||||
|
{state.sitemap?.map((cat: any, i: number) => (
|
||||||
|
<PDFView key={i} style={styles.categorySection} wrap={false}>
|
||||||
|
<PDFView style={styles.categoryHeader}>
|
||||||
|
<PDFView style={styles.categoryIcon} />
|
||||||
|
<PDFText style={styles.categoryTitle}>{cat.category}</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
|
||||||
|
<PDFView style={styles.pagesGrid}>
|
||||||
|
{cat.pages.map((p: any, j: number) => (
|
||||||
|
<PDFView
|
||||||
|
key={j}
|
||||||
|
style={[
|
||||||
|
styles.pageCard,
|
||||||
|
j % 2 === 1 ? { marginRight: 0 } : {},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<PDFText style={styles.pageTitle}>{p.title}</PDFText>
|
||||||
|
{p.desc && (
|
||||||
|
<PDFText style={styles.pageDesc}>{p.desc}</PDFText>
|
||||||
|
)}
|
||||||
|
</PDFView>
|
||||||
|
))}
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
))}
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
);
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import * as React from "react";
|
||||||
|
import {
|
||||||
|
View as PDFView,
|
||||||
|
Text as PDFText,
|
||||||
|
StyleSheet,
|
||||||
|
} from "@react-pdf/renderer";
|
||||||
|
import { DocumentTitle, COLORS, FONT_SIZES } from "../SharedUI.js";
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
section: { marginBottom: 24 },
|
||||||
|
ledgerRow: {
|
||||||
|
paddingVertical: 14,
|
||||||
|
borderBottomWidth: 1,
|
||||||
|
borderBottomColor: COLORS.GRID,
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "flex-start",
|
||||||
|
},
|
||||||
|
moduleLabel: {
|
||||||
|
fontSize: FONT_SIZES.LABEL,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
marginBottom: 4,
|
||||||
|
},
|
||||||
|
moduleDesc: {
|
||||||
|
fontSize: FONT_SIZES.SMALL,
|
||||||
|
color: COLORS.TEXT_DIM,
|
||||||
|
lineHeight: 1.5,
|
||||||
|
},
|
||||||
|
ledgerPrice: {
|
||||||
|
fontSize: FONT_SIZES.BODY,
|
||||||
|
fontWeight: "bold",
|
||||||
|
color: COLORS.CHARCOAL,
|
||||||
|
},
|
||||||
|
ledgerUnit: {
|
||||||
|
fontSize: FONT_SIZES.TINY,
|
||||||
|
color: COLORS.TEXT_LIGHT,
|
||||||
|
marginLeft: 2,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const TransparenzModule = ({ pricing }: any) => {
|
||||||
|
const sorglosPrice = (pricing.HOSTING_MONTHLY || 250) * 12;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<DocumentTitle title="Preis-Transparenz & Modell" isHero={true} />
|
||||||
|
<PDFView style={styles.section}>
|
||||||
|
<PDFView style={{ borderTopWidth: 1, borderTopColor: COLORS.CHARCOAL }}>
|
||||||
|
{[
|
||||||
|
{
|
||||||
|
l: "Fundament",
|
||||||
|
d: "Bereitstellung der techn. Infrastruktur & System-Umgebung.",
|
||||||
|
p: pricing.BASE_WEBSITE,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
l: "Einzelseiten",
|
||||||
|
d: "Individuelle Gestaltung, Layout & responsive Struktur.",
|
||||||
|
p: pricing.PAGE,
|
||||||
|
unit: "/ Stk",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
l: "Core Features",
|
||||||
|
d: "Geschlossene Datensysteme mit eigener Datenstruktur.",
|
||||||
|
p: pricing.FEATURE,
|
||||||
|
unit: "/ Stk",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
l: "Logik & Funktionen",
|
||||||
|
d: "Interaktive Funktions-Bausteine & Prozess-Logik.",
|
||||||
|
p: pricing.FUNCTION,
|
||||||
|
unit: "/ Stk",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
l: "Schnittstellen",
|
||||||
|
d: "Synchronisation mit externen Zielsystemen.",
|
||||||
|
p: pricing.API_INTEGRATION,
|
||||||
|
unit: "/ Stk",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
l: "Sprachversionen",
|
||||||
|
d: "Skalierung der System-Architektur auf Zweit-Sprachen.",
|
||||||
|
p: "+20%",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
l: "Initial-Pflege",
|
||||||
|
d: "Konvertierung & Aufbereitung von Bestandsdaten.",
|
||||||
|
p: pricing.NEW_DATASET,
|
||||||
|
unit: "/ Stk",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
l: "Sorglos Betrieb",
|
||||||
|
d: "Hosting, Instandhaltung, Security & techn. Support.",
|
||||||
|
p: sorglosPrice,
|
||||||
|
unit: "/ Jahr",
|
||||||
|
},
|
||||||
|
].map((item: any, i: number) => (
|
||||||
|
<PDFView key={i} style={styles.ledgerRow}>
|
||||||
|
<PDFView style={{ width: "25%" }}>
|
||||||
|
<PDFText style={styles.moduleLabel}>
|
||||||
|
{item.l.toUpperCase()}
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={{ width: "50%" }}>
|
||||||
|
<PDFText style={styles.moduleDesc}>{item.d}</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
<PDFView style={{ width: "25%", alignItems: "flex-end" }}>
|
||||||
|
<PDFText style={styles.ledgerPrice}>
|
||||||
|
{typeof item.p === "number"
|
||||||
|
? `${item.p.toLocaleString("de-DE")} €`
|
||||||
|
: item.p}
|
||||||
|
{item.unit && (
|
||||||
|
<PDFText style={styles.ledgerUnit}> {item.unit}</PDFText>
|
||||||
|
)}
|
||||||
|
</PDFText>
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
))}
|
||||||
|
</PDFView>
|
||||||
|
</PDFView>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
15
packages/pdf-library/src/index.ts
Normal file
15
packages/pdf-library/src/index.ts
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
export * from "./logic/pricing/types.js";
|
||||||
|
export * from "./logic/pricing/constants.js";
|
||||||
|
export * from "./logic/pricing/calculator.js";
|
||||||
|
export * from "./components/EstimationPDF.js";
|
||||||
|
export * from "./components/pdf/SimpleLayout.js";
|
||||||
|
export * from "./components/pdf/SharedUI.js";
|
||||||
|
export * from "./components/pdf/modules/FrontPageModule.js";
|
||||||
|
export * from "./components/pdf/modules/BriefingModule.js";
|
||||||
|
export * from "./components/pdf/modules/SitemapModule.js";
|
||||||
|
export * from "./components/pdf/modules/EstimationModule.js";
|
||||||
|
export * from "./components/pdf/modules/CommonModules.js";
|
||||||
|
export * from "./components/pdf/modules/BrandingModules.js";
|
||||||
|
export * from "./components/pdf/modules/TransparenzModule.js";
|
||||||
|
export * from "./components/AgbsPDF.js";
|
||||||
|
export * from "./components/CombinedQuotePDF.js";
|
||||||
224
packages/pdf-library/src/logic/pricing/calculator.ts
Normal file
224
packages/pdf-library/src/logic/pricing/calculator.ts
Normal file
@@ -0,0 +1,224 @@
|
|||||||
|
import { FormState, Position, Totals } from "./types.js";
|
||||||
|
import {
|
||||||
|
FEATURE_LABELS,
|
||||||
|
FUNCTION_LABELS,
|
||||||
|
API_LABELS,
|
||||||
|
PAGE_LABELS,
|
||||||
|
} from "./constants.js";
|
||||||
|
|
||||||
|
export function calculateTotals(state: FormState, pricing: any): Totals {
|
||||||
|
if (state.projectType !== "website") {
|
||||||
|
return {
|
||||||
|
totalPrice: 0,
|
||||||
|
monthlyPrice: 0,
|
||||||
|
totalPagesCount: 0,
|
||||||
|
totalFeatures: 0,
|
||||||
|
totalFunctions: 0,
|
||||||
|
totalApis: 0,
|
||||||
|
languagesCount: 0,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const sitemapPagesCount =
|
||||||
|
state.sitemap?.reduce(
|
||||||
|
(acc: number, cat: any) => acc + (cat.pages?.length || 0),
|
||||||
|
0,
|
||||||
|
) || 0;
|
||||||
|
const totalPagesCount = Math.max(
|
||||||
|
(state.selectedPages?.length || 0) +
|
||||||
|
(state.otherPages?.length || 0) +
|
||||||
|
(state.otherPagesCount || 0),
|
||||||
|
sitemapPagesCount,
|
||||||
|
);
|
||||||
|
|
||||||
|
const totalFeatures =
|
||||||
|
(state.features?.length || 0) +
|
||||||
|
(state.otherFeatures?.length || 0) +
|
||||||
|
(state.otherFeaturesCount || 0);
|
||||||
|
const totalFunctions =
|
||||||
|
(state.functions?.length || 0) +
|
||||||
|
(state.otherFunctions?.length || 0) +
|
||||||
|
(state.otherFunctionsCount || 0);
|
||||||
|
const totalApis =
|
||||||
|
(state.apiSystems?.length || 0) +
|
||||||
|
(state.otherTech?.length || 0) +
|
||||||
|
(state.otherTechCount || 0);
|
||||||
|
|
||||||
|
let total = pricing.BASE_WEBSITE;
|
||||||
|
total += totalPagesCount * pricing.PAGE;
|
||||||
|
total += totalFeatures * pricing.FEATURE;
|
||||||
|
total += totalFunctions * pricing.FUNCTION;
|
||||||
|
total += totalApis * pricing.API_INTEGRATION;
|
||||||
|
total += (state.newDatasets || 0) * pricing.NEW_DATASET;
|
||||||
|
|
||||||
|
if (state.cmsSetup) {
|
||||||
|
total += Math.max(1, totalFeatures) * pricing.CMS_CONNECTION_PER_FEATURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const languagesCount = state.languagesList?.length || 1;
|
||||||
|
if (languagesCount > 1) {
|
||||||
|
total *= 1 + (languagesCount - 1) * 0.2;
|
||||||
|
}
|
||||||
|
|
||||||
|
const monthlyPrice =
|
||||||
|
pricing.HOSTING_MONTHLY +
|
||||||
|
(state.storageExpansion || 0) * pricing.STORAGE_EXPANSION_MONTHLY;
|
||||||
|
|
||||||
|
return {
|
||||||
|
totalPrice: Math.round(total),
|
||||||
|
monthlyPrice: Math.round(monthlyPrice),
|
||||||
|
totalPagesCount,
|
||||||
|
totalFeatures,
|
||||||
|
totalFunctions,
|
||||||
|
totalApis,
|
||||||
|
languagesCount,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function calculatePositions(state: FormState, pricing: any): Position[] {
|
||||||
|
const positions: Position[] = [];
|
||||||
|
let pos = 1;
|
||||||
|
|
||||||
|
if (state.projectType === "website") {
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Das technische Fundament",
|
||||||
|
desc: "Projekt-Setup, Infrastruktur, Hosting-Bereitstellung, Grundstruktur & Design-Vorlage, technisches SEO-Basics, Analytics.",
|
||||||
|
qty: 1,
|
||||||
|
price: pricing.BASE_WEBSITE,
|
||||||
|
});
|
||||||
|
|
||||||
|
const sitemapPagesCount =
|
||||||
|
state.sitemap?.reduce(
|
||||||
|
(acc: number, cat: any) => acc + (cat.pages?.length || 0),
|
||||||
|
0,
|
||||||
|
) || 0;
|
||||||
|
const totalPagesCount = Math.max(
|
||||||
|
(state.selectedPages?.length || 0) +
|
||||||
|
(state.otherPages?.length || 0) +
|
||||||
|
(state.otherPagesCount || 0),
|
||||||
|
sitemapPagesCount,
|
||||||
|
);
|
||||||
|
|
||||||
|
const allPages = [
|
||||||
|
...(state.selectedPages || []).map((p: string) => PAGE_LABELS[p] || p),
|
||||||
|
...(state.otherPages || []),
|
||||||
|
...(state.sitemap?.flatMap((cat: any) =>
|
||||||
|
cat.pages?.map((p: any) => p.title),
|
||||||
|
) || []),
|
||||||
|
];
|
||||||
|
|
||||||
|
// Deduplicate labels
|
||||||
|
const uniquePages = Array.from(new Set(allPages));
|
||||||
|
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Individuelle Seiten",
|
||||||
|
desc: `Gestaltung und Umsetzung von ${totalPagesCount} individuellen Seiten-Layouts (${uniquePages.join(", ")}).`,
|
||||||
|
qty: totalPagesCount,
|
||||||
|
price: totalPagesCount * pricing.PAGE,
|
||||||
|
});
|
||||||
|
|
||||||
|
if ((state.features?.length || 0) > 0 || (state.otherFeatures?.length || 0) > 0) {
|
||||||
|
const allFeatures = [
|
||||||
|
...(state.features || []).map((f: string) => FEATURE_LABELS[f] || f),
|
||||||
|
...(state.otherFeatures || []),
|
||||||
|
];
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "System-Module (Features)",
|
||||||
|
desc: `Implementierung funktionaler Bereiche: ${allFeatures.join(", ")}. Inklusive Datenstruktur und Darstellung.`,
|
||||||
|
qty: allFeatures.length,
|
||||||
|
price: allFeatures.length * pricing.FEATURE,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((state.functions?.length || 0) > 0 || (state.otherFunctions?.length || 0) > 0) {
|
||||||
|
const allFunctions = [
|
||||||
|
...(state.functions || []).map((f: string) => FUNCTION_LABELS[f] || f),
|
||||||
|
...(state.otherFunctions || []),
|
||||||
|
];
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Logik-Funktionen",
|
||||||
|
desc: `Implementierung technischer Logik: ${allFunctions.join(", ")}.`,
|
||||||
|
qty: allFunctions.length,
|
||||||
|
price: allFunctions.length * pricing.FUNCTION,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((state.apiSystems?.length || 0) > 0 || (state.otherTech?.length || 0) > 0) {
|
||||||
|
const allApis = [
|
||||||
|
...(state.apiSystems || []).map((a: string) => API_LABELS[a] || a),
|
||||||
|
...(state.otherTech || []),
|
||||||
|
];
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Schnittstellen (API)",
|
||||||
|
desc: `Anbindung externer Systeme zur Datensynchronisation: ${allApis.join(", ")}.`,
|
||||||
|
qty: allApis.length,
|
||||||
|
price: allApis.length * pricing.API_INTEGRATION,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state.cmsSetup) {
|
||||||
|
const totalFeatures =
|
||||||
|
(state.features?.length || 0) +
|
||||||
|
(state.otherFeatures?.length || 0) +
|
||||||
|
(state.otherFeaturesCount || 0);
|
||||||
|
const qty = Math.max(1, totalFeatures);
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Inhalts-Verwaltung",
|
||||||
|
desc: "Anbindung der System-Module an das Redaktions-System zur eigenständigen Pflege von Inhalten.",
|
||||||
|
qty: qty,
|
||||||
|
price: qty * pricing.CMS_CONNECTION_PER_FEATURE,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state.newDatasets > 0) {
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Inhaltliche Initial-Pflege",
|
||||||
|
desc: `Manuelle Übernahme und Aufbereitung von ${state.newDatasets} Datensätzen (Produkte, Artikel) in das Zielsystem.`,
|
||||||
|
qty: state.newDatasets,
|
||||||
|
price: state.newDatasets * pricing.NEW_DATASET,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const languagesCount = state.languagesList?.length || 1;
|
||||||
|
if (languagesCount > 1) {
|
||||||
|
const subtotal = positions.reduce((sum, p) => sum + p.price, 0);
|
||||||
|
const factorPrice = subtotal * ((languagesCount - 1) * 0.2);
|
||||||
|
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Mehrsprachigkeit",
|
||||||
|
desc: `Erweiterung des Systems auf ${languagesCount} Sprachen (Struktur & Logik).`,
|
||||||
|
qty: languagesCount,
|
||||||
|
price: Math.round(factorPrice),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const monthlyRate =
|
||||||
|
pricing.HOSTING_MONTHLY +
|
||||||
|
state.storageExpansion * pricing.STORAGE_EXPANSION_MONTHLY;
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Sorglos Betrieb (1 Jahr)",
|
||||||
|
desc: `Inklusive 1 Jahr Sicherung des technischen Betriebs, Hosting, Instandhaltung, Sicherheits-Updates und techn. Support gemäß AGB Punkt 7a.`,
|
||||||
|
qty: 1,
|
||||||
|
price: monthlyRate * 12,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
positions.push({
|
||||||
|
pos: pos++,
|
||||||
|
title: "Web App / Software Entwicklung",
|
||||||
|
desc: "Individuelle Software-Entwicklung nach Aufwand. Abrechnung erfolgt auf Stundenbasis.",
|
||||||
|
qty: 1,
|
||||||
|
price: 0,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return positions;
|
||||||
|
}
|
||||||
332
packages/pdf-library/src/logic/pricing/constants.ts
Normal file
332
packages/pdf-library/src/logic/pricing/constants.ts
Normal file
@@ -0,0 +1,332 @@
|
|||||||
|
import { FormState } from "./types.js";
|
||||||
|
|
||||||
|
export const PRICING = {
|
||||||
|
BASE_WEBSITE: 5440, // Updated to match AI prompt requirement in Pass 1
|
||||||
|
PAGE: 600,
|
||||||
|
FEATURE: 1500,
|
||||||
|
FUNCTION: 800,
|
||||||
|
NEW_DATASET: 450,
|
||||||
|
HOSTING_MONTHLY: 250,
|
||||||
|
STORAGE_EXPANSION_MONTHLY: 10,
|
||||||
|
CMS_SETUP: 1500,
|
||||||
|
CMS_CONNECTION_PER_FEATURE: 1500,
|
||||||
|
API_INTEGRATION: 800,
|
||||||
|
APP_HOURLY: 120,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const initialState: FormState = {
|
||||||
|
projectType: "website",
|
||||||
|
// Company
|
||||||
|
companyName: "",
|
||||||
|
employeeCount: "",
|
||||||
|
// Existing Presence
|
||||||
|
existingWebsite: "",
|
||||||
|
socialMedia: [],
|
||||||
|
socialMediaUrls: {},
|
||||||
|
existingDomain: "",
|
||||||
|
wishedDomain: "",
|
||||||
|
// Project
|
||||||
|
websiteTopic: "",
|
||||||
|
selectedPages: ["Home"],
|
||||||
|
otherPages: [],
|
||||||
|
otherPagesCount: 0,
|
||||||
|
features: [],
|
||||||
|
otherFeatures: [],
|
||||||
|
otherFeaturesCount: 0,
|
||||||
|
functions: [],
|
||||||
|
otherFunctions: [],
|
||||||
|
otherFunctionsCount: 0,
|
||||||
|
apiSystems: [],
|
||||||
|
otherTech: [],
|
||||||
|
otherTechCount: 0,
|
||||||
|
assets: [],
|
||||||
|
otherAssets: [],
|
||||||
|
otherAssetsCount: 0,
|
||||||
|
newDatasets: 0,
|
||||||
|
cmsSetup: false,
|
||||||
|
storageExpansion: 0,
|
||||||
|
name: "",
|
||||||
|
email: "",
|
||||||
|
role: "",
|
||||||
|
message: "",
|
||||||
|
sitemapFile: null,
|
||||||
|
contactFiles: [],
|
||||||
|
// Design
|
||||||
|
designVibe: "minimal",
|
||||||
|
colorScheme: ["#ffffff", "#f8fafc", "#0f172a"],
|
||||||
|
references: [],
|
||||||
|
designWishes: "",
|
||||||
|
// Maintenance
|
||||||
|
expectedAdjustments: "low",
|
||||||
|
languagesList: ["Deutsch"],
|
||||||
|
personName: "",
|
||||||
|
// Timeline
|
||||||
|
deadline: "flexible",
|
||||||
|
// Web App specific
|
||||||
|
targetAudience: "internal",
|
||||||
|
userRoles: [],
|
||||||
|
dataSensitivity: "standard",
|
||||||
|
platformType: "web-only",
|
||||||
|
// Meta
|
||||||
|
dontKnows: [],
|
||||||
|
visualStaging: "standard",
|
||||||
|
complexInteractions: "standard",
|
||||||
|
// AI generated / Post-processed
|
||||||
|
briefingSummary: "",
|
||||||
|
designVision: "",
|
||||||
|
positionDescriptions: {},
|
||||||
|
taxId: "",
|
||||||
|
sitemap: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PAGE_SAMPLES = [
|
||||||
|
{ id: "Home", label: "Startseite", desc: "Der erste Eindruck Ihrer Marke." },
|
||||||
|
{ id: "About", label: "Über uns", desc: "Ihre Geschichte und Ihr Team." },
|
||||||
|
{ id: "Services", label: "Leistungen", desc: "Übersicht Ihres Angebots." },
|
||||||
|
{ id: "Contact", label: "Kontakt", desc: "Anlaufstelle für Ihre Kunden." },
|
||||||
|
{
|
||||||
|
id: "Landing",
|
||||||
|
label: "Landingpage",
|
||||||
|
desc: "Optimiert für Marketing-Kampagnen.",
|
||||||
|
},
|
||||||
|
{ id: "Legal", label: "Rechtliches", desc: "Impressum & Datenschutz." },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const FEATURE_OPTIONS = [
|
||||||
|
{
|
||||||
|
id: "blog_news",
|
||||||
|
label: "Blog / News",
|
||||||
|
desc: "Ein Bereich für aktuelle Beiträge und Neuigkeiten.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "products",
|
||||||
|
label: "Produktbereich",
|
||||||
|
desc: "Katalog Ihrer Leistungen oder Produkte.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "jobs",
|
||||||
|
label: "Karriere / Jobs",
|
||||||
|
desc: "Stellenanzeigen und Bewerbungsoptionen.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "refs",
|
||||||
|
label: "Referenzen / Cases",
|
||||||
|
desc: "Präsentation Ihrer Projekte.",
|
||||||
|
},
|
||||||
|
{ id: "events", label: "Events / Termine", desc: "Veranstaltungskalender." },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const FUNCTION_OPTIONS = [
|
||||||
|
{ id: "search", label: "Suche", desc: "Volltextsuche über alle Inhalte." },
|
||||||
|
{
|
||||||
|
id: "filter",
|
||||||
|
label: "Filter-Systeme",
|
||||||
|
desc: "Kategorisierung und Sortierung.",
|
||||||
|
},
|
||||||
|
{ id: "pdf", label: "PDF-Export", desc: "Automatisierte PDF-Erstellung." },
|
||||||
|
{
|
||||||
|
id: "forms",
|
||||||
|
label: "Individuelle Formular-Logik",
|
||||||
|
desc: "Smarte Validierung & mehrstufige Prozesse.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const API_OPTIONS = [
|
||||||
|
{
|
||||||
|
id: "crm",
|
||||||
|
label: "CRM System",
|
||||||
|
desc: "HubSpot, Salesforce, Pipedrive etc.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "erp",
|
||||||
|
label: "ERP / Warenwirtschaft",
|
||||||
|
desc: "SAP, Microsoft Dynamics, Xentral etc.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "stripe",
|
||||||
|
label: "Stripe / Payment",
|
||||||
|
desc: "Zahlungsabwicklung und Abonnements.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "newsletter",
|
||||||
|
label: "Newsletter / Marketing",
|
||||||
|
desc: "Mailchimp, Brevo, ActiveCampaign etc.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "ecommerce",
|
||||||
|
label: "E-Commerce / Shop",
|
||||||
|
desc: "Shopify, WooCommerce, Shopware Sync.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "hr",
|
||||||
|
label: "HR / Recruiting",
|
||||||
|
desc: "Personio, Workday, Recruitee etc.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "realestate",
|
||||||
|
label: "Immobilien",
|
||||||
|
desc: "OpenImmo, FlowFact, Immowelt Sync.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "calendar",
|
||||||
|
label: "Termine / Booking",
|
||||||
|
desc: "Calendly, Shore, Doctolib etc.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "social",
|
||||||
|
label: "Social Media Sync",
|
||||||
|
desc: "Automatisierte Posts oder Feeds.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "maps",
|
||||||
|
label: "Google Maps / Places",
|
||||||
|
desc: "Standortsuche und Kartenintegration.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "analytics",
|
||||||
|
label: "Custom Analytics",
|
||||||
|
desc: "Anbindung an spezialisierte Tracking-Tools.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const ASSET_OPTIONS = [
|
||||||
|
{
|
||||||
|
id: "existing_website",
|
||||||
|
label: "Bestehende Website",
|
||||||
|
desc: "Inhalte oder Struktur können übernommen werden.",
|
||||||
|
},
|
||||||
|
{ id: "logo", label: "Logo", desc: "Vektordatei Ihres Logos." },
|
||||||
|
{
|
||||||
|
id: "styleguide",
|
||||||
|
label: "Styleguide",
|
||||||
|
desc: "Farben, Schriften, Design-Vorgaben.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "content_concept",
|
||||||
|
label: "Inhalts-Konzept",
|
||||||
|
desc: "Struktur und Texte sind bereits geplant.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "media",
|
||||||
|
label: "Bild/Video-Material",
|
||||||
|
desc: "Professionelles Bildmaterial vorhanden.",
|
||||||
|
},
|
||||||
|
{ id: "icons", label: "Icons", desc: "Eigene Icon-Sets vorhanden." },
|
||||||
|
{
|
||||||
|
id: "illustrations",
|
||||||
|
label: "Illustrationen",
|
||||||
|
desc: "Eigene Illustrationen vorhanden.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "fonts",
|
||||||
|
label: "Fonts",
|
||||||
|
desc: "Lizenzen für Hausschriften vorhanden.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const DESIGN_OPTIONS = [
|
||||||
|
{
|
||||||
|
id: "minimal",
|
||||||
|
label: "Minimalistisch",
|
||||||
|
desc: "Viel Weißraum, klare Typografie.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "bold",
|
||||||
|
label: "Mutig & Laut",
|
||||||
|
desc: "Starke Kontraste, große Schriften.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "nature",
|
||||||
|
label: "Natürlich",
|
||||||
|
desc: "Sanfte Erdtöne, organische Formen.",
|
||||||
|
},
|
||||||
|
{ id: "tech", label: "Technisch", desc: "Präzise Linien, dunkle Akzente." },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const EMPLOYEE_OPTIONS = [
|
||||||
|
{ id: "1-5", label: "1-5 Mitarbeiter" },
|
||||||
|
{ id: "6-20", label: "6-20 Mitarbeiter" },
|
||||||
|
{ id: "21-100", label: "21-100 Mitarbeiter" },
|
||||||
|
{ id: "100+", label: "100+ Mitarbeiter" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const SOCIAL_MEDIA_OPTIONS = [
|
||||||
|
{ id: "instagram", label: "Instagram" },
|
||||||
|
{ id: "linkedin", label: "LinkedIn" },
|
||||||
|
{ id: "facebook", label: "Facebook" },
|
||||||
|
{ id: "twitter", label: "Twitter / X" },
|
||||||
|
{ id: "tiktok", label: "TikTok" },
|
||||||
|
{ id: "youtube", label: "YouTube" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const VIBE_LABELS: Record<string, string> = {
|
||||||
|
minimal: "Minimalistisch",
|
||||||
|
bold: "Mutig & Laut",
|
||||||
|
nature: "Natürlich",
|
||||||
|
tech: "Technisch",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const DEADLINE_LABELS: Record<string, string> = {
|
||||||
|
asap: "So schnell wie möglich",
|
||||||
|
"2-3-months": "In 2-3 Monaten",
|
||||||
|
"3-6-months": "In 3-6 Monaten",
|
||||||
|
flexible: "Flexibel",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ASSET_LABELS: Record<string, string> = {
|
||||||
|
existing_website: "Bestehende Website",
|
||||||
|
logo: "Logo",
|
||||||
|
styleguide: "Styleguide",
|
||||||
|
content_concept: "Inhalts-Konzept",
|
||||||
|
media: "Bild/Video-Material",
|
||||||
|
icons: "Icons",
|
||||||
|
illustrations: "Illustrationen",
|
||||||
|
fonts: "Fonts",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const FEATURE_LABELS: Record<string, string> = {
|
||||||
|
blog_news: "Blog / News",
|
||||||
|
products: "Produktbereich",
|
||||||
|
jobs: "Karriere / Jobs",
|
||||||
|
refs: "Referenzen / Cases",
|
||||||
|
events: "Events / Termine",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const FUNCTION_LABELS: Record<string, string> = {
|
||||||
|
search: "Suche",
|
||||||
|
filter: "Filter-Systeme",
|
||||||
|
pdf: "PDF-Export",
|
||||||
|
forms: "Individuelle Formular-Logik",
|
||||||
|
members: "Mitgliederbereich",
|
||||||
|
calendar: "Event-Kalender",
|
||||||
|
multilang: "Mehrsprachigkeit",
|
||||||
|
chat: "Echtzeit-Chat",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const API_LABELS: Record<string, string> = {
|
||||||
|
crm_erp: "CRM / ERP",
|
||||||
|
payment: "Payment",
|
||||||
|
marketing: "Marketing",
|
||||||
|
ecommerce: "E-Commerce",
|
||||||
|
maps: "Google Maps / Places",
|
||||||
|
social: "Social Media Sync",
|
||||||
|
analytics: "Custom Analytics",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SOCIAL_LABELS: Record<string, string> = {
|
||||||
|
instagram: "Instagram",
|
||||||
|
linkedin: "LinkedIn",
|
||||||
|
facebook: "Facebook",
|
||||||
|
twitter: "Twitter / X",
|
||||||
|
tiktok: "TikTok",
|
||||||
|
youtube: "YouTube",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const PAGE_LABELS: Record<string, string> = {
|
||||||
|
Home: "Startseite",
|
||||||
|
About: "Über uns",
|
||||||
|
Services: "Leistungen",
|
||||||
|
Contact: "Kontakt",
|
||||||
|
Landing: "Landingpage",
|
||||||
|
Legal: "Impressum & Datenschutz",
|
||||||
|
};
|
||||||
89
packages/pdf-library/src/logic/pricing/types.ts
Normal file
89
packages/pdf-library/src/logic/pricing/types.ts
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
export type ProjectType = 'website' | 'web-app';
|
||||||
|
|
||||||
|
export interface FormState {
|
||||||
|
projectType: ProjectType;
|
||||||
|
// Company
|
||||||
|
companyName: string;
|
||||||
|
employeeCount: string;
|
||||||
|
// Existing Presence
|
||||||
|
existingWebsite: string;
|
||||||
|
socialMedia: string[];
|
||||||
|
socialMediaUrls: Record<string, string>;
|
||||||
|
existingDomain: string;
|
||||||
|
wishedDomain: string;
|
||||||
|
// Project
|
||||||
|
websiteTopic: string;
|
||||||
|
selectedPages: string[];
|
||||||
|
otherPages: string[];
|
||||||
|
otherPagesCount: number;
|
||||||
|
features: string[];
|
||||||
|
otherFeatures: string[];
|
||||||
|
otherFeaturesCount: number;
|
||||||
|
functions: string[];
|
||||||
|
otherFunctions: string[];
|
||||||
|
otherFunctionsCount: number;
|
||||||
|
apiSystems: string[];
|
||||||
|
otherTech: string[];
|
||||||
|
otherTechCount: number;
|
||||||
|
assets: string[];
|
||||||
|
otherAssets: string[];
|
||||||
|
otherAssetsCount: number;
|
||||||
|
newDatasets: number;
|
||||||
|
cmsSetup: boolean;
|
||||||
|
storageExpansion: number;
|
||||||
|
name: string;
|
||||||
|
email: string;
|
||||||
|
role: string;
|
||||||
|
message: string;
|
||||||
|
sitemapFile: any;
|
||||||
|
contactFiles: any[];
|
||||||
|
// Design
|
||||||
|
designVibe: string;
|
||||||
|
colorScheme: string[];
|
||||||
|
references: string[];
|
||||||
|
designWishes: string;
|
||||||
|
// Maintenance
|
||||||
|
expectedAdjustments: string;
|
||||||
|
languagesList: string[];
|
||||||
|
// Timeline
|
||||||
|
deadline: string;
|
||||||
|
// Web App specific
|
||||||
|
targetAudience: string;
|
||||||
|
userRoles: string[];
|
||||||
|
dataSensitivity: string;
|
||||||
|
platformType: string;
|
||||||
|
// Meta
|
||||||
|
dontKnows: string[];
|
||||||
|
visualStaging: string;
|
||||||
|
complexInteractions: string;
|
||||||
|
gridDontKnows?: Record<string, string>;
|
||||||
|
briefingSummary?: string;
|
||||||
|
companyAddress?: string;
|
||||||
|
companyPhone?: string;
|
||||||
|
personName?: string;
|
||||||
|
taxId?: string;
|
||||||
|
designVision?: string;
|
||||||
|
positionDescriptions?: Record<string, string>;
|
||||||
|
sitemap?: {
|
||||||
|
category: string;
|
||||||
|
pages: { title: string; desc: string }[];
|
||||||
|
}[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Position {
|
||||||
|
pos: number;
|
||||||
|
title: string;
|
||||||
|
desc: string;
|
||||||
|
qty: number;
|
||||||
|
price: number;
|
||||||
|
isRecurring?: boolean;
|
||||||
|
}
|
||||||
|
export interface Totals {
|
||||||
|
totalPrice: number;
|
||||||
|
monthlyPrice: number;
|
||||||
|
totalPagesCount: number;
|
||||||
|
totalFeatures: number;
|
||||||
|
totalFunctions: number;
|
||||||
|
totalApis: number;
|
||||||
|
languagesCount: number;
|
||||||
|
}
|
||||||
3
packages/pdf-library/src/server.ts
Normal file
3
packages/pdf-library/src/server.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export * from "./index.js";
|
||||||
|
export * from "./services/AcquisitionService.js";
|
||||||
|
export * from "./services/PdfEngine.js";
|
||||||
153
packages/pdf-library/src/services/AcquisitionService.ts
Normal file
153
packages/pdf-library/src/services/AcquisitionService.ts
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
import { CheerioCrawler } from "@crawlee/cheerio";
|
||||||
|
import axios from "axios";
|
||||||
|
import { FileCacheAdapter } from "../utils/cache/FileCacheAdapter.js";
|
||||||
|
import { initialState } from "../logic/pricing/constants.js";
|
||||||
|
import { FormState } from "../logic/pricing/types.js";
|
||||||
|
|
||||||
|
export interface AcquisitionResult {
|
||||||
|
state: FormState;
|
||||||
|
usage: {
|
||||||
|
prompt: number;
|
||||||
|
completion: number;
|
||||||
|
cost: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export class AcquisitionService {
|
||||||
|
private cache: FileCacheAdapter;
|
||||||
|
private openRouterKey: string;
|
||||||
|
|
||||||
|
constructor(openRouterKey: string) {
|
||||||
|
this.openRouterKey = openRouterKey;
|
||||||
|
this.cache = new FileCacheAdapter({ prefix: "acq_" });
|
||||||
|
}
|
||||||
|
|
||||||
|
async runFullSequence(url: string, briefing: string, comments?: string): Promise<AcquisitionResult> {
|
||||||
|
console.log(`🚀 Starting Acquisition Sequence for: ${url}`);
|
||||||
|
|
||||||
|
// 1. Crawl
|
||||||
|
const crawlData = await this.performCrawl(url);
|
||||||
|
|
||||||
|
// 2. Distill
|
||||||
|
const distilledContext = await this.distillCrawlContext(crawlData);
|
||||||
|
|
||||||
|
// 3. AI Estimation (using parts of the original ai-estimate logic)
|
||||||
|
// For brevity in this initial port, I'll implement a combined prompt strategy
|
||||||
|
// or keep the multi-pass if needed.
|
||||||
|
|
||||||
|
const result = await this.getAiEstimation(briefing, distilledContext, comments || null);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async performCrawl(url: string): Promise<string> {
|
||||||
|
const pages: any[] = [];
|
||||||
|
const origin = new URL(url).origin;
|
||||||
|
|
||||||
|
const crawler = new CheerioCrawler({
|
||||||
|
maxRequestsPerCrawl: 15,
|
||||||
|
async requestHandler({ $, request, enqueueLinks }) {
|
||||||
|
const title = $("title").text();
|
||||||
|
const bodyText = $("body").text().replace(/\s+/g, " ").substring(0, 10000);
|
||||||
|
|
||||||
|
pages.push({
|
||||||
|
url: request.url,
|
||||||
|
content: `Title: ${title}\nText: ${bodyText}`,
|
||||||
|
});
|
||||||
|
|
||||||
|
await enqueueLinks({
|
||||||
|
limit: 10,
|
||||||
|
transformRequestFunction: (req) => {
|
||||||
|
try {
|
||||||
|
const reqUrl = new URL(req.url);
|
||||||
|
if (reqUrl.origin !== origin) return false;
|
||||||
|
if (reqUrl.pathname.match(/\.(pdf|zip|jpg|png|svg|webp)$/i)) return false;
|
||||||
|
return req;
|
||||||
|
} catch (_error) {
|
||||||
|
// Ignored - malformed URL in enqueueLinks
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await crawler.run([url]);
|
||||||
|
return pages.map((p) => `--- PAGE: ${p.url} ---\n${p.content}`).join("\n\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
private async distillCrawlContext(rawCrawl: string): Promise<string> {
|
||||||
|
const systemPrompt = `
|
||||||
|
You are a context distiller. Extract the "Company DNA" in 5-8 bullet points (GERMAN).
|
||||||
|
Focus on: Services, USPs, Target Audience, Tone.
|
||||||
|
`;
|
||||||
|
const resp = await axios.post(
|
||||||
|
"https://openrouter.ai/api/v1/chat/completions",
|
||||||
|
{
|
||||||
|
model: "google/gemini-3-flash-preview",
|
||||||
|
messages: [
|
||||||
|
{ role: "system", content: systemPrompt },
|
||||||
|
{ role: "user", content: `RAW_CRAWL:\n${rawCrawl.substring(0, 20000)}` },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: { Authorization: `Bearer ${this.openRouterKey}` },
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return resp.data.choices[0].message.content;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async getAiEstimation(briefing: string, context: string, comments: string | null): Promise<AcquisitionResult> {
|
||||||
|
// Porting a simplified version of Pass 1 and Pass 3 together for the "Audit"
|
||||||
|
const systemPrompt = `
|
||||||
|
You are a Digital Architect. Analyze the briefing and crawl context.
|
||||||
|
Generate a JSON state for a project estimation.
|
||||||
|
Language: GERMAN.
|
||||||
|
Format: ROOT LEVEL JSON matching FormState interface.
|
||||||
|
|
||||||
|
### PRICING RULES:
|
||||||
|
- Base: 5440 €
|
||||||
|
- Page: 600 €
|
||||||
|
- Feature: 1500 €
|
||||||
|
- Function/API: 800 €
|
||||||
|
|
||||||
|
Return ONLY the JSON.
|
||||||
|
`;
|
||||||
|
const userPrompt = `BRIEFING: ${briefing}\n\nCONTEXT: ${context}\n\nCOMMENTS: ${comments}`;
|
||||||
|
|
||||||
|
const resp = await axios.post(
|
||||||
|
"https://openrouter.ai/api/v1/chat/completions",
|
||||||
|
{
|
||||||
|
model: "google/gemini-3-flash-preview",
|
||||||
|
messages: [
|
||||||
|
{ role: "system", content: systemPrompt },
|
||||||
|
{ role: "user", content: userPrompt },
|
||||||
|
],
|
||||||
|
response_format: { type: "json_object" },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: { Authorization: `Bearer ${this.openRouterKey}` },
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
let state: FormState;
|
||||||
|
try {
|
||||||
|
state = JSON.parse(resp.data.choices[0].message.content);
|
||||||
|
} catch (_error) {
|
||||||
|
console.error("Failed to parse AI estimation JSON, returning initial state.");
|
||||||
|
state = initialState;
|
||||||
|
}
|
||||||
|
// Ensure it matches FormState defaults
|
||||||
|
const finalState = { ...initialState, ...state };
|
||||||
|
|
||||||
|
return {
|
||||||
|
state: finalState,
|
||||||
|
usage: {
|
||||||
|
prompt: resp.data.usage?.prompt_tokens || 0,
|
||||||
|
completion: resp.data.usage?.completion_tokens || 0,
|
||||||
|
cost: resp.data.usage?.cost || 0,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
24
packages/pdf-library/src/services/PdfEngine.ts
Normal file
24
packages/pdf-library/src/services/PdfEngine.ts
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import { renderToFile } from "@react-pdf/renderer";
|
||||||
|
import { createElement } from "react";
|
||||||
|
import { EstimationPDF } from "../components/EstimationPDF.js";
|
||||||
|
import { PRICING } from "../logic/pricing/constants.js";
|
||||||
|
import { calculateTotals } from "../logic/pricing/calculator.js";
|
||||||
|
|
||||||
|
export class PdfEngine {
|
||||||
|
constructor() { }
|
||||||
|
|
||||||
|
async generateEstimatePdf(state: any, outputPath: string): Promise<string> {
|
||||||
|
const totals = calculateTotals(state, PRICING);
|
||||||
|
|
||||||
|
await renderToFile(
|
||||||
|
createElement(EstimationPDF as any, {
|
||||||
|
state,
|
||||||
|
totalPrice: totals.totalPrice,
|
||||||
|
pricing: PRICING,
|
||||||
|
} as any) as any,
|
||||||
|
outputPath
|
||||||
|
);
|
||||||
|
|
||||||
|
return outputPath;
|
||||||
|
}
|
||||||
|
}
|
||||||
78
packages/pdf-library/src/utils/cache/FileCacheAdapter.ts
vendored
Normal file
78
packages/pdf-library/src/utils/cache/FileCacheAdapter.ts
vendored
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
import * as fs from 'node:fs/promises';
|
||||||
|
import * as path from 'node:path';
|
||||||
|
import { existsSync } from 'node:fs';
|
||||||
|
import * as crypto from 'node:crypto';
|
||||||
|
|
||||||
|
export class FileCacheAdapter {
|
||||||
|
private cacheDir: string;
|
||||||
|
private prefix: string;
|
||||||
|
private defaultTTL: number;
|
||||||
|
|
||||||
|
constructor(config?: { cacheDir?: string; prefix?: string; defaultTTL?: number }) {
|
||||||
|
this.cacheDir = config?.cacheDir || path.resolve(process.cwd(), '.cache');
|
||||||
|
this.prefix = config?.prefix || '';
|
||||||
|
this.defaultTTL = config?.defaultTTL || 3600;
|
||||||
|
|
||||||
|
if (!existsSync(this.cacheDir)) {
|
||||||
|
fs.mkdir(this.cacheDir, { recursive: true }).catch(() => { });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private sanitize(key: string): string {
|
||||||
|
const clean = key.replace(/[^a-z0-9]/gi, '_');
|
||||||
|
if (clean.length > 64) {
|
||||||
|
return crypto.createHash('md5').update(key).digest('hex');
|
||||||
|
}
|
||||||
|
return clean;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getFilePath(key: string): string {
|
||||||
|
const safeKey = this.sanitize(`${this.prefix}${key}`).toLowerCase();
|
||||||
|
return path.join(this.cacheDir, `${safeKey}.json`);
|
||||||
|
}
|
||||||
|
|
||||||
|
async get<T>(key: string): Promise<T | null> {
|
||||||
|
const filePath = this.getFilePath(key);
|
||||||
|
try {
|
||||||
|
if (!existsSync(filePath)) return null;
|
||||||
|
const content = await fs.readFile(filePath, 'utf8');
|
||||||
|
const data = JSON.parse(content);
|
||||||
|
|
||||||
|
if (data.expiry && Date.now() > data.expiry) {
|
||||||
|
await this.del(key);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return data.value;
|
||||||
|
} catch (_error) {
|
||||||
|
return null; // Keeping original return type Promise<T | null>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async set<T>(key: string, value: T, ttl?: number): Promise<void> {
|
||||||
|
const filePath = this.getFilePath(key);
|
||||||
|
const effectiveTTL = ttl !== undefined ? ttl : this.defaultTTL;
|
||||||
|
const data = {
|
||||||
|
value,
|
||||||
|
expiry: effectiveTTL > 0 ? Date.now() + effectiveTTL * 1000 : null,
|
||||||
|
updatedAt: new Date().toISOString()
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fs.writeFile(filePath, JSON.stringify(data, null, 2), 'utf8');
|
||||||
|
} catch (_error) {
|
||||||
|
// Ignore
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async del(key: string): Promise<void> {
|
||||||
|
const filePath = this.getFilePath(key);
|
||||||
|
try {
|
||||||
|
if (existsSync(filePath)) {
|
||||||
|
await fs.unlink(filePath);
|
||||||
|
}
|
||||||
|
} catch (_error) {
|
||||||
|
// Ignored - best effort cleanup
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
24
packages/pdf-library/tsconfig.json
Normal file
24
packages/pdf-library/tsconfig.json
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"target": "ESNext",
|
||||||
|
"module": "ESNext",
|
||||||
|
"moduleResolution": "node",
|
||||||
|
"jsx": "react-jsx",
|
||||||
|
"declaration": true,
|
||||||
|
"declarationDir": "dist",
|
||||||
|
"emitDeclarationOnly": true,
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"esModuleInterop": true,
|
||||||
|
"allowSyntheticDefaultImports": true,
|
||||||
|
"strict": true,
|
||||||
|
"outDir": "dist"
|
||||||
|
},
|
||||||
|
"include": [
|
||||||
|
"src/**/*"
|
||||||
|
],
|
||||||
|
"exclude": [
|
||||||
|
"node_modules",
|
||||||
|
"dist",
|
||||||
|
"build.mjs"
|
||||||
|
]
|
||||||
|
}
|
||||||
1
packages/people-manager/index.js
Normal file
1
packages/people-manager/index.js
Normal file
File diff suppressed because one or more lines are too long
30
packages/people-manager/package.json
Normal file
30
packages/people-manager/package.json
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "people-manager",
|
||||||
|
"description": "Custom High-Fidelity People Management for Directus",
|
||||||
|
"icon": "person",
|
||||||
|
"version": "1.7.12",
|
||||||
|
"type": "module",
|
||||||
|
"keywords": [
|
||||||
|
"directus",
|
||||||
|
"directus-extension",
|
||||||
|
"directus-extension-module"
|
||||||
|
],
|
||||||
|
"files": [
|
||||||
|
"dist"
|
||||||
|
],
|
||||||
|
"directus:extension": {
|
||||||
|
"type": "module",
|
||||||
|
"path": "index.js",
|
||||||
|
"source": "src/index.ts",
|
||||||
|
"host": "*",
|
||||||
|
"name": "People Manager"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "directus-extension build && (cp -f dist/index.js index.js 2>/dev/null || true)",
|
||||||
|
"dev": "directus-extension build -w"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@directus/extensions-sdk": "11.0.2",
|
||||||
|
"vue": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
14
packages/people-manager/src/index.ts
Normal file
14
packages/people-manager/src/index.ts
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import { defineModule } from '@directus/extensions-sdk';
|
||||||
|
import ModuleComponent from './module.vue';
|
||||||
|
|
||||||
|
export default defineModule({
|
||||||
|
id: 'people-manager',
|
||||||
|
name: 'People Manager',
|
||||||
|
icon: 'person',
|
||||||
|
routes: [
|
||||||
|
{
|
||||||
|
path: '',
|
||||||
|
component: ModuleComponent,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
296
packages/people-manager/src/module.vue
Normal file
296
packages/people-manager/src/module.vue
Normal file
@@ -0,0 +1,296 @@
|
|||||||
|
<template>
|
||||||
|
<private-view title="People Manager">
|
||||||
|
<template #navigation>
|
||||||
|
<v-list nav>
|
||||||
|
<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-content>
|
||||||
|
<v-text-overflow text="Neue Person anlegen" />
|
||||||
|
</v-list-item-content>
|
||||||
|
</v-list-item>
|
||||||
|
|
||||||
|
<v-divider />
|
||||||
|
|
||||||
|
<v-list-item
|
||||||
|
v-for="person in people"
|
||||||
|
:key="person.id"
|
||||||
|
:active="selectedPerson?.id === person.id"
|
||||||
|
class="person-item"
|
||||||
|
clickable
|
||||||
|
@click="selectPerson(person)"
|
||||||
|
>
|
||||||
|
<v-list-item-icon>
|
||||||
|
<v-icon name="person" />
|
||||||
|
</v-list-item-icon>
|
||||||
|
<v-list-item-content>
|
||||||
|
<v-text-overflow :text="`${person.first_name} ${person.last_name}`" />
|
||||||
|
</v-list-item-content>
|
||||||
|
</v-list-item>
|
||||||
|
</v-list>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<div class="content-wrapper">
|
||||||
|
<v-notice v-if="feedback" :type="feedback.type" @close="feedback = null" dismissible>
|
||||||
|
{{ feedback.message }}
|
||||||
|
</v-notice>
|
||||||
|
|
||||||
|
<div v-if="!selectedPerson" class="empty-state">
|
||||||
|
<v-info title="Person auswählen" icon="person" center>
|
||||||
|
Wähle eine Person in der Navigation aus oder
|
||||||
|
<v-button x-small @click="openCreateDrawer">erstelle eine neue Person</v-button>.
|
||||||
|
</v-info>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else>
|
||||||
|
<header class="header">
|
||||||
|
<div class="header-left">
|
||||||
|
<h1 class="title">{{ selectedPerson.first_name }} {{ selectedPerson.last_name }}</h1>
|
||||||
|
<p class="subtitle">{{ selectedPerson.email || 'Keine E-Mail angegeben' }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="header-right">
|
||||||
|
<v-button secondary rounded icon v-tooltip="'Person bearbeiten'" @click="openEditDrawer">
|
||||||
|
<v-icon name="edit" />
|
||||||
|
</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">Organisation</span>
|
||||||
|
<p class="value">{{ selectedPerson.company || '---' }}</p>
|
||||||
|
</div>
|
||||||
|
<div class="detail-item">
|
||||||
|
<span class="label">Telefon</span>
|
||||||
|
<p class="value">{{ selectedPerson.phone || '---' }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Create/Edit Drawer -->
|
||||||
|
<v-drawer
|
||||||
|
v-model="drawerActive"
|
||||||
|
:title="isEditing ? 'Person bearbeiten' : 'Neue Person anlegen'"
|
||||||
|
icon="person"
|
||||||
|
@cancel="drawerActive = false"
|
||||||
|
>
|
||||||
|
<template #default>
|
||||||
|
<div class="drawer-content">
|
||||||
|
<div class="form-section">
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Vorname</span>
|
||||||
|
<v-input v-model="form.first_name" placeholder="Vorname" autofocus />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Nachname</span>
|
||||||
|
<v-input v-model="form.last_name" placeholder="Nachname" />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">E-Mail</span>
|
||||||
|
<v-input v-model="form.email" placeholder="email@beispiel.de" type="email" />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Organisation / Firma</span>
|
||||||
|
<v-input v-model="form.company" placeholder="z.B. Mintel" />
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<span class="label">Telefon</span>
|
||||||
|
<v-input v-model="form.phone" placeholder="+49 ..." />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="drawer-actions">
|
||||||
|
<v-button primary block :loading="saving" @click="savePerson">
|
||||||
|
Person speichern
|
||||||
|
</v-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</v-drawer>
|
||||||
|
</private-view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, onMounted } from 'vue';
|
||||||
|
import { useApi } from '@directus/extensions-sdk';
|
||||||
|
|
||||||
|
const api = useApi();
|
||||||
|
const people = ref([]);
|
||||||
|
const selectedPerson = ref(null);
|
||||||
|
const feedback = ref(null);
|
||||||
|
const saving = ref(false);
|
||||||
|
const drawerActive = ref(false);
|
||||||
|
const isEditing = ref(false);
|
||||||
|
|
||||||
|
const form = ref({
|
||||||
|
id: null,
|
||||||
|
first_name: '',
|
||||||
|
last_name: '',
|
||||||
|
email: '',
|
||||||
|
company: '',
|
||||||
|
phone: ''
|
||||||
|
});
|
||||||
|
|
||||||
|
async function fetchPeople() {
|
||||||
|
try {
|
||||||
|
const response = await api.get('/items/people', {
|
||||||
|
params: { sort: 'last_name' }
|
||||||
|
});
|
||||||
|
people.ref = response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch people:', error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectPerson(person) {
|
||||||
|
selectedPerson.value = person;
|
||||||
|
}
|
||||||
|
|
||||||
|
function openCreateDrawer() {
|
||||||
|
isEditing.value = false;
|
||||||
|
form.value = { id: null, first_name: '', last_name: '', email: '', company: '', phone: '' };
|
||||||
|
drawerActive.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
function openEditDrawer() {
|
||||||
|
isEditing.value = true;
|
||||||
|
form.value = { ...selectedPerson.value };
|
||||||
|
drawerActive.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function savePerson() {
|
||||||
|
if (!form.value.first_name || !form.value.last_name) {
|
||||||
|
feedback.value = { type: 'danger', message: 'Vor- und Nachname sind erforderlich.' };
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
saving.value = true;
|
||||||
|
try {
|
||||||
|
if (isEditing.value) {
|
||||||
|
await api.patch(`/items/people/${form.value.id}`, form.value);
|
||||||
|
feedback.value = { type: 'success', message: 'Person aktualisiert!' };
|
||||||
|
} else {
|
||||||
|
await api.post('/items/people', form.value);
|
||||||
|
feedback.value = { type: 'success', message: 'Person angelegt!' };
|
||||||
|
}
|
||||||
|
drawerActive.value = false;
|
||||||
|
await fetchPeople();
|
||||||
|
if (isEditing.value) {
|
||||||
|
selectedPerson.value = form.value;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
feedback.value = { type: 'danger', message: error.message };
|
||||||
|
} finally {
|
||||||
|
saving.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deletePerson() {
|
||||||
|
if (!confirm('Soll diese Person wirklich gelöscht werden?')) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await api.delete(`/items/people/${selectedPerson.value.id}`);
|
||||||
|
feedback.value = { type: 'success', message: 'Person gelöscht.' };
|
||||||
|
selectedPerson.value = null;
|
||||||
|
await fetchPeople();
|
||||||
|
} catch (error) {
|
||||||
|
feedback.value = { type: 'danger', message: error.message };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(fetchPeople);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<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 {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(2, 1fr);
|
||||||
|
gap: 32px;
|
||||||
|
margin-top: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.detail-item {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.label {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: var(--theme--foreground-subdued);
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.value {
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.drawer-content {
|
||||||
|
padding: 24px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-section {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.field {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.drawer-actions {
|
||||||
|
margin-top: 24px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@mintel/tsconfig",
|
"name": "@mintel/tsconfig",
|
||||||
"version": "1.7.3",
|
"version": "1.7.12",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
"access": "public",
|
"access": "public",
|
||||||
"registry": "https://npm.infra.mintel.me"
|
"registry": "https://npm.infra.mintel.me"
|
||||||
|
|||||||
5231
pnpm-lock.yaml
generated
5231
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -37,7 +37,7 @@ esac
|
|||||||
|
|
||||||
# Detect local containers
|
# Detect local containers
|
||||||
echo "🔍 Detecting local database..."
|
echo "🔍 Detecting local database..."
|
||||||
LOCAL_DB_CONTAINER=$(docker compose ps -q directus-db)
|
LOCAL_DB_CONTAINER=$(docker compose ps -q at-mintel-directus-db)
|
||||||
if [ -z "$LOCAL_DB_CONTAINER" ]; then
|
if [ -z "$LOCAL_DB_CONTAINER" ]; then
|
||||||
echo "❌ Local directus-db container not found. Is it running? (npm run dev)"
|
echo "❌ Local directus-db container not found. Is it running? (npm run dev)"
|
||||||
exit 1
|
exit 1
|
||||||
|
|||||||
74
scripts/sync-extensions.sh
Executable file
74
scripts/sync-extensions.sh
Executable file
@@ -0,0 +1,74 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Configuration
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
REPO_ROOT="$(cd "$SCRIPT_DIR/.." && pwd)"
|
||||||
|
EXTENSIONS_ROOT="$REPO_ROOT/packages"
|
||||||
|
TARGET_DIR="$REPO_ROOT/directus/extensions"
|
||||||
|
|
||||||
|
# List of extensions to sync - including modules and endpoints
|
||||||
|
EXTENSIONS=(
|
||||||
|
"acquisition"
|
||||||
|
"acquisition-manager"
|
||||||
|
"customer-manager"
|
||||||
|
"feedback-commander"
|
||||||
|
"people-manager"
|
||||||
|
)
|
||||||
|
|
||||||
|
echo "🚀 Starting extension sync..."
|
||||||
|
|
||||||
|
# Ensure target directory exists
|
||||||
|
mkdir -p "$TARGET_DIR"
|
||||||
|
|
||||||
|
# Build the acquisition library first so extensions use the updated build
|
||||||
|
echo "📦 Building acquisition-library..."
|
||||||
|
(cd "$REPO_ROOT/packages/acquisition-library" && pnpm build)
|
||||||
|
|
||||||
|
for EXT in "${EXTENSIONS[@]}"; do
|
||||||
|
EXT_PATH="$EXTENSIONS_ROOT/$EXT"
|
||||||
|
|
||||||
|
if [ -d "$EXT_PATH" ]; then
|
||||||
|
echo "📦 Building $EXT..."
|
||||||
|
|
||||||
|
# Build the extension
|
||||||
|
# We use --if-present to avoid errors if build script is missing
|
||||||
|
(cd "$EXT_PATH" && pnpm build)
|
||||||
|
|
||||||
|
# Create target directory for this extension
|
||||||
|
# Directus expects extensions to be in subdirectories matching their name
|
||||||
|
mkdir -p "$TARGET_DIR/$EXT"
|
||||||
|
|
||||||
|
echo "🚚 Syncing $EXT to $TARGET_DIR/$EXT..."
|
||||||
|
|
||||||
|
# Clean target first to avoid ghost files
|
||||||
|
rm -rf "${TARGET_DIR:?}/$EXT"/*
|
||||||
|
|
||||||
|
# Copy build artifacts and package metadata
|
||||||
|
# Some extensions have index.js in root after build, some use dist/
|
||||||
|
# We check for index.js and package.json
|
||||||
|
if [ -f "$EXT_PATH/index.js" ]; then
|
||||||
|
cp "$EXT_PATH/index.js" "$TARGET_DIR/$EXT/"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -f "$EXT_PATH/package.json" ]; then
|
||||||
|
cp "$EXT_PATH/package.json" "$TARGET_DIR/$EXT/"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -d "$EXT_PATH/dist" ]; then
|
||||||
|
cp -r "$EXT_PATH/dist" "$TARGET_DIR/$EXT/"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Sync node_modules if they exist (sometimes needed if not everything is bundled)
|
||||||
|
# Deactivated: Causes global scope pollution and login issues in Directus
|
||||||
|
# if [ -d "$EXT_PATH/node_modules" ]; then
|
||||||
|
# echo "📚 Syncing node_modules for $EXT..."
|
||||||
|
# rsync -aL --delete "$EXT_PATH/node_modules/" "$TARGET_DIR/$EXT/node_modules/"
|
||||||
|
# fi
|
||||||
|
|
||||||
|
echo "✅ $EXT synced."
|
||||||
|
else
|
||||||
|
echo "❌ Extension source not found: $EXT_PATH"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "✨ Extension sync complete!"
|
||||||
Reference in New Issue
Block a user