2016-05-23 22:30:23 +01:00
|
|
|
// -*- C++ -*-
|
|
|
|
/**
|
|
|
|
* \file RefChanger.h
|
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
|
|
|
*
|
|
|
|
* \author Guillaume Munch
|
|
|
|
*
|
|
|
|
* Full author contact details are available in file CREDITS.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef LYX_REFCHANGER_H
|
|
|
|
#define LYX_REFCHANGER_H
|
|
|
|
|
|
|
|
#include "support/Changer.h"
|
|
|
|
|
|
|
|
|
|
|
|
namespace lyx {
|
|
|
|
|
|
|
|
/// A RefChanger records the current value of \param ref, allowing to
|
|
|
|
/// temporarily assign new values to it. The original value is restored
|
|
|
|
/// automatically when the object is destroyed, unless it is disabled.
|
|
|
|
///
|
|
|
|
/// RefChanger is movable, and doing so prolongs the duration of the temporary
|
|
|
|
/// assignment. This allows classes to supply their own changer methods.
|
|
|
|
///
|
|
|
|
/// Naturally, be careful not to extend the life of a RefChanger beyond that of
|
|
|
|
/// the reference it modifies. The RefChanger can be disabled by calling
|
|
|
|
/// ->keep() or ->revert(). Once disabled, the reference is never accessed
|
|
|
|
/// again.
|
|
|
|
template<typename X>
|
|
|
|
class RevertibleRef : public Revertible {
|
|
|
|
public:
|
|
|
|
RevertibleRef(X & ref) : ref(ref), old(ref), enabled(true) {}
|
|
|
|
//
|
|
|
|
~RevertibleRef() { revert(); }
|
|
|
|
//
|
|
|
|
void revert() { if (enabled) { enabled = false; ref = old; } }
|
|
|
|
//
|
|
|
|
void keep() { enabled = false; }
|
|
|
|
//
|
|
|
|
X & ref;
|
|
|
|
X const old;
|
|
|
|
private:
|
|
|
|
bool enabled;
|
|
|
|
};
|
|
|
|
|
2016-06-14 19:22:24 +01:00
|
|
|
|
|
|
|
//for gcc 4.6
|
|
|
|
#if defined(__GNUC__) && (__GNUC__ == 4) && (__GNUC_MINOR__ == 6)
|
|
|
|
template <typename X>
|
|
|
|
struct RefChanger : unique_ptr<RevertibleRef<X>>
|
|
|
|
{
|
|
|
|
RefChanger(unique_ptr<RevertibleRef<X>> p)
|
|
|
|
: unique_ptr<RevertibleRef<X>>(move(p))
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
#else
|
2016-05-23 22:30:23 +01:00
|
|
|
template <typename X> using RefChanger = unique_ptr<RevertibleRef<X>>;
|
2016-06-14 19:22:24 +01:00
|
|
|
#endif
|
2016-05-23 22:30:23 +01:00
|
|
|
|
|
|
|
|
|
|
|
/// Saves the value of \param ref in a movable object
|
|
|
|
template <typename X> RefChanger<X> make_save(X & ref)
|
|
|
|
{
|
|
|
|
return make_unique<RevertibleRef<X>>(ref);
|
|
|
|
}
|
|
|
|
|
2016-11-20 19:53:16 +01:00
|
|
|
/// Temporarily assign value val to reference ref.
|
|
|
|
/// To apply the change conditionnally, one can write:
|
|
|
|
/// Changer dummy = (cond) ? make_change(a, b) : Changer();
|
2016-05-23 22:30:23 +01:00
|
|
|
template <typename X>
|
2016-11-20 19:53:16 +01:00
|
|
|
RefChanger<X> make_change(X & ref, X const val)
|
2016-05-23 22:30:23 +01:00
|
|
|
{
|
|
|
|
auto rc = make_save(ref);
|
2016-11-20 19:53:16 +01:00
|
|
|
ref = val;
|
2016-05-23 22:30:23 +01:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-23 13:11:54 +02:00
|
|
|
} // namespace lyx
|
2016-05-23 22:30:23 +01:00
|
|
|
|
|
|
|
|
|
|
|
#endif //LYX_REFCHANGER_H
|