BioC++ core-0.7.0
The Modern C++ libraries for Bioinformatics.
 
Loading...
Searching...
No Matches
make_convertible.hpp
Go to the documentation of this file.
1// -----------------------------------------------------------------------------------------------------
2// Copyright (c) 2022 deCODE Genetics
3// Copyright (c) 2006-2020, Knut Reinert & Freie Universität Berlin
4// Copyright (c) 2016-2020, Knut Reinert & MPI für molekulare Genetik
5// This file may be used, modified and/or redistributed under the terms of the 3-clause BSD-License
6// shipped with this file and also available at: https://github.com/biocpp/biocpp-core/blob/main/LICENSE.md
7// -----------------------------------------------------------------------------------------------------
8
14#pragma once
15
16#include <algorithm>
17#include <concepts>
18#include <iterator>
19#include <ranges>
20#include <span>
21#include <type_traits>
22
30
31namespace bio::ranges::detail
32{
33
34// ============================================================================
35// view_take
36// ============================================================================
37
51template <std::ranges::view urng_t, bool exactly, bool or_throw>
52class view_take : public std::ranges::view_interface<view_take<urng_t, exactly, or_throw>>
53{
54private:
56 urng_t urange;
57
59 size_t target_size;
60
62 template <typename rng_t>
63 class basic_iterator;
64
65private:
70 using iterator = basic_iterator<urng_t>;
76 using const_iterator = basic_iterator<urng_t const>;
78
79public:
83 view_take() = default;
84 view_take(view_take const & rhs) = default;
85 view_take(view_take && rhs) noexcept = default;
86 view_take & operator=(view_take const & rhs) = default;
87 view_take & operator=(view_take && rhs) noexcept = default;
88 ~view_take() = default;
89
95 constexpr view_take(urng_t _urange, size_t const _size) : urange{std::move(_urange)}, target_size{_size}
96 {
97 if constexpr (std::ranges::sized_range<urng_t>)
98 {
99 if (std::ranges::size(urange) < target_size)
100 {
101 if constexpr (exactly && or_throw)
102 {
104 "You are trying to construct a views::take_exactly_or_throw from a range that is strictly "
105 "smaller."};
106 }
107 else
108 {
109 target_size = std::ranges::size(urange);
110 }
111 }
112 }
113 }
114
121 template <std::ranges::viewable_range rng_t>
122 requires std::constructible_from<rng_t, std::views::all_t<rng_t>>
123 constexpr view_take(rng_t && _urange, size_t const _size) :
124 view_take{std::views::all(std::forward<rng_t>(_urange)), _size}
125 {}
127
144 constexpr auto begin() noexcept
145 {
146 if constexpr (std::ranges::random_access_range<urng_t> && (std::ranges::sized_range<urng_t> || exactly))
147 return std::ranges::begin(urange);
148 else
149 return iterator{std::ranges::begin(urange), 0, target_size, this};
150 }
151
153 constexpr auto begin() const noexcept
154 requires const_iterable_range<urng_t>
155 {
156 if constexpr (std::ranges::random_access_range<urng_t> && (std::ranges::sized_range<urng_t> || exactly))
157 return std::ranges::cbegin(urange);
158 else
159 return const_iterator{std::ranges::cbegin(urange), 0, target_size};
160 }
161
175 constexpr auto end() noexcept
176 {
177 if constexpr (std::ranges::random_access_range<urng_t> && (std::ranges::sized_range<urng_t> || exactly))
178 return std::ranges::begin(urange) + target_size;
179 else
180 return std::ranges::end(urange);
181 }
182
184 constexpr auto end() const noexcept
185 requires const_iterable_range<urng_t>
186 {
187 if constexpr (std::ranges::random_access_range<urng_t> && (std::ranges::sized_range<urng_t> || exactly))
188 return std::ranges::cbegin(urange) + target_size;
189 else
190 return std::ranges::cend(urange);
191 }
193
209 constexpr auto size() const noexcept
210 requires exactly || std::ranges::sized_range<urng_t>
211 {
212 return target_size;
213 }
214};
215
218template <typename urng_t, bool exactly = false, bool or_throw = false>
219view_take(urng_t &&, size_t) -> view_take<std::views::all_t<urng_t>, exactly, or_throw>;
220
223template <std::ranges::view urng_t, bool exactly, bool or_throw>
224template <typename rng_t>
225class view_take<urng_t, exactly, or_throw>::basic_iterator :
226 public inherited_iterator_base<basic_iterator<rng_t>, std::ranges::iterator_t<rng_t>>
227{
228private:
230 using base_base_t = std::ranges::iterator_t<rng_t>;
232 using base_t = inherited_iterator_base<basic_iterator, std::ranges::iterator_t<rng_t>>;
233
235 using sentinel_type = std::ranges::sentinel_t<urng_t>;
236
238 size_t pos{};
239
241 size_t max_pos{};
242
245
246public:
251 basic_iterator() = default;
252 basic_iterator(basic_iterator const & rhs) = default;
253 basic_iterator(basic_iterator && rhs) noexcept = default;
254 basic_iterator & operator=(basic_iterator const & rhs) = default;
255 basic_iterator & operator=(basic_iterator && rhs) noexcept = default;
256 ~basic_iterator() = default;
257
259 constexpr basic_iterator(base_base_t const & it) noexcept(noexcept(base_t{it})) : base_t{std::move(it)} {}
260
262 constexpr basic_iterator(base_base_t it,
263 size_t const _pos,
264 size_t const _max_pos,
265 view_take * host = nullptr) noexcept(noexcept(base_t{it})) :
266 base_t{std::move(it)}, pos{_pos}, max_pos(_max_pos)
267 {
268 host_ptr = host;
269 }
271
278 using difference_type = std::iter_difference_t<base_base_t>;
280 using value_type = std::iter_value_t<base_base_t>;
282 using reference = std::iter_reference_t<base_base_t>;
284 using pointer = detail::iter_pointer_t<base_base_t>;
286 using iterator_category = detail::iterator_category_tag_t<base_base_t>;
288 using iterator_concept = detail::iterator_concept_tag_t<base_base_t>;
290
297 constexpr basic_iterator & operator++() noexcept(noexcept(++std::declval<base_t &>()))
298 {
299 base_t::operator++();
300 ++pos;
301 if constexpr (exactly && !std::forward_iterator<base_base_t>)
302 --host_ptr->target_size;
303 return *this;
304 }
305
307 constexpr basic_iterator operator++(int) noexcept(noexcept(++std::declval<basic_iterator &>()) &&
308 std::is_nothrow_copy_constructible_v<basic_iterator>)
309 {
310 basic_iterator cpy{*this};
311 ++(*this);
312 return cpy;
313 }
314
316 constexpr basic_iterator & operator--() noexcept(noexcept(--std::declval<base_base_t &>()))
317 requires std::bidirectional_iterator<base_base_t>
318 {
319 base_t::operator--();
320 --pos;
321 return *this;
322 }
323
325 constexpr basic_iterator operator--(int) noexcept(noexcept(--std::declval<basic_iterator &>()) &&
326 std::is_nothrow_copy_constructible_v<basic_iterator>)
327 requires std::bidirectional_iterator<base_base_t>
328 {
329 basic_iterator cpy{*this};
330 --(*this);
331 return cpy;
332 }
333
335 constexpr basic_iterator & operator+=(difference_type const skip) noexcept(noexcept(std::declval<base_t &>() +=
336 skip))
337 requires std::random_access_iterator<base_base_t>
338 {
339 base_t::operator+=(skip);
340 pos += skip;
341 return *this;
342 }
343
345 constexpr basic_iterator & operator-=(difference_type const skip) noexcept(noexcept(std::declval<base_t &>() -=
346 skip))
347 requires std::random_access_iterator<base_base_t>
348 {
349 base_t::operator-=(skip);
350 pos -= skip;
351 return *this;
352 }
354
361 constexpr bool operator==(basic_iterator const & rhs) const
362 noexcept(!or_throw && noexcept(std::declval<base_base_t &>() == std::declval<base_base_t &>()))
363 requires std::forward_iterator<base_base_t>
364 {
365 return *base_t::this_to_base() == *rhs.this_to_base();
366 }
367
369 constexpr bool operator==(sentinel_type const & rhs) const
370 noexcept(!or_throw && noexcept(std::declval<base_base_t const &>() == std::declval<sentinel_type const &>()))
371 {
372 if (pos >= max_pos)
373 return true;
374
375 if (*base_t::this_to_base() == rhs)
376 {
377 if constexpr (or_throw)
378 throw std::runtime_error{"Reached end of input before designated size."};
379
380 return true;
381 }
382 else
383 {
384 return false;
385 }
386 }
387
389 constexpr friend bool operator==(sentinel_type const & lhs,
390 basic_iterator const & rhs) noexcept(noexcept(rhs == lhs))
391 {
392 return rhs == lhs;
393 }
395
405 constexpr reference operator[](std::make_unsigned_t<difference_type> const n) const
406 noexcept(noexcept(std::declval<base_base_t &>()[0]))
407 requires std::random_access_iterator<base_base_t>
408 {
409 return base_base_t::operator[](n);
410 }
412};
413
414// ============================================================================
415// take_fn (adaptor definition)
416// ============================================================================
417
421template <bool exactly, bool or_throw>
422struct take_fn
423{
425 constexpr auto operator()(size_t const size) const { return adaptor_from_functor{*this, size}; }
426
430 template <std::ranges::range urng_t>
431 constexpr auto operator()(urng_t && urange, size_t target_size) const
432 {
433 static_assert(std::ranges::viewable_range<urng_t>,
434 "The views::take adaptor can only be passed viewable_ranges, i.e. Views or &-to-non-View.");
435
436 // safeguard against wrong size
437 if constexpr (std::ranges::sized_range<urng_t>)
438 {
439 if constexpr (or_throw)
440 {
441 if (target_size > std::ranges::size(urange))
442 {
444 "You are trying to construct a views::take_exactly_or_throw from a "
445 "range that is strictly smaller."};
446 }
447 }
448 else
449 {
450 target_size = std::min<size_t>(target_size, std::ranges::size(urange));
451 }
452 }
453
454 // string_view
455 if constexpr (meta::is_type_specialisation_of_v<std::remove_cvref_t<urng_t>, std::basic_string_view>)
456 {
457 return urange.substr(0, target_size);
458 }
459 // string const &
460 else if constexpr (meta::is_type_specialisation_of_v<std::remove_cvref_t<urng_t>, std::basic_string> &&
461 std::is_const_v<std::remove_reference_t<urng_t>>)
462 {
463 return std::basic_string_view{std::ranges::data(urange), target_size};
464 }
465 // contiguous
466 else if constexpr (std::ranges::borrowed_range<urng_t> && std::ranges::contiguous_range<urng_t> &&
467 std::ranges::sized_range<urng_t>)
468 {
469 return std::span{std::ranges::data(urange), target_size};
470 }
471 // random_access
472 else if constexpr (std::ranges::borrowed_range<urng_t> && std::ranges::random_access_range<urng_t> &&
473 std::ranges::sized_range<urng_t>)
474 {
475 return std::ranges::subrange<std::ranges::iterator_t<urng_t>, std::ranges::iterator_t<urng_t>>{
476 std::ranges::begin(urange),
477 std::ranges::begin(urange) + target_size,
478 target_size};
479 }
480 // our type
481 else
482 {
483 return view_take<std::views::all_t<urng_t>, exactly, or_throw>{std::forward<urng_t>(urange), target_size};
484 }
485 }
486};
487
488} // namespace bio::ranges::detail
T begin(T... args)
T end(T... args)
T forward(T... args)
constexpr auto size
A type trait that holds the size of a (semi-)alphabet.
Definition: concept.hpp:517
std::remove_cvref_t< decltype(std::ignore)> ignore_t
Return the type of std::ignore with const, volatile and references removed (type trait).
Definition: basic.hpp:48
Provides the bio::ranges::detail::inherited_iterator_base template.
T move(T... args)
Additional non-standard concepts for ranges.
Adaptations of concepts from the standard library.
Auxiliary header for the views submodule .
Provides various traits to inspect templates.
Provides bio::meta::transformation_trait_or.
Provides various transformation traits used by the range module.