2023-04-03 05:20:44 +00:00
|
|
|
import { Collapsible, CollapsibleContent } from "@radix-ui/react-collapsible";
|
2022-05-05 14:29:49 +00:00
|
|
|
import classNames from "classnames";
|
2023-07-24 10:09:03 +00:00
|
|
|
import { useSession } from "next-auth/react";
|
2023-05-02 18:57:01 +00:00
|
|
|
import type { TFunction } from "next-i18next";
|
2023-02-16 22:39:57 +00:00
|
|
|
import type { NextRouter } from "next/router";
|
|
|
|
import { useRouter } from "next/router";
|
|
|
|
import type { MutableRefObject, RefObject } from "react";
|
|
|
|
import { createRef, forwardRef, useRef, useState } from "react";
|
|
|
|
import type { ControlProps } from "react-select";
|
|
|
|
import { components } from "react-select";
|
2023-07-24 10:09:03 +00:00
|
|
|
import { shallow } from "zustand/shallow";
|
2022-05-05 14:29:49 +00:00
|
|
|
|
2023-07-24 10:09:03 +00:00
|
|
|
import type { Dayjs } from "@calcom/dayjs";
|
|
|
|
import dayjs from "@calcom/dayjs";
|
|
|
|
import { AvailableTimes } from "@calcom/features/bookings";
|
|
|
|
import { useInitializeBookerStore, useBookerStore } from "@calcom/features/bookings/Booker/store";
|
2023-06-14 09:22:44 +00:00
|
|
|
import type { BookerLayout } from "@calcom/features/bookings/Booker/types";
|
2023-07-24 10:09:03 +00:00
|
|
|
import { useEvent, useScheduleForEvent } from "@calcom/features/bookings/Booker/utils/event";
|
|
|
|
import { useTimePreferences } from "@calcom/features/bookings/lib/timePreferences";
|
|
|
|
import DatePicker from "@calcom/features/calendars/DatePicker";
|
2023-06-14 09:22:44 +00:00
|
|
|
import { useFlagMap } from "@calcom/features/flags/context/provider";
|
2023-07-24 10:09:03 +00:00
|
|
|
import { useNonEmptyScheduleDays } from "@calcom/features/schedules";
|
|
|
|
import { useSlotsForDate } from "@calcom/features/schedules/lib/use-schedule/useSlotsForDate";
|
|
|
|
import { CAL_URL } from "@calcom/lib/constants";
|
2023-06-26 15:24:19 +00:00
|
|
|
import { APP_NAME, EMBED_LIB_URL, IS_SELF_HOSTED, WEBAPP_URL } from "@calcom/lib/constants";
|
2023-07-24 10:09:03 +00:00
|
|
|
import { weekdayToWeekIndex } from "@calcom/lib/date-fns";
|
2022-05-05 14:29:49 +00:00
|
|
|
import { useLocale } from "@calcom/lib/hooks/useLocale";
|
2023-06-14 09:22:44 +00:00
|
|
|
import { BookerLayouts } from "@calcom/prisma/zod-utils";
|
2023-07-24 10:09:03 +00:00
|
|
|
import type { RouterOutputs } from "@calcom/trpc/react";
|
|
|
|
import { trpc } from "@calcom/trpc/react";
|
|
|
|
import { TimezoneSelect } from "@calcom/ui";
|
2022-11-23 02:55:25 +00:00
|
|
|
import {
|
|
|
|
Button,
|
|
|
|
Dialog,
|
|
|
|
DialogClose,
|
2023-06-29 07:47:16 +00:00
|
|
|
DialogFooter,
|
2022-11-23 02:55:25 +00:00
|
|
|
DialogContent,
|
|
|
|
HorizontalTabs,
|
|
|
|
Label,
|
|
|
|
showToast,
|
|
|
|
Switch,
|
|
|
|
TextArea,
|
|
|
|
TextField,
|
2023-02-17 20:27:29 +00:00
|
|
|
ColorPicker,
|
2023-05-02 18:57:01 +00:00
|
|
|
Select,
|
2022-11-23 02:55:25 +00:00
|
|
|
} from "@calcom/ui";
|
2023-07-24 10:09:03 +00:00
|
|
|
import { Code, Trello, Sun, ArrowLeft, ArrowDown, ArrowUp } from "@calcom/ui/components/icon";
|
2022-05-05 14:29:49 +00:00
|
|
|
|
2023-07-24 10:09:03 +00:00
|
|
|
type EventType = RouterOutputs["viewer"]["eventTypes"]["get"]["eventType"] | undefined;
|
|
|
|
type EmbedType = "inline" | "floating-popup" | "element-click" | "email";
|
2022-06-09 05:05:18 +00:00
|
|
|
type EmbedFramework = "react" | "HTML";
|
|
|
|
|
|
|
|
const enum Theme {
|
|
|
|
auto = "auto",
|
|
|
|
light = "light",
|
|
|
|
dark = "dark",
|
|
|
|
}
|
|
|
|
|
2023-06-26 15:24:19 +00:00
|
|
|
const EMBED_CAL_ORIGIN = WEBAPP_URL;
|
|
|
|
const EMBED_CAL_JS_URL = `${WEBAPP_URL}/embed/embed.js`;
|
|
|
|
|
2022-06-09 05:05:18 +00:00
|
|
|
type PreviewState = {
|
|
|
|
inline: {
|
|
|
|
width: string;
|
|
|
|
height: string;
|
|
|
|
};
|
|
|
|
theme: Theme;
|
2023-06-26 15:24:19 +00:00
|
|
|
floatingPopup: {
|
|
|
|
config?: {
|
|
|
|
layout: BookerLayouts;
|
|
|
|
};
|
|
|
|
[key: string]: string | boolean | undefined | Record<string, string>;
|
|
|
|
};
|
2022-06-09 05:05:18 +00:00
|
|
|
elementClick: Record<string, string>;
|
|
|
|
palette: {
|
|
|
|
brandColor: string;
|
|
|
|
};
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails: boolean;
|
2023-06-14 09:22:44 +00:00
|
|
|
layout: BookerLayouts;
|
2022-06-09 05:05:18 +00:00
|
|
|
};
|
2023-07-24 10:09:03 +00:00
|
|
|
const queryParamsForDialog = ["embedType", "embedTabName", "embedUrl", "eventId"];
|
2022-05-05 14:29:49 +00:00
|
|
|
|
2022-06-09 05:05:18 +00:00
|
|
|
const getDimension = (dimension: string) => {
|
|
|
|
if (dimension.match(/^\d+$/)) {
|
|
|
|
dimension = `${dimension}%`;
|
|
|
|
}
|
|
|
|
return dimension;
|
|
|
|
};
|
|
|
|
|
2022-09-19 09:47:46 +00:00
|
|
|
const goto = (router: NextRouter, searchParams: Record<string, string>) => {
|
|
|
|
const newQuery = new URLSearchParams(router.asPath.split("?")[1]);
|
|
|
|
Object.keys(searchParams).forEach((key) => {
|
|
|
|
newQuery.set(key, searchParams[key]);
|
|
|
|
});
|
|
|
|
router.push(`${router.asPath.split("?")[0]}?${newQuery.toString()}`, undefined, {
|
|
|
|
shallow: true,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const removeQueryParams = (router: NextRouter, queryParams: string[]) => {
|
2022-11-09 15:57:23 +00:00
|
|
|
const params = new URLSearchParams(window.location.search);
|
|
|
|
|
2022-09-19 09:47:46 +00:00
|
|
|
queryParams.forEach((param) => {
|
2022-11-09 15:57:23 +00:00
|
|
|
params.delete(param);
|
2022-09-19 09:47:46 +00:00
|
|
|
});
|
2022-11-09 15:57:23 +00:00
|
|
|
|
|
|
|
router.push(`${router.asPath.split("?")[0]}?${params.toString()}`);
|
2022-09-19 09:47:46 +00:00
|
|
|
};
|
|
|
|
|
2023-07-24 10:09:03 +00:00
|
|
|
const getQueryParam = (queryParam: string) => {
|
|
|
|
const params = new URLSearchParams(window.location.search);
|
|
|
|
|
|
|
|
return params.get(queryParam);
|
|
|
|
};
|
|
|
|
|
2022-06-09 05:05:18 +00:00
|
|
|
/**
|
|
|
|
* It allows us to show code with certain reusable blocks indented according to the block variable placement
|
|
|
|
* So, if you add a variable ${abc} with indentation of 4 spaces, it will automatically indent all newlines in `abc` with the same indent before constructing the final string
|
|
|
|
* `A${var}C` with var = "B" -> partsWithoutBlock=['A','C'] blocksOrVariables=['B']
|
|
|
|
*/
|
|
|
|
const code = (partsWithoutBlock: TemplateStringsArray, ...blocksOrVariables: string[]) => {
|
|
|
|
const constructedCode: string[] = [];
|
|
|
|
for (let i = 0; i < partsWithoutBlock.length; i++) {
|
|
|
|
const partWithoutBlock = partsWithoutBlock[i];
|
|
|
|
// blocksOrVariables length would always be 1 less than partsWithoutBlock
|
|
|
|
// So, last item should be concatenated as is.
|
|
|
|
if (i >= blocksOrVariables.length) {
|
|
|
|
constructedCode.push(partWithoutBlock);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const block = blocksOrVariables[i];
|
|
|
|
const indentedBlock: string[] = [];
|
|
|
|
let indent = "";
|
|
|
|
block.split("\n").forEach((line) => {
|
|
|
|
indentedBlock.push(line);
|
|
|
|
});
|
|
|
|
// non-null assertion is okay because we know that we are referencing last element.
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
|
|
const indentationMatch = partWithoutBlock
|
|
|
|
.split("\n")
|
|
|
|
.at(-1)!
|
|
|
|
.match(/(^[\t ]*).*$/);
|
|
|
|
if (indentationMatch) {
|
|
|
|
indent = indentationMatch[1];
|
|
|
|
}
|
|
|
|
constructedCode.push(partWithoutBlock + indentedBlock.join("\n" + indent));
|
|
|
|
}
|
|
|
|
return constructedCode.join("");
|
|
|
|
};
|
|
|
|
|
|
|
|
const getInstructionString = ({
|
|
|
|
apiName,
|
|
|
|
instructionName,
|
|
|
|
instructionArg,
|
|
|
|
}: {
|
|
|
|
apiName: string;
|
|
|
|
instructionName: string;
|
|
|
|
instructionArg: Record<string, unknown>;
|
|
|
|
}) => {
|
|
|
|
return `${apiName}("${instructionName}", ${JSON.stringify(instructionArg)});`;
|
|
|
|
};
|
|
|
|
|
|
|
|
const getEmbedUIInstructionString = ({
|
|
|
|
apiName,
|
|
|
|
theme,
|
|
|
|
brandColor,
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails,
|
2023-06-14 09:22:44 +00:00
|
|
|
layout,
|
2022-06-09 05:05:18 +00:00
|
|
|
}: {
|
|
|
|
apiName: string;
|
|
|
|
theme?: string;
|
|
|
|
brandColor: string;
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails: boolean;
|
2023-06-14 09:22:44 +00:00
|
|
|
layout?: string;
|
2022-06-09 05:05:18 +00:00
|
|
|
}) => {
|
|
|
|
theme = theme !== "auto" ? theme : undefined;
|
|
|
|
return getInstructionString({
|
|
|
|
apiName,
|
|
|
|
instructionName: "ui",
|
|
|
|
instructionArg: {
|
|
|
|
theme,
|
|
|
|
styles: {
|
|
|
|
branding: {
|
|
|
|
brandColor,
|
|
|
|
},
|
|
|
|
},
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails: hideEventTypeDetails,
|
2023-06-14 09:22:44 +00:00
|
|
|
layout,
|
2022-06-09 05:05:18 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
const Codes: Record<string, Record<string, (...args: any[]) => string>> = {
|
|
|
|
react: {
|
|
|
|
inline: ({
|
|
|
|
calLink,
|
|
|
|
uiInstructionCode,
|
|
|
|
previewState,
|
|
|
|
}: {
|
|
|
|
calLink: string;
|
|
|
|
uiInstructionCode: string;
|
|
|
|
previewState: PreviewState;
|
|
|
|
}) => {
|
|
|
|
const width = getDimension(previewState.inline.width);
|
|
|
|
const height = getDimension(previewState.inline.height);
|
|
|
|
return code`
|
|
|
|
import Cal, { getCalApi } from "@calcom/embed-react";
|
2023-03-28 18:17:40 +00:00
|
|
|
import { useEffect } from "react";
|
2023-06-26 15:24:19 +00:00
|
|
|
export default function MyApp() {
|
2022-06-09 05:05:18 +00:00
|
|
|
useEffect(()=>{
|
|
|
|
(async function () {
|
|
|
|
const cal = await getCalApi();
|
|
|
|
${uiInstructionCode}
|
|
|
|
})();
|
2022-07-23 00:39:50 +00:00
|
|
|
}, [])
|
2023-07-18 21:18:08 +00:00
|
|
|
return <Cal
|
|
|
|
calLink="${calLink}"
|
2023-06-26 15:24:19 +00:00
|
|
|
style={{width:"${width}",height:"${height}",overflow:"scroll"}}
|
|
|
|
${previewState.layout ? "config={{layout: '" + previewState.layout + "'}}" : ""}${
|
|
|
|
IS_SELF_HOSTED
|
|
|
|
? `
|
|
|
|
calOrigin="${EMBED_CAL_ORIGIN}"
|
|
|
|
calJsUrl="${EMBED_CAL_JS_URL}"`
|
|
|
|
: ""
|
|
|
|
}
|
|
|
|
/>;
|
2022-06-09 05:05:18 +00:00
|
|
|
};`;
|
|
|
|
},
|
|
|
|
"floating-popup": ({
|
|
|
|
floatingButtonArg,
|
|
|
|
uiInstructionCode,
|
|
|
|
}: {
|
|
|
|
floatingButtonArg: string;
|
|
|
|
uiInstructionCode: string;
|
|
|
|
}) => {
|
|
|
|
return code`
|
2023-06-26 15:24:19 +00:00
|
|
|
import { getCalApi } from "@calcom/embed-react";
|
2023-03-28 18:17:40 +00:00
|
|
|
import { useEffect } from "react";
|
2023-06-26 15:24:19 +00:00
|
|
|
export default function App() {
|
2022-06-09 05:05:18 +00:00
|
|
|
useEffect(()=>{
|
|
|
|
(async function () {
|
2023-06-26 15:24:19 +00:00
|
|
|
const cal = await getCalApi(${IS_SELF_HOSTED ? `"${EMBED_CAL_JS_URL}"` : ""});
|
2023-03-28 18:17:40 +00:00
|
|
|
cal("floatingButton", ${floatingButtonArg});
|
2022-06-09 05:05:18 +00:00
|
|
|
${uiInstructionCode}
|
|
|
|
})();
|
2022-07-23 00:39:50 +00:00
|
|
|
}, [])
|
2022-06-09 05:05:18 +00:00
|
|
|
};`;
|
|
|
|
},
|
2023-06-26 15:24:19 +00:00
|
|
|
"element-click": ({
|
|
|
|
calLink,
|
|
|
|
uiInstructionCode,
|
|
|
|
previewState,
|
|
|
|
}: {
|
|
|
|
calLink: string;
|
|
|
|
uiInstructionCode: string;
|
|
|
|
previewState: PreviewState;
|
|
|
|
}) => {
|
2022-06-09 05:05:18 +00:00
|
|
|
return code`
|
2023-06-26 15:24:19 +00:00
|
|
|
import { getCalApi } from "@calcom/embed-react";
|
2023-03-28 18:17:40 +00:00
|
|
|
import { useEffect } from "react";
|
2023-06-26 15:24:19 +00:00
|
|
|
export default function App() {
|
2022-06-09 05:05:18 +00:00
|
|
|
useEffect(()=>{
|
|
|
|
(async function () {
|
2023-06-26 15:24:19 +00:00
|
|
|
const cal = await getCalApi(${IS_SELF_HOSTED ? `"${EMBED_CAL_JS_URL}"` : ""});
|
2022-06-09 05:05:18 +00:00
|
|
|
${uiInstructionCode}
|
|
|
|
})();
|
2022-07-23 00:39:50 +00:00
|
|
|
}, [])
|
2023-07-18 21:18:08 +00:00
|
|
|
return <button
|
2023-06-26 15:24:19 +00:00
|
|
|
data-cal-link="${calLink}"${IS_SELF_HOSTED ? `\ndata-cal-origin="${EMBED_CAL_ORIGIN}"` : ""}
|
|
|
|
${`data-cal-config='${JSON.stringify({
|
|
|
|
layout: previewState.layout,
|
2023-07-18 21:18:08 +00:00
|
|
|
})}'`}
|
2023-06-26 15:24:19 +00:00
|
|
|
>Click me</button>;
|
2022-06-09 05:05:18 +00:00
|
|
|
};`;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
HTML: {
|
2023-06-26 15:24:19 +00:00
|
|
|
inline: ({
|
|
|
|
calLink,
|
|
|
|
uiInstructionCode,
|
|
|
|
previewState,
|
|
|
|
}: {
|
|
|
|
calLink: string;
|
|
|
|
uiInstructionCode: string;
|
|
|
|
previewState: PreviewState;
|
|
|
|
}) => {
|
2022-06-09 05:05:18 +00:00
|
|
|
return code`Cal("inline", {
|
|
|
|
elementOrSelector:"#my-cal-inline",
|
2023-06-26 15:24:19 +00:00
|
|
|
calLink: "${calLink}",
|
|
|
|
layout: "${previewState.layout}"
|
2022-06-09 05:05:18 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
${uiInstructionCode}`;
|
|
|
|
},
|
|
|
|
|
|
|
|
"floating-popup": ({
|
|
|
|
floatingButtonArg,
|
|
|
|
uiInstructionCode,
|
|
|
|
}: {
|
|
|
|
floatingButtonArg: string;
|
|
|
|
uiInstructionCode: string;
|
|
|
|
}) => {
|
|
|
|
return code`Cal("floatingButton", ${floatingButtonArg});
|
|
|
|
${uiInstructionCode}`;
|
|
|
|
},
|
2023-06-26 15:24:19 +00:00
|
|
|
"element-click": ({
|
|
|
|
calLink,
|
|
|
|
uiInstructionCode,
|
|
|
|
previewState,
|
|
|
|
}: {
|
|
|
|
calLink: string;
|
|
|
|
uiInstructionCode: string;
|
|
|
|
previewState: PreviewState;
|
|
|
|
}) => {
|
|
|
|
return code`
|
|
|
|
// Important: Please add following attributes to the element you want to open Cal on click
|
2023-07-18 21:18:08 +00:00
|
|
|
// \`data-cal-link="${calLink}"\`
|
2023-06-26 15:24:19 +00:00
|
|
|
// \`data-cal-config='${JSON.stringify({
|
|
|
|
layout: previewState.layout,
|
2023-07-18 21:18:08 +00:00
|
|
|
})}'\`
|
2023-06-26 15:24:19 +00:00
|
|
|
|
2022-06-09 05:05:18 +00:00
|
|
|
${uiInstructionCode}`;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const getEmbedTypeSpecificString = ({
|
|
|
|
embedFramework,
|
|
|
|
embedType,
|
|
|
|
calLink,
|
|
|
|
previewState,
|
|
|
|
}: {
|
|
|
|
embedFramework: EmbedFramework;
|
|
|
|
embedType: EmbedType;
|
|
|
|
calLink: string;
|
|
|
|
previewState: PreviewState;
|
|
|
|
}) => {
|
|
|
|
const frameworkCodes = Codes[embedFramework];
|
|
|
|
if (!frameworkCodes) {
|
|
|
|
throw new Error(`No code available for the framework:${embedFramework}`);
|
|
|
|
}
|
2023-07-24 10:09:03 +00:00
|
|
|
if (embedType === "email") return "";
|
2022-08-24 20:18:42 +00:00
|
|
|
let uiInstructionStringArg: {
|
|
|
|
apiName: string;
|
|
|
|
theme: PreviewState["theme"];
|
|
|
|
brandColor: string;
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails: boolean;
|
2023-06-14 09:22:44 +00:00
|
|
|
layout?: BookerLayout;
|
2022-08-24 20:18:42 +00:00
|
|
|
};
|
2022-06-09 05:05:18 +00:00
|
|
|
if (embedFramework === "react") {
|
|
|
|
uiInstructionStringArg = {
|
|
|
|
apiName: "cal",
|
|
|
|
theme: previewState.theme,
|
|
|
|
brandColor: previewState.palette.brandColor,
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails: previewState.hideEventTypeDetails,
|
2023-06-14 09:22:44 +00:00
|
|
|
layout: previewState.layout,
|
2022-06-09 05:05:18 +00:00
|
|
|
};
|
|
|
|
} else {
|
|
|
|
uiInstructionStringArg = {
|
|
|
|
apiName: "Cal",
|
|
|
|
theme: previewState.theme,
|
|
|
|
brandColor: previewState.palette.brandColor,
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails: previewState.hideEventTypeDetails,
|
2023-06-14 09:22:44 +00:00
|
|
|
layout: previewState.layout,
|
2022-06-09 05:05:18 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
if (!frameworkCodes[embedType]) {
|
|
|
|
throw new Error(`Code not available for framework:${embedFramework} and embedType:${embedType}`);
|
|
|
|
}
|
|
|
|
if (embedType === "inline") {
|
|
|
|
return frameworkCodes[embedType]({
|
|
|
|
calLink,
|
|
|
|
uiInstructionCode: getEmbedUIInstructionString(uiInstructionStringArg),
|
|
|
|
previewState,
|
|
|
|
});
|
|
|
|
} else if (embedType === "floating-popup") {
|
|
|
|
const floatingButtonArg = {
|
|
|
|
calLink,
|
2023-06-26 15:24:19 +00:00
|
|
|
...(IS_SELF_HOSTED ? { calOrigin: EMBED_CAL_ORIGIN } : null),
|
2022-06-09 05:05:18 +00:00
|
|
|
...previewState.floatingPopup,
|
|
|
|
};
|
|
|
|
return frameworkCodes[embedType]({
|
|
|
|
floatingButtonArg: JSON.stringify(floatingButtonArg),
|
|
|
|
uiInstructionCode: getEmbedUIInstructionString(uiInstructionStringArg),
|
|
|
|
previewState,
|
|
|
|
});
|
|
|
|
} else if (embedType === "element-click") {
|
|
|
|
return frameworkCodes[embedType]({
|
|
|
|
calLink,
|
|
|
|
uiInstructionCode: getEmbedUIInstructionString(uiInstructionStringArg),
|
|
|
|
previewState,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
};
|
|
|
|
|
2023-05-02 18:57:01 +00:00
|
|
|
const embeds = (t: TFunction) =>
|
|
|
|
(() => {
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
title: t("inline_embed"),
|
|
|
|
subtitle: t("load_inline_content"),
|
|
|
|
type: "inline",
|
|
|
|
illustration: (
|
|
|
|
<svg
|
|
|
|
width="100%"
|
|
|
|
height="100%"
|
|
|
|
className="rounded-md"
|
|
|
|
viewBox="0 0 308 265"
|
|
|
|
fill="none"
|
|
|
|
xmlns="http://www.w3.org/2000/svg">
|
|
|
|
<path
|
|
|
|
d="M0 1.99999C0 0.895423 0.895431 0 2 0H306C307.105 0 308 0.895431 308 2V263C308 264.105 307.105 265 306 265H2C0.895431 265 0 264.105 0 263V1.99999Z"
|
|
|
|
fill="white"
|
|
|
|
/>
|
|
|
|
<rect x="24" width="260" height="38.5" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="24.5" y="51" width="139" height="163" rx="1.5" fill="#F8F8F8" />
|
|
|
|
<rect opacity="0.8" x="48" y="74.5" width="80" height="8" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="48" y="86.5" width="48" height="4" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="49" y="99.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="61" y="99.5" width="6" height="6" rx="1" fill="#3E3E3E" />
|
|
|
|
<rect x="73" y="99.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="85" y="99.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="97" y="99.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="109" y="99.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="121" y="99.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="133" y="99.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="85" y="113.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="97" y="113.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="109" y="113.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="121" y="113.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="133" y="113.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="49" y="125.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="61" y="125.5" width="6" height="6" rx="1" fill="#3E3E3E" />
|
|
|
|
<path
|
|
|
|
d="M61 124.5H67V122.5H61V124.5ZM68 125.5V131.5H70V125.5H68ZM67 132.5H61V134.5H67V132.5ZM60 131.5V125.5H58V131.5H60ZM61 132.5C60.4477 132.5 60 132.052 60 131.5H58C58 133.157 59.3431 134.5 61 134.5V132.5ZM68 131.5C68 132.052 67.5523 132.5 67 132.5V134.5C68.6569 134.5 70 133.157 70 131.5H68ZM67 124.5C67.5523 124.5 68 124.948 68 125.5H70C70 123.843 68.6569 122.5 67 122.5V124.5ZM61 122.5C59.3431 122.5 58 123.843 58 125.5H60C60 124.948 60.4477 124.5 61 124.5V122.5Z"
|
|
|
|
fill="#3E3E3E"
|
|
|
|
/>
|
|
|
|
<rect x="73" y="125.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="85" y="125.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="97" y="125.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="109" y="125.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="121" y="125.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="133" y="125.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="49" y="137.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="61" y="137.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="73" y="137.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="85" y="137.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="97" y="137.5" width="6" height="6" rx="1" fill="#3E3E3E" />
|
|
|
|
<rect x="109" y="137.5" width="6" height="6" rx="1" fill="#3E3E3E" />
|
|
|
|
<rect x="121" y="137.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="133" y="137.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="49" y="149.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="61" y="149.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="73" y="149.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="85" y="149.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="97" y="149.5" width="6" height="6" rx="1" fill="#3E3E3E" />
|
|
|
|
<rect x="109" y="149.5" width="6" height="6" rx="1" fill="#3E3E3E" />
|
|
|
|
<rect x="121" y="149.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="133" y="149.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="49" y="161.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="61" y="161.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="73" y="161.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="85" y="161.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="97" y="161.5" width="6" height="6" rx="1" fill="#3E3E3E" />
|
|
|
|
<rect x="109" y="161.5" width="6" height="6" rx="1" fill="#C6C6C6" />
|
|
|
|
<rect x="24.5" y="51" width="139" height="163" rx="6" stroke="#292929" />
|
|
|
|
<rect x="176" y="50.5" width="108" height="164" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="24" y="226.5" width="260" height="38.5" rx="6" fill="#F3F4F6" />
|
|
|
|
</svg>
|
|
|
|
),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: t("floating_pop_up_button"),
|
|
|
|
subtitle: t("floating_button_trigger_modal"),
|
|
|
|
type: "floating-popup",
|
|
|
|
illustration: (
|
|
|
|
<svg
|
|
|
|
width="100%"
|
|
|
|
height="100%"
|
|
|
|
className="rounded-md"
|
|
|
|
viewBox="0 0 308 265"
|
|
|
|
fill="none"
|
|
|
|
xmlns="http://www.w3.org/2000/svg">
|
|
|
|
<path
|
|
|
|
d="M0 1.99999C0 0.895423 0.895431 0 2 0H306C307.105 0 308 0.895431 308 2V263C308 264.105 307.105 265 306 265H2C0.895431 265 0 264.105 0 263V1.99999Z"
|
|
|
|
fill="white"
|
|
|
|
/>
|
|
|
|
<rect x="24" width="260" height="38.5" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="24" y="50.5" width="120" height="76" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="24" y="138.5" width="120" height="76" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="156" y="50.5" width="128" height="164" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="24" y="226.5" width="260" height="38.5" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="226" y="223.5" width="66" height="26" rx="6" fill="#292929" />
|
|
|
|
<rect x="242" y="235.5" width="34" height="2" rx="1" fill="white" />
|
|
|
|
</svg>
|
|
|
|
),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: t("pop_up_element_click"),
|
|
|
|
subtitle: t("open_dialog_with_element_click"),
|
|
|
|
type: "element-click",
|
|
|
|
illustration: (
|
|
|
|
<svg
|
|
|
|
width="100%"
|
|
|
|
height="100%"
|
|
|
|
className="rounded-md"
|
|
|
|
viewBox="0 0 308 265"
|
|
|
|
fill="none"
|
|
|
|
xmlns="http://www.w3.org/2000/svg">
|
|
|
|
<path
|
|
|
|
d="M0 1.99999C0 0.895423 0.895431 0 2 0H306C307.105 0 308 0.895431 308 2V263C308 264.105 307.105 265 306 265H2C0.895431 265 0 264.105 0 263V1.99999Z"
|
|
|
|
fill="white"
|
|
|
|
/>
|
|
|
|
<rect x="24" y="0.50293" width="260" height="24" rx="6" fill="#F3F4F6" />
|
|
|
|
<rect x="24" y="35" width="259" height="192" rx="5.5" fill="#F9FAFB" />
|
|
|
|
<g filter="url(#filter0_i_3223_14162)">
|
|
|
|
<rect opacity="0.8" x="40" y="99" width="24" height="24" rx="2" fill="#E5E7EB" />
|
|
|
|
<rect x="40" y="127" width="48" height="8" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="40" y="139" width="82" height="8" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="40" y="151" width="34" height="4" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="40" y="159" width="34" height="4" rx="1" fill="#E5E7EB" />
|
|
|
|
</g>
|
|
|
|
<rect x="152" y="48" width="2" height="169" rx="2" fill="#E5E7EB" />
|
|
|
|
|
|
|
|
<rect opacity="0.8" x="176" y="84" width="80" height="8" rx="2" fill="#E5E7EB" />
|
|
|
|
<rect x="176" y="96" width="48" height="4" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="177" y="109" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="189" y="109" width="6" height="6" rx="1" fill="#0D121D" />
|
|
|
|
<rect x="201" y="109" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="213" y="109" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="225" y="109" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="237" y="109" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="249" y="109" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="261" y="109" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="213" y="123" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="225" y="123" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="237" y="123" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="249" y="123" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="261" y="123" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="177" y="135" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="189" y="135" width="6" height="6" rx="1" fill="#0D121D" />
|
|
|
|
<rect x="187.3" y="133.4" width="9" height="9" rx="1.5" stroke="#0D121D" />
|
|
|
|
<rect x="201" y="135" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="213" y="135" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="225" y="135" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="237" y="135" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="249" y="135" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="261" y="135" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="177" y="147" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="189" y="147" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="201" y="147" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="213" y="147" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="225" y="147" width="6" height="6" rx="1" fill="#0D121D" />
|
|
|
|
<rect x="237" y="147" width="6" height="6" rx="1" fill="#0D121D" />
|
|
|
|
<rect x="249" y="147" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="261" y="147" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="177" y="159" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="189" y="159" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="201" y="159" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="213" y="159" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="225" y="159" width="6" height="6" rx="1" fill="#0D121D" />
|
|
|
|
<rect x="237" y="159" width="6" height="6" rx="1" fill="#0D121D" />
|
|
|
|
<rect x="249" y="159" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="261" y="159" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="177" y="171" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="189" y="171" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="201" y="171" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="213" y="171" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="225" y="171" width="6" height="6" rx="1" fill="#0D121D" />
|
|
|
|
<rect x="237" y="171" width="6" height="6" rx="1" fill="#E5E7EB" />
|
|
|
|
<rect x="24" y="35" width="259" height="192" rx="5.5" stroke="#101010" />
|
|
|
|
<rect x="24" y="241.503" width="260" height="24" rx="6" fill="#F3F4F6" />
|
|
|
|
</svg>
|
|
|
|
),
|
|
|
|
},
|
2023-07-24 10:09:03 +00:00
|
|
|
{
|
|
|
|
title: t("email_embed"),
|
|
|
|
subtitle: t("add_times_to_your_email"),
|
|
|
|
type: "email",
|
|
|
|
illustration: (
|
|
|
|
<svg
|
|
|
|
width="100%"
|
|
|
|
height="100%"
|
|
|
|
className="rounded-md"
|
|
|
|
viewBox="0 0 308 265"
|
|
|
|
fill="none"
|
|
|
|
xmlns="http://www.w3.org/2000/svg">
|
2023-07-27 20:31:55 +00:00
|
|
|
<g clip-path="url(#clip0_457_1339)">
|
|
|
|
<rect width="308" height="265" rx="8" fill="white" />
|
|
|
|
<rect width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="19" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="38" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="57" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="76" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="95" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="114" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="133" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="152" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="171" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="190" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="209" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="228" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<rect y="247" width="308" height="18" rx="4" fill="#F3F4F6" />
|
|
|
|
<g clip-path="url(#clip1_457_1339)">
|
|
|
|
<rect x="107" y="64" width="189" height="189" rx="6" fill="white" />
|
|
|
|
<g clip-path="url(#clip2_457_1339)">
|
|
|
|
<path
|
|
|
|
d="M124.671 75.5243C124.671 75.1018 124.325 74.756 123.902 74.756H117.756C117.334 74.756 116.988 75.1018 116.988 75.5243M124.671 75.5243V80.1341C124.671 80.5567 124.325 80.9024 123.902 80.9024H117.756C117.334 80.9024 116.988 80.5567 116.988 80.1341V75.5243M124.671 75.5243L120.829 78.2134L116.988 75.5243"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
stroke-width="1.15244"
|
|
|
|
stroke-linecap="round"
|
|
|
|
stroke-linejoin="round"
|
|
|
|
/>
|
|
|
|
</g>
|
|
|
|
<rect x="130.049" y="75.5244" width="92.1951" height="4.60976" rx="2.30488" fill="#D1D5DB" />
|
|
|
|
<rect x="130.049" y="84.7439" width="55.3171" height="4.60976" rx="2.30488" fill="#E5E7EB" />
|
|
|
|
<rect
|
|
|
|
opacity="0.8"
|
|
|
|
x="107"
|
|
|
|
y="98.5732"
|
|
|
|
width="189"
|
|
|
|
height="1.15244"
|
|
|
|
rx="0.576219"
|
|
|
|
fill="#E5E7EB"
|
|
|
|
/>
|
|
|
|
<rect x="116.219" y="113.555" width="92.1951" height="4.60976" rx="2.30488" fill="#D1D5DB" />
|
|
|
|
<rect x="116.219" y="122.774" width="42.6402" height="4.60976" rx="2.30488" fill="#E5E7EB" />
|
|
|
|
<rect x="116.219" y="136.604" width="55.3171" height="4.60976" rx="2.30488" fill="#D1D5DB" />
|
|
|
|
<rect
|
|
|
|
x="116.719"
|
|
|
|
y="145.171"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="142.073"
|
|
|
|
y="145.171"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="167.427"
|
|
|
|
y="145.171"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="192.781"
|
|
|
|
y="145.171"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="218.134"
|
|
|
|
y="145.171"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect x="116.219" y="160.805" width="55.3171" height="4.60976" rx="2.30488" fill="#D1D5DB" />
|
|
|
|
<rect
|
|
|
|
x="116.719"
|
|
|
|
y="169.372"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="142.073"
|
|
|
|
y="169.372"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="167.427"
|
|
|
|
y="169.372"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="192.781"
|
|
|
|
y="169.372"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="218.134"
|
|
|
|
y="169.372"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect x="116.219" y="185.006" width="55.3171" height="4.60976" rx="2.30488" fill="#D1D5DB" />
|
|
|
|
<rect
|
|
|
|
x="116.719"
|
|
|
|
y="193.573"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="142.073"
|
|
|
|
y="193.573"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="167.427"
|
|
|
|
y="193.573"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="192.781"
|
|
|
|
y="193.573"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
x="218.134"
|
|
|
|
y="193.573"
|
|
|
|
width="22.0488"
|
|
|
|
height="5.91463"
|
|
|
|
rx="2.95732"
|
|
|
|
stroke="#9CA3AF"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
opacity="0.8"
|
|
|
|
x="107"
|
|
|
|
y="223.037"
|
|
|
|
width="189"
|
|
|
|
height="1.15244"
|
|
|
|
rx="0.576219"
|
|
|
|
fill="#E5E7EB"
|
|
|
|
/>
|
|
|
|
<rect width="189" height="28.811" transform="translate(107 224.189)" fill="#F9FAFB" />
|
|
|
|
<rect x="116.219" y="233.985" width="23.0488" height="9.21951" rx="2.30488" fill="#9CA3AF" />
|
|
|
|
<rect
|
|
|
|
opacity="0.8"
|
|
|
|
x="141.573"
|
|
|
|
y="233.985"
|
|
|
|
width="9.21951"
|
|
|
|
height="9.21951"
|
|
|
|
rx="2.30488"
|
|
|
|
fill="#E5E7EB"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
opacity="0.8"
|
|
|
|
x="153.098"
|
|
|
|
y="233.985"
|
|
|
|
width="9.21951"
|
|
|
|
height="9.21951"
|
|
|
|
rx="2.30488"
|
|
|
|
fill="#E5E7EB"
|
|
|
|
/>
|
|
|
|
<rect
|
|
|
|
opacity="0.8"
|
|
|
|
x="164.622"
|
|
|
|
y="233.985"
|
|
|
|
width="9.21951"
|
|
|
|
height="9.21951"
|
|
|
|
rx="2.30488"
|
|
|
|
fill="#E5E7EB"
|
|
|
|
/>
|
|
|
|
</g>
|
|
|
|
<rect
|
|
|
|
x="106.424"
|
|
|
|
y="63.4238"
|
|
|
|
width="190.152"
|
|
|
|
height="190.152"
|
|
|
|
rx="6.57622"
|
|
|
|
stroke="#101010"
|
|
|
|
stroke-width="1.15244"
|
|
|
|
/>
|
|
|
|
</g>
|
|
|
|
<rect x="0.5" y="0.5" width="307" height="264" rx="7.5" stroke="#E5E7EB" />
|
|
|
|
<defs>
|
|
|
|
<clipPath id="clip0_457_1339">
|
|
|
|
<rect width="308" height="265" rx="8" fill="white" />
|
|
|
|
</clipPath>
|
|
|
|
<clipPath id="clip1_457_1339">
|
|
|
|
<rect x="107" y="64" width="189" height="189" rx="6" fill="white" />
|
|
|
|
</clipPath>
|
|
|
|
<clipPath id="clip2_457_1339">
|
|
|
|
<rect width="9.21951" height="9.21951" fill="white" transform="translate(116.219 73.2195)" />
|
|
|
|
</clipPath>
|
|
|
|
</defs>
|
2023-07-24 10:09:03 +00:00
|
|
|
</svg>
|
|
|
|
),
|
|
|
|
},
|
2023-05-02 18:57:01 +00:00
|
|
|
];
|
|
|
|
})();
|
2023-03-20 07:52:54 +00:00
|
|
|
|
2022-06-09 05:05:18 +00:00
|
|
|
const tabs = [
|
|
|
|
{
|
|
|
|
name: "HTML",
|
2022-09-18 23:14:57 +00:00
|
|
|
href: "embedTabName=embed-code",
|
2023-04-12 15:26:31 +00:00
|
|
|
icon: Code,
|
2022-06-09 05:05:18 +00:00
|
|
|
type: "code",
|
|
|
|
Component: forwardRef<
|
|
|
|
HTMLTextAreaElement | HTMLIFrameElement | null,
|
|
|
|
{ embedType: EmbedType; calLink: string; previewState: PreviewState }
|
|
|
|
>(function EmbedHtml({ embedType, calLink, previewState }, ref) {
|
|
|
|
const { t } = useLocale();
|
|
|
|
if (ref instanceof Function || !ref) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (ref.current && !(ref.current instanceof HTMLTextAreaElement)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<>
|
2022-06-29 08:06:36 +00:00
|
|
|
<div>
|
2023-04-05 18:14:46 +00:00
|
|
|
<small className="text-subtle flex py-4">
|
2022-11-30 21:52:56 +00:00
|
|
|
{t("place_where_cal_widget_appear", { appName: APP_NAME })}
|
|
|
|
</small>
|
2022-06-29 08:06:36 +00:00
|
|
|
</div>
|
2022-06-09 05:05:18 +00:00
|
|
|
<TextArea
|
|
|
|
data-testid="embed-code"
|
|
|
|
ref={ref as typeof ref & MutableRefObject<HTMLTextAreaElement>}
|
|
|
|
name="embed-code"
|
2023-04-05 18:14:46 +00:00
|
|
|
className="text-default bg-default selection:bg-subtle h-[calc(100%-50px)] font-mono"
|
2022-07-27 02:24:00 +00:00
|
|
|
style={{ resize: "none", overflow: "auto" }}
|
2022-06-09 05:05:18 +00:00
|
|
|
readOnly
|
|
|
|
value={
|
|
|
|
`<!-- Cal ${embedType} embed code begins -->\n` +
|
|
|
|
(embedType === "inline"
|
|
|
|
? `<div style="width:${getDimension(previewState.inline.width)};height:${getDimension(
|
|
|
|
previewState.inline.height
|
|
|
|
)};overflow:scroll" id="my-cal-inline"></div>\n`
|
|
|
|
: "") +
|
|
|
|
`<script type="text/javascript">
|
|
|
|
${getEmbedSnippetString()}
|
|
|
|
${getEmbedTypeSpecificString({ embedFramework: "HTML", embedType, calLink, previewState })}
|
|
|
|
</script>
|
|
|
|
<!-- Cal ${embedType} embed code ends -->`
|
2022-07-13 21:14:16 +00:00
|
|
|
}
|
|
|
|
/>
|
2023-05-02 18:57:01 +00:00
|
|
|
<p className="text-subtle hidden text-sm">{t("need_help_embedding")}</p>
|
2022-06-09 05:05:18 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "React",
|
2022-09-18 23:14:57 +00:00
|
|
|
href: "embedTabName=embed-react",
|
2023-04-12 15:26:31 +00:00
|
|
|
icon: Code,
|
2022-06-09 05:05:18 +00:00
|
|
|
type: "code",
|
|
|
|
Component: forwardRef<
|
|
|
|
HTMLTextAreaElement | HTMLIFrameElement | null,
|
|
|
|
{ embedType: EmbedType; calLink: string; previewState: PreviewState }
|
|
|
|
>(function EmbedReact({ embedType, calLink, previewState }, ref) {
|
|
|
|
const { t } = useLocale();
|
|
|
|
if (ref instanceof Function || !ref) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (ref.current && !(ref.current instanceof HTMLTextAreaElement)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<>
|
2023-04-05 18:14:46 +00:00
|
|
|
<small className="text-subtle flex py-4">{t("create_update_react_component")}</small>
|
2022-06-09 05:05:18 +00:00
|
|
|
<TextArea
|
|
|
|
data-testid="embed-react"
|
|
|
|
ref={ref as typeof ref & MutableRefObject<HTMLTextAreaElement>}
|
|
|
|
name="embed-react"
|
2023-04-05 18:14:46 +00:00
|
|
|
className="text-default bg-default selection:bg-subtle h-[calc(100%-50px)] font-mono"
|
2022-06-09 05:05:18 +00:00
|
|
|
readOnly
|
2022-07-27 02:24:00 +00:00
|
|
|
style={{ resize: "none", overflow: "auto" }}
|
2022-06-09 05:05:18 +00:00
|
|
|
value={`/* First make sure that you have installed the package */
|
|
|
|
|
|
|
|
/* If you are using yarn */
|
|
|
|
// yarn add @calcom/embed-react
|
|
|
|
|
|
|
|
/* If you are using npm */
|
|
|
|
// npm install @calcom/embed-react
|
|
|
|
${getEmbedTypeSpecificString({ embedFramework: "react", embedType, calLink, previewState })}
|
2022-07-13 21:14:16 +00:00
|
|
|
`}
|
|
|
|
/>
|
2022-06-09 05:05:18 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Preview",
|
2022-09-18 23:14:57 +00:00
|
|
|
href: "embedTabName=embed-preview",
|
2023-04-12 15:26:31 +00:00
|
|
|
icon: Trello,
|
2022-06-09 05:05:18 +00:00
|
|
|
type: "iframe",
|
|
|
|
Component: forwardRef<
|
|
|
|
HTMLIFrameElement | HTMLTextAreaElement | null,
|
2022-08-24 20:18:42 +00:00
|
|
|
{ calLink: string; embedType: EmbedType; previewState: PreviewState }
|
2022-06-09 05:05:18 +00:00
|
|
|
>(function Preview({ calLink, embedType }, ref) {
|
|
|
|
if (ref instanceof Function || !ref) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (ref.current && !(ref.current instanceof HTMLIFrameElement)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<iframe
|
|
|
|
ref={ref as typeof ref & MutableRefObject<HTMLIFrameElement>}
|
|
|
|
data-testid="embed-preview"
|
2023-01-20 22:04:58 +00:00
|
|
|
className="h-[100vh] border"
|
2022-06-09 05:05:18 +00:00
|
|
|
width="100%"
|
|
|
|
height="100%"
|
|
|
|
src={`${WEBAPP_URL}/embed/preview.html?embedType=${embedType}&calLink=${calLink}`}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
];
|
2022-05-05 14:29:49 +00:00
|
|
|
|
|
|
|
function getEmbedSnippetString() {
|
|
|
|
// TODO: Import this string from @calcom/embed-snippet
|
2022-06-09 05:05:18 +00:00
|
|
|
return `(function (C, A, L) { let p = function (a, ar) { a.q.push(ar); }; let d = C.document; C.Cal = C.Cal || function () { let cal = C.Cal; let ar = arguments; if (!cal.loaded) { cal.ns = {}; cal.q = cal.q || []; d.head.appendChild(d.createElement("script")).src = A; cal.loaded = true; } if (ar[0] === L) { const api = function () { p(api, arguments); }; const namespace = ar[1]; api.q = api.q || []; typeof namespace === "string" ? (cal.ns[namespace] = api) && p(api, ar) : p(cal, ar); return; } p(cal, ar); }; })(window, "${EMBED_LIB_URL}", "init");
|
2022-05-06 15:56:26 +00:00
|
|
|
Cal("init", {origin:"${WEBAPP_URL}"});
|
2022-05-05 14:29:49 +00:00
|
|
|
`;
|
|
|
|
}
|
|
|
|
|
2022-06-09 05:05:18 +00:00
|
|
|
const ThemeSelectControl = ({ children, ...props }: ControlProps<{ value: Theme; label: string }, false>) => {
|
2022-05-05 14:29:49 +00:00
|
|
|
return (
|
|
|
|
<components.Control {...props}>
|
2023-05-02 18:57:01 +00:00
|
|
|
<Sun className="text-subtle mr-2 h-4 w-4" />
|
2022-05-05 14:29:49 +00:00
|
|
|
{children}
|
|
|
|
</components.Control>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const ChooseEmbedTypesDialogContent = () => {
|
|
|
|
const { t } = useLocale();
|
|
|
|
const router = useRouter();
|
2023-06-14 09:22:44 +00:00
|
|
|
|
2022-05-05 14:29:49 +00:00
|
|
|
return (
|
2023-03-20 07:52:54 +00:00
|
|
|
<DialogContent className="rounded-lg p-10" type="creation" size="lg">
|
|
|
|
<div className="mb-2">
|
2023-04-05 18:14:46 +00:00
|
|
|
<h3 className="font-cal text-emphasis mb-2 text-2xl font-bold leading-none" id="modal-title">
|
2022-11-30 21:52:56 +00:00
|
|
|
{t("how_you_want_add_cal_site", { appName: APP_NAME })}
|
2022-05-05 14:29:49 +00:00
|
|
|
</h3>
|
|
|
|
<div>
|
2023-04-05 18:14:46 +00:00
|
|
|
<p className="text-subtle text-sm">{t("choose_ways_put_cal_site", { appName: APP_NAME })}</p>
|
2022-05-05 14:29:49 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
2023-04-05 18:14:46 +00:00
|
|
|
<div className="items-start space-y-2 md:flex md:space-y-0">
|
2023-05-02 18:57:01 +00:00
|
|
|
{embeds(t).map((embed, index) => (
|
2022-05-05 14:29:49 +00:00
|
|
|
<button
|
2023-07-24 10:09:03 +00:00
|
|
|
className="hover:bg-subtle bg-muted w-full self-stretch rounded-md border border-transparent p-6 text-left hover:rounded-md ltr:mr-4 ltr:last:mr-0 rtl:ml-4 rtl:last:ml-0 lg:w-1/3"
|
2022-05-05 14:29:49 +00:00
|
|
|
key={index}
|
|
|
|
data-testid={embed.type}
|
|
|
|
onClick={() => {
|
2022-09-19 09:47:46 +00:00
|
|
|
goto(router, {
|
|
|
|
embedType: embed.type,
|
|
|
|
});
|
2022-05-05 14:29:49 +00:00
|
|
|
}}>
|
2023-04-05 18:14:46 +00:00
|
|
|
<div className="bg-default order-none box-border flex-none rounded-md border border-solid dark:bg-transparent dark:invert">
|
2022-05-05 14:29:49 +00:00
|
|
|
{embed.illustration}
|
|
|
|
</div>
|
2023-04-05 18:14:46 +00:00
|
|
|
<div className="text-emphasis mt-4 font-semibold">{embed.title}</div>
|
|
|
|
<p className="text-subtle mt-2 text-sm">{embed.subtitle}</p>
|
2022-05-05 14:29:49 +00:00
|
|
|
</button>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</DialogContent>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-07-24 10:09:03 +00:00
|
|
|
const EmailEmbed = ({ eventType, username }: { eventType?: EventType; username: string }) => {
|
|
|
|
const { t, i18n } = useLocale();
|
|
|
|
|
|
|
|
const [timezone] = useTimePreferences((state) => [state.timezone]);
|
|
|
|
|
|
|
|
const [selectTime, setSelectTime] = useState(false);
|
|
|
|
|
|
|
|
useInitializeBookerStore({
|
|
|
|
username,
|
|
|
|
eventSlug: eventType?.slug ?? "",
|
|
|
|
eventId: eventType?.id,
|
|
|
|
layout: BookerLayouts.MONTH_VIEW,
|
|
|
|
});
|
|
|
|
|
|
|
|
const [month, selectedDate, selectedDatesAndTimes] = useBookerStore(
|
|
|
|
(state) => [state.month, state.selectedDate, state.selectedDatesAndTimes],
|
|
|
|
shallow
|
|
|
|
);
|
|
|
|
const [setSelectedDate, setMonth, setSelectedDatesAndTimes, setSelectedTimeslot] = useBookerStore(
|
|
|
|
(state) => [
|
|
|
|
state.setSelectedDate,
|
|
|
|
state.setMonth,
|
|
|
|
state.setSelectedDatesAndTimes,
|
|
|
|
state.setSelectedTimeslot,
|
|
|
|
],
|
|
|
|
shallow
|
|
|
|
);
|
|
|
|
const event = useEvent();
|
|
|
|
const schedule = useScheduleForEvent();
|
|
|
|
const nonEmptyScheduleDays = useNonEmptyScheduleDays(schedule?.data?.slots);
|
|
|
|
|
|
|
|
const onTimeSelect = (time: string) => {
|
|
|
|
if (!eventType) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (selectedDatesAndTimes && selectedDatesAndTimes[eventType.slug]) {
|
|
|
|
const selectedDatesAndTimesForEvent = selectedDatesAndTimes[eventType.slug];
|
|
|
|
const selectedSlots = selectedDatesAndTimesForEvent[selectedDate as string] ?? [];
|
|
|
|
if (selectedSlots?.includes(time)) {
|
|
|
|
// Checks whether a user has removed all their timeSlots and thus removes it from the selectedDatesAndTimesForEvent state
|
|
|
|
if (selectedSlots?.length > 1) {
|
|
|
|
const updatedDatesAndTimes = {
|
|
|
|
...selectedDatesAndTimes,
|
|
|
|
[eventType.slug]: {
|
|
|
|
...selectedDatesAndTimesForEvent,
|
|
|
|
[selectedDate as string]: selectedSlots?.filter((slot: string) => slot !== time),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
setSelectedDatesAndTimes(updatedDatesAndTimes);
|
|
|
|
} else {
|
|
|
|
const updatedDatesAndTimesForEvent = { ...selectedDatesAndTimesForEvent };
|
|
|
|
delete updatedDatesAndTimesForEvent[selectedDate as string];
|
|
|
|
setSelectedTimeslot(null);
|
|
|
|
setSelectedDatesAndTimes({
|
|
|
|
...selectedDatesAndTimes,
|
|
|
|
[eventType.slug]: updatedDatesAndTimesForEvent,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const updatedDatesAndTimes = {
|
|
|
|
...selectedDatesAndTimes,
|
|
|
|
[eventType.slug]: {
|
|
|
|
...selectedDatesAndTimesForEvent,
|
|
|
|
[selectedDate as string]: [...selectedSlots, time],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
setSelectedDatesAndTimes(updatedDatesAndTimes);
|
|
|
|
} else if (!selectedDatesAndTimes) {
|
|
|
|
setSelectedDatesAndTimes({ [eventType.slug]: { [selectedDate as string]: [time] } });
|
|
|
|
} else {
|
|
|
|
setSelectedDatesAndTimes({
|
|
|
|
...selectedDatesAndTimes,
|
|
|
|
[eventType.slug]: { [selectedDate as string]: [time] },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
setSelectedTimeslot(time);
|
|
|
|
};
|
|
|
|
|
|
|
|
const slots = useSlotsForDate(selectedDate, schedule?.data?.slots);
|
|
|
|
|
|
|
|
if (!eventType) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="flex flex-col">
|
|
|
|
<div className="mb-[9px] font-medium">
|
|
|
|
<Collapsible open>
|
|
|
|
<CollapsibleContent>
|
|
|
|
<div className="text-default text-sm">{t("select_date")}</div>
|
|
|
|
<DatePicker
|
|
|
|
isLoading={schedule.isLoading}
|
|
|
|
onChange={(date: Dayjs) => {
|
|
|
|
setSelectedDate(date.format("YYYY-MM-DD"));
|
|
|
|
}}
|
|
|
|
onMonthChange={(date: Dayjs) => {
|
|
|
|
setMonth(date.format("YYYY-MM"));
|
|
|
|
setSelectedDate(date.format("YYYY-MM-DD"));
|
|
|
|
}}
|
|
|
|
includedDates={nonEmptyScheduleDays}
|
|
|
|
locale={i18n.language}
|
|
|
|
browsingDate={month ? dayjs(month) : undefined}
|
|
|
|
selected={dayjs(selectedDate)}
|
|
|
|
weekStart={weekdayToWeekIndex(event?.data?.users?.[0]?.weekStart)}
|
|
|
|
eventSlug={eventType?.slug}
|
|
|
|
/>
|
|
|
|
</CollapsibleContent>
|
|
|
|
</Collapsible>
|
|
|
|
</div>
|
|
|
|
{selectedDate ? (
|
|
|
|
<div className="mt-[9px] font-medium ">
|
|
|
|
<Collapsible open>
|
|
|
|
<CollapsibleContent>
|
|
|
|
<div
|
|
|
|
className="text-default mb-[9px] flex cursor-pointer items-center justify-between text-sm"
|
|
|
|
onClick={() => setSelectTime((prev) => !prev)}>
|
|
|
|
<p>{t("select_time")}</p>{" "}
|
|
|
|
<>
|
|
|
|
{!selectedDate || !selectTime ? <ArrowDown className="w-4" /> : <ArrowUp className="w-4" />}
|
|
|
|
</>
|
|
|
|
</div>
|
|
|
|
{selectTime && selectedDate ? (
|
|
|
|
<div className="flex h-full w-full flex-row gap-4">
|
|
|
|
<AvailableTimes
|
|
|
|
className="w-full"
|
|
|
|
date={dayjs(selectedDate)}
|
|
|
|
selectedSlots={
|
|
|
|
eventType.slug &&
|
|
|
|
selectedDatesAndTimes &&
|
|
|
|
selectedDatesAndTimes[eventType.slug] &&
|
|
|
|
selectedDatesAndTimes[eventType.slug][selectedDate as string]
|
|
|
|
? selectedDatesAndTimes[eventType.slug][selectedDate as string]
|
|
|
|
: undefined
|
|
|
|
}
|
|
|
|
onTimeSelect={onTimeSelect}
|
|
|
|
slots={slots}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</CollapsibleContent>
|
|
|
|
</Collapsible>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
<div className="mb-[9px] font-medium ">
|
|
|
|
<Collapsible open>
|
|
|
|
<CollapsibleContent>
|
|
|
|
<div className="text-default mb-[9px] text-sm">{t("duration")}</div>
|
|
|
|
<TextField
|
|
|
|
disabled
|
|
|
|
label={t("duration")}
|
|
|
|
defaultValue={eventType?.length ?? 15}
|
|
|
|
addOnSuffix={<>{t("minutes")}</>}
|
|
|
|
/>
|
|
|
|
</CollapsibleContent>
|
|
|
|
</Collapsible>
|
|
|
|
</div>
|
|
|
|
<div className="mb-[9px] font-medium ">
|
|
|
|
<Collapsible open>
|
|
|
|
<CollapsibleContent>
|
|
|
|
<div className="text-default mb-[9px] text-sm">{t("timezone")}</div>
|
|
|
|
<TimezoneSelect id="timezone" value={timezone} isDisabled />
|
|
|
|
</CollapsibleContent>
|
|
|
|
</Collapsible>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const EmailEmbedPreview = ({
|
|
|
|
eventType,
|
|
|
|
emailContentRef,
|
|
|
|
username,
|
|
|
|
month,
|
|
|
|
selectedDateAndTime,
|
|
|
|
}: {
|
|
|
|
eventType: EventType;
|
|
|
|
timezone?: string;
|
|
|
|
emailContentRef: RefObject<HTMLDivElement>;
|
|
|
|
username?: string;
|
|
|
|
month?: string;
|
|
|
|
selectedDateAndTime: { [key: string]: string[] };
|
|
|
|
}) => {
|
|
|
|
const { t } = useLocale();
|
|
|
|
const [timeFormat, timezone] = useTimePreferences((state) => [state.timeFormat, state.timezone]);
|
|
|
|
if (!eventType) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<div className="flex h-full items-center justify-center border p-5 last:font-medium">
|
|
|
|
<div className="border bg-white p-4">
|
|
|
|
<div
|
|
|
|
style={{
|
|
|
|
paddingBottom: "3px",
|
|
|
|
fontSize: "13px",
|
|
|
|
color: "black",
|
|
|
|
lineHeight: "1.4",
|
|
|
|
minWidth: "30vw",
|
|
|
|
maxHeight: "50vh",
|
|
|
|
overflowY: "auto",
|
|
|
|
backgroundColor: "white",
|
|
|
|
}}
|
|
|
|
ref={emailContentRef}>
|
|
|
|
<div
|
|
|
|
style={{
|
|
|
|
fontStyle: "normal",
|
|
|
|
fontSize: "20px",
|
|
|
|
fontWeight: "bold",
|
|
|
|
lineHeight: "19px",
|
|
|
|
marginTop: "15px",
|
|
|
|
marginBottom: "15px",
|
|
|
|
}}>
|
|
|
|
<b style={{ color: "black" }}> {eventType.title}</b>
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
style={{
|
|
|
|
fontStyle: "normal",
|
|
|
|
fontWeight: "normal",
|
|
|
|
fontSize: "14px",
|
|
|
|
lineHeight: "17px",
|
|
|
|
color: "#333333",
|
|
|
|
}}>
|
|
|
|
{t("duration")}: <b style={{ color: "black" }}>{eventType.length} mins</b>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<b style={{ color: "black" }}>
|
|
|
|
<span
|
|
|
|
style={{
|
|
|
|
fontStyle: "normal",
|
|
|
|
fontWeight: "normal",
|
|
|
|
fontSize: "14px",
|
|
|
|
lineHeight: "17px",
|
|
|
|
color: "#333333",
|
|
|
|
}}>
|
|
|
|
{t("timezone")}: <b style={{ color: "black" }}>{timezone}</b>
|
|
|
|
</span>
|
|
|
|
</b>
|
|
|
|
</div>
|
|
|
|
<b style={{ color: "black" }}>
|
|
|
|
<>
|
|
|
|
{selectedDateAndTime &&
|
|
|
|
Object.keys(selectedDateAndTime)
|
|
|
|
.sort()
|
|
|
|
.map((key) => {
|
|
|
|
const date = new Date(key);
|
|
|
|
return (
|
|
|
|
<table
|
|
|
|
key={key}
|
|
|
|
style={{
|
|
|
|
marginTop: "16px",
|
|
|
|
textAlign: "left",
|
|
|
|
borderCollapse: "collapse",
|
|
|
|
borderSpacing: "0px",
|
|
|
|
}}>
|
|
|
|
<tbody>
|
|
|
|
<tr>
|
|
|
|
<td style={{ textAlign: "left", marginTop: "16px" }}>
|
|
|
|
<span
|
|
|
|
style={{
|
|
|
|
fontSize: "14px",
|
|
|
|
lineHeight: "16px",
|
|
|
|
paddingBottom: "8px",
|
|
|
|
color: "rgb(26, 26, 26)",
|
|
|
|
fontWeight: "bold",
|
|
|
|
}}>
|
|
|
|
{date.toLocaleDateString("en-US", {
|
|
|
|
weekday: "long",
|
|
|
|
month: "long",
|
|
|
|
day: "numeric",
|
|
|
|
year: "numeric",
|
|
|
|
})}
|
|
|
|
|
|
|
|
</span>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>
|
|
|
|
<table style={{ borderCollapse: "separate", borderSpacing: "0px 4px" }}>
|
|
|
|
<tbody>
|
|
|
|
<tr style={{ height: "25px" }}>
|
|
|
|
{selectedDateAndTime[key]?.length > 0 &&
|
|
|
|
selectedDateAndTime[key].map((time) => {
|
|
|
|
const bookingURL = `${CAL_URL}/${username}/${eventType.slug}?duration=${eventType.length}&date=${key}&month=${month}&slot=${time}`;
|
|
|
|
return (
|
|
|
|
<td
|
|
|
|
key={time}
|
|
|
|
style={{
|
|
|
|
padding: "0px",
|
|
|
|
width: "64px",
|
|
|
|
display: "inline-block",
|
|
|
|
marginRight: "4px",
|
|
|
|
marginBottom: "4px",
|
|
|
|
height: "24px",
|
|
|
|
border: "1px solid #111827",
|
|
|
|
borderRadius: "3px",
|
|
|
|
}}>
|
|
|
|
<table style={{ height: "21px" }}>
|
|
|
|
<tbody>
|
|
|
|
<tr style={{ height: "21px" }}>
|
|
|
|
<td style={{ width: "7px" }} />
|
|
|
|
<td
|
|
|
|
style={{
|
|
|
|
width: "50px",
|
|
|
|
textAlign: "center",
|
|
|
|
marginRight: "1px",
|
|
|
|
}}>
|
|
|
|
<a
|
|
|
|
href={bookingURL}
|
|
|
|
className="spot"
|
|
|
|
style={{
|
|
|
|
fontFamily: '"Proxima Nova", sans-serif',
|
|
|
|
textDecoration: "none",
|
|
|
|
textAlign: "center",
|
|
|
|
color: "#111827",
|
|
|
|
fontSize: "12px",
|
|
|
|
lineHeight: "16px",
|
|
|
|
}}>
|
|
|
|
<b
|
|
|
|
style={{
|
|
|
|
fontWeight: "normal",
|
|
|
|
textDecoration: "none",
|
|
|
|
}}>
|
|
|
|
{dayjs.utc(time).tz(timezone).format(timeFormat)}
|
|
|
|
|
|
|
|
</b>
|
|
|
|
</a>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</td>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
<div style={{ marginTop: "13px" }}>
|
|
|
|
<a
|
|
|
|
className="more"
|
|
|
|
href={`${CAL_URL}/${username}/${eventType.slug}`}
|
|
|
|
style={{
|
|
|
|
textDecoration: "none",
|
|
|
|
cursor: "pointer",
|
|
|
|
color: "black",
|
|
|
|
}}>
|
|
|
|
{t("see_all_available_times")}
|
|
|
|
</a>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
</b>
|
|
|
|
<div
|
|
|
|
className="w-full text-right"
|
|
|
|
style={{
|
|
|
|
borderTop: "1px solid #CCCCCC",
|
|
|
|
marginTop: "8px",
|
|
|
|
paddingTop: "8px",
|
|
|
|
}}>
|
|
|
|
<span>{t("powered_by")}</span>{" "}
|
|
|
|
<b style={{ color: "black" }}>
|
|
|
|
<span> Cal.com</span>
|
|
|
|
</b>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<b style={{ color: "black" }} />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-05-05 14:29:49 +00:00
|
|
|
const EmbedTypeCodeAndPreviewDialogContent = ({
|
|
|
|
embedType,
|
2022-08-13 11:04:57 +00:00
|
|
|
embedUrl,
|
2022-05-05 14:29:49 +00:00
|
|
|
}: {
|
|
|
|
embedType: EmbedType;
|
2022-08-13 11:04:57 +00:00
|
|
|
embedUrl: string;
|
2022-05-05 14:29:49 +00:00
|
|
|
}) => {
|
|
|
|
const { t } = useLocale();
|
|
|
|
const router = useRouter();
|
|
|
|
const iframeRef = useRef<HTMLIFrameElement>(null);
|
2023-03-09 18:57:59 +00:00
|
|
|
const dialogContentRef = useRef<HTMLDivElement>(null);
|
2023-06-14 09:22:44 +00:00
|
|
|
const flags = useFlagMap();
|
|
|
|
const isBookerLayoutsEnabled = flags["booker-layouts"] === true;
|
2023-07-24 10:09:03 +00:00
|
|
|
const emailContentRef = useRef<HTMLDivElement>(null);
|
|
|
|
const { data } = useSession();
|
|
|
|
const [month, selectedDatesAndTimes] = useBookerStore(
|
|
|
|
(state) => [state.month, state.selectedDatesAndTimes],
|
|
|
|
shallow
|
|
|
|
);
|
|
|
|
const eventId = getQueryParam("eventId");
|
|
|
|
const calLink = decodeURIComponent(embedUrl);
|
|
|
|
const { data: eventTypeData } = trpc.viewer.eventTypes.get.useQuery(
|
|
|
|
{ id: parseInt(eventId as string) },
|
|
|
|
{ enabled: !!eventId && embedType === "email", refetchOnWindowFocus: false }
|
|
|
|
);
|
2023-03-09 18:57:59 +00:00
|
|
|
|
2022-09-18 23:14:57 +00:00
|
|
|
const s = (href: string) => {
|
2022-09-19 09:47:46 +00:00
|
|
|
const searchParams = new URLSearchParams(router.asPath.split("?")[1] || "");
|
2022-09-18 23:14:57 +00:00
|
|
|
const [a, b] = href.split("=");
|
|
|
|
searchParams.set(a, b);
|
|
|
|
return `${router.asPath.split("?")[0]}?${searchParams.toString()}`;
|
|
|
|
};
|
|
|
|
const parsedTabs = tabs.map((t) => ({ ...t, href: s(t.href) }));
|
2023-02-16 22:39:57 +00:00
|
|
|
const embedCodeRefs: Record<(typeof tabs)[0]["name"], RefObject<HTMLTextAreaElement>> = {};
|
2022-06-16 11:55:49 +00:00
|
|
|
tabs
|
|
|
|
.filter((tab) => tab.type === "code")
|
|
|
|
.forEach((codeTab) => {
|
|
|
|
embedCodeRefs[codeTab.name] = createRef();
|
|
|
|
});
|
|
|
|
|
|
|
|
const refOfEmbedCodesRefs = useRef(embedCodeRefs);
|
2023-05-02 18:57:01 +00:00
|
|
|
const embed = embeds(t).find((embed) => embed.type === embedType);
|
2022-05-05 14:29:49 +00:00
|
|
|
|
|
|
|
const [isEmbedCustomizationOpen, setIsEmbedCustomizationOpen] = useState(true);
|
|
|
|
const [isBookingCustomizationOpen, setIsBookingCustomizationOpen] = useState(true);
|
2023-06-14 09:22:44 +00:00
|
|
|
const [previewState, setPreviewState] = useState<PreviewState>({
|
2022-05-05 14:29:49 +00:00
|
|
|
inline: {
|
|
|
|
width: "100%",
|
|
|
|
height: "100%",
|
|
|
|
},
|
2022-06-09 05:05:18 +00:00
|
|
|
theme: Theme.auto,
|
2023-06-26 15:24:19 +00:00
|
|
|
layout: BookerLayouts.MONTH_VIEW,
|
2022-05-05 14:29:49 +00:00
|
|
|
floatingPopup: {},
|
|
|
|
elementClick: {},
|
2022-12-13 07:23:26 +00:00
|
|
|
hideEventTypeDetails: false,
|
2022-05-05 14:29:49 +00:00
|
|
|
palette: {
|
|
|
|
brandColor: "#000000",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
const close = () => {
|
2022-09-19 09:47:46 +00:00
|
|
|
removeQueryParams(router, ["dialog", ...queryParamsForDialog]);
|
|
|
|
};
|
2022-05-05 14:29:49 +00:00
|
|
|
|
2022-09-19 09:47:46 +00:00
|
|
|
// Use embed-code as default tab
|
|
|
|
if (!router.query.embedTabName) {
|
|
|
|
goto(router, {
|
|
|
|
embedTabName: "embed-code",
|
2022-05-05 14:29:49 +00:00
|
|
|
});
|
2022-09-19 09:47:46 +00:00
|
|
|
}
|
2022-05-05 14:29:49 +00:00
|
|
|
|
2022-08-13 11:04:57 +00:00
|
|
|
if (!embed || !embedUrl) {
|
2022-05-05 14:29:49 +00:00
|
|
|
close();
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-02-16 22:39:57 +00:00
|
|
|
const addToPalette = (update: (typeof previewState)["palette"]) => {
|
2022-05-05 14:29:49 +00:00
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
|
|
|
palette: {
|
|
|
|
...previewState.palette,
|
|
|
|
...update,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2022-05-17 20:43:27 +00:00
|
|
|
const previewInstruction = (instruction: { name: string; arg: unknown }) => {
|
2022-05-05 14:29:49 +00:00
|
|
|
iframeRef.current?.contentWindow?.postMessage(
|
|
|
|
{
|
|
|
|
mode: "cal:preview",
|
|
|
|
type: "instruction",
|
|
|
|
instruction,
|
|
|
|
},
|
|
|
|
"*"
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const inlineEmbedDimensionUpdate = ({ width, height }: { width: string; height: string }) => {
|
|
|
|
iframeRef.current?.contentWindow?.postMessage(
|
|
|
|
{
|
|
|
|
mode: "cal:preview",
|
|
|
|
type: "inlineEmbedDimensionUpdate",
|
|
|
|
data: {
|
|
|
|
width: getDimension(width),
|
|
|
|
height: getDimension(height),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
"*"
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
previewInstruction({
|
|
|
|
name: "ui",
|
|
|
|
arg: {
|
|
|
|
theme: previewState.theme,
|
2023-06-14 09:22:44 +00:00
|
|
|
layout: previewState.layout,
|
2023-06-26 15:24:19 +00:00
|
|
|
hideEventTypeDetails: previewState.hideEventTypeDetails,
|
2022-05-05 14:29:49 +00:00
|
|
|
styles: {
|
|
|
|
branding: {
|
|
|
|
...previewState.palette,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2023-07-24 10:09:03 +00:00
|
|
|
const handleCopyEmailText = () => {
|
|
|
|
const contentElement = emailContentRef.current;
|
|
|
|
if (contentElement !== null) {
|
|
|
|
const range = document.createRange();
|
|
|
|
range.selectNode(contentElement);
|
|
|
|
const selection = window.getSelection();
|
|
|
|
if (selection) {
|
|
|
|
selection.removeAllRanges();
|
|
|
|
selection.addRange(range);
|
|
|
|
document.execCommand("copy");
|
|
|
|
selection.removeAllRanges();
|
|
|
|
}
|
|
|
|
|
|
|
|
showToast(t("code_copied"), "success");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-05-05 14:29:49 +00:00
|
|
|
if (embedType === "floating-popup") {
|
|
|
|
previewInstruction({
|
|
|
|
name: "floatingButton",
|
|
|
|
arg: {
|
|
|
|
attributes: {
|
|
|
|
id: "my-floating-button",
|
|
|
|
},
|
|
|
|
...previewState.floatingPopup,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (embedType === "inline") {
|
|
|
|
inlineEmbedDimensionUpdate({
|
|
|
|
width: previewState.inline.width,
|
|
|
|
height: previewState.inline.height,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const ThemeOptions = [
|
2023-04-03 05:20:44 +00:00
|
|
|
{ value: Theme.auto, label: "Auto" },
|
2022-06-09 05:05:18 +00:00
|
|
|
{ value: Theme.dark, label: "Dark Theme" },
|
|
|
|
{ value: Theme.light, label: "Light Theme" },
|
2022-05-05 14:29:49 +00:00
|
|
|
];
|
|
|
|
|
2023-06-14 09:22:44 +00:00
|
|
|
const layoutOptions = [
|
|
|
|
{ value: BookerLayouts.MONTH_VIEW, label: t("bookerlayout_month_view") },
|
|
|
|
{ value: BookerLayouts.WEEK_VIEW, label: t("bookerlayout_week_view") },
|
|
|
|
{ value: BookerLayouts.COLUMN_VIEW, label: t("bookerlayout_column_view") },
|
|
|
|
];
|
|
|
|
|
2022-05-05 14:29:49 +00:00
|
|
|
const FloatingPopupPositionOptions = [
|
|
|
|
{
|
|
|
|
value: "bottom-right",
|
2023-05-02 18:57:01 +00:00
|
|
|
label: "Bottom right",
|
2022-05-05 14:29:49 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
value: "bottom-left",
|
2023-05-02 18:57:01 +00:00
|
|
|
label: "Bottom left",
|
2022-05-05 14:29:49 +00:00
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
return (
|
2023-04-03 05:20:44 +00:00
|
|
|
<DialogContent
|
2023-07-24 10:09:03 +00:00
|
|
|
enableOverflow
|
2023-04-03 05:20:44 +00:00
|
|
|
ref={dialogContentRef}
|
|
|
|
className="rounded-lg p-0.5 sm:max-w-[80rem]"
|
|
|
|
type="creation">
|
2022-05-05 14:29:49 +00:00
|
|
|
<div className="flex">
|
2023-07-24 10:09:03 +00:00
|
|
|
<div className="bg-muted flex h-[90vh] w-1/3 flex-col overflow-y-auto p-8">
|
2023-04-03 05:20:44 +00:00
|
|
|
<h3
|
2023-04-05 18:14:46 +00:00
|
|
|
className="text-emphasis mb-2.5 flex items-center text-xl font-semibold leading-5"
|
2023-04-03 05:20:44 +00:00
|
|
|
id="modal-title">
|
2022-05-05 14:29:49 +00:00
|
|
|
<button
|
2023-04-03 05:20:44 +00:00
|
|
|
className="h-6 w-6"
|
2022-05-05 14:29:49 +00:00
|
|
|
onClick={() => {
|
2022-09-19 09:47:46 +00:00
|
|
|
removeQueryParams(router, ["embedType", "embedTabName"]);
|
2022-05-05 14:29:49 +00:00
|
|
|
}}>
|
2023-04-12 15:26:31 +00:00
|
|
|
<ArrowLeft className="mr-4 w-4" />
|
2022-05-05 14:29:49 +00:00
|
|
|
</button>
|
|
|
|
{embed.title}
|
|
|
|
</h3>
|
2023-05-02 18:57:01 +00:00
|
|
|
<h4 className="text-subtle mb-6 text-sm font-normal">{embed.subtitle}</h4>
|
2023-07-24 10:09:03 +00:00
|
|
|
{eventTypeData?.eventType && embedType === "email" ? (
|
|
|
|
<EmailEmbed eventType={eventTypeData?.eventType} username={data?.user.username as string} />
|
|
|
|
) : (
|
|
|
|
<div className="flex flex-col">
|
|
|
|
<div className={classNames("font-medium", embedType === "element-click" ? "hidden" : "")}>
|
|
|
|
<Collapsible
|
|
|
|
open={isEmbedCustomizationOpen}
|
|
|
|
onOpenChange={() => setIsEmbedCustomizationOpen((val) => !val)}>
|
|
|
|
<CollapsibleContent className="text-sm">
|
|
|
|
<div className={classNames(embedType === "inline" ? "block" : "hidden")}>
|
|
|
|
{/*TODO: Add Auto/Fixed toggle from Figma */}
|
|
|
|
<div className="text-default mb-[9px] text-sm">Window sizing</div>
|
|
|
|
<div className="justify-left mb-6 flex items-center !font-normal ">
|
|
|
|
<div className="mr-[9px]">
|
|
|
|
<TextField
|
|
|
|
labelProps={{ className: "hidden" }}
|
|
|
|
className="focus:ring-offset-0"
|
|
|
|
required
|
|
|
|
value={previewState.inline.width}
|
|
|
|
onChange={(e) => {
|
|
|
|
setPreviewState((previewState) => {
|
|
|
|
const width = e.target.value || "100%";
|
|
|
|
|
|
|
|
return {
|
|
|
|
...previewState,
|
|
|
|
inline: {
|
|
|
|
...previewState.inline,
|
|
|
|
width,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
addOnLeading={<>W</>}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
2023-04-03 05:20:44 +00:00
|
|
|
<TextField
|
|
|
|
labelProps={{ className: "hidden" }}
|
|
|
|
className="focus:ring-offset-0"
|
2023-07-24 10:09:03 +00:00
|
|
|
value={previewState.inline.height}
|
2023-04-03 05:20:44 +00:00
|
|
|
required
|
|
|
|
onChange={(e) => {
|
2023-07-24 10:09:03 +00:00
|
|
|
const height = e.target.value || "100%";
|
2023-04-03 05:20:44 +00:00
|
|
|
|
2023-07-24 10:09:03 +00:00
|
|
|
setPreviewState((previewState) => {
|
2023-04-03 05:20:44 +00:00
|
|
|
return {
|
|
|
|
...previewState,
|
|
|
|
inline: {
|
|
|
|
...previewState.inline,
|
2023-07-24 10:09:03 +00:00
|
|
|
height,
|
2023-04-03 05:20:44 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
2023-07-24 10:09:03 +00:00
|
|
|
addOnLeading={<>H</>}
|
2023-04-03 05:20:44 +00:00
|
|
|
/>
|
|
|
|
</div>
|
2023-07-24 10:09:03 +00:00
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
className={classNames(
|
|
|
|
"items-center justify-between",
|
|
|
|
embedType === "floating-popup" ? "text-emphasis" : "hidden"
|
|
|
|
)}>
|
|
|
|
<div className="mb-2 text-sm">Button text</div>
|
|
|
|
{/* Default Values should come from preview iframe */}
|
2022-11-10 13:09:19 +00:00
|
|
|
<TextField
|
|
|
|
labelProps={{ className: "hidden" }}
|
|
|
|
onChange={(e) => {
|
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
2023-07-24 10:09:03 +00:00
|
|
|
floatingPopup: {
|
|
|
|
...previewState.floatingPopup,
|
|
|
|
buttonText: e.target.value,
|
2022-11-10 13:09:19 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
2023-07-24 10:09:03 +00:00
|
|
|
defaultValue={t("book_my_cal")}
|
|
|
|
required
|
2022-11-10 13:09:19 +00:00
|
|
|
/>
|
|
|
|
</div>
|
2023-07-24 10:09:03 +00:00
|
|
|
<div
|
|
|
|
className={classNames(
|
|
|
|
"mt-4 flex items-center justify-start",
|
|
|
|
embedType === "floating-popup"
|
|
|
|
? "text-emphasis space-x-2 rtl:space-x-reverse"
|
|
|
|
: "hidden"
|
|
|
|
)}>
|
|
|
|
<Switch
|
|
|
|
defaultChecked={true}
|
|
|
|
onCheckedChange={(checked) => {
|
2023-05-02 18:57:01 +00:00
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
2023-07-24 10:09:03 +00:00
|
|
|
floatingPopup: {
|
|
|
|
...previewState.floatingPopup,
|
|
|
|
hideButtonIcon: !checked,
|
|
|
|
},
|
2023-05-02 18:57:01 +00:00
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
/>
|
2023-07-24 10:09:03 +00:00
|
|
|
<div className="text-default my-2 text-sm">Display calendar icon</div>
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
className={classNames(
|
|
|
|
"mt-4 items-center justify-between",
|
|
|
|
embedType === "floating-popup" ? "text-emphasis" : "hidden"
|
|
|
|
)}>
|
|
|
|
<div className="mb-2">Position of button</div>
|
|
|
|
<Select
|
|
|
|
onChange={(position) => {
|
2022-12-13 07:23:26 +00:00
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
2023-07-24 10:09:03 +00:00
|
|
|
floatingPopup: {
|
|
|
|
...previewState.floatingPopup,
|
|
|
|
buttonPosition: position?.value,
|
|
|
|
},
|
2022-12-13 07:23:26 +00:00
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
2023-07-24 10:09:03 +00:00
|
|
|
defaultValue={FloatingPopupPositionOptions[0]}
|
|
|
|
options={FloatingPopupPositionOptions}
|
2022-12-13 07:23:26 +00:00
|
|
|
/>
|
|
|
|
</div>
|
2023-07-24 10:09:03 +00:00
|
|
|
<div className="mt-3 flex flex-col xl:flex-row xl:justify-between">
|
|
|
|
<div className={classNames("mt-4", embedType === "floating-popup" ? "" : "hidden")}>
|
|
|
|
<div className="whitespace-nowrap">Button color</div>
|
|
|
|
<div className="mt-2 w-40 xl:mt-0 xl:w-full">
|
2022-11-10 13:09:19 +00:00
|
|
|
<ColorPicker
|
2023-07-24 10:09:03 +00:00
|
|
|
className="w-[130px]"
|
2023-03-09 18:57:59 +00:00
|
|
|
popoverAlign="start"
|
|
|
|
container={dialogContentRef?.current ?? undefined}
|
2022-11-10 13:09:19 +00:00
|
|
|
defaultValue="#000000"
|
|
|
|
onChange={(color) => {
|
2023-07-24 10:09:03 +00:00
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
|
|
|
floatingPopup: {
|
|
|
|
...previewState.floatingPopup,
|
|
|
|
buttonColor: color,
|
|
|
|
},
|
|
|
|
};
|
2022-11-10 13:09:19 +00:00
|
|
|
});
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
2023-07-24 10:09:03 +00:00
|
|
|
</div>
|
|
|
|
<div className={classNames("mt-4", embedType === "floating-popup" ? "" : "hidden")}>
|
|
|
|
<div className="whitespace-nowrap">Text color</div>
|
|
|
|
<div className="mb-6 mt-2 w-40 xl:mt-0 xl:w-full">
|
|
|
|
<ColorPicker
|
|
|
|
className="w-[130px]"
|
|
|
|
popoverAlign="start"
|
|
|
|
container={dialogContentRef?.current ?? undefined}
|
|
|
|
defaultValue="#000000"
|
|
|
|
onChange={(color) => {
|
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
|
|
|
floatingPopup: {
|
|
|
|
...previewState.floatingPopup,
|
|
|
|
buttonTextColor: color,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</CollapsibleContent>
|
|
|
|
</Collapsible>
|
|
|
|
</div>
|
|
|
|
<div className="font-medium">
|
|
|
|
<Collapsible
|
|
|
|
open={isBookingCustomizationOpen}
|
|
|
|
onOpenChange={() => setIsBookingCustomizationOpen((val) => !val)}>
|
|
|
|
<CollapsibleContent>
|
|
|
|
<div className="text-sm">
|
2023-06-14 09:22:44 +00:00
|
|
|
<Label className="mb-6">
|
2023-07-24 10:09:03 +00:00
|
|
|
<div className="mb-2">Theme</div>
|
2023-06-14 09:22:44 +00:00
|
|
|
<Select
|
|
|
|
className="w-full"
|
2023-07-24 10:09:03 +00:00
|
|
|
defaultValue={ThemeOptions[0]}
|
|
|
|
components={{
|
|
|
|
Control: ThemeSelectControl,
|
|
|
|
IndicatorSeparator: () => null,
|
|
|
|
}}
|
2023-06-14 09:22:44 +00:00
|
|
|
onChange={(option) => {
|
|
|
|
if (!option) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
2023-07-24 10:09:03 +00:00
|
|
|
theme: option.value,
|
2023-06-14 09:22:44 +00:00
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
2023-07-24 10:09:03 +00:00
|
|
|
options={ThemeOptions}
|
2023-06-14 09:22:44 +00:00
|
|
|
/>
|
|
|
|
</Label>
|
2023-07-24 10:09:03 +00:00
|
|
|
<div className="mb-6 flex items-center justify-start space-x-2 rtl:space-x-reverse">
|
|
|
|
<Switch
|
|
|
|
checked={previewState.hideEventTypeDetails}
|
|
|
|
onCheckedChange={(checked) => {
|
|
|
|
setPreviewState((previewState) => {
|
|
|
|
return {
|
|
|
|
...previewState,
|
|
|
|
hideEventTypeDetails: checked,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<div className="text-default text-sm">{t("hide_eventtype_details")}</div>
|
|
|
|
</div>
|
|
|
|
{[
|
|
|
|
{ name: "brandColor", title: "Brand Color" },
|
|
|
|
// { name: "lightColor", title: "Light Color" },
|
|
|
|
// { name: "lighterColor", title: "Lighter Color" },
|
|
|
|
// { name: "lightestColor", title: "Lightest Color" },
|
|
|
|
// { name: "highlightColor", title: "Highlight Color" },
|
|
|
|
// { name: "medianColor", title: "Median Color" },
|
|
|
|
].map((palette) => (
|
|
|
|
<Label key={palette.name} className="mb-6">
|
|
|
|
<div className="mb-2">{palette.title}</div>
|
|
|
|
<div className="w-full">
|
|
|
|
<ColorPicker
|
|
|
|
popoverAlign="start"
|
|
|
|
container={dialogContentRef?.current ?? undefined}
|
|
|
|
defaultValue="#000000"
|
|
|
|
onChange={(color) => {
|
|
|
|
addToPalette({
|
|
|
|
[palette.name as keyof (typeof previewState)["palette"]]: color,
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</Label>
|
|
|
|
))}
|
|
|
|
{isBookerLayoutsEnabled && (
|
|
|
|
<Label className="mb-6">
|
|
|
|
<div className="mb-2">{t("layout")}</div>
|
|
|
|
<Select
|
|
|
|
className="w-full"
|
|
|
|
defaultValue={layoutOptions[0]}
|
|
|
|
onChange={(option) => {
|
|
|
|
if (!option) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setPreviewState((previewState) => {
|
|
|
|
const config = {
|
|
|
|
...(previewState.floatingPopup.config ?? {}),
|
|
|
|
layout: option.value,
|
|
|
|
};
|
|
|
|
return {
|
|
|
|
...previewState,
|
|
|
|
floatingPopup: {
|
|
|
|
config,
|
|
|
|
},
|
|
|
|
layout: option.value,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
options={layoutOptions}
|
|
|
|
/>
|
|
|
|
</Label>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</CollapsibleContent>
|
|
|
|
</Collapsible>
|
|
|
|
</div>
|
2022-11-10 13:09:19 +00:00
|
|
|
</div>
|
2023-07-24 10:09:03 +00:00
|
|
|
)}
|
2022-05-05 14:29:49 +00:00
|
|
|
</div>
|
2023-07-24 06:17:25 +00:00
|
|
|
<div className="flex w-2/3 flex-col px-8 pt-8">
|
2023-07-24 10:09:03 +00:00
|
|
|
<HorizontalTabs
|
|
|
|
data-testid="embed-tabs"
|
|
|
|
tabs={embedType === "email" ? parsedTabs.filter((tab) => tab.name === "Preview") : parsedTabs}
|
|
|
|
linkShallow
|
|
|
|
/>
|
2022-06-09 05:05:18 +00:00
|
|
|
{tabs.map((tab) => {
|
2023-07-24 10:09:03 +00:00
|
|
|
if (embedType !== "email") {
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
key={tab.href}
|
|
|
|
className={classNames(
|
|
|
|
router.query.embedTabName === tab.href.split("=")[1]
|
|
|
|
? "flex flex-grow flex-col"
|
|
|
|
: "hidden"
|
|
|
|
)}>
|
|
|
|
<div className="flex h-[55vh] flex-grow flex-col">
|
|
|
|
{tab.type === "code" ? (
|
|
|
|
<tab.Component
|
|
|
|
embedType={embedType}
|
|
|
|
calLink={calLink}
|
|
|
|
previewState={previewState}
|
|
|
|
ref={refOfEmbedCodesRefs.current[tab.name]}
|
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
<tab.Component
|
|
|
|
embedType={embedType}
|
|
|
|
calLink={calLink}
|
|
|
|
previewState={previewState}
|
|
|
|
ref={iframeRef}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<div className={router.query.embedTabName == "embed-preview" ? "mt-2 block" : "hidden"} />
|
|
|
|
<DialogFooter className="mt-10 flex-row-reverse gap-x-2" showDivider>
|
|
|
|
<DialogClose />
|
|
|
|
{tab.type === "code" ? (
|
|
|
|
<Button
|
|
|
|
type="submit"
|
|
|
|
onClick={() => {
|
|
|
|
const currentTabCodeEl = refOfEmbedCodesRefs.current[tab.name].current;
|
|
|
|
if (!currentTabCodeEl) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
navigator.clipboard.writeText(currentTabCodeEl.value);
|
|
|
|
showToast(t("code_copied"), "success");
|
|
|
|
}}>
|
|
|
|
{t("copy_code")}
|
|
|
|
</Button>
|
|
|
|
) : null}
|
|
|
|
</DialogFooter>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (embedType === "email" && (tab.name !== "Preview" || !eventTypeData?.eventType)) return;
|
|
|
|
|
2022-06-09 05:05:18 +00:00
|
|
|
return (
|
2023-07-24 10:09:03 +00:00
|
|
|
<div key={tab.href} className={classNames("flex flex-grow flex-col")}>
|
2022-09-15 05:34:11 +00:00
|
|
|
<div className="flex h-[55vh] flex-grow flex-col">
|
2023-07-24 10:09:03 +00:00
|
|
|
<EmailEmbedPreview
|
|
|
|
eventType={eventTypeData?.eventType}
|
|
|
|
emailContentRef={emailContentRef}
|
|
|
|
username={data?.user.username as string}
|
|
|
|
month={month as string}
|
|
|
|
selectedDateAndTime={
|
|
|
|
selectedDatesAndTimes
|
|
|
|
? selectedDatesAndTimes[eventTypeData?.eventType.slug as string]
|
|
|
|
: {}
|
|
|
|
}
|
|
|
|
/>
|
2022-06-09 05:05:18 +00:00
|
|
|
</div>
|
2023-06-29 07:47:16 +00:00
|
|
|
<div className={router.query.embedTabName == "embed-preview" ? "mt-2 block" : "hidden"} />
|
2023-07-24 06:17:25 +00:00
|
|
|
<DialogFooter className="mt-10 flex-row-reverse gap-x-2" showDivider>
|
2023-06-29 07:47:16 +00:00
|
|
|
<DialogClose />
|
2023-07-24 10:09:03 +00:00
|
|
|
<Button
|
|
|
|
onClick={() => {
|
|
|
|
handleCopyEmailText();
|
|
|
|
}}>
|
|
|
|
{embedType === "email" ? t("copy") : t("copy_code")}
|
|
|
|
</Button>
|
2023-06-29 07:47:16 +00:00
|
|
|
</DialogFooter>
|
2022-06-09 05:05:18 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
})}
|
2022-05-05 14:29:49 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</DialogContent>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const EmbedDialog = () => {
|
|
|
|
const router = useRouter();
|
2022-08-13 11:04:57 +00:00
|
|
|
const embedUrl: string = router.query.embedUrl as string;
|
2022-05-05 14:29:49 +00:00
|
|
|
return (
|
2023-06-06 23:52:13 +00:00
|
|
|
<Dialog name="embed" clearQueryParamsOnClose={queryParamsForDialog}>
|
2022-05-05 14:29:49 +00:00
|
|
|
{!router.query.embedType ? (
|
|
|
|
<ChooseEmbedTypesDialogContent />
|
|
|
|
) : (
|
|
|
|
<EmbedTypeCodeAndPreviewDialogContent
|
|
|
|
embedType={router.query.embedType as EmbedType}
|
2022-08-13 11:04:57 +00:00
|
|
|
embedUrl={embedUrl}
|
2022-05-05 14:29:49 +00:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</Dialog>
|
|
|
|
);
|
|
|
|
};
|
2022-08-13 11:04:57 +00:00
|
|
|
type EmbedButtonProps<T> = {
|
|
|
|
embedUrl: string;
|
|
|
|
children?: React.ReactNode;
|
2022-09-02 19:00:41 +00:00
|
|
|
className?: string;
|
2022-08-13 11:04:57 +00:00
|
|
|
as?: T;
|
2023-07-24 10:09:03 +00:00
|
|
|
eventId?: number;
|
2022-08-13 11:04:57 +00:00
|
|
|
};
|
2022-05-05 14:29:49 +00:00
|
|
|
|
2022-08-13 11:04:57 +00:00
|
|
|
export const EmbedButton = <T extends React.ElementType>({
|
|
|
|
embedUrl,
|
2022-07-27 02:24:00 +00:00
|
|
|
children,
|
2022-05-05 14:29:49 +00:00
|
|
|
className = "",
|
2022-08-13 11:04:57 +00:00
|
|
|
as,
|
2023-07-24 10:09:03 +00:00
|
|
|
eventId,
|
2022-05-05 14:29:49 +00:00
|
|
|
...props
|
2022-08-13 11:04:57 +00:00
|
|
|
}: EmbedButtonProps<T> & React.ComponentPropsWithoutRef<T>) => {
|
2022-05-05 14:29:49 +00:00
|
|
|
const router = useRouter();
|
2023-01-20 14:19:12 +00:00
|
|
|
className = classNames("hidden lg:inline-flex", className);
|
2022-05-05 14:29:49 +00:00
|
|
|
const openEmbedModal = () => {
|
2022-09-19 09:47:46 +00:00
|
|
|
goto(router, {
|
|
|
|
dialog: "embed",
|
2023-07-24 10:09:03 +00:00
|
|
|
eventId: eventId ? eventId.toString() : "",
|
2022-09-19 09:47:46 +00:00
|
|
|
embedUrl,
|
|
|
|
});
|
2022-05-05 14:29:49 +00:00
|
|
|
};
|
2022-08-13 11:04:57 +00:00
|
|
|
const Component = as ?? Button;
|
2022-05-05 14:29:49 +00:00
|
|
|
|
|
|
|
return (
|
2022-08-13 11:04:57 +00:00
|
|
|
<Component
|
2022-05-05 14:29:49 +00:00
|
|
|
{...props}
|
2022-08-13 11:04:57 +00:00
|
|
|
className={className}
|
|
|
|
data-test-embed-url={embedUrl}
|
|
|
|
data-testid="embed"
|
2022-09-02 19:00:41 +00:00
|
|
|
type="button"
|
|
|
|
onClick={() => {
|
|
|
|
openEmbedModal();
|
|
|
|
}}>
|
2022-07-27 02:24:00 +00:00
|
|
|
{children}
|
2022-08-13 11:04:57 +00:00
|
|
|
</Component>
|
2022-05-05 14:29:49 +00:00
|
|
|
);
|
|
|
|
};
|