feat(email): WYSIWYG template editor with variable substitution

- Add emailSubject{De,En} + emailBodyHtml{De,En} to ShopSettings
- New RichTextEditor component (TipTap) with toolbar + variable insert
- Settings UI: Email templates section per language
- email.server.ts: substitute {{var}} placeholders, fall back to defaults
- Default vars: invoiceNumber, customerName, customerFirstName, orderName,
  totalGross, dueDate, companyName, ownerName
This commit is contained in:
Gerhard Scheikl
2026-05-08 23:06:40 +02:00
parent 537dfd34cb
commit 04933fcac6
8 changed files with 1120 additions and 11 deletions
@@ -0,0 +1,55 @@
-- RedefineTables
PRAGMA defer_foreign_keys=ON;
PRAGMA foreign_keys=OFF;
CREATE TABLE "new_ShopSettings" (
"id" TEXT NOT NULL PRIMARY KEY,
"shopDomain" TEXT NOT NULL,
"companyName" TEXT NOT NULL DEFAULT '',
"legalForm" TEXT NOT NULL DEFAULT '',
"ownerName" TEXT NOT NULL DEFAULT '',
"addressLine1" TEXT NOT NULL DEFAULT '',
"addressLine2" TEXT NOT NULL DEFAULT '',
"postalCode" TEXT NOT NULL DEFAULT '',
"city" TEXT NOT NULL DEFAULT '',
"countryCode" TEXT NOT NULL DEFAULT 'AT',
"phone" TEXT NOT NULL DEFAULT '',
"email" TEXT NOT NULL DEFAULT '',
"website" TEXT NOT NULL DEFAULT '',
"vatId" TEXT NOT NULL DEFAULT '',
"taxNumber" TEXT NOT NULL DEFAULT '',
"registrationNo" TEXT NOT NULL DEFAULT '',
"registrationCourt" TEXT NOT NULL DEFAULT '',
"bankName" TEXT NOT NULL DEFAULT '',
"iban" TEXT NOT NULL DEFAULT '',
"bic" TEXT NOT NULL DEFAULT '',
"giroCodeEnabled" BOOLEAN NOT NULL DEFAULT true,
"numberingMode" TEXT NOT NULL DEFAULT 'shopify_order_number',
"invoicePrefix" TEXT NOT NULL DEFAULT 'RE-',
"invoiceSeed" INTEGER NOT NULL DEFAULT 1000,
"defaultLanguage" TEXT NOT NULL DEFAULT 'de',
"paymentTermDays" INTEGER NOT NULL DEFAULT 14,
"footerNote" TEXT NOT NULL DEFAULT '',
"footerNoteEn" TEXT NOT NULL DEFAULT '',
"kleinunternehmer" BOOLEAN NOT NULL DEFAULT false,
"logoUrl" TEXT NOT NULL DEFAULT '',
"smtpHost" TEXT NOT NULL DEFAULT '',
"smtpPort" INTEGER NOT NULL DEFAULT 587,
"smtpSecure" BOOLEAN NOT NULL DEFAULT false,
"smtpUser" TEXT NOT NULL DEFAULT '',
"smtpPassword" TEXT NOT NULL DEFAULT '',
"smtpFromName" TEXT NOT NULL DEFAULT '',
"smtpFromEmail" TEXT NOT NULL DEFAULT '',
"smtpReplyTo" TEXT NOT NULL DEFAULT '',
"emailSubjectDe" TEXT NOT NULL DEFAULT '',
"emailBodyHtmlDe" TEXT NOT NULL DEFAULT '',
"emailSubjectEn" TEXT NOT NULL DEFAULT '',
"emailBodyHtmlEn" TEXT NOT NULL DEFAULT '',
"createdAt" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updatedAt" DATETIME NOT NULL
);
INSERT INTO "new_ShopSettings" ("addressLine1", "addressLine2", "bankName", "bic", "city", "companyName", "countryCode", "createdAt", "defaultLanguage", "email", "footerNote", "footerNoteEn", "giroCodeEnabled", "iban", "id", "invoicePrefix", "invoiceSeed", "kleinunternehmer", "legalForm", "logoUrl", "numberingMode", "ownerName", "paymentTermDays", "phone", "postalCode", "registrationCourt", "registrationNo", "shopDomain", "smtpFromEmail", "smtpFromName", "smtpHost", "smtpPassword", "smtpPort", "smtpReplyTo", "smtpSecure", "smtpUser", "taxNumber", "updatedAt", "vatId", "website") SELECT "addressLine1", "addressLine2", "bankName", "bic", "city", "companyName", "countryCode", "createdAt", "defaultLanguage", "email", "footerNote", "footerNoteEn", "giroCodeEnabled", "iban", "id", "invoicePrefix", "invoiceSeed", "kleinunternehmer", "legalForm", "logoUrl", "numberingMode", "ownerName", "paymentTermDays", "phone", "postalCode", "registrationCourt", "registrationNo", "shopDomain", "smtpFromEmail", "smtpFromName", "smtpHost", "smtpPassword", "smtpPort", "smtpReplyTo", "smtpSecure", "smtpUser", "taxNumber", "updatedAt", "vatId", "website" FROM "ShopSettings";
DROP TABLE "ShopSettings";
ALTER TABLE "new_ShopSettings" RENAME TO "ShopSettings";
CREATE UNIQUE INDEX "ShopSettings_shopDomain_key" ON "ShopSettings"("shopDomain");
PRAGMA foreign_keys=ON;
PRAGMA defer_foreign_keys=OFF;
+6
View File
@@ -93,6 +93,12 @@ model ShopSettings {
smtpFromEmail String @default("")
smtpReplyTo String @default("")
// Email templates (HTML, with {{var}} placeholders). Empty = use defaults.
emailSubjectDe String @default("")
emailBodyHtmlDe String @default("")
emailSubjectEn String @default("")
emailBodyHtmlEn String @default("")
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt