5#include "../DataBinder/Core.hpp"
6#include "../DataBinder/SqlNullValue.hpp"
7#include "../SqlStatement.hpp"
24template <auto TheReferencedField, auto ColumnNameOverr
ideString = std::nullopt>
33 if constexpr (!std::same_as<
decltype(ColumnNameOverrideString), std::nullopt_t>)
34 return std::string_view{ColumnNameOverrideString};
36 return std::string_view {};
42 static_assert(std::remove_cvref_t<decltype(std::declval<ReferencedRecord>().*
ReferencedField)>::IsPrimaryKey,
43 "The referenced field must be a primary key.");
49 static constexpr auto IsOptional =
true;
50 static constexpr auto IsMandatory = !IsOptional;
51 static constexpr auto IsPrimaryKey =
false;
52 static constexpr auto IsAutoIncrementPrimaryKey =
false;
54 template <
typename... S>
55 requires std::constructible_from<
ValueType, S...>
56 constexpr BelongsTo(S&&... value)
noexcept:
57 _referencedFieldValue(std::forward<S>(value)...)
70 if (!_referencedFieldValue)
73 _record = std::nullopt;
74 _referencedFieldValue = {};
84 _record.emplace(other);
93 LIGHTWEIGHT_FORCE_INLINE
constexpr void SetModified(
bool value)
noexcept { _modified = value; }
96 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr bool IsModified() const noexcept {
return _modified; }
99 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr ValueType const&
Value() const noexcept {
return _referencedFieldValue; }
102 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr ValueType&
MutableValue() noexcept {
return _referencedFieldValue; }
105 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr ReferencedRecord&
Record() noexcept { RequireLoaded();
return _record.value(); }
108 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr ReferencedRecord const&
Record() const noexcept { RequireLoaded();
return _record.value(); }
111 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr bool IsLoaded() const noexcept {
return _loaded; }
114 LIGHTWEIGHT_FORCE_INLINE
void Unload() noexcept { _record = std::nullopt; _loaded =
false; }
120 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr ReferencedRecord const&
operator*() const noexcept { RequireLoaded();
return _record.value(); }
126 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr ReferencedRecord const*
operator->() const noexcept { RequireLoaded();
return &_record.value(); }
129 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr bool operator!() const noexcept {
return !_referencedFieldValue; }
132 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr explicit operator bool() const noexcept {
return static_cast<bool>(_referencedFieldValue); }
137 LIGHTWEIGHT_FORCE_INLINE
void BindOutputColumn(SQLSMALLINT outputIndex,
SqlStatement& stmt) { stmt.BindOutputColumn(outputIndex, &_referencedFieldValue); }
140 template <auto OtherReferencedField>
143 return _referencedFieldValue <=> other.Value();
146 template <detail::FieldElementType T, PrimaryKey IsPrimaryKeyValue = PrimaryKey::No>
149 return _referencedFieldValue <=> other.Value();
152 template <auto OtherReferencedField>
155 return (_referencedFieldValue <=> other.Value()) == std::weak_ordering::equivalent;
158 template <auto OtherReferencedField>
161 return (_referencedFieldValue <=> other.Value()) != std::weak_ordering::equivalent;
164 template <detail::FieldElementType T, PrimaryKey IsPrimaryKeyValue = PrimaryKey::No>
167 return (_referencedFieldValue <=> other.Value()) == std::weak_ordering::equivalent;
170 template <detail::FieldElementType T, PrimaryKey IsPrimaryKeyValue = PrimaryKey::No>
173 return (_referencedFieldValue <=> other.Value()) != std::weak_ordering::equivalent;
178 std::function<void()> loadReference {};
184 _loader = std::move(loader);
193 if (_loader.loadReference)
194 _loader.loadReference();
202 bool _loaded =
false;
203 bool _modified =
false;
204 std::optional<ReferencedRecord> _record {};
207template <auto ReferencedField>
210 return os << belongsTo.
Value();
216struct IsBelongsTo: std::false_type
220template <auto ReferencedField, auto ColumnNameOverr
ideString>
221struct IsBelongsTo<
BelongsTo<ReferencedField, ColumnNameOverrideString>>: std::true_type
228constexpr bool IsBelongsTo = detail::IsBelongsTo<std::remove_cvref_t<T>>::value;
230template <auto ReferencedField>
231struct SqlDataBinder<
BelongsTo<ReferencedField>>
234 using InnerType =
typename SelfType::ValueType;
236 static constexpr auto ColumnType = SqlDataBinder<InnerType>::ColumnType;
238 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN InputParameter(SQLHSTMT stmt,
240 SelfType
const& value,
243 return SqlDataBinder<InnerType>::InputParameter(stmt, column, value.Value(), cb);
246 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN
247 OutputColumn(SQLHSTMT stmt, SQLUSMALLINT column, SelfType* result, SQLLEN* indicator,
SqlDataBinderCallback& cb)
249 auto const sqlReturn =
250 SqlDataBinder<InnerType>::OutputColumn(stmt, column, &result->MutableValue(), indicator, cb);
251 cb.PlanPostProcessOutputColumn([result]() { result->SetModified(
true); });
255 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt,
261 auto const sqlReturn = SqlDataBinder<InnerType>::GetColumn(stmt, column, &result->emplace(), indicator, cb);
262 if (SQL_SUCCEEDED(sqlReturn))
263 result->SetModified(
true);
Represents a one-to-one relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & Record() noexcept
Retrieves a record from the relationship.
static constexpr std::string_view ColumnNameOverride
If not an empty string, this value will be used as the column name in the database.
LIGHTWEIGHT_FORCE_INLINE constexpr ValueType & MutableValue() noexcept
Retrieves the mutable reference to the value of the field.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & operator*() noexcept
Retrieves the record from the relationship.
MemberClassType< decltype(TheReferencedField)> ReferencedRecord
Represents the record type of the other field.
void SetAutoLoader(Loader loader) noexcept
Used internally to configure on-demand loading of the record.
static constexpr auto ReferencedField
The field in the other record that references the current record.
LIGHTWEIGHT_FORCE_INLINE constexpr void SetModified(bool value) noexcept
Marks the field as modified or unmodified.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & EmplaceRecord()
Emplaces a record into the relationship. This will mark the relationship as loaded.
LIGHTWEIGHT_FORCE_INLINE void Unload() noexcept
Unloads the record from memory.
LIGHTWEIGHT_FORCE_INLINE constexpr ValueType const & Value() const noexcept
Retrieves the reference to the value of the field.
typename std::remove_cvref_t< decltype(std::declval< ReferencedRecord >().*ReferencedField)>::ValueType ValueType
Represents the column type of the foreign key, matching the primary key of the other record.
LIGHTWEIGHT_FORCE_INLINE constexpr bool IsLoaded() const noexcept
Checks if the record is loaded into memory.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const & operator*() const noexcept
Retrieves the record from the relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr bool operator!() const noexcept
Checks if the field value is NULL.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord * operator->() noexcept
Retrieves the record from the relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr bool IsModified() const noexcept
Checks if the field is modified.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const * operator->() const noexcept
Retrieves the record from the relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const & Record() const noexcept
Retrieves an immutable reference to the record from the relationship.
Represents an error when a record is required to be loaded but is not.
High level API for (prepared) raw SQL statements.
Represents a single column in a table.