/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: set ts=8 sts=2 et sw=2 tw=80: * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef builtin_temporal_PlainDate_h #define builtin_temporal_PlainDate_h #include "mozilla/Assertions.h" #include "mozilla/Attributes.h" #include #include "builtin/temporal/Calendar.h" #include "builtin/temporal/TemporalTypes.h" #include "js/RootingAPI.h" #include "js/TypeDecls.h" #include "js/Value.h" #include "vm/NativeObject.h" class JS_PUBLIC_API JSTracer; namespace js { struct ClassSpec; } // namespace js namespace js::temporal { class PlainDateObject : public NativeObject { public: static const JSClass class_; static const JSClass& protoClass_; static constexpr uint32_t PACKED_DATE_SLOT = 0; static constexpr uint32_t CALENDAR_SLOT = 1; static constexpr uint32_t SLOT_COUNT = 2; /** * Extract the date fields from this PlainDate object. */ ISODate date() const { auto packed = PackedDate{getFixedSlot(PACKED_DATE_SLOT).toPrivateUint32()}; return PackedDate::unpack(packed); } CalendarValue calendar() const { return CalendarValue(getFixedSlot(CALENDAR_SLOT)); } private: static const ClassSpec classSpec_; }; #ifdef DEBUG /** * IsValidISODate ( year, month, day ) */ bool IsValidISODate(const ISODate& date); #endif /** * IsValidISODate ( year, month, day ) */ bool ThrowIfInvalidISODate(JSContext* cx, const ISODate& date); /** * IsValidISODate ( year, month, day ) */ bool ThrowIfInvalidISODate(JSContext* cx, double year, double month, double day); /** * ISODateWithinLimits ( isoDate ) */ bool ISODateWithinLimits(const ISODate& isoDate); class MOZ_STACK_CLASS PlainDate final { ISODate date_; CalendarValue calendar_; public: PlainDate() = default; PlainDate(const ISODate& date, const CalendarValue& calendar) : date_(date), calendar_(calendar) { MOZ_ASSERT(ISODateWithinLimits(date)); } explicit PlainDate(const PlainDateObject* date) : PlainDate(date->date(), date->calendar()) {} const auto& date() const { return date_; } const auto& calendar() const { return calendar_; } // Allow implicit conversion to an ISODate. operator const ISODate&() const { return date(); } explicit operator bool() const { return !!calendar_; } void trace(JSTracer* trc) { calendar_.trace(trc); } const auto* calendarDoNotUse() const { return &calendar_; } }; /** * CreateTemporalDate ( isoDate, calendar [ , newTarget ] ) */ PlainDateObject* CreateTemporalDate(JSContext* cx, const ISODate& isoDate, JS::Handle calendar); /** * CreateTemporalDate ( isoDate, calendar [ , newTarget ] ) */ PlainDateObject* CreateTemporalDate(JSContext* cx, JS::Handle date); /** * CreateTemporalDate ( isoDate, calendar [ , newTarget ] ) */ bool CreateTemporalDate(JSContext* cx, const ISODate& isoDate, JS::Handle calendar, JS::MutableHandle result); /** * CompareISODate ( y1, m1, d1, y2, m2, d2 ) */ int32_t CompareISODate(const ISODate& one, const ISODate& two); /** * BalanceISODate ( year, month, day ) */ bool BalanceISODate(JSContext* cx, const ISODate& date, int64_t days, ISODate* result); /** * BalanceISODate ( year, month, day ) */ ISODate BalanceISODate(const ISODate& date, int32_t days); } /* namespace js::temporal */ namespace js { template class WrappedPtrOperations { const auto& container() const { return static_cast(this)->get(); } public: explicit operator bool() const { return bool(container()); } const auto& date() const { return container().date(); } JS::Handle calendar() const { return JS::Handle::fromMarkedLocation( container().calendarDoNotUse()); } // Allow implicit conversion to an ISODate. operator const temporal::ISODate&() const { return date(); } }; } // namespace js #endif /* builtin_temporal_PlainDate_h */