2018-06-12 14:50:54 +10:00
|
|
|
/*
|
2018-08-04 15:14:06 +10:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
2018-06-12 14:50:54 +10:00
|
|
|
*
|
2018-06-13 15:43:01 +10:00
|
|
|
* Copyright 2012-2018, Danny Robson <danny@nerdcruft.net>
|
2018-06-12 14:50:54 +10:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2018-06-18 13:51:59 +10:00
|
|
|
#include <functional>
|
|
|
|
|
2018-06-12 14:50:54 +10:00
|
|
|
|
2018-08-05 14:42:02 +10:00
|
|
|
namespace cruft::scoped {
|
2018-06-12 14:50:54 +10:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
/// restores a referenced target variable to its original value at
|
|
|
|
/// destruction time. the type must be copyable.
|
|
|
|
template <typename ValueT>
|
|
|
|
class restore {
|
|
|
|
public:
|
2018-06-13 15:43:01 +10:00
|
|
|
explicit restore (ValueT &_target):
|
2018-06-12 14:50:54 +10:00
|
|
|
m_target (_target),
|
|
|
|
m_value (_target)
|
|
|
|
{ ; }
|
|
|
|
|
|
|
|
~restore ()
|
|
|
|
{
|
2019-04-12 16:40:17 +10:00
|
|
|
if (m_restore)
|
|
|
|
m_target = m_value;
|
2018-06-12 14:50:54 +10:00
|
|
|
}
|
|
|
|
|
2019-04-12 16:40:17 +10:00
|
|
|
/// Disable restoration of the value.
|
2019-08-12 12:58:44 +10:00
|
|
|
void release (void) { m_restore = false; }
|
2019-04-12 16:40:17 +10:00
|
|
|
|
2018-06-12 14:50:54 +10:00
|
|
|
private:
|
2019-04-12 16:40:17 +10:00
|
|
|
bool m_restore = true;
|
2018-06-12 14:50:54 +10:00
|
|
|
ValueT &m_target;
|
|
|
|
ValueT const m_value;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
template <typename ValueT>
|
|
|
|
restore (ValueT&) -> restore<ValueT>;
|
2018-06-13 15:43:01 +10:00
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
/// increments the referenced variable at construction time, and
|
|
|
|
/// decrements at destruction time.
|
|
|
|
template <typename ValueT>
|
|
|
|
class increment {
|
|
|
|
public:
|
|
|
|
explicit increment (ValueT &_target):
|
|
|
|
m_target (_target)
|
|
|
|
{
|
|
|
|
++m_target;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
~increment ()
|
|
|
|
{
|
|
|
|
--m_target;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
ValueT &m_target;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
template <typename ValueT>
|
|
|
|
increment (ValueT&) -> increment<ValueT>;
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
/// calls a function with the supplied arguments at destruction time
|
|
|
|
template <typename FuncT, typename ...Args>
|
|
|
|
class function {
|
|
|
|
public:
|
2019-04-25 09:22:02 +10:00
|
|
|
explicit function (FuncT _function, Args ..._args):
|
2018-06-18 13:51:59 +10:00
|
|
|
m_function (std::forward<FuncT> (_function)),
|
2018-06-13 15:43:01 +10:00
|
|
|
m_args (std::forward<Args> (_args)...)
|
|
|
|
{ ; }
|
|
|
|
|
2018-06-18 13:50:16 +10:00
|
|
|
function (function&&);
|
|
|
|
function (const function&);
|
2018-06-13 15:43:01 +10:00
|
|
|
|
|
|
|
~function ()
|
|
|
|
{
|
|
|
|
if (m_enabled)
|
2018-06-18 13:51:59 +10:00
|
|
|
std::apply (m_function, m_args);
|
2018-06-13 15:43:01 +10:00
|
|
|
}
|
|
|
|
|
2019-08-12 12:58:44 +10:00
|
|
|
void release (void) { m_enabled = false; }
|
2018-06-18 13:50:16 +10:00
|
|
|
|
2018-06-13 15:43:01 +10:00
|
|
|
private:
|
|
|
|
bool m_enabled = true;
|
|
|
|
FuncT m_function;
|
|
|
|
std::tuple<Args...> m_args;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
template <typename FuncT, typename ...Args>
|
2018-06-18 13:51:59 +10:00
|
|
|
function (FuncT,Args...) -> function<FuncT,Args...>;
|
2018-06-13 15:43:01 +10:00
|
|
|
};
|