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>
30struct AProperty: AObjectBase {
31 static_assert(!std::is_reference_v<T>,
"====================> AProperty: attempt to wrap a reference.");
50 requires aui::default_initializable<T>
53 template <aui::convertible_to<T> U>
54 AProperty(U&& value) noexcept(noexcept(T(std::forward<U>(value)))):
raw(std::forward<U>(value)) {}
60 const AObjectBase* boundObject()
const {
64 AProperty(
const AProperty& value):
raw(value.
raw) {
67 AProperty(AProperty&& value)
noexcept:
raw(std::move(value.raw)) {
71 AProperty& operator=(
const AProperty& value) {
79 AProperty& operator=(AProperty&& value)
noexcept {
83 operator=(std::move(value.raw));
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]] {
96 this->
raw = std::forward<U>(value);
101 template <ASignalInvokable SignalInvokable>
102 void operator^(SignalInvokable&& t) {
103 t.invokeSignal(
nullptr);
121 const T& value() const noexcept {
126 [[nodiscard]]
operator const T&()
const noexcept {
return value(); }
129 const T* operator->() const noexcept {
134 const T& operator*() const noexcept {
148 template<aui::invocable<const T&> Projection>
151 return aui::detail::property::makeReadonlyProjection(*
this, std::forward<Projection>(projection));
157 template<aui::invocable<const T&> ProjectionRead,
158 aui::invocable<const std::invoke_result_t<ProjectionRead, T>&> ProjectionWrite>
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));
169 template<aui::detail::property::ProjectionB
idirectional<T> Projection>
172 return aui::detail::property::makeBidirectionalProjection(*
this, projectionBidirectional);
181 auto assignment() noexcept {
182 return aui::detail::property::makeAssignment(*
this);
192 using Underlying = T;
193 PropertyModifier(
AProperty<T>& owner): mOwner(&owner) {}
194 ~PropertyModifier() {
195 if (mOwner ==
nullptr) {
202 Underlying& value()
const noexcept {
207 Underlying* operator->()
const noexcept {
262 using GetterReturnT =
decltype(std::invoke(
get,
base));
263 using Underlying = std::decay_t<GetterReturnT>;
286 template <aui::convertible_to<Underlying> U>
288 std::invoke(
set, *
const_cast<Model*
>(
base), std::forward<U>(u));
293 GetterReturnT value() const noexcept {
298 GetterReturnT operator*() const noexcept {
303 const Underlying* operator->() const noexcept {
304 return &std::invoke(
get,
base);
307 [[nodiscard]]
operator GetterReturnT() const noexcept {
return std::invoke(
get,
base); }
310 M* boundObject()
const {
311 return const_cast<M*
>(
base);
317 template <aui::invocable<const Underlying&> Projection>
320 return aui::detail::property::makeReadonlyProjection(std::move(*
this), std::forward<Projection>(projection));
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));
339 template <aui::detail::property::ProjectionB
idirectional<Underlying> Projection>
342 return aui::detail::property::makeBidirectionalProjection(std::move(*
this), projectionBidirectional);
349 if (
changed.hasOutgoingConnections()) {
360 auto assignment() noexcept {
361 return aui::detail::property::makeAssignment(std::move(*
this));
367template<AAnyProperty Lhs,
typename Rhs>
369inline auto operator==(
const Lhs& lhs, Rhs&& rhs) {
370 return *lhs == std::forward<Rhs>(rhs);
373template<AAnyProperty Lhs,
typename Rhs>
375inline auto operator!=(
const Lhs& lhs, Rhs&& rhs) {
376 return *lhs != std::forward<Rhs>(rhs);
379template<AAnyProperty Lhs,
typename Rhs>
381inline auto operator<<(
const Lhs& lhs, Rhs&& rhs) {
382 return *lhs << std::forward<Rhs>(rhs);
385template<AAnyProperty Lhs,
typename Rhs>
387inline auto operator>>(
const Lhs& lhs, Rhs&& rhs) {
388 return *lhs >> std::forward<Rhs>(rhs);
391template<AAnyProperty Lhs,
typename Rhs>
393inline auto operator<(
const Lhs& lhs, Rhs&& rhs) {
394 return *lhs < std::forward<Rhs>(rhs);
397template<AAnyProperty Lhs,
typename Rhs>
399inline auto operator>(
const Lhs& lhs, Rhs&& rhs) {
400 return *lhs > std::forward<Rhs>(rhs);
403template<AAnyProperty Lhs,
typename Rhs>
405inline auto operator<=(
const Lhs& lhs, Rhs&& rhs) {
406 return *lhs <= std::forward<Rhs>(rhs);
409template<AAnyProperty Lhs,
typename Rhs>
411inline auto operator>=(
const Lhs& lhs, Rhs&& rhs) {
412 return *lhs >= std::forward<Rhs>(rhs);
415template<AAnyProperty Lhs,
typename Rhs>
417inline auto operator+(
const Lhs& lhs, Rhs&& rhs) {
418 return *lhs + std::forward<Rhs>(rhs);
421template<AAnyProperty Lhs,
typename Rhs>
423inline auto operator-(
const Lhs& lhs, Rhs&& rhs) {
424 return *lhs - std::forward<Rhs>(rhs);
427template<AAnyProperty Lhs,
typename Rhs>
428inline decltype(
auto)
operator+=(Lhs& lhs, Rhs&& rhs) {
429 if constexpr (
requires { *lhs += std::forward<Rhs>(rhs); }) {
431 return *lhs += std::forward<Rhs>(rhs);
433 return *lhs.writeScope() += std::forward<Rhs>(rhs);
437template<AAnyProperty Lhs,
typename Rhs>
438inline decltype(
auto)
operator+=(Lhs&& lhs, Rhs&& rhs) {
439 return lhs = *lhs + std::forward<Rhs>(rhs);
472template <APropertyReadable T>
struct fmt::formatter<T> {
473 template<
typename ParseContext>
474 constexpr auto parse(ParseContext& ctx)
479 auto format(T& c, format_context& ctx)
const {
480 return fmt::format_to(ctx.out(),
"{}", *c);
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
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.