// This source code is released into the public domain. module; #include export module nihil.ucl:real; import nihil.std; import nihil.core; import :object; import :type; namespace nihil::ucl { export struct real final : object { using contained_type = double; static constexpr object_type ucl_type = object_type::real; // Create a real holding the value 0. Throws std::system_error // on failure. real() : real(0) { } // Create a real holding a specific value. Throws std::system_error // on failure. explicit real(contained_type value) : real(noref, [&] { auto *uobj = ::ucl_object_fromdouble(value); if (uobj == nullptr) throw std::system_error(std::make_error_code(sys_error())); return uobj; }()) { } // Create a new real from a UCL object. Throws type_mismatch // on failure. real(ref_t, ::ucl_object_t const *uobj) : object(nihil::ucl::ref, ensure_ucl_type(uobj, real::ucl_type)) { } real(noref_t, ::ucl_object_t *uobj) : object(nihil::ucl::noref, ensure_ucl_type(uobj, real::ucl_type)) { } // Return the value of this real. [[nodiscard]] auto value(this real const &self) -> contained_type { auto v = contained_type{}; auto const *uobj = self.get_ucl_object(); if (::ucl_object_todouble_safe(uobj, &v)) return v; throw std::runtime_error("ucl_object_todouble_safe failed"); } private: // // Comparison operators. // [[nodiscard]] friend auto operator==(real const &a, real const &b) -> bool { return a.value() == b.value(); } [[nodiscard]] friend auto operator<=>(real const &a, real const &b) -> std::partial_ordering { return a.value() <=> b.value(); } [[nodiscard]] friend auto operator==(real const &a, real::contained_type b) -> bool { return a.value() == b; } [[nodiscard]] friend auto operator<=>(real const &a, real::contained_type b) -> std::partial_ordering { return a.value() <=> b; } }; // Real constructor. This returns an error instead of throwing. export [[nodiscard]] auto make_real(real::contained_type value = 0) -> std::expected { auto *uobj = ::ucl_object_fromdouble(value); if (uobj == nullptr) return std::unexpected(error(errc::failed_to_create_object, error(sys_error()))); return real(noref, uobj); } // Literal operator. inline namespace literals { export constexpr auto operator""_ucl(long double d) -> real { if (d > static_cast(std::numeric_limits::max()) || d < static_cast(std::numeric_limits::min())) throw std::out_of_range("literal out of range"); return real(static_cast(d)); } } // namespace literals } // namespace nihil::ucl namespace nihil { inline namespace literals { export using namespace ::nihil::ucl::literals; } } // namespace nihil // std::formatter for a real. This provides the same format operations // as std::formatter; export template <> struct std::formatter { std::formatter base_formatter; template constexpr auto parse(ParseContext &ctx) -> ParseContext::iterator { return base_formatter.parse(ctx); } template auto format(nihil::ucl::real const &o, FmtContext &ctx) const -> FmtContext::iterator { return base_formatter.format(o.value(), ctx); } };