AUI Framework  master
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
AForEachUI.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 <range/v3/view/transform.hpp>
   15
   16#include "AScrollAreaViewport.h"
   17#include "AViewContainer.h"
   18#include <AUI/Model/IListModel.h>
   19#include <functional>
   20#include <AUI/Model/AListModelAdapter.h>
   21#include <AUI/Util/ADataBinding.h>
   22#include <AUI/Platform/AWindow.h>
   23#include <AUI/Traits/any_view.h>
   24
   25namespace aui::for_each_ui {
   26
   32using Key = std::size_t;
   33
   34template <typename T>
   35    requires requires(T& t) { std::hash<T> {}(t); }
   36constexpr aui::for_each_ui::Key defaultKey(const T& value, long primaryCandidate) {   // std::hash specialization
   37    return std::hash<T> {}(value);
   38}
   39
   40template <typename T>
   41constexpr aui::for_each_ui::Key
   42defaultKey(const _<T>& value, int secondaryCandidate) {   // specialization for shared pointers
   43    return reinterpret_cast<std::uintptr_t>(value.get());
   44}
   45
   46template <ranges::input_range T>
   47constexpr aui::for_each_ui::Key defaultKey(const T& value, int secondaryCandidate) {   // specialization for subranges
   48    auto key = std::hash<AByteBufferView>{}(AByteBufferView::fromRaw(value));
   49    for (const auto& i : value) {
   50        // sub produces order sensitive hash.
   51        // lshift distinguishes equal hashes.
   52        key = (key << 1) - defaultKey(i, 0L);
   53    }
   54    return key;
   55}
   56
   57namespace detail {
   58struct InflateOpts {
   59    bool backward = true;
   60    bool forward = true;
   61};
   62using ViewsSharedCache = AMap<aui::for_each_ui::Key, _<AView>>;
   63}   // namespace detail
   64}   // namespace aui::for_each_ui
   65
   66class API_AUI_VIEWS AForEachUIBase : public AViewContainerBase {
   67public:
   68    struct Entry {
   69        _<AView> view;
   70        aui::for_each_ui::Key id;
   71    };
   72
   73    using List = aui::any_view<Entry>;
   74    AForEachUIBase() {}
   75    ~AForEachUIBase() override = default;
   76    void setPosition(glm::ivec2 position) override;
   77
   78protected:
   79    struct Cache {
   80        struct LazyListItemInfo : Entry {
   81            List::iterator iterator;
   82        };
   83        AVector<LazyListItemInfo> items;
   84    };
   85
   86    AOptional<Cache> mCache;
   87
   88    void onViewGraphSubtreeChanged() override;
   89    void applyGeometryToChildren() override;
   90
   94    void setModelImpl(List model);
   95
  103    virtual aui::for_each_ui::detail::ViewsSharedCache* getViewsCache() = 0;
  104
  105private:
  106    _weak<AScrollAreaViewport> mViewport;
  107    List mViewsModel;
  108    aui::dyn_range_capabilities mViewsModelCapabilities;
  109    AOptional<glm::ivec2> mLastInflatedScroll {};
  110
  111    void addView(List::iterator iterator, AOptional<std::size_t> index = std::nullopt);
  112    void removeViews(aui::range<AVector<_<AView>>::iterator> iterators);
  113
  114    void inflate(aui::for_each_ui::detail::InflateOpts opts = {});
  115    glm::ivec2 calculateOffsetWithinViewportSlidingSurface();
  116    glm::ivec2 axisMask();
  117    void putOurViewsToSharedCache();
  118
  119};
  120
  121namespace aui::detail {
  122template <typename Factory, typename T>
  123concept RangeFactory = requires(Factory&& factory) {
  124    { factory } -> aui::invocable;
  125    { factory() } -> aui::range_consisting_of<T>;
  126};
  127}   // namespace aui::detail
  128
  129
  238template <typename T>
  239class AForEachUI : public AForEachUIBase, public aui::react::DependencyObserver {
  240public:
  241    friend class UIDeclarativeForTest;
  242
  243    static_assert(
  244        requires(T& t) { aui::for_each_ui::defaultKey(t, 0L); },
  245        "// ====================> AForEachUI: aui::for_each_ui::defaultKey overload or std::hash specialization is "
  246        "required for your type.");
  247
  248    using List = AForEachUIBase::List;
  249    using ListFactory = std::function<List()>;
  250    using ViewFactory = std::function<_<AView>(const T& value)>;
  251
  252    AForEachUI() {}
  253    ~AForEachUI() override = default;
  254
  255    template <aui::detail::RangeFactory<T> RangeFactory>
  256    AForEachUI(RangeFactory&& rangeFactory) {
  257        this->setModel(std::forward<RangeFactory>(rangeFactory));
  258    }
  259
  260    template <aui::detail::RangeFactory<T> RangeFactory>
  261    void setModel(RangeFactory&& rangeFactory) {
  262        mListFactory = [this, rangeFactory = std::forward<RangeFactory>(rangeFactory)] {
  263//            ALOG_DEBUG("AForEachUIBase") << this << "(" << AReflect::name(this) << ") range expression evaluation";
  264            aui::react::DependencyObserverRegistrar r(*this);
  265            decltype(auto) rng = rangeFactory();
  266            if (auto it = ranges::begin(rng); it != ranges::end(rng)) {
  267                [[maybe_unused]] auto discoverReferencedProperties = *it;
  268            }
  269            return rng | ranges::views::transform([this](const T& t) {
  270                       auto key = aui::for_each_ui::defaultKey(t, 0L);
  271                       _<AView> view;
  272                       if (mViewsSharedCache) {
  273                           if (auto c = mViewsSharedCache->contains(key)) {
  274//                               ALOG_DEBUG("AForEachUIBase")
  275//                                   << this << "(" << AReflect::name(this) << ") Trying to view from cache: " << key;
  276                               view = std::move(c->second);
  277                               mViewsSharedCache->erase(c);
  278                           }
  279                       }
  280                       if (!view) {
  281                           view = mFactory(t);
  282                       }
  283                       return AForEachUIBase::Entry { .view = std::move(view), .id = key };
  284                   });
  285        };
  286    }
  287
  292    template <aui::invocable<const T&> ViewFactoryT>
  293    void operator-(ViewFactoryT&& f) {
  294        mFactory = std::forward<ViewFactoryT>(f);
  295        mViewsSharedCache = &VIEWS_SHARED_CACHE<ViewFactoryT>;
  296        updateUnderlyingModel();
  297    }
  298
  302    void invalidate() override {
  303//        ALOG_DEBUG("AForEachUIBase") << this << "(" << AReflect::name(this) << ") invalidate";
  304        updateUnderlyingModel();
  305    }
  306
  307    using AViewContainerBase::setLayout;
  308
  309protected:
  310
  311    aui::for_each_ui::detail::ViewsSharedCache* getViewsCache() override { return mViewsSharedCache; }
  312
  313private:
  314    template <typename FactoryTypeTag>
  315    static aui::for_each_ui::detail::ViewsSharedCache VIEWS_SHARED_CACHE;
  316
  317    aui::for_each_ui::detail::ViewsSharedCache* mViewsSharedCache = nullptr;
  318    ListFactory mListFactory;
  319    ViewFactory mFactory;
  320
  321    void updateUnderlyingModel() {
  322        if (!mFactory) {
  323            return;
  324        }
  325
  326        this->setModelImpl(mListFactory());
  327    }
  328};
  329
  330template <typename T>
  331template <typename FactoryTypeTag>
  332aui::for_each_ui::detail::ViewsSharedCache AForEachUI<T>::VIEWS_SHARED_CACHE {};
  333
  334namespace aui::detail {
  335
  336template <typename Layout, aui::invocable RangeFactory>
  337auto makeForEach(RangeFactory&& rangeFactory)
  338    requires requires {
  339        { rangeFactory() } -> ranges::range;
  340    }
  341{
  342    using ImmediateValueType = decltype([&]() -> decltype(auto) {
  343        decltype(auto) rng = rangeFactory();
  344        return *ranges::begin(rng);
  345    }());
  346
  347    // | is_reference<ImmediateValueType> | is_const<ImmediateValueType> |   |
  348    // | -------------------------------- | ---------------------------- | - |
  349    // | 0                                | 0                            | 1 |
  350    // | 0                                | 1                            | 1 |
  351    // | 1                                | 0                            | 0 |
  352    // | 1                                | 1                            | 1 |
  353    //
  354    // This ensures that a borrowed container is constant, and throws a compile-time diagnostics if is not in most
  355    // scenarios:
  356    // ```cpp
  357    // class MyWindow {
  358    //   void test() {
  359    //     AUI_DECLARATIVE_FOR(i, ints, ... // error; mark ints as const
  360    //   }
  361    //   AVector<int> ints = { 1, 2, 3 };
  362    // };
  363    // ```
  364    //
  365    // The only known case where this check does not work:
  366    // ```cpp
  367    // class MyWindow {
  368    //   void test() {
  369    //     AUI_DECLARATIVE_FOR(i, ints | ranges::views::transform([](int i) { return true; }), ... // bad but compiles
  370    //   }
  371    //   AVector<int> ints = { 1, 2, 3 };
  372    // };
  373    // ```
  374    static_assert(!std::is_reference_v<ImmediateValueType> || std::is_const_v<std::remove_reference_t<ImmediateValueType>>,
  375                  "\n"
  376                  "====================> AUI_DECLARATIVE_FOR: attempt to use a non-const reference. You can:\n"
  377                  "====================> (1) transfer ownership of the container to AUI_DECLARATIVE_FOR by referencing a local, or\n"
  378                  "====================> (2) define your container as const field and manually make sure its lifetime exceeds "
  379                  "AUI_DECLARATIVE_FOR's, or\n"
  380                  "====================> (3) wrap your container as AProperty.\n"
  381                  "====================> Please consult with https://aui-framework.github.io/develop/classAForEachUI.html#AFOREACHUI_UPDATE for more info.");
  382
  383    using T = std::decay_t<ImmediateValueType>;
  384
  385    auto result = _new<AForEachUI<T>>(std::forward<RangeFactory>(rangeFactory));
  386    result->setLayout(std::make_unique<Layout>());
  387    return result;
  388}
  389}   // namespace aui::detail
  390
  391#define AUI_DECLARATIVE_FOR_EX(value, model, layout, ...)      \
  392    aui::detail::makeForEach<layout>([=]() -> decltype(auto) { \
  393        return (model);                                        \
  394    }) - [__VA_ARGS__](const auto& value) -> _<AView>
  395
  402#define AUI_DECLARATIVE_FOR(value, model, layout) AUI_DECLARATIVE_FOR_EX(value, model, layout, =)
Definition AForEachUI.h:66
void onViewGraphSubtreeChanged() override
Called when direct or indirect parent has changed.
virtual aui::for_each_ui::detail::ViewsSharedCache * getViewsCache()=0
Returns a cache of views, if any.
void setModelImpl(List model)
Notifies that range was changed or iterators might have invalidated.
aui::for_each_ui::detail::ViewsSharedCache * getViewsCache() override
Returns a cache of views, if any.
Definition AForEachUI.h:311
void invalidate() override
Notifies that range was changed or iterators might have invalidated.
Definition AForEachUI.h:302
Utility wrapper implementing the stack-allocated (fast) optional idiom.
Definition AOptional.h:33
A std::vector with AUI extensions.
Definition AVector.h:39
void setLayout(_unique< ALayout > layout)
Set new layout manager for this AViewContainerBase. DESTROYS OLD LAYOUT MANAGER WITH ITS VIEWS!...
void addView(const _< AView > &view)
Add view to the container.
void removeViews(aui::range< AVector< _< AView > >::iterator > views)
Remove views from the container.
An std::weak_ptr with AUI extensions.
Definition SharedPtrTypes.h:179
type_of< T > t
Selects views that are of the specified C++ types.
Definition type_of.h:71
Definition AForEachUI.h:80
Definition AForEachUI.h:79
Definition AForEachUI.h:68
An std::weak_ptr with AUI extensions.
Definition SharedPtrTypes.h:52
RTTI-wrapped range.
Definition any_view.h:64
Definition any_view.h:29
Definition iterators.h:50
Definition React.h:23