kongdeqiang
7 天以前 28ccfbfc51068a663a80764e14074df5edf2b5ba
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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
// -*- Mode: c++; c-basic-offset: 4; tab-width: 4; -*-
 
/******************************************************************************
 *
 *  file:  ArgTraits.h
 *
 *  Copyright (c) 2007, Daniel Aarno, Michael E. Smoot .
 *  Copyright (c) 2017 Google LLC
 *  All rights reserved.
 *
 *  See the file COPYING in the top directory of this distribution for
 *  more information.
 *
 *  THE SOFTWARE IS PROVIDED _AS IS_, WITHOUT WARRANTY OF ANY KIND, EXPRESS
 *  OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 *  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
 *  THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 *  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 *  FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 *  DEALINGS IN THE SOFTWARE.
 *
 *****************************************************************************/
 
// This is an internal tclap file, you should probably not have to
// include this directly
 
#ifndef TCLAP_ARGTRAITS_H
#define TCLAP_ARGTRAITS_H
 
namespace TCLAP {
 
// We use two empty structs to get compile type specialization
// function to work
 
/**
 * A value like argument value type is a value that can be set using
 * operator>>. This is the default value type.
 */
struct ValueLike {
    typedef ValueLike ValueCategory;
    virtual ~ValueLike() {}
};
 
/**
 * A string like argument value type is a value that can be set using
 * operator=(string). Useful if the value type contains spaces which
 * will be broken up into individual tokens by operator>>.
 */
struct StringLike {
    virtual ~StringLike() {}
};
 
/**
 * A class can inherit from this object to make it have string like
 * traits. This is a compile time thing and does not add any overhead
 * to the inherenting class.
 */
struct StringLikeTrait {
    typedef StringLike ValueCategory;
    virtual ~StringLikeTrait() {}
};
 
/**
 * A class can inherit from this object to make it have value like
 * traits. This is a compile time thing and does not add any overhead
 * to the inherenting class.
 */
struct ValueLikeTrait {
    typedef ValueLike ValueCategory;
    virtual ~ValueLikeTrait() {}
};
 
/**
 * Arg traits are used to get compile type specialization when parsing
 * argument values. Using an ArgTraits you can specify the way that
 * values gets assigned to any particular type during parsing. The two
 * supported types are StringLike and ValueLike. ValueLike is the
 * default and means that operator>> will be used to assign values to
 * the type.
 */
template<typename T>
class ArgTraits {
    // This is a bit silly, but what we want to do is:
    // 1) If there exists a specialization of ArgTraits for type X,
    // use it.
    //
    // 2) If no specialization exists but X has the typename
    // X::ValueCategory, use the specialization for X::ValueCategory.
    //
    // 3) If neither (1) nor (2) defines the trait, use the default
    // which is ValueLike.
 
    // This is the "how":
    //
    // test<T>(0) (where 0 is the NULL ptr) will match
    // test(typename C::ValueCategory*) iff type T has the
    // corresponding typedef. If it does not test(...) will be
    // matched. This allows us to determine if T::ValueCategory
    // exists by checking the sizeof for the test function (return
    // value must have different sizeof).
    template<typename C> static short test(typename C::ValueCategory*);
    template<typename C> static long  test(...);
    static const bool hasTrait = sizeof(test<T>(0)) == sizeof(short);
 
    template <typename C, bool>
    struct DefaultArgTrait {
        typedef ValueLike ValueCategory;
    };
 
    template <typename C>
    struct DefaultArgTrait<C, true> {
        typedef typename C::ValueCategory ValueCategory;
     };
 
public:
    typedef typename DefaultArgTrait<T, hasTrait>::ValueCategory ValueCategory;
};
 
} // namespace
 
#endif