/*
 * Scope Guard
 * Copyright (C) 2017-2018  offa
 *
 * This file is part of Scope Guard.
 *
 * Scope Guard is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * Scope Guard is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Scope Guard.  If not, see .
 */
#pragma once
#include 
#include 
namespace sr::detail
{
   template
   class Wrapper
   {
   public:
        template, int> = 0>
        Wrapper(TT&& value, G&& g) noexcept(std::is_nothrow_constructible_v) : m_value(std::forward(value))
        {
            g.release();
        }
        T& get() noexcept
        {
            return m_value;
        }
        const T& get() const noexcept
        {
            return m_value;
        }
        void reset(Wrapper&& other) noexcept
        {
            m_value = std::move(other.m_value);
        }
        void reset(T&& newValue) noexcept(std::is_nothrow_assignable_v)
        {
            m_value = std::forward(newValue);
        }
        void reset(const T& newValue) noexcept(std::is_nothrow_assignable_v)
        {
            m_value = newValue;
        }
        using type = T;
    private:
        T m_value;
    };
   template
   class Wrapper
   {
   public:
        template, int> = 0>
        Wrapper(TT&& value, G&& g) noexcept(std::is_nothrow_constructible_v) : m_value(static_cast(value))
        {
            g.release();
        }
        T& get() noexcept
        {
            return m_value.get();
        }
        const T& get() const noexcept
        {
            return m_value.get();
        }
        void reset(Wrapper&& other) noexcept
        {
            m_value = std::move(other.m_value);
        }
        void reset(T& newValue) noexcept
        {
            m_value = std::ref(newValue);
        }
        using type = std::reference_wrapper>;
    private:
        type m_value;
   };
}