mirror of
https://github.com/simstudioai/sim.git
synced 2026-01-09 06:58:07 -05:00
* feat(email): welcome email; improvement(emails): ui/ux * improvement(emails): links, accounts, preview * refactor(emails): file structure and wrapper components * added envvar for personal emails sent, added isHosted gate * fixed failing tests, added env mock * fix: removed comment --------- Co-authored-by: waleed <walif6@gmail.com>
106 lines
3.2 KiB
TypeScript
106 lines
3.2 KiB
TypeScript
import { Link, Text } from '@react-email/components'
|
|
import { baseStyles } from '@/components/emails/_styles'
|
|
import { EmailLayout } from '@/components/emails/components'
|
|
import { getBrandConfig } from '@/lib/branding/branding'
|
|
|
|
interface WorkspaceInvitation {
|
|
workspaceId: string
|
|
workspaceName: string
|
|
permission: 'admin' | 'write' | 'read'
|
|
}
|
|
|
|
interface BatchInvitationEmailProps {
|
|
inviterName: string
|
|
organizationName: string
|
|
organizationRole: 'admin' | 'member'
|
|
workspaceInvitations: WorkspaceInvitation[]
|
|
acceptUrl: string
|
|
}
|
|
|
|
const getPermissionLabel = (permission: string) => {
|
|
switch (permission) {
|
|
case 'admin':
|
|
return 'Admin (full access)'
|
|
case 'write':
|
|
return 'Editor (can edit workflows)'
|
|
case 'read':
|
|
return 'Viewer (read-only access)'
|
|
default:
|
|
return permission
|
|
}
|
|
}
|
|
|
|
const getRoleLabel = (role: string) => {
|
|
switch (role) {
|
|
case 'admin':
|
|
return 'Admin'
|
|
case 'member':
|
|
return 'Member'
|
|
default:
|
|
return role
|
|
}
|
|
}
|
|
|
|
export function BatchInvitationEmail({
|
|
inviterName = 'Someone',
|
|
organizationName = 'the team',
|
|
organizationRole = 'member',
|
|
workspaceInvitations = [],
|
|
acceptUrl,
|
|
}: BatchInvitationEmailProps) {
|
|
const brand = getBrandConfig()
|
|
const hasWorkspaces = workspaceInvitations.length > 0
|
|
|
|
return (
|
|
<EmailLayout
|
|
preview={`You've been invited to join ${organizationName}${hasWorkspaces ? ` and ${workspaceInvitations.length} workspace(s)` : ''}`}
|
|
>
|
|
<Text style={baseStyles.paragraph}>Hello,</Text>
|
|
<Text style={baseStyles.paragraph}>
|
|
<strong>{inviterName}</strong> has invited you to join <strong>{organizationName}</strong>{' '}
|
|
on {brand.name}.
|
|
</Text>
|
|
|
|
{/* Team Role Information */}
|
|
<Text style={baseStyles.paragraph}>
|
|
<strong>Team Role:</strong> {getRoleLabel(organizationRole)}
|
|
</Text>
|
|
<Text style={baseStyles.paragraph}>
|
|
{organizationRole === 'admin'
|
|
? "As a Team Admin, you'll be able to manage team members, billing, and workspace access."
|
|
: "As a Team Member, you'll have access to shared team billing and can be invited to workspaces."}
|
|
</Text>
|
|
|
|
{/* Workspace Invitations */}
|
|
{hasWorkspaces && (
|
|
<>
|
|
<Text style={baseStyles.paragraph}>
|
|
<strong>
|
|
Workspace Access ({workspaceInvitations.length} workspace
|
|
{workspaceInvitations.length !== 1 ? 's' : ''}):
|
|
</strong>
|
|
</Text>
|
|
{workspaceInvitations.map((ws) => (
|
|
<Text key={ws.workspaceId} style={{ ...baseStyles.paragraph, marginLeft: '20px' }}>
|
|
• <strong>{ws.workspaceName}</strong> - {getPermissionLabel(ws.permission)}
|
|
</Text>
|
|
))}
|
|
</>
|
|
)}
|
|
|
|
<Link href={acceptUrl} style={{ textDecoration: 'none' }}>
|
|
<Text style={baseStyles.button}>Accept Invitation</Text>
|
|
</Link>
|
|
|
|
{/* Divider */}
|
|
<div style={baseStyles.divider} />
|
|
|
|
<Text style={{ ...baseStyles.footerText, textAlign: 'left' }}>
|
|
Invitation expires in 7 days. If unexpected, you can ignore this email.
|
|
</Text>
|
|
</EmailLayout>
|
|
)
|
|
}
|
|
|
|
export default BatchInvitationEmail
|