🏨 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>
191 lines
5.1 KiB
PHP
191 lines
5.1 KiB
PHP
<?php
|
|
namespace Elementor;
|
|
|
|
use Elementor\Core\Editor\Editor;
|
|
use Elementor\Core\Utils\Collection;
|
|
|
|
if ( ! defined( 'ABSPATH' ) ) {
|
|
exit; // Exit if accessed directly.
|
|
}
|
|
|
|
/**
|
|
* Elementor images manager.
|
|
*
|
|
* Elementor images manager handler class is responsible for retrieving image
|
|
* details.
|
|
*
|
|
* @since 1.0.0
|
|
*/
|
|
class Images_Manager {
|
|
|
|
/**
|
|
* Get images details.
|
|
*
|
|
* Retrieve details for all the images.
|
|
*
|
|
* Fired by `wp_ajax_elementor_get_images_details` action.
|
|
*
|
|
* @since 1.0.0
|
|
* @access public
|
|
*/
|
|
public function get_images_details() {
|
|
if ( ! current_user_can( Editor::EDITING_CAPABILITY ) ) {
|
|
wp_send_json_error( 'Permission denied' );
|
|
}
|
|
|
|
// PHPCS - Already validated by wp_ajax.
|
|
$items = Utils::get_super_global_value( $_POST, 'items' ) ?? []; // phpcs:ignore WordPress.Security.NonceVerification.Missing
|
|
$urls = [];
|
|
|
|
foreach ( $items as $item ) {
|
|
$urls[ $item['id'] ] = $this->get_details( $item['id'], $item['size'], $item['is_first_time'] );
|
|
}
|
|
|
|
wp_send_json_success( $urls );
|
|
}
|
|
|
|
/**
|
|
* Get image details.
|
|
*
|
|
* Retrieve single image details.
|
|
*
|
|
* Fired by `wp_ajax_elementor_get_image_details` action.
|
|
*
|
|
* @since 1.0.0
|
|
* @access public
|
|
*
|
|
* @param string $id Image attachment ID.
|
|
* @param string|array $size Image size. Accepts any valid image
|
|
* size, or an array of width and height
|
|
* values in pixels (in that order).
|
|
* @param string $is_first_time Set 'true' string to force reloading
|
|
* all image sizes.
|
|
*
|
|
* @return array URLs with different image sizes.
|
|
*/
|
|
public function get_details( $id, $size, $is_first_time ) {
|
|
if ( ! class_exists( 'Group_Control_Image_Size' ) ) {
|
|
require_once ELEMENTOR_PATH . '/includes/controls/groups/image-size.php';
|
|
}
|
|
|
|
if ( 'true' === $is_first_time ) {
|
|
$sizes = get_intermediate_image_sizes();
|
|
$sizes[] = 'full';
|
|
} else {
|
|
$sizes = [];
|
|
}
|
|
|
|
$sizes[] = $size;
|
|
$urls = [];
|
|
foreach ( $sizes as $size ) {
|
|
if ( 0 === strpos( $size, 'custom_' ) ) {
|
|
preg_match( '/custom_(\d*)x(\d*)/', $size, $matches );
|
|
|
|
$matches[1] = (int) $matches[1];
|
|
$matches[2] = (int) $matches[2];
|
|
|
|
$instance = [
|
|
'image_size' => 'custom',
|
|
'image_custom_dimension' => [
|
|
'width' => $matches[1],
|
|
'height' => $matches[2],
|
|
],
|
|
];
|
|
|
|
$url = Group_Control_Image_Size::get_attachment_image_src( $id, 'image', $instance );
|
|
|
|
$thumbs_path = BFITHUMB_UPLOAD_DIR . '/' . basename( $url );
|
|
|
|
$image_meta = wp_get_attachment_metadata( $id );
|
|
|
|
// Attach custom image to original.
|
|
$image_meta['sizes'][ 'elementor_' . $size ] = [
|
|
'file' => $thumbs_path,
|
|
'width' => $matches[1],
|
|
'height' => $matches[2],
|
|
'mime-type' => get_post_mime_type( $id ),
|
|
];
|
|
|
|
wp_update_attachment_metadata( $id, $image_meta );
|
|
|
|
$urls[ $size ] = $url;
|
|
} else {
|
|
$urls[ $size ] = wp_get_attachment_image_src( $id, $size )[0];
|
|
}
|
|
}
|
|
|
|
return $urls;
|
|
}
|
|
|
|
/**
|
|
* Get Light-Box Image Attributes
|
|
*
|
|
* Used to retrieve an array of image attributes to be used for displaying an image in Elementor's Light Box module.
|
|
*
|
|
* @param int $id The ID of the image
|
|
*
|
|
* @return array An array of image attributes including `title` and `description`.
|
|
* @since 2.9.0
|
|
* @access public
|
|
*/
|
|
|
|
public function get_lightbox_image_attributes( $id ) {
|
|
$attributes = [];
|
|
$kit = Plugin::$instance->kits_manager->get_active_kit();
|
|
$lightbox_title_src = $kit->get_settings( 'lightbox_title_src' );
|
|
$lightbox_description_src = $kit->get_settings( 'lightbox_description_src' );
|
|
$attachment = get_post( $id );
|
|
|
|
if ( $attachment ) {
|
|
$image_data = [
|
|
'alt' => get_post_meta( $attachment->ID, '_wp_attachment_image_alt', true ),
|
|
'caption' => $attachment->post_excerpt,
|
|
'description' => $attachment->post_content,
|
|
'title' => $attachment->post_title,
|
|
];
|
|
|
|
if ( $lightbox_title_src && $image_data[ $lightbox_title_src ] ) {
|
|
$attributes['title'] = $image_data[ $lightbox_title_src ];
|
|
}
|
|
|
|
if ( $lightbox_description_src && $image_data[ $lightbox_description_src ] ) {
|
|
$attributes['description'] = $image_data[ $lightbox_description_src ];
|
|
}
|
|
}
|
|
|
|
return $attributes;
|
|
}
|
|
|
|
private function delete_custom_images( $post_id ) {
|
|
$image_meta = wp_get_attachment_metadata( $post_id );
|
|
if ( ! empty( $image_meta ) && ! empty( $image_meta['sizes'] ) ) {
|
|
( new Collection( $image_meta['sizes'] ) )
|
|
->filter( function ( $value, $key ) {
|
|
return ( 0 === strpos( $key, 'elementor_custom_' ) );
|
|
} )
|
|
->pluck( 'file' )
|
|
->each( function ( $path ) {
|
|
$base_dir = wp_get_upload_dir()['basedir'];
|
|
wp_delete_file( $base_dir . '/' . $path );
|
|
} );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Images manager constructor.
|
|
*
|
|
* Initializing Elementor images manager.
|
|
*
|
|
* @since 1.0.0
|
|
* @access public
|
|
*/
|
|
public function __construct() {
|
|
add_action( 'wp_ajax_elementor_get_images_details', [ $this, 'get_images_details' ] );
|
|
|
|
// Delete elementor thumbnail files on deleting its main image.
|
|
add_action( 'delete_attachment', function ( $post_id ) {
|
|
$this->delete_custom_images( $post_id );
|
|
} );
|
|
}
|
|
}
|