In the first post of this series, we created a page showing all the hotel room’s information. It normally has a button or several call-to-action areas to go to the booking page, which allows your customers to make an order. However, cause some businesses have strong direct channels, they want their sales to make booking orders for their customers as well. Along with that, to have you imagine how to create a booking page in the frontend later (which is in the 3rd post), I made this post to show you how to create one in the backend. That is for internal users only.

Before going ahead, let’s check some tools we need in this practice.
Video Version
Before Getting Started
Each booking will be saved as a post in a custom post type. All the related information, such as customer details, check-in/check-out dates, order info, payment, and booking status, will be stored in custom fields. We’ll also generate a unique booking ID automatically for each entry to make it easier to track and manage bookings in the backend.
To build a hotel booking website, we need many basic and advanced features. So, we highly recommend you use Meta Box AlO to have a framework to have everything. Also, it includes all the Meta Box extensions that you need for your creation.
For this case, we need:
- MB Custom Post Type: to create a custom post type for the booking;
- MB Builder: to have a UI on the backend to visually create custom fields;
- MB Conditional Logic: to display fields only when certain conditions are met;
- MB Group: to organize fields into groups for easy management;
- MB Admin Columns: to display the booking status as a column in the dashboard. It’s optional.
Let’s start now!
Create a Custom Post Type
We are going to create a new custom post type called Booking. Each booking made will be saved as a post in this post type.
Head to Meta Box > Post Types, and create a new one.

Each booking automatically uses its ID as the title and doesn’t need content or images. So I disable the default fields for easier control and just turn on the Revision option to allow tracking changes.

Create Custom Fields for the Booking
For the booking page, we need to create many different fields. You should clarify in advance what you want, how users fill in the data, how to show fields and their values, etc. This makes the process much smoother. As my supposition, all the fields I need are as follows:
| Name | Field type | Function | 
| Order Number | number | Is an array of numbers and auto-generated whenever the booking is created | 
| Booking Date | date | Date of creating the booking | 
| Person in Charge | user | Choose one from the list of users in WordPress | 
| Customer information | heading | Separate the sections | 
| · Full Name | text | |
| · Mobile | number | |
| · Address | text | |
| · Notes | textarea | |
| Booking Information | heading | Separate the sections | 
| Booking Details | group | Group of fields regarding the booking room. This group is cloneable for the case that the customer books multiple rooms | 
| · Room | post | Choose one from the list of the rooms | 
| · Price | number | |
| · Adults | number | |
| · Children | number | |
| · Age of Children (1, 2) | number | Each field is for a child. This field is auto-display whenever there is one more child | 
| · Extra Bed | number | |
| · Check-in Date | date | |
| · Check-out Date | date | |
| · Total Nights of Stay | number | Auto-calculated bases on the check-in and check-out date | 
| Total Number of Rooms | number | Auto-calculated bases on the number of the group about the room details | 
| Other Request | textarea | |
| Payment Information | heading | Separate the sections | 
| · Total Amount | number | Auto-calculated bases on room details and number of rooms | 
| · Paid Amount | number | Manual input by sales | 
| · Unpaid Amount | number | = Total amount - the paid amount | 
| Booking Status | heading | Separate the sections | 
| · Booking Status | select | Choose one from the provided statuses | 
| · Refund (if any) | select | Choose one level (%) from the provided levels | 

In the demo, you can see that the fields are grouped into sections based on their purposes, like Customer Details, Room Booking, Payment, and . Each section has a clear heading and inclBooking Statusudes related fields underneath, making the form more organized and easier to follow.
The Booking Information section is a cloneable group, useful for booking multiple rooms. We also set conditions for some fields to appear only when needed; for example, the Price field shows up after selecting a room, and the Age of Children fields only appear if there’s more than one child included in the booking. I’ll go into detail for those with special settings.
Since the process is similar, I’ll walk you through the key fields only.
Now, go to Meta Box, Custom Fields, to create a new field group.

Then, create fields one by one.
Create a Number Field for Order Number
Begin with a Number field for the order number. Since it is auto-generated information that users can not input or edit, tick the Disabled and Read Only options.

Create a field for Person in Charge
For the person in charge of the booking, typically a staff member with a WordPress admin account, we’ll use a User field. This allows you to select any user from the admin list.

Create a heading to separate the sections
Now, let’s organize the fields a bit to make things clearer and more well-structured. I use a Layout field called Heading to separate the sections or groups. This one is for customer information. Then, just add a few simple fields for the customer’s details, like name, phone number, and email.

Create a Group of Fields for Booking Details
In this section, we’ll create a Group of fields containing all booking-related information: type of rooms, price, number of adults, number of children, children’s ages, extra bed, check-in and check-out dates. To group them all, we need help from the MB Group extension we mentioned before.

And, enable the cloneable option for the case that the customer books multiple rooms. Then, set Custom CSS Class to track when the user clones the group to book additional rooms. This will help to calculate the total room later.

Besides that, each sub-field in this group has some remarks:
Fields for the Room Name
For the room, choose the field type as Post. In the field settings, choose the corresponding post type. This lets users choose from the list of rooms we created earlier in the previous part.
The Children field
As the regulation of each hotel, children may be charged or not depending on their ages. So we will create additional fields to fill in their ages. An additional field for age will appear when you add one more in the Children field. I restrict the maximum number of children is two as well.
I use MB Conditional Logic to set a condition for the age fields.


Like the above image, I set the Age of Children 1 field displays when Children >=1. It is likewise for the Age of Children 2, displays when Children >=2.
Check-in Date and Check-out Date
We create these fields in the type of normal date with a calendar. In the last post of this series, we will show the vacant room to the calendar in real time.
In the drop-down menu, choose Date Picker. This field will show a calendar to choose a date instead of typing.
The following field is for Extra Bed. Very simple with no special settings.
Once all fields are configured, move to the Settings tab, set the Location as Post type, and select Booking to apply these fields to it.

Now, navigate to your post editor of that post type, and you will see custom fields displayed.

Simply input values in these fields. When choosing any room, there will be a price section that appears as we set the condition before, but it does not display the corresponding price. It's the same when you choose the number of children.

We have an auto-saved booking with pending status shown as admin columns as well.

When the booking is approved, a booking confirmation number will be issued. Let’s add more functions to the custom fields in the next step.
Add Extra Functions to the Custom Fields
In this step, we are going to deal with some special fields to automatically calculate or generate values for them. Including:
- Automatically generate booking order number;
- Automatically calculate the total number of booking rooms;
- Automatically display the corresponding price of the booking room;
- Automatically calculate the total nights of stay;
- Automatically calculate the total amount.
We’ll need to code much in this part. Let’s go!
Go to your theme folder, and add some code to the functions.php file.

Automatically Generate Booking Order Number
Each post in the Booking post type will have neither a title nor a permalink. All of them will be kinds of the Auto Draft after saving.
This is the post before we create a title for it:

Now, add these codes to the functions.php in the theme folder:
add_action('save_post_booking', function($post_id) {
    if (wp_is_post_revision($post_id) || defined('DOING_AUTOSAVE')) return;
    $post = get_post($post_id);
    if ($post->post_title !== "#$post_id" || $post->post_name !== (string)$post_id) {
        wp_update_post([
            'ID'         => $post_id,
            'post_title' => "#$post_id",
            'post_name'  => $post_id,
        ]);
    }
    update_post_meta($post_id, 'order_number', $post_id);
}, 20);
In there, update_post_meta($post_id, 'order', $post_id) helps to set the post ID to be the value of the Order Number field.
Save the functions.php file, then create a new booking order for a try. You’ll see the title of order as below:


Automatically Calculate the Total Number of Booking Rooms
By default, the quantity of rooms will be 1, equivalent to 1 group of booking details for 1 room. Use this following code so that whenever users add or edit a booking, the JS file will be loaded. It helps handle things like disabling booked dates via AJAX.
function add_admin_scripts($hook) {
    $screen = get_current_screen();
    if ($hook == 'post-new.php' || $hook == 'post.php') {
        if (isset($screen->post_type) && 'booking' === $screen->post_type) {
        wp_register_script('booking-js', get_stylesheet_directory_uri() . '/js/booking.js');
        // Passing AJAX Data into JavaScript
        $ajax_data = array(
            'ajax_url' => admin_url('admin-ajax.php'),
            'nonce' => wp_create_nonce('booking_nonce')
        );
        wp_localize_script('booking-js', 'booking_ajax', $ajax_data);
        // Passing room data into JavaScript
            $rooms = get_posts([
                'post_type'      => 'room',
                'posts_per_page' => -1
            ]);
            $arr_rooms = [];
            foreach ($rooms as $room) {
                $disabled_dates = dates_disable($room->ID);
                $arr_rooms[] = [
                    'id'    => $room->ID,
                    'price' => rwmb_get_value('price', '', $room->ID),
                    'disabled_dates' => $disabled_dates
                ];
            }
            wp_localize_script('booking-js', 'rooms_data', $arr_rooms);
            wp_enqueue_script('booking-js');
        }
    }
}
add_action('admin_enqueue_scripts', 'add_admin_scripts');
Afterward, create a JavaScript file to handle calculations and dynamic interactions. Now add some script into that file.

When users add or remove a room group, this script updates the total number of rooms. And also refreshes the price, stay duration, and total cost for all existing room groups.
$('.group-booking .add-clone').on('click', function (e) {
    setTimeout(function () {
        var rooms = $('.group-booking .rwmb-group-clone:not(.rwmb-clone-template)').length;
        $('#total_number_of_rooms').val(rooms);
        // Update all rooms
        $(".group-booking .rwmb-field select[name*='[room]']").each(function() {
            updateRoom($(this));
        });
    }, 100);
});
// Handle room removal
$('.group-booking .remove-clone').on('click', function (e) {
    setTimeout(function () {
        var rooms = $('.group-booking .rwmb-group-clone:not(.rwmb-clone-template)').length;
        $('#total_number_of_rooms').val(rooms);
        update_total_payment();
    }, 100);
} );
In there:
- group-booking: is the Custom CSS Class of the group of fields about booking details;
- total_number_of_rooms: is the ID of the Total Number of Room field;
- add-clone,- remove-clone: is the Custom CSS Class of the Add Room or Remove button. To see this information, you may press F12:

Automatically Display the Corresponding Price of the Booking Room
First, we set the Price field to Read only with a condition to display is that the value of the Room field is not “empty”. It means that when the Room field has any value (not leave it blank), the Price field will appear.

To do it, add the following code to the JS file.
$(".group-booking").on('change', "select[name*='[room]']", function() {
    updateRoom($(this));
        } );
        // Initialize event for extra bed
    $(".group-booking").on('change', "input[name*='[extra_bed]']", function() {
        var roomSelect = $(this).closest('.rwmb-field').siblings().find("select[name*='[room]']");
        var check_out = $(this).closest('.rwmb-field').siblings().find("input[name*='[check_out]']");
        var total_nights = $(this).closest('.rwmb-field').siblings().find("input[name*='[total_nights_of_stay]']");
        var total_amount = $(this).closest('.rwmb-field').siblings().find("input[name*='[total_amount]']");
        if (check_out.val()) {
            var roomPrice = roomSelect.data('roomPrice') || 0;
            var extra = parseInt($(this).val()) || 0;
            var total = parseInt(total_nights.val()) || 0;
            var total_extra = extra * roomPrice;
            var finalAmount = (roomPrice + total_extra) * total;
            if (!isNaN(finalAmount)) {
                 total_amount.val(finalAmount);
                 update_total_payment();
            }
        }
} );
Automatically Calculate the Total Nights of Stay
Before calculating the total nights of stay, we should set a condition that the check-out date must be after the check-in date. I use js to set the minDate of the check-out date is the check-in date plus one unit.
We can see some specified code for special aims:
check_in.datepicker('destroy').datepicker({
    dateFormat: "yy-mm-dd",
    beforeShowDay: function(date) {
        var formattedDate = $.datepicker.formatDate('yy-mm-dd', date);
        var isDisabled = allDisabledDates.includes(formattedDate);
        var today = new Date();
        today.setHours(0, 0, 0, 0);
        var isPastDate = date < today;
        return [!isDisabled && !isPastDate, '', isDisabled ? 'This date cannot be booked' : isPastDate ? 'Cannot select past dates' : ''];
    },
    onSelect: function(date) {
        var d = new Date(date);
        var af_date = d.getDate() + 1;
        var af_month = d.getMonth() + 1;
        var af_year = d.getFullYear();
        var min_date = af_year + '-' + af_month + '-' + af_date;
       check_out.datepicker('option', 'minDate', min_date);
    }
});
When selecting the check-out date, the following code will call the calculate_total_day function to calculate the number of nights stayed and assign it to the total_nights_of_stay field.
check_out.datepicker('destroy').datepicker({
    dateFormat: "yy-mm-dd",
    beforeShowDay: function(date) {
        var formattedDate = $.datepicker.formatDate('yy-mm-dd', date);
        var isDisabled = allDisabledDates.includes(formattedDate);
        var today = new Date();
        today.setHours(0, 0, 0, 0);
        var isPastDate = date < today;
        return [!isDisabled && !isPastDate, '', isDisabled ? 'This date cannot be booked' : isPastDate ? 'Cannot select past dates' : ''];
    },
    onSelect: function(date) {
        var total = calculate_total_day(check_in.val(), date);
        if (!isNaN(total)) {
            total_nights.val(total);
            var extra = parseInt(extra_bed.val()) || 0;
            var roomPrice = roomSelect.data('roomPrice') || 0;
            var total_extra = extra * roomPrice;
            var finalAmount = (roomPrice + total_extra) * total;
            if (!isNaN(finalAmount)) {
                 total_amount.val(finalAmount);
                 update_total_payment();
            }
        }
    }
});
After that, add these codes to the booking.js file to calculate the total nights of stay:
function calculate_total_day(check_in, check_out) {
    var date1 = new Date(check_in);
    var date2 = new Date(check_out);
    return (date2.getTime() - date1.getTime()) / (1000 * 3600 * 24);
}
From now on, your Total Nights of Stay field’s value is automatically changed when you choose a date:

Automatically Calculate the Total Amount
The formula to calculate the total amount is as below:
Total Amount = (Price of room 1 + numbers of extra bed in room 1 * price) * nights
+ (Price of room 2 + numbers of extra bed in room 2 * price) * nights
+ …
+ (Price of room N + numbers of extra bed in room N * price) * nights
In there, I set the Extra Bed is a kind of room and has a fixed price. At the same time, I create a field for the Total Amount of Each Room:
Total Amount of Each Room = (Price of room + numbers of extra bed * price) * nights
We just need to summarize all the Total Amount of Each Room to get the Total Amount of the booking.
Add these codes to the booking.js file to calculate the total amount of each room:
function update_total_payment() {
    var total_payment = 0;
    $(".group-booking .rwmb-field input[name*='[total_amount]']").each(function () {
        var value = parseInt($(this).val()) || 0;
        total_payment += value;
    });
    $('#total').val(total_payment);
    var paid_amount = parseInt($('#paid_amount').val()) || 0;
    var unpaid = total_payment - paid_amount;
    $('#unpaid_amount').val(unpaid);
}
function update_paid_amount() {
    $("#paid_amount").on('change', function () {
        $('#unpaid_amount').val(parseInt($('#total').val()) - parseInt($(this).val()));
    });
}
update_paid_amount();
All code is updated on Github, so you can refer to it:
Therefore, all the fields will automatically run as your logic:
Now let’s book an order on the back end to choose a room and dates to see if the pricing and calculations are working correctly.

Last Words
With everything set up, your internal team can now create bookings from the backend with all necessary information, automatically calculated prices, disabled dates for fully booked rooms, and dynamic updates for total cost. This not only helps reduce human errors but also lays a solid foundation for building the frontend version later.
In the next part, we’ll bring this booking process to the frontend so your customers can book rooms themselves with the same smart features. Stay tuned!
- How to Build a Hotel Booking Website Using Meta Box - P1
- How to Build a Hotel Booking Website Using Meta Box - P2 - Booking Page in Backend
- How to Build a Hotel Booking Website Using Meta Box – P3 – Booking Page for Customer
- How to Build a Hotel Booking Website Using Meta Box - P4 - Booking Management Page
- Author Bio
- Better 404 Page
- Blogs for Developers
- Building a Simple Listing Website with Filters
- Building an Event Website
- Building Forms with MB Frontend Submission
- Coding
- Create a Chronological Timeline
- Custom Fields Fundamentals
- Design Patterns
- Displaying Posts with Filters
- Download and Preview Buttons
- Dynamic Banners
- Dynamic Landing Page
- FAQs Page
- Featured Products
- Filter Posts by Relationships
- Full Site Editing
- Google Fonts
- Gutenberg
- Hotel Booking
- Latest Products
- Logo Carousel
- MB Builder Applications
- MB Group Applications
- MB Views Applications
- Most Viewed Posts
- Opening Hours
- OTA Website
- Pricing Table Page
- Product Page
- Product Variations
- Querying and Showing Posts by Custom Fields
- Recipe
- Related Posts via Relationship
- Restaurant Menus
- SEO Analysis
- Simple LMS
- Speed Up Website
- Taxonomy Thumbnails
- Team Members
- User Profile
- Video Gallery
 
	
	
Is part 3 ever coming?
Yes, it's on the schedule.
Can you please provide an exported dump (.dat file) of the "Bookings" custom fields?
that shall help many to save time
The source code is available on Github. We've just added it recently.
I can create first booking, without adding the function (booking post name AUTO DRAFT).
Then i follow the tutorial to add the functions.
Then create new booking, it will lead to 500 server error.
If i delete that function, it goes normal.
I also tried to import your .dat file on github, it gives me error
Please help.
Anyone still looking for solution, try this:
add_action('save_post', function ($post_id) {
global $wpdb;
if (get_post_type($post_id) == 'booking') {
$title = $post_id;
$where = array('ID' => $post_id);
$wpdb->update($wpdb->posts, array('post_title' => $title), $where);
update_post_meta($post_id, 'order', $post_id);
}
});
Be careful what you write/allow to wpdb as it can lead to SQL injection attack.
Hi, I am trying to follow the tutorial to set up a billboard booking system. I am stuck at the stage of selecting the start and end dates of the reservation. the script does not work. Can you help me ?
$( ".group-reserv .rwmb-field input[name*='[date_de_debut]']" ).on('change', function(){
var d = new Date($(this).val());
var af_date = d.getDate() + 1;
var af_month = d.getMonth() + 1;
var af_year = d.getFullYear();
var min_date = af_year + '-' + af_month + '-' + af_date;
$(this).closest('.rwmb-field').siblings().find("input[name*='[date_de_fin]']").datepicker('option', 'minDate', min_date);
});
$( ".group-reserv .rwmb-field input[name*='[date_de_fin]']" ).on('change', function(){
var total = calculate_total_day($(this).closest('.rwmb-field').siblings().find("input[name*='[date_de_debut]']").val(), $(this).val());
$(this).closest('.rwmb-field').siblings().find("input[name*='[duree_de_campagne]']").val(total);
});
function calculate_total_day(date_de_debut, date_de_fin){
var date1 = new Date(date_de_debut);
var date2 = new Date(date_de_fin);
return (date2.getTime() - date1.getTime()) / (1000 * 3600 * 24);
}