AUI Framework  master
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
any_view.h
    1/*
    2 * AUI Framework - Declarative UI toolkit for modern C++20
    3 * Copyright (C) 2020-2025 Alex2772 and Contributors
    4 *
    5 * SPDX-License-Identifier: MPL-2.0
    6 *
    7 * This Source Code Form is subject to the terms of the Mozilla Public
    8 * License, v. 2.0. If a copy of the MPL was not distributed with this
    9 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
   10 */
   11
   12#pragma once
   13
   14#include <AUI/Common/AException.h>
   15#include <range/v3/range/concepts.hpp>
   16
   17
   18
   19namespace aui {
   20template<typename Rng, typename T>
   21concept range_consisting_of = requires(Rng&& rng) {
   22    { rng } -> ranges::range;
   23    { std::move(*::ranges::begin(rng)) } -> aui::convertible_to<T>;
   24};
   25}
   26
   27namespace aui {
   28
   30    bool implementsOperatorMinusMinus;
   31};
   32
   63template<typename T>
   64struct any_view {
   65    struct iterator {
   66        using value_type = T;
   67        using iterator_category = std::input_iterator_tag;
   68        using difference_type = std::ptrdiff_t;
   69
   70        template<ranges::input_range Rng, typename Iterator>
   71        iterator(Rng&& rng, Iterator&& it): mImpl(rttify(std::forward<Rng>(rng), std::forward<Iterator>(it))) {
   72
   73        }
   74        iterator(iterator&& rhs) noexcept = default;
   75        iterator& operator=(iterator&& rhs) noexcept = default;
   76
   77        iterator() = default;
   78
   79        iterator(iterator& rhs) {
   80            operator=(rhs);
   81        }
   82        iterator(const iterator& rhs) {
   83            operator=(rhs);
   84        }
   85        iterator& operator=(const iterator& rhs) {
   86            if (this == &rhs) {
   87                return *this;
   88            }
   89            if (rhs.mImpl) {
   90                mImpl = rhs.mImpl->clone();
   91            }
   92            return *this;
   93        }
   94
   95        iterator& operator++() {
   96            mImpl->next();
   97            return *this;
   98        }
   99
  100        iterator& operator++(int) { // not accurate
  101            mImpl->next();
  102            return *this;
  103        }
  104
  105        iterator& operator--() {
  106            mImpl->prev();
  107            return *this;
  108        }
  109
  110        iterator& operator--(int) { // not accurate
  111            mImpl->prev();
  112            return *this;
  113        }
  114
  115        T operator*() const {
  116            return mImpl->value();
  117        }
  118
  119        bool operator==(const iterator& rhs) const {
  120            return mImpl->isEquals(rhs.mImpl.get());
  121        }
  122
  123        bool operator!=(const iterator& rhs) const {
  124            return !mImpl->isEquals(rhs.mImpl.get());
  125        }
  126
  127    private:
  128        struct iface {
  129            virtual ~iface() = default;
  130            virtual void prev() = 0;
  131            virtual void next() = 0;
  132            virtual T value() = 0;
  133            virtual std::unique_ptr<iface> clone() = 0;
  134            virtual bool isEquals(iface* rhs) = 0;
  135        };
  136        std::unique_ptr<iface> mImpl;
  137
  138    template<ranges::input_range Rng, typename Iterator>
  139    std::unique_ptr<iface> rttify(Rng&& rng, Iterator&& it) {
  140        struct rttified: iface {
  141            rttified(std::remove_reference_t<Rng>& rng, std::decay_t<Iterator> it): rng(rng), it(std::move(it)) {}
  142            ~rttified() = default;
  143
  144            void prev() override {
  145                if constexpr (requires { --it; }) {
  146                    --it;
  147                } else {
  148                    throw AException("unimplemented");
  149                }
  150            }
  151
  152            void next() override {
  153                 ++it;
  154            }
  155
  156            T value() override {
  157                if constexpr (std::is_copy_constructible_v<T>) {
  158                    return *it;
  159                }
  160                throw AException("can't make a copy");
  161            }
  162
  163            std::unique_ptr<iface> clone() override {
  164                return std::make_unique<rttified>(rng, it);
  165            }
  166
  167            bool isEquals(iface* rhs) override {
  168                if (rhs == nullptr) {
  169                    // end sentinel; ask rng for end by ourselves
  170                    return it == ranges::end(rng);
  171                }
  172                if (auto t = dynamic_cast<rttified*>(rhs)) {
  173                     return it == t->it;
  174                }
  175                return false;
  176            }
  177
  178        private:
  179            std::decay_t<Iterator> it;
  180            std::remove_reference_t<Rng>& rng;
  181        };
  182
  183        return std::make_unique<rttified>(rng, it);
  184    }
  185    };
  186
  187    template<typename Rng>
  188    any_view(Rng&& rng): mImpl(rttify(std::forward<Rng>(rng))) {
  189
  190    }
  191
  192    any_view(): mImpl(nullptr) {}
  193
  194    any_view(any_view&& rhs) noexcept = default;
  195    any_view& operator=(any_view&& rhs) noexcept = default;
  196
  197    any_view(any_view& rhs) {
  198        operator=(rhs);
  199    }
  200    any_view(const any_view& rhs) {
  201        operator=(rhs);
  202    }
  203
  204    any_view& operator=(any_view& rhs) {
  205        return operator=(const_cast<const any_view&>(rhs));
  206    }
  207
  208    any_view& operator=(const any_view& rhs) {
  209        if (this == &rhs) {
  210            return *this;
  211        }
  212        mImpl = rhs.mImpl->clone();
  213        return *this;
  214    }
  215
  216    iterator begin() const {
  217        return mImpl->begin();
  218    }
  219
  220    iterator end() const {
  221        return mImpl->end();
  222    }
  223
  224    dyn_range_capabilities capabilities() const {
  225        if (!mImpl) {
  226            return {};
  227        }
  228        return mImpl->capabilities();
  229    }
  230
  231private:
  232    struct iface {
  233        virtual ~iface() = default;
  234
  235        virtual iterator begin() = 0;
  236        virtual iterator end() = 0;
  237        virtual std::unique_ptr<iface> clone() = 0;
  238        virtual dyn_range_capabilities capabilities() = 0;
  239    };
  240
  241    std::unique_ptr<iface> mImpl;
  242
  243    template<aui::range_consisting_of<T> Rng>
  244    std::unique_ptr<iface> rttify(Rng&& rng) {
  245        static_assert(!requires() { rng.mImpl; }, "rttifying itself?");
  246
  247        struct rttified: iface {
  248        public:
  249            rttified(Rng f): rng(std::forward<Rng>(f)) {}
  250            ~rttified() = default;
  251
  252            iterator begin() override {
  253                return { rng, ranges::begin(rng) };
  254
  255            }
  256
  257            iterator end() override {
  258                if constexpr (std::is_same_v<decltype(ranges::begin(rng)), decltype(ranges::end(rng))>) {
  259                    return { rng, ranges::end(rng) };
  260                } else {
  261                    // sentinel?
  262                    return {};
  263                }
  264            }
  265
  266            std::unique_ptr<iface> clone() override {
  267                if constexpr (std::is_copy_constructible_v<T>) {
  268                    return std::make_unique<rttified>(rng);
  269                }
  270                throw AException("can't make a copy of this range");
  271            }
  272
  273            dyn_range_capabilities capabilities() override {
  274                return {
  275                    .implementsOperatorMinusMinus = requires() { --++rng.begin(); },
  276                };
  277            }
  278
  279        private:
  280            Rng rng;
  281        };
  282        return std::make_unique<rttified>(std::forward<Rng>(rng));
  283    }
  284};
  285
  286static_assert(ranges::input_iterator<any_view<int>::iterator>);
  287static_assert(ranges::sentinel_for<any_view<int>::iterator, any_view<int>::iterator>);
  288static_assert(ranges::range<any_view<int>>);
  289
  290}
  291
Abstract AUI exception.
Definition AException.h:28
Definition concepts.h:42
Definition any_view.h:21
Definition any_view.h:65
RTTI-wrapped range.
Definition any_view.h:64
Definition any_view.h:29