Skip to content

Latest commit

 

History

History
685 lines (501 loc) · 31.9 KB

calendar-draft.md

File metadata and controls

685 lines (501 loc) · 31.9 KB

Draft Design of Temporal Calendar API

This doc describes a design for first-class support for non-Gregorian calendars in Temporal. Although most of this document is based on Temporal.PlainDate, most of this applies to Temporal.PlainDateTime, Temporal.PlainYearMonth, Temporal.PlainMonthDay, and Temporal.PlainTime as well.

The approach described in this document was largely adopted, but most of this document's information is nonetheless obsolete, due to removing the Temporal.Calendar interface.

Data Model

Temporal.PlainDate internal slots

Main issue: #290

Temporal.PlainDate currently has three internal slots: year, month, and day. (An "internal slot" refers to actual data, as opposed to "properties", which could be computed.) In this proposal, those slots are renamed to [[ISOYear]], [[ISOMonth]], and [[ISODay]], and an additional [[Calendar]] slot is added. The calendar slot contains an object implementing the Temporal.Calendar interface, described below.

No matter which calendar system is being represented, the data model in Temporal.PlainDate remains indexed in the ISO calendar. So, for instance, if you wanted to represent the Hebrew date 5 Nisan 5780, the data model would be 2020-03-30, and the calendar would be responsible for mapping that into the corresponding Hebrew fields, as described further down in this document.

This data model makes the simple assumption that the concept of a "day" is a solar day (main issues: #390, #389). Most or all modern-use calendars, even those with lunar month cycles, use a solar day, which is based on the time it takes for the Earth to complete one rotation relative to the Sun.

The acclaimed researchers Edward M. Reingold and Nachum Dershowitz discuss this subject in Calendrical Calculations:

The sun moves from east to west, and night follows day with predictable regularity. This apparent motion of the sun as viewed by an earthbound observer provided the earliest time-keeping standard for humankind. The day is, accordingly, the basic unit of time underlying all calendars, but various calendars use different conventions to structure days into larger units: weeks, months, years, and cycles of years.

Some historical calendars, such as the Hawaiian Moon Calendar, define a day as the time it takes for the Earth to complete one rotation relative to the moon (instead of the Sun), which is slightly shorter on average. For calendars that use a lunar day, a Temporal.PlainDateTime can be used instead of Temporal.PlainDate when the distinction is important.

At this time, Temporal does not support subdividing a solar day into anything other than hours, minutes, and seconds. However, we have taken a future-proof approach so that if a use case presents itself, we can add timePlus(), timeMinus(), and timeDifference() methods to Temporal.Calendar.

Temporal.PlainDateTime and Temporal.PlainTime internal slots

As with Temporal.PlainDate, all of these types will gain a [[Calendar]] slot, and year, month, and day will be renamed [[ISOYear]], [[ISOMonth]], and [[ISODay]].

Temporal.PlainYearMonth and Temporal.PlainMonthDay internal slots

Main issue: #391

For reasons explained above, using the ISO calendar as the internal data model has many advantages. However, there are several challenges for these two "incomplete" types: lunar months don't line up with solar months, and not every lunar month occurs in every solar year. After discussing several data model alternatives, we reached the conclusion that the simplest data model for Temporal.PlainYearMonth and Temporal.PlainMonthDay is to make it share the same data model as Temporal.PlainDate, with the same slots.

For calendars that use ISO-style months, such as Gregorian, Solar Buddhist, and Japanese, "ISODay" and "ISOYear" can be ignored for YearMonth and MonthDay respectively. However, for lunar and lunisolar calendars, such as Hebrew, Saudi Arabian Islamic, and Chinese, these fields allow those calendars to disambiguate which YearMonth and MonthDay are being represented.

Temporal.Calendar interface

Main issue: #289

The new Temporal.Calendar interface is a mechanism to allow arbitrary calendar systems to be implemented on top of Temporal. Most users will not encounter the Temporal.Calendar interface directly, unless they are building or using a non-built-in calendar system.

All built-in calendars will be instances of Temporal.Calendar (main issue: #300), and Temporal.Calendar can be subclassed. However, an object need not be a subclass of Temporal.Calendar to conform to the interface, which are the string methods listed below.

We had also considered using symbols, but settled on strings after discussion with the plenary (main issue: #310).

If a custom calendar needs to cache per-object information about the localized year, month, and day of a Temporal object, it can use a WeakMap for this purpose, with the Temporal object as the key into the WeakMap.

Methods on the Temporal.Calendar interface

All of the following methods return new Temporal objects.

class Temporal.Calendar {
	/** Constructs a Temporal.PlainDate from a free-form option bag */
	dateFromFields(
		fields: object,
		constructor: function
	) : Temporal.PlainDate;

	/** Constructs a Temporal.PlainYearMonth from a free-form option bag */
	yearMonthFromFields(
		fields: object,
		constructor: function
	) : Temporal.PlainYearMonth;

	/** Constructs a Temporal.PlainMonthDay from a free-form option bag */
	monthDayFromFields(
		fields: object,
		constructor: function
	) : Temporal.PlainMonthDay;

	/** A string identifier for this calendar */
	id : string;

	fields(
		fields: array<string>
	) : array<string>;

	//////////////////
	//  Arithmetic  //
	//////////////////

	/** Returns input plus duration according to the calendar rules. */
	dateAdd(
		input: Temporal.PlainDate,
		duration: Temporal.Duration,
		options: /* options bag */,
		constructor: function
	) : Temporal.PlainDate;

	/** Returns input minus duration according to the calendar rules. */
	dateSubtract(
		input: Temporal.PlainDate,
		duration: Temporal.Duration,
		options: /* options bag */,
		constructor: function
	) : Temporal.PlainDate;

	/** Returns larger minus smaller, which are dates in the same calendar. */
	dateUntil(
		smaller: Temporal.PlainDate,
		larger: Temporal.PlainDate,
		options: /* options bag */
	) : Temporal.Duration;

	////////////////////////////////////
	//  Accessors:                    //
	//  Semantics defined in date.md  //
	////////////////////////////////////

	year(
		input: Temporal.PlainDate
	) : number;

	month(
		input: Temporal.PlainDate
	) : number;

	day(
		input: Temporal.PlainDate
	) : number;

	dayOfWeek(
		input: Temporal.PlainDate
	) : number;

	weekOfYear(
		input: Temporal.PlainDate
	) : number;

	daysInMonth(
		input: Temporal.PlainDate
	) : number;

	daysInYear(
		input: Temporal.PlainDate
	) : number;

	inLeapYear(
		input: Temporal.PlainDate
	) : boolean;
}

The corresponding fields on Temporal.PlainDate.prototype should forward requests to the calendar as discussed in #291:

get foo(...args) {
  return this.calendar.foo?.(this, ...args);
}

Calendars can add additional calendar-specific accessors, such as the year type ("kesidran", "chaser", "maleh") in the Hebrew calendar, and may add conforming accessor methods to Temporal.PlainDate.prototype.

If any of these accessors are needed for constructing a Temporal.PlainDate from fields, then the calendar should implement fields() which, given an array of field names in the ISO calendar, returns an array of equivalent field names in the calendar. We are not aware of this being necessary for any built-in calendars.

An instance of MyCalendar is expected to have stateless behavior; i.e., calling a method with the same arguments should return the same result each time. There would be no mechanism for enforcing that user-land calendars are stateless; the calendar author should test this expectation on their own in order to prevent unexpected behavior such as the lack of round-tripping.

Enumerable Properties

Main issue: #403

If properties of Temporal.PlainDate, etc., are to be enumerable, the calendar should choose which properties to expose. This operation can cake place in the factory methods of the Temporal.Calendar protocol, such as .dateFromFields().

This is a work in progress, and this document will be updated once we reach consensus on #403.

Default Calendar

Main issue: #292

An open question is what the behavior should be if the programmer does not specify a calendar, or if we should require the programmer to always specify a calendar. Six choices are on the table:

  1. Default to full ISO (Gregorian) calendar.
  2. Require the user to explicitly specify the calendar.
  3. Default to a partial ISO calendar (explained below).
  4. Default to Intl.defaultCalendar (a new symbol), or ISO if that field doesn't exist.
  5. Create separate types for when a calendar is not present (explained below).
  6. Add factory methods for ISO and non-ISO (explained below).

As of October 2020, the Temporal champions chose to adopt Option 6 as the best balance between i18n correctness and ergonomics.

Partial ISO Calendar (Option 3)

A partial ISO calendar would be one implemented as follows:

const PartialIsoCalendar = {
	id: "iso",

	dateFromFields(fields, constructor) {
		const { year, month, day } = fields;
		return new constructor(year, month, day, this);
	}
	// Same for dateTimeFromFields, etc.

	// ALL OTHER METHODS:
	dateAdd() {
		throw new TypeError("Unsupported operation: full calendar required");
	}
	// Same for subtract, etc.
}

It would in effect render default Temporal.PlainDate (and Temporal.PlainDateTime) with fewer operations until you specify a calendar. The following methods/getters would throw:

  • .dayOfWeek
  • .weekOfYear
  • .daysInMonth
  • .daysInYear
  • .inLeapYear
  • .year
  • .month
  • .day
  • .add()*
  • .subtract()*
  • .since()*
  • .getYearMonth()
  • .getMonthDay()
  • .with()

* We could allow the arithmetic methods to work in Partial ISO if the duration units are days or smaller, with the same semantics as Temporal.Instant.

The following methods/getters would still work:

  • .inTimeZone()
  • .withTime()
  • .toString()
  • .toLocaleString()
  • .compare()

Although small, this set of operations still covers many of the recipes in the proposed Temporal Cookbook.

To enable the extended set of operations, the user would just use .withCalendar():

// Force the Gregorian calendar:
Temporal.PlainDate.from('2019-12-06').withCalendar('gregory').weekOfYear;

// Use a calendar from another source:
Temporal.PlainDate.from('2019-12-06').withCalendar(Intl.defaultCalendar).weekOfYear;
Temporal.PlainDate.from('2019-12-06').withCalendar(request.calendar).weekOfYear;

The calendar IDs are less clear. If the partial ISO calendar used ID "iso", then what would the full ISO calendar use? ID "gregory" (why not "gregorian"?) is misleading because there are Gregorian calendars that do not all agree on the same rules for things like weeks of the year. One solution could be to use a nullish ID like null or "" for the partial ISO calendar and "iso" for the full ISO calendar. Alternatively, "iso8601", the identifier defined by CLDR as "Gregorian calendar using the ISO 8601 calendar week rules", could be the identifier for the full ISO calendar.

New Non-Calendar Types (Option 5)

In this option, objects without a calendar would have their own type, and calendar-specific types would be used only when calendar-dependent functionality is required. This is similar in spirit to Partial ISO (Option 3), except that new types are used, rather than simply a null calendar on the existing type.

For example, name bikeshedding aside, Temporal.ZonedAbsolute (main issue: #569) could become an intermediate type between Temporal.Instant and Temporal.PlainDateTime that does not include arithmetic or calendar-dependent functionality. A calendar would be necessary when converting from Temporal.ZonedAbsolute into Temporal.PlainDateTime.

  • Temporal.Instant = a point in time, not specific to a certain place. - Data Model: [[EpochNanoseconds]]
  • Temporal.ZonedAbsolute = a point in time at a place on Earth. - Data Model: [[EpochNanoseconds]] + [[TimeZone]]
  • Temporal.PlainDateTime = a wall clock time, not specific to a certain place. - Data Model: [[Calendar]] + ISO fields

Conversion methods between these three types could be:

// Absolute <=> ZonedAbsolute
Temporal.Instant.prototype.withZone(tz) : Temporal.ZonedAbsolute;
Temporal.ZonedAbsolute.prototype.toInstant() : Temporal.Instant;

// ZonedAbsolute <=> DateTime
Temporal.ZonedAbsolute.prototype.withCalendar(cal) : Temporal.PlainDateTime;
Temporal.PlainDateTime.prototype.withZone(tz) : Temporal.ZonedAbsolute;

We could add a similar intermediate type for dates without times. Name bikeshedding aside, this second new type could be Temporal.AbstractDate. The semantics would be:

  • Temporal.AbstractDate = a solar day at a place on Earth - Data Model: [[EpochDays]] + [[TimeZone]]
  • Temporal.PlainDate = a wall clock date, not specific to a certain place. - Data Model: [[Calendar]] + ISO fields

Conversion methods:

// ZonedAbsolute <=> AbstractDate
Temporal.ZonedAbsolute.prototype.getDate() : Temporal.AbstractDate;
Temporal.AbstractDate.prototype.withTime() : Temporal.ZonedAbsolute;

// AbstractDate <=> Date
Temporal.AbstractDate.prototype.withCalendar(cal) : Temporal.PlainDate;
Temporal.PlainDate.prototype.withZone(tz) : Temporal.AbstractDate;

// Date <=> DateTime
Temporal.PlainDateTime.prototype.getDate() : Temporal.PlainDate;
Temporal.PlainDate.prototype.withTime() : Temporal.PlainDateTime;

An additional type, Temporal.EpochDays, could be added as an analog of Temporal.Instant but with days instead of nanoseconds since epoch.

Here is an illustrated version of this option:

Potential Temporal Data Types

New Factory Methods (Option 6)

With this option, separate methods would indicate whether the Full ISO calendar should be used versus a potentially non-ISO calendar. For example, Temporal.Now.plainDateISO would be added to supplement Temporal.Now.plainDate. See the full table below.

Methods of Construction

The main way the six options differ is by how and when a calendar is specified when creating calendar-sensitive Temporal objects.

The following table describes these semantics. Option 5 is not shown because the constructors will be different than in options 1-4 and 6.

Method Option 1 Option 2 Option 3 Option 4 Option 6
T.Date.from(string)* From String Explicit Partial ISO Environ. From String
T.Date.from(fields)** From Object Explicit Partial ISO Environ. From Object
new T.Date() Full ISO Full ISO Full ISO Full ISO Full ISO
T.Now.plainDate() Full ISO Explicit Partial ISO Environ. Explicit
T.Now.plainDateISO() N/A N/A N/A N/A Full ISO
instant.inTimeZone() Full ISO Explicit Partial ISO Environ. Explicit
instant.inZoneISO() N/A N/A N/A N/A Full ISO
date.getMonthDay()**** Inherit Inherit Explicit Inherit Inherit
HTML input*** Full ISO Full ISO Full ISO Full ISO Full ISO

Footnotes:

* from(string) may carry the calendar ID in the string (main issue: #293).

** @ptomato pointed out that "if you write Temporal.PlainDate.from({ year: 2020, month: 5, day: 19 }) with no calendar specified, then realistically what else do you mean besides the full ISO calendar?" Therefore, a calendar argument is not required in T.Date.from.

*** The HTML5 spec only supports ISO-8601 (reference).

**** An argument could be added to getMonthDay() in Options 3 and 5 to reduce boilerplate.

Preventing I18n Errors

As compared to option 1 (always default to Full ISO), the following are examples of programming errors that Partial ISO would help prevent.

These errors are most common in the four regions that use non-ISO calendars as their default: Saudi Arabia, Iran, Afghanistan, and Thailand. However, they could also manifest if the user has overridden their preferred calendar in browser settings.

For the purposes of illustration, the following examples will use the locale "en-SA" (English in Saudi Arabia) on 2020-05-17.

Format a month and day from a date

Buggy output:

Today is: May 17

Correct output:

Today is: Ramadan 24

Code:

/// Options 1 and 4: calendar implicit
const today = Temporal.Now.plainDate();
// OPTION 1 BUG: the ISO calendar is implicit when creating the MonthDay
const monthDay = today.getMonthDay();
console.log('Today is:', monthDay.toLocaleString());

/// Options 2 and 6: calendar in factory method
const calendar = navigator.locales[0].getLikelyCalendar();
const today = Temporal.Now.plainDate(calendar);
const monthDay = today.getMonthDay();
console.log('Today is:', monthDay.toLocaleString());

/// Option 3 and 5: getMonthDay() requires a calendar to be given
const today = Temporal.Now.plainDate();
const calendar = navigator.locales[0].getLikelyCalendar();
const monthDay = today.withCalendar(calendar).getMonthDay();
console.log('Today is:', monthDay.toLocaleString());

/// Option 3 and 5 alternative: add argument to getMonthDay()
const today = Temporal.Now.plainDate();
const calendar = navigator.locales[0].getLikelyCalendar();
const monthDay = today.getMonthDay(calendar);
console.log('Today is:', monthDay.toLocaleString());

/// Alternative using Intl settings
const today = Temporal.Now.plainDate();
console.log(
  'Today is:',
  today.toLocaleString(undefined, {
    month: 'long',
    day: 'numeric'
  })
);

Get this date next month

Buggy output:

Today is: Ramadan 24, 1441 AH
Next month is: Shawwal 25, 1441 AH

Correct output:

Today is: Ramadan 24, 1441 AH
Next month is: Shawwal 24, 1441 AH

Code:

/// Options 1 and 4: calendar implicit
const today = Temporal.Now.plainDate();
console.log('Today is:', today.toLocaleString());
// OPTION 1 BUG:
// Arithmetic in months must take place in the user calendar
const nextMonth = today.add({ months: 1 });
console.log('Next month is: ', nextMonth.toLocaleString());

/// Options 2 and 6: calendar in factory method
const calendar = navigator.locales[0].getLikelyCalendar();
const today = Temporal.Now.plainDate(calendar);
console.log('Today is:', today.toLocaleString());
const nextMonth = today.add({ months: 1 });
console.log('Next month is: ', nextMonth.toLocaleString());

/// Options 3 and 5: calendar only when needed
const today = Temporal.Now.plainDate();
console.log('Today is:', today.toLocaleString());
const calendar = navigator.locales[0].getLikelyCalendar();
const nextMonth = today.withCalendar(calendar).add({ months: 1 });
console.log('Next month is: ', nextMonth.toLocaleString());

Get a month and a day

Buggy output:

Today is: Ramadan 24, 1441 AH
Is May 18 your birthday?

Correct output:

Today is: Ramadan 24, 1441 AH
Is Ramadan 24 your birthday?

Code:

/// Options 1 and 4: calendar implicit
const date = Temporal.Now.plainDate();
console.log('Today is:', date.toLocaleString());
// OPTION 1 BUG:
// The MonthDay needs to be represented in the user calendar; otherwise,
// toLocaleString must format in the ISO calendar.
const monthDay = date.getMonthDay();
console.log(`Is ${monthDay.toLocaleString()} your birthday?`);

/// Options 2 and 6: calendar in factory method
const calendar = navigator.locales[0].getLikelyCalendar();
const date = Temporal.Now.plainDate(calendar);
console.log('Today is:', date.toLocaleString());
const monthDay = date.getMonthDay();
console.log(`Is ${monthDay.toLocaleString()} your birthday?`);

/// Options 3 and 5: calendar only when needed
const date = Temporal.Now.plainDate();
console.log('Today is:', date.toLocaleString());
const calendar = navigator.locales[0].getLikelyCalendar();
const monthDay = date.withCalendar(calendar).getMonthDay();
console.log(`Is ${monthDay.toLocaleString()} your birthday?`);

Get a month-related property

Buggy output:

Today is: Ramadan 24, 1441 AH
Number of days this month: 31

Correct output:

Today is: Ramadan 24, 1441 AH
Number of days this month: 30

Code:

/// Options 1 and 4: calendar implicit
const date = Temporal.Now.plainDate();
// OPTION 1 BUG:
// The YearMonth needs to be represented in the user calendar
const yearMonth = date.getYearMonth();
console.log('Today is:', date.toLocaleString());
console.log('Number of days this month:', yearMonth.daysInMonth);

/// Options 2 and 6: calendar in factory method
const calendar = navigator.locales[0].getLikelyCalendar();
const date = Temporal.Now.plainDate(calendar);
const yearMonth = date.getYearMonth();
console.log('Today is:', date.toLocaleString());
console.log('Number of days this month:', yearMonth.daysInMonth);

/// Options 3 and 5: calendar only when needed
const date = Temporal.Now.plainDate();
const calendar = navigator.locales[0].getLikelyCalendar();
const yearMonth = date.withCalendar(calendar).getYearMonth();
console.log('Today is:', date.toLocaleString());
console.log('Number of days this month:', yearMonth.daysInMonth);

Default Calendar Options: Pros and Cons

Legend:

  • 😃 = Strong advantage
  • 🙂 = Mixed advantage
  • 😐 = Surmountable disadvantage
  • ☹️ = Insurmountable disadvantage
Option API consistency & predictability Impact on Temporal call sites Impact on i18n correctness Impact on interoperability Logistics
Full ISO (option 1) 😃 Consistent and predictable 😃 No changes ☹️ Programmer needs to know to "opt in" to use the user's calendar preference 😃 ISO is the industry standard format 😃 No changes
No Default (option 2) 😃 Consistent and predictable 😐 All call sites require extra boilerplate 😃 All operations require an explicit choice 😃 Explicit choice 🙂 Minimal API changes
Partial ISO (option 3) 😐 Predictable behavior, but call sites may or may not require an explicit calendar 🙂 Most* operations work; some require extra boilerplate 😃 Calendar-sensitive operations require an explicit choice 😃 I/O operations operate in the ISO calendar space 🙂 No API changes, but edge cases to hash out
User Preference (option 4) 😐 Consistent API, but unpredictable behavior based on user's or server's location 😃 No changes 🙂 Correct on front end, but programmer needs to know to "opt in" on back end 😐 Certain operations implicitly use a non-ISO calendar 🙂 Minimal API changes
New Types (option 5) 🙂 Consistent, but the new types may increase mental load 🙂 Most* operations work; some require extra boilerplate 😃 Calendar-sensitive operations require an explicit choice 😃 I/O operations operate in the ISO calendar space 😐 Lots of new types and API changes
ISO Factories (option 6) 😐 Unclear whether to use ISO or non-ISO factories 🙂 Method names at call sites may change 🙂 Likely to be correct, but users could call the ISO factories incorrectly 😃 ISO-specific methods for interop 🙂 Minimal API changes

*See #240 (comment)

Temporal.PlainDate API changes

New Temporal.PlainDate instance methods

Temporal.PlainDate.prototype.with does not modify the calendar. A new method is added for that:

Temporal.PlainDate.prototype.withCalendar = function (newCalendar: Calendar): Temporal.PlainDate {
  const { year, month, day } = this.getISOFields();
  // note: call intrinsic version
  return new Temporal.PlainDate(year, month, day, newCalendar);
  // note: use species constructor
};

ISO strings with calendar hint

This is an open question being discussed in #293. The issue is that we do not want toString() to be lossy by losing the calendar field. One proposal is to append the Temporal.Calendar.id field in [c=ID] following the date. For example, 2019-12-06[c=hebrew] refers to 2019-12-06 projected into the Hebrew calendar.

Temporal.PlainDate.prototype.toString = function() {
	let calendarKeyword, isoDate;
	// For Default Calendar Option 3, check for the partial ISO calendar here
	if (/* this.calendar is the ISO calendar */) {
		calendarKeyword = null;
		isoDate = this;
	} else {
		calendarKeyword = this.calendar.id;
		isoDate = this.withCalendar(Temporal.Calendar.iso);  // call intrinsic
	}
	// return an ISO string for isoDate with calendar in brackets:
	// "2019-12-06[c=hebrew]"
}

In this scenario, Temporal.PlainDate.from() will call Temporal.Calendar.from() to resolve the ID into a Calendar object. Temporal.parse() will have to add a new calendar member to the object it returns, whose value is the string ID given in a [c=ID] comment, or null if not given.

Temporal.Calendar.from() can be monkeypatched by calendar implementors if it is necessary to make new time zones available globally. The expectation is that it would rarely be necessary to do so, because if you have implemented a custom calendar for a particular application, you probably don't need it to be available globally.

Example of monkeypatching for a custom calendar:

const fooCalendar = new FooCalendar();
const originalTemporalCalendarFrom = Temporal.Calendar.from;
Temporal.Calendar.from = function (item) {
  let id;
  if (item instanceof Temporal.Calendar) {
    ({ id } = item);
  } else {
    const string = `${item}`;
    try {
      id = Temporal.parse(item).calendar;
    } catch {
      id = string;
    }
  }
  if (id === 'foo') return fooCalendar;
  return originalTemporalCalendarFrom.call(this, item);
};

Temporal.PlainDate.from('2019-12-03[c=foo]');
// 2019-12-03 projected into the foo calendar

New behavior of Temporal.PlainDate.from

The exact behavior of this method depends on a few open discussions, but some logic will be passed to the Calendar object in order to project the date into the correct calendar system.

Temporal.PlainDate.from = function(thing: string | object, options: object) {
	if (typeof thing === "string") {
		let object = // components of string
		return Temporal.Calendar.iso.dateFromFields(object, this);
	} else {
		// Get the calendar object, either the default calendar or something based
		// on thing.calendar (string lookup or object)
		let calendar = // ...
		return calendar.dateFromFields(thing, this);
	}
}

Semantics of existing Temporal.PlainDate instance methods

As discussed earlier, Temporal.PlainDate will defer to Temporal.Calendar methods wherever necessary. Example implementation of selected Temporal.PlainDate methods:

Temporal.PlainDate.prototype.add = function (duration, options) {
  const constructor = ES.SpeciesConstructor(this, Temporal.PlainDate);
  return this.calendar.dateAdd(this, duration, options, constructor);
};

Temporal.PlainDate.prototype.until = function (other, options) {
  if (other.calendar !== this.calendar) {
    // Note: call intrinsic versions of this method
    other = other.withCalendar(this.calendar);
  }
  return this.calendar.dateUntil(this, other, options);
};

Temporal.PlainDate.prototype.with = function (overrides) {
  const newFields = Object.assign({}, this.getFields(), overrides);
  return this.calendar.dateFromFields(newFields);
};

Temporal.PlainDate.prototype.getYearMonth = function () {
  return this.calendar.yearMonthFromFields(this.getFields());
};

Note that the arguments to methods like with, withDate, etc., are interpreted in the instance's calendar.

Other Temporal.PlainDate constructors

Temporal.Instant.prototype.inTimeZone

The third way to get a Temporal.PlainDate (besides from a string and an object) is to convert it from a Temporal.Instant.

The API here would depend on the decision for whether to require an explicit default calendar. If we decide to use a default calendar (options 1, 3, and 4), no API change would be required for this method. If we decide to require an explicit calendar, then the API would likely be changed as follows:

// Default calendar option 2 only
Temporal.Instant.prototype.inTimeZone = function(timeZone, calendar) {
	const isoDate = // compute the ISO date from the time zone
	return isoDate.withCalendar(calendar);
}

Temporal.Now

The fourth way to get a Temporal.PlainDate is to get the current time according to the environment (or mocked for SES).

As above, this API depends on whether we decide to use a default calendar. If we require an explicit calendar, it would be similar to above:

Temporal.Now.plainDate = function (calendar) {
  const instant = Temporal.Now.instant(); // use intrinsic
  const timeZone = Temporal.Now.timeZone(); // use intrinsic
  return instant.inTimeZone(timeZone, calendar); // use intrinsic
};

Changes to other Temporal APIs

All of the following APIs would gain an internal slot for the calendar.

  • Temporal.PlainDateTime
  • Temporal.PlainTime
  • Temporal.PlainYearMonth
  • Temporal.PlainMonthDay

New behavior of Temporal.PlainMonthDay.withYear

In some calendars (such as the Japanese imperial calendar) a year is not enough information to be able to convert Temporal.PlainMonthDay to Temporal.PlainDate. Temporal.PlainMonthDay.withYear, which previously took a number argument (the year), can now be passed an object which must have properties that provide the needed information. (In the Japanese calendar, it needs to have era and year properties.)

Passing a numerical year argument num is equivalent to passing { year: num }. The calendar can decide what to do in the case of missing information, in dateFromFields.