Compare commits
	
		
			94 Commits
		
	
	
		
			fix-kirchd
			...
			5be5798681
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 5be5798681 | |||
| 1a24bc4572 | |||
| 068575f8e8 | |||
| e121cef17e | |||
| bdaea1106c | |||
| bc17213597 | |||
| 64b2b4bd85 | |||
| 8737508839 | |||
| 02f0500eac | |||
| 2d53224feb | |||
| 38964ad9a8 | |||
| d3f18f3b82 | |||
| 1a4265372e | |||
| eb20410849 | |||
| 2a492475e3 | |||
| 6a5272da7a | |||
| 4d5583df2d | |||
| de6f263817 | |||
| e4ce14da3f | |||
| b71c979a91 | |||
| 33f43fb83d | |||
| 1606a784a4 | |||
| 9f2cf613e1 | |||
| dbd9644047 | |||
| f8fef12b55 | |||
| 1dd81d4a56 | |||
| f93cf11d23 | |||
| 9d0ac4f8ce | |||
| 4ec5589f30 | |||
| c1d5ffcf3c | |||
| 7aded2b7fb | |||
| ed9e60a972 | |||
| 5d35331c83 | |||
| c7623429d9 | |||
| b596086245 | |||
| 24ca857b6f | |||
| ea0bcde413 | |||
| e93b08505a | |||
| 44efd44ede | |||
| 9d72574106 | |||
| d8ec41ebfc | |||
| f17c4f240b | |||
| 76bb1eb56a | |||
| 3b813e5eff | |||
| 436f5a109b | |||
| e17e54b315 | |||
| d07fdd8540 | |||
| 1fab458de3 | |||
| 1ac2ca91c5 | |||
| 1852f552a3 | |||
| e2c02420f0 | |||
| 1240abbbec | |||
| e20eb93abf | |||
| b0e7bd91b4 | |||
| 0cd4424434 | |||
| 1991853b66 | |||
| b31fcc5f4f | |||
| dbe649d0e4 | |||
| dfd74a455a | |||
| 10828e0708 | |||
| 2dbcc7018c | |||
| 18ed85430d | |||
| b70094abb5 | |||
| aa15d1c126 | |||
| cd159f2bbd | |||
| 475487c2ce | |||
| 8ff5b8e2b5 | |||
| b0c4ad0e2e | |||
| 7e3347b043 | |||
| 4e7fa83507 | |||
| ac76f194e1 | |||
| fe485b7b39 | |||
| e210224340 | |||
| afa62183fd | |||
| f241a87dc1 | |||
| 9b1cc49d34 | |||
| 449e618417 | |||
| aba38d2614 | |||
| 57d6b85f52 | |||
| 4f9c0422fc | |||
| a5b95d71b8 | |||
| fcbc8dcdc3 | |||
| ed99bae725 | |||
| 627d14204d | |||
| d8a4c4eaa7 | |||
| 4f45db4fde | |||
| a744a1ebb3 | |||
| df16bd7f9c | |||
| b751ba339e | |||
| 588a88455b | |||
| 92bfdced6a | |||
| 8bbec596c9 | |||
| 87b14ee3f8 | |||
| 3ad2c77467 | 
| @@ -84,6 +84,51 @@ struct CALCULATE_LIBRARY_API CalcState { | |||||||
|         return (m_status == State::SUCCESS); |         return (m_status == State::SUCCESS); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     QString toString() { | ||||||
|  |         QString s; | ||||||
|  |         switch (m_status) { | ||||||
|  |         case State::SUCCESS: | ||||||
|  |             s = "SUCCESS"; | ||||||
|  |             break; | ||||||
|  |         case State::ERROR_PARSING_ZONE_NR: | ||||||
|  |             s = "ERROR_PARSING_ZONE_NR"; | ||||||
|  |             break; | ||||||
|  |         case State::ERROR_LOADING_TARIFF: | ||||||
|  |             s = "ERROR_LOADING_TARIFF"; | ||||||
|  |             break; | ||||||
|  |         case State::ERROR_PARSING_TARIFF: | ||||||
|  |             s = "ERROR_PARSING_TARIFF"; | ||||||
|  |             break; | ||||||
|  |         case State::NEGATIVE_PARING_TIME: | ||||||
|  |             s = "NEGATIVE_PARKING_TIME"; | ||||||
|  |            break; | ||||||
|  |         case State::ABOVE_MAX_PARKING_TIME: | ||||||
|  |             s = "ABOVE_MAX_PARKING_TIME"; | ||||||
|  |             break; | ||||||
|  |         case State::WRONG_PARAM_VALUES: | ||||||
|  |             s = "WRONG_PARAM_VALUES"; | ||||||
|  |             break; | ||||||
|  |         case State::BELOW_MIN_PARKING_TIME: | ||||||
|  |             s = "BELOW_MIN_PARKING_TIME"; | ||||||
|  |             break; | ||||||
|  |         case State::BELOW_MIN_PARKING_PRICE: | ||||||
|  |             s = "BELOW_MIN_PARKING_PRICE"; | ||||||
|  |             break; | ||||||
|  |         case State::OVERPAID: | ||||||
|  |             s = "OVERPAID"; | ||||||
|  |             break; | ||||||
|  |         case State::INVALID_START_DATE: | ||||||
|  |             s = "INVALID_START_DATE"; | ||||||
|  |             break; | ||||||
|  |         case State::WRONG_ISO_TIME_FORMAT: | ||||||
|  |             s = "WRONG_ISO_TIME_FORMAT"; | ||||||
|  |             break; | ||||||
|  |         case State::OUTSIDE_ALLOWED_PARKING_TIME: | ||||||
|  |             s = "OUTSIDE_ALLOWED_PARKING_TIME"; | ||||||
|  |         } | ||||||
|  |         return s + ":" + m_desc; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     explicit operator QString () const noexcept { |     explicit operator QString () const noexcept { | ||||||
|         QString s; |         QString s; | ||||||
|         switch (m_status) { |         switch (m_status) { | ||||||
| @@ -130,6 +175,8 @@ struct CALCULATE_LIBRARY_API CalcState { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     CalcState &set(State s) { m_status = s; return *this; } |     CalcState &set(State s) { m_status = s; return *this; } | ||||||
|  |     CalcState &setStatus(State s) { return set(s); } | ||||||
|  |     State getStatus() const { return m_status; } | ||||||
|     CalcState &setDesc(QString s) { m_desc = s; return *this; } |     CalcState &setDesc(QString s) { m_desc = s; return *this; } | ||||||
|  |  | ||||||
|     void setAllowedTimeRange(QTime const &from, QTime const &until) { |     void setAllowedTimeRange(QTime const &from, QTime const &until) { | ||||||
| @@ -149,11 +196,29 @@ int CALCULATE_LIBRARY_API get_zone_nr(int zone = -1); | |||||||
| int CALCULATE_LIBRARY_API compute_next_timestep(parking_tariff_t *tariff, int currentTimeMinutes, int UpDown); | int CALCULATE_LIBRARY_API compute_next_timestep(parking_tariff_t *tariff, int currentTimeMinutes, int UpDown); | ||||||
|  |  | ||||||
| QList<int> CALCULATE_LIBRARY_API get_time_steps(Configuration *cfg); | QList<int> CALCULATE_LIBRARY_API get_time_steps(Configuration *cfg); | ||||||
| int CALCULATE_LIBRARY_API get_minimal_parkingtime(Configuration *cfg, PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING); |  | ||||||
| int CALCULATE_LIBRARY_API get_maximal_parkingtime(Configuration *cfg, PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING); | int CALCULATE_LIBRARY_API get_minimal_parkingtime(Configuration const *cfg, | ||||||
| int CALCULATE_LIBRARY_API get_minimal_parkingprice(Configuration *cfg, PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING); |                                                   PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING, | ||||||
| int CALCULATE_LIBRARY_API get_maximal_parkingprice(Configuration *cfg, PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING); |                                                   int paymentOptionIndex=0); | ||||||
| int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg, PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING); |  | ||||||
|  | int CALCULATE_LIBRARY_API get_maximal_parkingtime(Configuration const *cfg, | ||||||
|  |                                                   PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING, | ||||||
|  |                                                   int paymentOptionIndex=0); | ||||||
|  |  | ||||||
|  | int CALCULATE_LIBRARY_API get_minimal_parkingprice(Configuration *cfg, | ||||||
|  |                                                    PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING, | ||||||
|  |                                                    int paymentOptionIndex = 0, | ||||||
|  |                                                    QDateTime const &start = QDateTime::currentDateTime()); | ||||||
|  |  | ||||||
|  | int CALCULATE_LIBRARY_API get_maximal_parkingprice(Configuration *cfg, | ||||||
|  |                                                    PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING, | ||||||
|  |                                                    int paymentOptionIndex=0); | ||||||
|  |  | ||||||
|  | int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg, | ||||||
|  |                                                 PERMIT_TYPE permitType = PERMIT_TYPE::SHORT_TERM_PARKING, | ||||||
|  |                                                 QDateTime const &start = QDateTime::currentDateTime(), | ||||||
|  |                                                 QDateTime *productStart = nullptr, | ||||||
|  |                                                 QDateTime *productEnd = nullptr); | ||||||
|  |  | ||||||
| CalcState CALCULATE_LIBRARY_API compute_price_for_parking_ticket(       // deprecated | CalcState CALCULATE_LIBRARY_API compute_price_for_parking_ticket(       // deprecated | ||||||
|                                             parking_tariff_t *tariff, |                                             parking_tariff_t *tariff, | ||||||
|   | |||||||
| @@ -14,9 +14,19 @@ | |||||||
| using namespace std; | using namespace std; | ||||||
|  |  | ||||||
| class Calculator { | class Calculator { | ||||||
|     mutable QList<int> m_timeSteps; |     mutable QVector<QList<int>> m_timeSteps; | ||||||
|     mutable QList<int> m_priceSteps; |     mutable QList<int> m_priceSteps; | ||||||
|  |  | ||||||
|  |     CalcState isParkingAllowedForWeekDay(Configuration const *cfg, | ||||||
|  |                                          QDateTime const &start, | ||||||
|  |                                          int netto_parking_time, | ||||||
|  |                                          int paymentOptionIndex); | ||||||
|  |  | ||||||
|  |     CalcState isParkingAllowedForSpecialDay(Configuration const *cfg, | ||||||
|  |                                             QDateTime const &start, | ||||||
|  |                                             int netto_parking_time, | ||||||
|  |                                             int paymentOptionIndex); | ||||||
|  |  | ||||||
| protected: | protected: | ||||||
|     explicit Calculator() = default; |     explicit Calculator() = default; | ||||||
|  |  | ||||||
| @@ -29,13 +39,28 @@ public: | |||||||
|         return c; |         return c; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void ResetTimeSteps() { m_timeSteps.clear(); } |     void ResetTimeSteps(int paymentOptionIndex) { | ||||||
|     QList<int> timeSteps() const { return m_timeSteps; } |         if (m_timeSteps.size() > 0) { | ||||||
|  |             m_timeSteps[paymentOptionIndex].clear(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     QList<int> timeSteps(int paymentOptionIndex=0) const { | ||||||
|  |         if (m_timeSteps.size() > 0) { | ||||||
|  |             return m_timeSteps[paymentOptionIndex]; | ||||||
|  |         } | ||||||
|  |         return QList<int>(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     void ResetPriceSteps() { m_priceSteps.clear(); } |     void ResetPriceSteps() { m_priceSteps.clear(); } | ||||||
|     QList<int> priceSteps() const { return m_priceSteps; } |     QList<int> priceSteps() const { return m_priceSteps; } | ||||||
|  |  | ||||||
|     CalcState isParkingAllowed(Configuration const *cfg, QDateTime const &start); |     CalcState isParkingAllowed(Configuration const *cfg, | ||||||
|  |                                QDateTime const &start); | ||||||
|  |  | ||||||
|  |     CalcState isParkingAllowed(Configuration const *cfg, | ||||||
|  |                                QDateTime const &start, | ||||||
|  |                                int netto_parking_time, | ||||||
|  |                                int paymentOptionIndex); | ||||||
|  |  | ||||||
| 	/// <summary> | 	/// <summary> | ||||||
| 	/// Gets duration in seconds from cost | 	/// Gets duration in seconds from cost | ||||||
| @@ -66,8 +91,8 @@ public: | |||||||
|     // helper function to find time steps for a tariff with PaymentMethod::Steps |     // helper function to find time steps for a tariff with PaymentMethod::Steps | ||||||
|     // (e.g. Schoenau/Koenigsee) |     // (e.g. Schoenau/Koenigsee) | ||||||
|     // |     // | ||||||
|     QList<int> GetTimeSteps(Configuration *cfg) const; |     QList<int> GetTimeSteps(Configuration *cfg, int paymentOptionIndex=0) const; | ||||||
|     QList<int> GetSteps(Configuration *cfg) const { return GetTimeSteps(cfg); } |     QList<int> GetSteps(Configuration *cfg, int paymentOptionIndex=0) const { return GetTimeSteps(cfg, paymentOptionIndex); } | ||||||
|  |  | ||||||
|     QList<int> GetPriceSteps(Configuration *cfg) const; |     QList<int> GetPriceSteps(Configuration *cfg) const; | ||||||
|  |  | ||||||
| @@ -87,8 +112,8 @@ public: | |||||||
| // testing public: | // testing public: | ||||||
|     // Introduced for PaymentMethod::Steps (e.g. Schoenau) |     // Introduced for PaymentMethod::Steps (e.g. Schoenau) | ||||||
|     // For tariff of following structure: only steps, no special days, nonstop. |     // For tariff of following structure: only steps, no special days, nonstop. | ||||||
|     uint32_t GetCostFromDuration(Configuration *cfg, QDateTime const &start, quint64 durationMinutes) const; |     uint32_t GetCostFromDuration(Configuration *cfg, QDateTime const &start, quint64 durationMinutes, int paymentOptionIndex=0) const; | ||||||
|     uint32_t GetCostFromDuration(Configuration *cfg, QDateTime const &start, QDateTime const &end) const; |     uint32_t GetCostFromDuration(Configuration *cfg, QDateTime const &start, QDateTime const &end, int paymentOptionIndex=0) const; | ||||||
|  |  | ||||||
| private: | private: | ||||||
|     Ticket private_GetCostFromDuration(Configuration const* cfg, |     Ticket private_GetCostFromDuration(Configuration const* cfg, | ||||||
| @@ -104,7 +129,7 @@ private: | |||||||
|                               int durationMinutes); |                               int durationMinutes); | ||||||
|  |  | ||||||
|     // |     // | ||||||
|     uint32_t GetPriceForTimeStep(Configuration *cfg, int timeStep) const; |     uint32_t GetPriceForTimeStep(Configuration *cfg, int timeStep, int paymentOptionIndex=0) const; | ||||||
|     uint32_t GetPriceForStep(Configuration *cfg, int step) const { |     uint32_t GetPriceForStep(Configuration *cfg, int step) const { | ||||||
|         return GetPriceForTimeStep(cfg, step); |         return GetPriceForTimeStep(cfg, step); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -36,6 +36,12 @@ class Calculator; | |||||||
| class Configuration | class Configuration | ||||||
| { | { | ||||||
| public: | public: | ||||||
|  |     using SpecialDaysType = std::multimap<int, ATBSpecialDays>; | ||||||
|  |     using SpecialDaysWorktimeType = std::multimap<int, ATBSpecialDaysWorktime>; | ||||||
|  |     using TimeRangeType = std::multimap<int, ATBTimeRange>; | ||||||
|  |     using TariffProductType = std::multimap<int, ATBTariffProduct>; | ||||||
|  |     using ATBPaymentOptionType = std::multimap<int, ATBPaymentOption>; | ||||||
|  |  | ||||||
|     ATBProject project; |     ATBProject project; | ||||||
|     ATBCurrency Currency; |     ATBCurrency Currency; | ||||||
| 	ATBDuration duration; | 	ATBDuration duration; | ||||||
| @@ -43,18 +49,18 @@ public: | |||||||
| 	multimap<int, ATBDuration> Duration; | 	multimap<int, ATBDuration> Duration; | ||||||
| 	multimap<int, ATBPaymentMethod> PaymentMethod; | 	multimap<int, ATBPaymentMethod> PaymentMethod; | ||||||
| 	multimap<int, ATBPaymentRate> PaymentRate; | 	multimap<int, ATBPaymentRate> PaymentRate; | ||||||
| 	multimap<int, ATBSpecialDaysWorktime> SpecialDaysWorktime; |     SpecialDaysWorktimeType SpecialDaysWorktime; | ||||||
| 	multimap<int, ATBSpecialDays> SpecialDays; |     SpecialDaysType SpecialDays; | ||||||
| 	multimap<int, ATBWeekDays> WeekDays; | 	multimap<int, ATBWeekDays> WeekDays; | ||||||
| 	multimap<int, ATBPeriodYear> YearPeriod; | 	multimap<int, ATBPeriodYear> YearPeriod; | ||||||
| 	multimap<int, ATBWeekDaysWorktime> WeekDaysWorktime; | 	multimap<int, ATBWeekDaysWorktime> WeekDaysWorktime; | ||||||
| 	multimap<int, ATBPaymentOption> PaymentOption; |     ATBPaymentOptionType PaymentOption; | ||||||
|     multimap<int, ATBDailyTicket> DailyTicket; |     multimap<int, ATBDailyTicket> DailyTicket; | ||||||
|     multimap<int, ATBTimeRange> TimeRange; |     TimeRangeType TimeRange; | ||||||
|     multimap<int, ATBTimeStepConfig> TimeStepConfig; |     multimap<int, ATBTimeStepConfig> TimeStepConfig; | ||||||
|     multimap<int, ATBTimeBase> TimeBase; |     multimap<int, ATBTimeBase> TimeBase; | ||||||
|     multimap<int, ATBCustomer> Customer; |     multimap<int, ATBCustomer> Customer; | ||||||
|     multimap<int, ATBTariffProduct> TariffProduct; |     TariffProductType TariffProduct; | ||||||
|  |  | ||||||
| 	/// <summary> | 	/// <summary> | ||||||
| 	/// Parse JSON string | 	/// Parse JSON string | ||||||
| @@ -63,10 +69,18 @@ public: | |||||||
| 	/// <returns>Returns operation status bool (OK | FAIL) </returns> | 	/// <returns>Returns operation status bool (OK | FAIL) </returns> | ||||||
| 	bool ParseJson(Configuration* cfg, const char* json); | 	bool ParseJson(Configuration* cfg, const char* json); | ||||||
|  |  | ||||||
|     ATBPaymentOption &getPaymentOptions(); |     ATBPaymentOption &getPaymentOptions(int paymentOptionsIndex=0); | ||||||
|     ATBPaymentOption const &getPaymentOptions() const; |     ATBPaymentOption const &getPaymentOptions(int paymentOptionsIndex=0) const; | ||||||
|     QVector<ATBPaymentOption> &getAllPaymentOptions(); |     QVector<ATBPaymentOption> &getAllPaymentOptions(); | ||||||
|     QVector<ATBPaymentOption> const &getAllPaymentOptions() const; |     QVector<ATBPaymentOption> const &getAllPaymentOptions() const; | ||||||
|  |     int getPaymentOptionIndex(QDateTime const &dt) const; | ||||||
|  |     int getPaymentOptionIndexIfSpecialDay(QDateTime const &dt) const; | ||||||
|  |     bool isSpecialDay(QDateTime const &dt) const; | ||||||
|  |     int specialDayId(QDateTime const &dt) const; | ||||||
|  |     ATBSpecialDays specialDay(QDateTime const &dt) const; | ||||||
|  |     bool isDayIncluded(uint64_t businessHours, QDateTime const &dt) const; | ||||||
|  |     bool isDayIncludedAsSpecialDay(uint64_t businessHours, QDateTime const &dt) const; | ||||||
|  |     bool isDayIncludedAsSpecialDay(uint64_t businessHours, int specialDayId) const; | ||||||
|     std::optional<QVector<ATBPaymentRate>> getPaymentRateForAllKeys() const; |     std::optional<QVector<ATBPaymentRate>> getPaymentRateForAllKeys() const; | ||||||
|     std::optional<QVector<ATBPaymentRate>> getPaymentRateForKey(int key) const; |     std::optional<QVector<ATBPaymentRate>> getPaymentRateForKey(int key) const; | ||||||
|     std::optional<QVector<ATBDailyTicket>> getDailyTicketsForAllKeys() const; |     std::optional<QVector<ATBDailyTicket>> getDailyTicketsForAllKeys() const; | ||||||
| @@ -74,6 +88,7 @@ public: | |||||||
|     std::optional<QVector<ATBTariffProduct>> getTariffProductForAllKeys() const; |     std::optional<QVector<ATBTariffProduct>> getTariffProductForAllKeys() const; | ||||||
|     std::optional<QVector<ATBTariffProduct>> getTariffProductForProductId(int id) const; |     std::optional<QVector<ATBTariffProduct>> getTariffProductForProductId(int id) const; | ||||||
|     std::optional<QVector<ATBTariffProduct>> getTariffProductForProductId(PermitType permitType) const; |     std::optional<QVector<ATBTariffProduct>> getTariffProductForProductId(PermitType permitType) const; | ||||||
|  |     std::optional<QVector<ATBTariffProduct>> getTariffProductForProductTypeName(QString const &permitTypeName) const; | ||||||
|     std::optional<ATBCustomer> getCustomerForType(ATBCustomer::CustomerType customerType); |     std::optional<ATBCustomer> getCustomerForType(ATBCustomer::CustomerType customerType); | ||||||
|     std::optional<ATBWeekDaysWorktime> getWeekDayWorkTime(QTime const &time, Qt::DayOfWeek dayOfWeek); |     std::optional<ATBWeekDaysWorktime> getWeekDayWorkTime(QTime const &time, Qt::DayOfWeek dayOfWeek); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,11 @@ | |||||||
| #pragma once | #ifndef PAYMENT_OPT_H_INCLUDED | ||||||
|  | #define PAYMENT_OPT_H_INCLUDED | ||||||
|  |  | ||||||
|  |  | ||||||
| #include <string> | #include <string> | ||||||
|  | #include <cinttypes> | ||||||
|  |  | ||||||
|  | #include <QDateTime> | ||||||
|  |  | ||||||
| class ATBPaymentOption | class ATBPaymentOption | ||||||
| { | { | ||||||
| @@ -18,7 +24,10 @@ public: | |||||||
|         pop_min_price = 0; |         pop_min_price = 0; | ||||||
|         pop_max_price = 0; |         pop_max_price = 0; | ||||||
|         pop_carry_over = -1; |         pop_carry_over = -1; | ||||||
|  |         pop_carry_over_target = false; | ||||||
|         pop_carry_over_time_range_id = -1; |         pop_carry_over_time_range_id = -1; | ||||||
|  |         pop_carry_over_start_time_range = -1; | ||||||
|  |         pop_carry_over_end_time_range = -1; | ||||||
|         pop_daily_card_price = -1; |         pop_daily_card_price = -1; | ||||||
|         pop_business_hours = -1; |         pop_business_hours = -1; | ||||||
|         pop_time_step_config = -1; |         pop_time_step_config = -1; | ||||||
| @@ -35,8 +44,21 @@ public: | |||||||
|     double pop_min_price; |     double pop_min_price; | ||||||
|     double pop_max_price; |     double pop_max_price; | ||||||
|     int pop_carry_over; |     int pop_carry_over; | ||||||
|  |     bool pop_carry_over_target; | ||||||
|     int pop_carry_over_time_range_id; |     int pop_carry_over_time_range_id; | ||||||
|  |     int pop_carry_over_start_time_range; | ||||||
|  |     int pop_carry_over_end_time_range; | ||||||
|     int pop_daily_card_price; |     int pop_daily_card_price; | ||||||
|     int pop_business_hours; |     uint64_t pop_business_hours; | ||||||
|     int pop_time_step_config; |     int pop_time_step_config; | ||||||
|  |  | ||||||
|  |     struct ATBMaxDateTime { | ||||||
|  |         int direction; | ||||||
|  |         uint8_t week; | ||||||
|  |         uint8_t day; | ||||||
|  |         QTime time; | ||||||
|  |     } pop_min_date_time, | ||||||
|  |       pop_max_date_time; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | #endif // PAYMENT_OPT_H_INCLUDED | ||||||
|   | |||||||
| @@ -4,10 +4,12 @@ | |||||||
| class ATBSpecialDays | class ATBSpecialDays | ||||||
| { | { | ||||||
| public: | public: | ||||||
|  |     explicit ATBSpecialDays() = default; | ||||||
| 	int ped_id; | 	int ped_id; | ||||||
| 	std::string ped_label; | 	std::string ped_label; | ||||||
| 	std::string ped_date_start; | 	std::string ped_date_start; | ||||||
| 	std::string ped_date_end; | 	std::string ped_date_end; | ||||||
|     int ped_period_special_day_id; |     int ped_period_special_day_id; | ||||||
|  |     int ped_payment_option_id; | ||||||
| 	int ped_year; | 	int ped_year; | ||||||
| }; | }; | ||||||
| @@ -4,19 +4,87 @@ | |||||||
| /// <summary> | /// <summary> | ||||||
| /// Business hours (byte represents payment option id) | /// Business hours (byte represents payment option id) | ||||||
| /// </summary> | /// </summary> | ||||||
| enum BusinessHours | /// | ||||||
|  |  | ||||||
|  | #include <Qt> | ||||||
|  |  | ||||||
|  | #define _NO_RESTRICTION_24_7_               (uint64_t)(0ULL) | ||||||
|  | #define _MON_                               (uint64_t)(1ULL << 8) | ||||||
|  | #define _TUE_                               (uint64_t)(1ULL << 9) | ||||||
|  | #define _WED_                               (uint64_t)(1ULL << 10) | ||||||
|  | #define _THU_                               (uint64_t)(1ULL << 11) | ||||||
|  | #define _FRI_                               (uint64_t)(1ULL << 12) | ||||||
|  | #define _SAT_                               (uint64_t)(1ULL << 13) | ||||||
|  | #define _SUN_                               (uint64_t)(1ULL << 14) | ||||||
|  | #define _WEEK_DAYS_                         ((_MON_|_TUE_|_WED_|_THU_|_FRI_)) | ||||||
|  | #define _WORKING_DAYS_                      ((_MON_|_TUE_|_WED_|_THU_|_FRI_|_SAT_)) | ||||||
|  | #define _ALL_DAYS_                          ((_MON_|_TUE_|_WED_|_THU_|_FRI_|_SAT_|_SUN_)) | ||||||
|  | #define _OFFICIAL_HOLIDAY_                  (uint64_t)(1ULL << 15) | ||||||
|  | #define _ONLY_WEEKEND_                      ((_SAT_|_SUN_)) | ||||||
|  | #define _ONLY_OPEN_FOR_BUSINESS_DAYS_       (uint64_t)(1ULL << 16)    /* verkaufsoffen */ | ||||||
|  | #define _WITH_RESTRICTED_HOURS_             (uint64_t)(1ULL << 17) | ||||||
|  | #define _ALL_DAYS_WITH_RESTRICTED_HOURS_    ((_WITH_RESTRICTED_HOURS_|_ALL_DAYS_)) | ||||||
|  | #define _WEEKEND_WITH_RESTRICTED_HOURS_     ((_WITH_RESTRICTED_HOURS_|_FRI_|_SAT_|_SUN_)) | ||||||
|  | #define _FRI_WITH_RESTRICTED_HOURS_         ((_WITH_RESTRICTED_HOURS_|_FRI_)) | ||||||
|  | #define _SAT_WITH_RESTRICTED_HOURS_         ((_WITH_RESTRICTED_HOURS_|_SAT_)) | ||||||
|  | #define _SUN_WITH_RESTRICTED_HOURS_         ((_WITH_RESTRICTED_HOURS_|_SUN_)) | ||||||
|  | #define _NOT_DEFINED_                       (uint64_t)(~0ULL) | ||||||
|  |  | ||||||
|  | enum BusinessHours : std::uint64_t | ||||||
| { | { | ||||||
|     NoRestriction_24_7          = 0, |     NoRestriction_24_7          = 0, | ||||||
|     OnlyWorkingDays             = 1,    // [Monday-Friday] |     OnlyWorkingDays             = 1,    // [Monday-Friday] | ||||||
|     OnlyWeekDays                = 2,    // [Monday-Saturday] |     OnlyWeekDays                = 2,    // [Monday-Saturday] | ||||||
|     OnlyWeekEnd                 = 4,    // [Saturday+Sunday] |     OnlyWeekEnd                 = 4,    // [Saturday+Sunday] | ||||||
|     OnlyOfficialHolidays        = 8, |     OnlyOfficialHolidays        = 8, | ||||||
|  |     OnlySundaysAndHolidays      = 12,   // [Sun+Holiday] | ||||||
|     OnlySpecialDays             = 16, |     OnlySpecialDays             = 16, | ||||||
|     OnlySchoolHolidays          = 32, |     OnlySchoolHolidays          = 32, | ||||||
|     SpecialAndSchoolHolidays    = 48, |     SpecialAndSchoolHolidays    = 48, | ||||||
|     OnlyOpenForBusinessDays     = 64,   // verkaufsoffen |     OnlyOpenForBusinessDays     = 64,   // verkaufsoffen | ||||||
|     AllDaysWithRestrictedHours  = 128,  // every day, restricted to some time range |     AllDaysWithRestrictedHours  = 128,  // every day, restricted to some time range | ||||||
|     NoBusinessHoursDefined      = 255 |     NoBusinessHoursDefined      = 255, | ||||||
|  |  | ||||||
|  |     // new 12.04.2024 | ||||||
|  |     NO_RESTRICTION_24_7 = _NO_RESTRICTION_24_7_, | ||||||
|  |     MON = _MON_, | ||||||
|  |     TUE = _TUE_, | ||||||
|  |     WED = _WED_, | ||||||
|  |     THU = _THU_, | ||||||
|  |     FRI = _FRI_, | ||||||
|  |     SAT = _SAT_, | ||||||
|  |     SUN = _SUN_, | ||||||
|  |     WEEK_DAYS = _WEEK_DAYS_, | ||||||
|  |     WORKING_DAYS = _WORKING_DAYS_, | ||||||
|  |     ALL_DAYS = _ALL_DAYS_, | ||||||
|  |     OFFICIAL_HOLIDAY = _OFFICIAL_HOLIDAY_, | ||||||
|  |     ONLY_WEEKEND = _ONLY_WEEKEND_, | ||||||
|  |     ONLY_OPEN_FOR_BUSINESS_DAYS = _ONLY_OPEN_FOR_BUSINESS_DAYS_, | ||||||
|  |     ALL_DAYS_WITH_RESTRICTED_HOURS = _ALL_DAYS_WITH_RESTRICTED_HOURS_, | ||||||
|  |     WEEKEND_WITH_RESTRICTED_HOURS = _WEEKEND_WITH_RESTRICTED_HOURS_, | ||||||
|  |     FRI_WITH_RESTRICTED_HOURS = _FRI_WITH_RESTRICTED_HOURS_, | ||||||
|  |     SAT_WITH_RESTRICTED_HOURS = _SAT_WITH_RESTRICTED_HOURS_, | ||||||
|  |     SUN_WITH_RESTRICTED_HOURS = _SUN_WITH_RESTRICTED_HOURS_, | ||||||
|  |     NOT_DEFINED = _NOT_DEFINED_ | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #if 0 | ||||||
|  | static bool business(uint64_t businessHours, QDateTime &const dt) { | ||||||
|  |     switch (dayOfWeek) { | ||||||
|  |     case Qt::Monday: | ||||||
|  |         (businessHours & _MON_) == _MON_; | ||||||
|  |     case Qt::Tuesday: | ||||||
|  |     case Qt::Wednesday: | ||||||
|  |     case Qt::Thursday: | ||||||
|  |     case Qt::Saturday: | ||||||
|  |     case Qt::Sunday: | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | struct BusinessHours_struct { | ||||||
|  |     BusinessHours bh; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| #endif // TARIFF_BUSINESS_HOURS_H_INCLUDED | #endif // TARIFF_BUSINESS_HOURS_H_INCLUDED | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								library/include/mobilisis/tariff_global_defines.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								library/include/mobilisis/tariff_global_defines.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | #ifndef TARIFF_GLOBAL_DEFINES_H_INCLUDED | ||||||
|  | #define TARIFF_GLOBAL_DEFINES_H_INCLUDED | ||||||
|  |  | ||||||
|  | #define DBG_HEADER "(" << __func__ << ":" << __LINE__ << ")" | ||||||
|  |  | ||||||
|  | #endif // TARIFF_GLOBAL_DEFINES_H_INCLUDED | ||||||
| @@ -4,14 +4,16 @@ | |||||||
| #include <QString> | #include <QString> | ||||||
|  |  | ||||||
| enum class PERMIT_TYPE : quint8 { | enum class PERMIT_TYPE : quint8 { | ||||||
|     SHORT_TERM_PARKING, |     SHORT_TERM_PARKING=0, | ||||||
|     DAY_TICKET, |     DAY_TICKET=1, | ||||||
|     SZEGED_START, |     SZEGED_START=2, | ||||||
|     SZEGED_STOP, |     SZEGED_STOP=3, | ||||||
|     DAY_TICKET_ADULT, |     DAY_TICKET_ADULT=4, | ||||||
|     DAY_TICKET_TEEN, |     DAY_TICKET_TEEN=5, | ||||||
|     DAY_TICKET_CHILD, |     DAY_TICKET_CHILD=6, | ||||||
|     INVALID |     INVALID=7, | ||||||
|  |     FOOD_STAMP=8, | ||||||
|  |     TWENTY_FOUR_HOURS_TICKET=9 | ||||||
| }; | }; | ||||||
|  |  | ||||||
| struct PermitType { | struct PermitType { | ||||||
| @@ -41,6 +43,12 @@ struct PermitType { | |||||||
|         case 6: |         case 6: | ||||||
|             m_permitType = PERMIT_TYPE::DAY_TICKET_CHILD; |             m_permitType = PERMIT_TYPE::DAY_TICKET_CHILD; | ||||||
|             break; |             break; | ||||||
|  |         case 7: | ||||||
|  |             m_permitType = PERMIT_TYPE::FOOD_STAMP; | ||||||
|  |             break; | ||||||
|  |         case 8: | ||||||
|  |             m_permitType = PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET; | ||||||
|  |             break; | ||||||
|         default: |         default: | ||||||
|             m_permitType = PERMIT_TYPE::INVALID; |             m_permitType = PERMIT_TYPE::INVALID; | ||||||
|         } |         } | ||||||
| @@ -68,6 +76,10 @@ struct PermitType { | |||||||
|             return 5; |             return 5; | ||||||
|         case PERMIT_TYPE::DAY_TICKET_TEEN: |         case PERMIT_TYPE::DAY_TICKET_TEEN: | ||||||
|             return 6; |             return 6; | ||||||
|  |         case PERMIT_TYPE::FOOD_STAMP: | ||||||
|  |             return 7; | ||||||
|  |         case PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET: | ||||||
|  |             return 8; | ||||||
|         default: |         default: | ||||||
|             break; |             break; | ||||||
|         } |         } | ||||||
| @@ -75,7 +87,7 @@ struct PermitType { | |||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     operator QString() const { |     QString toString() { | ||||||
|         switch(m_permitType) { |         switch(m_permitType) { | ||||||
|         case PERMIT_TYPE::DAY_TICKET: |         case PERMIT_TYPE::DAY_TICKET: | ||||||
|             return QString("DAY_TICKET"); |             return QString("DAY_TICKET"); | ||||||
| @@ -91,11 +103,49 @@ struct PermitType { | |||||||
|             return QString("SZEGED_START"); |             return QString("SZEGED_START"); | ||||||
|         case PERMIT_TYPE::SZEGED_STOP: |         case PERMIT_TYPE::SZEGED_STOP: | ||||||
|             return QString("SZEGED_STOP"); |             return QString("SZEGED_STOP"); | ||||||
|  |         case PERMIT_TYPE::FOOD_STAMP: | ||||||
|  |             return QString("FOOD_STAMP"); | ||||||
|  |         case PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET: | ||||||
|  |             return QString("TWENTY_FOUR_HOURS_TICKET"); | ||||||
|         default: |         default: | ||||||
|             break; |             break; | ||||||
|         } |         } | ||||||
|         return QString("INVALID"); |         return QString("INVALID"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     QString toString() const { | ||||||
|  |         switch(m_permitType) { | ||||||
|  |         case PERMIT_TYPE::DAY_TICKET: | ||||||
|  |             return QString("DAY_TICKET"); | ||||||
|  |         case PERMIT_TYPE::DAY_TICKET_ADULT: | ||||||
|  |             return QString("DAY_TICKET_ADULT"); | ||||||
|  |         case PERMIT_TYPE::DAY_TICKET_CHILD: | ||||||
|  |             return QString("DAY_TICKET_CHILD"); | ||||||
|  |         case PERMIT_TYPE::DAY_TICKET_TEEN: | ||||||
|  |             return QString("DAY_TICKET_TEEN"); | ||||||
|  |         case PERMIT_TYPE::SHORT_TERM_PARKING: | ||||||
|  |             return QString("SHORT_TERM_PARKING"); | ||||||
|  |         case PERMIT_TYPE::SZEGED_START: | ||||||
|  |             return QString("SZEGED_START"); | ||||||
|  |         case PERMIT_TYPE::SZEGED_STOP: | ||||||
|  |             return QString("SZEGED_STOP"); | ||||||
|  |         case PERMIT_TYPE::FOOD_STAMP: | ||||||
|  |             return QString("FOOD_STAMP"); | ||||||
|  |         case PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET: | ||||||
|  |             return QString("TWENTY_FOUR_HOURS_TICKET"); | ||||||
|  |         default: | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         return QString("INVALID"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     operator QString () { | ||||||
|  |         return toString(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     operator QString () const { | ||||||
|  |         return toString(); | ||||||
|  |     } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| #endif // TARIFF_PERMIT_TYPE_H_INCLUDED | #endif // TARIFF_PERMIT_TYPE_H_INCLUDED | ||||||
|   | |||||||
| @@ -45,6 +45,8 @@ struct ATBTariffProduct { | |||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     uint32_t getProductPrice() const { return m_tariff_product_price; } | ||||||
|  |  | ||||||
|     friend QDebug operator<<(QDebug debug, ATBTariffProduct const &product) { |     friend QDebug operator<<(QDebug debug, ATBTariffProduct const &product) { | ||||||
|         QDebugStateSaver saver(debug); |         QDebugStateSaver saver(debug); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,6 +15,9 @@ | |||||||
| using namespace std; | using namespace std; | ||||||
|  |  | ||||||
| namespace Utilities { | namespace Utilities { | ||||||
|  |  | ||||||
|  |     bool isDayIncluded(uint64_t businessHours, QDateTime const &dt); | ||||||
|  |  | ||||||
| 	/// <summary> | 	/// <summary> | ||||||
| 	/// Get day of week from current date (Zeller's Algorithm), starting day is Sunday | 	/// Get day of week from current date (Zeller's Algorithm), starting day is Sunday | ||||||
| 	/// </summary> | 	/// </summary> | ||||||
| @@ -76,10 +79,12 @@ namespace Utilities { | |||||||
| 	double CalculatePricePerUnit(double pra_price, double durationUnit = -1); | 	double CalculatePricePerUnit(double pra_price, double durationUnit = -1); | ||||||
|  |  | ||||||
|     QTime SpecialDaysWorkTimeFrom(Configuration const *cfg, int specialDayId); |     QTime SpecialDaysWorkTimeFrom(Configuration const *cfg, int specialDayId); | ||||||
|  |     QTime SpecialDaysWorkTimeFrom(Configuration::SpecialDaysWorktimeType::const_iterator const &it); | ||||||
|     QTime SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId); |     QTime SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId); | ||||||
|     QTime WeekDaysWorkTimeFrom(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr); |     QTime SpecialDaysWorkTimeUntil(Configuration::SpecialDaysWorktimeType::const_iterator const &it); | ||||||
|     QTime WeekDaysWorkTimeUntil(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr); |     QTime WeekDaysWorkTimeFrom(std::multimap<int, ATBWeekDaysWorktime>::const_iterator const &itr); | ||||||
|     int WeekDayId(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr); |     QTime WeekDaysWorkTimeUntil(std::multimap<int, ATBWeekDaysWorktime>::const_iterator const &itr); | ||||||
|  |     int WeekDayId(std::multimap<int, ATBWeekDaysWorktime>::const_iterator const &itr); | ||||||
|     // PaymentRate GetPaymentRate(Configuration const *cfg, ); |     // PaymentRate GetPaymentRate(Configuration const *cfg, ); | ||||||
|     bool isCarryOverSet(Configuration const *cfg, PaymentMethod paymentMethodId); |     bool isCarryOverSet(Configuration const *cfg, PaymentMethod paymentMethodId); | ||||||
|     bool isCarryOverNotSet(Configuration const *cfg, PaymentMethod paymentMethodId); |     bool isCarryOverNotSet(Configuration const *cfg, PaymentMethod paymentMethodId); | ||||||
| @@ -94,4 +99,6 @@ namespace Utilities { | |||||||
|     BusinessHours getBusinessHours(Configuration const *cfg, PaymentMethod methodId); |     BusinessHours getBusinessHours(Configuration const *cfg, PaymentMethod methodId); | ||||||
|     uint32_t computeWeekDaysPrice(Configuration const *cfg, PaymentMethod id); |     uint32_t computeWeekDaysPrice(Configuration const *cfg, PaymentMethod id); | ||||||
|     double computeWeekDaysDurationUnit(Configuration const *cfg, PaymentMethod id); |     double computeWeekDaysDurationUnit(Configuration const *cfg, PaymentMethod id); | ||||||
|  |     QStringList dumpBusinessHours(uint64_t businessHours); | ||||||
|  |     uint32_t getDailyTicketCardPrice(Configuration const *cfg, PaymentMethod methodId); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -12,6 +12,10 @@ INCLUDEPATH += $$_PRO_FILE_PWD_/include/rapidjson | |||||||
| #Version is set in yocto recipe with "EXTRA_QMAKEVARS_PRE" | #Version is set in yocto recipe with "EXTRA_QMAKEVARS_PRE" | ||||||
| #VERSION=1.0.0 | #VERSION=1.0.0 | ||||||
|  |  | ||||||
|  | # 04.06.2024: Fix for Szeged: read price for daily ticket directly from entry | ||||||
|  | #             PaymentOptions in tariff-file if it is not given as part of a | ||||||
|  | #             Json-Product-Array in tariff-file. | ||||||
|  |  | ||||||
| CONFIG(debug, debug|release) { | CONFIG(debug, debug|release) { | ||||||
|     win32 { |     win32 { | ||||||
|         QMAKE_CXXFLAGS += -DCALCULATE_LIBRARY_EXPORTS |         QMAKE_CXXFLAGS += -DCALCULATE_LIBRARY_EXPORTS | ||||||
| @@ -79,7 +83,8 @@ HEADERS += \ | |||||||
|     include/mobilisis/tariff_timebase.h \ |     include/mobilisis/tariff_timebase.h \ | ||||||
|     include/mobilisis/tariff_timestep_config.h \ |     include/mobilisis/tariff_timestep_config.h \ | ||||||
|     include/mobilisis/tariff_product.h \ |     include/mobilisis/tariff_product.h \ | ||||||
|     include/mobilisis/tariff_permit_type.h |     include/mobilisis/tariff_permit_type.h \ | ||||||
|  |     include/mobilisis/tariff_global_defines.h | ||||||
|  |  | ||||||
| OTHER_FILES += src/main.cpp \ | OTHER_FILES += src/main.cpp \ | ||||||
|     ../tariffs/tariff_korneuburg.json \ |     ../tariffs/tariff_korneuburg.json \ | ||||||
|   | |||||||
| @@ -14,12 +14,14 @@ QList<int> CALCULATE_LIBRARY_API get_time_steps(Configuration *cfg) { | |||||||
|     return Calculator::GetInstance().GetTimeSteps(cfg); |     return Calculator::GetInstance().GetTimeSteps(cfg); | ||||||
| } | } | ||||||
|  |  | ||||||
| int CALCULATE_LIBRARY_API get_minimal_parkingtime(Configuration *cfg, PERMIT_TYPE permitType) { | int CALCULATE_LIBRARY_API get_minimal_parkingtime(Configuration const *cfg, | ||||||
|  |                                                   PERMIT_TYPE permitType, | ||||||
|  |                                                   int paymentOptionIndex) { | ||||||
|     int minTime = 0; |     int minTime = 0; | ||||||
|  |  | ||||||
|     switch(permitType) { |     switch(permitType) { | ||||||
|     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) |     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) | ||||||
|         minTime = cfg->getPaymentOptions().pop_min_time; |         minTime = cfg->getPaymentOptions(paymentOptionIndex).pop_min_time; | ||||||
|     } break; |     } break; | ||||||
|     case PERMIT_TYPE::DAY_TICKET_ADULT: { |     case PERMIT_TYPE::DAY_TICKET_ADULT: { | ||||||
|     } break; |     } break; | ||||||
| @@ -30,20 +32,22 @@ int CALCULATE_LIBRARY_API get_minimal_parkingtime(Configuration *cfg, PERMIT_TYP | |||||||
|     default: |     default: | ||||||
|         // for each new sell-procedure, recomute the timesteps. implicitly, set |         // for each new sell-procedure, recomute the timesteps. implicitly, set | ||||||
|         // the minimal parking time. |         // the minimal parking time. | ||||||
|         Calculator::GetInstance().ResetTimeSteps(); |         Calculator::GetInstance().ResetTimeSteps(paymentOptionIndex); | ||||||
|         Calculator::GetInstance().GetTimeSteps(cfg); |         Calculator::GetInstance().GetTimeSteps((Configuration *)cfg, paymentOptionIndex); | ||||||
|         minTime = qRound(cfg->getPaymentOptions().pop_min_time); |         minTime = qRound(cfg->getPaymentOptions(paymentOptionIndex).pop_min_time); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return minTime; |     return minTime; | ||||||
| } | } | ||||||
|  |  | ||||||
| int CALCULATE_LIBRARY_API get_maximal_parkingtime(Configuration *cfg, PERMIT_TYPE permitType) { | int CALCULATE_LIBRARY_API get_maximal_parkingtime(Configuration const *cfg, | ||||||
|  |                                                   PERMIT_TYPE permitType, | ||||||
|  |                                                   int paymentOptionIndex) { | ||||||
|     int maxTime = 0; |     int maxTime = 0; | ||||||
|  |  | ||||||
|     switch(permitType) { |     switch(permitType) { | ||||||
|     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) |     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) | ||||||
|         maxTime = cfg->getPaymentOptions().pop_max_time; |         maxTime = cfg->getPaymentOptions(paymentOptionIndex).pop_max_time; | ||||||
|     } break; |     } break; | ||||||
|     case PERMIT_TYPE::DAY_TICKET_ADULT: { |     case PERMIT_TYPE::DAY_TICKET_ADULT: { | ||||||
|     } break; |     } break; | ||||||
| @@ -55,14 +59,18 @@ int CALCULATE_LIBRARY_API get_maximal_parkingtime(Configuration *cfg, PERMIT_TYP | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     return maxTime; |     return maxTime; | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| int CALCULATE_LIBRARY_API get_minimal_parkingprice(Configuration *cfg, PERMIT_TYPE permitType) { | int CALCULATE_LIBRARY_API get_minimal_parkingprice(Configuration *cfg, | ||||||
|  |                                                    PERMIT_TYPE permitType, | ||||||
|  |                                                    int paymentOptionIndex, | ||||||
|  |                                                    QDateTime const &start) { | ||||||
|     int minPrice = -1; |     int minPrice = -1; | ||||||
|  |  | ||||||
|     switch(permitType) { |     switch(permitType) { | ||||||
|     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) |     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) | ||||||
|         minPrice = cfg->getPaymentOptions().pop_min_price; |         minPrice = cfg->getPaymentOptions(paymentOptionIndex).pop_min_price; | ||||||
|     } break; |     } break; | ||||||
|     case PERMIT_TYPE::DAY_TICKET_ADULT: { |     case PERMIT_TYPE::DAY_TICKET_ADULT: { | ||||||
|     } break; |     } break; | ||||||
| @@ -70,13 +78,20 @@ int CALCULATE_LIBRARY_API get_minimal_parkingprice(Configuration *cfg, PERMIT_TY | |||||||
|     } break; |     } break; | ||||||
|     case PERMIT_TYPE::DAY_TICKET_CHILD: { |     case PERMIT_TYPE::DAY_TICKET_CHILD: { | ||||||
|     } break; |     } break; | ||||||
|  |     case PERMIT_TYPE::DAY_TICKET: { | ||||||
|  |         minPrice = compute_product_price(cfg, permitType, start); | ||||||
|  |     } break; | ||||||
|     default: ; |     default: ; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return minPrice; |     return minPrice; | ||||||
| } | } | ||||||
|  |  | ||||||
| int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg, PERMIT_TYPE permitType) { | int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg, | ||||||
|  |                                                 PERMIT_TYPE permitType, | ||||||
|  |                                                 QDateTime const &start, | ||||||
|  |                                                 QDateTime *productStart, | ||||||
|  |                                                 QDateTime *productEnd) { | ||||||
|  |  | ||||||
|     switch(permitType) { |     switch(permitType) { | ||||||
|     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) |     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) | ||||||
| @@ -85,6 +100,8 @@ int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg, PERMIT | |||||||
|         // [[fallthrough]]; |         // [[fallthrough]]; | ||||||
|     case PERMIT_TYPE::DAY_TICKET_TEEN: |     case PERMIT_TYPE::DAY_TICKET_TEEN: | ||||||
|         // [[fallthrough]]; |         // [[fallthrough]]; | ||||||
|  |     case PERMIT_TYPE::FOOD_STAMP: | ||||||
|  |         // [[fallthrough]]; | ||||||
|     case PERMIT_TYPE::DAY_TICKET_ADULT: { |     case PERMIT_TYPE::DAY_TICKET_ADULT: { | ||||||
|         std::optional<QVector<ATBTariffProduct>> products = cfg->getTariffProductForProductId(permitType); |         std::optional<QVector<ATBTariffProduct>> products = cfg->getTariffProductForProductId(permitType); | ||||||
|         if (products) { |         if (products) { | ||||||
| @@ -92,9 +109,95 @@ int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg, PERMIT | |||||||
|             if (product.size() > 0) { |             if (product.size() > 0) { | ||||||
|                 ATBTariffProduct const &p = product[0]; |                 ATBTariffProduct const &p = product[0]; | ||||||
|                 return p.m_tariff_product_price; |                 return p.m_tariff_product_price; | ||||||
|  | #if 0 | ||||||
|  |                 // in case we do not have prepaid-option | ||||||
|  |                 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) << ")"; | ||||||
|  |                 } | ||||||
|  | #endif | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } break; |     } break; | ||||||
|  |     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(); | ||||||
|  |  | ||||||
|  |                 // qCritical() << __LINE__ << startTime.toString(Qt::ISODate); | ||||||
|  |                 // qCritical() << __LINE__ << endTime.toString(Qt::ISODate); | ||||||
|  |                 // qCritical() << __LINE__ << start.toString(Qt::ISODate); | ||||||
|  |  | ||||||
|  |                 if (start.time() >= startTime && start.time() < endTime) { | ||||||
|  |                     product_price = p.getProductPrice(); | ||||||
|  |                     if (productStart && productEnd) { | ||||||
|  |                         productStart->setTime(startTime); | ||||||
|  |                         productEnd->setTime(endTime); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return product_price; | ||||||
|  |         } 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; | ||||||
|  |         } | ||||||
|  |     } break; | ||||||
|  |     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) { | ||||||
|  |                     int pop_min_time = get_minimal_parkingtime(cfg); // in minutes | ||||||
|  |                     int pop_max_time = get_maximal_parkingtime(cfg); // in minutes | ||||||
|  |                     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; | ||||||
|     default: |     default: | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
| @@ -102,22 +205,24 @@ int CALCULATE_LIBRARY_API compute_product_price(Configuration const *cfg, PERMIT | |||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| int CALCULATE_LIBRARY_API get_maximal_parkingprice(Configuration *cfg, PERMIT_TYPE permitType) { | int CALCULATE_LIBRARY_API get_maximal_parkingprice(Configuration *cfg, | ||||||
|  |                                                    PERMIT_TYPE permitType, | ||||||
|  |                                                    int paymentOptionIndex) { | ||||||
|     int maxPrice = -1; |     int maxPrice = -1; | ||||||
|     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); |     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); | ||||||
|  |  | ||||||
|     switch(permitType) { |     switch(permitType) { | ||||||
|     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) |     case PERMIT_TYPE::SHORT_TERM_PARKING: { // e.g. szeged (customer_281) | ||||||
|         if (paymentMethodId == PaymentMethod::Progressive) { |         if (paymentMethodId == PaymentMethod::Progressive || paymentMethodId == PaymentMethod::Steps) { | ||||||
|             maxPrice = Utilities::getMaximalParkingPrice(cfg, paymentMethodId); |             maxPrice = Utilities::getMaximalParkingPrice(cfg, paymentMethodId); | ||||||
|         } else { // PaymentMethod::Linear -> e.g. szeged |         } else { // PaymentMethod::Linear -> e.g. szeged | ||||||
|             int const key = cfg->getPaymentOptions().pop_id; |             int const key = cfg->getPaymentOptions(paymentOptionIndex).pop_id; | ||||||
|             int const maxTime = cfg->getPaymentOptions().pop_max_time; // maxTime is given in minutes |             int const maxTime = cfg->getPaymentOptions(paymentOptionIndex).pop_max_time; // maxTime is given in minutes | ||||||
|             std::optional<QVector<ATBPaymentRate>> const &pv = cfg->getPaymentRateForKey(key); |             std::optional<QVector<ATBPaymentRate>> const &pv = cfg->getPaymentRateForKey(key); | ||||||
|             if (pv) { |             if (pv) { | ||||||
|                 QVector<ATBPaymentRate> const &paymentRate = pv.value(); |                 QVector<ATBPaymentRate> const &paymentRate = pv.value(); | ||||||
|                 if (paymentRate.size() > 0) { |                 if (paymentRate.size() > 0) { | ||||||
|                     int const price = paymentRate.at(0).pra_price; // price is given per hour |                     int const price = paymentRate.last().pra_price; // price is given per hour | ||||||
|                     maxPrice = qRound((maxTime * price) / 60.0f); |                     maxPrice = qRound((maxTime * price) / 60.0f); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -382,11 +487,15 @@ CalcState CALCULATE_LIBRARY_API compute_price_for_parking_ticket( | |||||||
|         bool prepaid) |         bool prepaid) | ||||||
| { | { | ||||||
|     CalcState calcState; |     CalcState calcState; | ||||||
|     double minMin = tariff->getPaymentOptions().pop_min_time; |  | ||||||
|     double maxMin = tariff->getPaymentOptions().pop_max_time; |     int paymentOptionIndex = tariff->getPaymentOptionIndex(start_parking_time); | ||||||
|  |  | ||||||
|  |     double minMin = tariff->getPaymentOptions(paymentOptionIndex).pop_min_time; | ||||||
|  |     double maxMin = tariff->getPaymentOptions(paymentOptionIndex).pop_max_time; | ||||||
|  |  | ||||||
|     // DEBUG |     // DEBUG | ||||||
|     qCritical() << "compute_price_for_parking_ticket() " << endl |     qCritical() << "compute_price_for_parking_ticket() " << endl | ||||||
|  |                 << "          paymentOptionIndex: " << paymentOptionIndex << endl | ||||||
|                 << "          start_parking_time: " << start_parking_time << endl |                 << "          start_parking_time: " << start_parking_time << endl | ||||||
|                 << "          netto_parking_time: " << netto_parking_time << endl |                 << "          netto_parking_time: " << netto_parking_time << endl | ||||||
|                 << "                      minMin: " << minMin << endl |                 << "                      minMin: " << minMin << endl | ||||||
| @@ -411,25 +520,46 @@ CalcState CALCULATE_LIBRARY_API compute_price_for_parking_ticket( | |||||||
|         return calcState.set(CalcState::State::SUCCESS); |         return calcState.set(CalcState::State::SUCCESS); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     double cost = -1; | ||||||
|     if (start_parking_time.isValid()) { |     if (start_parking_time.isValid()) { | ||||||
|         double cost = Calculator::GetInstance().GetCostFromDuration( |         if (tariff->getPaymentOptions(paymentOptionIndex).pop_payment_method_id == PaymentMethod::Steps) { | ||||||
|  |             // hier muesste man unterscheiden: uebertrag oder nicht? | ||||||
|  |             calcState = Calculator::GetInstance().isParkingAllowed(tariff, start_parking_time, | ||||||
|  |                                                                    netto_parking_time, paymentOptionIndex); | ||||||
|  |             if (calcState.getStatus() == CalcState::State::OUTSIDE_ALLOWED_PARKING_TIME) { | ||||||
|  |                 // qCritical() << "(" << __func__ << ":" << __LINE__ << ")" | ||||||
|  |                 //             << calcState.toString(); | ||||||
|  |                 return calcState; | ||||||
|  |             } | ||||||
|  |             cost = Calculator::GetInstance().GetCostFromDuration(tariff, start_parking_time, netto_parking_time, paymentOptionIndex); | ||||||
|  |             end_parking_time = start_parking_time.addSecs(netto_parking_time*60); | ||||||
|  |  | ||||||
|  |             // qCritical() << "(" << __func__ << ":" << __LINE__ << ")" | ||||||
|  |             //             << "end_parking_time" << end_parking_time.toString(Qt::ISODate); | ||||||
|  |  | ||||||
|  |         } else { | ||||||
|  |             cost = Calculator::GetInstance().GetCostFromDuration( | ||||||
|                         tariff, |                         tariff, | ||||||
|                         tariff->getPaymentOptions().pop_payment_method_id, |                         tariff->getPaymentOptions().pop_payment_method_id, | ||||||
|                         start_parking_time,         // starting time |                         start_parking_time,         // starting time | ||||||
|                         end_parking_time,           // return value: end time |                         end_parking_time,           // return value: end time | ||||||
|                         netto_parking_time,         // minutes, netto |                         netto_parking_time,         // minutes, netto | ||||||
|                         false, prepaid); |                         false, prepaid); | ||||||
|         double minCost = tariff->getPaymentOptions().pop_min_price; |         } | ||||||
|  |         double minCost = tariff->getPaymentOptions(paymentOptionIndex).pop_min_price; | ||||||
|         if (cost < minCost) { |         if (cost < minCost) { | ||||||
|             calcState.setDesc(QString("minCost=%1, cost=%2").arg(minCost, cost)); |             calcState.setDesc(QString("minCost=%1, cost=%2").arg(minCost, cost)); | ||||||
|             return calcState.set(CalcState::State::BELOW_MIN_PARKING_PRICE); |             return calcState.set(CalcState::State::BELOW_MIN_PARKING_PRICE); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // DEBUG |         // DEBUG | ||||||
|         qCritical() << "               -> calculated cost (price->netto) =  " << cost; |         qCritical() << "            end_parking_time: " << end_parking_time; | ||||||
|  |         qCritical() << "  -> calculated cost (netto): " << cost; | ||||||
|  |  | ||||||
|  |         price->brutto = price->vat = price->vat_percentage = 0; | ||||||
|         price->units = cost; |         price->units = cost; | ||||||
|         price->netto = cost; |         price->netto = cost; | ||||||
|  |  | ||||||
|     } else { |     } else { | ||||||
|         return calcState.set(CalcState::State::INVALID_START_DATE); |         return calcState.set(CalcState::State::INVALID_START_DATE); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ | |||||||
| #include "tariff_log.h" | #include "tariff_log.h" | ||||||
| #include "tariff_time_range.h" | #include "tariff_time_range.h" | ||||||
| #include "ticket.h" | #include "ticket.h" | ||||||
|  | #include "tariff_global_defines.h" | ||||||
|  |  | ||||||
| #include <sstream> | #include <sstream> | ||||||
| #include <algorithm> | #include <algorithm> | ||||||
| @@ -590,35 +591,334 @@ std::string Calculator::GetDurationFromCost(Configuration* cfg, | |||||||
|  |  | ||||||
| uint32_t Calculator::GetCostFromDuration(Configuration *cfg, | uint32_t Calculator::GetCostFromDuration(Configuration *cfg, | ||||||
|                                          QDateTime const &start, |                                          QDateTime const &start, | ||||||
|                                          quint64 timeStepInMinutes) const { |                                          quint64 timeStepInMinutes, | ||||||
|  |                                          int paymentOptionIndex) const { | ||||||
|     // for instance, a tariff as used in Schoenau, Koenigssee: only steps, no |     // for instance, a tariff as used in Schoenau, Koenigssee: only steps, no | ||||||
|     // special days, nonstop. |     // special days, nonstop. | ||||||
|     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); |     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); | ||||||
|     if (paymentMethodId == PaymentMethod::Steps) { |     if (paymentMethodId == PaymentMethod::Steps) { | ||||||
|         QDateTime const end = start.addSecs(timeStepInMinutes*60); |         QDateTime const end = start.addSecs(timeStepInMinutes*60); | ||||||
|         return GetCostFromDuration(cfg, start, end); |         return GetCostFromDuration(cfg, start, end, paymentOptionIndex); | ||||||
|     } |     } | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| uint32_t Calculator::GetCostFromDuration(Configuration * cfg, | uint32_t Calculator::GetCostFromDuration(Configuration * cfg, | ||||||
|                                          QDateTime const &start, |                                          QDateTime const &start, | ||||||
|                                          QDateTime const &end) const { |                                          QDateTime const &end, | ||||||
|  |                                          int paymentOptionIndex) const { | ||||||
|     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); |     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); | ||||||
|     if (paymentMethodId == PaymentMethod::Steps) { |     if (paymentMethodId == PaymentMethod::Steps) { | ||||||
|         int const timeStepInMinutes = start.secsTo(end) / 60; |         int const timeStepInMinutes = start.secsTo(end) / 60; | ||||||
|         return GetPriceForTimeStep(cfg, timeStepInMinutes); |         return GetPriceForTimeStep(cfg, timeStepInMinutes, paymentOptionIndex); | ||||||
|     } |     } | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | CalcState Calculator::isParkingAllowedForWeekDay(Configuration const *cfg, | ||||||
|  |                                                  QDateTime const &start, | ||||||
|  |                                                  int netto_parking_time, | ||||||
|  |                                                  int paymentOptionIndex) { | ||||||
|  |  | ||||||
| CalcState Calculator::isParkingAllowed(Configuration const *cfg, QDateTime const &start) { |     qCritical() << DBG_HEADER << "start" << start.toString(Qt::ISODate) | ||||||
|  |                 << "paymentOptionIndex" << paymentOptionIndex; | ||||||
|  |  | ||||||
|  |     QString errorStr = "UNKNOWN ERROR"; | ||||||
|  |  | ||||||
|  |     PaymentMethod const paymentMethodId = Utilities::getPaymentMethodId(cfg); | ||||||
|  |     if (paymentMethodId == PaymentMethod::Steps) { | ||||||
|  |         uint64_t const businessHours = cfg->getPaymentOptions(paymentOptionIndex).pop_business_hours; | ||||||
|  |  | ||||||
|  |         if (cfg->isDayIncluded(businessHours, start)) { | ||||||
|  |             if (businessHours == BusinessHours::NO_RESTRICTION_24_7) { | ||||||
|  |                 return CalcState(CalcState::State::SUCCESS, "PARKING_ALLOWED", | ||||||
|  |                          QTime(0, 0, 0), QTime(23, 59, 59)); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             int const weekdayId = start.date().dayOfWeek(); | ||||||
|  |  | ||||||
|  |             // qCritical() << DBG_HEADER | ||||||
|  |             //             << "weekdayId" << weekdayId | ||||||
|  |             //             << "count" << cfg->WeekDaysWorktime.count(weekdayId); | ||||||
|  |  | ||||||
|  |             if (cfg->WeekDaysWorktime.count(weekdayId) > 0) { | ||||||
|  |                 using WTIterator = std::multimap<int, ATBWeekDaysWorktime>::const_iterator; | ||||||
|  |                 std::pair<WTIterator, WTIterator> p = cfg->WeekDaysWorktime.equal_range(weekdayId); | ||||||
|  |  | ||||||
|  |                 for (WTIterator itr = p.first; itr != p.second; ++itr) { | ||||||
|  |                     QTime const &from = Utilities::WeekDaysWorkTimeFrom(itr); | ||||||
|  |                     QTime const &until = Utilities::WeekDaysWorkTimeUntil(itr); | ||||||
|  |  | ||||||
|  |                     qCritical() << DBG_HEADER | ||||||
|  |                                 << "CHECK IF PARKING IS ALLOWED IN TIME-RANGE (" | ||||||
|  |                                 << from.toString(Qt::ISODate) << "->" << until.toString(Qt::ISODate) << ") ..."; | ||||||
|  |  | ||||||
|  |                     QTime const &startTime = start.time(); | ||||||
|  |  | ||||||
|  |                     if (startTime >= from && startTime <= until) { | ||||||
|  |                         QDateTime const end = start.addSecs(netto_parking_time*60); | ||||||
|  |                         QTime const endTime = end.time(); | ||||||
|  |                         if (endTime <= until && start.date().dayOfWeek() == end.date().dayOfWeek()) { | ||||||
|  |                             qCritical() << DBG_HEADER; | ||||||
|  |                             return CalcState(CalcState::State::SUCCESS, "PARKING_ALLOWED", from, until); | ||||||
|  |                         } else { | ||||||
|  |                             errorStr = QString("%1 startTime not in range (%2 not in [%3, %4))") | ||||||
|  |                                                 .arg(__LINE__) | ||||||
|  |                                                 .arg(startTime.toString(Qt::ISODate)) | ||||||
|  |                                                 .arg(from.toString(Qt::ISODate)) | ||||||
|  |                                                 .arg(endTime.toString(Qt::ISODate)); | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|  |                         errorStr = QString("%1 startTime not in range (%2 not in [%3, %4))") | ||||||
|  |                                             .arg(__LINE__) | ||||||
|  |                                             .arg(startTime.toString(Qt::ISODate)) | ||||||
|  |                                             .arg(from.toString(Qt::ISODate)) | ||||||
|  |                                             .arg(until.toString(Qt::ISODate)); | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     int const pop_carry_over = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over; | ||||||
|  |  | ||||||
|  |                     // qCritical() << DBG_HEADER | ||||||
|  |                     //             << "paymentOptionIndex" << paymentOptionIndex | ||||||
|  |                     //             << "pop_carry_over" << pop_carry_over; | ||||||
|  |  | ||||||
|  |                     if (pop_carry_over == 1) { | ||||||
|  |                         // qCritical() << DBG_HEADER | ||||||
|  |                         //             << "NO. CHECK IF PARKING IS ALLOWED WITH CARRY-OVER ..."; | ||||||
|  |  | ||||||
|  |                         int const pop_carry_over_start_time_range = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over_start_time_range; | ||||||
|  |                         int const pop_carry_over_end_time_range = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over_end_time_range; | ||||||
|  |  | ||||||
|  |                         // qCritical() << DBG_HEADER | ||||||
|  |                         //             << "pop_carry_over_start_time_range" << pop_carry_over_start_time_range | ||||||
|  |                         //             << "pop_carry_over_end_time_range" << pop_carry_over_end_time_range; | ||||||
|  |  | ||||||
|  |                         if ((int)cfg->TimeRange.count(pop_carry_over_start_time_range) <= 0 && | ||||||
|  |                             (int)cfg->TimeRange.count(pop_carry_over_end_time_range) <= 0) { | ||||||
|  |  | ||||||
|  |                             qCritical() << DBG_HEADER << "PARKING_ALLOWED. startTime" << startTime.toString(Qt::ISODate); | ||||||
|  |                             return CalcState(CalcState::State::SUCCESS, "PARKING_ALLOWED", startTime); | ||||||
|  |  | ||||||
|  |                         } else | ||||||
|  |                         // search entry in time-range-field of tariff-file | ||||||
|  |                         if (cfg->TimeRange.count(pop_carry_over_start_time_range) == 1 && | ||||||
|  |                             cfg->TimeRange.count(pop_carry_over_end_time_range) == 1) { | ||||||
|  |  | ||||||
|  |                             ATBTimeRange s = cfg->TimeRange.find(pop_carry_over_start_time_range)->second; | ||||||
|  |                             ATBTimeRange e = cfg->TimeRange.find(pop_carry_over_end_time_range)->second; | ||||||
|  |  | ||||||
|  |                             // qCritical() << DBG_HEADER | ||||||
|  |                             //             << "startTime" << startTime.toString(Qt::ISODate); | ||||||
|  |  | ||||||
|  |                             if (startTime >= s.getTimeFrom() && startTime <= s.getTimeUntil()) { | ||||||
|  |  | ||||||
|  |                                 QDateTime sd = start; | ||||||
|  |                                 sd.setTime(s.getTimeUntil()); | ||||||
|  |  | ||||||
|  |                                 // qCritical() << DBG_HEADER << "jumpFrom" << sd.toString(Qt::ISODate); | ||||||
|  |  | ||||||
|  |                                 QDateTime ed = start.addDays(1); | ||||||
|  |                                 ed.setTime(e.getTimeFrom()); | ||||||
|  |  | ||||||
|  |                                 // qCritical() << DBG_HEADER << "to" << ed.toString(Qt::ISODate); | ||||||
|  |  | ||||||
|  |                                 int const jumpSecs = sd.secsTo(ed); | ||||||
|  |  | ||||||
|  |                                 // qCritical() << DBG_HEADER << "jumpSecs" << jumpSecs; | ||||||
|  |  | ||||||
|  |                                 QDateTime const end = start.addSecs(netto_parking_time*60 + jumpSecs); | ||||||
|  |  | ||||||
|  |                                 // qCritical() << DBG_HEADER << "new end" << end.toString(Qt::ISODate); | ||||||
|  |  | ||||||
|  |                                 if (end.time() <= e.getTimeUntil()) { | ||||||
|  |  | ||||||
|  |                                     qCritical() << DBG_HEADER | ||||||
|  |                                                 << "PARKING IS ALLOWED WITH CARRY-OVER (" | ||||||
|  |                                                 << start.toString(Qt::ISODate) << "->" << ed.toString(Qt::ISODate) << ")"; | ||||||
|  |  | ||||||
|  |                                     return CalcState(CalcState::State::SUCCESS, "PARKING_ALLOWED", | ||||||
|  |                                                      startTime, end.time()); | ||||||
|  |                                 } else { | ||||||
|  |                                     errorStr = QString("endTime %1 outside [%2, %3))") | ||||||
|  |                                                   .arg(end.toString(Qt::ISODate)) | ||||||
|  |                                                   .arg(sd.toString(Qt::ISODate)) | ||||||
|  |                                                   .arg(ed.toString(Qt::ISODate)); | ||||||
|  |                                 } | ||||||
|  |                             } else { | ||||||
|  |                                 errorStr = QString("startTime %1 outside [%2, %3))") | ||||||
|  |                                               .arg(startTime.toString(Qt::ISODate)) | ||||||
|  |                                               .arg(s.getTimeFrom().toString(Qt::ISODate)) | ||||||
|  |                                               .arg(s.getTimeUntil().toString(Qt::ISODate)); | ||||||
|  |                             } | ||||||
|  |                         } else { | ||||||
|  |                            errorStr = "no carry-over limits configured"; | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|  |                         errorStr = "no carry-over configured"; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 errorStr = QString("no weekday configured for day-id %1").arg(weekdayId); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         } else { | ||||||
|  |             errorStr = QString("start %1 not contained in business hours %2") | ||||||
|  |                             .arg(start.toString(Qt::ISODate)) | ||||||
|  |                             .arg(businessHours); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     qCritical() << DBG_HEADER << "errorStr" << errorStr; | ||||||
|  |  | ||||||
|  |     return CalcState(CalcState::State::OUTSIDE_ALLOWED_PARKING_TIME, errorStr, | ||||||
|  |                      QTime(), QTime()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CalcState Calculator::isParkingAllowedForSpecialDay(Configuration const *cfg, | ||||||
|  |                                                     QDateTime const &start, | ||||||
|  |                                                     int netto_parking_time, | ||||||
|  |                                                     int paymentOptionIndex) { | ||||||
|  |     QString errorStr = "UNKNOWN ERROR"; | ||||||
|  |  | ||||||
|  |     qCritical() << DBG_HEADER << "start" << start.toString(Qt::ISODate) | ||||||
|  |                 << "paymentOptionIndex" << paymentOptionIndex; | ||||||
|  |  | ||||||
|  |     PaymentMethod const paymentMethodId = Utilities::getPaymentMethodId(cfg); | ||||||
|  |     if (paymentMethodId == PaymentMethod::Steps) { | ||||||
|  |         //uint64_t const businessHours = cfg->getPaymentOptions(paymentOptionIndex).pop_business_hours; | ||||||
|  |         int const specialDayId = cfg->specialDayId(start); | ||||||
|  |         if ((specialDayId > 0) && (cfg->SpecialDaysWorktime.count(specialDayId) > 0)) { | ||||||
|  |             using SDIterator = Configuration::SpecialDaysWorktimeType::const_iterator; | ||||||
|  |             std::pair<SDIterator, SDIterator> p = cfg->SpecialDaysWorktime.equal_range(specialDayId); | ||||||
|  |  | ||||||
|  |             for (SDIterator it = p.first; it != p.second; ++it) { | ||||||
|  |                 QTime const &from = Utilities::SpecialDaysWorkTimeFrom(it); | ||||||
|  |                 QTime const &until = Utilities::SpecialDaysWorkTimeUntil(it); | ||||||
|  |  | ||||||
|  |                 qCritical() << DBG_HEADER | ||||||
|  |                             << "CHECK IF PARKING IS ALLOWED IN TIME-RANGE (" | ||||||
|  |                             << from.toString(Qt::ISODate) << "->" << until.toString(Qt::ISODate) << ") ..."; | ||||||
|  |  | ||||||
|  |                 QTime const &startTime = start.time(); | ||||||
|  |  | ||||||
|  |                 if (startTime >= from && startTime <= until) { | ||||||
|  |                     QTime const endTime = start.addSecs(netto_parking_time*60).time(); | ||||||
|  |                     if (endTime <= until) { | ||||||
|  |                         return CalcState(CalcState::State::SUCCESS, "PARKING_ALLOWED", from, until); | ||||||
|  |                     } else { | ||||||
|  |                         errorStr = QString("%1 startTime not in range (%2 not in [%3, %4))") | ||||||
|  |                                             .arg(__LINE__) | ||||||
|  |                                             .arg(startTime.toString(Qt::ISODate)) | ||||||
|  |                                             .arg(from.toString(Qt::ISODate)) | ||||||
|  |                                             .arg(endTime.toString(Qt::ISODate)); | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     errorStr = QString("%1 startTime not in range (%2 not in [%3, %4))") | ||||||
|  |                                         .arg(__LINE__) | ||||||
|  |                                         .arg(startTime.toString(Qt::ISODate)) | ||||||
|  |                                         .arg(from.toString(Qt::ISODate)) | ||||||
|  |                                         .arg(until.toString(Qt::ISODate)); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 int const pop_carry_over = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over; | ||||||
|  |                 if (pop_carry_over == 1) { | ||||||
|  |                     qCritical() << DBG_HEADER << "NO. CHECK IF PARKING IS ALLOWED WITH CARRY-OVER ..."; | ||||||
|  |  | ||||||
|  |                     int const pop_carry_over_start_time_range = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over_start_time_range; | ||||||
|  |                     int const pop_carry_over_end_time_range = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over_end_time_range; | ||||||
|  |  | ||||||
|  |                     if (cfg->TimeRange.count(pop_carry_over_start_time_range) == 1 && | ||||||
|  |                         cfg->TimeRange.count(pop_carry_over_end_time_range) == 1) { | ||||||
|  |  | ||||||
|  |                         ATBTimeRange s = cfg->TimeRange.find(pop_carry_over_start_time_range)->second; | ||||||
|  |                         ATBTimeRange e = cfg->TimeRange.find(pop_carry_over_end_time_range)->second; | ||||||
|  |  | ||||||
|  |                         if (startTime >= s.getTimeFrom() && startTime <= s.getTimeUntil()) { | ||||||
|  |                             QDateTime sd = start; | ||||||
|  |                             sd.setTime(s.getTimeUntil()); | ||||||
|  |  | ||||||
|  |                             QDateTime ed = start.addDays(1); | ||||||
|  |                             ed.setTime(e.getTimeFrom()); | ||||||
|  |  | ||||||
|  |                             int const jumpSecs = sd.secsTo(ed); | ||||||
|  |                             QDateTime const end = start.addSecs(netto_parking_time*60 + jumpSecs); | ||||||
|  |                             if (end.time() <= e.getTimeUntil()) { | ||||||
|  |  | ||||||
|  |                                 ed.setTime(e.getTimeUntil()); // for printing | ||||||
|  |  | ||||||
|  |                                 qCritical() << DBG_HEADER | ||||||
|  |                                             << "PARKING IS ALLOWED WITH CARRY-OVER (" | ||||||
|  |                                             << start.toString(Qt::ISODate) << "->" << ed.toString(Qt::ISODate) << ")"; | ||||||
|  |  | ||||||
|  |                                 return CalcState(CalcState::State::SUCCESS, "PARKING_ALLOWED", | ||||||
|  |                                                  startTime, end.time()); | ||||||
|  |                             } else { | ||||||
|  |                                 ed.setTime(e.getTimeUntil()); // for printing | ||||||
|  |  | ||||||
|  |                                 errorStr = QString("endTime %1 exceeds [%2, %3))") | ||||||
|  |                                               .arg(end.toString(Qt::ISODate)) | ||||||
|  |                                               .arg(sd.toString(Qt::ISODate)) | ||||||
|  |                                               .arg(ed.toString(Qt::ISODate)); | ||||||
|  |                             } | ||||||
|  |                         } else { | ||||||
|  |                             errorStr = QString("startTime %1 exceeds [%2, %3))") | ||||||
|  |                                           .arg(startTime.toString(Qt::ISODate)) | ||||||
|  |                                           .arg(s.getTimeFrom().toString(Qt::ISODate)) | ||||||
|  |                                           .arg(s.getTimeUntil().toString(Qt::ISODate)); | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|  |                        errorStr = "no carry-over limits configured"; | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     errorStr = "no carry-over configured"; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return CalcState(CalcState::State::OUTSIDE_ALLOWED_PARKING_TIME, errorStr, | ||||||
|  |                      QTime(), QTime()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | CalcState Calculator::isParkingAllowed(Configuration const *cfg, | ||||||
|  |                                        QDateTime const &start, | ||||||
|  |                                        int netto_parking_time, | ||||||
|  |                                        int paymentOptionIndex) { | ||||||
|  |  | ||||||
|  |     qCritical() << DBG_HEADER << "CHECK IF PARKING IS ALLOWED AT" | ||||||
|  |                 << start.toString(Qt::ISODate) << "..."; | ||||||
|  |  | ||||||
|  |     CalcState cs; | ||||||
|  |  | ||||||
|  |     if ((cs = isParkingAllowedForWeekDay(cfg, start, netto_parking_time, paymentOptionIndex))) { | ||||||
|  |         return cs; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     qCritical() << DBG_HEADER << QString(cs); | ||||||
|  |  | ||||||
|  |     if ((cs = isParkingAllowedForSpecialDay(cfg, start, netto_parking_time, paymentOptionIndex))) { | ||||||
|  |         return cs; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     qCritical() << DBG_HEADER << QString(cs); | ||||||
|  |  | ||||||
|  |     return CalcState(CalcState::State::OUTSIDE_ALLOWED_PARKING_TIME, "UNKNOWN ERROR", | ||||||
|  |                      QTime(), QTime()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | CalcState Calculator::isParkingAllowed(Configuration const *cfg, | ||||||
|  |                                        QDateTime const &start) { | ||||||
|     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); |     static const PaymentMethod paymentMethodId = Utilities::getPaymentMethodId(cfg); | ||||||
|  |  | ||||||
|     if (paymentMethodId == PaymentMethod::Steps) { |     if (paymentMethodId == PaymentMethod::Steps) { | ||||||
|         int const weekdayId = start.date().dayOfWeek(); |         int const weekdayId = start.date().dayOfWeek(); | ||||||
|         BusinessHours businessHours = Utilities::getBusinessHours(cfg, paymentMethodId); |         BusinessHours businessHours = Utilities::getBusinessHours(cfg, paymentMethodId); | ||||||
|  |         if (businessHours == BusinessHours::NoRestriction_24_7) { | ||||||
|  |             return CalcState(CalcState::State::SUCCESS, "PARKING_ALLOWED", | ||||||
|  |                              QTime(0, 0, 0), QTime(23, 59, 59)); | ||||||
|  |         } else | ||||||
|         if (businessHours == BusinessHours::OnlyWeekDays) { |         if (businessHours == BusinessHours::OnlyWeekDays) { | ||||||
|             if (weekdayId != (int)Qt::Saturday && weekdayId != (int)Qt::Sunday) { // e.g. Neuhauser, Linsinger Maschinenbau (741) |             if (weekdayId != (int)Qt::Saturday && weekdayId != (int)Qt::Sunday) { // e.g. Neuhauser, Linsinger Maschinenbau (741) | ||||||
|                 if (cfg->WeekDaysWorktime.count(weekdayId) > 0) { |                 if (cfg->WeekDaysWorktime.count(weekdayId) > 0) { | ||||||
| @@ -1569,24 +1869,28 @@ QList<int> Calculator::GetPriceSteps(Configuration * /*cfg*/) const { | |||||||
|     return QList<int>(); |     return QList<int>(); | ||||||
| } | } | ||||||
|  |  | ||||||
| QList<int> Calculator::GetTimeSteps(Configuration *cfg) const { | QList<int> Calculator::GetTimeSteps(Configuration *cfg, int paymentOptionIndex) const { | ||||||
|     if (m_timeSteps.size() > 0) { |     qCritical() << "(" << __func__ << ":" << __LINE__ << ")" << "paymentOptionIndex:" << paymentOptionIndex; | ||||||
|  |  | ||||||
|  |     if (m_timeSteps.size() > paymentOptionIndex) { | ||||||
|         //qCritical() << __PRETTY_FUNCTION__ << "timeSteps:" << m_timeSteps; |         //qCritical() << __PRETTY_FUNCTION__ << "timeSteps:" << m_timeSteps; | ||||||
|         return m_timeSteps; |         return m_timeSteps[paymentOptionIndex]; | ||||||
|  |     } else { | ||||||
|  |         m_timeSteps.push_back(QList<int>()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     QDateTime start = QDateTime::currentDateTime(); |     QDateTime start = QDateTime::currentDateTime(); | ||||||
|     start.setTime(QTime(start.time().hour(), start.time().minute(), 0)); |     start.setTime(QTime(start.time().hour(), start.time().minute(), 0)); | ||||||
|  |  | ||||||
|     int const pop_id = cfg->getPaymentOptions().pop_id; |     int const pop_id = cfg->getPaymentOptions(paymentOptionIndex).pop_id; | ||||||
|     int const pop_carry_over = cfg->getPaymentOptions().pop_carry_over; |     int const pop_carry_over = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over; | ||||||
|     int const pop_time_step_config = cfg->getPaymentOptions().pop_time_step_config; |     int const pop_time_step_config = cfg->getPaymentOptions(paymentOptionIndex).pop_time_step_config; | ||||||
|  |  | ||||||
|     static PaymentMethod const paymentMethodId = Utilities::getPaymentMethodId(cfg); |     static PaymentMethod const paymentMethodId = Utilities::getPaymentMethodId(cfg); | ||||||
|  |  | ||||||
|     qCritical() << __func__ << ":" << __LINE__ << "       start parking time:" << start.toString(Qt::ISODate); |     qCritical() << "(" << __func__ << ":" << __LINE__ << ")        start parking time:" << start.toString(Qt::ISODate); | ||||||
|     qCritical() << __func__ << ":" << __LINE__ << "        payment option id:" << pop_id; |     qCritical() << "(" << __func__ << ":" << __LINE__ << ")         payment option id:" << pop_id; | ||||||
|     qCritical() << __func__ << ":" << __LINE__ << "payment option carry over:" << pop_carry_over; |     qCritical() << "(" << __func__ << ":" << __LINE__ << ") payment option carry over:" << pop_carry_over; | ||||||
|  |  | ||||||
|     if (pop_time_step_config == (int)ATBTimeStepConfig::TimeStepConfig::DYNAMIC) { |     if (pop_time_step_config == (int)ATBTimeStepConfig::TimeStepConfig::DYNAMIC) { | ||||||
|         //qCritical() << __PRETTY_FUNCTION__ << "payment option time step config:" << "TimeStepConfig::DYNAMIC"; |         //qCritical() << __PRETTY_FUNCTION__ << "payment option time step config:" << "TimeStepConfig::DYNAMIC"; | ||||||
| @@ -1595,13 +1899,13 @@ QList<int> Calculator::GetTimeSteps(Configuration *cfg) const { | |||||||
|             std::size_t const s = cfg->TimeRange.size(); |             std::size_t const s = cfg->TimeRange.size(); | ||||||
|             for (std::size_t id = 1; id <= s; ++id) { |             for (std::size_t id = 1; id <= s; ++id) { | ||||||
|                 int const step = Utilities::getTimeRangeStep(cfg, id, paymentMethodId); |                 int const step = Utilities::getTimeRangeStep(cfg, id, paymentMethodId); | ||||||
|                 m_timeSteps.append(step); |                 m_timeSteps[paymentOptionIndex].append(step); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             uint16_t timeStepCompensation = 0; |             uint16_t timeStepCompensation = 0; | ||||||
|  |  | ||||||
|             if (pop_carry_over) { |             if (pop_carry_over) { | ||||||
|                 int const pop_carry_over_time_range_id = cfg->getPaymentOptions().pop_carry_over_time_range_id; |                 int const pop_carry_over_time_range_id = cfg->getPaymentOptions(paymentOptionIndex).pop_carry_over_time_range_id; | ||||||
|                 QTime const carryOverTimeRangeFrom = cfg->TimeRange.find(pop_carry_over_time_range_id)->second.time_range_from; |                 QTime const carryOverTimeRangeFrom = cfg->TimeRange.find(pop_carry_over_time_range_id)->second.time_range_from; | ||||||
|                 QTime const carryOverTimeRangeTo = cfg->TimeRange.find(pop_carry_over_time_range_id)->second.time_range_to; |                 QTime const carryOverTimeRangeTo = cfg->TimeRange.find(pop_carry_over_time_range_id)->second.time_range_to; | ||||||
|  |  | ||||||
| @@ -1625,19 +1929,19 @@ QList<int> Calculator::GetTimeSteps(Configuration *cfg) const { | |||||||
|                                                 .arg(timeStep).arg(duration.pun_duration_max); |                                                 .arg(timeStep).arg(duration.pun_duration_max); | ||||||
|                                         break; |                                         break; | ||||||
|                                     } |                                     } | ||||||
|                                     qCritical() << __PRETTY_FUNCTION__ << "configured minimal parking time:" << cfg->getPaymentOptions().pop_min_time; |                                     qCritical() << "(" << __func__ << ":" << __LINE__ << ") configured minimal parking time:" << cfg->getPaymentOptions(paymentOptionIndex).pop_min_time; | ||||||
|  |  | ||||||
|                                     // set dynamic minimal parking time |                                     // set dynamic minimal parking time | ||||||
|                                     cfg->getPaymentOptions().pop_min_time = timeStep; |                                     cfg->getPaymentOptions(paymentOptionIndex).pop_min_time = timeStep; | ||||||
|  |  | ||||||
|                                     qCritical() << __PRETTY_FUNCTION__ << "  computed minimal parking time:" << cfg->getPaymentOptions().pop_min_time; |                                     qCritical() << "(" << __func__ << ":" << __LINE__ << ") computed minimal parking time:" << cfg->getPaymentOptions(paymentOptionIndex).pop_min_time; | ||||||
|  |  | ||||||
|                                     duration.pun_duration = timeStep; |                                     duration.pun_duration = timeStep; | ||||||
|                                     timeStepCompensation = duration.pun_duration_max - duration.pun_duration; |                                     timeStepCompensation = duration.pun_duration_max - duration.pun_duration; | ||||||
|                                     m_timeSteps << duration.pun_duration; |                                     m_timeSteps[paymentOptionIndex] << duration.pun_duration; | ||||||
|                                 } else { |                                 } else { | ||||||
|                                     duration.pun_duration = duration.pun_duration_max - timeStepCompensation; |                                     duration.pun_duration = duration.pun_duration_max - timeStepCompensation; | ||||||
|                                     m_timeSteps << duration.pun_duration;; |                                     m_timeSteps[paymentOptionIndex] << duration.pun_duration;; | ||||||
|                                 } |                                 } | ||||||
|  |  | ||||||
|                                 cfg->Duration.erase(search); |                                 cfg->Duration.erase(search); | ||||||
| @@ -1658,24 +1962,31 @@ QList<int> Calculator::GetTimeSteps(Configuration *cfg) const { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         qCritical() << __PRETTY_FUNCTION__ << "payment option time step config:" << "TimeStepConfig::STATIC"; |         qCritical() << "(" << __func__ << ":" << __LINE__ << ") payment option time step config:" << "TimeStepConfig::STATIC"; | ||||||
|  |  | ||||||
|         for (auto[itr, rangeEnd] = cfg->PaymentRate.equal_range(pop_id); itr != rangeEnd; ++itr) |         for (auto[itr, rangeEnd] = cfg->PaymentRate.equal_range(pop_id); itr != rangeEnd; ++itr) | ||||||
|         { |         { | ||||||
|             int const durationId = itr->second.pra_payment_unit_id; |             int const durationId = itr->second.pra_payment_unit_id; | ||||||
|             int const durationUnit = cfg->Duration.find(durationId)->second.pun_duration; |             int const durationUnit = cfg->Duration.find(durationId)->second.pun_duration; | ||||||
|             m_timeSteps << durationUnit; |             int size = m_timeSteps.size(); | ||||||
|  |  | ||||||
|  |             while (size <= paymentOptionIndex) { | ||||||
|  |                 m_timeSteps.push_back(QList<int>()); | ||||||
|  |                 size = m_timeSteps.size(); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             m_timeSteps[paymentOptionIndex] << durationUnit; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     qCritical() << __PRETTY_FUNCTION__ << "NEW timeSteps:" << m_timeSteps; |     qCritical() << "(" << __func__ << ":" << __LINE__ << ")" << "NEW timeSteps:" << m_timeSteps; | ||||||
|  |  | ||||||
|     return m_timeSteps; |     return m_timeSteps[paymentOptionIndex]; | ||||||
| } | } | ||||||
|  |  | ||||||
| uint32_t Calculator::GetPriceForTimeStep(Configuration *cfg, int timeStep) const { | uint32_t Calculator::GetPriceForTimeStep(Configuration *cfg, int timeStep, int paymentOptionIndex) const { | ||||||
|  |  | ||||||
|     int const pop_id = cfg->getPaymentOptions().pop_id; |     int const pop_id = cfg->getPaymentOptions(paymentOptionIndex).pop_id; | ||||||
|  |  | ||||||
|     for (auto[itr, rangeEnd] = cfg->PaymentRate.equal_range(pop_id); itr != rangeEnd; ++itr) |     for (auto[itr, rangeEnd] = cfg->PaymentRate.equal_range(pop_id); itr != rangeEnd; ++itr) | ||||||
|     { |     { | ||||||
| @@ -1736,6 +2047,12 @@ Calculator::GetDailyTicketPrice(Configuration* cfg, | |||||||
|         if (dailyTickets) { |         if (dailyTickets) { | ||||||
|             QVector<ATBDailyTicket> const tickets = dailyTickets.value(); |             QVector<ATBDailyTicket> const tickets = dailyTickets.value(); | ||||||
|             switch (permitType) { |             switch (permitType) { | ||||||
|  |                 case PERMIT_TYPE::TWENTY_FOUR_HOURS_TICKET: { | ||||||
|  |                     // TODO | ||||||
|  |                 } break; | ||||||
|  |                 case PERMIT_TYPE::FOOD_STAMP: { | ||||||
|  |                     // TODO | ||||||
|  |                 } break; | ||||||
|                 case PERMIT_TYPE::DAY_TICKET_ADULT: { |                 case PERMIT_TYPE::DAY_TICKET_ADULT: { | ||||||
|                     std::optional<ATBCustomer> c = cfg->getCustomerForType(ATBCustomer::CustomerType::ADULT); |                     std::optional<ATBCustomer> c = cfg->getCustomerForType(ATBCustomer::CustomerType::ADULT); | ||||||
|                     if (c) { |                     if (c) { | ||||||
|   | |||||||
| @@ -1,11 +1,15 @@ | |||||||
| #include "configuration.h" | #include "configuration.h" | ||||||
|  | #include "utilities.h" | ||||||
| #include "tariff_timebase.h" | #include "tariff_timebase.h" | ||||||
| #include "time_range_header.h" | #include "time_range_header.h" | ||||||
| #include "tariff_timestep_config.h" | #include "tariff_timestep_config.h" | ||||||
| #include "tariff_permit_type.h" | #include "tariff_permit_type.h" | ||||||
|  | #include "tariff_business_hours.h" | ||||||
|  | #include "tariff_global_defines.h" | ||||||
|  |  | ||||||
| #include <QString> | #include <QString> | ||||||
| #include <QDebug> | #include <QDebug> | ||||||
|  | #include <QRegularExpression> | ||||||
|  |  | ||||||
| /// <inheritdoc/> | /// <inheritdoc/> | ||||||
| MemberType Configuration::IdentifyJsonMember(const char* member_name) | MemberType Configuration::IdentifyJsonMember(const char* member_name) | ||||||
| @@ -413,12 +417,80 @@ bool Configuration::ParseJson(Configuration* cfg, const char* json) | |||||||
|                             this->currentPaymentOptions.last().pop_carry_over = k->value.GetInt(); |                             this->currentPaymentOptions.last().pop_carry_over = k->value.GetInt(); | ||||||
|                         } else if (strcmp(inner_obj_name, "pop_carry_over_time_range_id") == 0) { |                         } else if (strcmp(inner_obj_name, "pop_carry_over_time_range_id") == 0) { | ||||||
|                             this->currentPaymentOptions.last().pop_carry_over_time_range_id = k->value.GetInt(); |                             this->currentPaymentOptions.last().pop_carry_over_time_range_id = k->value.GetInt(); | ||||||
|  |                         } else if (strcmp(inner_obj_name, "pop_carry_over_start_time_range") == 0) { | ||||||
|  |                             this->currentPaymentOptions.last().pop_carry_over_start_time_range = k->value.GetInt(); | ||||||
|  |                         } else if (strcmp(inner_obj_name, "pop_carry_over_end_time_range") == 0) { | ||||||
|  |                             this->currentPaymentOptions.last().pop_carry_over_end_time_range = k->value.GetInt(); | ||||||
|                         } else if (strcmp(inner_obj_name, "pop_daily_card_price") == 0) { |                         } else if (strcmp(inner_obj_name, "pop_daily_card_price") == 0) { | ||||||
|                             this->currentPaymentOptions.last().pop_daily_card_price = k->value.GetInt(); |                             this->currentPaymentOptions.last().pop_daily_card_price = k->value.GetInt(); | ||||||
|                         } else if (strcmp(inner_obj_name, "pop_business_hours") == 0) { |                         } else if (strcmp(inner_obj_name, "pop_business_hours") == 0) { | ||||||
|                             this->currentPaymentOptions.last().pop_business_hours = k->value.GetInt(); |                             if (k->value.IsInt()) { | ||||||
|  |                                 int const v = k->value.GetInt(); | ||||||
|  |                                 this->currentPaymentOptions.last().pop_business_hours = v; | ||||||
|  |                             } else | ||||||
|  |                             if (k->value.IsString()) { | ||||||
|  |                                 bool ok; | ||||||
|  |                                 uint64_t const v = QString::fromStdString(k->value.GetString()).toLongLong(&ok); | ||||||
|  |                                 if (ok) { | ||||||
|  |                                     this->currentPaymentOptions.last().pop_business_hours = v; | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|                         } else if (strcmp(inner_obj_name, "pop_time_step_config") == 0) { |                         } else if (strcmp(inner_obj_name, "pop_time_step_config") == 0) { | ||||||
|                             this->currentPaymentOptions.last().pop_time_step_config = k->value.GetInt(); |                             this->currentPaymentOptions.last().pop_time_step_config = k->value.GetInt(); | ||||||
|  |                         } else if ((strcmp(inner_obj_name, "pop_min_date_time") == 0) | ||||||
|  |                                 || (strcmp(inner_obj_name, "pop_max_date_time") == 0)) { | ||||||
|  |                             if (k->value.IsString()) { // -w0dFriT16:20:00 or +w0dMonT00:00:00 | ||||||
|  |                                 static const QRegularExpression re(R"(([+-])w([0-9]+)d([A-Za-z]{3})T([0-9]{2}:[0-9]{2}:[0-9]{2}))"); | ||||||
|  |                                 QString const &s = QString::fromStdString(k->value.GetString()); | ||||||
|  |                                 QRegularExpressionMatch match = re.match(s); | ||||||
|  |                                 if (match.hasMatch()) { | ||||||
|  |                                     ATBPaymentOption::ATBMaxDateTime dt; | ||||||
|  |                                     int lastCaptured = match.lastCapturedIndex(); | ||||||
|  |                                     if (lastCaptured == 4) { | ||||||
|  |                                         dt.direction = (match.captured(1) == "-") ? -1 : +1; | ||||||
|  |  | ||||||
|  |                                         bool ok; | ||||||
|  |                                         uint8_t week = match.captured(2).toUInt(&ok); | ||||||
|  |                                         if (ok) { | ||||||
|  |                                             dt.week = week; | ||||||
|  |                                         } | ||||||
|  |  | ||||||
|  |                                         QString const &day = match.captured(3); | ||||||
|  |                                         if (day.compare("Mon", Qt::CaseInsensitive) == 0) { | ||||||
|  |                                             dt.day = (int)Qt::Monday; | ||||||
|  |                                         } else if (day.compare("Tue", Qt::CaseInsensitive) == 0) { | ||||||
|  |                                             dt.day = (int)Qt::Tuesday; | ||||||
|  |                                         } else if (day.compare("Wed", Qt::CaseInsensitive) == 0) { | ||||||
|  |                                             dt.day = (int)Qt::Wednesday; | ||||||
|  |                                         } else if (day.compare("Thu", Qt::CaseInsensitive) == 0) { | ||||||
|  |                                             dt.day = (int)Qt::Thursday; | ||||||
|  |                                         } else if (day.compare("Fri", Qt::CaseInsensitive) == 0) { | ||||||
|  |                                             dt.day = (int)Qt::Friday; | ||||||
|  |                                         } else if (day.compare("Sat", Qt::CaseInsensitive) == 0) { | ||||||
|  |                                             dt.day = (int)Qt::Saturday; | ||||||
|  |                                         } else if (day.compare("Sun", Qt::CaseInsensitive) == 0) { | ||||||
|  |                                             dt.day = (int)Qt::Sunday; | ||||||
|  |                                         } | ||||||
|  |  | ||||||
|  |                                         QTime t = QTime::fromString(match.captured(4), Qt::ISODate); | ||||||
|  |                                         if (t.isValid()) { | ||||||
|  |                                             dt.time = t; | ||||||
|  |                                         } | ||||||
|  |                                     } | ||||||
|  |  | ||||||
|  |                                     if (strcmp(inner_obj_name, "pop_min_date_time") == 0) { | ||||||
|  |                                         this->currentPaymentOptions.last().pop_min_date_time = dt; | ||||||
|  |                                     } else | ||||||
|  |                                     if (strcmp(inner_obj_name, "pop_max_date_time") == 0) { | ||||||
|  |                                         this->currentPaymentOptions.last().pop_max_date_time = dt; | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                         } else if (strcmp(inner_obj_name, "pop_carry_over_target") == 0) { | ||||||
|  |                             if (k->value.IsBool()) { | ||||||
|  |                                 bool const v = k->value.GetBool(); | ||||||
|  |                                 this->currentPaymentOptions.last().pop_carry_over_target = v; | ||||||
|  |                             } | ||||||
|                         } |                         } | ||||||
|                         break; |                         break; | ||||||
| 					case MemberType::DurationType: | 					case MemberType::DurationType: | ||||||
| @@ -453,6 +525,7 @@ bool Configuration::ParseJson(Configuration* cfg, const char* json) | |||||||
| 						else if (strcmp(inner_obj_name, "ped_date_start") == 0) SpecialDays.ped_date_start = k->value.GetString(); | 						else if (strcmp(inner_obj_name, "ped_date_start") == 0) SpecialDays.ped_date_start = k->value.GetString(); | ||||||
| 						else if (strcmp(inner_obj_name, "ped_date_end") == 0) SpecialDays.ped_date_end = k->value.GetString(); | 						else if (strcmp(inner_obj_name, "ped_date_end") == 0) SpecialDays.ped_date_end = k->value.GetString(); | ||||||
| 						else if (strcmp(inner_obj_name, "ped_period_special_day_id") == 0) SpecialDays.ped_period_special_day_id = k->value.GetInt(); | 						else if (strcmp(inner_obj_name, "ped_period_special_day_id") == 0) SpecialDays.ped_period_special_day_id = k->value.GetInt(); | ||||||
|  |                         else if (strcmp(inner_obj_name, "ped_payment_option_id") == 0) SpecialDays.ped_payment_option_id = k->value.GetInt(); | ||||||
|                         else if (strcmp(inner_obj_name, "ped_year") == 0) SpecialDays.ped_year = k->value.GetInt(); |                         else if (strcmp(inner_obj_name, "ped_year") == 0) SpecialDays.ped_year = k->value.GetInt(); | ||||||
| 						break; | 						break; | ||||||
| 					case MemberType::PeriodYearType: | 					case MemberType::PeriodYearType: | ||||||
| @@ -542,15 +615,135 @@ bool Configuration::ParseJson(Configuration* cfg, const char* json) | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | int Configuration::getPaymentOptionIndexIfSpecialDay(QDateTime const &dt) const { | ||||||
| ATBPaymentOption const &Configuration::getPaymentOptions() const { |     if (isSpecialDay(dt)) { | ||||||
|     Q_ASSERT(!this->currentPaymentOptions.isEmpty()); |         int const numOptions = getAllPaymentOptions().size(); | ||||||
|     return this->currentPaymentOptions.at(0); |         for (int opt=0; opt < numOptions; ++opt) { | ||||||
|  |             uint64_t const pop_business_hours = getPaymentOptions(opt).pop_business_hours; | ||||||
|  |             if ((pop_business_hours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |                 return opt; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return -1; | ||||||
| } | } | ||||||
|  |  | ||||||
| ATBPaymentOption &Configuration::getPaymentOptions() { | int Configuration::getPaymentOptionIndex(QDateTime const &dt) const { | ||||||
|  |     int const numOptions = getAllPaymentOptions().size(); | ||||||
|  |  | ||||||
|  |     // special days are handled before usual week days | ||||||
|  |     int const sid = specialDayId(dt); | ||||||
|  |     if (sid > 0) { | ||||||
|  |         ATBSpecialDays const sd = specialDay(dt); | ||||||
|  |         if (sd.ped_id != 0) { | ||||||
|  |             for (int opt=0; opt < numOptions; ++opt) { | ||||||
|  |                 uint64_t const pop_id = getPaymentOptions(opt).pop_id; | ||||||
|  |                 if (pop_id == (uint64_t)sd.ped_payment_option_id) { | ||||||
|  |                     return opt; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     for (int opt=0; opt < numOptions; ++opt) { | ||||||
|  |         uint64_t const pop_business_hours = getPaymentOptions(opt).pop_business_hours; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         uint64_t p = 0; | ||||||
|  |         int const dayOfWeek = dt.date().dayOfWeek(); | ||||||
|  |  | ||||||
|  |         switch (dayOfWeek) { | ||||||
|  |         case (int)Qt::Monday: { | ||||||
|  |             p = BusinessHours::MON; | ||||||
|  |         } break; | ||||||
|  |         case (int)Qt::Tuesday: { | ||||||
|  |             p = BusinessHours::TUE; | ||||||
|  |         } break; | ||||||
|  |         case (int)Qt::Wednesday: { | ||||||
|  |             p = BusinessHours::WED; | ||||||
|  |         } break; | ||||||
|  |         case (int)Qt::Thursday: { | ||||||
|  |             p = BusinessHours::THU; | ||||||
|  |         } break; | ||||||
|  |         case (int)Qt::Friday: { | ||||||
|  |             p = BusinessHours::FRI; | ||||||
|  |         } break; | ||||||
|  |         case (int)Qt::Saturday: { | ||||||
|  |             p = BusinessHours::SAT; | ||||||
|  |         } break; | ||||||
|  |         case (int)Qt::Sunday: { | ||||||
|  |             p = BusinessHours::SUN; | ||||||
|  |             qCritical() << DBG_HEADER << Utilities::dumpBusinessHours(pop_business_hours) << pop_business_hours << p; | ||||||
|  |         } break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if ((pop_business_hours & p) == p) { | ||||||
|  |             return opt; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     qCritical() << DBG_HEADER << "ERROR: DEFAULT VALUE OF '0' RETURNED AS STATIC VALUE"; | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ATBSpecialDays Configuration::specialDay(QDateTime const &dt) const { | ||||||
|  |     SpecialDaysType::const_iterator it; | ||||||
|  |     for (it = SpecialDays.cbegin(); it != SpecialDays.cend(); ++it) { | ||||||
|  |         if (dt.date().toString(Qt::ISODate) == QString::fromStdString(it->second.ped_date_start)) { | ||||||
|  |             return it->second; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return ATBSpecialDays(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int Configuration::specialDayId(QDateTime const &dt) const { | ||||||
|  |     SpecialDaysType::const_iterator it; | ||||||
|  |     for (it = SpecialDays.cbegin(); it != SpecialDays.cend(); ++it) { | ||||||
|  |         if (dt.date().toString(Qt::ISODate) == QString::fromStdString(it->second.ped_date_start)) { | ||||||
|  |             int const specialDayId = it->second.ped_id; | ||||||
|  |             return specialDayId; // must be > 0 | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool Configuration::isSpecialDay(QDateTime const &dt) const { | ||||||
|  |     return (specialDayId(dt) > 0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool Configuration::isDayIncludedAsSpecialDay(uint64_t businessHours, int specialDayId) const { | ||||||
|  |     if (specialDayId > 0) { | ||||||
|  |         bool const &r = ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY); | ||||||
|  |         return r; | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool Configuration::isDayIncludedAsSpecialDay(uint64_t businessHours, QDateTime const &dt) const { | ||||||
|  |     // included in 'businessHours' | ||||||
|  |  | ||||||
|  |     if (isSpecialDay(dt)) { | ||||||
|  |         bool const &r = ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY); | ||||||
|  |         qCritical() << "XXXXXXXXXXXXXXXXXX r" << r << businessHours; | ||||||
|  |         return r; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool Configuration::isDayIncluded(uint64_t businessHours, QDateTime const &dt) const { | ||||||
|  |     return Utilities::isDayIncluded(businessHours, dt); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ATBPaymentOption const &Configuration::getPaymentOptions(int paymentOptionsIndex) const { | ||||||
|     Q_ASSERT(!this->currentPaymentOptions.isEmpty()); |     Q_ASSERT(!this->currentPaymentOptions.isEmpty()); | ||||||
|     return this->currentPaymentOptions[0]; |     return this->currentPaymentOptions.at(paymentOptionsIndex); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ATBPaymentOption &Configuration::getPaymentOptions(int paymentOptionsIndex) { | ||||||
|  |     Q_ASSERT(!this->currentPaymentOptions.isEmpty()); | ||||||
|  |     return this->currentPaymentOptions[paymentOptionsIndex]; | ||||||
| } | } | ||||||
|  |  | ||||||
| QVector<ATBPaymentOption> const &Configuration::getAllPaymentOptions() const { | QVector<ATBPaymentOption> const &Configuration::getAllPaymentOptions() const { | ||||||
| @@ -581,12 +774,38 @@ Configuration::getTariffProductForAllKeys() const { | |||||||
| } | } | ||||||
|  |  | ||||||
| std::optional<QVector<ATBTariffProduct>> | std::optional<QVector<ATBTariffProduct>> | ||||||
| Configuration::getTariffProductForProductId(PermitType permitType) const { | Configuration::getTariffProductForProductTypeName(QString const &permitTypeName) const { | ||||||
|     QVector<ATBTariffProduct> products; |     QVector<ATBTariffProduct> products; | ||||||
|     std::optional<QVector<ATBTariffProduct>> value; |     std::optional<QVector<ATBTariffProduct>> value; | ||||||
|  |  | ||||||
|     products.clear(); |     products.clear(); | ||||||
|  |  | ||||||
|  |     for(const auto &product: this->TariffProduct) { | ||||||
|  |         ATBTariffProduct const &v = product.second; | ||||||
|  |         if (v.m_tariff_product_name == permitTypeName) { | ||||||
|  |             products.append(v); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (products.size() > 0) { | ||||||
|  |         value = value.value_or(products); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | std::optional<QVector<ATBTariffProduct>> | ||||||
|  | Configuration::getTariffProductForProductId(PermitType permitType) const { | ||||||
|  |     QString permitTypeName(permitType); | ||||||
|  |     return getTariffProductForProductTypeName(permitTypeName); | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |     QVector<ATBTariffProduct> products; | ||||||
|  |     std::optional<QVector<ATBTariffProduct>> value; | ||||||
|  |  | ||||||
|  |     products.clear(); | ||||||
|  |  | ||||||
|  |  | ||||||
|     for (auto[it, rangeEnd] = this->TariffProduct.equal_range(permitType); it != rangeEnd; ++it) { |     for (auto[it, rangeEnd] = this->TariffProduct.equal_range(permitType); it != rangeEnd; ++it) { | ||||||
|         products.append(it->second); |         products.append(it->second); | ||||||
|     } |     } | ||||||
| @@ -596,6 +815,7 @@ Configuration::getTariffProductForProductId(PermitType permitType) const { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     return value; |     return value; | ||||||
|  |     */ | ||||||
| } | } | ||||||
|  |  | ||||||
| std::optional<QVector<ATBTariffProduct>> | std::optional<QVector<ATBTariffProduct>> | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| #include "utilities.h" | #include "utilities.h" | ||||||
| #include "tariff_log.h" | #include "tariff_log.h" | ||||||
|  | #include "tariff_business_hours.h" | ||||||
|  |  | ||||||
| #include <QDebug> | #include <QDebug> | ||||||
| #include <algorithm> | #include <algorithm> | ||||||
| @@ -260,7 +261,7 @@ bool Utilities::CheckSpecialDay(Configuration* cfg, const char* currentDateTimeS | |||||||
| 						//cout << "CheckSpecialDay() => Month is in range between start and end" << 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)) | 						if ((current_tm.tm_mday >= start_tm.tm_mday) && (current_tm.tm_mday <= end_tm.tm_mday)) | ||||||
| 						{ | 						{ | ||||||
| 							LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY"); |                             //LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY"); | ||||||
| 							*specialDayId = spec_days_itr->second.ped_id; | 							*specialDayId = spec_days_itr->second.ped_id; | ||||||
| 							*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; | 							*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; | ||||||
| 							return true; | 							return true; | ||||||
| @@ -275,7 +276,7 @@ bool Utilities::CheckSpecialDay(Configuration* cfg, const char* currentDateTimeS | |||||||
| 					//cout << "CheckSpecialDay() => Month is in range between start and end" << 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)) | 					if ((current_tm.tm_mday >= start_tm.tm_mday) && (current_tm.tm_mday <= end_tm.tm_mday)) | ||||||
| 					{ | 					{ | ||||||
| 						LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY"); |                         //LOG_DEBUG("CheckSpecialDay() => SPECIAL DAY"); | ||||||
| 						*specialDayId = spec_days_itr->second.ped_id; | 						*specialDayId = spec_days_itr->second.ped_id; | ||||||
| 						*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; | 						*specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; | ||||||
| 						return true; | 						return true; | ||||||
| @@ -317,7 +318,7 @@ bool Utilities::CheckSpecialDay(Configuration const *cfg, | |||||||
|                             continue; |                             continue; | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     qDebug() << "CheckSpecialDay() => SPECIAL DAY"; |                     //qDebug() << "CheckSpecialDay() => SPECIAL DAY"; | ||||||
|                     *specialDayId = spec_days_itr->second.ped_id; |                     *specialDayId = spec_days_itr->second.ped_id; | ||||||
|                     *specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; |                     *specialDayPrice = cfg->SpecialDaysWorktime.find(*specialDayId)->second.pedwt_price; | ||||||
|                     return true; |                     return true; | ||||||
| @@ -333,19 +334,27 @@ QTime Utilities::SpecialDaysWorkTimeFrom(Configuration const *cfg, int specialDa | |||||||
|     return QTime::fromString(cfg->SpecialDaysWorktime.find(specialDayId)->second.pedwt_time_from.c_str(), Qt::ISODate); |     return QTime::fromString(cfg->SpecialDaysWorktime.find(specialDayId)->second.pedwt_time_from.c_str(), Qt::ISODate); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | QTime Utilities::SpecialDaysWorkTimeFrom(Configuration::SpecialDaysWorktimeType::const_iterator const &it) { | ||||||
|  |     return QTime::fromString(it->second.pedwt_time_from.c_str(), Qt::ISODate); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | QTime Utilities::SpecialDaysWorkTimeUntil(Configuration::SpecialDaysWorktimeType::const_iterator const &it) { | ||||||
|  |     return QTime::fromString(it->second.pedwt_time_to.c_str(), Qt::ISODate); | ||||||
|  | } | ||||||
|  |  | ||||||
| QTime Utilities::SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId) { | QTime Utilities::SpecialDaysWorkTimeUntil(Configuration const *cfg, int specialDayId) { | ||||||
|     return QTime::fromString(cfg->SpecialDaysWorktime.find(specialDayId)->second.pedwt_time_to.c_str(), Qt::ISODate); |     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) { | QTime Utilities::WeekDaysWorkTimeFrom(std::multimap<int, ATBWeekDaysWorktime>::const_iterator const &itr) { | ||||||
|     return QTime::fromString(itr->second.pwd_time_from.c_str(), Qt::ISODate); |     return QTime::fromString(itr->second.pwd_time_from.c_str(), Qt::ISODate); | ||||||
| } | } | ||||||
|  |  | ||||||
| QTime Utilities::WeekDaysWorkTimeUntil(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr) { | QTime Utilities::WeekDaysWorkTimeUntil(std::multimap<int, ATBWeekDaysWorktime>::const_iterator const &itr) { | ||||||
|     return QTime::fromString(itr->second.pwd_time_to.c_str(), Qt::ISODate); |     return QTime::fromString(itr->second.pwd_time_to.c_str(), Qt::ISODate); | ||||||
| } | } | ||||||
|  |  | ||||||
| int Utilities::WeekDayId(std::multimap<int, ATBWeekDaysWorktime>::const_iterator itr) { | int Utilities::WeekDayId(std::multimap<int, ATBWeekDaysWorktime>::const_iterator const &itr) { | ||||||
|     return itr->second.pwd_period_day_in_week_id; |     return itr->second.pwd_period_day_in_week_id; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -397,9 +406,16 @@ uint32_t Utilities::getMaximalParkingPrice(Configuration const *cfg, PaymentMeth | |||||||
|     return std::max((int)cfg->PaymentOption.find(methodId)->second.pop_max_price, 0); |     return std::max((int)cfg->PaymentOption.find(methodId)->second.pop_max_price, 0); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | uint32_t Utilities::getDailyTicketCardPrice(Configuration const *cfg, PaymentMethod methodId) { | ||||||
|  |     return std::max((int)cfg->PaymentOption.find(methodId)->second.pop_daily_card_price, 0); | ||||||
|  | } | ||||||
|  |  | ||||||
| uint32_t Utilities::getTimeRangeStep(Configuration const *cfg, int step, PaymentMethod methodId) { | uint32_t Utilities::getTimeRangeStep(Configuration const *cfg, int step, PaymentMethod methodId) { | ||||||
|     if (methodId == PaymentMethod::Progressive) { |     if (methodId == PaymentMethod::Progressive) { | ||||||
|         return std::max((int)cfg->TimeRange.find(step)->second.time_range_to_in_minutes_from_start, 0); |         Configuration::TimeRangeType::const_iterator it = cfg->TimeRange.find(step); | ||||||
|  |         if (it != cfg->TimeRange.cend()) { | ||||||
|  |             return std::max((int)(it->second.time_range_to_in_minutes_from_start), 0); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return 0; |     return 0; | ||||||
| @@ -420,10 +436,10 @@ uint32_t Utilities::getFirstDurationStep(Configuration const *cfg, PaymentMethod | |||||||
| } | } | ||||||
|  |  | ||||||
| BusinessHours Utilities::getBusinessHours(Configuration const *cfg, PaymentMethod methodId) { | BusinessHours Utilities::getBusinessHours(Configuration const *cfg, PaymentMethod methodId) { | ||||||
|     int businessHours = cfg->PaymentOption.find(methodId)->second.pop_business_hours; |     uint64_t businessHours = cfg->PaymentOption.find(methodId)->second.pop_business_hours; | ||||||
|  |  | ||||||
|     switch (businessHours) { |     switch (businessHours) { | ||||||
|     case NoRestriction_24_7: return BusinessHours::NoRestriction_24_7; |     //case NoRestriction_24_7: return BusinessHours::NoRestriction_24_7; | ||||||
|     case OnlyWorkingDays: return BusinessHours::OnlyWorkingDays; |     case OnlyWorkingDays: return BusinessHours::OnlyWorkingDays; | ||||||
|     case OnlyWeekDays: return BusinessHours::OnlyWeekDays; |     case OnlyWeekDays: return BusinessHours::OnlyWeekDays; | ||||||
|     case OnlyWeekEnd: return BusinessHours::OnlyWeekEnd; |     case OnlyWeekEnd: return BusinessHours::OnlyWeekEnd; | ||||||
| @@ -433,6 +449,21 @@ BusinessHours Utilities::getBusinessHours(Configuration const *cfg, PaymentMetho | |||||||
|     case SpecialAndSchoolHolidays: return BusinessHours::SpecialAndSchoolHolidays; |     case SpecialAndSchoolHolidays: return BusinessHours::SpecialAndSchoolHolidays; | ||||||
|     case OnlyOpenForBusinessDays: return BusinessHours::OnlyOpenForBusinessDays; |     case OnlyOpenForBusinessDays: return BusinessHours::OnlyOpenForBusinessDays; | ||||||
|     case AllDaysWithRestrictedHours: return BusinessHours::AllDaysWithRestrictedHours; |     case AllDaysWithRestrictedHours: return BusinessHours::AllDaysWithRestrictedHours; | ||||||
|  |     case _NO_RESTRICTION_24_7_: return BusinessHours::NO_RESTRICTION_24_7; | ||||||
|  |     case _MON_: return BusinessHours::MON; | ||||||
|  |     case _TUE_: return BusinessHours::TUE; | ||||||
|  |     case _WED_: return BusinessHours::WED; | ||||||
|  |     case _THU_: return BusinessHours::THU; | ||||||
|  |     case _FRI_: return BusinessHours::FRI; | ||||||
|  |     case _SAT_: return BusinessHours::SAT; | ||||||
|  |     case _SUN_: return BusinessHours::SUN; | ||||||
|  |     case _WEEK_DAYS_: return BusinessHours::WEEK_DAYS; | ||||||
|  |     case _WORKING_DAYS_: return BusinessHours::WORKING_DAYS; | ||||||
|  |     case _ALL_DAYS_: return BusinessHours::ALL_DAYS; | ||||||
|  |     case _OFFICIAL_HOLIDAY_: return BusinessHours::OFFICIAL_HOLIDAY; | ||||||
|  |     case _ONLY_WEEKEND_: return BusinessHours::ONLY_WEEKEND; | ||||||
|  |     case _ONLY_OPEN_FOR_BUSINESS_DAYS_: return BusinessHours::ONLY_OPEN_FOR_BUSINESS_DAYS; | ||||||
|  |     case _NOT_DEFINED_: return BusinessHours::NOT_DEFINED; | ||||||
|     } |     } | ||||||
|     return BusinessHours::NoBusinessHoursDefined; |     return BusinessHours::NoBusinessHoursDefined; | ||||||
| } | } | ||||||
| @@ -447,3 +478,82 @@ double Utilities::computeWeekDaysDurationUnit(Configuration const *cfg, PaymentM | |||||||
|     int durationId = cfg->PaymentRate.find(pop_id)->second.pra_payment_unit_id; |     int durationId = cfg->PaymentRate.find(pop_id)->second.pra_payment_unit_id; | ||||||
|     return (double)(cfg->Duration.find(durationId)->second.pun_duration); |     return (double)(cfg->Duration.find(durationId)->second.pun_duration); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | bool Utilities::isDayIncluded(uint64_t businessHours, QDateTime const &dt) { | ||||||
|  |     int dayOfWeek = dt.date().dayOfWeek(); | ||||||
|  |     switch (dayOfWeek) { | ||||||
|  |     case Qt::Monday: | ||||||
|  |         return ((businessHours & BusinessHours::MON) == BusinessHours::MON); | ||||||
|  |     case Qt::Tuesday: | ||||||
|  |         return ((businessHours & BusinessHours::TUE) == BusinessHours::TUE); | ||||||
|  |     case Qt::Wednesday: | ||||||
|  |         return ((businessHours & BusinessHours::WED) == BusinessHours::WED); | ||||||
|  |     case Qt::Thursday: | ||||||
|  |         return ((businessHours & BusinessHours::THU) == BusinessHours::THU); | ||||||
|  |     case Qt::Friday: | ||||||
|  |         return ((businessHours & BusinessHours::FRI) == BusinessHours::FRI); | ||||||
|  |     case Qt::Saturday: | ||||||
|  |         return ((businessHours & BusinessHours::SAT) == BusinessHours::SAT); | ||||||
|  |     case Qt::Sunday: | ||||||
|  |         return ((businessHours & BusinessHours::SUN) == BusinessHours::SUN); | ||||||
|  |     default:; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | QStringList Utilities::dumpBusinessHours(uint64_t businessHours) { | ||||||
|  |     QStringList s; | ||||||
|  |  | ||||||
|  |     if ((businessHours & BusinessHours::MON) == BusinessHours::MON) { | ||||||
|  |         if ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |             s << "MON (Holiday)"; | ||||||
|  |         } else { | ||||||
|  |             s << "MON"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     if ((businessHours & BusinessHours::TUE) == BusinessHours::TUE) { | ||||||
|  |         if ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |             s << "TUE (Holiday)"; | ||||||
|  |         } else { | ||||||
|  |             s << "TUE"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     if ((businessHours & BusinessHours::WED) == BusinessHours::WED) { | ||||||
|  |         if ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |             s << "WED (Holiday)"; | ||||||
|  |         } else { | ||||||
|  |             s << "WED"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     if ((businessHours & BusinessHours::THU) == BusinessHours::THU) { | ||||||
|  |         if ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |             s << "THU (Holiday)"; | ||||||
|  |         } else { | ||||||
|  |             s << "THU"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     if ((businessHours & BusinessHours::FRI) == BusinessHours::FRI) { | ||||||
|  |         if ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |             s << "FRI (Holiday)"; | ||||||
|  |         } else { | ||||||
|  |             s << "FRI"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     if ((businessHours & BusinessHours::SAT) == BusinessHours::SAT) { | ||||||
|  |         if ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |             s << "SAT (Holiday)"; | ||||||
|  |         } else { | ||||||
|  |             s << "SAT"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     if ((businessHours & BusinessHours::SUN) == BusinessHours::SUN) { | ||||||
|  |         if ((businessHours & BusinessHours::OFFICIAL_HOLIDAY) == BusinessHours::OFFICIAL_HOLIDAY) { | ||||||
|  |             s << "SUN (Holiday)"; | ||||||
|  |         } else { | ||||||
|  |             s << "SUN"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return s; | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										1418
									
								
								main/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										1418
									
								
								main/main.cpp
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user