OpenVolumeMeshHandle.hh 7.71 KB
Newer Older
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
/*===========================================================================*\
 *                                                                           *
 *                            OpenVolumeMesh                                 *
 *        Copyright (C) 2011 by Computer Graphics Group, RWTH Aachen         *
 *                        www.openvolumemesh.org                             *
 *                                                                           *
 *---------------------------------------------------------------------------*
 *  This file is part of OpenVolumeMesh.                                     *
 *                                                                           *
 *  OpenVolumeMesh is free software: you can redistribute it and/or modify   *
 *  it under the terms of the GNU Lesser General Public License as           *
 *  published by the Free Software Foundation, either version 3 of           *
 *  the License, or (at your option) any later version with the              *
 *  following exceptions:                                                    *
 *                                                                           *
 *  If other files instantiate templates or use macros                       *
 *  or inline functions from this file, or you compile this file and         *
 *  link it with other files to produce an executable, this file does        *
 *  not by itself cause the resulting executable to be covered by the        *
 *  GNU Lesser General Public License. This exception does not however       *
 *  invalidate any other reasons why the executable file might be            *
 *  covered by the GNU Lesser General Public License.                        *
 *                                                                           *
 *  OpenVolumeMesh 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 Lesser General Public License for more details.                      *
 *                                                                           *
 *  You should have received a copy of the GNU LesserGeneral Public          *
 *  License along with OpenVolumeMesh.  If not,                              *
 *  see <http://www.gnu.org/licenses/>.                                      *
 *                                                                           *
\*===========================================================================*/

35
#pragma once
36

37
#include <algorithm>
38
39
#include <iosfwd>
#include <vector>
40
41
#include <cassert>
#include <limits>
42

43
#include "Entities.hh"
44
#include "../System/FunctionalInclude.hh"
45
#include "../System/Deprecation.hh"
Mike Kremer's avatar
Mike Kremer committed
46

47
48
49
50
51
52
namespace OpenVolumeMesh {

// Define handle types in order to distinguish different entities by their indices
class OpenVolumeMeshHandle {
public:
    // Default constructor
53
    explicit constexpr OpenVolumeMeshHandle(int _idx) : idx_(_idx) {}
54
55
56
57
58
59

	OpenVolumeMeshHandle& operator=(int _idx) {
		idx_ = _idx;
		return *this;
	}

60
61
    OpenVolumeMeshHandle(const OpenVolumeMeshHandle& _idx) = default;
    OpenVolumeMeshHandle& operator=(const OpenVolumeMeshHandle& _idx) = default;
62
63
64
65
66
67
68

	inline bool is_valid() const { return idx_ != -1; }

	inline bool operator<(const OpenVolumeMeshHandle& _idx) const { return (this->idx_ < _idx.idx_); }

	inline bool operator<(int _idx) const { return idx_ < _idx; }

69
70
71
72
73
74
75
76
	inline bool operator>(const OpenVolumeMeshHandle& _idx) const { return (this->idx_ > _idx.idx_); }

    inline bool operator>(int _idx) const { return idx_ > _idx; }

	inline bool operator==(const OpenVolumeMeshHandle& _h) const { return _h.idx_ == this->idx_; }

	inline bool operator!=(const OpenVolumeMeshHandle& _h) const { return _h.idx_ != this->idx_; }

Mike Kremer's avatar
Mike Kremer committed
77
	inline const int& idx() const { return idx_; }
78

79
80
81
    /// return unsigned idx - handle must be valid
    inline size_t uidx() const { assert(is_valid()); return static_cast<size_t>(idx_); }

82
83
	void idx(const int& _idx) { idx_ = _idx; }

84
85
    OVM_DEPRECATED("use explicit .idx() instead")
    inline operator int() const { return idx_; }
86

87
88
89
90
91
92
	void reset() { idx_ = -1; }

private:
	int idx_;
};

93
94
template<typename EntityTag,
    typename = typename std::enable_if<is_entity<EntityTag>::value>::type>
95
96
97
98
99
100
101
102
103
104
105
class PropHandleTag {};

template <typename T> struct is_prop_handle_tag : std::false_type {};
template<typename T>
struct is_prop_handle_tag<PropHandleTag<T>> : std::true_type {};

template<typename T>
using is_handle_tag = std::enable_if<is_entity<T>::value || is_prop_handle_tag<T>::value>;


template<typename EntityTag, typename = typename is_handle_tag<EntityTag>::type>
106
107
108
109
class HandleT : public OpenVolumeMeshHandle
{
public:
    using Entity = EntityTag;
110
    explicit constexpr HandleT(int _idx = -1) : OpenVolumeMeshHandle(_idx) {}
111
112
113
114
115
116
117
118
119
120
121

    static HandleT<EntityTag>
    from_unsigned(size_t _idx)
    {
        if (_idx <= std::numeric_limits<int>::max()) {
            return HandleT<EntityTag>(static_cast<int>(_idx));
        } else {
            assert(false);
            return HandleT<EntityTag>(-1);
        }
    }
122
123
};

124
125
// Default entity handles

126
127
128
129
130
131
132
using VertexHandle   = HandleT<Entity::Vertex>;
using HalfEdgeHandle = HandleT<Entity::HalfEdge>;
using EdgeHandle     = HandleT<Entity::Edge>;
using HalfFaceHandle = HandleT<Entity::HalfFace>;
using FaceHandle     = HandleT<Entity::Face>;
using CellHandle     = HandleT<Entity::Cell>;
using MeshHandle     = HandleT<Entity::Mesh>;
133

134
135
136
137
138
// Helper class that is used to decrease all handles
// exceeding a certain threshold

class VHandleCorrection {
public:
Jan Möbius's avatar
Jan Möbius committed
139
    explicit VHandleCorrection(VertexHandle _thld) : thld_(_thld) {}
140
141
142
143
144
145
146
147
    void correctValue(VertexHandle& _h) {
        if(_h > thld_) _h.idx(_h.idx() - 1);
    }
private:
    VertexHandle thld_;
};
class HEHandleCorrection {
public:
Jan Möbius's avatar
Jan Möbius committed
148
    explicit HEHandleCorrection(HalfEdgeHandle _thld) : thld_(_thld) {}
149
    void correctVecValue(std::vector<HalfEdgeHandle>& _vec) {
Jan Möbius's avatar
Jan Möbius committed
150
#if defined(__clang_major__) && (__clang_major__ >= 5)
151
152
153
154
        for(std::vector<HalfEdgeHandle>::iterator it = _vec.begin(), end = _vec.end(); it != end; ++it) {
            correctValue(*it);
        }
#else
155
        std::for_each(_vec.begin(), _vec.end(), fun::bind(&HEHandleCorrection::correctValue, this, fun::placeholders::_1));
156
#endif
157
158
159
160
161
162
163
164
165
    }
    void correctValue(HalfEdgeHandle& _h) {
        if(_h > thld_) _h.idx(_h.idx() - 2);
    }
private:
    HalfEdgeHandle thld_;
};
class HFHandleCorrection {
public:
Jan Möbius's avatar
Jan Möbius committed
166
    explicit HFHandleCorrection(HalfFaceHandle _thld) : thld_(_thld) {}
167
    void correctVecValue(std::vector<HalfFaceHandle>& _vec) {
Jan Möbius's avatar
Jan Möbius committed
168
#if defined(__clang_major__) && (__clang_major__ >= 5)
169
170
171
172
        for(std::vector<HalfFaceHandle>::iterator it = _vec.begin(), end = _vec.end(); it != end; ++it) {
            correctValue(*it);
        }
#else
173
        std::for_each(_vec.begin(), _vec.end(), fun::bind(&HFHandleCorrection::correctValue, this, fun::placeholders::_1));
174
#endif
175
176
177
178
179
180
181
182
183
    }
    void correctValue(HalfFaceHandle& _h) {
        if(_h > thld_) _h.idx(_h.idx() - 2);
    }
private:
    HalfFaceHandle thld_;
};
class CHandleCorrection {
public:
Jan Möbius's avatar
Jan Möbius committed
184
    explicit CHandleCorrection(CellHandle _thld) : thld_(_thld) {}
185
186
187
188
189
190
191
    void correctValue(CellHandle& _h) {
        if(_h > thld_) _h.idx(_h.idx() - 1);
    }
private:
    CellHandle thld_;
};

192
193
194
195
196
197
198
199
bool operator==(const int& _lhs, const OpenVolumeMeshHandle& _rhs);

bool operator==(const unsigned int& _lhs, const OpenVolumeMeshHandle& _rhs);

bool operator!=(const int& _lhs, const OpenVolumeMeshHandle& _rhs);

bool operator!=(const unsigned int& _lhs, const OpenVolumeMeshHandle& _rhs);

200
201
202
203
std::ostream& operator<<(std::ostream& _ostr, const OpenVolumeMeshHandle& _handle);

std::istream& operator>>(std::istream& _istr, OpenVolumeMeshHandle& _handle);

204
205
} // Namespace OpenVolumeMesh