/*
 * Scope Guard
 * Copyright (C) 2017  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(noexcept(Wrapper{value})) : Wrapper(std::forward(value))
       {
           g.release();
       }
       T& get() noexcept
       {
           return m_value;
       }
       const T& get() const noexcept
       {
           return 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:
       Wrapper(const T& value) noexcept(noexcept(T{value})) : m_value(value)
       {
       }
       Wrapper(T&& value) noexcept(noexcept(T{std::move_if_noexcept(value)})) : m_value(std::move_if_noexcept(value))
       {
       }
       T m_value;
   };
   template
   class Wrapper
   {
   public:
       template, int> = 0>
       Wrapper(TT&& value, G&& g) noexcept(noexcept(static_cast(value))) : 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(T& newValue) noexcept
       {
           m_value = std::ref(newValue);
       }
       using type = std::reference_wrapper;
   private:
       std::reference_wrapper m_value;
   };
}