AUI Framework  master
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
AProperty.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/ASignal.h>
   15#include <AUI/Common/APropertyPrecomputed.h>
   16#include <AUI/Common/PropertyModifier.h>
   17#include <AUI/Common/detail/property.h>
   18
   29template <typename T>
   30struct AProperty: AObjectBase {
   31    static_assert(!std::is_reference_v<T>, "====================> AProperty: attempt to wrap a reference.");
   32
   33    using Underlying = T;
   34
   42    T raw{};
   43
   48
   49    AProperty()
   50        requires aui::default_initializable<T>
   51    = default;
   52
   53    template <aui::convertible_to<T> U>
   54    AProperty(U&& value) noexcept(noexcept(T(std::forward<U>(value)))): raw(std::forward<U>(value)) {}
   55
   56    AObjectBase* boundObject() {
   57        return this;
   58    }
   59
   60    const AObjectBase* boundObject() const {
   61        return this;
   62    }
   63
   64    AProperty(const AProperty& value): raw(value.raw) {
   65    }
   66
   67    AProperty(AProperty&& value) noexcept: raw(std::move(value.raw)) {
   68        value.notify();
   69    }
   70
   71    AProperty& operator=(const AProperty& value) {
   72        if (this == &value) {
   73            return *this;
   74        }
   75        operator=(value.raw);
   76        return *this;
   77    }
   78
   79    AProperty& operator=(AProperty&& value) noexcept {
   80        if (this == &value) {
   81            return *this;
   82        }
   83        operator=(std::move(value.raw));
   84        value.notify();
   85        return *this;
   86    }
   87
   88    template <aui::convertible_to<T> U>
   89    AProperty& operator=(U&& value) noexcept {
   90        static constexpr auto IS_COMPARABLE = requires { this->raw == value; };
   91        if constexpr (IS_COMPARABLE) {
   92            if (this->raw == value) [[unlikely]] {
   93                return *this;
   94            }
   95        }
   96        this->raw = std::forward<U>(value);
   97        notify();
   98        return *this;
   99    }
  100
  101    template <ASignalInvokable SignalInvokable>
  102    void operator^(SignalInvokable&& t) {
  103        t.invokeSignal(nullptr);
  104    }
  105
  116    void notify() {
  117        emit changed(this->raw);
  118    }
  119
  120    [[nodiscard]]
  121    const T& value() const noexcept {
  123        return raw;
  124    }
  125
  126    [[nodiscard]] operator const T&() const noexcept { return value(); }
  127
  128    [[nodiscard]]
  129    const T* operator->() const noexcept {
  130        return &value();
  131    }
  132
  133    [[nodiscard]]
  134    const T& operator*() const noexcept {
  135        return value();
  136    }
  137
  142        return { *this };
  143    }
  144
  148    template<aui::invocable<const T&> Projection>
  149    [[nodiscard]]
  150    auto readProjected(Projection&& projection) const noexcept {
  151        return aui::detail::property::makeReadonlyProjection(*this, std::forward<Projection>(projection));
  152    }
  153
  157    template<aui::invocable<const T&> ProjectionRead,
  158             aui::invocable<const std::invoke_result_t<ProjectionRead, T>&> ProjectionWrite>
  159    [[nodiscard]]
  160    auto biProjected(ProjectionRead&& projectionRead, ProjectionWrite&& projectionWrite) noexcept {
  161        return aui::detail::property::makeBidirectionalProjection(*this,
  162                                                                  std::forward<ProjectionRead>(projectionRead),
  163                                                                  std::forward<ProjectionWrite>(projectionWrite));
  164    }
  165
  169    template<aui::detail::property::ProjectionBidirectional<T> Projection>
  170    [[nodiscard]]
  171    auto biProjected(Projection&& projectionBidirectional) noexcept {
  172        return aui::detail::property::makeBidirectionalProjection(*this, projectionBidirectional);
  173    }
  174
  175private:
  176    friend class AObject;
  180    [[nodiscard]]
  181    auto assignment() noexcept {
  182        return aui::detail::property::makeAssignment(*this);
  183    }
  184};
  185static_assert(AAnyProperty<AProperty<int>>, "AProperty does not conform AAnyProperty concept");
  186static_assert(AAnyProperty<AProperty<int>&>, "AProperty does not conform AAnyProperty concept");
  187static_assert(APropertyReadable<const AProperty<int>>, "const AProperty does not conform AAnyProperty concept");
  188
  189template<typename T>
  191public:
  192    using Underlying = T;
  193    PropertyModifier(AProperty<T>& owner): mOwner(&owner) {}
  194    ~PropertyModifier() {
  195        if (mOwner == nullptr) {
  196            return;
  197        }
  198        mOwner->notify();
  199    }
  200
  201    [[nodiscard]]
  202    Underlying& value() const noexcept {
  203        return mOwner->raw;
  204    }
  205
  206    [[nodiscard]]
  207    Underlying* operator->() const noexcept {
  208        return &value();
  209    }
  210
  211private:
  212    AProperty<T>* mOwner;
  213};
  214
  215
  231template <
  233    typename SignalArg>
  234struct APropertyDef {
  238    const M* base;
  239    using Model = M;
  240
  244    Getter get;
  245
  261    Setter set;
  262    using GetterReturnT = decltype(std::invoke(get, base));
  263    using Underlying = std::decay_t<GetterReturnT>;
  264
  269    static_assert(aui::same_as<Underlying , std::decay_t<SignalArg>>, "different getter result and signal arg?");
  270
  271    // this ctor effectively prohibits designated initialization, i.e., this one is not possible:
  272    //
  273    // auto size() const {
  274    //     return APropertyDef {
  275    //         .base = this,
  276    //         .get = &AView::mSize,
  277    //         .set = &AView::setSize,
  278    //         .changed = mSizeChanged,
  279    //     };
  280    // }
  281    //
  282    // deduction in designated initializers is relatively recent feature.
  283    APropertyDef(const M* base, Getter get, Setter set, const emits<SignalArg>& changed)
  284      : base(base), get(std::move(get)), set(std::move(set)), changed(changed) {}
  285
  286    template <aui::convertible_to<Underlying> U>
  287    APropertyDef& operator=(U&& u) {
  288        std::invoke(set, *const_cast<Model*>(base), std::forward<U>(u));
  289        return *this;
  290    }
  291
  292    [[nodiscard]]
  293    GetterReturnT value() const noexcept {
  294        return std::invoke(get, base);
  295    }
  296
  297    [[nodiscard]]
  298    GetterReturnT operator*() const noexcept {
  299        return std::invoke(get, base);
  300    }
  301
  302    [[nodiscard]]
  303    const Underlying* operator->() const noexcept {
  304        return &std::invoke(get, base);
  305    }
  306
  307    [[nodiscard]] operator GetterReturnT() const noexcept { return std::invoke(get, base); }
  308
  309    [[nodiscard]]
  310    M* boundObject() const {
  311        return const_cast<M*>(base);
  312    }
  313
  317    template <aui::invocable<const Underlying&> Projection>
  318    [[nodiscard]]
  319    auto readProjected(Projection&& projection) noexcept {
  320        return aui::detail::property::makeReadonlyProjection(std::move(*this), std::forward<Projection>(projection));
  321    }
  322
  326    template <
  327        aui::invocable<const Underlying&> ProjectionRead,
  329    [[nodiscard]]
  330    auto biProjected(ProjectionRead&& projectionRead, ProjectionWrite&& projectionWrite) noexcept {
  331        return aui::detail::property::makeBidirectionalProjection(
  332            std::move(*this), std::forward<ProjectionRead>(projectionRead),
  333            std::forward<ProjectionWrite>(projectionWrite));
  334    }
  335
  339    template <aui::detail::property::ProjectionBidirectional<Underlying> Projection>
  340    [[nodiscard]]
  341    auto biProjected(Projection&& projectionBidirectional) noexcept {
  342        return aui::detail::property::makeBidirectionalProjection(std::move(*this), projectionBidirectional);
  343    };
  344
  348    void notify() {
  349        if (changed.hasOutgoingConnections()) {
  350            emit changed(this->value());
  351        }
  352    }
  353
  354private:
  355    friend class AObject;
  359    [[nodiscard]]
  360    auto assignment() noexcept {
  361        return aui::detail::property::makeAssignment(std::move(*this));
  362    }
  363};
  364
  365// binary operations for properties.
  366// note: sync this PropertyModifier.h
  367template<AAnyProperty Lhs, typename Rhs>
  368[[nodiscard]]
  369inline auto operator==(const Lhs& lhs, Rhs&& rhs) {
  370    return *lhs == std::forward<Rhs>(rhs);
  371}
  372
  373template<AAnyProperty Lhs, typename Rhs>
  374[[nodiscard]]
  375inline auto operator!=(const Lhs& lhs, Rhs&& rhs) {
  376    return *lhs != std::forward<Rhs>(rhs);
  377}
  378
  379template<AAnyProperty Lhs, typename Rhs>
  380[[nodiscard]]
  381inline auto operator<<(const Lhs& lhs, Rhs&& rhs) {
  382    return *lhs << std::forward<Rhs>(rhs);
  383}
  384
  385template<AAnyProperty Lhs, typename Rhs>
  386[[nodiscard]]
  387inline auto operator>>(const Lhs& lhs, Rhs&& rhs) {
  388    return *lhs >> std::forward<Rhs>(rhs);
  389}
  390
  391template<AAnyProperty Lhs, typename Rhs>
  392[[nodiscard]]
  393inline auto operator<(const Lhs& lhs, Rhs&& rhs) {
  394    return *lhs < std::forward<Rhs>(rhs);
  395}
  396
  397template<AAnyProperty Lhs, typename Rhs>
  398[[nodiscard]]
  399inline auto operator>(const Lhs& lhs, Rhs&& rhs) {
  400    return *lhs > std::forward<Rhs>(rhs);
  401}
  402
  403template<AAnyProperty Lhs, typename Rhs>
  404[[nodiscard]]
  405inline auto operator<=(const Lhs& lhs, Rhs&& rhs) {
  406    return *lhs <= std::forward<Rhs>(rhs);
  407}
  408
  409template<AAnyProperty Lhs, typename Rhs>
  410[[nodiscard]]
  411inline auto operator>=(const Lhs& lhs, Rhs&& rhs) {
  412    return *lhs >= std::forward<Rhs>(rhs);
  413}
  414
  415template<AAnyProperty Lhs, typename Rhs>
  416[[nodiscard]]
  417inline auto operator+(const Lhs& lhs, Rhs&& rhs) {
  418    return *lhs + std::forward<Rhs>(rhs);
  419}
  420
  421template<AAnyProperty Lhs, typename Rhs>
  422[[nodiscard]]
  423inline auto operator-(const Lhs& lhs, Rhs&& rhs) {
  424    return *lhs - std::forward<Rhs>(rhs);
  425}
  426
  427template<AAnyProperty Lhs, typename Rhs>
  428inline decltype(auto) operator+=(Lhs& lhs, Rhs&& rhs)  {
  429    if constexpr (requires { *lhs += std::forward<Rhs>(rhs); }) {
  430        // const operator?
  431        return *lhs += std::forward<Rhs>(rhs);
  432    } else {
  433        return *lhs.writeScope() += std::forward<Rhs>(rhs);
  434    }
  435}
  436
  437template<AAnyProperty Lhs, typename Rhs>
  438inline decltype(auto) operator+=(Lhs&& lhs, Rhs&& rhs) {
  439    return lhs = *lhs + std::forward<Rhs>(rhs);
  440}
  441
  442// simple check above operators work.
  443static_assert(requires { AProperty<int>() + 1; });
  444
  445
  446/*
  447// UNCOMMENT THIS to test biProjected
  448static_assert(requires (AProperty<int>& intProperty) {
  449    { intProperty.biProjected(aui::lambda_overloaded {
  450      [](int) -> AString { return ""; },
  451      [](const AString&) -> int { return 0; },
  452    }).value() } -> aui::convertible_to<AString>;
  453
  454    { intProperty.biProjected(aui::lambda_overloaded {
  455        [](int) -> AString { return ""; },
  456        [](const AString&) -> int { return 0; },
  457    }) = "AString" };
  458
  459
  460    { intProperty.biProjected(aui::lambda_overloaded {
  461        [](int) -> AString { return ""; },
  462        [](const AString&) -> int { return 0; },
  463    }) };
  464
  465    { intProperty.biProjected(aui::lambda_overloaded {
  466        [](int) -> AString { return ""; },
  467        [](const AString&) -> int { return 0; },
  468    }).assignment() } -> aui::invocable<AString>;
  469});
  470*/
  471
  472template <APropertyReadable T> struct fmt::formatter<T> {
  473    template<typename ParseContext>
  474    constexpr auto parse(ParseContext& ctx)
  475    {
  476        return ctx.begin();
  477    }
  478
  479    auto format(T& c, format_context& ctx) const {
  480        return fmt::format_to(ctx.out(), "{}", *c);
  481    }
  482};
Definition AObjectBase.h:24
A base object class.
Definition AObject.h:39
Temporary transparent object that gains write access to underlying property's value,...
Definition PropertyModifier.h:26
Definition concepts.h:226
Definition concepts.h:195
Invokable concept.
Definition concepts.h:37
Definition concepts.h:70
type_of< T > t
Selects views that are of the specified C++ types.
Definition type_of.h:71
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
Property implementation to use with custom getter/setter.
Definition AProperty.h:234
void notify()
Notify observers that a change was occurred (no preconditions).
Definition AProperty.h:348
const M * base
AObject which this property belongs to.
Definition AProperty.h:238
Getter get
Getter. Can be pointer-to-member(function or field) or lambda.
Definition AProperty.h:244
Setter set
Setter. Can be pointer-to-member(function or field) or lambda.
Definition AProperty.h:261
auto biProjected(ProjectionRead &&projectionRead, ProjectionWrite &&projectionWrite) noexcept
Makes a bidirectional projection of this property.
Definition AProperty.h:330
const emits< SignalArg > & changed
Reference to underlying signal emitting on value changes.
Definition AProperty.h:268
auto biProjected(Projection &&projectionBidirectional) noexcept
Makes a bidirectional projection of this property (by a single aui::lambda_overloaded).
Definition AProperty.h:341
auto readProjected(Projection &&projection) noexcept
Makes a readonly projection of this property.
Definition AProperty.h:319
Basic easy-to-use property implementation containing T.
Definition AProperty.h:30
void notify()
Notify observers that a change was occurred (no preconditions).
Definition AProperty.h:116
auto readProjected(Projection &&projection) const noexcept
Makes a readonly projection of this property.
Definition AProperty.h:150
emits< T > changed
Signal that notifies data changes.
Definition AProperty.h:47
auto biProjected(Projection &&projectionBidirectional) noexcept
Makes a bidirectional projection of this property (by a single aui::lambda_overloaded).
Definition AProperty.h:171
aui::PropertyModifier< AProperty > writeScope() noexcept
Definition AProperty.h:141
auto biProjected(ProjectionRead &&projectionRead, ProjectionWrite &&projectionWrite) noexcept
Makes a bidirectional projection of this property.
Definition AProperty.h:160
T raw
Stored value.
Definition AProperty.h:42
static void addDependency(const AAbstractSignal &signal)
Adds observer to the specified signal, if called inside a reactive expression evaluation.