<?php
/**
* @package VikBooking
* @subpackage core
* @author Alessio Gaggii - E4J s.r.l.
* @copyright Copyright (C) 2022 E4J s.r.l. All Rights Reserved.
* @license http://www.gnu.org/licenses/gpl-2.0.html GNU/GPL
* @link https://vikwp.com
*/
// No direct access
defined('ABSPATH') or die('No script kiddies please!');
/**
* Helper class to handle rooms data.
*
* @since 1.15.1 (J) - 1.5.2 (WP)
*/
final class VBORoomHelper extends JObject
{
/**
* The singleton instance of the class.
*
* @var VBORoomHelper
*/
private static $instance = null;
/**
* Proxy to construct the object.
*
* @param array|object $data optional data to bind.
* @param boolean $anew true for forcing a new instance.
*
* @return self
*/
public static function getInstance($data = [], $anew = false)
{
if (is_null(static::$instance) || $anew) {
static::$instance = new static($data);
}
return static::$instance;
}
/**
* Checks whether a room has been configured with LOS pricing rules.
* VCM comes with a similar built-in method, but we need this feature
* to be available also for those who only use VBO. Moreover, this method
* can identify the first night with a non-proportional rate.
*
* @param int $idroom the ID of the room in VBO.
* @param int $idprice the optional rate plan ID in VBO.
* @param bool $get_nights whether to return the number of nights when LOS starts.
*
* @return bool|int false on failure or if no LOS prices found, true or int otherwise.
*/
public static function hasLosRecords($idroom, $idprice = 0, $get_nights = false)
{
if (empty($idroom)) {
return false;
}
$dbo = JFactory::getDbo();
$q = "SELECT * FROM `#__vikbooking_dispcost` WHERE `idroom`=" . (int)$idroom . (!empty($idprice) ? " AND `idprice`=" . (int)$idprice : '') . " ORDER BY `days` ASC;";
$dbo->setQuery($q);
$los_data = $dbo->loadAssocList();
if (!$los_data) {
return false;
}
$los_pricing = array();
foreach ($los_data as $cost) {
if (!isset($los_pricing[$cost['days']])) {
$los_pricing[$cost['days']] = array();
}
array_push($los_pricing[$cost['days']], $cost);
}
// sort by number of nights
ksort($los_pricing);
// compose lowest costs per rate plan
$base_costs = array();
foreach ($los_pricing as $nights => $costs) {
foreach ($costs as $rplan_cost) {
$base_costs[$rplan_cost['idprice']] = ($rplan_cost['cost'] / $rplan_cost['days']);
}
// we take the costs for the lowest number of nights
break;
}
// check if rates change depending on the number of nights of stay
foreach ($los_pricing as $nights => $costs) {
foreach ($costs as $rplan_cost) {
$base_cost = ($rplan_cost['cost'] / $rplan_cost['days']);
if (isset($base_costs[$rplan_cost['idprice']]) && round($base_costs[$rplan_cost['idprice']], 2) != round($base_cost, 2)) {
/**
* Average rates should be compared after applying rounding or we may face issues.
* For example, 383.97 / 3 = 127.99, but it's actually = 127.99000000000001 with
* an absolute number for the difference with 127.99 of 1.4210854715202004E-14
* which results to be greater than 0 but less than 1. Therefore, we also allow
* an absolute number for the difference of 0.05 cents for a proper check.
*/
$price_diff = abs($base_costs[$rplan_cost['idprice']] - $base_cost);
if ($price_diff > 0.05) {
// this is a non-proportional cost per night, so LOS records have been defined
return $get_nights ? $nights : true;
}
}
}
}
// all costs per night were proportional
return false;
}
/**
* Calculates the effective Min LOS from the Rates Table
* for the given room and rate plan ID.
*
* @param int $idroom the ID of the room-type on the website.
* @param int $idprice the ID of the rate plan on the website.
*
* @return int the effective Min LOS or 0.
*
* @since 1.18.0 (J) - 1.8.0 (WP)
*/
public static function calcEffectiveMinLOS($idroom, $idprice)
{
if (empty($idroom) || empty($idprice)) {
return 0;
}
$dbo = JFactory::getDbo();
$dbo->setQuery(
$dbo->getQuery(true)
->select('MIN(' . $dbo->qn('days') . ')')
->from($dbo->qn('#__vikbooking_dispcost'))
->where($dbo->qn('idroom') . ' = ' . (int) $idroom)
->where($dbo->qn('idprice') . ' = ' . (int) $idprice)
, 0, 1);
return (int) $dbo->loadResult();
}
/**
* Calculates the effective Max LOS from the Rates Table
* for the given room and rate plan ID.
*
* @param int $idroom the ID of the room-type on the website.
* @param int $idprice the ID of the rate plan on the website.
*
* @return int the effective Max LOS or 0.
*
* @since 1.18.0 (J) - 1.8.0 (WP)
*/
public static function calcEffectiveMaxLOS($idroom, $idprice)
{
if (empty($idroom) || empty($idprice)) {
return 0;
}
$dbo = JFactory::getDbo();
$dbo->setQuery(
$dbo->getQuery(true)
->select('MAX(' . $dbo->qn('days') . ')')
->from($dbo->qn('#__vikbooking_dispcost'))
->where($dbo->qn('idroom') . ' = ' . (int) $idroom)
->where($dbo->qn('idprice') . ' = ' . (int) $idprice)
, 0, 1);
return (int) $dbo->loadResult();
}
/**
* Gets the available room upgrade options, if any.
*
* @param VikBookingTranslator $vbo_tn the translator object.
*
* @return array list of available upgrade options,
* or empty array if nothing availabe.
*
* @since 1.16.0 (J) - 1.6.0 (WP)
*/
public function getUpgradeOptions($vbo_tn = null)
{
$booking = $this->get('booking', []);
$rooms = $this->get('rooms', []);
if (!$booking || !$rooms || $booking['status'] != 'confirmed') {
return [];
}
$dbo = JFactory::getDbo();
$config = VBOFactory::getConfig();
$upgrade_options = [];
$room_ids = [];
foreach ($rooms as $num => $broom) {
if (empty($broom['idroom']) || empty($broom['idtar'])) {
// room must have a valid tariff assigned
continue;
}
$room_upgrade_options = $config->getArray('room_upgrade_options_' . $broom['idroom'], []);
if (empty($room_upgrade_options) || empty($room_upgrade_options['rooms'])) {
// no relations for this room
continue;
}
// fetch the original tariff for this room
$orig_tariff = $this->getTariffData($broom['idtar']);
if (!$orig_tariff) {
// unable to get the original tariff information for this room booked
continue;
}
// push suitable rooms
$upgrade_options[$num] = [
'rooms' => array_map('intval', array_filter(array_unique($room_upgrade_options['rooms']))),
'discount' => (!empty($room_upgrade_options['discount']) ? (float)$room_upgrade_options['discount'] : 0),
'tariff' => $orig_tariff,
'r_costs' => [],
];
$room_ids = array_merge($room_ids, $room_upgrade_options['rooms']);
}
if (!$upgrade_options) {
return [];
}
// get all room IDs involved
$room_ids = array_map('intval', array_filter(array_unique($room_ids)));
$q = "SELECT * FROM `#__vikbooking_rooms` WHERE `id` IN (" . implode(', ', $room_ids) . ") AND `avail`=1;";
$dbo->setQuery($q);
$room_records = $dbo->loadAssocList();
if (!$room_records) {
return [];
}
if ($vbo_tn) {
// translate rooms
$vbo_tn->translateContents($room_records, '#__vikbooking_rooms');
}
// build up an associative array of room infos
$room_infos = [];
foreach ($room_records as $room_record) {
$room_infos[$room_record['id']] = $this->prepareCMSContents($room_record, ['info', 'smalldesc']);
}
unset($room_records);
// keep the count of the room units suggested
$room_units_counter = [];
// filter the suitable rooms by rate plan, and calculate the costs
foreach ($upgrade_options as $num => $upgrade_option) {
// build the costs for each upgrade room option
$upgrade_room_costs = [];
// parse all rooms compatible
foreach ($upgrade_option['rooms'] as $rkey => $rid) {
// find the same tariff for this room and nights
$room_same_tariff = $this->findTariff($rid, $upgrade_option['tariff']['days'], $upgrade_option['tariff']['idprice']);
if (!$room_same_tariff || !isset($room_infos[$rid])) {
// this room is not suited
unset($upgrade_options[$num]['rooms'][$rkey]);
continue;
}
// count the actual number of room remaining units
$use_room_units = $room_infos[$rid]['units'];
if (isset($room_units_counter[$rid])) {
$use_room_units -= $room_units_counter[$rid];
}
// make sure the room is bookable on these dates (restrictions are ignored)
if (!VikBooking::roomBookable($rid, $use_room_units, $booking['checkin'], $booking['checkout'])) {
// room is not available for upgrade
unset($upgrade_options[$num]['rooms'][$rkey]);
continue;
}
// update room units counter
if (!isset($room_units_counter[$rid])) {
$room_units_counter[$rid] = 0;
}
$room_units_counter[$rid]++;
// apply seasonal rates
$tar = VikBooking::applySeasonsRoom([$room_same_tariff], $booking['checkin'], $booking['checkout']);
// apply OBP rules
$tar = $this->applyOBPRules($tar, $room_infos[$rid], $rooms[$num]['adults']);
// apply upgrade discount (if any) and calculate upgrade cost
foreach ($tar as $tk => $tv) {
$tar[$tk]['upgrade_cost'] = $upgrade_option['discount'] > 0 ? round(($tv['cost'] * (100 - $upgrade_option['discount']) / 100), 2) : $tv['cost'];
}
// push room tariff (just one rate plan, the originally booked one)
$upgrade_room_costs[$rid] = $tar[0];
}
if (!count($upgrade_options[$num]['rooms'])) {
// no more suitable rooms
unset($upgrade_options[$num]);
continue;
}
// sort by price descending (most expensive on top)
$sort_map = [];
foreach ($upgrade_room_costs as $rid => $tar) {
$sort_map[$rid] = $tar['upgrade_cost'];
}
arsort($sort_map);
// replace values with sorted ordering
$cp_upgrade_room_costs = [];
foreach ($sort_map as $rid => $sorted) {
$cp_upgrade_room_costs[$rid] = $upgrade_room_costs[$rid];
}
$upgrade_room_costs = $cp_upgrade_room_costs;
// set upgrade room costs
$upgrade_options[$num]['r_costs'] = $upgrade_room_costs;
}
if (!count($upgrade_options)) {
return [];
}
// return the associative array information
return [
'upgrade' => $upgrade_options,
'rooms' => $room_infos,
];
}
/**
* Gets the record details about a specific tariff ID.
*
* @param int $idtar the ID of the room-tariff.
*
* @return array record found, or empty array.
*
* @since 1.16.0 (J) - 1.6.0 (WP)
*/
public function getTariffData($idtar)
{
$dbo = JFactory::getDbo();
$dbo->setQuery("SELECT * FROM `#__vikbooking_dispcost` WHERE `id` = " . (int)$idtar, 0, 1);
$tariff = $dbo->loadAssoc();
if (!$tariff) {
return [];
}
return $tariff;
}
/**
* Finds a tariff for the given rate plan ID, room and nights.
*
* @param int $rid the room ID.
* @param int $nights the number of nights of stay.
* @param int $idprice the rate plan ID.
*
* @return array record found or empty array.
*
* @since 1.16.0 (J) - 1.6.0 (WP)
*/
public function findTariff($rid, $nights, $idprice)
{
$dbo = JFactory::getDbo();
$q = "SELECT `t`.*, `p`.`name` AS `rate_plan_name` FROM `#__vikbooking_dispcost` AS `t`
LEFT JOIN `#__vikbooking_prices` AS `p` ON `t`.`idprice`=`p`.`id`
WHERE `t`.`idroom` = " . (int)$rid . " AND `t`.`days`=" . (int)$nights . " AND `t`.`idprice`=" . (int)$idprice;
$dbo->setQuery($q, 0, 1);
$tariff = $dbo->loadAssoc();
if (!$tariff) {
return [];
}
return $tariff;
}
/**
* Applies the OBP rules over an array of tariffs.
*
* @param array $tar list of tariff records, one per rate plan, after seasonal rates.
* @param array $room the room (or order-room) record for which tariffs where loaded.
* @param int $adults the number of adults to consider.
*
* @return array original tariffs array with OBP costs applied.
*
* @since 1.16.0 (J) - 1.6.0 (WP)
*/
public function applyOBPRules(array $tar, array $room, $adults = 2)
{
// check for different usage
if (!isset($room['fromadult']) || $room['fromadult'] > $adults || $room['toadult'] < $adults) {
return $tar;
}
// check for room ID
$use_room_id = isset($room['idroom']) ? $room['idroom'] : $room['id'];
// different usage
$diffusageprice = VikBooking::loadAdultsDiff($use_room_id, $adults);
/**
* Memorize immediately the OBP rules defined at room-level in order to avoid conflicts
* with rate plans with and without OBP overrides defined at rate plan level through SP.
*/
$orig_diffusage = $diffusageprice;
// occupancy overrides
$occ_ovr = VikBooking::occupancyOverrideExists($tar, $adults);
$diffusageprice = $occ_ovr !== false ? $occ_ovr : $diffusageprice;
if (!$diffusageprice) {
return $tar;
}
// set a charge or discount to the price(s) for the different usage of the room
foreach ($tar as $kpr => $vpr) {
// occupancy override
$diffusageprice = isset($vpr['occupancy_ovr']) && isset($vpr['occupancy_ovr'][$adults]) ? $vpr['occupancy_ovr'][$adults] : $orig_diffusage;
// set usage of the room
$tar[$kpr]['diffusage'] = $adults;
if ($diffusageprice['chdisc'] == 1) {
// charge
if ($diffusageprice['valpcent'] == 1) {
// fixed value
$tar[$kpr]['diffusagecostpernight'] = $diffusageprice['pernight'] == 1 ? 1 : 0;
$aduseval = $diffusageprice['pernight'] == 1 ? $diffusageprice['value'] * $tar[$kpr]['days'] : $diffusageprice['value'];
$tar[$kpr]['diffusagecost'] = "+" . $aduseval;
$tar[$kpr]['room_base_cost'] = $vpr['cost'];
$tar[$kpr]['cost'] = $vpr['cost'] + $aduseval;
} else {
// percentage value
$tar[$kpr]['diffusagecostpernight'] = $diffusageprice['pernight'] == 1 ? $vpr['cost'] : 0;
$aduseval = $diffusageprice['pernight'] == 1 ? round(($vpr['cost'] * $diffusageprice['value'] / 100) * $tar[$kpr]['days'] + $vpr['cost'], 2) : round(($vpr['cost'] * (100 + $diffusageprice['value']) / 100), 2);
$tar[$kpr]['diffusagecost'] = "+" . $diffusageprice['value'] . "%";
$tar[$kpr]['room_base_cost'] = $vpr['cost'];
$tar[$kpr]['cost'] = $aduseval;
}
} else {
// discount
if ($diffusageprice['valpcent'] == 1) {
// fixed value
$tar[$kpr]['diffusagecostpernight'] = $diffusageprice['pernight'] == 1 ? 1 : 0;
$aduseval = $diffusageprice['pernight'] == 1 ? $diffusageprice['value'] * $tar[$kpr]['days'] : $diffusageprice['value'];
$tar[$kpr]['diffusagecost'] = "-" . $aduseval;
$tar[$kpr]['room_base_cost'] = $vpr['cost'];
$tar[$kpr]['cost'] = $vpr['cost'] - $aduseval;
} else {
// percentage value
$tar[$kpr]['diffusagecostpernight'] = $diffusageprice['pernight'] == 1 ? $vpr['cost'] : 0;
$aduseval = $diffusageprice['pernight'] == 1 ? round($vpr['cost'] - ((($vpr['cost'] / $tar[$kpr]['days']) * $diffusageprice['value'] / 100) * $tar[$kpr]['days']), 2) : round(($vpr['cost'] * (100 - $diffusageprice['value']) / 100), 2);
$tar[$kpr]['diffusagecost'] = "-" . $diffusageprice['value'] . "%";
$tar[$kpr]['room_base_cost'] = $vpr['cost'];
$tar[$kpr]['cost'] = $aduseval;
}
}
}
// return the array of tariffs with OBP included
return $tar;
}
/**
* Prepares some description strings for the current CMS, by triggering
* the necessary platform-related functions for third party plugins.
*
* @param array $room_record the room record to prepare.
* @param array $keys list of record keys to prepare.
*
* @return array the original array given with keys prepared.
*
* @since 1.16.0 (J) - 1.6.0 (WP)
*/
public function prepareCMSContents(array $room_record, array $keys)
{
foreach ($keys as $key) {
if (!isset($room_record[$key])) {
continue;
}
if (VBOPlatformDetection::isWordPress()) {
/**
* @wponly we try to parse any shortcode inside the HTML description of the room
*/
$room_record[$key] = do_shortcode(wpautop($room_record[$key]));
} else {
// BEGIN: Joomla Content Plugins Rendering
JPluginHelper::importPlugin('content');
$myItem = JTable::getInstance('content');
$myItem->text = $room_record[$key];
$objparams = array();
if (class_exists('JEventDispatcher')) {
$dispatcher = JEventDispatcher::getInstance();
$dispatcher->trigger('onContentPrepare', array('com_vikbooking.roomdetails', &$myItem, &$objparams, 0));
} else {
/**
* @joomla4only
*/
$dispatcher = JFactory::getApplication();
if (method_exists($dispatcher, 'triggerEvent')) {
$dispatcher->triggerEvent('onContentPrepare', array('com_vikbooking.roomdetails', &$myItem, &$objparams, 0));
}
}
$room_record[$key] = $myItem->text;
// END: Joomla Content Plugins Rendering
}
}
return $room_record;
}
/**
* Gets an associative list of rate plans with a few pricing information for the given room.
*
* @param int $rid the VBO room id.
* @param int $rplan_id optional rate plan ID to get.
*
* @return array associative list of rate plans for the given room or specific rate plan.
*
* @since 1.16.3 (J) - 1.6.3 (WP)
*/
public function getRatePlans($rid = 0, $rplan_id = 0)
{
if (empty($rid)) {
$rid = $this->get('id', 0);
}
$dbo = JFactory::getDbo();
$q = $dbo->getQuery(true)
->select([
$dbo->qn('r.id'),
$dbo->qn('r.idroom'),
$dbo->qn('r.days'),
$dbo->qn('r.idprice'),
$dbo->qn('r.cost'),
$dbo->qn('p.name'),
$dbo->qn('p.minlos'),
$dbo->qn('p.derived_id'),
])
->from($dbo->qn('#__vikbooking_dispcost', 'r'))
->leftJoin($dbo->qn('#__vikbooking_prices', 'p') . ' ON ' . $dbo->qn('r.idprice') . ' = ' . $dbo->qn('p.id'))
->where($dbo->qn('r.idroom') . ' = ' . (int)$rid)
->order($dbo->qn('r.days') . ' ASC')
->order($dbo->qn('r.cost') . ' ASC');
$dbo->setQuery($q, 0, 50);
$tariffs = $dbo->loadObjectList();
if (!$tariffs) {
return [];
}
$parsed_room_prices = [];
foreach ($tariffs as $rrk => $rrv) {
if (isset($parsed_room_prices[$rrv->idprice])) {
unset($tariffs[$rrk]);
continue;
}
$tariffs[$rrk]->cost = round(($rrv->cost / $rrv->days), 2);
$tariffs[$rrk]->days = 1;
$parsed_room_prices[$rrv->idprice] = 1;
}
$tariffs = array_values($tariffs);
$room_rate_plans = [];
foreach ($tariffs as $rplan) {
if ($rplan_id && $rplan_id == $rplan->idprice) {
return (array)$rplan;
}
$room_rate_plans[] = [
'id' => $rplan->idprice,
'name' => $rplan->name,
'cost' => $rplan->cost,
'minlos' => $rplan->minlos,
'derived_id' => $rplan->derived_id,
];
}
if ($rplan_id) {
return [];
}
return $room_rate_plans;
}
/**
* Calculates if the provided booking information require a split payment for the damage deposit.
*
* @param array $booking The booking record data.
* @param array $booking_rooms The rooms booking list.
*
* @return array Associative list of damage deposit details.
*
* @since 1.17.6 (J) - 1.7.6 (WP)
*/
public function getDamageDepositSplitPayment(array $booking, array $booking_rooms)
{
// load all option records of type damage deposit
$dbo = JFactory::getDbo();
$dbo->setQuery(
$dbo->getQuery(true)
->select('*')
->from($dbo->qn('#__vikbooking_optionals'))
->where($dbo->qn('forcesel') . ' = 1')
->where($dbo->qn('oparams') . ' LIKE ' . $dbo->q('%' . str_replace(['{', '}'], '', json_encode(['damagedep' => 1])) . '%'))
);
$dd_records = $dbo->loadAssocList();
// scan all records for validation, if any
foreach ($dd_records as &$dd_record) {
// make sure to decode the option params
$dd_record['oparams'] = (array) json_decode($dd_record['oparams'], true);
if (empty($dd_record['oparams']['damagedep_settings']['paywhen'])) {
// no separate payment defined
unset($dd_record);
continue;
}
// validate maximum nights of stay
if (!empty($dd_record['oparams']['damagedep_settings']['bmaxlos']) && ($booking['days'] ?? 1) > $dd_record['oparams']['damagedep_settings']['bmaxlos']) {
// limit exceeded
unset($dd_record);
continue;
}
// validate payment method ID
if (empty($dd_record['oparams']['damagedep_settings']['payid']) && empty($booking['idpayment'])) {
// no payment method defined anywhere
unset($dd_record);
continue;
}
// calculate and set the payment window values
$dd_record['payment_window'] = [];
if (!strlen((string) $dd_record['oparams']['damagedep_settings']['paywind'])) {
// payable from today (always)
$dd_record['payment_window']['payment_from_dt'] = date('Y-m-d');
$dd_record['payment_window']['payable'] = true;
} elseif (empty($dd_record['oparams']['damagedep_settings']['paywind'])) {
// payable from the check-in day
$dd_record['payment_window']['payment_from_dt'] = date('Y-m-d', $booking['checkin']);
$dd_record['payment_window']['payable'] = strtotime($dd_record['payment_window']['payment_from_dt']) <= strtotime(date('Y-m-d'));
} else {
// calculate the payable date
$window_days = (int) $dd_record['oparams']['damagedep_settings']['paywind'];
$dd_record['payment_window']['payment_from_dt'] = date('Y-m-d', strtotime(sprintf('-%d days', $window_days), $booking['checkin']));
$dd_record['payment_window']['payable'] = strtotime($dd_record['payment_window']['payment_from_dt']) <= strtotime(date('Y-m-d'));
}
// check if a custom payment method should be used
if (!empty($dd_record['oparams']['damagedep_settings']['payid'])) {
$dd_record['payment_window']['pay_id'] = $dd_record['oparams']['damagedep_settings']['payid'];
}
// ensure damage deposit amount was not paid already
if (empty($booking['idorderota']) && !empty($booking['totpaid']) && $booking['totpaid'] >= ($booking['total'] ?? 0)) {
// payment window not available because damage deposit already paid
$dd_record['payment_window'] = [];
}
}
// unset last reference
unset($dd_record);
if (!$dd_records) {
// unable to proceed
return [];
}
// always reset array keys
$dd_records = array_values($dd_records);
// list of damage deposit option IDs
$dd_record_ids = array_column($dd_records, 'id');
// room reservation IDs affected
$room_reservation_dd = [];
// collect all damage deposit options from the booked rooms
$rooms_dd_data = [];
foreach ($booking_rooms as $or) {
if (empty($or['optionals'])) {
continue;
}
$stepo = array_filter(explode(";", $or['optionals']));
foreach ($stepo as $roptkey => $one) {
$stept = explode(":", $one);
if (in_array($stept[0], $dd_record_ids)) {
// push damage deposit ID and room record
$rooms_dd_data[] = [
'dd_id' => $stept[0],
'rr' => $or,
];
// push room reservation ID
$room_reservation_dd[] = $or['idroom'] ?? 0;
}
}
}
if (!$rooms_dd_data) {
// no damage deposit options were booked
return [];
}
// get the unique array
$rooms_dd_unique = array_values(array_unique(array_column($rooms_dd_data, 'dd_id')));
// turn the records into an associative list
$dd_records_assoc = [];
foreach ($dd_records as $dd_record) {
$dd_records_assoc[$dd_record['id']] = $dd_record;
}
// calculate amounts and damage deposit payment window
$tot_dd_amount_gross = 0;
$tot_dd_amount_net = 0;
$tot_dd_amount_tax = 0;
$payment_window = [];
foreach ($rooms_dd_data as $room_dd_data) {
$opt_id = $room_dd_data['dd_id'];
if (!($dd_records_assoc[$opt_id] ?? [])) {
continue;
}
// calculate damage deposit price
$dd_price = (float) $dd_records_assoc[$opt_id]['cost'];
if (!empty($dd_records_assoc[$opt_id]['pcentroom'])) {
// percent cost of the room reservation
$room_cost = ($room_dd_data['rr']['room_cost'] ?? 0) ?: ($room_dd_data['rr']['cust_cost'] ?? 0) ?: 0;
$dd_price = $room_cost * $dd_price / 100;
}
if ($dd_price <= 0) {
// invalid damage deposit cost
continue;
}
if ($dd_records_assoc[$opt_id]['perday'] == 1) {
// cost per night
$dd_price = $dd_price * ($booking['days'] ?? 1);
}
if (($dd_records_assoc[$opt_id]['maxprice'] ?? 0) > 0 && $dd_price > $dd_records_assoc[$opt_id]['maxprice']) {
// maximum cost
$dd_price = (float) $dd_records_assoc[$opt_id]['maxprice'];
}
if ($dd_records_assoc[$opt_id]['perperson'] == 1) {
// cost per person
$dd_price = $dd_price * ((int) $room_dd_data['rr']['adults']);
}
/**
* Trigger event to allow third party plugins to apply a custom calculation for the option/extra fee or tax.
*
* @since 1.17.7 (J) - 1.7.7 (WP)
*/
$custom_calculation = VBOFactory::getPlatform()->getDispatcher()->filter('onCalculateBookingOptionFeeCost', [$dd_price, &$dd_records_assoc[$opt_id], $booking, $booking_rooms]);
if ($custom_calculation) {
$dd_price = (float) $custom_calculation[0];
}
if ($dd_price <= 0) {
// invalid damage deposit cost
continue;
}
// calculate taxes, if any
$dd_amount_gross = VikBooking::sayOptionalsPlusIva($dd_price, $dd_records_assoc[$opt_id]['idiva']);
$dd_amount_net = VikBooking::sayOptionalsMinusIva($dd_price, $dd_records_assoc[$opt_id]['idiva']);
$dd_amount_tax = $dd_amount_gross - $dd_amount_net;
// increase global values
$tot_dd_amount_gross += $dd_amount_gross;
$tot_dd_amount_net += $dd_amount_net;
$tot_dd_amount_tax += $dd_amount_tax;
// update payment window (one for all option records)
$payment_window = (array) $dd_records_assoc[$opt_id]['payment_window'];
}
if (!$tot_dd_amount_gross) {
// no compliant damage deposit option found for separate payment
return [];
}
return [
'damagedep_gross' => $tot_dd_amount_gross,
'damagedep_net' => $tot_dd_amount_net,
'damagedep_tax' => $tot_dd_amount_tax,
'payment_window' => $payment_window,
'damagedep_rids' => $room_reservation_dd,
];
}
/**
* Given a list of room records, returns an associative list
* of room IDs and corresponding mini thumbnail URLs, if any.
*
* @param array $rooms List of room records.
*
* @return array
*
* @since 1.17.6 (J) - 1.7.6 (WP)
*/
public function loadMiniThumbnails(array $rooms, string $def_uri = '')
{
$mini_thumbnails = [];
$base_img_path = implode(DIRECTORY_SEPARATOR, [VBO_SITE_PATH, 'resources', 'uploads']) . DIRECTORY_SEPARATOR;
$base_img_uri = VBO_SITE_URI . 'resources/uploads/';
foreach ($rooms as $room) {
if (empty($room['id'])) {
continue;
}
if (!empty($room['img']) && is_file($base_img_path . 'mini_' . $room['img'])) {
$mini_thumbnails[$room['id']] = $base_img_uri . 'mini_' . $room['img'];
} elseif ($def_uri) {
$mini_thumbnails[$room['id']] = $def_uri;
}
}
return $mini_thumbnails;
}
}