Files
orion/app/templates/vendor/email-templates.html
Samir Boulahtit 0b4291d893 refactor(js): migrate JavaScript files to module directories
Move 47 JS files from static/{admin,vendor,shared}/js/ to their
respective module directories app/modules/*/static/*/js/:

- Orders: orders.js, order-detail.js
- Catalog: products.js (renamed from vendor-products.js), product-*.js
- Inventory: inventory.js (admin & vendor)
- Customers: customers.js, users.js, user-*.js
- Billing: billing-history.js, subscriptions.js, subscription-tiers.js,
  billing.js, invoices.js, feature-store.js, upgrade-prompts.js
- Messaging: messages.js, notifications.js, email-templates.js
- Marketplace: marketplace*.js, letzshop*.js, onboarding.js
- Monitoring: monitoring.js, background-tasks.js, imports.js, logs.js
- Dev Tools: testing-*.js, code-quality-*.js

Update 39 templates to reference new module static paths using
url_for('{module}_static', path='...') pattern.

Files staying in static/ (platform core):
- admin: dashboard, login, platforms, vendors, companies, admin-users,
  settings, components, init-alpine, module-config
- vendor: dashboard, login, profile, settings, team, media, init-alpine
- shared: api-client, utils, money, icons, log-config, vendor-selector,
  media-picker

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 22:08:20 +01:00

331 lines
16 KiB
HTML

{# app/templates/vendor/email-templates.html #}
{% extends "vendor/base.html" %}
{% from 'shared/macros/headers.html' import page_header_flex %}
{% from 'shared/macros/alerts.html' import loading_state, error_state %}
{% from 'shared/macros/modals.html' import modal_dialog %}
{% block title %}Email Templates{% endblock %}
{% block alpine_data %}vendorEmailTemplates(){% endblock %}
{% block content %}
<!-- Page Header -->
{% call page_header_flex(title='Email Templates', subtitle='Customize email templates sent to your customers') %}
{% endcall %}
{{ loading_state('Loading email templates...') }}
{{ error_state('Error loading templates') }}
<!-- Main Content -->
<div x-show="!loading && !error" class="space-y-6">
<!-- Info Banner -->
<div class="p-4 bg-blue-50 dark:bg-blue-900/20 rounded-lg border border-blue-200 dark:border-blue-800">
<div class="flex items-start gap-3">
<span x-html="$icon('information-circle', 'w-5 h-5 text-blue-600 dark:text-blue-400 flex-shrink-0 mt-0.5')"></span>
<div>
<p class="text-sm text-blue-800 dark:text-blue-300">
Customize how emails appear to your customers. Platform templates are used by default,
and you can override them with your own versions. Some templates (billing, subscriptions)
are platform-only and cannot be customized.
</p>
</div>
</div>
</div>
<!-- Templates Table -->
{# noqa: FE-005 - Table has custom header section and styling not compatible with table_wrapper #}
<div class="bg-white rounded-lg shadow-xs dark:bg-gray-800 overflow-hidden">
<div class="p-4 border-b dark:border-gray-700">
<h3 class="text-lg font-semibold text-gray-800 dark:text-gray-200">Available Templates</h3>
<p class="text-sm text-gray-500 dark:text-gray-400">Click a template to customize it</p>
</div>
<div class="overflow-x-auto">
<table class="w-full">
<thead class="bg-gray-50 dark:bg-gray-700">
<tr>
<th class="px-4 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Template</th>
<th class="px-4 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Category</th>
<th class="px-4 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Languages</th>
<th class="px-4 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Status</th>
<th class="px-4 py-3 text-right text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Actions</th>
</tr>
</thead>
<tbody class="divide-y divide-gray-200 dark:divide-gray-700">
<template x-for="template in templates" :key="template.code">
<tr class="hover:bg-gray-50 dark:hover:bg-gray-700/50">
<td class="px-4 py-4">
<p class="text-sm font-medium text-gray-800 dark:text-gray-200" x-text="template.name"></p>
<p class="text-xs text-gray-500 dark:text-gray-400 font-mono" x-text="template.code"></p>
</td>
<td class="px-4 py-4">
<span
:class="getCategoryClass(template.category)"
class="px-2 py-1 text-xs font-medium rounded-full"
x-text="template.category"
></span>
</td>
<td class="px-4 py-4">
<div class="flex flex-wrap gap-1">
<template x-for="lang in supportedLanguages" :key="lang">
<span
:class="template.override_languages.includes(lang)
? 'bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-400'
: 'bg-gray-100 text-gray-600 dark:bg-gray-700 dark:text-gray-400'"
class="px-2 py-0.5 text-xs font-medium rounded uppercase"
x-text="lang"
></span>
</template>
</div>
</td>
<td class="px-4 py-4">
<template x-if="template.has_override">
<span class="inline-flex items-center gap-1 px-2 py-1 text-xs font-medium text-green-700 bg-green-100 rounded-full dark:bg-green-900/30 dark:text-green-400">
<span x-html="$icon('check-circle', 'w-3 h-3')"></span>
Customized
</span>
</template>
<template x-if="!template.has_override">
<span class="px-2 py-1 text-xs font-medium text-gray-600 bg-gray-100 rounded-full dark:bg-gray-700 dark:text-gray-400">
Platform Default
</span>
</template>
</td>
<td class="px-4 py-4 text-right">
<button
@click="editTemplate(template)"
class="px-3 py-1.5 text-sm font-medium text-purple-600 hover:text-purple-700 hover:bg-purple-50 rounded-lg dark:text-purple-400 dark:hover:bg-purple-900/20"
>
Customize
</button>
</td>
</tr>
</template>
</tbody>
</table>
</div>
<template x-if="templates.length === 0">
<div class="p-8 text-center">
<span x-html="$icon('mail', 'w-12 h-12 mx-auto text-gray-400 dark:text-gray-500')"></span>
<p class="mt-2 text-gray-500 dark:text-gray-400">No customizable templates available</p>
</div>
</template>
</div>
</div>
<!-- Edit Template Modal -->
{% call modal_dialog(
show_var="showEditModal",
title_var="editingTemplate ? 'Customize: ' + editingTemplate.name : 'Edit Template'",
size="4xl"
) %}
<template x-if="editingTemplate">
<div class="space-y-6">
<!-- Language Tabs -->
<div class="border-b dark:border-gray-700">
<div class="flex flex-wrap gap-1">
<template x-for="lang in supportedLanguages" :key="lang">
<button
@click="editLanguage = lang; loadTemplateLanguage()"
:class="editLanguage === lang
? 'border-purple-500 text-purple-600 dark:text-purple-400'
: 'border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300 dark:text-gray-400'"
class="px-4 py-2 text-sm font-medium border-b-2 uppercase"
x-text="lang"
></button>
</template>
</div>
</div>
<!-- Loading -->
<div x-show="loadingTemplate" class="flex items-center justify-center py-8">
<span x-html="$icon('loading', 'w-8 h-8 animate-spin text-purple-600')"></span>
</div>
<!-- Edit Form -->
<div x-show="!loadingTemplate" class="space-y-4">
<!-- Source Indicator -->
<div class="flex items-center gap-2 text-sm">
<template x-if="templateSource === 'vendor_override'">
<span class="text-green-600 dark:text-green-400">Using your customized version</span>
</template>
<template x-if="templateSource === 'platform'">
<span class="text-gray-500 dark:text-gray-400">Using platform default - edit to create your version</span>
</template>
</div>
<!-- Subject -->
<div>
<label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">
Subject Line
</label>
<input
type="text"
x-model="editForm.subject"
class="w-full px-4 py-2 text-sm text-gray-700 bg-white border border-gray-200 rounded-lg dark:text-gray-300 dark:bg-gray-800 dark:border-gray-600 focus:border-purple-400 focus:outline-none focus:ring-1 focus:ring-purple-400"
placeholder="Email subject..."
/>
</div>
<!-- Variables Info -->
<div x-show="editingTemplate.variables?.length > 0" class="p-3 bg-gray-50 dark:bg-gray-700 rounded-lg">
<p class="text-xs font-medium text-gray-700 dark:text-gray-300 mb-2">Available Variables:</p>
<div class="flex flex-wrap gap-2">
<template x-for="variable in editingTemplate.variables" :key="variable">
<code class="px-2 py-0.5 text-xs bg-white dark:bg-gray-600 rounded border dark:border-gray-500" x-text="'{{ ' + variable + ' }}'"></code>
</template>
</div>
</div>
<!-- HTML Body -->
<div>
<label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">
HTML Content
</label>
<textarea
x-model="editForm.body_html"
rows="12"
class="w-full px-4 py-2 text-sm font-mono text-gray-700 bg-white border border-gray-200 rounded-lg dark:text-gray-300 dark:bg-gray-800 dark:border-gray-600 focus:border-purple-400 focus:outline-none focus:ring-1 focus:ring-purple-400"
placeholder="<html>...</html>"
></textarea>
</div>
<!-- Plain Text Body -->
<div>
<label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">
Plain Text (Optional)
</label>
<textarea
x-model="editForm.body_text"
rows="4"
class="w-full px-4 py-2 text-sm font-mono text-gray-700 bg-white border border-gray-200 rounded-lg dark:text-gray-300 dark:bg-gray-800 dark:border-gray-600 focus:border-purple-400 focus:outline-none focus:ring-1 focus:ring-purple-400"
placeholder="Plain text fallback..."
></textarea>
</div>
</div>
<!-- Actions -->
<div class="flex items-center justify-between pt-4 border-t dark:border-gray-700">
<div>
<!-- Revert to Default Button -->
<template x-if="templateSource === 'vendor_override'">
<button
@click="revertToDefault()"
:disabled="reverting"
class="px-4 py-2 text-sm font-medium text-red-600 hover:text-red-700 hover:bg-red-50 rounded-lg dark:text-red-400 dark:hover:bg-red-900/20"
>
<span x-show="!reverting">Revert to Platform Default</span>
<span x-show="reverting">Reverting...</span>
</button>
</template>
</div>
<div class="flex items-center gap-3">
<button
@click="previewTemplate()"
class="px-4 py-2 text-sm font-medium text-gray-600 hover:text-gray-700 hover:bg-gray-100 rounded-lg dark:text-gray-400 dark:hover:bg-gray-700"
>
Preview
</button>
<button
@click="sendTestEmail()"
class="px-4 py-2 text-sm font-medium text-gray-600 hover:text-gray-700 hover:bg-gray-100 rounded-lg dark:text-gray-400 dark:hover:bg-gray-700"
>
Send Test
</button>
<button
@click="closeEditModal()"
class="px-4 py-2 text-sm font-medium text-gray-600 hover:text-gray-700 rounded-lg dark:text-gray-400"
>
Cancel
</button>
<button
@click="saveTemplate()"
:disabled="saving"
class="px-4 py-2 text-sm font-medium text-white bg-purple-600 rounded-lg hover:bg-purple-700 disabled:opacity-50"
>
<span x-show="!saving">Save Override</span>
<span x-show="saving">Saving...</span>
</button>
</div>
</div>
</div>
</template>
{% endcall %}
<!-- Preview Modal -->
{% call modal_dialog(
show_var="showPreviewModal",
title="Email Preview",
size="4xl"
) %}
<template x-if="previewData">
<div class="space-y-4">
<div class="p-3 bg-gray-100 dark:bg-gray-700 rounded-lg">
<p class="text-sm"><strong>Subject:</strong> <span x-text="previewData.subject"></span></p>
</div>
<div class="border dark:border-gray-700 rounded-lg overflow-hidden">
<iframe
:srcdoc="previewData.body_html"
class="w-full h-96 bg-white"
sandbox="allow-same-origin"
></iframe>
</div>
<div class="flex justify-end">
<button
@click="showPreviewModal = false"
class="px-4 py-2 text-sm font-medium text-gray-600 hover:text-gray-700 rounded-lg dark:text-gray-400"
>
Close
</button>
</div>
</div>
</template>
{% endcall %}
<!-- Test Email Modal -->
{% call modal_dialog(
show_var="showTestEmailModal",
title="Send Test Email",
size="md"
) %}
<div class="space-y-4">
<div>
<label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">
Send test email to:
</label>
<input
type="email"
x-model="testEmailAddress"
class="w-full px-4 py-2 text-sm text-gray-700 bg-white border border-gray-200 rounded-lg dark:text-gray-300 dark:bg-gray-800 dark:border-gray-600 focus:border-purple-400 focus:outline-none focus:ring-1 focus:ring-purple-400"
placeholder="your@email.com"
/>
</div>
<p class="text-sm text-gray-500 dark:text-gray-400">
A test email will be sent using sample data for template variables.
</p>
<div class="flex justify-end gap-3">
<button
@click="showTestEmailModal = false"
class="px-4 py-2 text-sm font-medium text-gray-600 hover:text-gray-700 rounded-lg dark:text-gray-400"
>
Cancel
</button>
<button
@click="confirmSendTestEmail()"
:disabled="sendingTest || !testEmailAddress"
class="px-4 py-2 text-sm font-medium text-white bg-purple-600 rounded-lg hover:bg-purple-700 disabled:opacity-50"
>
<span x-show="!sendingTest">Send Test</span>
<span x-show="sendingTest">Sending...</span>
</button>
</div>
</div>
{% endcall %}
{% endblock %}
{% block extra_scripts %}
<script src="{{ url_for('messaging_static', path='vendor/js/email-templates.js') }}"></script>
{% endblock %}