You can add new mail templates to Shopware by using the administration. However, you might want to ship a mail template with your plugin, so using the administration is no option.
This guide will cover how to add a custom mail template with your plugin.
The namespaces used in the examples of this guide are the same as the namespace from our plugin base guide, so you might want to have a look at it first.
Furthermore, this guide will use database migrations in order to add a custom mail template, which is not explained in depth here. Make sure to understand those first!
As already mentioned, adding a mail template is done by using a plugin database migration. To be precise, those are the steps necessary:
- Create a new mail template type or fetch an existing mail template type ID
- Add an entry to
mail_template
using the said template type ID - Add an entry to
mail_template_translation
for each language you want to support
The following example will create a new template of type "contact form", which is already available. There will be an example to create a custom mail template type though.
Let's have a look at an example, which will:
- Use the "contact form" type
- Add a mail template entry
- Add a mail template translation for en_GB and de_DE
{% code title="/src/Migration/Migration1616418675AddMailTemplate.php" %}
<?php declare(strict_types=1);
namespace Swag\BasicExample\Migration;
use DateTime;
use Doctrine\DBAL\Connection;
use Shopware\Core\Framework\Migration\MigrationStep;
use Shopware\Core\Defaults;
use Shopware\Core\Framework\Uuid\Uuid;
class Migration1616418675AddMailTemplate extends MigrationStep
{
public function getCreationTimestamp(): int
{
return 1616418675;
}
public function update(Connection $connection): void
{
$mailTemplateTypeId = $this->getMailTemplateTypeId($connection);
$this->createMailTemplate($connection, $mailTemplateTypeId);
}
public function updateDestructive(Connection $connection): void
{
}
private function getMailTemplateTypeId(Connection $connection): string
{
$sql = <<<SQL
SELECT id
FROM mail_template_type
WHERE technical_name = "contact_form"
SQL;
return Uuid::fromBytesToHex($connection->fetchOne($sql));
}
private function getLanguageIdByLocale(Connection $connection, string $locale): ?string
{
$sql = <<<SQL
SELECT `language`.`id`
FROM `language`
INNER JOIN `locale` ON `locale`.`id` = `language`.`locale_id`
WHERE `locale`.`code` = :code
SQL;
$languageId = $connection->executeQuery($sql, ['code' => $locale])->fetchOne();
if (empty($languageId)) {
return null;
}
return $languageId;
}
private function createMailTemplate(Connection $connection, string $mailTemplateTypeId): void
{
$mailTemplateId = Uuid::randomHex();
$enGbLangId = $this->getLanguageIdByLocale($connection, 'en-GB');
$deDeLangId = $this->getLanguageIdByLocale($connection, 'de-DE');
$connection->executeStatement("
INSERT IGNORE INTO `mail_template`
(id, mail_template_type_id, system_default, created_at)
VALUES
(:id, :mailTemplateTypeId, :systemDefault, :createdAt)
",[
'id' => Uuid::fromHexToBytes($mailTemplateId),
'mailTemplateTypeId' => Uuid::fromHexToBytes($mailTemplateTypeId),
'systemDefault' => 0,
'createdAt' => (new DateTime())->format(Defaults::STORAGE_DATE_TIME_FORMAT),
]);
if (!empty($enGbLangId)) {
$connection->executeStatement("
INSERT IGNORE INTO `mail_template_translation`
(mail_template_id, language_id, sender_name, subject, description, content_html, content_plain, created_at)
VALUES
(:mailTemplateId, :languageId, :senderName, :subject, :description, :contentHtml, :contentPlain, :createdAt)
",[
'mailTemplateId' => Uuid::fromHexToBytes($mailTemplateId),
'languageId' => $enGbLangId,
'senderName' => '{{ salesChannel.name }}',
'subject' => 'Example mail template subject',
'description' => 'Example mail template description',
'contentHtml' => $this->getContentHtmlEn(),
'contentPlain' => $this->getContentPlainEn(),
'createdAt' => (new DateTime())->format(Defaults::STORAGE_DATE_TIME_FORMAT),
]);
}
if (!empty($deDeLangId)) {
$connection->executeStatement("
INSERT IGNORE INTO `mail_template_translation`
(mail_template_id, language_id, sender_name, subject, description, content_html, content_plain, created_at)
VALUES
(:mailTemplateId, :languageId, :senderName, :subject, :description, :contentHtml, :contentPlain, :createdAt)
",[
'mailTemplateId' => Uuid::fromHexToBytes($mailTemplateId),
'languageId' => $deDeLangId,
'senderName' => '{{ salesChannel.name }}',
'subject' => 'Beispiel E-Mail Template Titel',
'description' => 'Beispiel E-Mail Template Beschreibung',
'contentHtml' => $this->getContentHtmlDe(),
'contentPlain' => $this->getContentPlainDe(),
'createdAt' => (new DateTime())->format(Defaults::STORAGE_DATE_TIME_FORMAT),
]);
}
}
private function getContentHtmlEn(): string
{
return <<<MAIL
<div style="font-family:arial; font-size:12px;">
<p>
Example HTML content!
</p>
</div>
MAIL;
}
private function getContentPlainEn(): string
{
return <<<MAIL
Example plain content!
MAIL;
}
private function getContentHtmlDe(): string
{
return <<<MAIL
<div style="font-family:arial; font-size:12px;">
<p>
Beispiel HTML Inhalt!
</p>
</div>
MAIL;
}
private function getContentPlainDe(): string
{
return <<<MAIL
Beispiel Plain Inhalt!
MAIL;
}
}
{% endcode %}
First of all, let's have a look at the small update
method. It's mainly just fetching the mail template type ID using a short SQL statement and afterwards it executes the method createMailTemplate
, which will cover all the other steps.
Now on to the createMailTemplate
method, which looks big, but isn't that scary. First of all, we're fetching the language IDs for both en-GB
and de-DE
.
We then create the entry for the mail_template
table. Make sure to set system_default
to 0 here!
Afterwards we're inserting the entries into the mail_template_translation
table. For compatibility reasons we have to check whether the languages exist in the database so we can insert our translations for these languages. The same principle applies to other ISO languages.
The variables for the English and the German subject and description, may be changed to fit your needs.
Each of those calls uses a little helper method getContentHtml
or getContentPlain
respectively, where you can use your template.
And that's it, once your plugin is installed, the mail template will be added to Shopware.
{% hint style="warning" %}
Do not remove e-mail templates in your plugin, e.g. when it is uninstalled. This may lead to data inconsistency, since those templates can be associated to other entities. Beware to use IGNORE
before INTO
Statements so no exception will be thrown upon uninstallation and reinstallation of your plugin.
{% endhint %}
In order to not only use an existing mail template type, but to create a custom one, you have to adjust the update
method and create a new method.
Let's have a look:
{% code title="/src/Migration/Migration1616418675AddMailTemplate.php" %}
<?php declare(strict_types=1);
namespace Swag\BasicExample\Migration;
use DateTime;
use Doctrine\DBAL\Connection;
use Shopware\Core\Framework\Migration\MigrationStep;
use Shopware\Core\Defaults;
use Shopware\Core\Framework\Uuid\Uuid;
class Migration1616418675AddMailTemplate extends MigrationStep
{
public function getCreationTimestamp(): int
{
return 1616418675;
}
public function update(Connection $connection): void
{
$mailTemplateTypeId = $this->createMailTemplateType($connection);
$this->createMailTemplate($connection, $mailTemplateTypeId);
}
private function createMailTemplateType(Connection $connection): string
{
$mailTemplateTypeId = Uuid::randomHex();
$enGbLangId = $this->getLanguageIdByLocale($connection, 'en-GB');
$deDeLangId = $this->getLanguageIdByLocale($connection, 'de-DE');
$englishName = 'Example mail template type name';
$germanName = 'Beispiel E-Mail Template Name';
$connection->executeStatement("
INSERT IGNORE INTO `mail_template_type`
(id, technical_name, available_entities, created_at)
VALUES
(:id, :technicalName, :availableEntities, :createdAt)
",[
'id' => Uuid::fromHexToBytes($mailTemplateTypeId),
'technicalName' => 'custom_mail_template_type',
'availableEntities' => json_encode(['product' => 'product']),
'createdAt' => (new DateTime())->format(Defaults::STORAGE_DATE_TIME_FORMAT),
]);
if (!empty($enGbLangId)) {
$connection->executeStatement("
INSERT IGNORE INTO `mail_template_type_translation`
(mail_template_type_id, language_id, name, created_at)
VALUES
(:mailTemplateTypeId, :languageId, :name, :createdAt)
",[
'mailTemplateTypeId' => Uuid::fromHexToBytes($mailTemplateTypeId),
'languageId' => $enGbLangId,
'name' => $englishName,
'createdAt' => (new DateTime())->format(Defaults::STORAGE_DATE_TIME_FORMAT),
]);
}
if (!empty($deDeLangId)) {
$connection->executeStatement("
INSERT IGNORE INTO `mail_template_type_translation`
(mail_template_type_id, language_id, name, created_at)
VALUES
(:mailTemplateTypeId, :languageId, :name, :createdAt)
",[
'mailTemplateTypeId' => Uuid::fromHexToBytes($mailTemplateTypeId),
'languageId' => $deDeLangId,
'name' => $germanName,
'createdAt' => (new DateTime())->format(Defaults::STORAGE_DATE_TIME_FORMAT),
]);
}
return $mailTemplateTypeId;
}
// ...
}
{% endcode %}
First of all we changed the getMailTemplateTypeId
method call to createMailTemplateType
, a new method which we will create afterwards. Again, this method then has to return the ID of the newly created mail template ID.
So having a look at the createMailTemplateType
method, you will see some similarities:
- First of all we're fetching the language IDs for
en-GB
andde-DE
- Then we define the translated names for the mail template type
- And then the respective
mail_template_type
entry, as well as the translatedmail_template_type_translation
entries are created
Note the available_entities
column when creating the mail template type itself though. In here, you define which entities should be available for the respective mail template, in this example we'll just provide the ProductEntity
.
Now that you know how to add custom mail templates, you might wonder how you can actually add new mail template data to existing mail templates.
For that case, we've created a separate guide about adding data to mail templates.