mirror of
https://github.com/formbricks/formbricks.git
synced 2025-12-19 20:50:55 -06:00
Co-authored-by: Johannes <johannes@formbricks.com> Co-authored-by: pandeymangg <anshuman.pandey9999@gmail.com> Co-authored-by: Matthias Nannt <mail@matthiasnannt.com>
224 lines
7.8 KiB
TypeScript
224 lines
7.8 KiB
TypeScript
import { Column, Container, Hr, Img, Link, Row, Section, Text } from "@react-email/components";
|
|
import { FileDigitIcon, FileType2Icon } from "lucide-react";
|
|
import { getQuestionResponseMapping } from "@formbricks/lib/responses";
|
|
import { getOriginalFileNameFromUrl } from "@formbricks/lib/storage/utils";
|
|
import type { TOrganization } from "@formbricks/types/organizations";
|
|
import type { TResponse } from "@formbricks/types/responses";
|
|
import {
|
|
type TSurvey,
|
|
type TSurveyQuestionType,
|
|
TSurveyQuestionTypeEnum,
|
|
} from "@formbricks/types/surveys/types";
|
|
import { EmailButton } from "../general/email-button";
|
|
|
|
export const renderEmailResponseValue = (
|
|
response: string | string[],
|
|
questionType: TSurveyQuestionType
|
|
): React.JSX.Element => {
|
|
switch (questionType) {
|
|
case TSurveyQuestionTypeEnum.FileUpload:
|
|
return (
|
|
<Container>
|
|
{Array.isArray(response) &&
|
|
response.map((responseItem) => (
|
|
<Link
|
|
className="mt-2 flex flex-col items-center justify-center rounded-lg bg-gray-200 p-2 text-black shadow-sm"
|
|
href={responseItem}
|
|
key={responseItem}>
|
|
<FileIcon />
|
|
<Text className="mx-auto mb-0 truncate">{getOriginalFileNameFromUrl(responseItem)}</Text>
|
|
</Link>
|
|
))}
|
|
</Container>
|
|
);
|
|
|
|
case TSurveyQuestionTypeEnum.PictureSelection:
|
|
return (
|
|
<Container>
|
|
<Row>
|
|
{Array.isArray(response) &&
|
|
response.map((responseItem) => (
|
|
<Column key={responseItem}>
|
|
<Img alt={responseItem.split("/").pop()} className="m-2 h-28" src={responseItem} />
|
|
</Column>
|
|
))}
|
|
</Row>
|
|
</Container>
|
|
);
|
|
|
|
case TSurveyQuestionTypeEnum.Ranking:
|
|
return (
|
|
<Container>
|
|
<Row className="my-1 font-semibold text-slate-700" dir="auto">
|
|
{Array.isArray(response) &&
|
|
response.map(
|
|
(item, index) =>
|
|
item && (
|
|
<Row key={index} className="mb-1 flex items-center">
|
|
<Column className="w-6 text-gray-400">#{index + 1}</Column>
|
|
<Column className="rounded bg-gray-100 px-2 py-1">{item}</Column>
|
|
</Row>
|
|
)
|
|
)}
|
|
</Row>
|
|
</Container>
|
|
);
|
|
|
|
default:
|
|
return <Text className="mt-0 whitespace-pre-wrap break-words font-bold">{response}</Text>;
|
|
}
|
|
};
|
|
|
|
interface ResponseFinishedEmailProps {
|
|
survey: TSurvey;
|
|
responseCount: number;
|
|
response: TResponse;
|
|
WEBAPP_URL: string;
|
|
environmentId: string;
|
|
organization: TOrganization;
|
|
}
|
|
|
|
export function ResponseFinishedEmail({
|
|
survey,
|
|
responseCount,
|
|
response,
|
|
WEBAPP_URL,
|
|
environmentId,
|
|
organization,
|
|
}: ResponseFinishedEmailProps): React.JSX.Element {
|
|
const questions = getQuestionResponseMapping(survey, response);
|
|
|
|
return (
|
|
<Container>
|
|
<Row>
|
|
<Column>
|
|
<Text className="mb-4 text-3xl font-bold">Hey 👋</Text>
|
|
<Text className="mb-4">
|
|
Congrats, you received a new response to your survey! Someone just completed your survey{" "}
|
|
<strong>{survey.name}</strong>:
|
|
</Text>
|
|
<Hr />
|
|
{questions.map((question) => {
|
|
if (!question.response) return;
|
|
return (
|
|
<Row key={question.question}>
|
|
<Column className="w-full">
|
|
<Text className="mb-2 font-medium">{question.question}</Text>
|
|
{renderEmailResponseValue(question.response, question.type)}
|
|
</Column>
|
|
</Row>
|
|
);
|
|
})}
|
|
{survey.variables.map((variable) => {
|
|
const variableResponse = response.variables[variable.id];
|
|
if (variableResponse && ["number", "string"].includes(typeof variable)) {
|
|
return (
|
|
<Row key={variable.id}>
|
|
<Column className="w-full">
|
|
<Text className="mb-2 flex items-center gap-2 font-medium">
|
|
{variable.type === "number" ? (
|
|
<FileDigitIcon className="h-4 w-4" />
|
|
) : (
|
|
<FileType2Icon className="h-4 w-4" />
|
|
)}
|
|
{variable.name}
|
|
</Text>
|
|
<Text className="mt-0 whitespace-pre-wrap break-words font-bold">{variableResponse}</Text>
|
|
</Column>
|
|
</Row>
|
|
);
|
|
}
|
|
return null;
|
|
})}
|
|
{survey.hiddenFields.fieldIds?.map((hiddenFieldId) => {
|
|
const hiddenFieldResponse = response.data[hiddenFieldId];
|
|
if (hiddenFieldResponse && typeof hiddenFieldResponse === "string") {
|
|
return (
|
|
<Row key={hiddenFieldId}>
|
|
<Column className="w-full">
|
|
<Text className="mb-2 flex items-center gap-2 font-medium">
|
|
{hiddenFieldId} <EyeOffIcon />
|
|
</Text>
|
|
<Text className="mt-0 whitespace-pre-wrap break-words font-bold">
|
|
{hiddenFieldResponse}
|
|
</Text>
|
|
</Column>
|
|
</Row>
|
|
);
|
|
}
|
|
return null;
|
|
})}
|
|
<EmailButton
|
|
href={`${WEBAPP_URL}/environments/${environmentId}/surveys/${survey.id}/responses?utm_source=email_notification&utm_medium=email&utm_content=view_responses_CTA`}
|
|
label={
|
|
responseCount > 1
|
|
? `View ${String(responseCount - 1).toString()} more ${responseCount === 2 ? "response" : "responses"}`
|
|
: `View survey summary`
|
|
}
|
|
/>
|
|
<Hr />
|
|
<Section className="mt-4 text-center text-sm">
|
|
<Text className="font-bold">Don't want to get these notifications?</Text>
|
|
<Text className="mb-0">
|
|
Turn off notifications for{" "}
|
|
<Link
|
|
className="text-black underline"
|
|
href={`${WEBAPP_URL}/environments/${environmentId}/settings/notifications?type=alert&elementId=${survey.id}`}>
|
|
this form
|
|
</Link>
|
|
</Text>
|
|
<Text className="mt-0">
|
|
Turn off notifications for{" "}
|
|
<Link
|
|
className="text-black underline"
|
|
href={`${WEBAPP_URL}/environments/${environmentId}/settings/notifications?type=unsubscribedOrganizationIds&elementId=${organization.id}`}>
|
|
all newly created forms{" "}
|
|
</Link>
|
|
</Text>
|
|
</Section>
|
|
</Column>
|
|
</Row>
|
|
</Container>
|
|
);
|
|
}
|
|
|
|
function FileIcon(): React.JSX.Element {
|
|
return (
|
|
<svg
|
|
className="lucide lucide-file"
|
|
fill="none"
|
|
height="24"
|
|
stroke="currentColor"
|
|
strokeLinecap="round"
|
|
strokeLinejoin="round"
|
|
strokeWidth="2"
|
|
viewBox="0 0 24 24"
|
|
width="24"
|
|
xmlns="http://www.w3.org/2000/svg">
|
|
<path d="M15 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7Z" />
|
|
<path d="M14 2v4a2 2 0 0 0 2 2h4" />
|
|
</svg>
|
|
);
|
|
}
|
|
|
|
function EyeOffIcon(): React.JSX.Element {
|
|
return (
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
width="24"
|
|
height="24"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
stroke="currentColor"
|
|
strokeWidth="2"
|
|
strokeLinecap="round"
|
|
strokeLinejoin="round"
|
|
className="lucide lucide-eye-off h-4 w-4 rounded-lg bg-slate-200 p-1">
|
|
<path d="M9.88 9.88a3 3 0 1 0 4.24 4.24" />
|
|
<path d="M10.73 5.08A10.43 10.43 0 0 1 12 5c7 0 10 7 10 7a13.16 13.16 0 0 1-1.67 2.68" />
|
|
<path d="M6.61 6.61A13.526 13.526 0 0 0 2 12s3 7 10 7a9.74 9.74 0 0 0 5.39-1.61" />
|
|
<line x1="2" x2="22" y1="2" y2="22" />
|
|
</svg>
|
|
);
|
|
}
|