14#include <range/v3/view/transform.hpp>
16#include "AScrollAreaViewport.h"
17#include "AViewContainer.h"
18#include <AUI/Model/IListModel.h>
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>
25namespace aui::for_each_ui {
32using Key = std::size_t;
35 requires requires(T&
t) { std::hash<T> {}(
t); }
36constexpr aui::for_each_ui::Key defaultKey(
const T& value,
long primaryCandidate) {
37 return std::hash<T> {}(value);
41constexpr aui::for_each_ui::Key
42defaultKey(
const _<T>& value,
int secondaryCandidate) {
43 return reinterpret_cast<std::uintptr_t
>(value.get());
46template <ranges::input_range T>
47constexpr aui::for_each_ui::Key defaultKey(
const T& value,
int secondaryCandidate) {
48 auto key = std::hash<AByteBufferView>{}(AByteBufferView::fromRaw(value));
49 for (
const auto& i : value) {
52 key = (key << 1) - defaultKey(i, 0L);
62using ViewsSharedCache = AMap<aui::for_each_ui::Key, _<AView>>;
66class API_AUI_VIEWS AForEachUIBase :
public AViewContainerBase {
70 aui::for_each_ui::Key id;
76 void setPosition(glm::ivec2 position)
override;
81 List::iterator iterator;
89 void applyGeometryToChildren()
override;
114 void inflate(aui::for_each_ui::detail::InflateOpts opts = {});
115 glm::ivec2 calculateOffsetWithinViewportSlidingSurface();
116 glm::ivec2 axisMask();
117 void putOurViewsToSharedCache();
121namespace aui::detail {
122template <
typename Factory,
typename T>
123concept RangeFactory =
requires(Factory&& factory) {
124 { factory } -> aui::invocable;
125 { factory() } -> aui::range_consisting_of<T>;
241 friend class UIDeclarativeForTest;
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.");
248 using List = AForEachUIBase::List;
249 using ListFactory = std::function<List()>;
250 using ViewFactory = std::function<_<AView>(
const T& value)>;
253 ~AForEachUI()
override =
default;
255 template <aui::detail::RangeFactory<T> RangeFactory>
256 AForEachUI(RangeFactory&& rangeFactory) {
257 this->setModel(std::forward<RangeFactory>(rangeFactory));
260 template <aui::detail::RangeFactory<T> RangeFactory>
261 void setModel(RangeFactory&& rangeFactory) {
262 mListFactory = [
this, rangeFactory = std::forward<RangeFactory>(rangeFactory)] {
265 decltype(
auto) rng = rangeFactory();
266 if (
auto it = ranges::begin(rng); it != ranges::end(rng)) {
267 [[maybe_unused]]
auto discoverReferencedProperties = *it;
269 return rng | ranges::views::transform([
this](
const T& t) {
270 auto key = aui::for_each_ui::defaultKey(t, 0L);
272 if (mViewsSharedCache) {
273 if (
auto c = mViewsSharedCache->contains(key)) {
276 view = std::move(c->second);
277 mViewsSharedCache->erase(c);
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();
304 updateUnderlyingModel();
311 aui::for_each_ui::detail::ViewsSharedCache*
getViewsCache()
override {
return mViewsSharedCache; }
314 template <
typename FactoryTypeTag>
315 static aui::for_each_ui::detail::ViewsSharedCache VIEWS_SHARED_CACHE;
317 aui::for_each_ui::detail::ViewsSharedCache* mViewsSharedCache =
nullptr;
318 ListFactory mListFactory;
319 ViewFactory mFactory;
321 void updateUnderlyingModel() {
331template <
typename FactoryTypeTag>
332aui::for_each_ui::detail::ViewsSharedCache AForEachUI<T>::VIEWS_SHARED_CACHE {};
334namespace aui::detail {
336template <
typename Layout, aui::invocable RangeFactory>
337auto makeForEach(RangeFactory&& rangeFactory)
339 { rangeFactory() } -> ranges::range;
342 using ImmediateValueType =
decltype([&]() ->
decltype(
auto) {
343 decltype(
auto) rng = rangeFactory();
344 return *ranges::begin(rng);
374 static_assert(!std::is_reference_v<ImmediateValueType> || std::is_const_v<std::remove_reference_t<ImmediateValueType>>,
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.");
383 using T = std::decay_t<ImmediateValueType>;
385 auto result = _new<AForEachUI<T>>(std::forward<RangeFactory>(rangeFactory));
386 result->setLayout(std::make_unique<Layout>());
391#define AUI_DECLARATIVE_FOR_EX(value, model, layout, ...) \
392 aui::detail::makeForEach<layout>([=]() -> decltype(auto) { \
394 }) - [__VA_ARGS__](const auto& value) -> _<AView>
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 iterators.h:50