Hotel Raxa - Advanced Booking System Implementation
🏨 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>
This commit is contained in:
@@ -0,0 +1,8 @@
|
||||
<?php
|
||||
/**
|
||||
* Silence is golden.
|
||||
*
|
||||
* @package Redux Framework
|
||||
*/
|
||||
|
||||
echo null;
|
||||
@@ -0,0 +1,244 @@
|
||||
/*global redux_change, wp, redux */
|
||||
|
||||
/**
|
||||
* Media Uploader
|
||||
* Dependencies : jquery, wp media uploader
|
||||
* Feature added by : Smartik - http://smartik.ws/
|
||||
* Date : 05.28.2013
|
||||
*
|
||||
* Modified by : Kevin Provance (kprovance/svl-studios)
|
||||
* Date : 07.07.2021
|
||||
*/
|
||||
|
||||
( function ( $ ) {
|
||||
'use strict';
|
||||
|
||||
var isFiltered;
|
||||
|
||||
redux.field_objects = redux.field_objects || {};
|
||||
redux.field_objects.media = redux.field_objects.media || {};
|
||||
|
||||
redux.field_objects.media.init = function ( selector ) {
|
||||
if ( ! selector ) {
|
||||
selector = $( document ).find( '.redux-group-tab:visible' ).find( '.redux-container-media:visible' );
|
||||
}
|
||||
|
||||
$( selector ).each(
|
||||
function () {
|
||||
var el = $( this );
|
||||
var parent = el;
|
||||
|
||||
if ( ! el.hasClass( 'redux-field-container' ) ) {
|
||||
parent = el.parents( '.redux-field-container:first' );
|
||||
}
|
||||
|
||||
if ( parent.is( ':hidden' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( parent.hasClass( 'redux-field-init' ) ) {
|
||||
parent.removeClass( 'redux-field-init' );
|
||||
} else {
|
||||
return;
|
||||
}
|
||||
|
||||
if ( undefined !== redux.field_objects.image_filters) {
|
||||
redux.field_objects.image_filters.sliderInit( el, 'media' );
|
||||
redux.field_objects.image_filters.checkbox( el, 'media' );
|
||||
}
|
||||
|
||||
isFiltered = false;
|
||||
|
||||
// Remove the image button.
|
||||
el.find( '.remove-image, .remove-file' ).off( 'click' ).on(
|
||||
'click',
|
||||
function () {
|
||||
redux.field_objects.media.removeFile( $( this ).parents( 'fieldset.redux-field:first' ) );
|
||||
}
|
||||
);
|
||||
|
||||
// Upload media button.
|
||||
el.find( '.media_upload_button' ).off().on(
|
||||
'click',
|
||||
function ( event ) {
|
||||
redux.field_objects.media.addFile( event, $( this ).parents( 'fieldset.redux-field:first' ) );
|
||||
}
|
||||
);
|
||||
}
|
||||
);
|
||||
};
|
||||
|
||||
// Add a file via the wp.media function.
|
||||
redux.field_objects.media.addFile = function ( event, selector ) {
|
||||
var frame;
|
||||
var libFilter;
|
||||
var filter;
|
||||
var data;
|
||||
var thumbSrc;
|
||||
var height;
|
||||
var key;
|
||||
var object;
|
||||
|
||||
var jQueryel = $( this );
|
||||
|
||||
event.preventDefault();
|
||||
|
||||
// If the media frame already exists, reopen it.
|
||||
if ( frame ) {
|
||||
frame.open();
|
||||
return;
|
||||
}
|
||||
|
||||
// Get library filter data.
|
||||
filter = $( selector ).find( '.library-filter' ).data( 'lib-filter' );
|
||||
|
||||
// Must exist to do decoding.
|
||||
if ( undefined !== filter ) {
|
||||
if ( '' !== filter ) {
|
||||
libFilter = [];
|
||||
isFiltered = true;
|
||||
filter = decodeURIComponent( filter );
|
||||
filter = JSON.parse( filter );
|
||||
|
||||
$.each(
|
||||
filter,
|
||||
function ( index, value ) {
|
||||
index = null;
|
||||
libFilter.push( value );
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// Create the media frame.
|
||||
frame = wp.media(
|
||||
{ multiple: false,
|
||||
library: { type: libFilter }, // Only allow images.
|
||||
|
||||
// Set the title of the modal.
|
||||
title: jQueryel.data( 'choose' ),
|
||||
|
||||
// Customize the submit button.
|
||||
button: {
|
||||
|
||||
// Set the text of the button.
|
||||
text: jQueryel.data( 'update' )
|
||||
|
||||
// Tell the button not to close the modal, since we're
|
||||
// going to refresh the page when the image is selected.
|
||||
}
|
||||
}
|
||||
);
|
||||
|
||||
// When an image is selected, run a callback.
|
||||
frame.on(
|
||||
'select',
|
||||
function () {
|
||||
|
||||
// Grab the selected attachment.
|
||||
var attachment = frame.state().get( 'selection' ).first();
|
||||
frame.close();
|
||||
|
||||
data = $( selector ).find( '.data' ).data();
|
||||
|
||||
if ( 'undefined' === typeof redux.field_objects.media || undefined === typeof redux.field_objects.media ) {
|
||||
redux.field_objects.media = {};
|
||||
}
|
||||
|
||||
if ( undefined === data || 'undefined' === data.mode ) {
|
||||
data = {};
|
||||
data.mode = 'image';
|
||||
}
|
||||
|
||||
if ( true === isFiltered ) {
|
||||
data.mode = 0;
|
||||
}
|
||||
|
||||
if ( 0 === data.mode ) {
|
||||
|
||||
} else {
|
||||
if ( false !== data.mode ) {
|
||||
if ( attachment.attributes.type !== data.mode ) {
|
||||
if ( attachment.attributes.subtype !== data.mode ) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
selector.find( '.upload' ).val( attachment.attributes.url );
|
||||
selector.find( '.upload-id' ).val( attachment.attributes.id );
|
||||
selector.find( '.upload-height' ).val( attachment.attributes.height );
|
||||
selector.find( '.upload-width' ).val( attachment.attributes.width );
|
||||
|
||||
redux_change( $( selector ).find( '.upload-id' ) );
|
||||
|
||||
thumbSrc = attachment.attributes.url;
|
||||
|
||||
if ( 'undefined' !== typeof attachment.attributes.sizes && 'undefined' !== typeof attachment.attributes.sizes.thumbnail ) {
|
||||
if ( 'thumbnail' === data.previewSize ) {
|
||||
thumbSrc = attachment.attributes.sizes.thumbnail.url;
|
||||
}
|
||||
} else if ( 'undefined' !== typeof attachment.attributes.sizes ) {
|
||||
height = attachment.attributes.height;
|
||||
|
||||
for ( key in attachment.attributes.sizes ) {
|
||||
if ( attachment.attributes.sizes.hasOwnProperty( key ) ) {
|
||||
object = attachment.attributes.sizes[ key ];
|
||||
|
||||
if ( object.height < height ) {
|
||||
height = object.height;
|
||||
thumbSrc = object.url;
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
thumbSrc = attachment.attributes.icon;
|
||||
}
|
||||
|
||||
selector.find( '.upload-thumbnail' ).val( thumbSrc );
|
||||
if ( ! selector.find( '.upload' ).hasClass( 'noPreview' ) ) {
|
||||
selector.find( '.screenshot' ).empty().hide().append( '<img class="redux-option-image" src="' + thumbSrc + '">' ).slideDown( 'fast' );
|
||||
}
|
||||
|
||||
selector.find( '.remove-image' ).removeClass( 'hide' ); // Show "Remove" button.
|
||||
selector.find( '.redux-background-properties' ).slideDown();
|
||||
}
|
||||
);
|
||||
|
||||
// Finally, open the modal.
|
||||
frame.open();
|
||||
};
|
||||
|
||||
// Function to remove the image on click. Still requires a save.
|
||||
redux.field_objects.media.removeFile = function ( selector ) {
|
||||
var screenshot;
|
||||
|
||||
// This shouldn't have been run...
|
||||
if ( ! selector.find( '.remove-image' ).addClass( 'hide' ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
selector.find( '.remove-image' ).addClass( 'hide' ); // Hide "Remove" button.
|
||||
selector.find( '.upload' ).val( '' );
|
||||
selector.find( '.upload-id' ).val( '' );
|
||||
selector.find( '.upload-height' ).val( '' );
|
||||
selector.find( '.upload-width' ).val( '' );
|
||||
selector.find( '.upload-thumbnail' ).val( '' );
|
||||
redux_change( $( selector ).find( '.upload-id' ) );
|
||||
selector.find( '.redux-background-properties' ).hide();
|
||||
|
||||
screenshot = selector.find( '.screenshot' );
|
||||
|
||||
// Hide the screenshot.
|
||||
screenshot.slideUp();
|
||||
|
||||
selector.find( '.remove-file' ).off();
|
||||
|
||||
// We don't display the upload button if .upload-notice is present.
|
||||
// This means the user doesn't have the WordPress 3.5 Media Library Support.
|
||||
if ( selector.find( '.section-upload .upload-notice' ).length > 0 ) {
|
||||
selector.find( '.media_upload_button' ).remove();
|
||||
}
|
||||
};
|
||||
} )( jQuery );
|
||||
1
wp-content/plugins/eagle-booking/include/redux/assets/js/media/media.min.js
vendored
Normal file
1
wp-content/plugins/eagle-booking/include/redux/assets/js/media/media.min.js
vendored
Normal file
@@ -0,0 +1 @@
|
||||
!function(u){"use strict";var f;redux.field_objects=redux.field_objects||{},redux.field_objects.media=redux.field_objects.media||{},redux.field_objects.media.init=function(e){e=e||u(document).find(".redux-group-tab:visible").find(".redux-container-media:visible"),u(e).each(function(){var e=u(this),i=e;(i=e.hasClass("redux-field-container")?i:e.parents(".redux-field-container:first")).is(":hidden")||i.hasClass("redux-field-init")&&(i.removeClass("redux-field-init"),void 0!==redux.field_objects.image_filters&&(redux.field_objects.image_filters.sliderInit(e,"media"),redux.field_objects.image_filters.checkbox(e,"media")),f=!1,e.find(".remove-image, .remove-file").off("click").on("click",function(){redux.field_objects.media.removeFile(u(this).parents("fieldset.redux-field:first"))}),e.find(".media_upload_button").off().on("click",function(e){redux.field_objects.media.addFile(e,u(this).parents("fieldset.redux-field:first"))}))})},redux.field_objects.media.addFile=function(e,i){var d,t,a,s,o,r,l,n=u(this);e.preventDefault(),void 0!==(e=u(i).find(".library-filter").data("lib-filter"))&&""!==e&&(t=[],f=!0,e=decodeURIComponent(e),e=JSON.parse(e),u.each(e,function(e,i){t.push(i)})),(d=wp.media({multiple:!1,library:{type:t},title:n.data("choose"),button:{text:n.data("update")}})).on("select",function(){var e=d.state().get("selection").first();if(d.close(),a=u(i).find(".data").data(),void 0===redux.field_objects.media?redux.field_objects.media={}:redux.field_objects.media,void 0!==a&&"undefined"!==a.mode||(a={mode:"image"}),!0===f&&(a.mode=0),0===a.mode||!1===a.mode||e.attributes.type===a.mode||e.attributes.subtype===a.mode){if(i.find(".upload").val(e.attributes.url),i.find(".upload-id").val(e.attributes.id),i.find(".upload-height").val(e.attributes.height),i.find(".upload-width").val(e.attributes.width),redux_change(u(i).find(".upload-id")),s=e.attributes.url,void 0!==e.attributes.sizes&&void 0!==e.attributes.sizes.thumbnail)"thumbnail"===a.previewSize&&(s=e.attributes.sizes.thumbnail.url);else if(void 0!==e.attributes.sizes)for(r in o=e.attributes.height,e.attributes.sizes)e.attributes.sizes.hasOwnProperty(r)&&(l=e.attributes.sizes[r]).height<o&&(o=l.height,s=l.url);else s=e.attributes.icon;i.find(".upload-thumbnail").val(s),i.find(".upload").hasClass("noPreview")||i.find(".screenshot").empty().hide().append('<img class="redux-option-image" src="'+s+'">').slideDown("fast"),i.find(".remove-image").removeClass("hide"),i.find(".redux-background-properties").slideDown()}}),d.open()},redux.field_objects.media.removeFile=function(e){e.find(".remove-image").addClass("hide")&&(e.find(".remove-image").addClass("hide"),e.find(".upload").val(""),e.find(".upload-id").val(""),e.find(".upload-height").val(""),e.find(".upload-width").val(""),e.find(".upload-thumbnail").val(""),redux_change(u(e).find(".upload-id")),e.find(".redux-background-properties").hide(),e.find(".screenshot").slideUp(),e.find(".remove-file").off(),0<e.find(".section-upload .upload-notice").length)&&e.find(".media_upload_button").remove()}}(jQuery);
|
||||
Reference in New Issue
Block a user