AUI Framework  master
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
APropertyPrecomputed.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 "AObjectBase.h"
   15#include "ASignal.h"
   16#include "AUI/Util/AEvaluationLoopException.h"
   17#include <AUI/Common/detail/property.h>
   18#include <AUI/Common/React.h>
   19
   50template<typename T>
   51struct APropertyPrecomputed final : aui::react::DependencyObserver {
   52    using Underlying = T;
   53    using Factory = std::function<T()>;
   54
   55    template<aui::factory<T> Factory>
   56    APropertyPrecomputed(Factory&& expression): mCurrentValue([this, expression = std::forward<Factory>(expression)] { // NOLINT(*-explicit-constructor)
   57      aui::react::DependencyObserverRegistrar r(*this);
   58      return expression();
   59    }) {
   60
   61    }
   62
   63    APropertyPrecomputed(const APropertyPrecomputed&) = delete;
   64    APropertyPrecomputed(APropertyPrecomputed&&) noexcept = delete;
   65
   66    template <ASignalInvokable SignalInvokable>
   67    void operator^(SignalInvokable&& t) {
   68        t.invokeSignal(nullptr);
   69    }
   70
   77    void invalidate() override {
   78        mCurrentValue.reset();
   79        if (this->changed) {
   80            if (this->changed.isAtSignalEmissionState()) {
   81                mCurrentValue.setEvaluationLoopTrap();
   82                return;
   83            }
   84            emit this->changed(value());
   85        }
   86    }
   87
   88    AObjectBase* boundObject() {
   89        return this;
   90    }
   91
   92    [[nodiscard]]
   93    const T& value() const {
   95        return mCurrentValue;
   96    }
   97
   98    [[nodiscard]]
   99    operator const T&() const {
  100        return value();
  101    }
  102
  103    [[nodiscard]]
  104    const T& operator*() const {
  105        return value();
  106    }
  107
  111    template <aui::invocable<const Underlying&> Projection>
  112    [[nodiscard]]
  113    auto readProjected(Projection&& projection) noexcept {
  114        return aui::detail::property::makeReadonlyProjection(*this, std::forward<Projection>(projection));
  115    }
  116
  117    [[nodiscard]]
  118    const T* operator->() const noexcept {
  119        return &value();
  120    }
  121
  122signals:
  123    emits<T> changed;
  124
  125private:
  126    aui::lazy<T> mCurrentValue;
  127};
  128
  129template<aui::invocable<> Factory>
  131
  132static_assert(APropertyReadable<APropertyPrecomputed<int>>, "APropertyPrecomputed must be a APropertyReadable");
Definition AObjectBase.h:24
Definition Factory.h:18
Definition concepts.h:195
ASignal< Args... > emits
A signal declaration.
Definition ASignal.h:572
#define emit
emits the specified signal in context of this object.
Definition AObject.h:343
Readonly property that holds a value computed by an expression.
Definition APropertyPrecomputed.h:51
auto readProjected(Projection &&projection) noexcept
Makes a readonly projection of this property.
Definition APropertyPrecomputed.h:113
void invalidate() override
Marks this precomputed property to be reevaluated.
Definition APropertyPrecomputed.h:77
static void addDependency(const AAbstractSignal &signal)
Adds observer to the specified signal, if called inside a reactive expression evaluation.
Definition React.h:23