#pragma once

#include "routing/road_point.hpp"
#include "routing/route_weight.hpp"

#include <optional>
#include <string>
#include <vector>

#include "3party/skarupke/flat_hash_map.hpp"
#include "3party/opening_hours/opening_hours.hpp"

namespace routing
{
// This class provides information about road access classes.
// One instance of RoadAccess holds information about one
// mwm and one router type (also known as VehicleType).
class RoadAccess final
{
public:
  // The road access types are selected by analyzing the most
  // popular tags used when mapping roads in OSM.
  enum class Type : uint8_t
  {
    // Moving through the road is prohibited.
    No,

    // General public access is not allowed. Access is granted with individual permission only.
    Private,

    // No pass through the road is allowed; however, it can
    // be used if it is close enough to the destination point
    // of the route.
    Destination,

    // No restrictions, as in "access=yes".
    Yes,

    // The number of different road types.
    Count,

    /// @name Generator only, not serialized options.
    /// @{
    // https://github.com/organicmaps/organicmaps/issues/2600
    // Open only to people who have obtained a permit granting them access, but permit is ordinarily granted.
    Permit,
    // https://github.com/organicmaps/organicmaps/issues/4442
    // locked=yes, will be transformed into Private.
    Locked,
    /// @}
  };

  enum class Confidence
  {
    Maybe,
    Sure
  };

  class Conditional
  {
  public:
    struct Access
    {
      Access(RoadAccess::Type type, osmoh::OpeningHours && openingHours)
          : m_type(type), m_openingHours(std::move(openingHours))
      {
      }

      bool operator==(Access const & rhs) const
      {
        return m_type == rhs.m_type && m_openingHours == rhs.m_openingHours;
      }

      RoadAccess::Type m_type = RoadAccess::Type::Count;
      osmoh::OpeningHours m_openingHours;
    };

    Conditional() = default;

    void Insert(RoadAccess::Type type, osmoh::OpeningHours && openingHours)
    {
      m_accesses.emplace_back(type, std::move(openingHours));
    }

    size_t Size() const { return m_accesses.size(); }
    bool IsEmpty() const { return m_accesses.empty(); }
    std::vector<Access> const & GetAccesses() const { return m_accesses; }

    bool operator==(Conditional const & rhs) const { return m_accesses == rhs.m_accesses; }
    bool operator!=(Conditional const & rhs) const { return !(*this == rhs); }

  private:
    std::vector<Access> m_accesses;
  };

  using WayToAccess = ska::flat_hash_map<uint32_t, RoadAccess::Type>;
  using PointToAccess = ska::flat_hash_map<RoadPoint, RoadAccess::Type, RoadPoint::Hash>;
  using WayToAccessConditional = ska::flat_hash_map<uint32_t, Conditional>;
  using PointToAccessConditional = ska::flat_hash_map<RoadPoint, Conditional, RoadPoint::Hash>;

  RoadAccess();

  WayToAccess const & GetWayToAccess() const { return m_wayToAccess; }
  PointToAccess const & GetPointToAccess() const { return m_pointToAccess; }

  WayToAccessConditional const & GetWayToAccessConditional() const
  {
    return m_wayToAccessConditional;
  }

  PointToAccessConditional const & GetPointToAccessConditional() const
  {
    return m_pointToAccessConditional;
  }

  std::pair<Type, Confidence> GetAccess(uint32_t featureId,
                                        RouteWeight const & weightToFeature) const;
  std::pair<Type, Confidence> GetAccess(RoadPoint const & point,
                                        RouteWeight const & weightToPoint) const;

  std::pair<Type, Confidence> GetAccessWithoutConditional(uint32_t featureId) const;
  std::pair<Type, Confidence> GetAccessWithoutConditional(RoadPoint const & point) const;

  void SetWayAccess(WayToAccess && access, WayToAccessConditional && condAccess)
  {
    m_wayToAccess = std::move(access);
    m_wayToAccessConditional = std::move(condAccess);
  }

  void SetPointAccess(PointToAccess && access, PointToAccessConditional && condAccess)
  {
    m_pointToAccess = std::move(access);
    m_pointToAccessConditional = std::move(condAccess);
  }

  void SetAccess(WayToAccess && wayAccess, PointToAccess && pointAccess)
  {
    m_wayToAccess = std::move(wayAccess);
    m_pointToAccess = std::move(pointAccess);
  }

  void SetAccessConditional(WayToAccessConditional && wayAccess, PointToAccessConditional && pointAccess)
  {
    m_wayToAccessConditional = std::move(wayAccess);
    m_pointToAccessConditional = std::move(pointAccess);
  }

  bool operator==(RoadAccess const & rhs) const;

  template <typename WayToAccess>
  void SetWayToAccessForTests(WayToAccess && wayToAccess)
  {
    m_wayToAccess = std::forward<WayToAccess>(wayToAccess);
  }

  template <typename T>
  void SetCurrentTimeGetter(T && getter) { m_currentTimeGetter = std::forward<T>(getter); }

private:
  // When we check access:conditional, we check such interval in fact:
  // is access:conditional is open at: curTime - |kConfidenceIntervalSeconds| / 2
  // is access:conditional is open at: curTime + |kConfidenceIntervalSeconds| / 2
  // If at both ends access:conditional is open we say, we are sure that this access:conditional is open,
  // if only one is open, we say access:conditional is maybe open.
  inline static time_t constexpr kConfidenceIntervalSeconds = 2 * 3600;  // 2 hours

  static std::optional<Confidence> GetConfidenceForAccessConditional(
      time_t momentInTime, osmoh::OpeningHours const & openingHours);

  std::pair<Type, Confidence> GetAccess(uint32_t featureId, double weight) const;
  std::pair<Type, Confidence> GetAccess(RoadPoint const & point, double weight) const;

  std::function<time_t()> m_currentTimeGetter;

  // If segmentIdx of a key in this map is 0, it means the
  // entire feature has the corresponding access type.
  // Otherwise, the information is about the segment with number (segmentIdx-1).
  WayToAccess m_wayToAccess;
  PointToAccess m_pointToAccess;
  WayToAccessConditional m_wayToAccessConditional;
  PointToAccessConditional m_pointToAccessConditional;
};

time_t GetCurrentTimestamp();

std::string ToString(RoadAccess::Type type);
void FromString(std::string_view s, RoadAccess::Type & result);

std::string DebugPrint(RoadAccess::Confidence confidence);
std::string DebugPrint(RoadAccess::Conditional const & conditional);
std::string DebugPrint(RoadAccess::Type type);
std::string DebugPrint(RoadAccess const & r);
}  // namespace routing