Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Cannot output only era or timeZoneName #461

Open
FrankYFTang opened this issue Jun 18, 2020 · 19 comments
Open

Cannot output only era or timeZoneName #461

FrankYFTang opened this issue Jun 18, 2020 · 19 comments
Labels
c: datetime Component: dates, times, timezones s: help wanted Status: help wanted; needs proposal champion
Milestone

Comments

@FrankYFTang
Copy link
Contributor

FrankYFTang commented Jun 18, 2020

Currently, we can use the option in Date.toLocale*String or Intl.DateTimeFormat to output most of the property in "Table 6: Components of date and time formats"
https://ecma-international.org/ecma-402/#datetimeformat-objects

d8> (new Date()).toLocaleString("en", {weekday: "long"})
"Thursday"
d8> (new Date()).toLocaleString("en", {year: "numeric"})
"2020"
d8> (new Date()).toLocaleString("en", {month: "long"})
"June"
d8> (new Date()).toLocaleString("en", {day: "numeric"})
"18"
d8> (new Date()).toLocaleString("en", {hour: "numeric"})
"12 AM"
d8> (new Date()).toLocaleString("en", {hour: "numeric"})
"12 AM"
d8> (new Date()).toLocaleString("en", {minute: "numeric"})
"3"
d8> (new Date()).toLocaleString("en", {second: "numeric"})
"51"

But if we put era or timeZoneName into the option, it will output more than just era / timeZoneName due to the ToDateTimeOptions

d8> (new Date()).toLocaleString("en", {era: "long"})
"6 18, 2020 Anno Domini, 12:01:54 AM"
d8> (new Date()).toLocaleString("en", {timeZoneName: "long"})
"6/18/2020, 12:04:05 AM Pacific Daylight Time"

Should we change ToDateTimeOptions to consider also era and timeZoneName so we can only output era or timeZoneName as the following. Would that be better?

d8> (new Date()).toLocaleString("en", {era: "long"})
"Anno Domini"
d8> (new Date()).toLocaleString("en", {timeZoneName: "long"})
"Pacific Daylight Time"
@FrankYFTang
Copy link
Contributor Author

@anba
Copy link
Contributor

anba commented Jun 19, 2020

A workaround to get the time zone resp. era name is to invoke formatToParts:

js> new Intl.DateTimeFormat("en", {timeZoneName:"long"}).formatToParts().filter(({type}) => type === "timeZoneName")[0].value               
"Pacific Daylight Time"
js> new Intl.DateTimeFormat("en", {era:"long"}).formatToParts().filter(({type}) => type === "era")[0].value                   
"Anno Domini"

I wonder if there are any meeting notes or es-discuss mails why "era" and "timeZoneName" aren't allowed as standalone elements in ToDateTimeOptions.

@FrankYFTang
Copy link
Contributor Author

A workaround to get the time zone resp. era name is to invoke formatToParts:

js> new Intl.DateTimeFormat("en", {timeZoneName:"long"}).formatToParts().filter(({type}) => type === "timeZoneName")[0].value               
"Pacific Daylight Time"
js> new Intl.DateTimeFormat("en", {era:"long"}).formatToParts().filter(({type}) => type === "era")[0].value                   
"Anno Domini"

I wonder if there are any meeting notes or es-discuss mails why "era" and "timeZoneName" aren't allowed as standalone elements in ToDateTimeOptions.

yea, but that is a very clumsy way to get it, right?

@sffc
Copy link
Contributor

sffc commented Jun 22, 2020

This sounds like something for DisplayNames, no?

DateTimeFormat should be for formatting dates and times. If anything, we should do more to make sure there is enough information in a string to decipher a date, not less.

@FrankYFTang
Copy link
Contributor Author

This sounds like something for DisplayNames, no?

First of all, the reason I raise this is not just for the functionality, but for the coherence / consistency of the Intl.DateTimeFormat
Second, the issue could be solved by DisplayNames IF we also know the timeZone itself. But if we do not know the timeZone ID itself, for example, the following case, how could we get the "Pacific Daylight Time" back?
(new Date()).toLocaleString("en", {timeZoneName: "long"})

@sffc
Copy link
Contributor

sffc commented Jul 7, 2020

In the case, the time zone implicitly defaults to the system time zone, so Temporal.now.timeZone() should suffice to get the time zone ID, and then you can pass that to Intl.DisplayNames.

@sffc sffc added s: discuss Status: TG2 must discuss to move forward c: datetime Component: dates, times, timezones labels Jul 8, 2020
@jswalden
Copy link
Collaborator

jswalden commented Jul 9, 2020

A workaround to get the time zone resp. era name is to invoke formatToParts:

That assumes the presentation of the era/time zone in broader context, is suitable for use in standalone context as well. That may not be a universal assumption, nor one that will hold in the future, seems like.

@sffc sffc added s: help wanted Status: help wanted; needs proposal champion and removed s: discuss Status: TG2 must discuss to move forward labels Jul 9, 2020
@sffc sffc added this to the ES 2021 milestone Jul 9, 2020
@anba
Copy link
Contributor

anba commented Jul 9, 2020

CLDR doesn't seem to provide standalone formats for time zone resp. era in <availableFormats>. (I'm not even sure there's an explicit standalone support for era and time zones at all in CLDR. https://unicode.org/reports/tr35/tr35-dates.html#months_days_quarters_eras doesn't mention standalone contexts for eras. The same seems to apply for https://unicode.org/reports/tr35/tr35-dates.html#Time_Zone_Names.)

@sffc
Copy link
Contributor

sffc commented Jul 10, 2020

I could imagine that the time zone or era name could be context-dependent, like grammatical case, pluralization rules, etc., although I don't know of this being done in practice.

@sffc
Copy link
Contributor

sffc commented Sep 10, 2020

@sffc sffc modified the milestones: ES 2021, ES 2022 Mar 22, 2021
@adriancorcoran
Copy link

Just adding to this - it would be amazing if we could obtain only the Timezone name from Date object. It seems strange to me that when we look for timezone name, we get the date thrown in as well, because of the different display options for the date (slashes, dots, spaces, before / after the timezone name) it makes it very difficult to strip out just the timezone name from the resulting string - here's a few examples in different languages:

Here's the code I am using:

    const timezoneFormat = new Intl.DateTimeFormat(userLocale, {
      timeZoneName: 'long',
    });

    const tz = timezoneFormat.format(new Date());
    const tf = timezoneFormat.format(new Date()).split(' ').slice(1).join(' ');

    console.log(`tz: ${tz}`);
    console.log(`tf: ${tf}`);
    console.log(` `);

As you can see, I find it strange that the date is thrown into the result and attempts to split it out are hampered in some timezones due to the formatting. I'm glad to see the workaround above but it would be great to address this :)

@sffc sffc modified the milestones: ES 2022, ES 2023 Jun 1, 2022
@mohd-akram
Copy link

Running into this as well while trying to use the longOffset format. Would be useful when presenting the user with a list of time zones and their offsets.

@ptomato
Copy link
Contributor

ptomato commented Dec 6, 2024

Note that this doesn't apply to dayPeriod, which one might expect to behave similarly to era:

> new Date().toLocaleString('en-CA', {dayPeriod: 'short'})
'in the afternoon'

@ptomato
Copy link
Contributor

ptomato commented Dec 6, 2024

If CLDR doesn't provide standalone formats for era and timeZoneName, the format matcher algorithm could still return a format that includes other fields, if I understand correctly? But the current spec text precludes ever returning a standalone format if CLDR were to gain one.

@anba
Copy link
Contributor

anba commented Dec 6, 2024

The minimum available format subsets for each locale contain neither era nor timeZoneName, so we first need to check if CLDR provides sufficient data for all locales.

@sffc
Copy link
Contributor

sffc commented Dec 9, 2024

I don't think availableFormats is necessary for time zone name. It is usually always just glued to the end.

I can imagine that availableFormats could be useful for era name. It might let a certain locale or calendar override short and long, for example. However, it is very likely a reasonable fallback if the input era name length is also used as the output era name length.

If this isn't in UTS 35, it could be added.

@anba
Copy link
Contributor

anba commented Dec 10, 2024

Looks like <appendItems> as described here should handle additional fields which aren't present in <availableFormats>.

But just to make sure there's no confusion, I was referring to this required list of available formats → https://tc39.es/ecma402/#sec-intl.datetimeformat-internal-slots and not CLDR's <availableFormats>.

@sffc
Copy link
Contributor

sffc commented Jan 9, 2025

In ECMA-402, the culprit lines seem to be

If required is date or any, then
       i. For each property name prop of « "weekday", "year", "month", "day" », do
          1. Let value be formatOptions.[[<prop>]].
          2. If value is not undefined, set needDefaults to false

Thanks @ptomato for your investigation on this.

Why is "era" not in that list?

ptomato added a commit to ptomato/ecma402 that referenced this issue Feb 6, 2025
Previously, if `era` or `timeZoneName` were given by themselves in the
formatting options, the default formatting for the other date/time
components would be added. This is inconsistent with what we do for
`dayPeriod` where a lone `dayPeriod` produces formats such as "in the
afternoon".

The use case of obtaining the localized era name has also been mentioned
several times.

This change makes a lone `era` or `timeZoneName` be passed to the format
matcher algorithm without added date/time components.

Closes: tc39#461
ptomato added a commit to ptomato/ecma402 that referenced this issue Feb 7, 2025
Previously, if `era` was given by itself in the formatting options, the
default formatting for the other date/time components would be added.
This is inconsistent with what we do for `dayPeriod` where a lone
`dayPeriod` produces formats such as "in the afternoon".

The use case of obtaining the localized era name has also been mentioned
several times.

This change makes a lone `era` be passed to the format matcher algorithm
without added date/time components, and requires implementations to have
at least a year-month-day-era and year-month-era format available (which
looks to be widely available across languages in CLDR.)

See: tc39#461
ptomato added a commit to ptomato/ecma402 that referenced this issue Feb 7, 2025
Previously, if `timeZoneName` was given by itself in the formatting
options, the default formatting for the other date/time components would
be added. This is inconsistent with what we do for `dayPeriod` where a
lone `dayPeriod` produces formats such as "in the afternoon".

The use case of obtaining the localized time zone name in all its
possible variants such as specific/generic, has been mentioned several
times.

This change makes a lone `timeZoneName` be passed to the format matcher
algorithm without added date/time components, and requires
implementations to have at least a
year-month-day-hour-minute-second-timeZoneName format available.

See: tc39#461
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
c: datetime Component: dates, times, timezones s: help wanted Status: help wanted; needs proposal champion
Projects
Archived in project
Development

No branches or pull requests

7 participants