Embeds
This folder contains all the various flavours of embeds.
core contains the core library written in vanilla JS that manages the embed.
snippet contains the Vanilla JS Code Snippet that can be installed on any website and would automatically fetch the core library.
Please see the respective folder READMEs for details on them.
Publishing to NPM. It will soon be automated using changesets github action
To publish the packages. Following steps should be followed. All commands are to be run at the root.
yarn changeset-> Creates changelog files and adds summary to changelog. Select embed packages only here.yarn changeset version-> Bumps the versions as required- Get the PR reviewed and merged
yarn publish-embed-> Releases all packages. We can't useyarn changeset publishbecause it doesn't support workspace: prefix removal yet. See https://github.com/changesets/changesets/issues/432#issuecomment-1016365428
Skeleton Loader
Skeleton loader is shown for supported page types. For all other page types, default non-skeleton loader is shown. Status:
- Layout
- Responsive
- Mobile Layout
- month_view Layout
- week_view Layout
- column_view Layout
- Theming
- Dark and Light theme NOTE: If user has preference for theme configured within app, that has to be communicated clearly in the embed too for skeleton to work
- Change in system theme should reflect without page refresh
- Page Types supported
- user.event.booking.slots
- team.event.booking.slots
- [Partially supported] user.event.booking.form - Shows skeleton but of the slots page
- [Partially supported] team.event.booking.form - Shows skeleton but of the slots page
- user.profile
- team.profile
How Routing Prerendering works
- Use API to prerender a booking link for "modal"
- When CTA is clicked by user, we check if there is a "prerendered"/"being prerendered" modal for this namespace.
- If yes, we open up the modal showing the skeleton loader and send the POST request to /api/router endpoint
- When we get the response from the endpoint, we pass on all the query params to the already rendered/being rendered iframe and embed-iframe updates the URL of the iframe to have the new query params through history.replaceState(i.e. without reloading the page)
Prerendering vs Preloading
- Preloading loads the calLink in iframe with the sole purpose of preloading the static assets, so that when the embed actually opens, it uses the static assets from browser cache.
- Prerendering means continuing over the preloaded iframe, so that the user books on the prerendered iframe only. So, it is much more complex than preloading and gives much more benefits in terms of performance.
Note: API wise
prerenderdelegates its task topreloadAPI which then identifies whether to preload or prerender.
Modalbox re-opening performance optimization
- ModalBox supports reusing the same cal-modal-box element and thus same iframe and thus providing a lightning fast experience when the same modal is opened multiple times [This feature is currently disabled in code because of stale booking page UI issues]
Embed Core Architecture and Features
Architecture Overview
Initialization and Bootstrap Process
The embed system initializes through a multi-step process:
- The embed script is loaded on the parent page
- It creates a global
Calobject that acts as the entry point - The system initializes necessary custom elements (
cal-modal-box,cal-floating-button,cal-inline) - A namespace-based action manager is created for event handling
Parent-Iframe Communication System
Communication between the parent page and the embedded iframe uses a message-based system:
// Parent to Iframe communication example
interface InterfaceWithParent {
ui: (config: UiConfig) => void;
connect: (config: PrefillAndIframeAttrsConfig) => void;
}
// Event data structure
type EventData<T> = {
type: string;
namespace: string;
fullType: string;
data: EventDataMap[T];
};
The system uses namespaced events to ensure multiple embeds on the same page don't interfere with each other.
Instruction Queue System
Commands are queued before the iframe is ready:
type Instruction = SingleInstruction | SingleInstruction[];
type InstructionQueue = Instruction[];
// Commands are queued if iframe isn't ready
if (!this.iframeReady) {
this.iframeDoQueue.push(doInIframeArg);
return;
}
Embedding Methods
Inline Embedding
Embeds the calendar directly within the page flow:
Cal.inline({
elementOrSelector: "#my-cal-inline",
calLink: "organization/event-type"
});
Modal Embedding
Creates a modal dialog with the calendar:
Cal.modal({
calLink: "organization/event-type",
config: {
// Optional configuration
useSlotsViewOnSmallScreen: "true"
}
});
FloatingButton Embedding
Adds a floating action button that opens the calendar in a modal. It uses modal embedding under the hood.
Cal.floatingButton({
calLink: "organization/event-type",
buttonText: "Book meeting",
buttonPosition: "bottom-right"
});
Configuration and Customization
Prefill System
Allows pre-filling form fields:
Cal.inline({
calLink: "organization/event-type",
config: {
name: "John Doe",
email: "john@example.com",
notes: "Initial discussion",
useSlotsViewOnSmallScreen: "true"
}
});
Query Parameter Handling
The system allows automatically forwarding query params to the iframe, by setting. This code must be present right after the embed snippet is added to the page.
Cal.config = Cal.config || {};
Cal.config.forwardQueryParams=true
Enabling Logging
You can enable logging for the embed by adding the cal.embed.logging=1 query parameter to the URL of the page where the embed is placed. This is useful for debugging issues with the embed. It will log all important things in parent as well as in the iframe(i.e. child)
For example, if your page is https.example.com/contact, you can enable logging by visiting https.example.com/contact?cal.embed.logging=1.
Advanced Features
Routing Prerendering System
The prerendering system optimizes the initial load:
Cal("prerender", {
calLink,
type: "modal"
});
Key aspects:
- Creates a hidden iframe
- Loads the booking page but doesn't send the slots availability request
- Tries to reuse whenever it makes sense and do a fresh load otherwise
Modal's Iframe Reuse and Reload Conditions. There could be four situations:
- Show modal as is. No connect and no requests happen - Corresponding action is "noAction"
- Connect but don't fetch the slots- Corresponding action is "connect-no-slots-fetch"
- Connect and fetch the slots - Corresponding action is "connect"
- Do a fresh load in iframe. Corresponding action is "fullReload"
-
Show modal as is:
- Modal is not in a failed state
- config, params are same as the last time
- No time threshold violations
-
Connect but don't fetch the slots:
- Only embed
configchanges (handled via "connect" flow) - Query query params changes (handled via "connect" flow)
- Crossed slots stale time threshold (EMBED_MODAL_IFRAME_SLOT_STALE_TIME)
- Only embed
-
Connect and fetch the slots:
- Slots are stale
- Crossed slots stale time threshold (EMBED_MODAL_IFRAME_SLOT_STALE_TIME)
-
Do a fresh load in iframe:
- Different path being loaded(i.e. /pro vs /free)
- Modal is in a failed state
- Time since last render exceeds EMBED_MODAL_IFRAME_FORCE_RELOAD_THRESHOLD_MS
Prerendering headless router
Without namespace: Prerender when there are high chances of user clicking the CTA.
Cal('prerender', {
calLink: "router?formId=123&ONLY_THOSE_FIELDS_THAT_ARE_REQUIRED_BY_ROUTING_RULES_SHOULD_BE_PRESENT_HERE",
// Prerender right now works only with "modal", so 'element click' embed is able to reuse this prerendered iframe
type: "modal",
// Shows skeleton loader for a Team Event's booking slots page
pageType: "team.event.booking.slots"
});
Using the prerendered iframe with a CTA:
<button data-cal-link="router?formId=123&ALL_FIELDS_HERE">Demo</button>
With namespace: Prerender when there are high changes of user clicking the CTA.
Cal.ns.myNamespace('prerender', {
calLink: "router?formId=123&ONLY_THOSE_FIELDS_THAT_ARE_REQUIRED_BY_ROUTING_RULES_SHOULD_BE_PRESENT_HERE",
// Prerender right now works only with "modal", so 'element click' embed is able to reuse this prerendered iframe
type: "modal"
// Shows skeleton loader for a Team Event's booking slots page
pageType: "team.event.booking.slots"
});
Using the prerendered iframe with a CTA:
<button data-cal-namespace="myNamespace" data-cal-link="router?formId=123&ALL_FIELDS_HERE">Demo</button>