Compare commits

...

9 Commits

Author SHA1 Message Date
7a5d797ae0 Start reimplementation of
private_GetCostFromDuration() in terms of ticket.
2023-11-27 16:21:28 +01:00
215667af61 Implement findWorkTimeRange() and indNextWorkTimeRange(). 2023-11-27 16:19:39 +01:00
2b6d116200 Implement GetCostFromDuration() in terms of private_GetCostFromDuration()
using a ticket as return value.
2023-11-27 16:18:06 +01:00
2d9ed3c1c8 testing ... 2023-11-27 16:16:50 +01:00
9ca45e613f Return a ticket from private_GetCostFromDuration().
Add helpers findWorkTimeRange() and findNextWorkTimeRange().
2023-11-27 16:15:18 +01:00
6a3d183129 Minor: overload constructor 2023-11-27 16:14:27 +01:00
684de4acd1 Reformat to Unix.
Add overload for IsYearPeriodActive().
Add several helpers.
2023-11-27 16:12:07 +01:00
e9047f995a Minor: format to Unix. 2023-11-27 16:09:38 +01:00
20cdb8b07f Add overload for IsYearPeriodActive().
Add helpers WeekDaysWorkTimeFrom(), WeekDaysWorkTimeUntil(), isCarryOverSet()
sCarryOverNotSet() and getPaymentMethodId().
2023-11-27 16:07:30 +01:00
7 changed files with 727 additions and 537 deletions

View File

@ -1,7 +1,14 @@
#pragma once #ifndef CALCULATOR_FUNCTIONS_H_INCLUDED
#define CALCULATOR_FUNCTIONS_H_INCLUDED
#include <iostream> #include <iostream>
#include <optional>
#include "configuration.h" #include "configuration.h"
#include "payment_method.h" #include "payment_method.h"
#include "ticket.h"
#include "tariff_time_range.h"
#include <QDateTime> #include <QDateTime>
using namespace std; using namespace std;
@ -49,13 +56,13 @@ private:
int getMinimalParkingTime(Configuration const *cfg, PaymentMethod methodId); int getMinimalParkingTime(Configuration const *cfg, PaymentMethod methodId);
int getMaximalParkingTime(Configuration const *cfg, PaymentMethod methodId); int getMaximalParkingTime(Configuration const *cfg, PaymentMethod methodId);
uint32_t private_GetCostFromDuration(Configuration const* cfg, Ticket private_GetCostFromDuration(Configuration const* cfg,
QDateTime const &start, QDateTime const &start,
QDateTime &end, QDateTime &end,
int durationMinutes, int &durationMinutes,
bool nextDay = false, bool nextDay = false,
bool prepaid = false, bool prepaid = false,
bool overtime = false); bool overtime = false);
bool checkDurationMinutes(bool overTime, bool checkDurationMinutes(bool overTime,
int minParkingTime, int maxParkingTime, int minParkingTime, int maxParkingTime,
@ -64,4 +71,13 @@ private:
// //
uint32_t GetPriceForTimeStep(Configuration *cfg, int timeStep) const; uint32_t GetPriceForTimeStep(Configuration *cfg, int timeStep) const;
uint32_t GetDurationForPrice(Configuration *cfg, int price) const; uint32_t GetDurationForPrice(Configuration *cfg, int price) const;
int findWorkTimeRange(QDateTime const &dt,
QScopedArrayPointer<TariffTimeRange> const &worktime,
size_t size);
int findNextWorkTimeRange(QDateTime const &dt,
QScopedArrayPointer<TariffTimeRange> const &worktime,
size_t size);
}; };
#endif // CALCULATOR_FUNCTIONS_H_INCLUDED

View File

@ -21,7 +21,10 @@ public:
using Status = std::tuple<int, char const*, char const*>; using Status = std::tuple<int, char const*, char const*>;
explicit Ticket(); explicit Ticket();
explicit Ticket(QDateTime const &s, QDateTime const &e,
int durationMinutesNetto, int durationMinutesBrutto,
uint32_t price, Status status);
explicit operator bool() { return std::get<CODE>(m_status) == VALID; } explicit operator bool() { return std::get<CODE>(m_status) == VALID; }
operator QString(); operator QString();
@ -35,6 +38,15 @@ public:
void setValidUntil(QDateTime const &validUnil); void setValidUntil(QDateTime const &validUnil);
void setPrice(uint32_t price); void setPrice(uint32_t price);
bool isValid() { return operator bool(); }
static constexpr const Status s[STATUS_END] = {
{NOT_INITIALIZED , "NOT_INITIALIZED" , "Ticket not initialized" },
{VALID , "VALID" , "Ticket is valid" },
{INVALID_FROM_DATETIME , "INVALID_FROM_DATETIME" , "Ticket has invalid start datetime"},
{INVALID_UNTIL_DATETIME, "INVALID_UNTIL_DATETIME", "Ticket has invalid end datetime" }
};
private: private:
Status m_status; Status m_status;
@ -45,20 +57,8 @@ private:
int m_durationMinutesBrutto; int m_durationMinutesBrutto;
uint32_t m_price; uint32_t m_price;
static constexpr const Status s[STATUS_END] = {
{NOT_INITIALIZED , "NOT_INITIALIZED" , "Ticket not initialized" },
{VALID , "VALID" , "Ticket is valid" },
{INVALID_FROM_DATETIME , "INVALID_FROM_DATETIME" , "Ticket has invalid start datetime"},
{INVALID_UNTIL_DATETIME, "INVALID_UNTIL_DATETIME", "Ticket has invalid end datetime" }
};
}; };
QDebug operator<<(QDebug debug, Ticket::Status const &status) { QDebug operator<<(QDebug debug, Ticket::Status const &status);
QDebugStateSaver saver(debug);
debug << "Ticket-Status: " << std::get<1>(status)
<< "(" << std::get<2>(status) << ")";
return debug;
}
#endif // TICKET_H_INCLUDED #endif // TICKET_H_INCLUDED

View File

@ -1,78 +1,85 @@
#pragma once #pragma once
#include <cstring> #include <cstring>
#include <string.h> #include <string.h>
#include <ctime> #include <ctime>
#include <iostream> #include <iostream>
#include <cmath> #include <cmath>
#include "day_of_week.h" #include "day_of_week.h"
#include "configuration.h" #include "configuration.h"
#include "time_range.h" #include "time_range.h"
#include "payment_method.h"
#include <QDateTime>
#include <QDateTime>
using namespace std;
using namespace std;
namespace Utilities {
/// <summary> namespace Utilities {
/// Get day of week from current date (Zeller's Algorithm), starting day is Sunday /// <summary>
/// </summary> /// Get day of week from current date (Zeller's Algorithm), starting day is Sunday
/// <param name="date"></param> /// </summary>
/// <returns></returns> /// <param name="date"></param>
DayOfWeek GetDayOfWeek(struct tm* tm); /// <returns></returns>
DayOfWeek GetDayOfWeek(struct tm* tm);
/// <summary>
/// Date and time parse helper function /// <summary>
/// </summary> /// Date and time parse helper function
/// <returns>Returns time (tm) structure</returns> /// </summary>
struct tm DateTimeToStructTm(const char* dateTimeStr); /// <returns>Returns time (tm) structure</returns>
struct tm DateTimeToStructTm(const char* dateTimeStr);
/// <summary>
/// Date parse helper function /// <summary>
/// </summary> /// Date parse helper function
/// <returns>Returns time (tm) structure</returns> /// </summary>
struct tm DateToStructTm(const char* dateStr); /// <returns>Returns time (tm) structure</returns>
struct tm DateToStructTm(const char* dateStr);
/// <summary>
/// Time parse helper function /// <summary>
/// </summary> /// Time parse helper function
/// <returns>Returns time (tm) structure</returns> /// </summary>
struct tm TimeToStructTm(const char* timeStr, int year, int mon, int mday, int wday); /// <returns>Returns time (tm) structure</returns>
struct tm TimeToStructTm(const char* timeStr, int year, int mon, int mday, int wday);
/// <summary>
/// Get current local time /// <summary>
/// </summary> /// Get current local time
/// <returns>Returns time_t structure</returns> /// </summary>
time_t GetCurrentLocalTime(); /// <returns>Returns time_t structure</returns>
time_t GetCurrentLocalTime();
/// <summary>
/// Zeller's algorithm for determining day of week /// <summary>
/// </summary> /// Zeller's algorithm for determining day of week
int ZellersAlgorithm(int day, int month, int year); /// </summary>
int ZellersAlgorithm(int day, int month, int year);
/// <summary>
/// Checks if current datetime is in range between start and end month of parking worktime /// <summary>
/// </summary> /// Checks if current datetime is in range between start and end month of parking worktime
/// <param name="tariff_cfg"></param> /// </summary>
/// <param name="currentDateTime"></param> /// <param name="tariff_cfg"></param>
/// <returns></returns> /// <param name="currentDateTime"></param>
bool IsYearPeriodActive(Configuration* cfg, struct tm* currentDateTime); /// <returns></returns>
bool IsYearPeriodActive(Configuration* cfg, struct tm* currentDateTime);
/// <summary> bool IsYearPeriodActive(Configuration const *cfg, QDateTime const &currentDateTime);
/// Check permissions
/// </summary> /// <summary>
bool CheckSpecialDay(Configuration* cfg, const char* currentDateTimeStr, int* specialDayId, double* specialDayPrice); /// Check permissions
bool CheckSpecialDay(Configuration const *cfg, /// </summary>
QDateTime const &currentDateTimeS, bool CheckSpecialDay(Configuration* cfg, const char* currentDateTimeStr, int* specialDayId, double* specialDayPrice);
int* specialDayId, uint32_t *specialDayPrice); bool CheckSpecialDay(Configuration const *cfg,
QDateTime const &currentDateTimeS,
/// <summary> int* specialDayId, uint32_t *specialDayPrice);
/// Calculates price per unit
/// </summary> /// <summary>
/// <param name="pra_price"></param> /// Calculates price per unit
/// <returns></returns> /// </summary>
double CalculatePricePerUnit(double pra_price, double durationUnit = -1); /// <param name="pra_price"></param>
/// <returns></returns>
QTime SpecialDaysWorkTimeFrom(Configuration const *cfg, int specialDayId); double CalculatePricePerUnit(double pra_price, double durationUnit = -1);
QTime SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId);
} QTime SpecialDaysWorkTimeFrom(Configuration const *cfg, int specialDayId);
QTime SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId);
QTime WeekDaysWorkTimeFrom(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr);
QTime WeekDaysWorkTimeUntil(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr);
bool isCarryOverSet(Configuration const *cfg, PaymentMethod paymentMethodId);
bool isCarryOverNotSet(Configuration const *cfg, PaymentMethod paymentMethodId);
PaymentMethod getPaymentMethodId(Configuration const *cfg);
}

View File

@ -3,6 +3,7 @@
#include "utilities.h" #include "utilities.h"
#include "tariff_log.h" #include "tariff_log.h"
#include "tariff_time_range.h" #include "tariff_time_range.h"
#include "ticket.h"
#include <sstream> #include <sstream>
#include <algorithm> #include <algorithm>
@ -404,9 +405,18 @@ double Calculator::GetCostFromDuration(Configuration* cfg,
return GetCostFromDuration(cfg, start_datetime, end_datetime); return GetCostFromDuration(cfg, start_datetime, end_datetime);
} }
return private_GetCostFromDuration(cfg, start_datetime, QDateTime start = start_datetime;
end_datetime, durationMinutes,
nextDay, prepaid); Ticket t = private_GetCostFromDuration(cfg, start,
end_datetime, durationMinutes,
nextDay, prepaid);
if (t) {
qCritical().noquote() << t;
return t.getPrice();
}
return -1;
} }
int Calculator::getMinimalParkingTime(Configuration const *cfg, PaymentMethod methodId) { int Calculator::getMinimalParkingTime(Configuration const *cfg, PaymentMethod methodId) {
@ -434,66 +444,95 @@ bool Calculator::checkDurationMinutes(bool overtime,
return true; return true;
} }
int Calculator::findWorkTimeRange(QDateTime const &dt,
QScopedArrayPointer<TariffTimeRange> const &worktime,
size_t size) {
for (size_t w = 0; w < size; ++w) {
QTime const &worktime_from = worktime[w].getTimeFrom();
QTime const &worktime_to = worktime[w].getTimeUntil();
if ((dt.time() >= worktime_from) && (dt.time() < worktime_to)) {
return w;
}
}
return -1;
}
int Calculator::findNextWorkTimeRange(QDateTime const &dt,
QScopedArrayPointer<TariffTimeRange> const &worktime,
size_t size) {
int nextWorkTimeRange = -1;
for (size_t w = 0; w < size; ++w) {
QTime const &worktime_from = worktime[w].getTimeFrom();
if (dt.time() < worktime_from) {
nextWorkTimeRange = w;
continue;
}
}
return nextWorkTimeRange;
}
using namespace Utilities; using namespace Utilities;
uint32_t Calculator::private_GetCostFromDuration(Configuration const* cfg, Ticket Calculator::private_GetCostFromDuration(Configuration const* cfg,
QDateTime const &start, QDateTime const &start,
QDateTime &end, QDateTime &end,
int durationMinutes, int &durationMinutes,
bool nextDay, bool nextDay,
bool prepaid, bool prepaid,
bool overtime) { bool overtime) {
// TODO
static const PaymentMethod paymentMethodId = PaymentMethod::Linear;
static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg);
static const bool carryOverNotSet = isCarryOverNotSet(cfg, paymentMethodId);
static int const minParkingTimeMinutes = getMinimalParkingTime(cfg, paymentMethodId); static int const minParkingTimeMinutes = getMinimalParkingTime(cfg, paymentMethodId);
static int const maxParkingTimeMinutes = getMaximalParkingTime(cfg, paymentMethodId); static int const maxParkingTimeMinutes = getMaximalParkingTime(cfg, paymentMethodId);
static bool const checkMinMaxMinutes = (minParkingTimeMinutes < maxParkingTimeMinutes); static bool const checkMinMaxMinutes = (minParkingTimeMinutes < maxParkingTimeMinutes);
static const int durationMinutesNetto = durationMinutes;
if (!checkMinMaxMinutes) { if (!checkMinMaxMinutes) {
qCritical() << QString( qCritical() << QString(
"ERROR: CONDITION minMin < maxMin (%1 < %2) IS NOT VALID") "ERROR: CONDITION minMin < maxMin (%1 < %2) IS NOT VALID")
.arg(minParkingTimeMinutes).arg(maxParkingTimeMinutes); .arg(minParkingTimeMinutes).arg(maxParkingTimeMinutes);
return 0; return Ticket();
} }
if (!checkDurationMinutes(overtime, minParkingTimeMinutes, if (!checkDurationMinutes(overtime, minParkingTimeMinutes,
maxParkingTimeMinutes, durationMinutes)) { maxParkingTimeMinutes, durationMinutes)) {
return 0; return Ticket();
} }
// Get input date
QDateTime inputDate = start; QDateTime inputDate = start;
// Get day of week
int const weekdayId = inputDate.date().dayOfWeek(); int const weekdayId = inputDate.date().dayOfWeek();
uint32_t price = 0;
uint32_t day_price = 0; double durationUnit = 60.0;
uint32_t price_per_unit = 0;
int current_special_day_id = -1; int current_special_day_id = -1;
// there might be more than 1 worktime ranges per day
int const timeRanges = std::max((int)cfg->WeekDaysWorktime.count(weekdayId), 1); int const timeRanges = std::max((int)cfg->WeekDaysWorktime.count(weekdayId), 1);
QScopedArrayPointer<TariffTimeRange> worktime(new TariffTimeRange[timeRanges]); QScopedArrayPointer<TariffTimeRange> worktime(new TariffTimeRange[timeRanges]);
int index = 0; int ranges = 0;
if(Utilities::CheckSpecialDay(cfg, inputDate, &current_special_day_id, &day_price)) { if(Utilities::CheckSpecialDay(cfg, inputDate, &current_special_day_id, &price)) {
// Set special day price: // Set special day price:
// ACHTUNG: price_per_unit ist eigentlich immer preis pro minute !!! durationUnit = 60.0;
price_per_unit = CalculatePricePerUnit(day_price); worktime[ranges].setTimeRange(SpecialDaysWorkTimeFrom(cfg, current_special_day_id),
worktime[index].setTimeRange(SpecialDaysWorkTimeFrom(cfg, current_special_day_id), SpecialDaysWorkTimeUntil(cfg, current_special_day_id));
SpecialDaysWorkTimeUntil(cfg, current_special_day_id)); ranges = 1;
} else { } else {
// Set new price for the normal day // Set new price for the normal day: do not use a floating-point type
// for the price, rather compute with integers. Only at the very end of
// the computation the price is divided by durationUnit.
int pop_id = cfg->PaymentOption.find(paymentMethodId)->second.pop_id; int pop_id = cfg->PaymentOption.find(paymentMethodId)->second.pop_id;
day_price = cfg->PaymentRate.find(pop_id)->second.pra_price; price = cfg->PaymentRate.find(pop_id)->second.pra_price;
int durationId = cfg->PaymentRate.find(pop_id)->second.pra_payment_unit_id; int durationId = cfg->PaymentRate.find(pop_id)->second.pra_payment_unit_id;
double durationUnit = cfg->Duration.find(durationId)->second.pun_duration; durationUnit = cfg->Duration.find(durationId)->second.pun_duration;
price_per_unit = Utilities::CalculatePricePerUnit(day_price,durationUnit);
// If no working day found, skip it (recursively call method again) // If no working day found, skip it (recursively call method again)
if (cfg->WeekDaysWorktime.count(weekdayId) <= 0) { if (cfg->WeekDaysWorktime.count(weekdayId) <= 0) {
@ -503,114 +542,162 @@ uint32_t Calculator::private_GetCostFromDuration(Configuration const* cfg,
return private_GetCostFromDuration(cfg, inputDate, end, durationMinutes, true, prepaid, overtime); return private_GetCostFromDuration(cfg, inputDate, end, durationMinutes, true, prepaid, overtime);
} }
for (auto[itr, rangeEnd] = cfg->WeekDaysWorktime.equal_range(weekdayId); itr != rangeEnd; ++itr) { using WTIterator = std::multimap<int, ATBWeekDaysWorktime>::const_iterator;
qCritical() << itr->first << itr->second.pwd_time_from.c_str() << itr->second.pwd_time_to.c_str(); std::pair<WTIterator, WTIterator> p = cfg->WeekDaysWorktime.equal_range(weekdayId);
worktime[index].setTimeRange(QTime::fromString(itr->second.pwd_time_from.c_str()),
QTime::fromString(itr->second.pwd_time_to.c_str())); for (WTIterator itr = p.first; itr != p.second; ++itr) {
index += 1; worktime[ranges].setTimeRange(WeekDaysWorkTimeFrom(itr),
WeekDaysWorkTimeUntil(itr));
ranges += 1;
} }
} }
if (price_per_unit < 0) price_per_unit = 1.0f; uint32_t costFromDuration = 0;
qDebug() << "Calculated price per minute=" << price_per_unit; QTime const &lastWorktimeTo = worktime[ranges-1].getTimeUntil();
int currentRange = -1;
double costFromDuration = 0.0; if (nextDay) { // this means the function has been called recursively
for (int w = 0; w < index; ++w) { currentRange = 0;
QTime worktime_from = worktime[w].getTimeFrom(); inputDate.setTime(worktime[currentRange].getTimeFrom());
QTime worktime_to = worktime[w].getTimeUntil(); } else {
// check if inputDate is located inside a valid worktime-range...
if (price_per_unit == 0) { currentRange = findWorkTimeRange(inputDate, worktime, ranges);
inputDate = inputDate.addDays(1); if (currentRange == -1) { // no...
inputDate.setTime(worktime_from); if (!prepaid) { // parking is not allowed
uint32_t const partialCost = private_GetCostFromDuration(cfg, inputDate, end, durationMinutes, true, prepaid); return Ticket(start, end, durationMinutesNetto, 0,
if (partialCost == 0) { 0, Ticket::s[INVALID_FROM_DATETIME]);
return 0;
} }
costFromDuration += partialCost; // find the next worktime-range (on the same day), and start from there
continue; currentRange = findNextWorkTimeRange(inputDate, worktime, ranges);
inputDate.setTime(worktime[currentRange].getTimeFrom());
} }
}
// If overtime flag is set for (int w = currentRange; w < ranges; ++w) {
if (overtime || nextDay) { if (durationMinutes > 0) {
inputDate.setTime(worktime_from); QTime const &worktime_from = worktime[w].getTimeFrom();
overtime = false; QTime const &worktime_to = worktime[w].getTimeUntil();
}
// Check prepaid qCritical() << "from" << worktime_from;
if (!prepaid) { qCritical() << "until" << worktime_to;
if ((inputDate.time() < worktime_from) || (inputDate.time() > worktime_to)) {
qDebug() << "[STOP] * Ticket is not valid * "; //if (w > 0) { // durationMinutes are always meant as netto time and
return 0.0f; // // the time between worktime-ranges are free.
} // inputDate.setTime(worktime_from);
} else { //}
qDebug() << "* PREPAID MODE ACTIVE *";
if (inputDate.time() < worktime_from) { // TODO: hier muss dann der preis hin
inputDate.setTime(worktime_from); if (price == 0) {
} else if(inputDate.time() > worktime_to) {
qDebug() << " *** PREPAID *** Current time is past the time range end, searching for next available day";
inputDate = inputDate.addDays(1); inputDate = inputDate.addDays(1);
uint32_t const partialCost = private_GetCostFromDuration(cfg, inputDate, end, durationMinutes, true, prepaid); inputDate.setTime(worktime_from);
if (partialCost == 0) { Ticket t = private_GetCostFromDuration(
return 0; cfg, // TODO: erklaerung
inputDate,
end, durationMinutes,
true, prepaid);
if (!t.isValid()) {
return t;
} }
costFromDuration += partialCost; costFromDuration += t.getPrice();
continue; continue;
} }
}
while(durationMinutes > 0) { // If overtime flag is set
// Check for active year period // TODO: ueberpruefen, kann man wohl nach oben ziehen
if (std::none_of(cfg->YearPeriod.begin(), //if (overtime || nextDay) {
cfg->YearPeriod.end(), // inputDate.setTime(worktime_from);
[&inputDate](std::pair<int, ATBPeriodYear> const &year) { // overtime = false;
QDate const input(2004, // 2004 is a leap year //}
inputDate.date().month(),
inputDate.date().day()); qCritical() << "inputDate.time()=" << inputDate.time();
QDate const s(2004, year.second.pye_start_day, year.second.pye_start_month);
QDate const e(2004, year.second.pye_end_day, year.second.pye_end_month); // Check prepaid
return (input >= s && input <= e); if (!prepaid) {
})) { if ((inputDate.time() < worktime_from) || (inputDate.time() > worktime_to)) {
qCritical() << "NO VALID YEAR PERIOD"; qDebug() << "[STOP] * Ticket is not valid * ";
return 0.0; return Ticket();
}
} else {
qDebug() << "* PREPAID MODE ACTIVE *";
if (inputDate.time() < worktime_from) {
inputDate.setTime(worktime_from);
//} else if(inputDate.time() > worktime_to) {
} else if(inputDate.time() > lastWorktimeTo) {
qDebug() << " *** PREPAID *** Current time is past the time range end, searching for next available day";
// wieso ist hier overtime nicht gesetzt
inputDate = inputDate.addDays(1);
Ticket t = private_GetCostFromDuration(
cfg, inputDate, end,
durationMinutes, true, prepaid);
if (!t.isValid()) {
return t;
}
costFromDuration += t.getPrice();
continue;
}
} }
// Go to next day if minutes not spent while(durationMinutes > 0) {
if(inputDate.time() >= worktime_to) { // Check for active year period
// check for carry_over status if (!IsYearPeriodActive(cfg, inputDate)) {
if (cfg->PaymentOption.find(paymentMethodId)->second.pop_carry_over < 1) { return Ticket();
break;
} }
qDebug() << "Reached end of worktime, searching for the next working day"; qDebug() << "inputDate" << inputDate.toString(Qt::ISODate);
inputDate = inputDate.addDays(1); if(inputDate.time() >= lastWorktimeTo) {
overtime = true; // Go to next day if minutes not spent
uint32_t const partialCost = private_GetCostFromDuration(cfg, inputDate, end, durationMinutes, true, prepaid, overtime); if (carryOverNotSet) {
if (partialCost == 0) { // no carry_over, so stop computation
return 0; break;
}
qDebug() << "Reached end of worktime, searching for the next working day";
inputDate = inputDate.addDays(1);
inputDate.setTime(QTime());
overtime = true;
Ticket t = private_GetCostFromDuration(
cfg, inputDate, end,
durationMinutes, true, prepaid, overtime);
if (!t.isValid()) {
return t;
}
costFromDuration += t.getPrice();
break; // stop while, and continue in outer loop
} else {
if(inputDate.time() < worktime_to) {
// Increment input date minutes for each monetary unit
inputDate = inputDate.addSecs(60);
qDebug() << "inputDate" << inputDate.toString(Qt::ISODate);
durationMinutes -= 1;
//costFromDuration += price_per_unit;
costFromDuration += price;
} else break;
} }
costFromDuration += partialCost;
break; // stop while, and continue in outer loop
} else {
// Increment input date minutes for each monetary unit
inputDate = inputDate.addSecs(60);
durationMinutes -= 1;
costFromDuration += price_per_unit;
} }
} }
} }
qDebug() << "GetCostFromDuration(): Valid until:" << inputDate.toString(Qt::ISODate);
end = inputDate; if (inputDate >= end) {
end = inputDate;
}
//double ret_val = total_cost; if (nextDay == false) {
//total_cost = 0.0f; qDebug() << "GetCostFromDuration(): Valid until:" << end.toString(Qt::ISODate);
//return ceil(ret_val); }
// TODO: runden nur falls oberster stack-rahmen return
Ticket(start, end,
return ceil(costFromDuration); durationMinutesNetto,
start.secsTo(end) / 60,
nextDay ?
costFromDuration :
llround(Utilities::CalculatePricePerUnit(costFromDuration, durationUnit)),
Ticket::s[VALID]);
} }

View File

@ -11,7 +11,19 @@ Ticket::Ticket()
qDebug() << *this; qDebug() << *this;
qDebug() << m_status; qDebug() << m_status;
} }
Ticket::Ticket(QDateTime const &s, QDateTime const &e,
int durationMinutesNetto, int durationMinutesBrutto,
uint32_t price, Ticket::Status status)
: m_status(status)
, m_validFrom(s)
, m_validUntil(e)
, m_durationMinutesNetto(durationMinutesNetto)
, m_durationMinutesBrutto(durationMinutesBrutto)
, m_price(price) {
}
Ticket::Status Ticket::setStatus(Status status) { Ticket::Status Ticket::setStatus(Status status) {
Status old = m_status; Status old = m_status;
m_status = status; m_status = status;
@ -54,14 +66,23 @@ void Ticket::setPrice(uint32_t price) {
Ticket::operator QString() { Ticket::operator QString() {
QStringList status; QStringList status;
status << QString("Status .............. : %1 (%2)") status << QString("**********************");
status << QString("Status ............. : %1 (%2)")
.arg(std::get<0>(m_status)) .arg(std::get<0>(m_status))
.arg(std::get<2>(m_status)); .arg(std::get<2>(m_status));
status << QString("Valid from ......... : %1").arg(m_validFrom.toString(Qt::ISODate)); status << QString("Valid from ......... : %1").arg(m_validFrom.toString(Qt::ISODate));
status << QString("Valid until ........ : %1").arg(m_validUntil.toString(Qt::ISODate)); status << QString("Valid until ........ : %1").arg(m_validUntil.toString(Qt::ISODate));
status << QString("Duration (netto) ... : %1").arg(m_durationMinutesNetto); status << QString("Duration (netto) ... : %1").arg(m_durationMinutesNetto);
status << QString("Duration (brutto)... : %1").arg(m_durationMinutesBrutto); status << QString("Duration (brutto)... : %1").arg(m_durationMinutesBrutto);
status << QString("Price ......... : %1").arg(m_price); status << QString("Price .............. : %1").arg(m_price);
status << QString("**********************");
return status.join('\n');; return status.join('\n');
}
QDebug operator<<(QDebug debug, Ticket::Status const &status) {
QDebugStateSaver saver(debug);
debug << "Ticket-Status: " << std::get<1>(status)
<< "(" << std::get<2>(status) << ")";
return debug;
} }

View File

@ -1,319 +1,375 @@
#include "utilities.h" #include "utilities.h"
#include "tariff_log.h" #include "tariff_log.h"
#include <QDebug> #include <QDebug>
#include <algorithm>
static int protection_counter = 0;
static int protection_counter = 0;
/// <summary>
/// Helper function /// <summary>
/// </summary> /// Helper function
/// <param name="pra_price"></param> /// </summary>
/// <returns></returns> /// <param name="pra_price"></param>
double Utilities::CalculatePricePerUnit(double pra_price, double durationUnit) /// <returns></returns>
{ double Utilities::CalculatePricePerUnit(double pra_price, double durationUnit)
try {
{ try
double price_per_unit = pra_price; {
double unit = durationUnit; double price_per_unit = pra_price;
double unit = durationUnit;
if(unit < 0 || unit > 65535 ) unit = 60.0f;
price_per_unit /= unit; // Divided by 60 because price per unit is set per hour and we are using minutes if(unit < 0 || unit > 65535 ) unit = 60.0f;
//printf("Price per unit (min) is: %lf\n", price_per_unit); price_per_unit /= unit; // Divided by 60 because price per unit is set per hour and we are using minutes
return price_per_unit; //printf("Price per unit (min) is: %lf\n", price_per_unit);
} return price_per_unit;
catch (...) }
{ catch (...)
throw std::invalid_argument("An error has occurred in CalculatePricePerUnit() function\n"); {
} throw std::invalid_argument("An error has occurred in CalculatePricePerUnit() function\n");
} }
}
/// <inheritdoc/>
time_t Utilities::GetCurrentLocalTime() /// <inheritdoc/>
{ time_t Utilities::GetCurrentLocalTime()
try {
{ try
time_t curr_time = time(NULL); {
tm tm_curr_time = {}; time_t curr_time = time(NULL);
memset(&tm_curr_time, '\0', sizeof(struct tm)); tm tm_curr_time = {};
memset(&tm_curr_time, '\0', sizeof(struct tm));
tm_curr_time = *localtime(&curr_time);
curr_time = mktime(&tm_curr_time); //- timezone; tm_curr_time = *localtime(&curr_time);
return curr_time; curr_time = mktime(&tm_curr_time); //- timezone;
} return curr_time;
catch (...) }
{ catch (...)
throw std::invalid_argument("An error has occurred in GetCurrentLocalTime() function\n"); {
} throw std::invalid_argument("An error has occurred in GetCurrentLocalTime() function\n");
} }
}
/// <inheritdoc/>
int Utilities::ZellersAlgorithm(int day, int month, int year) /// <inheritdoc/>
{ int Utilities::ZellersAlgorithm(int day, int month, int year)
int mon; {
if (month > 2) mon = month; //for march to december month code is same as month int mon;
else { if (month > 2) mon = month; //for march to december month code is same as month
mon = (12 + month); //for Jan and Feb, month code will be 13 and 14 else {
year--; //decrease year for month Jan and Feb mon = (12 + month); //for Jan and Feb, month code will be 13 and 14
} year--; //decrease year for month Jan and Feb
int y = year % 100; //last two digit }
int c = year / 100; //first two digit int y = year % 100; //last two digit
int w = (day + floor((13 * (mon + 1)) / 5) + y + floor(y / 4) + floor(c / 4) + (5 * c)); int c = year / 100; //first two digit
w = ((w + 5) % 7) + 1; //w % 7; int w = (day + floor((13 * (mon + 1)) / 5) + y + floor(y / 4) + floor(c / 4) + (5 * c));
return w; w = ((w + 5) % 7) + 1; //w % 7;
} return w;
}
/// <inheritdoc/>
struct tm Utilities::DateToStructTm(const char* dateStr) /// <inheritdoc/>
{ struct tm Utilities::DateToStructTm(const char* dateStr)
struct tm t = {}; {
memset(&t, '\0', sizeof(struct tm)); struct tm t = {};
memset(&t, '\0', sizeof(struct tm));
if (dateStr == nullptr || strlen(dateStr) <= 0) throw std::invalid_argument("DateToStructTm has failed parsing date string (null or empty)\n");
try if (dateStr == nullptr || strlen(dateStr) <= 0) throw std::invalid_argument("DateToStructTm has failed parsing date string (null or empty)\n");
{ try
int success = sscanf(dateStr, "%d-%d-%d", &t.tm_year, &t.tm_mon, &t.tm_mday); {
if (success != 3) throw std::invalid_argument("DateToStructTm() has failed parsing datetime string\n"); int success = sscanf(dateStr, "%d-%d-%d", &t.tm_year, &t.tm_mon, &t.tm_mday);
if (success != 3) throw std::invalid_argument("DateToStructTm() has failed parsing datetime string\n");
t.tm_year = t.tm_year - 1900;
t.tm_mon = t.tm_mon - 1; t.tm_year = t.tm_year - 1900;
t.tm_isdst = 0; t.tm_mon = t.tm_mon - 1;
return t; t.tm_isdst = 0;
} return t;
catch (...) }
{ catch (...)
throw std::invalid_argument("An error has occurred in DateToStructTm() function\n"); {
} throw std::invalid_argument("An error has occurred in DateToStructTm() function\n");
} }
}
/// <inheritdoc/>
struct tm Utilities::TimeToStructTm(const char* timeStr, int year, int mon, int mday, int wday) /// <inheritdoc/>
{ struct tm Utilities::TimeToStructTm(const char* timeStr, int year, int mon, int mday, int wday)
struct tm t = {}; {
memset(&t, '\0', sizeof(struct tm)); struct tm t = {};
memset(&t, '\0', sizeof(struct tm));
if (timeStr == nullptr || strlen(timeStr) <= 0) throw std::invalid_argument("TimeToStructTm() has failed parsing time string (null or empty)\n");
try if (timeStr == nullptr || strlen(timeStr) <= 0) throw std::invalid_argument("TimeToStructTm() has failed parsing time string (null or empty)\n");
{ try
int success_time = sscanf(timeStr, "%d:%d:%d", &t.tm_hour, &t.tm_min, &t.tm_sec); {
if (success_time != 3) throw std::invalid_argument("TimeToStructTm() has failed parsing time string\n"); int success_time = sscanf(timeStr, "%d:%d:%d", &t.tm_hour, &t.tm_min, &t.tm_sec);
if (success_time != 3) throw std::invalid_argument("TimeToStructTm() has failed parsing time string\n");
struct tm tm_struct;
t.tm_year = year; struct tm tm_struct;
t.tm_mon = mon; t.tm_year = year;
t.tm_mday = mday; t.tm_mon = mon;
t.tm_wday = wday; t.tm_mday = mday;
t.tm_isdst = 0; t.tm_wday = wday;
return t; t.tm_isdst = 0;
} return t;
catch (...) }
{ catch (...)
throw std::invalid_argument("An error has occurred in TimeToStructTm() function\n"); {
} throw std::invalid_argument("An error has occurred in TimeToStructTm() function\n");
} }
}
/// <inheritdoc/>
struct tm Utilities::DateTimeToStructTm(const char* dateTimeStr) /// <inheritdoc/>
{ struct tm Utilities::DateTimeToStructTm(const char* dateTimeStr)
struct tm t = {}; {
memset(&t, '\0', sizeof(struct tm)); struct tm t = {};
memset(&t, '\0', sizeof(struct tm));
if (dateTimeStr == nullptr || strlen(dateTimeStr) <= 0) throw std::invalid_argument("DateTimeToStructTm() has failed parsing date string (null or empty)");
try if (dateTimeStr == nullptr || strlen(dateTimeStr) <= 0) throw std::invalid_argument("DateTimeToStructTm() has failed parsing date string (null or empty)");
{ try
int success = sscanf(dateTimeStr, "%d-%d-%dT%d:%d:%dZ", &t.tm_year, &t.tm_mon, &t.tm_mday, &t.tm_hour, &t.tm_min, &t.tm_sec); {
if (success != 6) throw std::invalid_argument("DateTimeToStructTm() has failed parsing datetime string\n"); int success = sscanf(dateTimeStr, "%d-%d-%dT%d:%d:%dZ", &t.tm_year, &t.tm_mon, &t.tm_mday, &t.tm_hour, &t.tm_min, &t.tm_sec);
if (success != 6) throw std::invalid_argument("DateTimeToStructTm() has failed parsing datetime string\n");
t.tm_year = t.tm_year - 1900;
t.tm_mon = t.tm_mon - 1; t.tm_year = t.tm_year - 1900;
t.tm_isdst = 0; t.tm_mon = t.tm_mon - 1;
return t; t.tm_isdst = 0;
} return t;
catch (...) }
{ catch (...)
throw std::invalid_argument("An error has occurred in DateTimeToStructTm() function\n"); {
} throw std::invalid_argument("An error has occurred in DateTimeToStructTm() function\n");
} }
}
/// <inheritdoc/>
DayOfWeek Utilities::GetDayOfWeek(struct tm* t) /// <inheritdoc/>
{ DayOfWeek Utilities::GetDayOfWeek(struct tm* t)
if (t == nullptr) throw std::invalid_argument("GetDayOfWeekFromDate() => parameter 't' is null\n"); {
try if (t == nullptr) throw std::invalid_argument("GetDayOfWeekFromDate() => parameter 't' is null\n");
{ try
int d = t->tm_mday; {
int m = t->tm_mon + 1; int d = t->tm_mday;
int y = t->tm_year + 1900; int m = t->tm_mon + 1;
int y = t->tm_year + 1900;
int wd = Utilities::ZellersAlgorithm(d, m, y);
return static_cast<DayOfWeek>(wd); int wd = Utilities::ZellersAlgorithm(d, m, y);
} return static_cast<DayOfWeek>(wd);
catch (...) }
{ catch (...)
throw std::invalid_argument("An error has occurred in GetDayOfWeekFromDate() function\n"); {
} throw std::invalid_argument("An error has occurred in GetDayOfWeekFromDate() function\n");
} }
}
/// <inheritdoc/>
bool Utilities::IsYearPeriodActive(Configuration* cfg, struct tm* currentDateTime_tm) /// <inheritdoc/>
{ bool Utilities::IsYearPeriodActive(Configuration* cfg, struct tm* currentDateTime_tm)
if (cfg == nullptr) throw std::invalid_argument("IsYearPeriodActive() = > Configuration not set\n"); {
if (currentDateTime_tm == nullptr) throw std::invalid_argument("IsYearPeriodActive() = > Current datetime not set\n"); if (cfg == nullptr) throw std::invalid_argument("IsYearPeriodActive() = > Configuration not set\n");
if (currentDateTime_tm == nullptr) throw std::invalid_argument("IsYearPeriodActive() = > Current datetime not set\n");
try
{ try
//// Parse input date {
int dayCurrent = currentDateTime_tm->tm_mday; //// Parse input date
int monthCurrent = currentDateTime_tm->tm_mon + 1; int dayCurrent = currentDateTime_tm->tm_mday;
int monthCurrent = currentDateTime_tm->tm_mon + 1;
// Current date time
int cdt = (monthCurrent * 100) + dayCurrent; // Current date time
int cdt = (monthCurrent * 100) + dayCurrent;
multimap<int, ATBPeriodYear>::iterator year_period_itr;
for (year_period_itr = cfg->YearPeriod.begin(); year_period_itr != cfg->YearPeriod.end(); ++year_period_itr) multimap<int, ATBPeriodYear>::iterator year_period_itr;
{ for (year_period_itr = cfg->YearPeriod.begin(); year_period_itr != cfg->YearPeriod.end(); ++year_period_itr)
int dStart = year_period_itr->second.pye_start_day; {
int dEnd = year_period_itr->second.pye_end_day; int dStart = year_period_itr->second.pye_start_day;
int dEnd = year_period_itr->second.pye_end_day;
int mStart = year_period_itr->second.pye_start_month;
int mEnd = year_period_itr->second.pye_end_month; int mStart = year_period_itr->second.pye_start_month;
int mEnd = year_period_itr->second.pye_end_month;
int start = (mStart * 100) + dStart;
int end = (mEnd * 100) + dEnd; int start = (mStart * 100) + dStart;
int end = (mEnd * 100) + dEnd;
if (cdt >= start && cdt <= end)
{ if (cdt >= start && cdt <= end)
return true; {
} return true;
} }
return false; }
} return false;
catch (...) }
{ catch (...)
cout << "IsYearPeriodActive() => An exception has occurred, ignoring check, returning true" << endl; {
return true; cout << "IsYearPeriodActive() => An exception has occurred, ignoring check, returning true" << endl;
} return true;
} }
}
/// <inheritdoc/>
bool Utilities::CheckSpecialDay(Configuration* cfg, const char* currentDateTimeStr, int* specialDayId, double* specialDayPrice) bool Utilities::IsYearPeriodActive(Configuration const *cfg, QDateTime const &dt) {
{ if (std::none_of(cfg->YearPeriod.cbegin(),
try cfg->YearPeriod.cend(),
{ [&dt](std::pair<int, ATBPeriodYear> const &year) {
*specialDayId = -1; QDate const d(2004, // 2004 is a leap year
*specialDayPrice = 0.0f; dt.date().month(),
dt.date().day());
if (cfg == nullptr) throw std::invalid_argument("CheckSpecialDay() => configuration is not set\n"); QDate const s(2004, year.second.pye_start_month, year.second.pye_start_day);
if (currentDateTimeStr == nullptr) throw std::invalid_argument("CheckSpecialDay() => invalid date/time string set\n"); QDate const e(2004, year.second.pye_end_month, year.second.pye_end_day);
return (d >= s && d <= e);
})) {
struct tm current_tm = Utilities::DateTimeToStructTm(currentDateTimeStr); qCritical() << "NO VALID YEAR PERIOD";
//cout << "CheckSpecialDay() => Current: " << asctime(&current_tm) << endl; return false;
}
multimap<int, ATBSpecialDays>::iterator spec_days_itr; return true;
}
for (spec_days_itr = cfg->SpecialDays.begin(); spec_days_itr != cfg->SpecialDays.end(); spec_days_itr++)
{ /// <inheritdoc/>
int repeat_every_year = 0; bool Utilities::CheckSpecialDay(Configuration* cfg, const char* currentDateTimeStr, int* specialDayId, double* specialDayPrice)
repeat_every_year = spec_days_itr->second.ped_year; {
try
string start = spec_days_itr->second.ped_date_start; {
if (start.length() <= 0) continue; *specialDayId = -1;
//cout << "CheckSpecialDay() => Start: " << start << endl; *specialDayPrice = 0.0f;
string end = spec_days_itr->second.ped_date_end; if (cfg == nullptr) throw std::invalid_argument("CheckSpecialDay() => configuration is not set\n");
if (end.length() <= 0) continue; if (currentDateTimeStr == nullptr) throw std::invalid_argument("CheckSpecialDay() => invalid date/time string set\n");
//cout << "CheckSpecialDay() => End: " << end << endl;
struct tm start_tm = Utilities::DateToStructTm(start.c_str()); struct tm current_tm = Utilities::DateTimeToStructTm(currentDateTimeStr);
//cout << "CheckSpecialDay() => Start: " << asctime(&start_tm) << endl; //cout << "CheckSpecialDay() => Current: " << asctime(&current_tm) << endl;
struct tm end_tm = Utilities::DateToStructTm(end.c_str()); multimap<int, ATBSpecialDays>::iterator spec_days_itr;
//cout << "CheckSpecialDay() => End: " << asctime(&end_tm) << endl;
for (spec_days_itr = cfg->SpecialDays.begin(); spec_days_itr != cfg->SpecialDays.end(); spec_days_itr++)
if (repeat_every_year <= 0) {
{ int repeat_every_year = 0;
//cout << "CheckSpecialDay() => Repeat every year is: 0" << endl; repeat_every_year = spec_days_itr->second.ped_year;
if ((current_tm.tm_year == start_tm.tm_year) && (current_tm.tm_year == end_tm.tm_year))
{ string start = spec_days_itr->second.ped_date_start;
if ((current_tm.tm_mon >= start_tm.tm_mon) && (current_tm.tm_mon <= end_tm.tm_mon)) if (start.length() <= 0) continue;
{ //cout << "CheckSpecialDay() => Start: " << start << endl;
//cout << "CheckSpecialDay() => Month is in range between start and end" << endl;
if ((current_tm.tm_mday >= start_tm.tm_mday) && (current_tm.tm_mday <= end_tm.tm_mday)) string end = spec_days_itr->second.ped_date_end;
{ if (end.length() <= 0) continue;
LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY"); //cout << "CheckSpecialDay() => End: " << end << endl;
*specialDayId = spec_days_itr->second.ped_id;
*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; struct tm start_tm = Utilities::DateToStructTm(start.c_str());
return true; //cout << "CheckSpecialDay() => Start: " << asctime(&start_tm) << endl;
}
} struct tm end_tm = Utilities::DateToStructTm(end.c_str());
} //cout << "CheckSpecialDay() => End: " << asctime(&end_tm) << endl;
}
else if (repeat_every_year <= 0)
{ {
if ((current_tm.tm_mon >= start_tm.tm_mon) && (current_tm.tm_mon <= end_tm.tm_mon)) //cout << "CheckSpecialDay() => Repeat every year is: 0" << endl;
{ if ((current_tm.tm_year == start_tm.tm_year) && (current_tm.tm_year == end_tm.tm_year))
//cout << "CheckSpecialDay() => Month is in range between start and end" << endl; {
if ((current_tm.tm_mday >= start_tm.tm_mday) && (current_tm.tm_mday <= end_tm.tm_mday)) if ((current_tm.tm_mon >= start_tm.tm_mon) && (current_tm.tm_mon <= end_tm.tm_mon))
{ {
LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY"); //cout << "CheckSpecialDay() => Month is in range between start and end" << endl;
*specialDayId = spec_days_itr->second.ped_id; if ((current_tm.tm_mday >= start_tm.tm_mday) && (current_tm.tm_mday <= end_tm.tm_mday))
*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; {
return true; LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY");
} *specialDayId = spec_days_itr->second.ped_id;
} *specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price;
} return true;
} }
//cout << "CheckSpecialDay() => NOT SPECIAL DAY" << endl; }
return false; }
} }
catch (...) else
{ {
throw std::invalid_argument("CheckSpecialDay() => An error has occurred\n"); if ((current_tm.tm_mon >= start_tm.tm_mon) && (current_tm.tm_mon <= end_tm.tm_mon))
return false; {
} //cout << "CheckSpecialDay() => Month is in range between start and end" << endl;
} if ((current_tm.tm_mday >= start_tm.tm_mday) && (current_tm.tm_mday <= end_tm.tm_mday))
{
bool Utilities::CheckSpecialDay(Configuration const *cfg, LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY");
QDateTime const &currentDateTime, *specialDayId = spec_days_itr->second.ped_id;
int* specialDayId, *specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price;
uint32_t *specialDayPrice) { return true;
*specialDayId = -1; }
*specialDayPrice = 0; }
}
std::multimap<int, ATBSpecialDays>::const_iterator spec_days_itr; }
//cout << "CheckSpecialDay() => NOT SPECIAL DAY" << endl;
for (spec_days_itr = cfg->SpecialDays.cbegin(); spec_days_itr != cfg->SpecialDays.cend(); ++spec_days_itr) { return false;
int repeat_every_year = spec_days_itr->second.ped_year; }
QDate start = QDate::fromString(spec_days_itr->second.ped_date_start.c_str(), Qt::ISODate); catch (...)
QDate end = QDate::fromString(spec_days_itr->second.ped_date_end.c_str(), Qt::ISODate); {
if (start.isValid() && end.isValid()) { throw std::invalid_argument("CheckSpecialDay() => An error has occurred\n");
if ((currentDateTime.date().month() >= start.month()) && return false;
(currentDateTime.date().month() <= end.month())) { }
if ((currentDateTime.date().day() >= start.day()) && }
(currentDateTime.date().day() <= end.day())) {
if (repeat_every_year <= 0) { bool Utilities::CheckSpecialDay(Configuration const *cfg,
if ((currentDateTime.date().year() != start.year()) || QDateTime const &currentDateTime,
(currentDateTime.date().year() != end.year())) { int* specialDayId,
continue; uint32_t *specialDayPrice) {
} *specialDayId = -1;
} *specialDayPrice = 0;
qDebug() << "CheckSpecialDay() => SPECIAL DAY";
*specialDayId = spec_days_itr->second.ped_id; std::multimap<int, ATBSpecialDays>::const_iterator spec_days_itr;
*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price;
return true; for (spec_days_itr = cfg->SpecialDays.cbegin(); spec_days_itr != cfg->SpecialDays.cend(); ++spec_days_itr) {
} int repeat_every_year = spec_days_itr->second.ped_year;
} QDate start = QDate::fromString(spec_days_itr->second.ped_date_start.c_str(), Qt::ISODate);
} QDate end = QDate::fromString(spec_days_itr->second.ped_date_end.c_str(), Qt::ISODate);
} if (start.isValid() && end.isValid()) {
if ((currentDateTime.date().month() >= start.month()) &&
return false; (currentDateTime.date().month() <= end.month())) {
} if ((currentDateTime.date().day() >= start.day()) &&
(currentDateTime.date().day() <= end.day())) {
QTime Utilities::SpecialDaysWorkTimeFrom(Configuration const *cfg, int specialDayId) { if (repeat_every_year <= 0) {
return QTime::fromString(cfg->SpecialDaysWorktime.find(specialDayId)->second.pedwt_time_from.c_str(), Qt::ISODate); if ((currentDateTime.date().year() != start.year()) ||
} (currentDateTime.date().year() != end.year())) {
continue;
QTime Utilities::SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId) { }
return QTime::fromString(cfg->SpecialDaysWorktime.find(specialDayId)->second.pedwt_time_to.c_str(), Qt::ISODate); }
} qDebug() << "CheckSpecialDay() => SPECIAL DAY";
*specialDayId = spec_days_itr->second.ped_id;
*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price;
return true;
}
}
}
}
return false;
}
QTime Utilities::SpecialDaysWorkTimeFrom(Configuration const *cfg, int specialDayId) {
return QTime::fromString(cfg->SpecialDaysWorktime.find(specialDayId)->second.pedwt_time_from.c_str(), Qt::ISODate);
}
QTime Utilities::SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId) {
return QTime::fromString(cfg->SpecialDaysWorktime.find(specialDayId)->second.pedwt_time_to.c_str(), Qt::ISODate);
}
QTime Utilities::WeekDaysWorkTimeFrom(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr) {
return QTime::fromString(itr->second.pwd_time_from.c_str(), Qt::ISODate);
}
QTime Utilities::WeekDaysWorkTimeUntil(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr) {
return QTime::fromString(itr->second.pwd_time_to.c_str(), Qt::ISODate);
}
bool Utilities::isCarryOverSet(Configuration const *cfg, PaymentMethod paymentMethodId) {
return !isCarryOverNotSet(cfg, paymentMethodId);
}
bool Utilities::isCarryOverNotSet(Configuration const *cfg, PaymentMethod paymentMethodId) {
return (cfg->PaymentOption.find(paymentMethodId)->second.pop_carry_over < 1);
}
PaymentMethod Utilities::getPaymentMethodId(Configuration const *cfg) {
if (cfg->PaymentOption.size() != 1) {
return PaymentMethod::Undefined;
}
std::multimap<int, ATBPaymentOption>::const_iterator it =
cfg->PaymentOption.cbegin();
switch (it->first) {
case PaymentMethod::Linear:
return PaymentMethod::Linear;
case PaymentMethod::Steps:
return PaymentMethod::Steps;
case PaymentMethod::Degressive:
return PaymentMethod::Degressive;
case PaymentMethod::Progressive:
return PaymentMethod::Progressive;
}
return PaymentMethod::Undefined;
}

View File

@ -50,9 +50,12 @@ int main() {
if (isParsed) if (isParsed)
{ {
QDateTime start = QDateTime::fromString("2023-05-11T08:00:00",Qt::ISODate); QDateTime start = QDateTime::fromString("2023-11-27T17:50:00",Qt::ISODate);
//QDateTime start = QDateTime::currentDateTime();
QDateTime end = start.addSecs(120); QDateTime end = start.addSecs(120);
calculator.GetCostFromDuration(&cfg, 3, start, end, 60); uint32_t cost = calculator.GetCostFromDuration(&cfg, 3, start, end, 60);
qCritical() << "cost=" << cost;
qCritical() << "end=" << end;
} }
return 0; return 0;