Cldr.Interval (Cldr Dates & Times v2.25.3)
View SourceInterval formats allow for software to format intervals like "Jan 10-12, 2008" as a shorter and more natural format than "Jan 10, 2008 - Jan 12, 2008". They are designed to take a start and end date, time or datetime plus a formatting pattern and use that information to produce a localized format.
The interval functions in the library will determine the calendar field with the greatest difference between the two datetimes before using the format pattern.
For example, the greatest difference in "Jan 10-12, 2008" is the day field, while the greatest difference in "Jan 10 - Feb 12, 2008" is the month field. This is used to pick the exact format pattern to be used.
Interval Formats
CLDR provides a three-level mapping to a concrete format pattern.
The first level determines which fields are included in the interval format. This level is called a format style.
The second level is grouped by standard formats (
:long,:medium,:short) which reflect the length of the specified fields - espeially the format length of themonthfield for dates.The third level is a format skeleton which can then be resolved to a format pattern.
Here's an example of the format styles, encapsulating standard formats the resolve to format skeletons.
iex> Cldr.Date.Interval.styles()
%{
date: %{long: :yMMMEd, medium: :yMMMd, short: :yMd},
month: %{long: :MMM, medium: :MMM, short: :M},
month_and_day: %{long: :MMMEd, medium: :MMMd, short: :Md},
year_and_month: %{long: :yMMMM, medium: :yMMM, short: :yM}
}
# The *time* interval styles are also segments by whether 12-hour
# or 24-hour time formatting is requested.
iex> Cldr.Time.Interval.styles()
%{
flex: %{h12: %{long: :Bhm, medium: :Bhm, short: :Bh}, h23: %{long: :Bhm, medium: :Bhm, short: :Bh}},
time: %{h12: %{long: :hm, medium: :hm, short: :h}, h23: %{long: :Hm, medium: :Hm, short: :H}},
zone: %{h12: %{long: :hmv, medium: :hmv, short: :hv}, h23: %{long: :Hmv, medium: :Hmv, short: :Hv}}
}Here the format style is the key of the map. For a date they
keys are :date, :month, :month_and_day and year_and_month.
These are then mapped to one of the standard interval formats :short,
:medium or :long returning a format skeleton.
Interval formats
In a manner similar to formatting discrete dates, times and date_times, standard formats are introduced to simplify common usage. For all intervals the following standard formats are:
:short:medium(the default):long
In each case, the mapping is from a format style to a standard format which then resolves to a format skeleton. The format skeleton is then used to resolve a format pattern for the interval.
These maps can be examined as follows where "en" is any configured
locale name and :gregorian is the underlying CLDR calendar type. In
common use the :gregorian calendar is the standard. However other
calendar types are also supported. For example:
iex> Cldr.known_calendars()
[:buddhist, :chinese, :coptic, :dangi, :ethiopic, :ethiopic_amete_alem,
:gregorian, :hebrew, :indian, :islamic, :islamic_civil, :islamic_rgsa,
:islamic_tbla, :islamic_umalqura, :japanese, :persian, :roc]To examine the available interval formats, Cldr.DateTime.Format.interval_formats/2
can be used although its use is primarily internal to the implementation of
to_string/3 and would not normally be called directly.
Cldr.DateTime.Format.interval_formats(:en, :gregorian)
=> {:ok,
%{
Bh: %{
B: ["h B – ", "h B"],
h: ["h – ", "h B"]
},
Bhm: %{
B: ["h:mm B – ", "h:mm B"],
h: ["h:mm – ", "h:mm B"],
m: ["h:mm – ", "h:mm B"]
},
Gy: %{
G: ["y G – ", "y G"],
y: ["y – ", "y G"]
},
GyM: %{
G: ["M/y G – ", "M/y G"],
M: ["M/y – ", "M/y G"],
y: ["M/y – ", "M/y G"]
},
GyMEd: %{
G: ["E, M/d/y G – ", "E, M/d/y G"],
M: ["E, M/d/y – ", "E, M/d/y G"],
d: ["E, M/d/y – ", "E, M/d/y G"],
y: ["E, M/d/y – ", "E, M/d/y G"]
},
...
}
}At this point we can see that the path to resolving a format (using
the :en locale for this example) is:
- Apply the format style. For dates, this is
:date, returning%{long: :yMMMEd, medium: :yMMMd, short: :yMd} - Apply the standard format. For dates, the default is
:mediumreturning the format skeleton:yMMMd. - Using the returned format skeleton, resolve the interval formats returning:
%{
d: ["MMM d – ", "d, y"],
m: ["MMM d – ", "MMM d, y"],
y: ["MMM d, y – ", "MMM d, y"]
}The field with the greatest difference
There remains one more choice to make - and that choice is made
based upon the highest order date field that is different between the
from and to dates.
With two dates 2020-02-02 and 2021-01-01 the highest order
difference is :year. With 2020-02-02 and 2020-01-01 it is :month.
Formatting the interval
Using this greatest difference information we can now resolve the
final format. With the :year field being the greatest difference then
the format is y: ["MMM d, y – ", "MMM d, y"].
Finally, formatting can proceed for the from date being formatted with
the "MMM d, y – " format pattern and the to date being formatted with the
"MMM d, y" format pattern and the two results then being concatenated to
form the final string.
Other ways to specify an interval format
So far we have considered formats that are resolved from standard styles
and stamdard formats. This is the typical usage and they are specified
as parameters to the to_string/3 function. For example:
iex> Cldr.Date.Interval.to_string ~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr
{:ok, "Jan 1 – 12, 2020"}
iex> Cldr.Date.Interval.to_string ~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr, format: :long
{:ok, "Wed, Jan 1 – Sun, Jan 12, 2020"}
iex> Cldr.Date.Interval.to_string ~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr,
...> style: :month_and_day
{:ok, "Jan 1 – 12"}Direct use of format skeletons
It is also possible to directly specify a format skeleton. For example:
iex> Cldr.Date.Interval.to_string ~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr, format: :GyMMMd
{:ok, "Jan 1 – 12, 2020 AD"}Using format patterns
In the unusual situation where one of the standard formats and format skeletons does not meet requirements, a format pattern can also be specified. For example:
iex> Cldr.Date.Interval.to_string ~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr,
...> format: "E, M/d/y – E, M/d/y"
{:ok, "Wed, 1/1/2020 – Sun, 1/12/2020"}In this case, the steps to formatting are:
Split the format pattern at the point at which the first repeating formatting code is detected. In the pattern above it is where the second
Eis detected. The result in this case will be["E, M/d/y – ", "E, M/d/y"]For the purposes of splitting, duplicates are ignored. Therefore "EEE, M/d/y – E, M/d/y" will split into["EEE, M/d/y – ", "E, M/d/y"].Each part of the pattern is parsed.
The two dates, times or date_times are formatted.
This is a more expensive operation than using the standard formats and styles since the underlying formats for these types are precompiled into an efficient runtime format.
Configuring precompiled interval formats
If there is a requirement for repeated use of format strings then they can be configured in the backend module so that they are precompiled and therefore not suffer a runtime performance penalty.
In a backend module, configure the required formats as a list under the
:precompile_interval_formats key:
defmodule MyApp.Cldr do
use Cldr,
locales: ["en", "fr"],
default_locale: "en",
precompile_interval_formats: ["E, MMM d/y – d/y"]
end
Summary
Functions
Returns a Date.Range.t/0 or CalendarInterval.t/0 as
a localised string.
Returns a string representing the formatted interval formed by two dates.
Returns a Date.Range.t/0 or CalendarInterval.t/0 as
a localised string or raises an exception.
Returns a string representing the formatted interval formed by two dates or raises an exception.
Types
@type datetime() :: Calendar.date() | Calendar.datetime() | Calendar.naive_datetime() | Calendar.time()
Any date, time or datetime
@type range() :: Date.Range.t() | CalendarInterval.t()
A Date.Range or CalendarInterval range
Functions
@spec to_string(range(), Cldr.backend(), Keyword.t()) :: {:ok, String.t()} | {:error, {module(), String.t()}}
Returns a Date.Range.t/0 or CalendarInterval.t/0 as
a localised string.
Arguments
rangeis either aDate.Range.t/0returned fromDate.range/2or aCalendarInterval.t.backendis any module that includesuse Cldrand is therefore aCldrbackend moduleoptionsis a keyword list of options. The default is[format: :medium, style: :date | :time | nil].
Options
:formatis one of:short,:mediumor:longor a specific format skeleton of format pattern representing an interval format. The default is:medium.:stylesupports different formatting styles. The valid styles depends on whether formatting is for a date, time or date_time. Since the functions in this module will make a determination as to which formatter to be used based upon the data passed to them it is recommended the style option be omitted. If styling is important then callto_string/3directly onCldr.Date.Interval,Cldr.Time.IntervalorCldr.DateTime.Interval.For a date the alternatives are
:date,:month_and_day,:monthand:year_and_month. The default is:date.For a time the alternatives are
:time,:zoneand:flex. The default is:time.For a date_time there are no style options, the default for each of the date and time parts is used.
localeis any valid locale name returned byCldr.known_locale_names/0or aCldr.LanguageTag.t/0struct. The default isCldr.get_locale/0.number_system:a number system into which the formatted date digits should be transliterated.
Returns
{:ok, string}or{:error, {exception, reason}}
Notes
to_string/3will decide which formatter to call based upon the arguments provided to it.A
Date.Range.t/0will callCldr.Date.Interval.to_string/3.A
CalendarInterval.t/0will callCldr.Date.Interval.to_string/3if its:precisionis:year,:monthor:day. Otherwise it will callCldr.Time.Interval.to_string/3.If
fromandtoboth conform to theCalendar.datetime/0type thenCldr.DateTime.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.date/0type thenCldr.Date.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.time/0type thenCldr.Time.Interval.to_string/3is called.
CalendarIntervalsupport requires adding the dependency calendar_interval to thedepsconfiguration inmix.exs.For more information on interval format patterns see
Cldr.Interval.The available format skeletons that can be applied are the keys of the map returned by
Cldr.DateTime.Format.interval_formats("en", :gregorian)where"en"can be replaced by any configuration locale name and:gregorianis the underlying CLDR calendar type.In the case where
fromandtoare equal, a single date, time or date_time is formatted instead of an interval.
Examples
iex> Cldr.Interval.to_string(Date.range(~D[2020-01-01], ~D[2020-01-12]), MyApp.Cldr,
...> format: :long)
{:ok, "Wed, Jan 1 – Sun, Jan 12, 2020"}
iex> use CalendarInterval
iex> Cldr.Interval.to_string(~I"2020-01-01/12", MyApp.Cldr,
...> format: :long)
{:ok, "Wed, Jan 1 – Sun, Jan 12, 2020"}
@spec to_string(datetime(), datetime(), Cldr.backend(), Keyword.t()) :: {:ok, String.t()} | {:error, {module(), String.t()}}
Returns a string representing the formatted interval formed by two dates.
Arguments
fromis any map that conforms to the any one of theCalendartypes.tois any map that conforms to the any one of theCalendartypes.tomust occur on or afterfrom.backendis any module that includesuse Cldrand is therefore aCldrbackend module.optionsis a keyword list of options. The default is[format: :medium, style: :date | :time | nil].
Options
:formatis one of:short,:mediumor:longor a specific format skeleton or a format pattern representing of an interval format. The default is:medium.:stylesupports different formatting styles. The valid styles depends on whether formatting is for a date, time or datetime. Since the functions in this module will make a determination as to which formatter to be used based upon the data passed to them it is recommended the style option be omitted. If styling is important then callto_string/3directly onCldr.Date.Interval,Cldr.Time.IntervalorCldr.DateTime.Interval.For a date the alternatives are
:date,:month_and_day,:monthand:year_and_month. The default is:date.For a time the alternatives are
:time,:zoneand:flex. The default is:time.For a datetime there are no style options, the default for each of the date and time part is used.
localeis any valid locale name returned byCldr.known_locale_names/0or aCldr.LanguageTag.t/0struct. The default isCldr.get_locale/0.number_system:a number system into which the formatted date digits should be transliterated.
Returns
{:ok, string}or{:error, {exception, reason}}
Notes
to_string/3will decide which formatter to call based upon the arguments provided to it.A
Date.Range.t/0will callCldr.Date.Interval.to_string/3.A
CalendarInterval.t/0will callCldr.Date.Interval.to_string/3if its:precisionis:year,:monthor:day. Otherwise it will callCldr.Time.Interval.to_string/3.If
fromandtoboth conform to either of theCalendar.datetime/0orCalendar.naive_datetime/0types thenCldr.DateTime.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.date/0type thenCldr.Date.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.time/0type thenCldr.Time.Interval.to_string/3is called.
CalendarIntervalsupport requires adding the dependency calendar_interval to thedepsconfiguration inmix.exs.For more information on interval format pattern see
Cldr.Interval.The available format skeletons that can be applied are the keys of the map returned by
Cldr.DateTime.Format.interval_formats(:en, :gregorian)where:encan be replaced by any configuration locale name and:gregorianis the underlying CLDR calendar type.In the case where
fromandtoare considered equal, a single date, time or date_time is formatted instead of an interval.
Examples
iex> Cldr.Interval.to_string(~D[2020-01-01], ~D[2020-12-31], MyApp.Cldr)
{:ok, "Jan 1 – Dec 31, 2020"}
iex> Cldr.Interval.to_string(~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr)
{:ok, "Jan 1 – 12, 2020"}
iex> Cldr.Interval.to_string(~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr,
...> format: :long)
{:ok, "Wed, Jan 1 – Sun, Jan 12, 2020"}
iex> Cldr.Interval.to_string(~D[2020-01-01], ~D[2020-12-01], MyApp.Cldr,
...> format: :long, style: :year_and_month)
{:ok, "January – December 2020"}
iex> Cldr.Interval.to_string(~U[2020-01-01 00:00:00.0Z], ~U[2020-12-01 10:05:00.0Z],
...> MyApp.Cldr, format: :long)
{:ok, "January 1, 2020, 12:00:00 AM UTC – December 1, 2020, 10:05:00 AM UTC"}
iex> Cldr.Interval.to_string(~U[2020-01-01 00:00:00.0Z], ~U[2020-01-01 10:05:00.0Z],
...> MyApp.Cldr, format: :long)
{:ok, "January 1, 2020, 12:00:00 AM UTC – 10:05:00 AM UTC"}
iex> Cldr.Interval.to_string(~N[2025-10-05 14:50:00], ~N[2025-10-05 16:30:00],
...> format: :short)
{:ok, "10/5/25, 2:50 PM – 4:30 PM"}
iex> Cldr.Interval.to_string(~N[2025-10-05 14:50:00], ~N[2025-10-05 16:30:00],
...> format: :medium)
{:ok, "Oct 5, 2025, 2:50:00 PM – 4:30:00 PM"}
@spec to_string!(range(), Cldr.backend(), Keyword.t()) :: String.t() | no_return()
Returns a Date.Range.t/0 or CalendarInterval.t/0 as
a localised string or raises an exception.
Arguments
rangeis either aDate.Range.t/0returned fromDate.range/2or aCalendarInterval.t.backendis any module that includesuse Cldrand is therefore aCldrbackend module.optionsis a keyword list of options. The default is[format: :medium, style: :date | :time | nil].
Options
:formatis one of:short,:mediumor:longor a specific format skeleton or a format pattern representing an interval format. The default is:medium.:stylesupports different formatting styles. The valid styles depends on whether formatting is for a date, time or datetime. Since the functions in this module will make a determination as to which formatter to be used based upon the data passed to them it is recommended the style option be omitted. If styling is important then callto_string/3directly onCldr.Date.Interval,Cldr.Time.IntervalorCldr.DateTime.Interval.For a date the alternatives are
:date,:month_and_day,:monthand:year_and_month. The default is:date.For a time the alternatives are
:time,:zoneand:flex. The default is:time.For a date_time there are no style options, the default for each of the date and time part is used.
localeis any valid locale name returned byCldr.known_locale_names/0or aCldr.LanguageTag.t/0struct. The default isCldr.get_locale/0.number_system:a number system into which the formatted date digits should be transliterated.
Returns
formatted_stringorraises an exception.
Notes
to_string/3will decide which formatter to call based upon the arguments provided to it.A
Date.Range.t/0will callCldr.Date.Interval.to_string/3.A
CalendarInterval.t/0will callCldr.Date.Interval.to_string/3if its:precisionis:year,:monthor:day. Otherwise it will callCldr.Time.Interval.to_string/3.If
fromandtoboth conform to either of theCalendar.datetime/0orCalendar.naive_datetime/0types thenCldr.DateTime.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.date/0type thenCldr.Date.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.time/0type thenCldr.Time.Interval.to_string/3is called.
CalendarIntervalsupport requires adding the dependency calendar_interval to thedepsconfiguration inmix.exs.For more information on interval format pattern see
Cldr.Interval.The available format skeletons that can be applied are the keys of the map returned by
Cldr.DateTime.Format.interval_formats("en", :gregorian)where"en"can be replaced by any configuration locale name and:gregorianis the underlying CLDR calendar type.In the case where
fromandtoare equal, a single date, time or date_time is formatted instead of an interval.
Examples
iex> use CalendarInterval
iex> Cldr.Interval.to_string!(~I"2020-01-01/12", MyApp.Cldr,
...> format: :long)
"Wed, Jan 1 – Sun, Jan 12, 2020"
iex> Cldr.Interval.to_string!(Date.range(~D[2020-01-01], ~D[2020-01-12]), MyApp.Cldr,
...> format: :long)
"Wed, Jan 1 – Sun, Jan 12, 2020"
Returns a string representing the formatted interval formed by two dates or raises an exception.
Arguments
fromis any map that conforms to the any one of theCalendartypes.tois any map that conforms to the any one of theCalendartypes.tomust occur on or afterfrom.backendis any module that includesuse Cldrand is therefore aCldrbackend moduleoptionsis a keyword list of options. The default is[format: :medium, style: :date | :time | nil].
Options
:formatis one of:short,:mediumor:longor a specific format skeleton or a format pattern representing an interval format. The default is:medium.:stylesupports different formatting styles. The valid styles depends on whether formatting is for a date, time or datetime. Since the functions in this module will make a determination as to which formatter to be used based upon the data passed to them it is recommended the style option be omitted. If styling is important then callto_string/3directly onCldr.Date.Interval,Cldr.Time.IntervalorCldr.DateTime.Interval.For a date the alternatives are
:date,:month_and_day,:monthand:year_and_month. The default is:date.For a time the alternatives are
:time,:zoneand:flex. The default is:time.For a date_time there are no style options, the default for each of the date and time part is used.
localeis any valid locale name returned byCldr.known_locale_names/0or aCldr.LanguageTag.t/0struct. The default isCldr.get_locale/0.number_system:a number system into which the formatted date digits should be transliterated.
Returns
formatted_stringorraises an exception.
Notes
to_string!/3will decide which formatter to call based upon the arguments provided to it.A
Date.Range.t/0will callCldr.Date.Interval.to_string/3A
CalendarInterval.t/0will callCldr.Date.Interval.to_string/3if its:precisionis:year,:monthor:day. Otherwise it will callCldr.Time.Interval.to_string/3.If
fromandtoboth conform to either of theCalendar.datetime/0orCalendar.naive_datetime/0types thenCldr.DateTime.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.date/0type thenCldr.Date.Interval.to_string/3is called.Otherwise if
fromandtoconform to theCalendar.time/0type thenCldr.Time.Interval.to_string/3is called.
CalendarIntervalsupport requires adding the dependency calendar_interval to thedepsconfiguration inmix.exs.For more information on interval format pattern see
Cldr.Interval.The available format skeletons that can be applied are the keys of the map returned by
Cldr.DateTime.Format.interval_formats("en", :gregorian)where"en"can be replaced by any configuration locale name and:gregorianis the underlying CLDR calendar type.In the case where
fromandtoare equal, a single date, time or datetime is formatted instead of an interval.
Examples
iex> Cldr.Interval.to_string!(~D[2020-01-01], ~D[2020-12-31], MyApp.Cldr)
"Jan 1 – Dec 31, 2020"
iex> Cldr.Interval.to_string!(~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr)
"Jan 1 – 12, 2020"
iex> Cldr.Interval.to_string!(~D[2020-01-01], ~D[2020-01-12], MyApp.Cldr,
...> format: :long)
"Wed, Jan 1 – Sun, Jan 12, 2020"
iex> Cldr.Interval.to_string!(~D[2020-01-01], ~D[2020-12-01], MyApp.Cldr,
...> format: :long, style: :year_and_month)
"January – December 2020"
iex> Cldr.Interval.to_string!(~U[2020-01-01 00:00:00.0Z], ~U[2020-12-01 10:05:00.0Z],
...> MyApp.Cldr, format: :long)
"January 1, 2020, 12:00:00 AM UTC – December 1, 2020, 10:05:00 AM UTC"
iex> Cldr.Interval.to_string!(~U[2020-01-01 00:00:00.0Z], ~U[2020-01-01 10:05:00.0Z],
...> MyApp.Cldr, format: :long)
"January 1, 2020, 12:00:00 AM UTC – 10:05:00 AM UTC"