PSA/server/migrations/utils/templates/email/time/timeEntryApproved.cjs
Hermes 284313f908
Some checks are pending
Bidi Control Character Guard / bidi-control-guard (push) Waiting to run
Circular Dependency Check / Check for new circular dependencies (push) Waiting to run
Citus Migration Smoke / Combined migrations on single-node Citus (push) Waiting to run
E2E Fresh Install Tests / fresh-install-e2e (push) Waiting to run
ext-v2 guardrails / Run ext-v2 guard and ESLint (push) Waiting to run
Integration Tests / Check for relevant changes (push) Waiting to run
Integration Tests / ${{ (github.event_name == 'schedule' || github.event.inputs.suite == 'full') && 'Full integration suite' || 'Tier-1 integration subset' }} (push) Blocked by required conditions
Mobile checks / Mobile lint + typecheck (push) Waiting to run
Mobile checks / Mobile unit tests (push) Waiting to run
Mobile checks / Mobile dependency audit (report) (push) Waiting to run
Mobile checks / Mobile reproducibility checks (push) Waiting to run
Secrets guard (env backups) / Ensure no tracked env backup files (push) Waiting to run
Temporal Readiness / fast-readiness (push) Waiting to run
Temporal Readiness / docker-parity (push) Waiting to run
TypeScript Type Check / Nx affected typecheck (push) Waiting to run
Unit Tests / Skipped-test budget (push) Waiting to run
Unit Tests / Nx affected unit tests (push) Waiting to run
Unit Tests / Server unit coverage (informational) (push) Waiting to run
Validate Tenant Management Schema / Check for relevant changes (push) Waiting to run
Validate Tenant Management Schema / Validate Tenant Management Schema (push) Blocked by required conditions
EE Workflows Build Guard / ee-workflows-build-guard (push) Waiting to run
Initial import of AlgaPSA codebase from PSA server
Excluded: .git, node_modules, secrets/, compose.env, assemblyscript tgz

Source: /opt/alga-psa on psa.joliet.tech
2026-06-22 16:12:17 -05:00

189 lines
6.8 KiB
JavaScript

/**
* Source-of-truth: time-entry-approved email template.
*
* Uses the shared email layout wrapper. Body content is built from
* per-language translated strings so that only text differs between locales.
*/
const { wrapEmailLayout } = require('../../_shared/emailLayout.cjs');
const { BRAND_PRIMARY } = require('../../_shared/constants.cjs');
const TEMPLATE_NAME = 'time-entry-approved';
const SUBTYPE_NAME = 'Time Entry Approved';
const SUBJECTS = {
en: 'Time Entry Approved',
fr: 'Saisie de temps approuv\u00e9e',
es: 'Registro de tiempo aprobado',
de: 'Zeiteintrag genehmigt',
nl: 'Tijdregistratie goedgekeurd',
it: 'Voce di tempo approvata',
pl: 'Wpis czasu zatwierdzony',
};
/* eslint-disable max-len */
const COPY = {
en: {
headerLabel: 'Time Entry Approved',
intro: 'Your time entry has been approved.',
date: 'Date',
duration: 'Duration',
project: 'Project',
task: 'Task',
approvedBy: 'Approved By',
viewButton: 'View Time Entry',
footer: 'Powered by Alga PSA · Keeping teams aligned',
textHeader: 'Time Entry Approved',
textIntro: 'Your time entry has been approved:',
textView: 'View time entry at',
},
fr: {
headerLabel: 'Saisie de temps approuv\u00e9e',
intro: 'Votre saisie de temps a \u00e9t\u00e9 approuv\u00e9e.',
date: 'Date',
duration: 'Dur\u00e9e',
project: 'Projet',
task: 'T\u00e2che',
approvedBy: 'Approuv\u00e9e par',
viewButton: 'Voir la saisie',
footer: 'Powered by Alga PSA · Gardons les \u00e9quipes align\u00e9es',
textHeader: 'Saisie de temps approuv\u00e9e',
textIntro: 'Votre saisie de temps a \u00e9t\u00e9 approuv\u00e9e :',
textView: 'Voir la saisie sur',
},
es: {
headerLabel: 'Registro de tiempo aprobado',
intro: 'Su registro de tiempo ha sido aprobado.',
date: 'Fecha',
duration: 'Duraci\u00f3n',
project: 'Proyecto',
task: 'Tarea',
approvedBy: 'Aprobado por',
viewButton: 'Ver registro',
footer: 'Powered by Alga PSA · Manteniendo a los equipos alineados',
textHeader: 'Registro de tiempo aprobado',
textIntro: 'Su registro de tiempo ha sido aprobado:',
textView: 'Ver registro en',
},
de: {
headerLabel: 'Zeiteintrag genehmigt',
intro: 'Ihr Zeiteintrag wurde genehmigt.',
date: 'Datum',
duration: 'Dauer',
project: 'Projekt',
task: 'Aufgabe',
approvedBy: 'Genehmigt von',
viewButton: 'Zeiteintrag anzeigen',
footer: 'Powered by Alga PSA · Teams auf Kurs halten',
textHeader: 'Zeiteintrag genehmigt',
textIntro: 'Ihr Zeiteintrag wurde genehmigt:',
textView: 'Zeiteintrag anzeigen unter',
},
nl: {
headerLabel: 'Tijdregistratie goedgekeurd',
intro: 'Uw tijdregistratie is goedgekeurd.',
date: 'Datum',
duration: 'Duur',
project: 'Project',
task: 'Taak',
approvedBy: 'Goedgekeurd door',
viewButton: 'Tijdregistratie bekijken',
footer: 'Powered by Alga PSA · Teams op \u00e9\u00e9n lijn houden',
textHeader: 'Tijdregistratie goedgekeurd',
textIntro: 'Uw tijdregistratie is goedgekeurd:',
textView: 'Tijdregistratie bekijken op',
},
it: {
headerLabel: 'Voce di tempo approvata',
intro: 'La Sua voce di tempo \u00e8 stata approvata.',
date: 'Data',
duration: 'Durata',
project: 'Progetto',
task: 'Attivit\u00e0',
approvedBy: 'Approvata da',
viewButton: 'Visualizza voce di tempo',
footer: 'Powered by Alga PSA · Manteniamo i team allineati',
textHeader: 'Voce di tempo approvata',
textIntro: 'La Sua voce di tempo \u00e8 stata approvata:',
textView: 'Visualizza la voce di tempo su',
},
pl: {
headerLabel: 'Wpis czasu zatwierdzony',
intro: 'Twój wpis czasu zosta\u0142 zatwierdzony.',
date: 'Data',
duration: 'Czas trwania',
project: 'Projekt',
task: 'Zadanie',
approvedBy: 'Zatwierdzi\u0142(a)',
viewButton: 'Zobacz wpis czasu',
footer: 'Powered by Alga PSA',
textHeader: 'Wpis czasu zatwierdzony',
textIntro: 'Twój wpis czasu zosta\u0142 zatwierdzony:',
textView: 'Zobacz wpis czasu na',
},
};
/* eslint-enable max-len */
function buildBodyHtml(c) {
return `<p style="margin:0 0 16px 0;font-size:15px;color:#1f2933;line-height:1.5;">${c.intro}</p>
<table role="presentation" width="100%" cellpadding="0" cellspacing="0" style="border-collapse:collapse;font-size:14px;color:#1f2933;">
<tr>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;width:160px;font-weight:600;color:#475467;">${c.date}</td>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;">{{timeEntry.date}}</td>
</tr>
<tr>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;font-weight:600;color:#475467;">${c.duration}</td>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;">{{timeEntry.duration}}</td>
</tr>
<tr>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;font-weight:600;color:#475467;">${c.project}</td>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;">{{timeEntry.project}}</td>
</tr>
<tr>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;font-weight:600;color:#475467;">${c.task}</td>
<td style="padding:12px 0;border-bottom:1px solid #eef2ff;">{{timeEntry.task}}</td>
</tr>
<tr>
<td style="padding:12px 0;font-weight:600;color:#475467;">${c.approvedBy}</td>
<td style="padding:12px 0;">{{timeEntry.approvedBy}}</td>
</tr>
</table>
<div style="margin-top:24px;">
<a href="{{timeEntry.url}}" style="display:inline-block;background:${BRAND_PRIMARY};color:#ffffff;text-decoration:none;padding:12px 24px;border-radius:10px;font-weight:600;">${c.viewButton}</a>
</div>`;
}
function buildText(c) {
return `${c.textHeader}
${c.textIntro}
${c.date}: {{timeEntry.date}}
${c.duration}: {{timeEntry.duration}}
${c.project}: {{timeEntry.project}}
${c.task}: {{timeEntry.task}}
${c.approvedBy}: {{timeEntry.approvedBy}}
${c.textView}: {{timeEntry.url}}`;
}
function getTemplate() {
return {
templateName: TEMPLATE_NAME,
subtypeName: SUBTYPE_NAME,
translations: Object.entries(COPY).map(([lang, copy]) => ({
language: lang,
subject: SUBJECTS[lang],
htmlContent: wrapEmailLayout({
language: lang,
headerLabel: copy.headerLabel,
bodyHtml: buildBodyHtml(copy),
footerText: copy.footer,
}),
textContent: buildText(copy),
})),
};
}
module.exports = { TEMPLATE_NAME, SUBTYPE_NAME, getTemplate };