2023-04-24 15:31:46 +02:00
|
|
|
#include "calculate_price.h"
|
|
|
|
#include "configuration.h"
|
|
|
|
#include "calculator_functions.h"
|
|
|
|
#include "payment_option.h"
|
2023-12-07 16:28:17 +01:00
|
|
|
#include "utilities.h"
|
2024-08-01 16:00:10 +02:00
|
|
|
#include "tariff_global_defines.h"
|
|
|
|
#include "period_year.h"
|
2023-04-24 15:31:46 +02:00
|
|
|
|
|
|
|
#include <QFile>
|
|
|
|
#include <QFileInfo>
|
|
|
|
#include <QDateTime>
|
|
|
|
#include <QDebug>
|
2024-01-22 14:46:40 +01:00
|
|
|
#include <QList>
|
2023-04-24 15:31:46 +02:00
|
|
|
|
2024-06-05 16:58:12 +02:00
|
|
|
QString const CalcState::SUCCESS = "SUCCESS";
|
|
|
|
QString const CalcState::ERROR_PARSING_ZONE_NR = "ERROR_PARSING_ZONE_NR";
|
|
|
|
QString const CalcState::ERROR_LOADING_TARIFF = "ERROR_LOADING_TARIFF";
|
|
|
|
QString const CalcState::ERROR_PARSING_TARIFF = "ERROR_PARSING_TARIFF";
|
|
|
|
QString const CalcState::NEGATIVE_PARKING_TIME = "NEGATIVE_PARKING_TIME";
|
|
|
|
QString const CalcState::INVALID_START_DATE = "INVALID_START_DATE";
|
|
|
|
QString const CalcState::WRONG_PARAM_VALUES = "WRONG_PARAM_VALUES";
|
|
|
|
QString const CalcState::WRONG_ISO_TIME_FORMAT = "WRONG_ISO_TIME_FORMAT";
|
|
|
|
QString const CalcState::ABOVE_MAX_PARKING_TIME = "ABOVE_MAX_PARKING_TIME";
|
|
|
|
QString const CalcState::BELOW_MIN_PARKING_TIME = "BELOW_MIN_PARKING_TIME";
|
|
|
|
QString const CalcState::BELOW_MIN_PARKING_PRICE = "BELOW_MIN_PARKING_PRICE";
|
|
|
|
QString const CalcState::ABOVE_MAX_PARKING_PRICE = "ABOVE_MAX_PARKING_PRICE";
|
|
|
|
QString const CalcState::OVERPAID = "OVERPAID";
|
|
|
|
QString const CalcState::OUTSIDE_ALLOWED_PARKING_TIME = "OUTSIDE_ALLOWED_PARKING_TIME";
|
|
|
|
|
2024-01-22 15:41:20 +01:00
|
|
|
QList<int> CALCULATE_LIBRARY_API get_time_steps(Configuration *cfg) {
|
2024-01-23 10:55:00 +01:00
|
|
|
return Calculator::GetInstance().GetTimeSteps(cfg);
|
2024-01-22 15:41:20 +01:00
|
|
|
}
|
|
|
|
|
2024-04-30 14:08:08 +02:00
|
|
|
int CALCULATE_LIBRARY_API get_minimal_parkingtime(Configuration const *cfg,
|
2024-04-12 14:27:08 +02:00
|
|
|
PERMIT_TYPE permitType,
|
|
|
|
int paymentOptionIndex) {
|
2024-01-30 10:39:24 +01:00
|
|
|
int minTime = 0;
|
|
|
|
|
2024-09-02 17:09:26 +02:00
|
|
|
paymentOptionIndex = getPaymentOptionIndex(*cfg);
|
|
|
|
if (paymentOptionIndex == -1) {
|
|
|
|
paymentOptionIndex = cfg->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
2024-07-25 09:48:12 +02:00
|
|
|
|
2024-07-26 10:59:45 +02:00
|
|
|
qCritical() << __func__ << __LINE__ << "paymentOptionIndex" << paymentOptionIndex;
|
|
|
|
qCritical() << __func__ << __LINE__ << "permit" << PermitType(permitType).toString();
|
|
|
|
|
2024-01-30 10:39:24 +01:00
|
|
|
switch(permitType) {
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281)
|
2024-07-26 10:59:45 +02:00
|
|
|
qCritical() << __LINE__ << Calculator::GetInstance().GetTimeSteps((Configuration *)cfg, paymentOptionIndex);
|
2024-04-12 14:27:08 +02:00
|
|
|
minTime = cfg->getPaymentOptions(paymentOptionIndex).pop_min_time;
|
2024-07-26 10:59:45 +02:00
|
|
|
qCritical() << __func__ << __LINE__ << minTime;
|
2024-01-30 10:39:24 +01:00
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_ADULT: {
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_TEEN: {
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_CHILD: {
|
|
|
|
} break;
|
|
|
|
default:
|
|
|
|
// for each new sell-procedure, recomute the timesteps. implicitly, set
|
|
|
|
// the minimal parking time.
|
2024-04-12 14:27:08 +02:00
|
|
|
Calculator::GetInstance().ResetTimeSteps(paymentOptionIndex);
|
2024-07-26 10:59:45 +02:00
|
|
|
qCritical() << __LINE__ << Calculator::GetInstance().GetTimeSteps((Configuration *)cfg, paymentOptionIndex);
|
2024-04-12 14:27:08 +02:00
|
|
|
minTime = qRound(cfg->getPaymentOptions(paymentOptionIndex).pop_min_time);
|
2024-01-30 10:39:24 +01:00
|
|
|
}
|
|
|
|
|
2024-07-26 10:59:45 +02:00
|
|
|
qCritical() << "minTime" << minTime;
|
2024-01-30 10:39:24 +01:00
|
|
|
return minTime;
|
|
|
|
}
|
|
|
|
|
2024-04-30 14:08:08 +02:00
|
|
|
int CALCULATE_LIBRARY_API get_maximal_parkingtime(Configuration const *cfg,
|
2024-04-12 14:27:08 +02:00
|
|
|
PERMIT_TYPE permitType,
|
|
|
|
int paymentOptionIndex) {
|
2024-07-25 09:48:12 +02:00
|
|
|
|
2024-09-02 17:09:26 +02:00
|
|
|
paymentOptionIndex = getPaymentOptionIndex(*cfg);
|
|
|
|
if (paymentOptionIndex == -1) {
|
|
|
|
paymentOptionIndex = cfg->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
2024-01-30 10:39:24 +01:00
|
|
|
int maxTime = 0;
|
|
|
|
|
|
|
|
switch(permitType) {
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281)
|
2024-04-12 14:27:08 +02:00
|
|
|
maxTime = cfg->getPaymentOptions(paymentOptionIndex).pop_max_time;
|
2024-01-30 10:39:24 +01:00
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_ADULT: {
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_TEEN: {
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_CHILD: {
|
|
|
|
} break;
|
|
|
|
default: ;
|
2024-07-25 09:48:12 +02:00
|
|
|
maxTime = cfg->getPaymentOptions(paymentOptionIndex).pop_max_time;
|
2024-01-30 10:39:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return maxTime;
|
|
|
|
}
|
|
|
|
|
2024-04-12 14:27:08 +02:00
|
|
|
int CALCULATE_LIBRARY_API get_minimal_parkingprice(Configuration *cfg,
|
|
|
|
PERMIT_TYPE permitType,
|
2024-05-15 15:21:38 +02:00
|
|
|
int paymentOptionIndex,
|
|
|
|
QDateTime const &start) {
|
2024-01-30 10:39:24 +01:00
|
|
|
int minPrice = -1;
|
2024-09-02 17:09:26 +02:00
|
|
|
|
2024-09-06 12:05:41 +02:00
|
|
|
if ((paymentOptionIndex = getPaymentOptionIndex(*cfg, start)) == -1) {
|
2024-09-02 17:09:26 +02:00
|
|
|
paymentOptionIndex = cfg->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
2024-01-30 10:39:24 +01:00
|
|
|
|
2024-07-30 15:29:47 +02:00
|
|
|
int payment_method_id = cfg->getPaymentOptions(paymentOptionIndex).pop_payment_method_id;
|
|
|
|
|
|
|
|
if (payment_method_id == PaymentMethod::Degressive) {
|
|
|
|
// Degressive: new for Fuchs Technik (500), ValserAlm (Fane):
|
|
|
|
// the minimal price has to be calculated, in cannot be hard coded into
|
|
|
|
// the tariff file.
|
|
|
|
// The working times have a reference into the payment rates. Two special
|
|
|
|
// entries (with the numbers 1000/1001) point to the respective prices.
|
|
|
|
switch(permitType) {
|
|
|
|
default: {
|
|
|
|
// find the correct work time range
|
|
|
|
int weekDay = start.date().dayOfWeek();
|
|
|
|
std::optional<QVector<ATBWeekDaysWorktime>> const &wd = cfg->getAllWeekDayWorkTimes();
|
|
|
|
if (wd.has_value()) {
|
|
|
|
QVector<ATBWeekDaysWorktime> const &vec = wd.value();
|
|
|
|
for (int i = 0; i < vec.size(); ++i) {
|
|
|
|
ATBWeekDaysWorktime const &wt = vec[i];
|
|
|
|
if (wt.pwd_period_day_in_week_id == weekDay) {
|
|
|
|
if (start.time() >= QTime::fromString(QString::fromStdString(wt.pwd_time_from), Qt::ISODate)
|
|
|
|
&& start.time() <= QTime::fromString(QString::fromStdString(wt.pwd_time_to), Qt::ISODate)) {
|
|
|
|
// found worktime range
|
|
|
|
int pop_id = wt.pwd_pop_id; // 1000 or 1001
|
|
|
|
for (auto[itr, rangeEnd] = cfg->PaymentRate.equal_range(pop_id); itr != rangeEnd; ++itr) {
|
|
|
|
i = vec.size(); // to leave outer loop
|
|
|
|
minPrice = itr->second.pra_price; // this is now the minimal price
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch(permitType) {
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281)
|
|
|
|
minPrice = cfg->getPaymentOptions(paymentOptionIndex).pop_min_price;
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_ADULT: {
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_TEEN: {
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_CHILD: {
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET: {
|
|
|
|
minPrice = compute_product_price(cfg, permitType, start);
|
|
|
|
} break;
|
|
|
|
default:
|
|
|
|
minPrice = cfg->getPaymentOptions(paymentOptionIndex).pop_min_price;
|
|
|
|
}
|
2024-01-30 10:39:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return minPrice;
|
|
|
|
}
|
|
|
|
|
2024-04-16 12:10:34 +02:00
|
|
|
int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg,
|
|
|
|
PERMIT_TYPE permitType,
|
|
|
|
QDateTime const &start,
|
|
|
|
QDateTime *productStart,
|
|
|
|
QDateTime *productEnd) {
|
2024-02-22 16:38:41 +01:00
|
|
|
|
|
|
|
switch(permitType) {
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281)
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_CHILD:
|
|
|
|
// [[fallthrough]];
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_TEEN:
|
|
|
|
// [[fallthrough]];
|
2024-02-28 12:06:02 +01:00
|
|
|
case PERMIT_TYPE::FOOD_STAMP:
|
|
|
|
// [[fallthrough]];
|
2024-02-22 16:38:41 +01:00
|
|
|
case PERMIT_TYPE::DAY_TICKET_ADULT: {
|
|
|
|
std::optional<QVector<ATBTariffProduct>> products = cfg->getTariffProductForProductId(permitType);
|
|
|
|
if (products) {
|
|
|
|
QVector<ATBTariffProduct> product = products.value();
|
|
|
|
if (product.size() > 0) {
|
|
|
|
ATBTariffProduct const &p = product[0];
|
2024-02-28 09:58:00 +01:00
|
|
|
return p.m_tariff_product_price;
|
|
|
|
#if 0
|
|
|
|
// in case we do not have prepaid-option
|
2024-02-28 09:26:51 +01:00
|
|
|
QTime const ¤tTime = QDateTime::currentDateTime().time();
|
|
|
|
|
|
|
|
if (p.m_tariff_product_start <= currentTime && currentTime <= p.m_tariff_product_end) {
|
|
|
|
return p.m_tariff_product_price;
|
|
|
|
} else {
|
|
|
|
qCritical() << "(" << __func__ << ":" << __LINE__ << ")"
|
|
|
|
<< "ERROR currentTime"
|
|
|
|
<< currentTime.toString(Qt::ISODate)
|
|
|
|
<< "INVALID ("
|
|
|
|
<< p.m_tariff_product_start.toString(Qt::ISODate)
|
|
|
|
<< p.m_tariff_product_end.toString(Qt::ISODate) << ")";
|
|
|
|
}
|
2024-02-28 09:58:00 +01:00
|
|
|
#endif
|
2024-02-22 16:38:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} break;
|
2024-04-16 12:10:34 +02:00
|
|
|
case PERMIT_TYPE::INVALID:
|
|
|
|
// [[fallthrough]];
|
|
|
|
case PERMIT_TYPE::DAY_TICKET: {
|
|
|
|
std::optional<QVector<ATBTariffProduct>> products = cfg->getTariffProductForProductId(permitType);
|
|
|
|
if (products) {
|
|
|
|
QVector<ATBTariffProduct> product = products.value();
|
|
|
|
int product_price = 0;
|
|
|
|
|
|
|
|
if (productStart && productEnd) {
|
|
|
|
*productStart = start;
|
|
|
|
*productEnd = start;
|
|
|
|
if (product.size() > 0) {
|
|
|
|
productStart->setTime(product[0].getTimeStart());
|
|
|
|
productEnd->setTime(product[0].getTimeEnd());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (QVector<ATBTariffProduct>::size_type i=0; i<product.size(); ++i) {
|
|
|
|
ATBTariffProduct const &p = product[i];
|
|
|
|
QTime const &startTime = p.getTimeStart();
|
|
|
|
QTime const &endTime = p.getTimeEnd();
|
|
|
|
|
2024-05-15 15:23:12 +02:00
|
|
|
// qCritical() << __LINE__ << startTime.toString(Qt::ISODate);
|
|
|
|
// qCritical() << __LINE__ << endTime.toString(Qt::ISODate);
|
|
|
|
// qCritical() << __LINE__ << start.toString(Qt::ISODate);
|
|
|
|
|
2024-04-16 12:10:34 +02:00
|
|
|
if (start.time() >= startTime && start.time() < endTime) {
|
|
|
|
product_price = p.getProductPrice();
|
|
|
|
if (productStart && productEnd) {
|
|
|
|
productStart->setTime(startTime);
|
|
|
|
productEnd->setTime(endTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return product_price;
|
2024-06-04 11:24:15 +02:00
|
|
|
} else {
|
|
|
|
// SZEGED
|
|
|
|
int const pop_daily_card_price = cfg->getPaymentOptions().pop_daily_card_price;
|
|
|
|
|
|
|
|
qDebug() << QString("(%1:%2) no products defined in tariff-file").arg(__func__).arg(__LINE__);
|
|
|
|
qDebug() << QString("(%1:%2) pop_daily_card_price=%3").arg(__func__).arg(__LINE__).arg(pop_daily_card_price);
|
|
|
|
|
|
|
|
// static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg);
|
|
|
|
// return Utilities::getDailyTicketCardPrice(cfg, paymentMethodId);
|
|
|
|
|
|
|
|
return pop_daily_card_price;
|
2024-04-16 12:10:34 +02:00
|
|
|
}
|
|
|
|
} break;
|
2024-04-30 14:08:08 +02:00
|
|
|
case PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET: {
|
|
|
|
std::optional<QVector<ATBTariffProduct>> products = cfg->getTariffProductForProductId(permitType);
|
|
|
|
if (products) {
|
|
|
|
int product_price = 0;
|
|
|
|
QVector<ATBTariffProduct> product = products.value();
|
|
|
|
|
|
|
|
if (product.size() > 0) {
|
|
|
|
if (productStart && productEnd) {
|
2024-07-29 11:22:22 +02:00
|
|
|
int pop_min_time = get_minimal_parkingtime(cfg, PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET); // in minutes
|
|
|
|
int pop_max_time = get_maximal_parkingtime(cfg, PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET); // in minutes
|
2024-04-30 14:08:08 +02:00
|
|
|
if (pop_max_time >= pop_min_time) {
|
|
|
|
*productStart = start;
|
|
|
|
*productEnd = start.addSecs(pop_max_time*60);
|
|
|
|
product_price = product[0].getProductPrice();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return product_price;
|
|
|
|
}
|
|
|
|
|
|
|
|
} break;
|
2024-07-23 11:45:34 +02:00
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING_PKW: {
|
2024-07-25 09:48:12 +02:00
|
|
|
PermitType p(permitType);
|
|
|
|
std::optional<ATBPaymentOption> const paymentOption = cfg->getPaymentOptionForKey(p.get());
|
|
|
|
if (paymentOption.has_value()) {
|
|
|
|
ATBPaymentOption option = paymentOption.value();
|
|
|
|
int const pop_daily_card_price = option.pop_daily_card_price;
|
|
|
|
qCritical() << "SHORT_TERM_PARKING_PKW: daily ticket price" << pop_daily_card_price;
|
|
|
|
return pop_daily_card_price;
|
|
|
|
}
|
2024-07-23 11:45:34 +02:00
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING_BUS: {
|
|
|
|
qCritical() << "TODO: SHORT_TERM_PARKING_BUS";
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING_CAMPER: {
|
|
|
|
qCritical() << "TODO: SHORT_TERM_PARKING_CAMPER";
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_PKW: {
|
2024-07-25 09:48:12 +02:00
|
|
|
PermitType p(permitType);
|
|
|
|
std::optional<ATBPaymentOption> const paymentOption = cfg->getPaymentOptionForKey(p.get());
|
|
|
|
if (paymentOption.has_value()) {
|
|
|
|
ATBPaymentOption option = paymentOption.value();
|
|
|
|
int const pop_daily_card_price = option.pop_daily_card_price;
|
|
|
|
qCritical() << "DAY_TICKET_PKW: daily ticket price" << pop_daily_card_price;
|
|
|
|
return pop_daily_card_price;
|
|
|
|
}
|
2024-07-23 11:45:34 +02:00
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_BUS: {
|
2024-07-25 09:48:12 +02:00
|
|
|
PermitType p(permitType);
|
|
|
|
std::optional<ATBPaymentOption> const paymentOption = cfg->getPaymentOptionForKey(p.get());
|
|
|
|
if (paymentOption.has_value()) {
|
|
|
|
ATBPaymentOption option = paymentOption.value();
|
|
|
|
int const pop_daily_card_price = option.pop_daily_card_price;
|
|
|
|
qCritical() << "DAY_TICKET_BUS: daily ticket price" << pop_daily_card_price;
|
|
|
|
return pop_daily_card_price;
|
|
|
|
}
|
2024-07-23 11:45:34 +02:00
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_CAMPER: {
|
2024-07-25 09:48:12 +02:00
|
|
|
PermitType p(permitType);
|
|
|
|
std::optional<ATBPaymentOption> const paymentOption = cfg->getPaymentOptionForKey(p.get());
|
|
|
|
if (paymentOption.has_value()) {
|
|
|
|
ATBPaymentOption option = paymentOption.value();
|
|
|
|
int const pop_daily_card_price = option.pop_daily_card_price;
|
|
|
|
qCritical() << "DAY_TICKET_CAMPER: daily ticket price" << pop_daily_card_price;
|
|
|
|
return pop_daily_card_price;
|
|
|
|
}
|
2024-07-23 11:45:34 +02:00
|
|
|
} break;
|
2024-02-22 16:38:41 +01:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-04-12 14:27:08 +02:00
|
|
|
int CALCULATE_LIBRARY_API get_maximal_parkingprice(Configuration *cfg,
|
|
|
|
PERMIT_TYPE permitType,
|
|
|
|
int paymentOptionIndex) {
|
2024-01-30 10:39:24 +01:00
|
|
|
int maxPrice = -1;
|
2024-02-20 15:47:02 +01:00
|
|
|
static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg);
|
2024-01-30 10:39:24 +01:00
|
|
|
|
2024-09-02 17:09:26 +02:00
|
|
|
if ((paymentOptionIndex = getPaymentOptionIndex(*cfg)) == -1) {
|
|
|
|
paymentOptionIndex = cfg->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
2024-07-25 09:48:12 +02:00
|
|
|
|
2024-01-30 10:39:24 +01:00
|
|
|
switch(permitType) {
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281)
|
2024-04-08 13:57:40 +02:00
|
|
|
if (paymentMethodId == PaymentMethod::Progressive || paymentMethodId == PaymentMethod::Steps) {
|
2024-02-20 15:47:02 +01:00
|
|
|
maxPrice = Utilities::getMaximalParkingPrice(cfg, paymentMethodId);
|
|
|
|
} else { // PaymentMethod::Linear -> e.g. szeged
|
2024-04-12 14:27:08 +02:00
|
|
|
int const key = cfg->getPaymentOptions(paymentOptionIndex).pop_id;
|
|
|
|
int const maxTime = cfg->getPaymentOptions(paymentOptionIndex).pop_max_time; // maxTime is given in minutes
|
2024-02-20 15:47:02 +01:00
|
|
|
std::optional<QVector<ATBPaymentRate>> const &pv = cfg->getPaymentRateForKey(key);
|
|
|
|
if (pv) {
|
|
|
|
QVector<ATBPaymentRate> const &paymentRate = pv.value();
|
|
|
|
if (paymentRate.size() > 0) {
|
2024-03-07 08:17:32 +01:00
|
|
|
int const price = paymentRate.last().pra_price; // price is given per hour
|
2024-02-20 15:47:02 +01:00
|
|
|
maxPrice = qRound((maxTime * price) / 60.0f);
|
|
|
|
}
|
2024-01-30 10:39:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_ADULT:
|
|
|
|
break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_TEEN:
|
|
|
|
break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_CHILD:
|
|
|
|
break;
|
2024-07-25 09:48:12 +02:00
|
|
|
case PERMIT_TYPE::DAY_TICKET_BUS:
|
|
|
|
break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_CAMPER:
|
|
|
|
break;
|
|
|
|
case PERMIT_TYPE::DAY_TICKET_PKW:
|
|
|
|
break;
|
2024-07-29 11:24:05 +02:00
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING_BUS: {
|
|
|
|
std::optional<ATBPaymentOption> po = cfg->getPaymentOptionForKey(permitType);
|
|
|
|
if (po.has_value()) {
|
|
|
|
ATBPaymentOption option = po.value();
|
|
|
|
return option.pop_max_price;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING_PKW: {
|
|
|
|
std::optional<ATBPaymentOption> po = cfg->getPaymentOptionForKey(permitType);
|
|
|
|
if (po.has_value()) {
|
|
|
|
ATBPaymentOption option = po.value();
|
|
|
|
return option.pop_max_price;
|
|
|
|
}
|
|
|
|
} break;
|
2024-07-25 09:48:12 +02:00
|
|
|
case PERMIT_TYPE::SHORT_TERM_PARKING_CAMPER:
|
|
|
|
break;
|
2024-01-30 10:39:24 +01:00
|
|
|
default: ;
|
|
|
|
}
|
|
|
|
|
|
|
|
return maxPrice;
|
2024-01-22 14:29:02 +01:00
|
|
|
}
|
|
|
|
|
2023-05-12 09:20:46 +02:00
|
|
|
int CALCULATE_LIBRARY_API get_zone_nr(int zone)
|
|
|
|
{
|
|
|
|
if(zone > -1) return zone;
|
|
|
|
else
|
2023-05-15 14:05:55 +02:00
|
|
|
{
|
2023-05-12 09:20:46 +02:00
|
|
|
QFile zone("/etc/zone_nr");
|
|
|
|
if (zone.exists()) {
|
|
|
|
QFileInfo finfo(zone);
|
|
|
|
if (finfo.size() <= 4) { // decimal 000\n
|
|
|
|
if (zone.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
|
|
|
QTextStream in(&zone);
|
|
|
|
return in.readLine(100).toInt();
|
|
|
|
}
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
|
|
|
}
|
2023-05-12 09:20:46 +02:00
|
|
|
return -1;
|
2023-05-15 14:05:55 +02:00
|
|
|
}
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
|
|
|
|
2023-05-08 12:34:01 +02:00
|
|
|
CalcState CALCULATE_LIBRARY_API init_tariff(parking_tariff_t **tariff, char const *config_file) {
|
2023-04-24 15:31:46 +02:00
|
|
|
*tariff = new Configuration();
|
|
|
|
|
2023-05-08 12:34:01 +02:00
|
|
|
CalcState calcState;
|
2023-05-11 13:57:31 +02:00
|
|
|
#if __linux__
|
|
|
|
|
2023-04-24 15:31:46 +02:00
|
|
|
int const zone = get_zone_nr();
|
2023-05-05 13:19:02 +02:00
|
|
|
|
|
|
|
// DEBUG
|
|
|
|
qCritical() << "init_tariff:";
|
|
|
|
qCritical() << " ... zone = " << zone;
|
|
|
|
|
|
|
|
if (zone <= 0) {
|
2023-05-09 13:05:02 +02:00
|
|
|
delete *tariff;
|
|
|
|
*tariff = nullptr;
|
2023-05-08 12:34:01 +02:00
|
|
|
return calcState.set(CalcState::State::ERROR_PARSING_ZONE_NR);
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QString confFile(config_file);
|
|
|
|
if (!confFile.endsWith(QChar('/'))) {
|
|
|
|
confFile += "/";
|
|
|
|
}
|
|
|
|
|
2023-05-05 13:19:02 +02:00
|
|
|
char buffer[32];
|
2023-04-24 15:31:46 +02:00
|
|
|
memset(buffer, 0x00, sizeof(buffer));
|
|
|
|
snprintf(buffer, sizeof(buffer)-1, "tariff%02d.json", zone);
|
|
|
|
confFile += buffer;
|
2023-05-11 13:57:31 +02:00
|
|
|
#else // windows
|
|
|
|
QString confFile(config_file);
|
|
|
|
#endif
|
2023-04-24 15:31:46 +02:00
|
|
|
|
2023-05-05 13:19:36 +02:00
|
|
|
// DEBUG
|
|
|
|
qCritical() << " ... confFile = " << confFile;
|
|
|
|
|
2023-04-24 15:31:46 +02:00
|
|
|
QFile fname(confFile);
|
2023-05-08 12:34:01 +02:00
|
|
|
if (fname.exists() &&
|
2023-05-15 14:05:55 +02:00
|
|
|
fname.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
2023-05-05 13:19:36 +02:00
|
|
|
// DEBUG
|
2023-05-08 12:34:01 +02:00
|
|
|
qCritical() << " ... confFile is open";
|
2023-05-05 13:19:36 +02:00
|
|
|
|
2023-05-08 12:34:01 +02:00
|
|
|
QString json = fname.readAll();
|
|
|
|
if (! (*tariff)->ParseJson(*tariff, json.toStdString().c_str())) {
|
2023-05-09 13:05:02 +02:00
|
|
|
delete *tariff;
|
|
|
|
*tariff = nullptr;
|
2024-07-26 10:59:45 +02:00
|
|
|
qCritical() << " ... error parsing tariff";
|
2023-05-09 11:52:17 +02:00
|
|
|
return calcState.set(CalcState::State::ERROR_PARSING_TARIFF);
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
2023-05-08 12:34:01 +02:00
|
|
|
} else {
|
2023-05-09 13:05:02 +02:00
|
|
|
delete *tariff;
|
|
|
|
*tariff = nullptr;
|
2024-07-26 10:59:45 +02:00
|
|
|
qCritical() << " ... error loading tariff";
|
2023-05-09 11:52:17 +02:00
|
|
|
return calcState.set(CalcState::State::ERROR_LOADING_TARIFF);
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
|
|
|
|
2023-05-09 11:52:17 +02:00
|
|
|
qCritical() << "init_tariff: Parsing tariff config (" << confFile << ")";
|
2023-05-05 13:19:36 +02:00
|
|
|
|
2023-05-08 12:34:01 +02:00
|
|
|
return calcState;
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
|
|
|
|
2023-05-02 09:46:17 +02:00
|
|
|
void CALCULATE_LIBRARY_API free_tariff(parking_tariff_t *tariff) {
|
2023-05-09 13:05:02 +02:00
|
|
|
if (tariff != nullptr) {
|
|
|
|
delete tariff;
|
|
|
|
}
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
|
|
|
|
2023-12-07 16:28:17 +01:00
|
|
|
//
|
|
|
|
// UpDown 1 -> up; 0 -> down
|
2024-07-25 09:48:12 +02:00
|
|
|
int CALCULATE_LIBRARY_API compute_next_timestep(parking_tariff_t *tariff, int currentTimeMinutes,
|
|
|
|
int UpDown, PermitType const &permitType)
|
2023-12-07 16:28:17 +01:00
|
|
|
{
|
2024-07-25 09:48:12 +02:00
|
|
|
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() currentTimeMinutes: " << currentTimeMinutes;
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() up/down (1=up, 0=down): " << UpDown;
|
2024-01-23 10:55:00 +01:00
|
|
|
|
2024-07-29 17:31:33 +02:00
|
|
|
// FIXME
|
|
|
|
//std::optional<ATBPaymentOption> paymentOption = tariff->getPaymentOptionForKey(permitType.get());
|
|
|
|
//if (!paymentOption.has_value()) {
|
|
|
|
// qCritical() << " compute_next_timestep() ERROR";
|
|
|
|
// return currentTimeMinutes;
|
|
|
|
//}
|
2024-07-25 09:48:12 +02:00
|
|
|
|
2024-09-02 17:09:26 +02:00
|
|
|
int paymentOptionIndex = getPaymentOptionIndex(*tariff);
|
|
|
|
if (paymentOptionIndex == -1) {
|
|
|
|
paymentOptionIndex = tariff->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
|
|
|
|
2024-08-01 16:00:10 +02:00
|
|
|
int const &pop_plus_steps = tariff->getPaymentOptions(paymentOptionIndex).pop_plus_steps;
|
|
|
|
int const &pop_minus_steps = tariff->getPaymentOptions(paymentOptionIndex).pop_minus_steps;
|
|
|
|
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() payment option index: " << paymentOptionIndex;
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() plus steps: " << pop_plus_steps;
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() minus steps: " << pop_minus_steps;
|
2024-07-25 09:48:12 +02:00
|
|
|
|
2023-12-07 17:00:28 +01:00
|
|
|
Configuration const *cfg = tariff;
|
|
|
|
|
|
|
|
// compute payment method id (e.g. Linear=3, Steps=4)
|
|
|
|
PaymentMethod const paymentMethodId = Utilities::getPaymentMethodId(cfg);
|
|
|
|
switch (paymentMethodId) {
|
|
|
|
case PaymentMethod::Progressive:
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() paymentMethodId: Progressive";
|
2023-12-07 17:00:28 +01:00
|
|
|
break;
|
|
|
|
case PaymentMethod::Degressive:
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() paymentMethodId: Degressive";
|
2023-12-07 17:00:28 +01:00
|
|
|
break;
|
|
|
|
case PaymentMethod::Linear:
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() paymentMethodId: Linear";
|
2023-12-07 17:00:28 +01:00
|
|
|
break;
|
|
|
|
case PaymentMethod::Steps:
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() paymentMethodId: Steps";
|
2023-12-07 17:00:28 +01:00
|
|
|
break;
|
|
|
|
case PaymentMethod::Undefined:
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() paymentMethodId: Undefined";
|
2023-12-07 17:00:28 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2023-12-07 16:28:17 +01:00
|
|
|
// use tariff with structure as for instance Schnau, Koenigsee:
|
|
|
|
// without given YearPeriod, SpecialDays and SpecialDaysWorktime
|
2024-02-21 11:11:09 +01:00
|
|
|
if ((paymentMethodId == PaymentMethod::Steps) ||
|
|
|
|
// progressive tariff: e.g. Neuhauser, Kirchdorf (743)
|
2024-08-02 10:50:25 +02:00
|
|
|
(paymentMethodId == PaymentMethod::Progressive) ||
|
|
|
|
// degressive tariff: e.g. Fuchs Technik (500)
|
|
|
|
(paymentMethodId == PaymentMethod::Degressive))
|
2023-12-07 16:28:17 +01:00
|
|
|
{
|
2024-07-25 09:48:12 +02:00
|
|
|
QList<int> &stepList = Calculator::GetInstance().GetTimeSteps(tariff, paymentOptionIndex);
|
2024-08-02 10:50:25 +02:00
|
|
|
|
2024-07-16 16:50:11 +02:00
|
|
|
int const size = stepList.size();
|
|
|
|
if (size == 0) {
|
|
|
|
qCritical() << "compute_next_timestep() *ERROR empty step-list*";
|
|
|
|
return currentTimeMinutes;
|
|
|
|
}
|
|
|
|
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() first time step:" << stepList[0];
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() timeSteps:" << stepList;
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() currentTimeInMinutes:" << currentTimeMinutes;
|
2024-07-16 16:50:11 +02:00
|
|
|
|
|
|
|
// consider time shift: the step-list might have been computed at a
|
|
|
|
// slightly different time point
|
2024-08-01 16:00:10 +02:00
|
|
|
|
|
|
|
int currentStepIndex = stepList.indexOf(currentTimeMinutes);
|
|
|
|
if (currentStepIndex == -1) {
|
|
|
|
unsigned minimalDistance = ~0;
|
|
|
|
int j = -1;
|
|
|
|
for (int i = 0; i < stepList.size(); ++i) {
|
|
|
|
unsigned distance = std::abs(stepList[i] - currentTimeMinutes);
|
|
|
|
if (distance < minimalDistance) {
|
|
|
|
minimalDistance = distance;
|
|
|
|
j = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// max. tolerance set to 3 minutes
|
2024-09-02 17:11:48 +02:00
|
|
|
// unsigned const tolerance = std::min(minimalDistance, (unsigned)(3));
|
2024-08-01 16:00:10 +02:00
|
|
|
if (j != -1) {
|
|
|
|
stepList[j] = currentTimeMinutes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
2024-07-16 16:50:11 +02:00
|
|
|
int maxStep = -1;
|
|
|
|
if (size >= 2) {
|
|
|
|
maxStep = stepList[1] - stepList[0];
|
|
|
|
}
|
2024-08-01 16:00:10 +02:00
|
|
|
// max. tolerance set to 5 minutes
|
|
|
|
int const tolerance = (maxStep == -1) ? 5 : std::min(maxStep, 5);
|
2024-07-23 12:48:50 +02:00
|
|
|
for (int i=0; i < stepList.size(); ++i) {
|
|
|
|
if (std::abs(stepList[i] - currentTimeMinutes) <= tolerance) {
|
|
|
|
qCritical().noquote()
|
2024-08-01 16:00:10 +02:00
|
|
|
<< __LINE__ << QString("compute_next_timestep() correction stepList[%1]=%2 -> %3:")
|
2024-07-23 12:48:50 +02:00
|
|
|
.arg(i).arg(stepList[0]).arg(currentTimeMinutes);
|
|
|
|
stepList[i] = currentTimeMinutes;
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() NEW timeSteps:" << stepList;
|
2024-07-23 12:48:50 +02:00
|
|
|
}
|
2024-07-16 16:50:11 +02:00
|
|
|
}
|
2023-12-07 16:28:17 +01:00
|
|
|
|
|
|
|
int currentStepIndex = stepList.indexOf(currentTimeMinutes);
|
2024-08-01 16:00:10 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
currentStepIndex = stepList.indexOf(currentTimeMinutes);
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() currentStepIndex (" << currentStepIndex << ")";
|
2023-12-07 16:28:17 +01:00
|
|
|
|
|
|
|
if (currentStepIndex == -1) {
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() *NO STEP* for currentTimeMinutes (" << currentTimeMinutes << ")";
|
2023-12-07 16:28:17 +01:00
|
|
|
return currentTimeMinutes;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (UpDown == 1) { // UP
|
|
|
|
if (stepList[currentStepIndex] == stepList.last()) {
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() *NO NEXT STEP* for currentTimeMinutes (" << currentTimeMinutes << ")";
|
2023-12-07 16:28:17 +01:00
|
|
|
return currentTimeMinutes;
|
|
|
|
}
|
|
|
|
else {
|
2024-08-01 16:00:10 +02:00
|
|
|
int const rest = currentStepIndex % pop_plus_steps;
|
|
|
|
|
|
|
|
if (rest) {
|
|
|
|
currentStepIndex -= rest;
|
|
|
|
}
|
|
|
|
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() currentStepIndex (" << currentStepIndex << ")";
|
|
|
|
|
|
|
|
int const nextStepIndex = currentStepIndex + pop_plus_steps;
|
|
|
|
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() next step index:" << nextStepIndex;
|
|
|
|
if (nextStepIndex >= 0 && nextStepIndex < stepList.size()) {
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() return next time step:" << stepList[nextStepIndex];
|
|
|
|
return stepList[nextStepIndex];
|
|
|
|
}
|
2023-12-07 16:28:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (UpDown == 0) { // DOWN
|
|
|
|
if (stepList[currentStepIndex] == stepList.first()) {
|
2024-08-01 16:00:10 +02:00
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() *NO PREVIOUS STEP* for currentTimeMinutes (" << currentTimeMinutes << ")";
|
2023-12-07 16:28:17 +01:00
|
|
|
return currentTimeMinutes;
|
|
|
|
}
|
|
|
|
else {
|
2024-08-01 16:00:10 +02:00
|
|
|
int const nextStepIndex = currentStepIndex - pop_minus_steps;
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() next step index:" << nextStepIndex;
|
|
|
|
if (nextStepIndex >= 0 && nextStepIndex < stepList.size()) {
|
|
|
|
qCritical() << __LINE__ << "compute_next_timestep() return next time step:" << stepList[nextStepIndex];
|
|
|
|
return stepList[nextStepIndex];
|
|
|
|
}
|
2023-12-07 16:28:17 +01:00
|
|
|
}
|
|
|
|
}
|
2023-12-07 17:00:28 +01:00
|
|
|
} else
|
|
|
|
if (paymentMethodId == PaymentMethod::Linear) {
|
2023-12-07 16:28:17 +01:00
|
|
|
|
|
|
|
// currentTimeMinutes is the number of minutes actually used. This
|
|
|
|
// value is an offset from the start time and cannot be used as a
|
|
|
|
// QDateTime.
|
|
|
|
|
2023-12-08 10:16:52 +01:00
|
|
|
qCritical() << "compute_next_timestep() up/down (1=up, 0=down):" << UpDown;
|
2023-12-07 16:28:17 +01:00
|
|
|
|
|
|
|
// get minimal and maximal parking times
|
|
|
|
int const minParkingTime = Utilities::getMinimalParkingTime(cfg, paymentMethodId);
|
|
|
|
int const maxParkingTime = Utilities::getMaximalParkingTime(cfg, paymentMethodId);
|
|
|
|
|
2023-12-08 10:16:52 +01:00
|
|
|
qCritical() << " compute_next_timestep() maxParkingTime:" << maxParkingTime;
|
|
|
|
qCritical() << " compute_next_timestep() minParkingTime:" << minParkingTime;
|
2023-12-07 16:28:17 +01:00
|
|
|
|
|
|
|
// use the first (i.e. main duration step contained in the tariff json-file)
|
|
|
|
int firstDurationStep = Utilities::getFirstDurationStep(cfg, paymentMethodId);
|
2023-12-08 10:16:52 +01:00
|
|
|
firstDurationStep = ((UpDown == 1) ? firstDurationStep : -firstDurationStep);
|
|
|
|
|
|
|
|
qCritical() << " compute_next_timestep() firstDurationStep:" << firstDurationStep;
|
2023-12-07 16:28:17 +01:00
|
|
|
|
2023-12-11 09:00:24 +01:00
|
|
|
int const nextTimeStep = currentTimeMinutes + firstDurationStep;
|
2023-12-08 10:16:52 +01:00
|
|
|
|
2023-12-11 09:00:24 +01:00
|
|
|
if (nextTimeStep >= minParkingTime && nextTimeStep <= maxParkingTime) {
|
2023-12-08 10:16:52 +01:00
|
|
|
qCritical() << " compute_next_timestep() nextTimeStep:" << nextTimeStep;
|
2023-12-07 16:28:17 +01:00
|
|
|
return nextTimeStep;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qCritical() << "compute_next_timestep() *CAN NOT COMPUTE* for currentTimeMinutes (" << currentTimeMinutes << ")";
|
|
|
|
return currentTimeMinutes;
|
|
|
|
}
|
2023-05-16 15:31:53 +02:00
|
|
|
|
|
|
|
// this is currently not used
|
2023-05-08 12:34:01 +02:00
|
|
|
CalcState CALCULATE_LIBRARY_API compute_price_for_parking_ticket(
|
2023-05-15 14:05:55 +02:00
|
|
|
parking_tariff_t *tariff,
|
2024-07-25 09:48:12 +02:00
|
|
|
time_t start_parking_time, // in minutes
|
|
|
|
time_t end_parking_time, // netto time in minutes
|
|
|
|
struct price_t *price,
|
|
|
|
PermitType permitType) { // permitType maps to product
|
2024-07-29 17:32:15 +02:00
|
|
|
|
2023-05-08 12:34:01 +02:00
|
|
|
CalcState calcState;
|
2024-07-25 09:48:12 +02:00
|
|
|
|
2024-09-06 12:05:41 +02:00
|
|
|
QDate const d(1970, 1, 1);
|
|
|
|
QTime const t(0, 0, 0);
|
|
|
|
QDateTime start(d, t, Qt::UTC);
|
|
|
|
start = start.toLocalTime().addSecs(start_parking_time * 60);
|
|
|
|
QDateTime end(start);
|
|
|
|
|
|
|
|
int paymentOptionIndex = getPaymentOptionIndex(*tariff, start);
|
2024-09-02 17:09:26 +02:00
|
|
|
if (paymentOptionIndex == -1) {
|
|
|
|
paymentOptionIndex = tariff->getPaymentOptionIndex(permitType.get());
|
|
|
|
}
|
2024-07-25 09:48:12 +02:00
|
|
|
|
|
|
|
double minMin = tariff->PaymentOption.find(tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id)->second.pop_min_time;
|
|
|
|
double maxMin = tariff->PaymentOption.find(tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id)->second.pop_max_time;
|
2023-05-08 12:34:01 +02:00
|
|
|
|
|
|
|
if (minMin < 0 || maxMin < 0 || maxMin < minMin) {
|
2023-05-10 13:33:28 +02:00
|
|
|
calcState.setDesc(QString("minMin=%1, maxMin=%2").arg(minMin).arg(maxMin));
|
2023-05-08 12:34:01 +02:00
|
|
|
return calcState.set(CalcState::State::WRONG_PARAM_VALUES);
|
|
|
|
}
|
|
|
|
|
2023-04-24 15:31:46 +02:00
|
|
|
int const duration = end_parking_time - start_parking_time;
|
2023-05-08 12:34:01 +02:00
|
|
|
if (duration < 0) {
|
|
|
|
calcState.setDesc(QString("end=%1, start=%2")
|
2023-05-15 14:05:55 +02:00
|
|
|
.arg(end_parking_time, start_parking_time));
|
2024-06-05 17:03:10 +02:00
|
|
|
return calcState.set(CalcState::State::NEGATIVE_PARKING_TIME);
|
2023-05-08 12:34:01 +02:00
|
|
|
}
|
|
|
|
if (duration > maxMin) {
|
2023-05-10 13:33:28 +02:00
|
|
|
calcState.setDesc(QString("duration=%1, maxMin=%2").arg(duration).arg(maxMin));
|
2023-05-08 12:34:01 +02:00
|
|
|
return calcState.set(CalcState::State::ABOVE_MAX_PARKING_TIME);
|
|
|
|
}
|
|
|
|
if (duration < minMin) {
|
2023-05-10 13:33:28 +02:00
|
|
|
calcState.setDesc(QString("duration=%1, minMin=%2").arg(duration).arg(minMin));
|
2023-05-08 12:34:01 +02:00
|
|
|
return calcState.set(CalcState::State::BELOW_MIN_PARKING_TIME);
|
|
|
|
}
|
2023-04-24 15:31:46 +02:00
|
|
|
if (duration == 0) {
|
2023-05-08 12:34:01 +02:00
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
2023-05-08 12:34:01 +02:00
|
|
|
|
2024-09-06 12:05:41 +02:00
|
|
|
//QDate const d(1970, 1, 1);
|
|
|
|
//QTime const t(0, 0, 0);
|
|
|
|
//QDateTime start(d, t, Qt::UTC);
|
|
|
|
//start = start.toLocalTime().addSecs(start_parking_time * 60);
|
|
|
|
//QDateTime end(start);
|
2023-05-08 12:34:01 +02:00
|
|
|
if (start.isValid()) {
|
2024-01-23 10:55:00 +01:00
|
|
|
double cost = Calculator::GetInstance().GetCostFromDuration(
|
2023-11-22 16:27:41 +01:00
|
|
|
tariff,
|
2024-07-25 09:48:12 +02:00
|
|
|
tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id,
|
2023-05-16 15:31:53 +02:00
|
|
|
start,
|
|
|
|
end,
|
2023-05-15 14:05:55 +02:00
|
|
|
duration, false, true);
|
2024-07-25 09:48:12 +02:00
|
|
|
double minCost = tariff->PaymentOption.find(tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id)->second.pop_min_price;
|
2023-05-08 12:34:01 +02:00
|
|
|
if (cost < minCost) {
|
2024-07-25 09:48:12 +02:00
|
|
|
calcState.setDesc(QString("line=%1 minCost=%2, cost=%3").arg(__LINE__).arg(minCost).arg(cost));
|
2023-05-08 12:34:01 +02:00
|
|
|
return calcState.set(CalcState::State::BELOW_MIN_PARKING_PRICE);
|
|
|
|
}
|
|
|
|
price->units = cost;
|
|
|
|
price->netto = cost;
|
2023-05-09 11:40:14 +02:00
|
|
|
} else {
|
|
|
|
return calcState.set(CalcState::State::INVALID_START_DATE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
|
|
|
}
|
|
|
|
|
2023-05-10 16:16:24 +02:00
|
|
|
CalcState CALCULATE_LIBRARY_API compute_price_for_parking_ticket(
|
2023-05-15 14:05:55 +02:00
|
|
|
parking_tariff_t *tariff,
|
2024-08-01 16:04:57 +02:00
|
|
|
QDateTime &start_parking_time_,
|
2023-05-16 15:31:53 +02:00
|
|
|
int netto_parking_time,
|
|
|
|
QDateTime &end_parking_time,
|
2024-02-22 09:42:02 +01:00
|
|
|
struct price_t *price,
|
2024-07-26 10:59:45 +02:00
|
|
|
PermitType permitType,
|
2024-02-22 09:42:02 +01:00
|
|
|
bool prepaid)
|
2023-05-16 15:31:53 +02:00
|
|
|
{
|
2023-05-10 16:16:24 +02:00
|
|
|
CalcState calcState;
|
2024-04-19 13:28:35 +02:00
|
|
|
|
2024-08-01 16:04:57 +02:00
|
|
|
QDateTime start_parking_time(start_parking_time_);
|
|
|
|
|
2024-09-06 12:05:41 +02:00
|
|
|
int paymentOptionIndex = getPaymentOptionIndex(*tariff, start_parking_time);
|
2024-09-02 17:09:26 +02:00
|
|
|
if (paymentOptionIndex == -1) {
|
|
|
|
paymentOptionIndex = tariff->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
2024-04-19 13:28:35 +02:00
|
|
|
|
|
|
|
double minMin = tariff->getPaymentOptions(paymentOptionIndex).pop_min_time;
|
|
|
|
double maxMin = tariff->getPaymentOptions(paymentOptionIndex).pop_max_time;
|
2023-05-10 16:16:24 +02:00
|
|
|
|
2023-05-11 09:58:15 +02:00
|
|
|
// DEBUG
|
|
|
|
qCritical() << "compute_price_for_parking_ticket() " << endl
|
2024-05-08 16:34:07 +02:00
|
|
|
<< " paymentOptionIndex: " << paymentOptionIndex << endl
|
2023-05-11 09:58:15 +02:00
|
|
|
<< " start_parking_time: " << start_parking_time << endl
|
2023-05-16 15:31:53 +02:00
|
|
|
<< " netto_parking_time: " << netto_parking_time << endl
|
2024-07-24 12:34:05 +02:00
|
|
|
<< " start + netto: " << start_parking_time.addSecs(netto_parking_time * 60) << endl
|
2023-05-11 09:58:15 +02:00
|
|
|
<< " minMin: " << minMin << endl
|
2024-07-29 17:32:15 +02:00
|
|
|
<< " maxMin: " << maxMin
|
2024-08-01 16:04:57 +02:00
|
|
|
<< " prepaid: " << prepaid
|
2024-07-29 17:32:15 +02:00
|
|
|
<< " permitType: " << permitType.toString();
|
2023-05-11 09:58:15 +02:00
|
|
|
|
2023-05-16 15:31:53 +02:00
|
|
|
if (netto_parking_time < 0) {
|
2023-05-10 16:16:24 +02:00
|
|
|
calcState.setDesc(QString("end=%1, start=%2")
|
2023-05-15 14:05:55 +02:00
|
|
|
.arg(end_parking_time.toString(Qt::ISODate),
|
|
|
|
start_parking_time.toString(Qt::ISODate)));
|
2024-06-05 17:03:10 +02:00
|
|
|
return calcState.set(CalcState::State::NEGATIVE_PARKING_TIME);
|
2023-05-10 16:16:24 +02:00
|
|
|
}
|
2023-05-16 15:31:53 +02:00
|
|
|
if (netto_parking_time > maxMin) {
|
|
|
|
calcState.setDesc(QString("duration=%1, maxMin=%2").arg(netto_parking_time).arg(maxMin));
|
2023-05-10 16:16:24 +02:00
|
|
|
return calcState.set(CalcState::State::ABOVE_MAX_PARKING_TIME);
|
|
|
|
}
|
2023-05-16 15:31:53 +02:00
|
|
|
if (netto_parking_time < minMin) {
|
|
|
|
calcState.setDesc(QString("duration=%1, minMin=%2").arg(netto_parking_time).arg(minMin));
|
2023-05-10 16:16:24 +02:00
|
|
|
return calcState.set(CalcState::State::BELOW_MIN_PARKING_TIME);
|
|
|
|
}
|
2023-05-16 15:31:53 +02:00
|
|
|
if (netto_parking_time == 0) {
|
2023-05-10 16:16:24 +02:00
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
|
|
|
}
|
|
|
|
|
2024-04-08 13:58:24 +02:00
|
|
|
double cost = -1;
|
2023-05-10 16:16:24 +02:00
|
|
|
if (start_parking_time.isValid()) {
|
2024-07-26 10:59:45 +02:00
|
|
|
if (tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id == PaymentMethod::Steps ||
|
|
|
|
tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id == PaymentMethod::Degressive) {
|
2024-04-19 13:28:35 +02:00
|
|
|
// hier muesste man unterscheiden: uebertrag oder nicht?
|
|
|
|
calcState = Calculator::GetInstance().isParkingAllowed(tariff, start_parking_time,
|
|
|
|
netto_parking_time, paymentOptionIndex);
|
2024-04-08 13:58:24 +02:00
|
|
|
if (calcState.getStatus() == CalcState::State::OUTSIDE_ALLOWED_PARKING_TIME) {
|
2024-08-01 16:04:57 +02:00
|
|
|
qCritical() << "(" << __func__ << ":" << __LINE__ << ")"
|
|
|
|
<< calcState.toString();
|
2024-04-08 13:58:24 +02:00
|
|
|
return calcState;
|
|
|
|
}
|
2024-07-26 10:59:45 +02:00
|
|
|
|
2024-08-01 16:04:57 +02:00
|
|
|
QList<int> tlist = Calculator::GetInstance().GetTimeSteps(tariff);
|
|
|
|
Q_UNUSED(tlist);
|
2024-07-26 10:59:45 +02:00
|
|
|
|
2024-08-02 11:08:32 +02:00
|
|
|
// compute cost (price)
|
2024-04-19 13:28:35 +02:00
|
|
|
cost = Calculator::GetInstance().GetCostFromDuration(tariff, start_parking_time, netto_parking_time, paymentOptionIndex);
|
2024-07-26 17:01:44 +02:00
|
|
|
|
|
|
|
int weekDay = start_parking_time.date().dayOfWeek();
|
2024-08-01 16:04:57 +02:00
|
|
|
int pop_carry_over_option_id = tariff->getPaymentOptions(paymentOptionIndex).pop_carry_over_option_id;
|
2024-08-02 11:08:32 +02:00
|
|
|
qCritical() << __func__ << __LINE__ << "configured carry-over-id" << pop_carry_over_option_id;
|
2024-08-01 16:04:57 +02:00
|
|
|
|
|
|
|
std::optional<ATBPeriodYear> yperiod = Utilities::GetYearPeriodActive(tariff, start_parking_time);
|
|
|
|
if (yperiod.has_value()) {
|
|
|
|
ATBPeriodYear const &period = yperiod.value();
|
|
|
|
pop_carry_over_option_id = period.pye_id;
|
2024-08-02 11:08:32 +02:00
|
|
|
qCritical() << __func__ << __LINE__ << "re-computed carry-over-id" << pop_carry_over_option_id;
|
2024-08-01 16:04:57 +02:00
|
|
|
}
|
|
|
|
|
2024-07-26 17:01:44 +02:00
|
|
|
QTime const carryOverStart = tariff->TariffCarryOverOptions.find(pop_carry_over_option_id)->second.carryover[weekDay].static_start;
|
|
|
|
int const carryOverDuration = tariff->TariffCarryOverOptions.find(pop_carry_over_option_id)->second.carryover[weekDay].duration;
|
|
|
|
|
2024-08-01 16:04:57 +02:00
|
|
|
qCritical() << __func__ << __LINE__ << "carryOverStart" << carryOverStart.toString(Qt::ISODate);
|
|
|
|
qCritical() << __func__ << __LINE__ << "carryOverDuration" << carryOverDuration;
|
|
|
|
|
|
|
|
// handle prepaid option
|
|
|
|
QDateTime effectiveStartTime(start_parking_time);
|
|
|
|
int const prepaid_option_id = tariff->getPaymentOptions(paymentOptionIndex).pop_prepaid_option_id;
|
|
|
|
std::optional<ATBPrepaid> prepaidOption = tariff->getPrepaidType(prepaid_option_id);
|
|
|
|
if (prepaidOption.has_value()) {
|
|
|
|
ATBPrepaid const &p = prepaidOption.value();
|
2024-08-02 11:08:52 +02:00
|
|
|
if (p.never) {
|
|
|
|
qCritical() << __func__ << __LINE__ << "prepaid: no";
|
|
|
|
} else {
|
|
|
|
if (start_parking_time.time() < p.static_end) { // static_end: e.g. 08:00:00
|
|
|
|
effectiveStartTime.setTime(p.static_end);
|
|
|
|
} else
|
|
|
|
if (start_parking_time.time() > p.static_start) { // static_start: e.g. 22:00:00
|
|
|
|
effectiveStartTime.setTime(p.static_start);
|
|
|
|
}
|
2024-08-01 16:04:57 +02:00
|
|
|
}
|
2024-07-26 17:01:44 +02:00
|
|
|
}
|
|
|
|
|
2024-08-01 16:04:57 +02:00
|
|
|
// handle carry over
|
|
|
|
int minutesUntilCarryOver = effectiveStartTime.time().secsTo(carryOverStart) / 60;
|
|
|
|
if (netto_parking_time > minutesUntilCarryOver) {
|
|
|
|
int const rest = netto_parking_time - minutesUntilCarryOver;
|
|
|
|
QDateTime s(effectiveStartTime);
|
|
|
|
s = s.addSecs(minutesUntilCarryOver * 60);
|
|
|
|
s = s.addSecs(carryOverDuration * 60);
|
|
|
|
end_parking_time = s.addSecs(rest * 60);
|
|
|
|
} else {
|
|
|
|
end_parking_time = effectiveStartTime.addSecs(netto_parking_time*60);
|
|
|
|
}
|
2024-05-15 15:23:12 +02:00
|
|
|
|
2024-09-02 17:11:48 +02:00
|
|
|
|
|
|
|
end_parking_time.setTime(QTime(end_parking_time.time().hour(),
|
|
|
|
end_parking_time.time().minute(), 0));
|
|
|
|
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "end-parking-time:" << end_parking_time.toString(Qt::ISODate);
|
|
|
|
|
|
|
|
weekDay = end_parking_time.date().dayOfWeek();
|
|
|
|
|
|
|
|
for (auto[itr, rangeEnd] = tariff->WeekDays.equal_range((Qt::DayOfWeek)weekDay); itr != rangeEnd; ++itr) {
|
|
|
|
ATBWeekDay const &wd = itr->second;
|
2024-09-03 11:19:23 +02:00
|
|
|
bool const parkTimeLimitViolated = wd.getTariffCarryOverSettings().parkingTimeLimitExceeded(effectiveStartTime,
|
|
|
|
end_parking_time,
|
|
|
|
paymentOptionIndex);
|
2024-09-02 17:11:48 +02:00
|
|
|
if (parkTimeLimitViolated) {
|
|
|
|
calcState.setDesc(QString("line=%1 endTime=%2: park-time-limit violated").arg(__LINE__)
|
|
|
|
.arg(end_parking_time.time().toString(Qt::ISODate)));
|
|
|
|
return calcState.set(CalcState::State::ABOVE_MAX_PARKING_TIME);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-08 13:58:24 +02:00
|
|
|
} else {
|
|
|
|
cost = Calculator::GetInstance().GetCostFromDuration(
|
2023-11-22 16:27:41 +01:00
|
|
|
tariff,
|
|
|
|
tariff->getPaymentOptions().pop_payment_method_id,
|
2023-05-16 15:31:53 +02:00
|
|
|
start_parking_time, // starting time
|
|
|
|
end_parking_time, // return value: end time
|
|
|
|
netto_parking_time, // minutes, netto
|
2024-02-22 09:42:02 +01:00
|
|
|
false, prepaid);
|
2024-04-08 13:58:24 +02:00
|
|
|
}
|
2024-07-26 10:59:45 +02:00
|
|
|
// qCritical() << __func__ << __LINE__;
|
2024-04-19 13:28:35 +02:00
|
|
|
double minCost = tariff->getPaymentOptions(paymentOptionIndex).pop_min_price;
|
2023-05-10 16:16:24 +02:00
|
|
|
if (cost < minCost) {
|
2024-07-25 09:48:12 +02:00
|
|
|
calcState.setDesc(QString("line=%1 minCost=%2, cost=%3").arg(__LINE__).arg(minCost).arg(cost));
|
2023-05-10 16:16:24 +02:00
|
|
|
return calcState.set(CalcState::State::BELOW_MIN_PARKING_PRICE);
|
|
|
|
}
|
2023-05-11 09:58:15 +02:00
|
|
|
|
|
|
|
// DEBUG
|
2024-08-01 16:04:57 +02:00
|
|
|
//qCritical() << __LINE__ << " end_parking_time: " << end_parking_time;
|
|
|
|
//qCritical() << __LINE__ << " -> calculated cost (netto): " << cost;
|
2023-05-11 09:58:15 +02:00
|
|
|
|
2024-04-08 13:58:24 +02:00
|
|
|
price->brutto = price->vat = price->vat_percentage = 0;
|
2023-05-10 16:16:24 +02:00
|
|
|
price->units = cost;
|
|
|
|
price->netto = cost;
|
2024-04-08 13:58:24 +02:00
|
|
|
|
2023-05-10 16:16:24 +02:00
|
|
|
} else {
|
|
|
|
return calcState.set(CalcState::State::INVALID_START_DATE);
|
|
|
|
}
|
|
|
|
|
2024-08-02 11:09:20 +02:00
|
|
|
if (end_parking_time.time().hour() == 0 && end_parking_time.time().minute() == 0) {
|
|
|
|
end_parking_time = end_parking_time.addDays(-1);
|
|
|
|
end_parking_time.setTime(QTime(23, 59, 0));
|
|
|
|
}
|
|
|
|
qCritical() << __func__ << __LINE__ << "end_parking_time" << end_parking_time.toString(Qt::ISODate);
|
|
|
|
|
2023-05-10 16:16:24 +02:00
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
|
|
|
}
|
|
|
|
|
2023-05-09 11:40:14 +02:00
|
|
|
CalcState CALCULATE_LIBRARY_API compute_duration_for_parking_ticket(
|
2023-05-15 14:05:55 +02:00
|
|
|
parking_tariff_t *tariff,
|
|
|
|
time_t start_parking_time,
|
|
|
|
double price,
|
2024-07-25 09:48:12 +02:00
|
|
|
QString &duration,
|
|
|
|
PermitType permitType) {
|
2023-05-09 11:40:14 +02:00
|
|
|
CalcState calcState;
|
|
|
|
QDate const d(1970, 1, 1);
|
|
|
|
QTime const t(0, 0, 0);
|
|
|
|
QDateTime start(d, t, Qt::UTC);
|
|
|
|
start = start.toLocalTime().addSecs(start_parking_time * 60);
|
|
|
|
if (start.isValid()) {
|
|
|
|
QString cs = start.toString(Qt::ISODate);
|
2023-05-11 09:58:15 +02:00
|
|
|
|
2024-07-30 15:32:58 +02:00
|
|
|
bool prepaid = true;
|
2024-09-06 12:05:41 +02:00
|
|
|
int paymentOptionIndex = getPaymentOptionIndex(*tariff, start);
|
|
|
|
qCritical() << " payment option index: " << paymentOptionIndex;
|
2024-09-02 17:09:26 +02:00
|
|
|
if (paymentOptionIndex == -1) {
|
|
|
|
paymentOptionIndex = tariff->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
|
|
|
|
2024-07-30 15:32:58 +02:00
|
|
|
int prepaid_option_id = tariff->getPaymentOptions(paymentOptionIndex).pop_prepaid_option_id;
|
|
|
|
if (prepaid_option_id == 2) { // see tariff03.json for 502: 2 means no prepaid-option
|
|
|
|
prepaid = false;
|
|
|
|
}
|
|
|
|
bool const nextDay = false;
|
|
|
|
|
2023-05-11 09:58:15 +02:00
|
|
|
// DEBUG
|
2024-07-30 15:32:58 +02:00
|
|
|
qCritical() << "compute_duration_for_parking_ticket(1): ";
|
|
|
|
qCritical() << " payment option index: " << paymentOptionIndex;
|
|
|
|
qCritical() << " prepaid: " << prepaid;
|
|
|
|
qCritical() << " start (cs): " << cs;
|
|
|
|
qCritical() << " price: " << price;
|
2023-05-11 09:58:15 +02:00
|
|
|
|
2024-01-23 10:55:00 +01:00
|
|
|
duration = Calculator::GetInstance().GetDurationFromCost(tariff,
|
2024-07-30 15:32:58 +02:00
|
|
|
tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id,
|
2023-05-09 11:40:14 +02:00
|
|
|
cs.toLocal8Bit().constData(),
|
2024-07-30 15:32:58 +02:00
|
|
|
price, permitType, nextDay, prepaid).c_str();
|
2023-05-12 14:10:16 +02:00
|
|
|
QDateTime d = QDateTime::fromString(duration, Qt::ISODate);
|
2023-05-10 16:16:24 +02:00
|
|
|
if (!d.isValid()) {
|
|
|
|
calcState.setDesc(QString("ticketEndTime=%1").arg(duration));
|
|
|
|
return calcState.set(CalcState::State::WRONG_ISO_TIME_FORMAT);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return calcState.set(CalcState::State::INVALID_START_DATE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
CalcState CALCULATE_LIBRARY_API compute_duration_for_parking_ticket(
|
2023-05-15 14:05:55 +02:00
|
|
|
parking_tariff_t *tariff,
|
|
|
|
QDateTime const &start_parking_time,
|
|
|
|
double price,
|
2024-07-25 09:48:12 +02:00
|
|
|
QDateTime &ticketEndTime,
|
|
|
|
PermitType permitType)
|
2023-05-16 11:07:21 +02:00
|
|
|
{
|
2023-05-10 16:16:24 +02:00
|
|
|
CalcState calcState;
|
2024-07-30 15:32:58 +02:00
|
|
|
|
|
|
|
bool prepaid = true;
|
2024-09-02 17:09:26 +02:00
|
|
|
|
2024-09-06 12:05:41 +02:00
|
|
|
int paymentOptionIndex = getPaymentOptionIndex(*tariff, start_parking_time);
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "payment option index: " << paymentOptionIndex;
|
2024-09-02 17:09:26 +02:00
|
|
|
if (paymentOptionIndex == -1) {
|
|
|
|
paymentOptionIndex = tariff->getPaymentOptionIndex(permitType);
|
|
|
|
}
|
|
|
|
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << " payment option index: " << paymentOptionIndex;
|
|
|
|
|
2024-07-30 15:32:58 +02:00
|
|
|
int prepaid_option_id = tariff->getPaymentOptions(paymentOptionIndex).pop_prepaid_option_id;
|
2024-08-21 17:46:35 +02:00
|
|
|
|
2024-07-30 15:32:58 +02:00
|
|
|
if (prepaid_option_id == 2) {
|
|
|
|
prepaid = false;
|
|
|
|
}
|
|
|
|
bool const nextDay = false;
|
|
|
|
|
|
|
|
// DEBUG
|
2024-08-01 16:04:57 +02:00
|
|
|
if (DBG_LEVEL >= DBG_DEBUG) {
|
|
|
|
qCritical() << "compute_duration_for_parking_ticket(2): ";
|
|
|
|
qCritical() << " payment option index: " << paymentOptionIndex;
|
|
|
|
qCritical() << " prepaid: " << prepaid;
|
|
|
|
qCritical() << " price: " << price;
|
|
|
|
}
|
2024-07-30 15:32:58 +02:00
|
|
|
|
2023-05-10 16:16:24 +02:00
|
|
|
if (start_parking_time.isValid()) {
|
2024-08-21 15:39:44 +02:00
|
|
|
|
2024-08-21 17:46:35 +02:00
|
|
|
int const pop_time_step_config = tariff->getPaymentOptions(paymentOptionIndex).pop_time_step_config;
|
|
|
|
if (pop_time_step_config == (int)ATBTimeStepConfig::TimeStepConfig::STATIC) {
|
|
|
|
// handle prepaid option
|
|
|
|
QDateTime effectiveStartTime(start_parking_time);
|
|
|
|
int const prepaid_option_id = tariff->getPaymentOptions(paymentOptionIndex).pop_prepaid_option_id;
|
|
|
|
std::optional<ATBPrepaid> prepaidOption = tariff->getPrepaidType(prepaid_option_id);
|
|
|
|
if (prepaidOption.has_value()) {
|
|
|
|
ATBPrepaid const &p = prepaidOption.value();
|
|
|
|
if (p.never) {
|
|
|
|
qCritical() << __func__ << __LINE__ << "prepaid: no";
|
|
|
|
} else {
|
|
|
|
if (start_parking_time.time() < p.static_end) { // static_end: e.g. 08:00:00
|
|
|
|
effectiveStartTime.setTime(p.static_end);
|
|
|
|
} else
|
|
|
|
if (start_parking_time.time() > p.static_start) { // static_start: e.g. 22:00:00
|
|
|
|
effectiveStartTime.setTime(p.static_start);
|
|
|
|
}
|
2024-08-21 15:39:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-10 16:16:24 +02:00
|
|
|
QString cs = start_parking_time.toString(Qt::ISODate);
|
2024-01-23 10:55:00 +01:00
|
|
|
QString endTime = Calculator::GetInstance().GetDurationFromCost(
|
2023-11-22 16:27:41 +01:00
|
|
|
tariff,
|
|
|
|
tariff->getPaymentOptions().pop_payment_method_id,
|
2023-05-15 14:05:55 +02:00
|
|
|
cs.toLocal8Bit().constData(),
|
2024-07-30 15:32:58 +02:00
|
|
|
price, permitType, nextDay, prepaid).c_str();
|
2023-05-11 09:58:15 +02:00
|
|
|
|
2024-06-05 17:03:33 +02:00
|
|
|
if (endTime == CalcState::SUCCESS) {
|
|
|
|
calcState.setDesc(QString("SUCCESS"));
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::ERROR_PARSING_ZONE_NR) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::ERROR_LOADING_TARIFF) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::ERROR_PARSING_TARIFF) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::NEGATIVE_PARKING_TIME) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::INVALID_START_DATE) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::WRONG_PARAM_VALUES) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::WRONG_ISO_TIME_FORMAT) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::ABOVE_MAX_PARKING_TIME) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::BELOW_MIN_PARKING_TIME) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::BELOW_MIN_PARKING_PRICE) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::ABOVE_MAX_PARKING_PRICE) {
|
|
|
|
calcState.setDesc(CalcState::ABOVE_MAX_PARKING_PRICE);
|
|
|
|
calcState.setStatus(CalcState::ABOVE_MAX_PARKING_PRICE);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::OVERPAID) {
|
|
|
|
calcState.setDesc(CalcState::OVERPAID);
|
|
|
|
calcState.setStatus(CalcState::OVERPAID);
|
|
|
|
return calcState;
|
|
|
|
} else
|
|
|
|
if (endTime == CalcState::OUTSIDE_ALLOWED_PARKING_TIME) {
|
|
|
|
calcState.setStatus(endTime);
|
|
|
|
return calcState;
|
|
|
|
} else {
|
|
|
|
ticketEndTime = QDateTime::fromString(endTime,Qt::ISODate);
|
2023-05-11 09:58:15 +02:00
|
|
|
|
2024-09-02 17:13:11 +02:00
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "ticketEndTime:" << ticketEndTime.toString(Qt::ISODate);
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "step-config:" << pop_time_step_config;
|
|
|
|
|
2024-06-05 17:03:33 +02:00
|
|
|
if (!ticketEndTime.isValid()) {
|
|
|
|
calcState.setDesc(QString("ticketEndTime=%1").arg(endTime));
|
|
|
|
return calcState.set(CalcState::State::WRONG_ISO_TIME_FORMAT);
|
|
|
|
}
|
2024-08-02 13:15:11 +02:00
|
|
|
|
2024-08-21 17:46:35 +02:00
|
|
|
if (pop_time_step_config == (int)ATBTimeStepConfig::TimeStepConfig::STATIC) {
|
|
|
|
// handle carry over for ticket-end-time
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "ticketEndTime:" << ticketEndTime.toString(Qt::ISODate);
|
2024-08-21 15:40:55 +02:00
|
|
|
|
2024-08-21 17:46:35 +02:00
|
|
|
int weekDay = start_parking_time.date().dayOfWeek();
|
|
|
|
int pop_carry_over_option_id = tariff->getPaymentOptions(paymentOptionIndex).pop_carry_over_option_id;
|
2024-09-02 17:13:11 +02:00
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "configured carry-over-id" << pop_carry_over_option_id;
|
2024-08-21 15:40:55 +02:00
|
|
|
|
2024-08-21 17:46:35 +02:00
|
|
|
std::optional<ATBPeriodYear> yperiod = Utilities::GetYearPeriodActive(tariff, start_parking_time);
|
|
|
|
if (yperiod.has_value()) {
|
|
|
|
ATBPeriodYear const &period = yperiod.value();
|
|
|
|
pop_carry_over_option_id = period.pye_id;
|
2024-09-02 17:13:11 +02:00
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "re-computed carry-over-id" << pop_carry_over_option_id;
|
2024-08-21 17:46:35 +02:00
|
|
|
}
|
2024-08-21 15:40:55 +02:00
|
|
|
|
2024-08-21 17:46:35 +02:00
|
|
|
QTime carryOverStart;
|
|
|
|
QTime carryOverEnd;
|
|
|
|
int carryOverDuration = -1;
|
|
|
|
|
|
|
|
// using TariffCarryOverType = std::multimap<int, ATBCarryOver>;
|
|
|
|
std::multimap<int, ATBCarryOver>::const_iterator it;
|
|
|
|
if ((it = tariff->TariffCarryOverOptions.find(pop_carry_over_option_id)) !=
|
|
|
|
tariff->TariffCarryOverOptions.cend()) {
|
|
|
|
carryOverStart = it->second.carryover[weekDay].static_start;
|
|
|
|
carryOverEnd = it->second.carryover[weekDay].static_end;
|
|
|
|
carryOverDuration = it->second.carryover[weekDay].duration;
|
|
|
|
}
|
2024-08-21 15:40:55 +02:00
|
|
|
|
2024-08-21 17:46:35 +02:00
|
|
|
if (carryOverStart.isValid() && carryOverEnd.isValid()) {
|
2024-09-02 17:11:48 +02:00
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "carryOverStart" << carryOverStart.toString(Qt::ISODate);
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "carryOverEnd" << carryOverEnd.toString(Qt::ISODate);
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "carryOverDuration" << carryOverDuration;
|
2024-08-21 17:46:35 +02:00
|
|
|
}
|
2024-08-21 15:40:55 +02:00
|
|
|
|
2024-08-21 17:46:35 +02:00
|
|
|
if (carryOverStart.isValid() && carryOverEnd.isValid() && carryOverDuration != -1) {
|
2024-09-02 17:11:48 +02:00
|
|
|
// qCritical() << __func__ << __LINE__ << "ticketEndTime.time():" << ticketEndTime.time().toString(Qt::ISODate);
|
2024-08-21 17:46:35 +02:00
|
|
|
if (ticketEndTime.time() > carryOverStart) {
|
2024-09-02 17:11:48 +02:00
|
|
|
// qCritical() << __func__ << __LINE__ << "ticketEndTime.time():" << ticketEndTime.time().toString(Qt::ISODate);
|
2024-08-21 15:40:55 +02:00
|
|
|
ticketEndTime = ticketEndTime.addSecs(carryOverDuration * 60);
|
2024-08-21 17:46:35 +02:00
|
|
|
} else {
|
2024-09-02 17:11:48 +02:00
|
|
|
// qCritical() << __func__ << __LINE__ << "ticketEndTime.time():" << ticketEndTime.time().toString(Qt::ISODate);
|
2024-08-21 17:46:35 +02:00
|
|
|
if (ticketEndTime.time() < carryOverEnd) {
|
2024-09-02 17:11:48 +02:00
|
|
|
// qCritical() << __func__ << __LINE__ << "ticketEndTime.time():" << ticketEndTime.time().toString(Qt::ISODate);
|
2024-08-21 17:46:35 +02:00
|
|
|
ticketEndTime = ticketEndTime.addSecs(carryOverDuration * 60);
|
|
|
|
}
|
2024-08-21 15:40:55 +02:00
|
|
|
}
|
2024-08-21 17:46:35 +02:00
|
|
|
} else {
|
2024-09-02 17:11:48 +02:00
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "WARNING: wrong carry-over-settings";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ticketEndTime.setTime(QTime(ticketEndTime.time().hour(),
|
|
|
|
ticketEndTime.time().minute(), 0));
|
|
|
|
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "ticketEndTime:" << ticketEndTime.toString(Qt::ISODate);
|
|
|
|
|
|
|
|
for (auto[itr, rangeEnd] = tariff->WeekDays.equal_range((Qt::DayOfWeek)(ticketEndTime.date().dayOfWeek()));
|
|
|
|
itr != rangeEnd;
|
|
|
|
++itr) {
|
|
|
|
ATBWeekDay const &wd = itr->second;
|
2024-09-03 11:19:23 +02:00
|
|
|
bool const parkTimeLimitViolated = wd.getTariffCarryOverSettings().parkingTimeLimitExceeded(start_parking_time,
|
|
|
|
ticketEndTime,
|
|
|
|
paymentOptionIndex);
|
2024-09-02 17:11:48 +02:00
|
|
|
if (parkTimeLimitViolated) {
|
|
|
|
calcState.setDesc(QString("line=%1 endTime=%2: park-time-limit violated").arg(__LINE__)
|
|
|
|
.arg(ticketEndTime.time().toString(Qt::ISODate)));
|
|
|
|
return calcState.set(CalcState::State::ABOVE_MAX_PARKING_TIME);
|
2024-08-21 15:40:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-02 13:15:11 +02:00
|
|
|
if (ticketEndTime.time().hour() == 0 && ticketEndTime.time().minute() == 0) {
|
|
|
|
ticketEndTime = ticketEndTime.addDays(-1);
|
|
|
|
ticketEndTime.setTime(QTime(23, 59, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
// DEBUG
|
2024-09-02 17:13:11 +02:00
|
|
|
qCritical() << __func__ << ":" << __LINE__ << " endTime:" << endTime;
|
|
|
|
qCritical() << __func__ << ":" << __LINE__ << "ticketEndTime:" << ticketEndTime.toString(Qt::ISODate);
|
2023-05-10 16:16:24 +02:00
|
|
|
}
|
2023-05-09 11:40:14 +02:00
|
|
|
} else {
|
|
|
|
return calcState.set(CalcState::State::INVALID_START_DATE);
|
2023-05-08 12:34:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
2023-04-24 15:31:46 +02:00
|
|
|
}
|
2023-05-15 14:05:55 +02:00
|
|
|
|
2024-07-25 09:48:12 +02:00
|
|
|
CalcState CALCULATE_LIBRARY_API compute_duration_for_daily_ticket(parking_tariff_t *tariff,
|
|
|
|
QDateTime const &start_parking_time,
|
|
|
|
QDateTime &ticketEndTime,
|
2024-09-02 17:13:11 +02:00
|
|
|
PermitType /* PermitType */)
|
2023-05-15 14:05:55 +02:00
|
|
|
{
|
|
|
|
CalcState calcState;
|
2023-05-16 11:07:21 +02:00
|
|
|
if (start_parking_time.isValid()) {
|
|
|
|
|
2024-01-23 10:55:00 +01:00
|
|
|
ticketEndTime = Calculator::GetInstance().GetDailyTicketDuration(tariff,
|
2023-05-16 16:43:45 +02:00
|
|
|
start_parking_time,
|
2023-11-22 16:27:41 +01:00
|
|
|
tariff->getPaymentOptions().pop_payment_method_id,
|
2023-05-16 11:07:21 +02:00
|
|
|
false); // carry over
|
|
|
|
|
|
|
|
// DEBUG
|
|
|
|
qCritical() << "compute_duration_for_daily_ticket(): ";
|
|
|
|
qCritical() << " ticketEndTime: " << ticketEndTime;
|
|
|
|
|
|
|
|
if (!ticketEndTime.isValid()) {
|
2023-05-16 16:43:45 +02:00
|
|
|
calcState.setDesc(QString("ticketEndTime=%1").arg(ticketEndTime.toString(Qt::ISODate)));
|
2023-05-16 11:07:21 +02:00
|
|
|
return calcState.set(CalcState::State::WRONG_ISO_TIME_FORMAT);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
return calcState.set(CalcState::State::INVALID_START_DATE);
|
|
|
|
}
|
2023-05-15 14:05:55 +02:00
|
|
|
|
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
|
|
|
}
|
|
|
|
|
2024-01-30 10:41:29 +01:00
|
|
|
|
|
|
|
CalcState CALCULATE_LIBRARY_API compute_price_for_daily_ticket(
|
|
|
|
parking_tariff_t *tariff,
|
|
|
|
QDateTime const &startDatetime,
|
|
|
|
QDateTime &endDatetime,
|
|
|
|
PERMIT_TYPE permitType,
|
|
|
|
struct price_t *price) {// return value
|
|
|
|
CalcState calcState;
|
|
|
|
|
|
|
|
|
|
|
|
if (startDatetime.isValid()) {
|
|
|
|
if (std::optional<struct price_t> p =
|
|
|
|
Calculator::GetInstance().GetDailyTicketPrice(tariff,
|
|
|
|
startDatetime,
|
|
|
|
endDatetime,
|
|
|
|
permitType)) {
|
|
|
|
*price = p.value();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return calcState.set(CalcState::State::INVALID_START_DATE);
|
|
|
|
}
|
|
|
|
return calcState.set(CalcState::State::SUCCESS);
|
|
|
|
}
|
|
|
|
|