🏨 Hotel Booking Enhancements: - Implemented Eagle Booking Advanced Pricing add-on - Added Booking.com-style rate management system - Created professional calendar interface for pricing - Integrated deals and discounts functionality 💰 Advanced Pricing Features: - Dynamic pricing models (per room, per person, per adult) - Base rates, adult rates, and child rates management - Length of stay discounts and early bird deals - Mobile rates and secret deals implementation - Seasonal promotions and flash sales 📅 Availability Management: - Real-time availability tracking - Stop sell and restriction controls - Closed to arrival/departure functionality - Minimum/maximum stay requirements - Automatic sold-out management 💳 Payment Integration: - Maintained Redsys payment gateway integration - Seamless integration with existing Eagle Booking - No modifications to core Eagle Booking plugin 🛠️ Technical Implementation: - Custom database tables for advanced pricing - WordPress hooks and filters integration - AJAX-powered admin interface - Data migration from existing Eagle Booking - Professional calendar view for revenue management 📊 Admin Interface: - Booking.com-style management dashboard - Visual rate and availability calendar - Bulk operations for date ranges - Statistics and analytics dashboard - Modal dialogs for quick editing 🔧 Code Quality: - WordPress coding standards compliance - Secure database operations with prepared statements - Proper input validation and sanitization - Error handling and logging - Responsive admin interface 🤖 Generated with Claude Code (https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
177 lines
7.7 KiB
PHP
177 lines
7.7 KiB
PHP
<?php
|
|
/* --------------------------------------------------------------------------
|
|
* Eagle Booking / Calendar Availability
|
|
* Author: Eagle Themes (Jomin Muskaj)
|
|
* Since: 1.1.5
|
|
* Modified: 1.3.3.8
|
|
---------------------------------------------------------------------------*/
|
|
|
|
defined('ABSPATH') || exit;
|
|
|
|
function eb_availability_calendar($eb_room_id, $eb_checkin, $eb_checkout) {
|
|
|
|
global $wpdb;
|
|
|
|
/**
|
|
* Room ID based on used multi-language plugin (WPML or Polylang)
|
|
*/
|
|
if ( function_exists('wpml_loaded') ) {
|
|
|
|
$eb_room_id = apply_filters('wpml_object_id', $eb_room_id, 'eagle_rooms', true, apply_filters('wpml_default_language', NULL ));
|
|
|
|
} elseif ( function_exists('pll_the_languages') ) {
|
|
|
|
$eb_room_id = pll_get_post( $eb_room_id, pll_default_language() );
|
|
|
|
} else {
|
|
|
|
$eb_room_id = $eb_room_id ;
|
|
|
|
}
|
|
|
|
// Get Room Exceptions
|
|
$eagle_booking_exceptions_array = get_post_meta($eb_room_id, 'eagle_booking_mtb_room_block_exceptions', true);
|
|
|
|
// Room Quantity
|
|
$eagle_booking_meta_box_qnt = get_post_meta($eb_room_id, 'eagle_booking_mtb_room_quantity', true);
|
|
if (empty($eagle_booking_meta_box_qnt)) {
|
|
$eagle_booking_meta_box_qnt = 1;
|
|
}
|
|
|
|
// Set the check circle
|
|
$eagle_booking_check_start_date_format = DateTime::createFromFormat("Y/m/d", $eb_checkin)->format('m/d/Y');
|
|
$eagle_booking_check_end_date_format = DateTime::createFromFormat("Y/m/d", $eb_checkout)->format('m/d/Y');
|
|
$eagle_booking_check_circle = eb_total_booking_nights($eagle_booking_check_start_date_format, $eagle_booking_check_end_date_format);
|
|
|
|
// DB Query
|
|
$eagle_booking_booked_dates = $wpdb->get_results("SELECT date_from, date_to FROM ".EAGLE_BOOKING_TABLE." WHERE id_post = $eb_room_id AND (paypal_payment_status = 'Completed' OR paypal_payment_status = 'Pending Payment') ");
|
|
|
|
$eagle_booking_avaiability_string = '';
|
|
|
|
if ($eagle_booking_booked_dates) {
|
|
|
|
foreach ($eagle_booking_booked_dates as $eagle_booking_booked_date) {
|
|
|
|
// Checked Checkin
|
|
$eagle_booking_check_start_date = DateTime::createFromFormat("Y/m/d", $eb_checkin)->format('Y/m/d');
|
|
|
|
// Booked Checkin & Checkout
|
|
$eagle_booking_booked_checkin = $eagle_booking_booked_date->date_from;
|
|
$eagle_booking_booked_checkout = $eagle_booking_booked_date->date_to;
|
|
|
|
// Set the booked circle
|
|
$eagle_booking_booked_circle = eb_total_booking_nights($eagle_booking_booked_checkin, $eagle_booking_booked_checkout);
|
|
|
|
// Start loop the sheck dates
|
|
for ($eagle_booking_i_1 = 1; $eagle_booking_i_1 <= $eagle_booking_check_circle; $eagle_booking_i_1++) {
|
|
|
|
$eagle_booking_booked_start_date = DateTime::createFromFormat("m/d/Y", $eagle_booking_booked_checkin)->format('Y/m/d');
|
|
|
|
// Start Check the booked date
|
|
for ($eagle_booking_i_2 = 1; $eagle_booking_i_2 <= $eagle_booking_booked_circle; $eagle_booking_i_2++) {
|
|
|
|
if ($eagle_booking_check_start_date == $eagle_booking_booked_start_date) {
|
|
|
|
$eagle_booking_avaiability_string .= $eagle_booking_booked_start_date . '-';
|
|
|
|
}
|
|
|
|
// Add 1 day to check the next booked date
|
|
$eagle_booking_booked_start_date = date('Y/m/d', strtotime($eagle_booking_booked_start_date . ' + 1 days'));
|
|
|
|
}
|
|
|
|
// Add 1 day to check the next check date
|
|
$eagle_booking_check_start_date = date('Y/m/d', strtotime($eagle_booking_check_start_date . ' + 1 days'));
|
|
|
|
}
|
|
}
|
|
|
|
$eagle_booking_avaiability_count = $eagle_booking_avaiability_string;
|
|
|
|
} else {
|
|
|
|
$eagle_booking_avaiability_count = '';
|
|
|
|
}
|
|
|
|
$booked_dates = [];
|
|
|
|
// Check the Room Quantity
|
|
$eagle_booking_quantity_availability = $eb_checkin;
|
|
|
|
for ($eagle_booking_i = 1; $eagle_booking_i <= $eagle_booking_check_circle; $eagle_booking_i++) {
|
|
|
|
$eagle_booking_num_reservations_per_day = substr_count($eagle_booking_avaiability_count, $eagle_booking_quantity_availability);
|
|
|
|
if ($eagle_booking_num_reservations_per_day >= $eagle_booking_meta_box_qnt) {
|
|
|
|
$booked_dates[] = $eagle_booking_quantity_availability;
|
|
}
|
|
|
|
$eagle_booking_quantity_availability = date('Y/m/d', strtotime($eagle_booking_quantity_availability . ' + 1 days'));
|
|
}
|
|
|
|
// Check the Room Blocks
|
|
if (!empty($eagle_booking_exceptions_array)) {
|
|
|
|
for ($eagle_booking_exceptions_array_i = 0; $eagle_booking_exceptions_array_i < count($eagle_booking_exceptions_array); $eagle_booking_exceptions_array_i++) {
|
|
|
|
// Get exception path
|
|
$eagle_booking_page_by_path = get_post($eagle_booking_exceptions_array[$eagle_booking_exceptions_array_i], OBJECT, 'eagle_exceptions');
|
|
|
|
// Exceptions MTB
|
|
$eagle_booking_exception_id = $eagle_booking_page_by_path->ID;
|
|
$eagle_booking_exception_name = get_the_title($eagle_booking_exception_id);
|
|
$eagle_booking_mtb_exception_date_from = get_post_meta($eagle_booking_exception_id, 'eagle_booking_mtb_exception_date_from', true);
|
|
$eagle_booking_mtb_exception_date_to = get_post_meta($eagle_booking_exception_id, 'eagle_booking_mtb_exception_date_to', true);
|
|
// $eagle_booking_mbt_exception_repeat = get_post_meta($eagle_booking_exception_id, 'eagle_booking_mtb_exception_repeat', true);
|
|
|
|
// Check if the exception still exists (in case it get deleted without updating the room)
|
|
if ( !empty( $eagle_booking_mtb_exception_date_from ) && !empty( $eagle_booking_mtb_exception_date_to ) ) {
|
|
|
|
// Get the total blocked booking nights
|
|
$eagle_booking_blocked_booking_nights = eb_total_booking_nights($eagle_booking_mtb_exception_date_from, $eagle_booking_mtb_exception_date_to);
|
|
|
|
// Format Exception Dates
|
|
$eagle_booking_new_date_from_ex_format = DateTime::createFromFormat("m/d/Y", $eagle_booking_mtb_exception_date_from)->format('Y/m/d');
|
|
$eagle_booking_new_date_to_ex_format = DateTime::createFromFormat("m/d/Y", $eagle_booking_mtb_exception_date_to)->format('Y/m/d');
|
|
|
|
// Set the check circle
|
|
$eagle_booking_checked_date = DateTime::createFromFormat("Y/m/d", $eb_checkin)->format('m/d/Y');
|
|
|
|
//Bug: Further investigation needed (it check checkin + 1)
|
|
$eagle_booking_checked_date = date('m/d/Y', strtotime($eagle_booking_checked_date . ' - 1 days'));
|
|
|
|
for ($eagle_booking_i_1 = 0; $eagle_booking_i_1 <= $eagle_booking_check_circle; $eagle_booking_i_1++) {
|
|
|
|
$eagle_booking_blocked_date = $eagle_booking_new_date_from_ex_format;
|
|
|
|
for ($eagle_booking_i_2 = 0; $eagle_booking_i_2 <= $eagle_booking_blocked_booking_nights; $eagle_booking_i_2++) {
|
|
|
|
|
|
if ($eagle_booking_checked_date == $eagle_booking_blocked_date) {
|
|
|
|
$booked_dates[] = $eagle_booking_checked_date;
|
|
|
|
}
|
|
|
|
$eagle_booking_blocked_date = date('Y/m/d', strtotime($eagle_booking_blocked_date . ' + 1 days'));
|
|
|
|
}
|
|
|
|
$eagle_booking_checked_date = date('Y/m/d', strtotime($eagle_booking_checked_date . ' + 1 days'));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Return the booked & blocked dates (remove the first date to allow it as a checkout date of the previous booking: further instigation needed)
|
|
return array_slice($booked_dates, 0);
|
|
|
|
}
|