Source code

Revision control

Copy as Markdown

Other Tools

/* -*- 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_PlainDateTime_h
#define builtin_temporal_PlainDateTime_h
#include "mozilla/Assertions.h"
#include "mozilla/Attributes.h"
#include "mozilla/Casting.h"
#include <stdint.h>
#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 PlainDateTimeObject : public NativeObject {
public:
static const JSClass class_;
static const JSClass& protoClass_;
static constexpr uint32_t PACKED_DATE_SLOT = 0;
static constexpr uint32_t PACKED_TIME_SLOT = 1;
static constexpr uint32_t CALENDAR_SLOT = 2;
static constexpr uint32_t SLOT_COUNT = 3;
/**
* Extract the date fields from this PlainDateTime object.
*/
ISODate date() const {
auto packed = PackedDate{getFixedSlot(PACKED_DATE_SLOT).toPrivateUint32()};
return PackedDate::unpack(packed);
}
/**
* Extract the time fields from this PlainDateTime object.
*/
Time time() const {
auto packed = PackedTime{mozilla::BitwiseCast<uint64_t>(
getFixedSlot(PACKED_TIME_SLOT).toDouble())};
return PackedTime::unpack(packed);
}
/**
* Extract the date-time fields from this PlainDateTime object.
*/
ISODateTime dateTime() const { return {date(), time()}; }
CalendarValue calendar() const {
return CalendarValue(getFixedSlot(CALENDAR_SLOT));
}
private:
static const ClassSpec classSpec_;
};
struct DifferenceSettings;
class Increment;
class CalendarFields;
enum class TemporalOverflow;
enum class TemporalRoundingMode;
enum class TemporalUnit;
#ifdef DEBUG
/**
* IsValidISODate ( year, month, day )
* IsValidTime ( hour, minute, second, millisecond, microsecond, nanosecond )
*/
bool IsValidISODateTime(const ISODateTime& isoDateTime);
#endif
/**
* ISODateTimeWithinLimits ( isoDateTime )
*/
bool ISODateTimeWithinLimits(const ISODateTime& isoDateTime);
class MOZ_STACK_CLASS PlainDateTime final {
ISODateTime dateTime_;
CalendarValue calendar_;
public:
PlainDateTime() = default;
PlainDateTime(const ISODateTime& dateTime, const CalendarValue& calendar)
: dateTime_(dateTime), calendar_(calendar) {
MOZ_ASSERT(ISODateTimeWithinLimits(dateTime));
}
explicit PlainDateTime(const PlainDateTimeObject* dateTime)
: PlainDateTime(dateTime->dateTime(), dateTime->calendar()) {}
const auto& dateTime() const { return dateTime_; }
const auto& date() const { return dateTime_.date; }
const auto& time() const { return dateTime_.time; }
const auto& calendar() const { return calendar_; }
// Allow implicit conversion to an ISODateTime.
operator const ISODateTime&() const { return dateTime(); }
void trace(JSTracer* trc) { calendar_.trace(trc); }
const auto* calendarDoNotUse() const { return &calendar_; }
};
/**
* CreateTemporalDateTime ( isoDateTime, calendar [ , newTarget ] )
*/
PlainDateTimeObject* CreateTemporalDateTime(JSContext* cx,
const ISODateTime& dateTime,
JS::Handle<CalendarValue> calendar);
/**
* InterpretTemporalDateTimeFields ( calendar, fields, overflow )
*/
bool InterpretTemporalDateTimeFields(JSContext* cx,
JS::Handle<CalendarValue> calendar,
JS::Handle<CalendarFields> fields,
TemporalOverflow overflow,
ISODateTime* result);
/**
* RoundISODateTime ( isoDateTime, increment, unit, roundingMode )
*/
ISODateTime RoundISODateTime(const ISODateTime& isoDateTime,
Increment increment, TemporalUnit unit,
TemporalRoundingMode roundingMode);
/**
* DifferencePlainDateTimeWithRounding ( isoDateTime1, isoDateTime2, calendar,
* largestUnit, roundingIncrement, smallestUnit, roundingMode )
*/
bool DifferencePlainDateTimeWithRounding(JSContext* cx,
const ISODateTime& isoDateTime1,
const ISODateTime& isoDateTime2,
JS::Handle<CalendarValue> calendar,
const DifferenceSettings& settings,
InternalDuration* result);
/**
* DifferencePlainDateTimeWithTotal ( isoDateTime1, isoDateTime2, calendar, unit
* )
*/
bool DifferencePlainDateTimeWithTotal(JSContext* cx,
const ISODateTime& isoDateTime1,
const ISODateTime& isoDateTime2,
JS::Handle<CalendarValue> calendar,
TemporalUnit unit, double* result);
} /* namespace js::temporal */
namespace js {
template <typename Wrapper>
class WrappedPtrOperations<temporal::PlainDateTime, Wrapper> {
const auto& container() const {
return static_cast<const Wrapper*>(this)->get();
}
public:
const auto& dateTime() const { return container().dateTime(); }
const auto& date() const { return container().date(); }
const auto& time() const { return container().time(); }
auto calendar() const {
return JS::Handle<temporal::CalendarValue>::fromMarkedLocation(
container().calendarDoNotUse());
}
// Allow implicit conversion to an ISODateTime.
operator const temporal::ISODateTime&() const { return dateTime(); }
};
} // namespace js
#endif /* builtin_temporal_PlainDateTime_h */