#ifndef OPENMW_COMPONENTS_MISC_FRAMERATELIMITER_H
#define OPENMW_COMPONENTS_MISC_FRAMERATELIMITER_H

#include <chrono>
#include <thread>

namespace Misc
{
    class FrameRateLimiter
    {
    public:
        template <class Rep, class Ratio>
        explicit FrameRateLimiter(std::chrono::duration<Rep, Ratio> maxFrameDuration,
            std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now())
            : mMaxFrameDuration(std::chrono::duration_cast<std::chrono::steady_clock::duration>(maxFrameDuration))
            , mLastMeasurement(now)
            , mLastFrameDuration(0)
        {
        }

        std::chrono::steady_clock::duration getLastFrameDuration() const { return mLastFrameDuration; }

        void limit(std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now())
        {
            const auto passed = now - mLastMeasurement;
            const auto left = mMaxFrameDuration - passed;
            if (left > left.zero())
            {
                std::this_thread::sleep_for(left);
                mLastMeasurement = now + left;
                mLastFrameDuration = mMaxFrameDuration;
            }
            else
            {
                mLastMeasurement = now;
                mLastFrameDuration = passed;
            }
        }

    private:
        std::chrono::steady_clock::duration mMaxFrameDuration;
        std::chrono::steady_clock::time_point mLastMeasurement;
        std::chrono::steady_clock::duration mLastFrameDuration;
    };

    inline Misc::FrameRateLimiter makeFrameRateLimiter(float frameRateLimit)
    {
        if (frameRateLimit > 0.0f)
            return Misc::FrameRateLimiter(std::chrono::duration<float>(1.0f / frameRateLimit));
        else
            return Misc::FrameRateLimiter(std::chrono::steady_clock::duration::zero());
    }
}

#endif