// Copyright 2025 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { fabric } from 'fabric'; import type { DonationReceipt } from '../types/Donations'; import type { LocalizerType } from '../types/Util'; import { strictAssert } from './assert'; import { getDateTimeFormatter } from './formatTimestamp'; import { isStagingServer } from './isStagingServer'; const SCALING_FACTOR = 4.17; // Color constants matching SCSS variables const COLORS = { WHITE: '#ffffff', GRAY_20: '#c6c6c6', GRAY_45: '#848484', GRAY_60: '#5e5e5e', GRAY_90: '#1b1b1b', GRAY_95: '#121212', } as const; /** * Helper function to scale font sizes, heights, and letter spacing for the receipt * @param params - Object containing original values to scale * @param params.fontSize - Original font size in pixels * @param params.height - Optional original height/margin/padding in pixels * @param params.letterSpacing - Optional original letter spacing in pixels * @returns Scaled values for use in FabricJS */ function scaleValues(params: { fontSize: number; height?: number; letterSpacing?: number; }): { fontSize: number; height?: number; charSpacing?: number; } { const result: { fontSize: number; height?: number; charSpacing?: number; } = { fontSize: params.fontSize * SCALING_FACTOR, }; if (params.height !== undefined) { result.height = params.height * SCALING_FACTOR; } if (params.letterSpacing !== undefined) { // FabricJS charSpacing is in thousandths of em units // Formula: (letterSpacingPx * 1000) / fontSizePx // This converts pixel-based letter spacing to em-based units // For example: -0.13px letter spacing on 12px font = // (-0.13 * 1000) / 12 = -10.83 thousandths of em result.charSpacing = (params.letterSpacing * 1000) / params.fontSize; } return result; } const SIGNAL_LOGO_SVG = ` `; export async function generateDonationReceiptBlob( receipt: DonationReceipt, i18n: LocalizerType ): Promise { const width = 2550; const height = 3300; const canvas = new fabric.StaticCanvas(null, { width, height, backgroundColor: COLORS.WHITE, }); const fontFamily = 'Inter'; const paddingTop = 70 * SCALING_FACTOR; const paddingX = 66 * SCALING_FACTOR; const contentWidth = width - paddingX * 2; let currentY = paddingTop; // Create an image from the SVG const logo = await new Promise((resolve, reject) => { const logoDataUrl = `data:image/svg+xml;base64,${btoa(SIGNAL_LOGO_SVG)}`; fabric.Image.fromURL(logoDataUrl, fabricImg => { if (!fabricImg) { reject(new Error('Failed to load logo')); return; } // Position the logo fabricImg.set({ left: paddingX, top: currentY, }); resolve(fabricImg); }); }); canvas.add(logo); const dateFormatter = getDateTimeFormatter({ month: 'short', day: '2-digit', year: 'numeric', }); const dateStr = dateFormatter.format(new Date()); const dateText = new fabric.Text(dateStr, { left: width - paddingX, top: currentY + (logo.height ?? 0), fontFamily, fill: COLORS.GRAY_60, originX: 'right', originY: 'bottom', ...scaleValues({ fontSize: 12, letterSpacing: -0.03 }), }); canvas.add(dateText); currentY += (logo.height ?? 0) + 16 * SCALING_FACTOR; const divider1 = new fabric.Rect({ left: paddingX, top: currentY, width: contentWidth, height: 1 * SCALING_FACTOR, fill: COLORS.GRAY_20, }); canvas.add(divider1); strictAssert(divider1.height != null, 'Divider1 height must be defined'); currentY += divider1.height; currentY += 167; const title = new fabric.Text(i18n('icu:DonationReceipt__title'), { left: paddingX, top: currentY, fontFamily, fill: COLORS.GRAY_90, ...scaleValues({ fontSize: 20, letterSpacing: -0.34 }), }); canvas.add(title); strictAssert(title.height != null, 'Title height must be defined'); currentY += title.height + 29 * SCALING_FACTOR; // Amount section const amountLabel = new fabric.Text( i18n('icu:DonationReceipt__amount-label'), { left: paddingX, top: currentY, fontFamily, fill: COLORS.GRAY_90, ...scaleValues({ fontSize: 14, letterSpacing: -0.34 }), } ); canvas.add(amountLabel); // Format currency const preferredSystemLocales = window.SignalContext.getPreferredSystemLocales(); const localeOverride = window.SignalContext.getLocaleOverride(); const locales = localeOverride != null ? [localeOverride] : preferredSystemLocales; const formatter = new Intl.NumberFormat(locales, { style: 'currency', currency: receipt.currencyType, }); const amountStr = formatter.format(receipt.paymentAmount / 100); const amountValue = new fabric.Text(amountStr, { left: width - paddingX, top: currentY, fontFamily, fill: COLORS.GRAY_90, originX: 'right', ...scaleValues({ fontSize: 14, letterSpacing: -0.34 }), }); canvas.add(amountValue); strictAssert( amountLabel.height != null, 'Amount label height must be defined' ); strictAssert( amountValue.height != null, 'Amount value height must be defined' ); currentY += Math.max(amountLabel.height, amountValue.height) + 25 * SCALING_FACTOR; const boldDivider = new fabric.Rect({ left: paddingX, top: currentY, width: contentWidth, height: 1 * SCALING_FACTOR, fill: COLORS.GRAY_90, }); canvas.add(boldDivider); strictAssert( boldDivider.height != null, 'Bold divider height must be defined' ); currentY += boldDivider.height; // Details section (margin-top: 50px) currentY += 12 * SCALING_FACTOR; // Detail row 1 - Type (padding: 50px 0) currentY += 12 * SCALING_FACTOR; const typeLabel = new fabric.Text(i18n('icu:DonationReceipt__type-label'), { left: paddingX, top: currentY, fontFamily, fill: COLORS.GRAY_95, ...scaleValues({ fontSize: 14, letterSpacing: -0.34 }), }); canvas.add(typeLabel); strictAssert(typeLabel.height != null, 'Type label height must be defined'); currentY += 4 * SCALING_FACTOR + typeLabel.height; // margin-bottom + actual height const typeValue = new fabric.Text( i18n('icu:DonationReceipt__type-value--one-time'), { left: paddingX, top: currentY, fontFamily, fill: COLORS.GRAY_45, ...scaleValues({ fontSize: 12, letterSpacing: -0.08 }), } ); canvas.add(typeValue); strictAssert(typeValue.height != null, 'Type value height must be defined'); currentY += typeValue.height + 50; // actual height + bottom padding const rowDivider = new fabric.Rect({ left: paddingX, top: currentY, width: contentWidth, height: 1 * SCALING_FACTOR, fill: COLORS.GRAY_20, }); canvas.add(rowDivider); strictAssert(rowDivider.height != null, 'Row divider height must be defined'); currentY += rowDivider.height; // Detail row 2 - Date Paid currentY += 12 * SCALING_FACTOR; const dateLabel = new fabric.Text( i18n('icu:DonationReceipt__date-paid-label'), { left: paddingX, top: currentY, fontFamily, fill: COLORS.GRAY_95, ...scaleValues({ fontSize: 14, letterSpacing: -0.34 }), } ); canvas.add(dateLabel); strictAssert(dateLabel.height != null, 'Date label height must be defined'); currentY += 4 * SCALING_FACTOR + dateLabel.height; const paymentDateFormatter = getDateTimeFormatter({ month: 'short', day: 'numeric', year: 'numeric', }); const paymentDate = paymentDateFormatter.format(new Date(receipt.timestamp)); const dateValue = new fabric.Text(paymentDate, { left: paddingX, top: currentY, fontFamily, fill: COLORS.GRAY_45, ...scaleValues({ fontSize: 12, letterSpacing: -0.08 }), }); canvas.add(dateValue); strictAssert(dateValue.height != null, 'Date value height must be defined'); currentY += dateValue.height + 50; currentY += 10 * SCALING_FACTOR; const footerText = i18n('icu:DonationReceipt__footer-text'); const footer = new fabric.Textbox(footerText, { left: paddingX, top: currentY, width: contentWidth, fontFamily, fill: COLORS.GRAY_60, lineHeight: 1.45, ...scaleValues({ fontSize: 11 }), }); canvas.add(footer); // Add staging indicator if in staging environment if (isStagingServer()) { strictAssert(footer.height != null, 'Footer height must be defined'); currentY += footer.height + 100 * SCALING_FACTOR; const stagingText = new fabric.Text( 'NOT A REAL RECEIPT / FOR TESTING ONLY', { left: width / 2, top: currentY, fontFamily, fontSize: 24 * SCALING_FACTOR, fontWeight: 'bold', fill: '#7C3AED', originX: 'center', textAlign: 'center', } ); canvas.add(stagingText); } canvas.renderAll(); // Convert canvas to PNG blob // First, get the canvas as a data URL (base64 encoded string) const dataURL = canvas.toDataURL({ format: 'png', multiplier: 1, }); // Extract the base64 encoded data from the data URL // Data URL format: "..." const base64Data = dataURL.split(',')[1]; // Decode the base64 string to binary data const binaryString = atob(base64Data); // Convert the binary string directly to a typed array const byteArray = new Uint8Array(binaryString.length); for (let i = 0; i < binaryString.length; i += 1) { byteArray[i] = binaryString.charCodeAt(i); } const blob = new Blob([byteArray], { type: 'image/png' }); return blob; }