Integration with DHTMLX Scheduler
This guide will show how to integrate the DHTMLX Booking widget with DHTMLX Scheduler.
Main concepts
The integration primarily focuses on converting the Scheduler data into Booking slots.
- Scheduler events vs. Booking slots:
- Scheduler handles events (e.g., single or recurring).
- Booking generates available time slots from those events.
So what you actually need is to generate booking slots from the schedule (the snippet below shows how to generate booking slots from the doctor's schedule by converting JSON data on the server-side).
-
Recurring events limitation:
- Booking supports only weekly recurring events (defined as INTERVAL=1;FREQ=WEEKLY in Scheduler).
- Scheduler can handle any recurring pattern so you will need to limit creating other recurring events in Scheduler config
-
Timezone handling:
- Booking interprets timestamps in the local timezone.
- If you use global timestamps, you need to convert them to local timezones before sending them to Booking (and vice versa before saving the data back). For conversion instructions, refer to Working with UTC data.
-
Booking slot strategies:
- Use
slots
andusedSlots
to build the schedule, ensuring that used slots are excluded (we'll focus on this strategy) - Use only
availableSlots
, which is suitable for events without recurrences.
- Use
Example
The snippet below demonstrates how to integrate Booking with the Scheduler widget by converting doctors' schedules into booking slots. Key data endpoints used for integration:
/doctors/worktime
- Scheduler data (doctor schedules) that includes recurring and single-time events. These events are used to create time slots for the Booking system./units
- final Booking slots generated from the Schedulerworktime
data. The slots are generated on the server-side. Please, also refer to backend./doctors/reservations
- an auxiliary collection used to visualizeusedSlots
in the timeline view. This data comes from the Booking form, containing information about already reserved slots for doctors./doctors
- contains all doctors, including their names and IDs. It is used for displaying doctor information in both the Scheduler and Booking widgets.
Converting Scheduler events to Booking slots is the major part of integration and the rules for handling the events and converting them to slots are described in the section below.
Rules for converting Scheduler events to Booking slots
We will show how to generate booking slots from the doctor's schedule using JSON data. Data is converted on the server-side. In all example below the schedule for the next period is considered: from 2025-03-13 to 2027-03-13.
Rule 1. Single event slot creation.
For each single event in the schedule, convert the start and end times to Booking slots by creating an entry in the slots array, including the corresponding date (dates).
Scheduler event:
{
"doctor_id": 1,
"start_date": "2025-03-18 02:00:00",
"end_date": "2025-03-18 06:00:00"
}
Booking slot:
{
"id": 1,
"slotSize": 20,
"slotGap": 5,
"slots": [
{
"from": "02:00",
"to": "06:00",
"dates": [
1742256000000 // 2025-03-18 00:00:00 (timestamp)
]
}
]
}
Rule 2. Recurring events.
For recurring events, we use a weekly pattern. The start date and end date of each recurring event in Scheduler should be equal to Booking start and end dates, otherwise create placeholders for dates before and after the recurring event (see Rule 7).
Scheduler event (weekly on weekdays): the recurrence rule (rrule) specifies that the event repeats weekly on Monday, Tuesday, Wednesday, Thursday, and Friday.
{
"doctor_id": 1,
"start_date": "2025-03-13 09:00:00",
"end_date": "2027-03-13 00:00:00",
"rrule": "INTERVAL=1;FREQ=WEEKLY;BYDAY=MO,TU,WE,TH,FR",
"duration": 28800
}
Booking slots: In Booking, the weekly schedule is represented as a single rule, with the same start and end times for all recurring events:
{
"id": 1,
"slotSize": 20,
"slotGap": 5,
"slots": [
{
"from": "09:00",
"to": "17:00",
"days": [1, 2, 3, 4, 5] // Monday to Friday
}
]
}
Rule 3. Scheduling an event that spans multiple days.
If an event spans across multiple days (e.g., starts at 8 PM and ends at 4 AM), it should be split into two slots — one for each day.
For example, when a doctor's shift starts on Saturday evening and lasts into Sunday morning, Booking can only generate slots within one day. In this case, we need to split the event into two separate rules: one for Saturday and another for Sunday.
Scheduler event:
{
"doctor_id": 2,
"start_date": "2025-03-13 20:00:00",
"end_date": "2027-03-13 00:00:00",
"rrule": "INTERVAL=1;FREQ=WEEKLY;BYDAY=SA",
"duration": 28800
}
Booking slots:
{
"id": 2,
"slotSize": 45,
"slotGap": 5,
"slots": [
{
"from": "20:00",
"to": "24:10",
"days": [6] // Saturday
},
{
"from": "00:10",
"to": "04:00",
"days": [0] // Sunday
}
]
}
Rule 4. Additional single events added to recurring events.
In this case, a single event is added to a recurring schedule. The Booking slots are generated for both the recurring and the single events. The single event dates are added to the recurring event's dates array.
Scheduler events:
- Recurring event: a doctor’s availability from 9:00 AM to 5:00 PM on weekdays.
- Single event: a doctor is also available from 2:00 AM to 6:00 AM on March 18th and 19th.
[
// recurring event
{
"doctor_id": 1,
"start_date": "2025-03-13 09:00:00",
"end_date": "2027-03-13 00:00:00",
"rrule": "INTERVAL=1;FREQ=WEEKLY;BYDAY=MO,TU,WE,TH,FR",
"duration": 28800
},
// single events
{
"doctor_id": 1,
"start_date": "2025-03-18 02:00:00",
"end_date": "2025-03-18 06:00:00"
},
{
"doctor_id": 1,
"start_date": "2025-03-19 02:00:00",
"end_date": "2025-03-19 06:00:00"
}
]
Booking slots:
- Merging events: the recurring event and single events are combined into one Booking rule.
- If a single event has priority, its specific dates (March 18th and 19th) are added to the recurring event's rule. Please, refer to Defining the slot rules
{
"id": 1,
"slotSize": 20,
"slotGap": 20,
"slots": [
{
"from": "02:00",
"to": "06:00",
"dates": [
1742256000000, // 2025-03-18 00:00:00
1742342400000 // 2025-03-19 00:00:00
]
},
{
"from": "09:00",
"to": "17:00",
"days": [1, 2, 3, 4, 5],
"dates": [
1742256000000, // 2025-03-18 00:00:00
1742342400000 // 2025-03-19 00:00:00
]
}
]
}
Rule 5. Modifying a single instance of a recurring event.
If a single instance of a recurring event is edited (e.g., time change for a specific date), generate a new slot with the updated time and date in the dates array, overriding the days array.
Scheduler event:
[
{
"doctor_id": 1,
"start_date": "2025-03-13 09:00:00",
"end_date": "2027-03-13 00:00:00",
"rrule": "INTERVAL=1;FREQ=WEEKLY;BYDAY=MO,TU,WE,TH,FR",
"duration": 28800
},
{
"doctor_id": 1,
"start_date": "2025-03-14 03:00:00",
"end_date": "2025-03-14 11:00:00",
"recurring_event_id": "1",
"original_start": "2025-03-14 09:00"
}
]
Booking slots:
{
"id": 1,
"slotSize": 20,
"slotGap": 5,
"slots": [
{
"from": "09:00",
"to": "17:00",
"days": [1, 2, 3, 4, 5]
},
{
"from": "03:00",
"to": "11:00",
"dates": [
1741910400000 // 2025-03-14 03:00:00 (modified)
]
}
]
}
Rule 6. Deleting a single instance of a recurring event.
When a single occurrence is removed from a recurring event in Scheduler, we need to update Booking rules to reflect this removal. This is done by creating a special rule for the removed date, using an empty time interval and the dates property (which has higher priority than days).
Scheduler events:
[
{
"doctor_id": 5,
"start_date": "2025-03-14 09:00:00",
"end_date": "2027-03-13 00:00:00",
"rrule": "INTERVAL=1;FREQ=WEEKLY;BYDAY=TH,FR,SA,SU",
"duration": 28800
},
{
"doctor_id": 5,
"start_date": "2025-03-23 09:00:00",
"end_date": "2025-03-23 17:00:00",
"recurring_event_id": "15",
"original_start": "2025-03-23 09:00",
"deleted": true
}
]
Booking slots:
{
"id": 5,
"slotSize":60,
"slotGap":10,
"slots":[
{
"from": "09:00",
"to": "17:00",
"days": [4, 5, 6, 0] // Thursday to Sunday
},
{
"from": "00:00",
"to": "00:00",
"dates": [
1742688000000 // 2025-03-23 00:00:00 (deleted occurrence)
]
}
]
}
Rule 7. Events starting later than Booking start date.
If a recurring event starts after the Booking start date (default is today which is 2025-03-13 in all examples), create rules with empty time intervals for the dates prior to the event's start date. This simulates the dates being "removed" from the recurrence.
Scheduler event:
{
"doctor_id": 5,
"start_date": "2025-03-17 09:00:00",
"end_date": "2027-03-13 00:00:00",
"rrule": "INTERVAL=1;FREQ=WEEKLY;BYDAY=SU,MO,TU,WE,TH,FR,SA",
"duration": 28800
}
Booking slots:
{
"id": 5,
"slotSize":60,
"slotGap":10,
"slots": [
{ "from": "09:00", "to": "17:00", "days": [0, 1, 2, 3, 4, 5, 6] },
{ "from": "00:00", "to": "00:00", "dates": [
1741820400000, // March 13, 2025
1741906800000, // March 14, 2025
1741993200000, // March 15, 2025
1742079600000 // March 16, 2025
]}
]
}