forked from CleverRaven/Cataclysm-DDA
-
Notifications
You must be signed in to change notification settings - Fork 0
/
pimpl.h
80 lines (71 loc) · 2.82 KB
/
pimpl.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
#pragma once
#ifndef CATA_SRC_PIMPL_H
#define CATA_SRC_PIMPL_H
#include <memory>
#include <type_traits>
class JsonValue;
template<typename T>
class pimpl;
template<typename ...T>
class is_pimpl_helper : public std::false_type
{
};
template<typename T>
class is_pimpl_helper<pimpl<T>> : public std::true_type
{
};
template<typename T>
class is_pimpl : public is_pimpl_helper<typename std::decay<T>::type>
{
};
/**
* This is a wrapper for implementing the pointer-to-implementation technique,
* see for example http://en.cppreference.com/w/cpp/language/pimpl
*
* It's a container for one element of type @p T, which is *always* present
* (one can not create a @ref pimpl instance without a contained @ref T instance).
*
* The contained @ref T can be access like a pointer via @ref operator-> and @ref operator.
*
* The constructor forwards all its arguments to the constructor of @ref T.
*/
template <typename T>
class pimpl : private std::unique_ptr<T>
{
public:
// Original second template arguments was `std::enable_if<std::is_constructible<T, Args>::value>`,
// but this caused errors in few compilers when `T` was not a complete type.
// The new argument serves the same purpose: this constructor should *not* be available when the
// argument is a `pimpl` itself (the other copy constructors should be used instead).
explicit pimpl() : std::unique_ptr<T>( new T() ) { }
template < typename P, typename ...Args,
typename = typename std::enable_if < !is_pimpl<P>::value >::type >
explicit pimpl( P && head, Args &&
... args ) : std::unique_ptr<T>( new T( std::forward<P>( head ), std::forward<Args>( args )... ) ) { }
pimpl( const pimpl<T> &rhs ) : std::unique_ptr<T>( new T( *rhs ) ) { }
pimpl( pimpl<T> &&rhs ) noexcept : std::unique_ptr<T>( new T( std::move( *rhs ) ) ) { }
pimpl<T> &operator=( const pimpl<T> &rhs ) {
operator*() = *rhs;
return *this;
}
pimpl<T> &operator=( pimpl<T> &&rhs ) noexcept {
operator*() = std::move( *rhs );
return *this;
}
bool operator==( const pimpl<T> &rhs ) const {
return operator*() == *rhs;
}
using std::unique_ptr<T>::operator->;
using std::unique_ptr<T>::operator*;
/// Forwards the stream to `T::deserialize`.
template<typename Value = JsonValue, std::enable_if_t<std::is_same<std::decay_t<Value>, JsonValue>::value>* = nullptr>
void deserialize( const Value &stream ) {
operator*().deserialize( stream );
}
/// Forwards the stream to `T::serialize`.
template<typename JsonStream>
void serialize( JsonStream &stream ) const {
operator*().serialize( stream );
}
};
#endif // CATA_SRC_PIMPL_H