import dayjs, { Dayjs } from "dayjs"; import localizedFormat from "dayjs/plugin/localizedFormat"; import timezone from "dayjs/plugin/timezone"; import toArray from "dayjs/plugin/toArray"; import utc from "dayjs/plugin/utc"; import { createEvent, DateArray, Person } from "ics"; import rrule from "rrule"; import { getAppName } from "@calcom/app-store/utils"; import { getCancelLink, getRichDescription } from "@calcom/lib/CalEventParser"; import type { CalendarEvent, RecurringEvent } from "@calcom/types/Calendar"; import BaseEmail from "@lib/emails/templates/_base-email"; import { emailBodyLogo, emailHead, emailScheduledBodyHeaderContent, emailSchedulingBodyDivider, emailSchedulingBodyHeader, linkIcon, } from "./common"; dayjs.extend(utc); dayjs.extend(timezone); dayjs.extend(localizedFormat); dayjs.extend(toArray); export default class OrganizerScheduledEmail extends BaseEmail { calEvent: CalendarEvent; recurringEvent: RecurringEvent; constructor(calEvent: CalendarEvent, recurringEvent: RecurringEvent) { super(); this.name = "SEND_BOOKING_CONFIRMATION"; this.calEvent = calEvent; this.recurringEvent = recurringEvent; } protected getiCalEventAsString(): string | undefined { // Taking care of recurrence rule beforehand let recurrenceRule: string | undefined = undefined; if (this.recurringEvent?.count) { recurrenceRule = new rrule(this.recurringEvent).toString().replace("RRULE:", ""); } const icsEvent = createEvent({ start: dayjs(this.calEvent.startTime) .utc() .toArray() .slice(0, 6) .map((v, i) => (i === 1 ? v + 1 : v)) as DateArray, startInputType: "utc", productId: "calendso/ics", title: this.calEvent.organizer.language.translate("ics_event_title", { eventType: this.calEvent.type, name: this.calEvent.attendees[0].name, }), description: this.getTextBody(), duration: { minutes: dayjs(this.calEvent.endTime).diff(dayjs(this.calEvent.startTime), "minute") }, organizer: { name: this.calEvent.organizer.name, email: this.calEvent.organizer.email }, ...{ recurrenceRule }, attendees: this.calEvent.attendees.map((attendee: Person) => ({ name: attendee.name, email: attendee.email, })), status: "CONFIRMED", }); if (icsEvent.error) { throw icsEvent.error; } return icsEvent.value; } protected getNodeMailerPayload(): Record { const toAddresses = [this.calEvent.organizer.email]; if (this.calEvent.team) { this.calEvent.team.members.forEach((member) => { const memberAttendee = this.calEvent.attendees.find((attendee) => attendee.name === member); if (memberAttendee) { toAddresses.push(memberAttendee.email); } }); } return { icalEvent: { filename: "event.ics", content: this.getiCalEventAsString(), }, from: `Cal.com <${this.getMailerOptions().from}>`, to: toAddresses.join(","), subject: `${this.calEvent.organizer.language.translate("confirmed_event_type_subject", { eventType: this.calEvent.type, name: this.calEvent.attendees[0].name, date: `${this.getOrganizerStart().format("h:mma")} - ${this.getOrganizerEnd().format( "h:mma" )}, ${this.calEvent.organizer.language.translate( this.getOrganizerStart().format("dddd").toLowerCase() )}, ${this.calEvent.organizer.language.translate( this.getOrganizerStart().format("MMMM").toLowerCase() )} ${this.getOrganizerStart().format("D")}, ${this.getOrganizerStart().format("YYYY")}`, })}`, html: this.getHtmlBody(), text: this.getTextBody(), }; } protected getTextBody(): string { return ` ${this.calEvent.organizer.language.translate( this.recurringEvent?.count ? "new_event_scheduled_recurring" : "new_event_scheduled" )} ${this.calEvent.organizer.language.translate("emailed_you_and_any_other_attendees")} ${getRichDescription(this.calEvent)} `.trim(); } protected getHtmlBody(): string { const headerContent = this.calEvent.organizer.language.translate("confirmed_event_type_subject", { eventType: this.calEvent.type, name: this.calEvent.attendees[0].name, date: `${this.getOrganizerStart().format("h:mma")} - ${this.getOrganizerEnd().format( "h:mma" )}, ${this.calEvent.organizer.language.translate( this.getOrganizerStart().format("dddd").toLowerCase() )}, ${this.calEvent.organizer.language.translate( this.getOrganizerStart().format("MMMM").toLowerCase() )} ${this.getOrganizerStart().format("D")}, ${this.getOrganizerStart().format("YYYY")}`, }); return ` ${emailHead(headerContent)}
${emailSchedulingBodyHeader("checkCircle")} ${emailScheduledBodyHeaderContent( this.calEvent.organizer.language.translate( this.recurringEvent?.count ? "new_event_scheduled_recurring" : "new_event_scheduled" ), this.calEvent.organizer.language.translate("emailed_you_and_any_other_attendees") )} ${emailSchedulingBodyDivider()}
${this.getWhat()} ${this.getWhen()} ${this.getWho()} ${this.getLocation()} ${this.getDescription()} ${this.getAdditionalNotes()} ${this.getCustomInputs()}
${emailSchedulingBodyDivider()}
${this.getManageLink()}
${emailBodyLogo()}
`; } protected getManageLink(): string { const manageText = this.calEvent.organizer.language.translate("manage_this_event"); return `

${this.calEvent.organizer.language.translate( "need_to_reschedule_or_cancel" )}

${manageText}

`; } protected getWhat(): string { return `

${this.calEvent.organizer.language.translate("what")}

${this.calEvent.type}

`; } protected getRecurringWhen(): string { if (this.recurringEvent?.freq) { return ` - ${this.calEvent.attendees[0].language.translate("every_for_freq", { freq: this.calEvent.attendees[0].language.translate( `${rrule.FREQUENCIES[this.recurringEvent.freq].toString().toLowerCase()}` ), })} ${this.recurringEvent.count} ${this.calEvent.attendees[0].language.translate( `${rrule.FREQUENCIES[this.recurringEvent.freq].toString().toLowerCase()}`, { count: this.recurringEvent.count } )}`; } else { return ""; } } protected getWhen(): string { return `

${this.calEvent.organizer.language.translate("when")}${ this.recurringEvent?.count ? this.getRecurringWhen() : "" }

${this.recurringEvent?.count ? `${this.calEvent.attendees[0].language.translate("starting")} ` : ""} ${this.calEvent.organizer.language.translate( this.getOrganizerStart().format("dddd").toLowerCase() )}, ${this.calEvent.organizer.language.translate( this.getOrganizerStart().format("MMMM").toLowerCase() )} ${this.getOrganizerStart().format("D")}, ${this.getOrganizerStart().format( "YYYY" )} | ${this.getOrganizerStart().format("h:mma")} - ${this.getOrganizerEnd().format( "h:mma" )} (${this.getTimezone()})

`; } protected getWho(): string { const attendees = this.calEvent.attendees .map((attendee) => { return `
${ attendee?.name || `${this.calEvent.organizer.language.translate("guest")}` } ${ attendee.email }
`; }) .join(""); const organizer = `
${ this.calEvent.organizer.name } - ${this.calEvent.organizer.language.translate( "organizer" )} ${this.calEvent.organizer.email}
`; return `

${this.calEvent.organizer.language.translate("who")}

${organizer + attendees}
`; } protected getAdditionalNotes(): string { if (!this.calEvent.additionalNotes) return ""; return `

${this.calEvent.organizer.language.translate("additional_notes")}

${ this.calEvent.additionalNotes }

`; } protected getCustomInputs(): string { const { customInputs } = this.calEvent; if (!customInputs) return ""; const customInputsString = Object.keys(customInputs) .map((key) => { if (customInputs[key] !== "") { return `

${key}

${customInputs[key]}

`; } }) .join(""); return customInputsString; } protected getDescription(): string { if (!this.calEvent.description) return ""; return `

${this.calEvent.organizer.language.translate("description")}

${ this.calEvent.description }

`; } protected getLocation(): string { let providerName = this.calEvent.location && getAppName(this.calEvent.location); // This returns null if nothing is found if (this.calEvent.location && this.calEvent.location.includes("integrations:")) { const location = this.calEvent.location.split(":")[1]; providerName = location[0].toUpperCase() + location.slice(1); } // If location its a url, probably we should be validating it with a custom library if (this.calEvent.location && /^https?:\/\//.test(this.calEvent.location)) { providerName = this.calEvent.location; } if (this.calEvent.videoCallData) { const meetingId = this.calEvent.videoCallData.id; const meetingPassword = this.calEvent.videoCallData.password; const meetingUrl = this.calEvent.videoCallData.url; return `

${this.calEvent.organizer.language.translate("where")}

${providerName} ${ meetingUrl && `` }

${ meetingId && `
${this.calEvent.organizer.language.translate( "meeting_id" )}: ${meetingId}
` } ${ meetingPassword && `
${this.calEvent.organizer.language.translate( "meeting_password" )}: ${meetingPassword}
` } ${ meetingUrl && `
${this.calEvent.organizer.language.translate( "meeting_url" )}: ${meetingUrl}
` }
`; } if (this.calEvent.additionInformation?.hangoutLink) { const hangoutLink: string = this.calEvent.additionInformation.hangoutLink; return `

${this.calEvent.organizer.language.translate("where")}

${providerName} ${ hangoutLink && `` }

${hangoutLink}
`; } return `

${this.calEvent.organizer.language.translate("where")}

${ providerName || this.calEvent.location }

${ (providerName === "Zoom" || providerName === "Google") && ` ${this.calEvent.organizer.language.translate("meeting_url_provided_after_confirmed")} ` }

`; } protected getTimezone(): string { return this.calEvent.organizer.timeZone; } protected getOrganizerStart(): Dayjs { return dayjs(this.calEvent.startTime).tz(this.getTimezone()); } protected getOrganizerEnd(): Dayjs { return dayjs(this.calEvent.endTime).tz(this.getTimezone()); } }