Line data Source code
1 : /******************************************************************************
2 : * Project: GDAL Vector abstraction
3 : * Purpose:
4 : * Author: Javier Jimenez Shaw
5 : *
6 : ******************************************************************************
7 : * Copyright (c) 2024, Javier Jimenez Shaw
8 : *
9 : * SPDX-License-Identifier: MIT
10 : ****************************************************************************/
11 :
12 : #ifndef GDAL_VECTORX_H_INCLUDED
13 : #define GDAL_VECTORX_H_INCLUDED
14 :
15 : /*! @cond Doxygen_Suppress */
16 :
17 : #include <algorithm>
18 : #include <array>
19 : #include <functional>
20 :
21 : namespace gdal
22 : {
23 :
24 : /**
25 : * @brief Template class to abstract a vector
26 : *
27 : * Inspired by Eigen3 Vector class, but much simpler.
28 : * For GDAL internal use for now.
29 : *
30 : * @tparam T type of the values stored
31 : * @tparam N size of the container/vector
32 : */
33 : template <typename T, std::size_t N> class VectorX
34 : {
35 : public:
36 : using value_type = T;
37 : using size_type = std::size_t;
38 : using self_type = VectorX<T, N>;
39 :
40 : /** Size of the container */
41 7 : static constexpr size_type size() noexcept
42 : {
43 7 : return N;
44 : }
45 :
46 1776699 : VectorX()
47 1776699 : {
48 1776699 : }
49 :
50 : // cppcheck-suppress noExplicitConstructor
51 1121300 : template <typename... Args> VectorX(Args... args) : _values({args...})
52 : {
53 : static_assert(N == sizeof...(Args),
54 : "Invalid number of constructor params");
55 1121300 : }
56 :
57 : /** Container as an std::array */
58 1 : const std::array<T, N> &array() const
59 : {
60 1 : return _values;
61 : }
62 :
63 3046256 : T &operator[](const size_type &pos)
64 : {
65 3046256 : return _values[pos];
66 : }
67 :
68 1015782 : const T &operator[](const size_type &pos) const
69 : {
70 1015782 : return _values[pos];
71 : }
72 :
73 2134945 : T x() const
74 : {
75 : static_assert(N >= 1, "Invalid template size for x()");
76 2134945 : return _values[0];
77 : }
78 :
79 1796632 : T y() const
80 : {
81 : static_assert(N >= 2, "Invalid template size for y()");
82 1796632 : return _values[1];
83 : }
84 :
85 : T z() const
86 : {
87 : static_assert(N >= 3, "Invalid template size for z()");
88 : return _values[2];
89 : }
90 :
91 1521749 : T &x()
92 : {
93 : static_assert(N >= 1, "Invalid template size for x()");
94 1521749 : return _values[0];
95 : }
96 :
97 1268177 : T &y()
98 : {
99 : static_assert(N >= 2, "Invalid template size for y()");
100 1268177 : return _values[1];
101 : }
102 :
103 6 : T &z()
104 : {
105 : static_assert(N >= 3, "Invalid template size for z()");
106 6 : return _values[2];
107 : }
108 :
109 : /** Fill all elements of the vector with the same value
110 : * @return this
111 : */
112 2 : self_type &fill(T arg)
113 : {
114 8 : for (size_t i = 0; i < N; i++)
115 6 : (*this)[i] = arg;
116 :
117 2 : return *this;
118 : }
119 :
120 : /** Apply the unary operator to all the elements
121 : * @param op unary operator to apply to every element
122 : * @return a new object with the computed values
123 : */
124 253619 : template <class UnaryOp> self_type apply(UnaryOp op) const
125 : {
126 253619 : self_type res;
127 760857 : for (size_t i = 0; i < N; i++)
128 507238 : res[i] = op(_values[i]);
129 253619 : return res;
130 : }
131 :
132 253616 : self_type floor() const
133 : {
134 760848 : return apply([](const value_type &v) { return std::floor(v); });
135 : }
136 :
137 1 : self_type ceil() const
138 : {
139 3 : return apply([](const value_type &v) { return std::ceil(v); });
140 : }
141 :
142 : /** Compute the scalar product of two vectors */
143 2 : T scalarProd(const self_type &arg) const
144 : {
145 2 : T accum{};
146 6 : for (size_t i = 0; i < N; i++)
147 4 : accum += _values[i] * arg[i];
148 2 : return accum;
149 : }
150 :
151 : /** Compute the norm squared of the vector */
152 1 : T norm2() const
153 : {
154 1 : return scalarProd(*this);
155 : }
156 :
157 : /**
158 : * @brief cast the type to a different one, and convert the elements
159 : *
160 : * @tparam U output value type
161 : * @return VectorX<U, N> object of new type, where all elements are casted
162 : */
163 507955 : template <typename U> VectorX<U, N> cast() const
164 : {
165 507955 : VectorX<U, N> res;
166 1523865 : for (size_t i = 0; i < N; i++)
167 1015910 : res[i] = static_cast<U>(_values[i]);
168 507955 : return res;
169 : }
170 :
171 1 : self_type operator+(T arg) const
172 : {
173 1 : return operatorImpl(arg, std::plus());
174 : }
175 :
176 2 : self_type &operator+=(T arg)
177 : {
178 2 : return operatorEqImpl(arg, std::plus());
179 : }
180 :
181 253656 : self_type operator-(T arg) const
182 : {
183 253656 : return operatorImpl(arg, std::minus());
184 : }
185 :
186 2 : self_type &operator-=(T arg)
187 : {
188 2 : return operatorEqImpl(arg, std::minus());
189 : }
190 :
191 1 : self_type operator-() const
192 : {
193 3 : return apply([](const value_type &v) { return -v; });
194 : }
195 :
196 1 : template <typename U> self_type operator*(U arg) const
197 : {
198 1 : self_type res;
199 3 : for (size_t i = 0; i < N; i++)
200 2 : res[i] = T(_values[i] * arg);
201 1 : return res;
202 : }
203 :
204 1 : self_type operator*(T arg) const
205 : {
206 1 : return operatorImpl(arg, std::multiplies());
207 : }
208 :
209 1 : template <typename U> self_type operator/(U arg) const
210 : {
211 1 : self_type res;
212 3 : for (size_t i = 0; i < N; i++)
213 2 : res[i] = T(_values[i] / arg);
214 1 : return res;
215 : }
216 :
217 1 : self_type operator/(T arg) const
218 : {
219 1 : return operatorImpl(arg, std::divides());
220 : }
221 :
222 1 : self_type operator+(const self_type &arg) const
223 : {
224 1 : return operatorImpl(arg, std::plus());
225 : }
226 :
227 507885 : self_type operator-(const self_type &arg) const
228 : {
229 507885 : return operatorImpl(arg, std::minus());
230 : }
231 :
232 1 : friend VectorX<T, N> operator+(T t, const VectorX<T, N> &arg)
233 : {
234 1 : VectorX<T, N> res;
235 3 : for (size_t i = 0; i < N; i++)
236 2 : res._values[i] = t + arg._values[i];
237 1 : return res;
238 : }
239 :
240 253570 : friend VectorX<T, N> operator-(T t, const VectorX<T, N> &arg)
241 : {
242 253570 : VectorX<T, N> res;
243 760710 : for (size_t i = 0; i < N; i++)
244 507140 : res._values[i] = t - arg._values[i];
245 253570 : return res;
246 : }
247 :
248 : private:
249 : std::array<T, N> _values{};
250 :
251 253659 : template <class Op> self_type operatorImpl(T arg, Op op) const
252 : {
253 253659 : self_type res;
254 760977 : for (size_t i = 0; i < N; i++)
255 507318 : res[i] = op(_values[i], arg);
256 253659 : return res;
257 : }
258 :
259 4 : template <class Op> self_type &operatorEqImpl(T arg, Op op)
260 : {
261 12 : for (size_t i = 0; i < N; i++)
262 8 : _values[i] = op(_values[i], arg);
263 4 : return *this;
264 : }
265 :
266 : template <class Op>
267 507886 : self_type operatorImpl(const self_type &arg, Op op) const
268 : {
269 507886 : self_type res;
270 1523658 : for (size_t i = 0; i < N; i++)
271 1015772 : res[i] = op(_values[i], arg[i]);
272 507886 : return res;
273 : }
274 :
275 : template <class Op> self_type &operatorEqImpl(const self_type &arg, Op op)
276 : {
277 : for (size_t i = 0; i < N; i++)
278 : _values[i] = op(_values[i], arg[i]);
279 : return *this;
280 : }
281 : };
282 :
283 : using Vector2d = VectorX<double, 2>;
284 : using Vector2i = VectorX<int, 2>;
285 : using Vector3d = VectorX<double, 3>;
286 : using Vector3i = VectorX<int, 3>;
287 : } // namespace gdal
288 :
289 : /*! @endcond */
290 :
291 : #endif /* ndef GDAL_VECTORX_H_INCLUDED */
|