File "es_guardia_civil.php"
Full Path: /home/romayxjt/public_html/wp-content/plugins/vikbooking/admin/helpers/report/es_guardia_civil.php
File size: 44.13 KB
MIME-type: text/x-php
Charset: utf-8
<?php
/**
* @package VikBooking
* @subpackage com_vikbooking
* @author E4J srl
* @copyright Copyright (C) 2020 E4J srl. All rights reserved.
* @license GNU General Public License version 2 or later; see LICENSE
* @link https://vikwp.com
*/
defined('ABSPATH') or die('No script kiddies please!');
class VikBookingReportEsGuardiaCivil extends VikBookingReport
{
/**
* Property 'defaultKeySort' is used by the View that renders the report.
*/
public $defaultKeySort = 'idbooking';
/**
* Property 'defaultKeyOrder' is used by the View that renders the report.
*/
public $defaultKeyOrder = 'ASC';
/**
* Property 'customExport' is used by the View to display custom export buttons.
*/
public $customExport = '';
/**
* Debug mode is activated by passing the value 'e4j_debug' > 0
*/
private $debug;
/**
* Other private vars of this sub-class.
*/
private $nazioni = [];
private $documenti = [];
/**
* Class constructor should define the name of the report and
* other vars. Call the parent constructor to define the DB object.
*/
function __construct()
{
$this->reportFile = basename(__FILE__, '.php');
$this->reportName = "Guardia Civil";
$this->reportFilters = [];
$this->cols = [];
$this->rows = [];
$this->footerRow = [];
$this->debug = (VikRequest::getInt('e4j_debug', 0, 'request') > 0);
$this->registerExportFileName();
parent::__construct();
}
/**
* Returns the name of this report.
*
* @return string
*/
public function getName()
{
return $this->reportName;
}
/**
* Returns the name of this file without .php.
*
* @return string
*/
public function getFileName()
{
return $this->reportFile;
}
/**
* Returns the filters of this report.
*
* @return array
*/
public function getFilters()
{
if (count($this->reportFilters)) {
//do not run this method twice, as it could load JS and CSS files.
return $this->reportFilters;
}
//get VBO Application Object
$vbo_app = VikBooking::getVboApplication();
//load the jQuery UI Datepicker
$this->loadDatePicker();
//custom export button
$this->customExport = '<a href="JavaScript: void(0);" onclick="vboDownloadSchedaGuardiaCivil();" class="vbcsvexport"><i class="'.VikBookingIcons::i('download').'"></i> <span>Download File</span></a>';
//build the hidden values for the selection of Comuni & Province.
$this->nazioni = $this->loadNazioni();
$this->documenti = $this->loadDocumenti();
$hidden_vals = '<div id="vbo-report-guardiacivil-hidden" style="display: none;">';
//Nazioni
$hidden_vals .= ' <div id="vbo-report-guardiacivil-nazione" class="vbo-report-guardiacivil-selcont" style="display: none;">';
$hidden_vals .= ' <select id="choose-nazione" onchange="vboReportChosenNazione(this);"><option value=""></option>';
if (count($this->nazioni)) {
foreach ($this->nazioni as $code => $nazione) {
$hidden_vals .= ' <option value="' . $nazione['name'] . '" data-threecode="' . $nazione['three_code'] . '">' . $nazione['name'] . '</option>'."\n";
}
}
$hidden_vals .= ' </select>';
$hidden_vals .= ' </div>';
//Documenti
$hidden_vals .= ' <div id="vbo-report-guardiacivil-doctype" class="vbo-report-guardiacivil-selcont" style="display: none;">';
$hidden_vals .= ' <select id="choose-documento" onchange="vboReportChosenDocumento(this);"><option value=""></option>';
if (count($this->documenti)) {
foreach ($this->documenti as $code => $documento) {
$hidden_vals .= ' <option value="'.$code.'">'.$documento.'</option>'."\n";
}
}
$hidden_vals .= ' </select>';
$hidden_vals .= ' </div>';
//Sesso
$hidden_vals .= ' <div id="vbo-report-guardiacivil-sesso" class="vbo-report-guardiacivil-selcont" style="display: none;">';
$hidden_vals .= ' <select id="choose-sesso" onchange="vboReportChosenSesso(this);"><option value=""></option>';
$sessos = array(
1 => 'M',
2 => 'F'
);
foreach ($sessos as $code => $ses) {
$hidden_vals .= ' <option value="'.$code.'">'.$ses.'</option>'."\n";
}
$hidden_vals .= ' </select>';
$hidden_vals .= ' </div>';
//Numero Documento
$hidden_vals .= ' <div id="vbo-report-guardiacivil-docnum" class="vbo-report-guardiacivil-selcont" style="display: none;">';
$hidden_vals .= ' <input type="text" size="40" id="choose-docnum" placeholder="Número del Documento..." value="" /><br/>';
$hidden_vals .= ' <button type="button" class="btn" onclick="vboReportChosenDocnum(document.getElementById(\'choose-docnum\').value);">'.JText::translate('VBAPPLY').'</button>';
$hidden_vals .= ' </div>';
//Data di Nascita
$hidden_vals .= ' <div id="vbo-report-guardiacivil-dbirth" class="vbo-report-guardiacivil-selcont" style="display: none;">';
$hidden_vals .= ' <input type="text" size="40" id="choose-dbirth" placeholder="'.JText::translate('VBCUSTOMERBDATE').'" value="" /><br/>';
$hidden_vals .= ' <button type="button" class="btn" onclick="vboReportChosenDbirth(document.getElementById(\'choose-dbirth\').value);">'.JText::translate('VBAPPLY').'</button>';
$hidden_vals .= ' </div>';
//Data di Emissione documento
$hidden_vals .= ' <div id="vbo-report-guardiacivil-docissuedt" class="vbo-report-guardiacivil-selcont" style="display: none;">';
$hidden_vals .= ' <input type="text" size="40" id="choose-docissuedt" placeholder="Fecha de expedición del documento" value="" /><br/>';
$hidden_vals .= ' <button type="button" class="btn" onclick="vboReportChosenDocissuedt(document.getElementById(\'choose-docissuedt\').value);">'.JText::translate('VBAPPLY').'</button>';
$hidden_vals .= ' </div>';
$hidden_vals .= '</div>';
//From Date Filter (plus hidden values for the dropdown menus of custom values)
$filter_opt = array(
'label' => '<label for="fromdate">'.JText::translate('VBOREPORTSDATEFROM').'</label>',
'html' => '<input type="text" id="fromdate" name="fromdate" value="" class="vbo-report-datepicker vbo-report-datepicker-from" />'.$hidden_vals,
'type' => 'calendar',
'name' => 'fromdate'
);
array_push($this->reportFilters, $filter_opt);
//To Date Filter
$filter_opt = array(
'label' => '<label for="todate">'.JText::translate('VBOREPORTSDATETO').'</label>',
'html' => '<input type="text" id="todate" name="todate" value="" class="vbo-report-datepicker vbo-report-datepicker-to" />',
'type' => 'calendar',
'name' => 'todate'
);
array_push($this->reportFilters, $filter_opt);
// Filtro Nome Hotel
$photelname = VikRequest::getString('hotelname', '', 'request');
$filter_opt = array(
'label' => '<label for="hotelname">Nombre del hotel</label>',
'html' => '<input type="text" id="hotelname" name="hotelname" value="'.$photelname.'" size="10" />',
'type' => 'text',
'name' => 'hotelname'
);
array_push($this->reportFilters, $filter_opt);
// Filtro Codice Hotel
$photelcode = VikRequest::getString('hotelcode', '', 'request');
$filter_opt = array(
'label' => '<label for="hotelcode">Código del hotel</label>',
'html' => '<input type="text" id="hotelcode" name="hotelcode" value="'.$photelcode.'" size="10" />',
'type' => 'text',
'name' => 'hotelcode'
);
array_push($this->reportFilters, $filter_opt);
// append button to save the data when creating manual values
$filter_opt = array(
'label' => '<label class="vbo-report-guardiacivil-manualsave" style="display: none;">Data</label>',
'html' => '<button type="button" class="btn vbo-config-btn vbo-report-guardiacivil-manualsave" style="display: none;" onclick="vboGuardiacivilSaveData();"><i class="' . VikBookingIcons::i('save') . '"></i> ' . JText::translate('VBSAVE') . '</button>',
);
array_push($this->reportFilters, $filter_opt);
//jQuery code for the datepicker calendars, select2 and triggers for the dropdown menus
$pfromdate = VikRequest::getString('fromdate', '', 'request');
$ptodate = VikRequest::getString('todate', '', 'request');
$js = 'var reportActiveCell = null, reportObj = {};
var vbo_guardiacivil_ajax_uri = "' . VikBooking::ajaxUrl('index.php?option=com_vikbooking&task=invoke_report&report=' . $this->reportFile) . '";
var vbo_guardiacivil_save_icn = "' . VikBookingIcons::i('save') . '";
var vbo_guardiacivil_saving_icn = "' . VikBookingIcons::i('circle-notch', 'fa-spin fa-fw') . '";
var vbo_guardiacivil_saved_icn = "' . VikBookingIcons::i('check-circle') . '";
jQuery(function() {
//prepare main filters
jQuery(".vbo-report-datepicker:input").datepicker({
maxDate: 0,
dateFormat: "'.$this->getDateFormat('jui').'",
onSelect: vboReportCheckDates
});
'.(!empty($pfromdate) ? 'jQuery(".vbo-report-datepicker-from").datepicker("setDate", "'.$pfromdate.'");' : '').'
'.(!empty($ptodate) ? 'jQuery(".vbo-report-datepicker-to").datepicker("setDate", "'.$ptodate.'");' : '').'
//prepare filler helpers
jQuery("#vbo-report-guardiacivil-hidden").children().detach().appendTo(".vbo-info-overlay-report");
jQuery("#choose-nazione").select2({placeholder: "- Selecciona una Nacion -", width: "200px"});
jQuery("#choose-documento").select2({placeholder: "- Selecciona un Documento -", width: "200px"});
jQuery("#choose-sesso").select2({placeholder: "- Selecciona Sexo -", width: "200px"});
jQuery("#choose-dbirth, #choose-docissuedt").datepicker({
maxDate: 0,
dateFormat: "dd/mm/yy",
changeMonth: true,
changeYear: true,
yearRange: "'.(date('Y') - 100).':'.date('Y').'"
});
// click events
jQuery(".vbo-report-load-nazione, .vbo-report-load-cittadinanza").click(function() {
reportActiveCell = this;
jQuery(".vbo-report-guardiacivil-selcont").hide();
jQuery("#vbo-report-guardiacivil-nazione").show();
vboShowOverlay();
});
jQuery(".vbo-report-load-doctype").click(function() {
reportActiveCell = this;
jQuery(".vbo-report-guardiacivil-selcont").hide();
jQuery("#vbo-report-guardiacivil-doctype").show();
vboShowOverlay();
});
jQuery(".vbo-report-load-docplace").click(function() {
reportActiveCell = this;
jQuery(".vbo-report-guardiacivil-selcont").hide();
jQuery("#vbo-report-guardiacivil-nazione").show();
vboShowOverlay();
});
jQuery(".vbo-report-load-sesso").click(function() {
reportActiveCell = this;
jQuery(".vbo-report-guardiacivil-selcont").hide();
jQuery("#vbo-report-guardiacivil-sesso").show();
vboShowOverlay();
});
jQuery(".vbo-report-load-docnum").click(function() {
reportActiveCell = this;
jQuery(".vbo-report-guardiacivil-selcont").hide();
jQuery("#vbo-report-guardiacivil-docnum").show();
vboShowOverlay();
setTimeout(function(){jQuery("#choose-docnum").focus();}, 500);
});
jQuery(".vbo-report-load-dbirth").click(function() {
reportActiveCell = this;
jQuery(".vbo-report-guardiacivil-selcont").hide();
jQuery("#vbo-report-guardiacivil-dbirth").show();
vboShowOverlay();
});
jQuery(".vbo-report-load-docissuedt").click(function() {
reportActiveCell = this;
jQuery(".vbo-report-guardiacivil-selcont").hide();
jQuery("#vbo-report-guardiacivil-docissuedt").show();
vboShowOverlay();
});
});
function vboReportCheckDates(selectedDate, inst) {
if (selectedDate === null || inst === null) {
return;
}
var cur_from_date = jQuery(this).val();
if (jQuery(this).hasClass("vbo-report-datepicker-from") && cur_from_date.length) {
var nowstart = jQuery(this).datepicker("getDate");
var nowstartdate = new Date(nowstart.getTime());
jQuery(".vbo-report-datepicker-to").datepicker("option", {minDate: nowstartdate});
}
}
function vboReportChosenNazione(naz) {
var c_code = naz.value;
var c_val = naz.options[naz.selectedIndex].text;
var c_threecode = jQuery(naz).find("option:selected").attr("data-threecode");
if (reportActiveCell !== null) {
var nowindex = jQuery(".vbo-reports-output table tbody tr").index(jQuery(reportActiveCell).closest("tr"));
if (isNaN(nowindex) || parseInt(nowindex) < 0) {
alert("Error, cannot find element to update.");
} else {
var rep_act_cell = jQuery(reportActiveCell);
rep_act_cell.addClass("vbo-report-load-elem-filled").find("span").text(c_val);
var rep_guest_bid = rep_act_cell.closest("tr").find("a[data-bid]").attr("data-bid");
if (!reportObj.hasOwnProperty(nowindex)) {
reportObj[nowindex] = {
bid: rep_guest_bid,
bid_index: jQuery(".vbo-reports-output table tbody tr").index(jQuery("a[data-bid=\"" + rep_guest_bid + "\"]").first().closest("tr"))
};
}
if (jQuery(reportActiveCell).hasClass("vbo-report-load-nazione")) {
reportObj[nowindex].stabirth = c_code;
} else if (jQuery(reportActiveCell).hasClass("vbo-report-load-docplace")) {
reportObj[nowindex].docplace = c_code;
} else {
reportObj[nowindex].country = c_code;
reportObj[nowindex].country_code = c_threecode;
}
}
}
reportActiveCell = null;
vboHideOverlay();
jQuery("#choose-nazione").val("").select2("data", null, false);
jQuery(".vbo-report-guardiacivil-manualsave").show();
}
function vboReportChosenDocumento(doctype) {
var c_code = doctype.value;
var c_val = doctype.options[doctype.selectedIndex].text;
if (reportActiveCell !== null) {
var nowindex = jQuery(".vbo-reports-output table tbody tr").index(jQuery(reportActiveCell).closest("tr"));
if (isNaN(nowindex) || parseInt(nowindex) < 0) {
alert("Error, cannot find element to update.");
} else {
var rep_act_cell = jQuery(reportActiveCell);
rep_act_cell.addClass("vbo-report-load-elem-filled").find("span").text(c_val);
var rep_guest_bid = rep_act_cell.closest("tr").find("a[data-bid]").attr("data-bid");
if (!reportObj.hasOwnProperty(nowindex)) {
reportObj[nowindex] = {
bid: rep_guest_bid,
bid_index: jQuery(".vbo-reports-output table tbody tr").index(jQuery("a[data-bid=\"" + rep_guest_bid + "\"]").first().closest("tr"))
};
}
reportObj[nowindex].doctype = c_code;
}
}
reportActiveCell = null;
vboHideOverlay();
jQuery("#choose-documento").val("").select2("data", null, false);
jQuery(".vbo-report-guardiacivil-manualsave").show();
}
function vboReportChosenSesso(sesso) {
var c_code = sesso.value;
var c_val = sesso.options[sesso.selectedIndex].text;
if (reportActiveCell !== null) {
var nowindex = jQuery(".vbo-reports-output table tbody tr").index(jQuery(reportActiveCell).closest("tr"));
if (isNaN(nowindex) || parseInt(nowindex) < 0) {
alert("Error, cannot find element to update.");
} else {
var rep_act_cell = jQuery(reportActiveCell);
rep_act_cell.addClass("vbo-report-load-elem-filled").find("span").text(c_val);
var rep_guest_bid = rep_act_cell.closest("tr").find("a[data-bid]").attr("data-bid");
if (!reportObj.hasOwnProperty(nowindex)) {
reportObj[nowindex] = {
bid: rep_guest_bid,
bid_index: jQuery(".vbo-reports-output table tbody tr").index(jQuery("a[data-bid=\"" + rep_guest_bid + "\"]").first().closest("tr"))
};
}
reportObj[nowindex].gender = c_code;
}
}
reportActiveCell = null;
vboHideOverlay();
jQuery("#choose-sesso").val("").select2("data", null, false);
jQuery(".vbo-report-guardiacivil-manualsave").show();
}
function vboReportChosenDocnum(val) {
var c_code = val, c_val = val;
if (reportActiveCell !== null) {
var nowindex = jQuery(".vbo-reports-output table tbody tr").index(jQuery(reportActiveCell).closest("tr"));
if (isNaN(nowindex) || parseInt(nowindex) < 0) {
alert("Error, cannot find element to update.");
} else {
var rep_act_cell = jQuery(reportActiveCell);
rep_act_cell.addClass("vbo-report-load-elem-filled").find("span").text(c_val);
var rep_guest_bid = rep_act_cell.closest("tr").find("a[data-bid]").attr("data-bid");
if (!reportObj.hasOwnProperty(nowindex)) {
reportObj[nowindex] = {
bid: rep_guest_bid,
bid_index: jQuery(".vbo-reports-output table tbody tr").index(jQuery("a[data-bid=\"" + rep_guest_bid + "\"]").first().closest("tr"))
};
}
reportObj[nowindex].docnum = c_code;
}
}
reportActiveCell = null;
vboHideOverlay();
jQuery("#choose-docnum").val("");
jQuery(".vbo-report-guardiacivil-manualsave").show();
}
function vboReportChosenDbirth(val) {
var c_code = val, c_val = val;
if (reportActiveCell !== null) {
var nowindex = jQuery(".vbo-reports-output table tbody tr").index(jQuery(reportActiveCell).closest("tr"));
if (isNaN(nowindex) || parseInt(nowindex) < 0) {
alert("Error, cannot find element to update.");
} else {
var rep_act_cell = jQuery(reportActiveCell);
rep_act_cell.addClass("vbo-report-load-elem-filled").find("span").text(c_val);
var rep_guest_bid = rep_act_cell.closest("tr").find("a[data-bid]").attr("data-bid");
if (!reportObj.hasOwnProperty(nowindex)) {
reportObj[nowindex] = {
bid: rep_guest_bid,
bid_index: jQuery(".vbo-reports-output table tbody tr").index(jQuery("a[data-bid=\"" + rep_guest_bid + "\"]").first().closest("tr"))
};
}
reportObj[nowindex]["date_birth"] = c_code;
}
}
reportActiveCell = null;
vboHideOverlay();
jQuery("#choose-dbirth").val("");
jQuery(".vbo-report-guardiacivil-manualsave").show();
}
function vboReportChosenDocissuedt(val) {
var c_code = val, c_val = val;
if (reportActiveCell !== null) {
var nowindex = jQuery(".vbo-reports-output table tbody tr").index(jQuery(reportActiveCell).closest("tr"));
if (isNaN(nowindex) || parseInt(nowindex) < 0) {
alert("Error, cannot find element to update.");
} else {
var rep_act_cell = jQuery(reportActiveCell);
rep_act_cell.addClass("vbo-report-load-elem-filled").find("span").text(c_val);
var rep_guest_bid = rep_act_cell.closest("tr").find("a[data-bid]").attr("data-bid");
if (!reportObj.hasOwnProperty(nowindex)) {
reportObj[nowindex] = {
bid: rep_guest_bid,
bid_index: jQuery(".vbo-reports-output table tbody tr").index(jQuery("a[data-bid=\"" + rep_guest_bid + "\"]").first().closest("tr"))
};
}
reportObj[nowindex].docissuedt = c_code;
}
}
reportActiveCell = null;
vboHideOverlay();
jQuery("#choose-docissuedt").val("");
jQuery(".vbo-report-guardiacivil-manualsave").show();
}
//download function
function vboDownloadSchedaGuardiaCivil() {
if (!confirm("¿Está seguro de haber ingresado todos los datos requeridos por la Guardia Civil?")) {
return false;
}
document.adminForm.target = "_blank";
document.adminForm.action += "&tmpl=component";
vboSetFilters({exportreport: "1", filler: JSON.stringify(reportObj)}, true);
setTimeout(function() {
document.adminForm.target = "";
document.adminForm.action = document.adminForm.action.replace("&tmpl=component", "");
vboSetFilters({exportreport: "0", filler: ""}, false);
}, 1000);
}
// save data function
function vboGuardiacivilSaveData() {
jQuery("button.vbo-report-guardiacivil-manualsave").find("i").attr("class", vbo_guardiacivil_saving_icn);
VBOCore.doAjax(
vbo_guardiacivil_ajax_uri,
{
call: "updatePaxData",
params: reportObj,
tmpl: "component"
},
function(response) {
if (!response || !response[0]) {
alert("An error occurred.");
return false;
}
jQuery("button.vbo-report-guardiacivil-manualsave").addClass("btn-success").find("i").attr("class", vbo_guardiacivil_saved_icn);
},
function(error) {
alert(error.responseText);
jQuery("button.vbo-report-guardiacivil-manualsave").removeClass("btn-success").find("i").attr("class", vbo_guardiacivil_save_icn);
}
);
}
';
$this->setScript($js);
return $this->reportFilters;
}
/**
* Loads the report data from the DB.
* Returns true in case of success, false otherwise.
* Sets the columns and rows for the report to be displayed.
*
* @return boolean
*/
public function getReportData()
{
if (strlen($this->getError())) {
//Export functions may set errors rather than exiting the process, and the View may continue the execution to attempt to render the report.
return false;
}
$all_countries = VikBooking::getCountriesArray();
//Input fields and other vars
$pfromdate = VikRequest::getString('fromdate', '', 'request');
$ptodate = VikRequest::getString('todate', '', 'request');
$photelname = VikRequest::getString('hotelname', '', 'request');
$photelcode = VikRequest::getString('hotelcode', '', 'request');
$pkrsort = VikRequest::getString('krsort', $this->defaultKeySort, 'request');
$pkrsort = empty($pkrsort) ? $this->defaultKeySort : $pkrsort;
$pkrorder = VikRequest::getString('krorder', $this->defaultKeyOrder, 'request');
$pkrorder = empty($pkrorder) ? $this->defaultKeyOrder : $pkrorder;
$pkrorder = $pkrorder == 'DESC' ? 'DESC' : 'ASC';
$currency_symb = VikBooking::getCurrencySymb();
$df = $this->getDateFormat();
$datesep = VikBooking::getDateSeparator();
if (empty($ptodate)) {
$ptodate = $pfromdate;
}
if (empty($photelcode) || empty($photelname)) {
$this->setError(JText::translate('VBOREPORTERRNODATA'));
return false;
}
//Get dates timestamps
$from_ts = VikBooking::getDateTimestamp($pfromdate, 0, 0);
$to_ts = VikBooking::getDateTimestamp($ptodate, 23, 59, 59);
if (empty($pfromdate) || empty($from_ts) || empty($to_ts)) {
$this->setError(JText::translate('VBOREPORTSERRNODATES'));
return false;
}
//Query to obtain the records (all check-ins within the dates filter)
$records = array();
$q = "SELECT `o`.`id`,`o`.`custdata`,`o`.`ts`,`o`.`days`,`o`.`checkin`,`o`.`checkout`,`o`.`totpaid`,`o`.`roomsnum`,`o`.`total`,`o`.`idorderota`,`o`.`channel`,`o`.`country`,".
"`or`.`idorder`,`or`.`idroom`,`or`.`adults`,`or`.`children`,`or`.`t_first_name`,`or`.`t_last_name`,`or`.`cust_cost`,`or`.`cust_idiva`,`or`.`extracosts`,`or`.`room_cost`,".
"`co`.`idcustomer`,`co`.`pax_data`,`c`.`first_name`,`c`.`last_name`,`c`.`country` AS `customer_country`,`c`.`doctype`,`c`.`docnum`,`c`.`gender`,`c`.`bdate`,`c`.`pbirth` ".
"FROM `#__vikbooking_orders` AS `o` LEFT JOIN `#__vikbooking_ordersrooms` AS `or` ON `or`.`idorder`=`o`.`id` ".
"LEFT JOIN `#__vikbooking_customers_orders` AS `co` ON `co`.`idorder`=`o`.`id` LEFT JOIN `#__vikbooking_customers` AS `c` ON `c`.`id`=`co`.`idcustomer` ".
"WHERE `o`.`status`='confirmed' AND `o`.`closure`=0 AND `o`.`checkin`>=".$from_ts." AND `o`.`checkin`<=".$to_ts." ".
"ORDER BY `o`.`checkin` ASC, `o`.`id` ASC, `or`.`id` ASC;";
$this->dbo->setQuery($q);
$records = $this->dbo->loadAssocList();
if (!$records) {
$this->setError(JText::translate('VBOREPORTSERRNORESERV'));
$this->setError('No llegan clientes en las fechas seleccionadas.');
return false;
}
//nest records with multiple rooms booked inside sub-array
$bookings = array();
foreach ($records as $v) {
if (!isset($bookings[$v['id']])) {
$bookings[$v['id']] = array();
}
array_push($bookings[$v['id']], $v);
}
//define the columns of the report
$this->cols = array(
//check-in
array(
'key' => 'checkin',
'attr' => array(
'class="center"'
),
'label' => JText::translate('VBPICKUPAT')
),
//cognome
array(
'key' => 'last_name',
'label' => JText::translate('VBTRAVELERLNAME')
),
//nome
array(
'key' => 'first_name',
'label' => JText::translate('VBTRAVELERNAME')
),
//sesso
array(
'key' => 'gender',
'attr' => array(
'class="center"'
),
'label' => JText::translate('VBCUSTOMERGENDER')
),
//data di nascita
array(
'key' => 'date_birth',
'attr' => array(
'class="center"'
),
'label' => JText::translate('VBCUSTOMERBDATE')
),
//cittadinanza
array(
'key' => 'country',
'attr' => array(
'class="center"'
),
'label' => JText::translate('VBCUSTOMERNATION')
),
//tipo documento
array(
'key' => 'doctype',
'attr' => array(
'class="center"'
),
'label' => JText::translate('VBCUSTOMERDOCTYPE')
),
//numero documento
array(
'key' => 'docnum',
'attr' => array(
'class="center"'
),
'label' => JText::translate('VBCUSTOMERDOCNUM')
),
//numero documento
array(
'key' => 'docissuedt',
'attr' => array(
'class="center"'
),
'label' => JText::translate('VBCUSTOMERDOCISSUE')
),
//id booking
array(
'key' => 'idbooking',
'attr' => array(
'class="center"'
),
'label' => 'ID'
)
);
//loop over the bookings to build the rows of the report
$from_info = getdate($from_ts);
foreach ($bookings as $gbook) {
$guests_rows = array($gbook[0]);
$tot_guests_rows = 1;
// count the total number of guests (only adults count) for all rooms of this booking
$tot_booking_guests = 0;
$room_guests = array();
foreach ($gbook as $rbook) {
$tot_booking_guests += $rbook['adults'];
$room_guests[] = $rbook['adults'];
}
if (!empty($gbook[0]['pax_data'])) {
$pax_data = json_decode($gbook[0]['pax_data'], true);
if (is_array($pax_data) && count($pax_data)) {
$guests_rows[0]['pax_data'] = $pax_data;
$tot_guests_rows = 0;
foreach ($pax_data as $roomguests) {
$tot_guests_rows += count($roomguests);
}
for ($i = 1; $i < $tot_guests_rows; $i++) {
array_push($guests_rows, $guests_rows[0]);
}
}
}
//create one row for each guest
$guest_ind = 1;
foreach ($guests_rows as $ind => $guests) {
$insert_row = array();
$use_pax_data = !empty($guests['pax_data']) && is_array($guests['pax_data']) ? $guests['pax_data'] : [];
//Data Arrivo
array_push($insert_row, array(
'key' => 'checkin',
'attr' => array(
'class="center"'
),
'callback' => function ($val) {
return $val;
},
'callback_export' => function ($val) {
return str_replace(['/', '-', '.'], '', $val);
},
'value' => date('Y-m-d', $guests['checkin'])
));
//Cognome
$pax_cognome = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'last_name');
$pax_cognome = empty($pax_cognome) && !empty($guests['t_last_name']) ? $guests['t_last_name'] : $pax_cognome;
array_push($insert_row, array(
'key' => 'last_name',
'value' => $pax_cognome,
));
//Nome
$pax_nome = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'first_name');
$pax_nome = empty($pax_nome) && !empty($guests['t_first_name']) ? $guests['t_first_name'] : $pax_nome;
array_push($insert_row, array(
'key' => 'first_name',
'value' => $pax_nome,
));
//Sesso
$gender = !empty($guests['gender']) && $guest_ind < 2 ? strtoupper($guests['gender']) : '';
$pax_gender = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'gender');
$gender = !empty($pax_gender) ? $pax_gender : $gender;
if (!empty($gender) && $gender == '1') {
$gender = 'M';
} elseif (!empty($gender) && $gender == '2') {
$gender = 'F';
}
array_push($insert_row, array(
'key' => 'gender',
'attr' => array(
'class="center'.(empty($gender) ? ' vbo-report-load-sesso' : '').'"'
),
'callback' => function ($val) {
return !empty($val) ? $val : '?';
},
'no_export_callback' => 1,
'value' => $gender
));
//Data di nascita
$dbirth = !empty($guests['bdate']) && $guest_ind < 2 ? VikBooking::getDateTimestamp($guests['bdate'], 0, 0) : '';
$pax_dbirth = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'date_birth');
$dbirth = !empty($pax_dbirth) ? $pax_dbirth : $dbirth;
$dbirth = (strpos($dbirth, '/') === false && strpos($dbirth, VikBooking::getDateSeparator()) === false) ? $dbirth : VikBooking::getDateTimestamp($dbirth, 0, 0);
array_push($insert_row, array(
'key' => 'date_birth',
'attr' => array(
'class="center'.(empty($dbirth) ? ' vbo-report-load-dbirth' : '').'"'
),
'callback' => function ($val) {
if (!empty($val) && strpos($val, '/') === false && strpos($val, VikBooking::getDateSeparator()) === false) {
return date('d/m/Y', $val);
}
if (!empty($val) && strpos($val, '/') !== false) {
return $val;
}
return '?';
},
'callback_export' => function ($val) {
return (strpos($val, '/') === false && strpos($val, VikBooking::getDateSeparator()) === false) ? date('Ymd', $val) : date('Ymd', VikBooking::getDateTimestamp($val, 0, 0));
},
'value' => $dbirth
));
//Cittadinanza
$citizen = !empty($guests['country']) && $guest_ind < 2 ? $guests['country'] : '';
$citizenval = '';
if (!empty($citizen) && $guest_ind < 2) {
$citizenval = $this->checkCountry($citizen);
}
if (!empty($citizenval) && isset($this->nazioni[$citizenval])) {
$citizenval = $this->nazioni[$citizenval]['name'];
}
// give higher important to info collected through custom fields
$pax_citizen = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'country');
if (!empty($pax_citizen)) {
$pax_citizen = strlen($pax_citizen) == 3 ? $this->checkCountry($pax_citizen) : $pax_citizen;
$citizenval = isset($this->nazioni[$pax_citizen]) ? $this->nazioni[$pax_citizen]['name'] : $pax_citizen;
}
array_push($insert_row, array(
'key' => 'country',
'attr' => array(
'class="center'.(empty($citizenval) ? ' vbo-report-load-cittadinanza' : '').'"'
),
'callback' => function ($val) {
return !empty($val) ? strtoupper($val) : '?';
},
'no_export_callback' => 1,
'value' => $citizenval
));
//Tipo documento
$pax_doctype = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'doctype');
$all_doc_types = $this->loadDocumenti();
$doctype = $pax_doctype;
array_push($insert_row, array(
'key' => 'doctype',
'attr' => array(
'class="center'.(empty($doctype) ? ' vbo-report-load-doctype' : '').'"'
),
'callback' => function ($val) use ($all_doc_types) {
return !empty($val) && isset($all_doc_types[$val]) ? $all_doc_types[$val] : '?';
},
'no_export_callback' => 1,
'value' => $pax_doctype,
));
//Numero documento
$pax_docnum = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'docnum');
$docnum = empty($pax_docnum) && !empty($guests['docnum']) && $guest_ind < 2 ? $guests['docnum'] : $pax_docnum;
array_push($insert_row, array(
'key' => 'docnum',
'attr' => array(
'class="center'.(empty($docnum) ? ' vbo-report-load-docnum' : '').'"'
),
'callback' => function ($val) {
return empty($val) ? '?' : $val;
},
'value' => $docnum,
));
// Document issue date
$pax_docissuedt = $this->getGuestPaxDataValue($use_pax_data, $room_guests, $guest_ind, 'docissuedt');
array_push($insert_row, array(
'key' => 'docissuedt',
'attr' => array(
'class="center'.(empty($pax_docissuedt) ? ' vbo-report-load-docissue vbo-report-load-docissuedt' : '').'"'
),
'callback' => function ($val) {
return !empty($val) ? $val : '?';
},
'callback_export' => function ($val) {
return (strpos($val, '/') === false && strpos($val, VikBooking::getDateSeparator()) === false) ? date('Ymd', $val) : date('Ymd', VikBooking::getDateTimestamp($val, 0, 0));
},
'value' => $pax_docissuedt,
));
//id booking
array_push($insert_row, array(
'key' => 'idbooking',
'attr' => array(
'class="center"'
),
'callback' => function ($val) {
// make sure to keep the data-bid attribute as it's used by JS to identify the booking ID
return '<a data-bid="' . $val . '" href="index.php?option=com_vikbooking&task=editorder&cid[]=' . $val . '" target="_blank"><i class="' . VikBookingIcons::i('external-link') . '"></i> ' . $val . '</a>';
},
'ignore_export' => 1,
'value' => $guests['id']
));
//push fields in the rows array as a new row
array_push($this->rows, $insert_row);
//increment guest index
$guest_ind++;
}
}
// do not sort the rows for this report because the lines of the guests of the same booking must be consecutive
//the footer row will just print the amount of records to export
array_push($this->footerRow, array(
array(
'attr' => array(
'class="vbo-report-total"'
),
'value' => '<h3>'.JText::translate('VBOREPORTSTOTALROW').'</h3>'
),
array(
'attr' => array(
'colspan="'.(count($this->cols) - 1).'"'
),
'value' => count($this->rows)
)
));
//Debug
if ($this->debug) {
$this->setWarning('path to report file = '.urlencode(dirname(__FILE__)).'<br/>');
$this->setWarning('$total_rooms_units = '.$total_rooms_units.'<br/>');
$this->setWarning('$bookings:<pre>'.print_r($bookings, true).'</pre><br/>');
}
//
return true;
}
/**
* Generates the text file for the Italian Police,
* then it sends it to output for download.
* In case of errors, the process is not terminated (exit)
* to let the View display the error message.
*
* @param string $export_type Differentiates the type of export requested.
*
* @return void|bool Void in case of script termination, boolean otherwise.
*/
public function customExport($export_type = 0)
{
if (!$this->getReportData()) {
return false;
}
$photelname = VikRequest::getString('hotelname', '', 'request');
$photelcode = VikRequest::getString('hotelcode', '', 'request');
$pfiller = VikRequest::getString('filler', '', 'request', VIKREQUEST_ALLOWRAW);
$pfiller = !empty($pfiller) ? json_decode($pfiller, true) : array();
$pfiller = !is_array($pfiller) ? array() : $pfiller;
//pool of booking IDs to update their history
$booking_ids = array();
//array of lines (one line for each guest)
$lines = array();
$customers = array();
$customerCount = 0;
$keyorder = array(
"docnum", // Doc id for DNI
"doctype",
"docissuedt",
"cognome1",
"cognome2",
"first_name",
"gender",
"date_birth",
"country",
"checkin"
);
//Push the lines of the Text file
foreach ($this->rows as $ind => $row) {
$line_cont = '';
foreach ($row as $field) {
if ($field['key'] == 'idbooking' && !in_array($field['value'], $booking_ids)) {
array_push($booking_ids, $field['value']);
}
if (isset($field['ignore_export'])) {
continue;
}
//report value
if (is_array($pfiller) && isset($pfiller[$ind]) && isset($pfiller[$ind][$field['key']])) {
if (strlen($pfiller[$ind][$field['key']])) {
$field['value'] = $pfiller[$ind][$field['key']];
}
}
// values set to -1 are usually empty and should have been filled in manually
if ($field['value'] === -1) {
// we raise an error in this case without stopping the process
$field['value'] = 0;
VikError::raiseWarning('', 'Line #' . $ind . ' has got an empty value that should have been manually filled by clicking the red box. File may be broken due to this line.');
}
if (isset($field['callback_export'])) {
$field['callback'] = $field['callback_export'];
}
$value = !isset($field['no_export_callback']) && isset($field['callback']) && is_callable($field['callback']) ? $field['callback']($field['value']) : $field['value'];
//0 or '---' should be changed to an empty string (case of "-- Estero --" or field to be filled with Blank)
$value = empty($value) || $value == '---' ? '' : $value;
//this is necessary since the customer may have multiple last names.
if ($field['key'] == 'last_name') {
$cognomi = explode(" ", $value);
$customers[$customerCount]['cognome1'] = $cognomi[0];
$customers[$customerCount]['cognome2'] = isset($cognomi[1]) ? $cognomi[1] : "";
continue;
}
//concatenate the field to the customer record
$customers[$customerCount][$field['key']] = $this->valueFiller($value, '');
}
$customerCount++;
}
$separator = '|';
$line_cont = implode($separator, [
'1',
$photelcode,
$photelname,
date('Ymd'),
date('Hi'),
str_pad((string)count($customers), 5, '0', STR_PAD_LEFT),
]);
array_push($lines, $line_cont);
foreach ($customers as $customer) {
$line_cont = '2|';
foreach ($keyorder as $key) {
if ($key == 'docnum') {
$line_cont .= strtoupper($customer['doctype']) == 'D' ? (strtoupper($customer[$key]) . $separator . $separator) : ($separator . strtoupper($customer[$key]) . $separator);
continue;
}
$line_cont .= strtoupper($customer[$key]) . $separator;
}
$line_cont = rtrim($line_cont, $separator);
//push the line in the array of lines
array_push($lines, $line_cont);
}
//update the history for all bookings affected
foreach ($booking_ids as $bid) {
VikBooking::getBookingHistoryInstance()->setBid($bid)->store('RP', $this->reportName);
}
/**
* Custom export method supports a custom export handler, if previously set.
*
* @since 1.16.1 (J) - 1.6.1 (WP)
*/
if ($this->hasExportHandler()) {
// write data onto the custom file handler
$fp = $this->getExportCSVHandler();
fwrite($fp, implode("\r\n", $lines));
fclose($fp);
return true;
}
// force text file download
header('Content-Type: text/plain; charset=utf-8');
header('Cache-Control: no-store, no-cache');
header('Content-Disposition: attachment; filename="' . $this->getExportCSVFileName() . '"');
echo implode("\r\n", $lines);
exit;
}
/**
* Helper method invoked via AJAX by the controller.
* Needed to save the manual entries for the pax data.
*
* @param array $manual_data the object representation of the manual entries.
*
* @return array one boolean value array with the operation result.
*/
public function updatePaxData($manual_data = array())
{
if (!is_array($manual_data) || !$manual_data) {
VBOHttpDocument::getInstance()->close(400, 'Nothing to save!');
}
// re-build manual entries object representation
$bids_guests = array();
foreach ($manual_data as $guest_ind => $guest_data) {
if (!is_numeric($guest_ind) || !is_array($guest_data) || empty($guest_data['bid']) || !isset($guest_data['bid_index']) || count($guest_data) < 2) {
// empty or invalid manual entries array
continue;
}
// the guest index in the reportObj starts from 0
$use_guest_ind = ($guest_ind + 1 - (int)$guest_data['bid_index']);
if (!isset($bids_guests[$guest_data['bid']])) {
$bids_guests[$guest_data['bid']] = array();
}
// set manual entries for this guest number
$bids_guests[$guest_data['bid']][$use_guest_ind] = $guest_data;
// remove the "bid" and "bid_index" keys
unset($bids_guests[$guest_data['bid']][$use_guest_ind]['bid'], $bids_guests[$guest_data['bid']][$use_guest_ind]['bid_index']);
}
if (!count($bids_guests)) {
VBOHttpDocument::getInstance()->close(400, 'No manual entries to save found');
}
// loop through all bookings to update the data for the various rooms and guests
$bids_updated = 0;
foreach ($bids_guests as $bid => $entries) {
$b_rooms = VikBooking::loadOrdersRoomsData($bid);
if (empty($b_rooms)) {
continue;
}
// count adults per room
$room_guests = array();
foreach ($b_rooms as $b_room) {
$room_guests[] = $b_room['adults'];
}
// get current booking pax data
$pax_data = VBOCheckinPax::getBookingPaxData($bid);
$pax_data = empty($pax_data) ? array() : $pax_data;
foreach ($entries as $guest_ind => $guest_data) {
if (!empty($guest_data['country']) && !empty($guest_data['country_code'])) {
// adjust manually entered country to store just the code for compliance with field of type "country"
$guest_data['country'] = $guest_data['country_code'];
unset($guest_data['country_code']);
}
// find room index for this guest
$room_num = 0;
$use_guest_ind = $guest_ind;
foreach ($room_guests as $room_index => $tot_guests) {
// find the proper guest index for the room to which this belongs
if ($use_guest_ind <= $tot_guests) {
// proper room index found for this guest
$room_num = $room_index;
break;
} else {
// it's probably in a next room
$use_guest_ind -= $tot_guests;
}
}
// push new pax data for this room and guest
if (!isset($pax_data[$room_num])) {
$pax_data[$room_num] = array();
}
if (!isset($pax_data[$room_num][$use_guest_ind])) {
$pax_data[$room_num][$use_guest_ind] = $guest_data;
} else {
$pax_data[$room_num][$use_guest_ind] = array_merge($pax_data[$room_num][$use_guest_ind], $guest_data);
}
}
// update booking pax data
if (VBOCheckinPax::setBookingPaxData($bid, $pax_data)) {
$bids_updated++;
}
}
return $bids_updated ? [true] : [false];
}
/**
* Parses the file Documenti.csv and returns an associative
* array with the code and name of the Documento.
* Every line of the CSV is composed of: Codice, Documento.
*
* @return array
*/
public function loadDocumenti()
{
return [
"D" => "Documento Nacional de Identidad",
"P" => "Pasaporte",
"C" => "Permiso de Conducir",
"I" => "Carta o Documento de Identidad",
"N" => "Permiso de Residencia Español",
"X" => "Permiso de Residencia de otro Estado Miembro de la Unión Europea",
];
}
/**
* Registers the name to give to the file being exported.
*
* @return void
*
* @since 1.16.1 (J) - 1.6.1 (WP)
*/
private function registerExportFileName()
{
$pfromdate = VikRequest::getString('fromdate', '', 'request');
$ptodate = VikRequest::getString('todate', '', 'request');
$this->setExportCSVFileName($this->reportName . '-' . str_replace('/', '_', $pfromdate) . '-' . str_replace('/', '_', $ptodate) . '.txt');
}
/**
* This method adds blank spaces to the string
* until the passed length of string is reached.
*
* @param string $val
* @param int $len
*
* @return string
*/
private function valueFiller($val, $len)
{
$len = empty($len) || (int)$len <= 0 ? strlen($val) : (int)$len;
//clean up $val in case there is still a CR or LF
$val = str_replace(array("\r\n", "\r", "\n"), '', $val);
if (strlen($val) < $len) {
while (strlen($val) < $len) {
$val .= ' ';
}
} elseif (strlen($val) > $len) {
$val = substr($val, 0, $len);
}
return $val;
}
/**
* Loads the country names from DB
*
* @return array
*/
private function loadNazioni()
{
$nazioni = [];
$records = [];
$db = JFactory::getDbo();
$q = "SELECT `country_3_code`, `country_name`,`country_2_code` FROM `#__vikbooking_countries`;";
$db->setQuery($q);
$db->execute();
if ($db->getNumRows()>0) {
$records = $db->loadAssocList();
}
foreach ($records as $key => $value) {
$nazioni[$value['country_2_code']]['name'] = $value['country_name'];
$nazioni[$value['country_2_code']]['three_code'] = $value['country_3_code'];
}
return $nazioni;
}
/**
*
* Returns the key of the state selected by the user.
*
* @return string
*
*/
private function checkCountry($country) {
$found = false;
$staval = '';
if (!count($this->nazioni)) {
$this->nazioni = $this->loadNazioni();
}
foreach ($this->nazioni as $key => $value) {
if (trim($value['three_code']) == trim($country)) {
$staval = $key;
$found = true;
break;
}
}
if ($found !== true) {
$staval = '';
}
return $staval;
}
/**
* Helper method to quickly get a pax_data property for the guest.
*
* @param array $pax_data the current pax_data stored.
* @param array $guests list of total guests per room.
* @param int $guest_ind the guest index.
* @param string $key the pax_data key to look for.
*
* @return mixed null on failure or value fetched.
*
* @since 1.15.2 (J) - 1.5.4 (WP)
*/
private function getGuestPaxDataValue($pax_data, $guests, $guest_ind, $key)
{
if (!is_array($pax_data) || !count($pax_data) || empty($key)) {
return null;
}
// find room index for this guest number
$room_num = 0;
$use_guest_ind = $guest_ind;
foreach ($guests as $room_index => $room_tot_guests) {
// find the proper guest index for the room to which this belongs
if ($use_guest_ind <= $room_tot_guests) {
// proper room index found for this guest
$room_num = $room_index;
break;
} else {
// it's probably in a next room
$use_guest_ind -= $room_tot_guests;
}
}
// check if a value exists for the requested key in the found room and guest indexes
if (isset($pax_data[$room_num]) && isset($pax_data[$room_num][$use_guest_ind])) {
if (isset($pax_data[$room_num][$use_guest_ind][$key])) {
// we've got a value previously stored
return $pax_data[$room_num][$use_guest_ind][$key];
}
}
// nothing was found
return null;
}
}