Home Reference Source

packages/core/src/YearMonth.js

  1. /*
  2. * @copyright (c) 2016, Philipp Thürwächter & Pattrick Hüper
  3. * @license BSD-3-Clause (see LICENSE.md in the root directory of this source tree)
  4. */
  5.  
  6. import { requireNonNull, requireInstance } from './assert';
  7. import { DateTimeException, UnsupportedTemporalTypeException } from './errors';
  8. import { MathUtil } from './MathUtil';
  9.  
  10. import { ChronoField } from './temporal/ChronoField';
  11. import { ChronoUnit } from './temporal/ChronoUnit';
  12. import { Clock } from './Clock';
  13. import { DateTimeFormatterBuilder } from './format/DateTimeFormatterBuilder';
  14. import { IsoChronology } from './chrono/IsoChronology';
  15. import { LocalDate } from './LocalDate';
  16. import { Month } from './Month';
  17. import { SignStyle } from './format/SignStyle';
  18. import { Temporal } from './temporal/Temporal';
  19. import { TemporalField } from './temporal/TemporalField';
  20. import { TemporalQueries } from './temporal/TemporalQueries';
  21. import { TemporalQuery } from './temporal/TemporalQuery';
  22. import { TemporalUnit } from './temporal/TemporalUnit';
  23. import { createTemporalQuery } from './temporal/TemporalQuery';
  24. import { ValueRange } from './temporal/ValueRange';
  25. import { Year } from './Year';
  26. import { ZoneId } from './ZoneId';
  27.  
  28. /**
  29. * A year-month in the ISO-8601 calendar system, such as `2007-12`.
  30. *
  31. * {@link YearMonth} is an immutable date-time object that represents the combination
  32. * of a year and month. Any field that can be derived from a year and month, such as
  33. * quarter-of-year, can be obtained.
  34. *
  35. * This class does not store or represent a day, time or time-zone.
  36. * For example, the value "October 2007" can be stored in a {@link YearMonth}.
  37. *
  38. * The ISO-8601 calendar system is the modern civil calendar system used today
  39. * in most of the world. It is equivalent to the proleptic Gregorian calendar
  40. * system, in which today's rules for leap years are applied for all time.
  41. * For most applications written today, the ISO-8601 rules are entirely suitable.
  42. * However, any application that makes use of historical dates, and requires them
  43. * to be accurate will find the ISO-8601 approach unsuitable.
  44. *
  45. * ### Specification for implementors
  46. *
  47. * This class is immutable and thread-safe.
  48. */
  49. export class YearMonth extends Temporal {
  50. //-----------------------------------------------------------------------
  51. /**
  52. * function overloading for {@link YearMonth.now}
  53. *
  54. * if called with 0 argument {@link YearMonth.now0} is executed,
  55. *
  56. * if called with 1 argument and first argument is an instance of ZoneId, then {@link YearMonth.nowZoneId} is executed,
  57. *
  58. * otherwise {@link YearMonth.nowClock} is executed
  59. *
  60. * @param {?(ZoneId|Clock)} zoneIdOrClock
  61. * @returns {YearMonth}
  62. */
  63. static now(zoneIdOrClock) {
  64. if (arguments.length === 0) {
  65. return YearMonth.now0();
  66. } else if (arguments.length === 1 && zoneIdOrClock instanceof ZoneId) {
  67. return YearMonth.nowZoneId(zoneIdOrClock);
  68. } else {
  69. return YearMonth.nowClock(zoneIdOrClock);
  70. }
  71. }
  72.  
  73. /**
  74. * Obtains the current year-month from the system clock in the default time-zone.
  75. *
  76. * This will query the system clock (see {@link Clock#systemDefaultZone}) in the default
  77. * time-zone to obtain the current year-month.
  78. * The zone and offset will be set based on the time-zone in the clock.
  79. *
  80. * Using this method will prevent the ability to use an alternate clock for testing
  81. * because the clock is hard-coded.
  82. *
  83. * @return {YearMonth} the current year-month using the system clock and default time-zone, not null
  84. */
  85. static now0() {
  86. return YearMonth.nowClock(Clock.systemDefaultZone());
  87. }
  88.  
  89. /**
  90. * Obtains the current year-month from the system clock in the specified time-zone.
  91. *
  92. * This will query the system clock (see {@link Clock#system}) to obtain the current year-month.
  93. * Specifying the time-zone avoids dependence on the default time-zone.
  94. *
  95. * Using this method will prevent the ability to use an alternate clock for testing
  96. * because the clock is hard-coded.
  97. *
  98. * @param {ZoneId} zone the zone ID to use, not null
  99. * @return {YearMonth} the current year-month using the system clock, not null
  100. */
  101. static nowZoneId(zone) {
  102. return YearMonth.nowClock(Clock.system(zone));
  103. }
  104.  
  105. /**
  106. * Obtains the current year-month from the specified clock.
  107. *
  108. * This will query the specified clock to obtain the current year-month.
  109. * Using this method allows the use of an alternate clock for testing.
  110. * The alternate clock may be introduced using dependency injection.
  111. *
  112. * @param {Clock} clock the clock to use, not null
  113. * @return {YearMonth} the current year-month, not null
  114. */
  115. static nowClock(clock) {
  116. const now = LocalDate.now(clock);
  117. return YearMonth.of(now.year(), now.month());
  118. }
  119.  
  120. //-----------------------------------------------------------------------
  121. /**
  122. * function overloading for {@link YearMonth.of}
  123. *
  124. * if called with 2 argument and first argument is an instance of Month, then {@link YearMonth.ofNumberMonth} is executed,
  125. *
  126. * otherwise {@link YearMonth.ofNumberNumber} is executed
  127. *
  128. * @param {!number} year
  129. * @param {!(Month|number)} monthOrNumber
  130. * @returns {YearMonth}
  131. */
  132. static of(year, monthOrNumber) {
  133. if (arguments.length === 2 && monthOrNumber instanceof Month) {
  134. return YearMonth.ofNumberMonth(year, monthOrNumber);
  135. } else {
  136. return YearMonth.ofNumberNumber(year, monthOrNumber);
  137. }
  138. }
  139.  
  140. /**
  141. * Obtains an instance of {@link YearMonth} from a year and month.
  142. *
  143. * @param {number} year the year to represent, from MIN_YEAR to MAX_YEAR
  144. * @param {Month} month the month-of-year to represent, not null
  145. * @return {YearMonth} the year-month, not null
  146. * @throws DateTimeException if the year value is invalid
  147. */
  148. static ofNumberMonth(year, month) {
  149. requireNonNull(month, 'month');
  150. requireInstance(month, Month, 'month');
  151. return YearMonth.ofNumberNumber(year, month.value());
  152. }
  153.  
  154. /**
  155. * Obtains an instance of {@link YearMonth} from a year and month.
  156. *
  157. * @param {number} year the year to represent, from MIN_YEAR to MAX_YEAR
  158. * @param {number} month the month-of-year to represent, from 1 (January) to 12 (December)
  159. * @return {YearMonth} the year-month, not null
  160. * @throws DateTimeException if either field value is invalid
  161. */
  162. static ofNumberNumber(year, month) {
  163. requireNonNull(year, 'year');
  164. requireNonNull(month, 'month');
  165. ChronoField.YEAR.checkValidValue(year);
  166. ChronoField.MONTH_OF_YEAR.checkValidValue(month);
  167. return new YearMonth(year, month);
  168. }
  169.  
  170. //-----------------------------------------------------------------------
  171. /**
  172. * Obtains an instance of {@link YearMonth} from a temporal object.
  173. *
  174. * A {@link TemporalAccessor} represents some form of date and time information.
  175. * This factory converts the arbitrary temporal object to an instance of {@link YearMonth}.
  176. *
  177. * The conversion extracts the {@link ChronoField#YEAR} and
  178. * {@link ChronoField#MONTH_OF_YEAR} fields.
  179. * The extraction is only permitted if the temporal object has an ISO
  180. * chronology, or can be converted to a {@link LocalDate}.
  181. *
  182. * This method matches the signature of the functional interface {@link TemporalQuery}
  183. * allowing it to be used in queries via method reference, {@link YearMonth::from}.
  184. *
  185. * @param {TemporalAccessor} temporal the temporal object to convert, not null
  186. * @return {YearMonth} the year-month, not null
  187. * @throws DateTimeException if unable to convert to a {@link YearMonth}
  188. */
  189. static from(temporal) {
  190. requireNonNull(temporal, 'temporal');
  191. if (temporal instanceof YearMonth) {
  192. return temporal;
  193. }
  194. try {
  195. /* TODO: only IsoChronology for now
  196. if (IsoChronology.INSTANCE.equals(Chronology.from(temporal)) == false) {
  197. temporal = LocalDate.from(temporal);
  198. }*/
  199. return YearMonth.of(temporal.get(ChronoField.YEAR), temporal.get(ChronoField.MONTH_OF_YEAR));
  200. } catch (ex) {
  201. throw new DateTimeException(`Unable to obtain YearMonth from TemporalAccessor: ${
  202. temporal}, type ${temporal && temporal.constructor != null ? temporal.constructor.name : ''}`);
  203. }
  204. }
  205. //-----------------------------------------------------------------------
  206. /**
  207. * function overloading for {@link YearMonth.parse}
  208. *
  209. * if called with 2 argument and first argument is an instance of Month, then {@link YearMonth.parseString} is executed,
  210. *
  211. * otherwise {@link YearMonth.parseStringFormatter} is executed
  212. *
  213. * @param {!(String)} text
  214. * @param {?DateTimeFormatter} formatter
  215. * @returns {YearMonth}
  216. */
  217. static parse(text, formatter) {
  218. if (arguments.length === 1) {
  219. return YearMonth.parseString(text);
  220. } else {
  221. return YearMonth.parseStringFormatter(text, formatter);
  222. }
  223. }
  224.  
  225. /**
  226. * Obtains an instance of {@link YearMonth} from a text string such as `2007-12`.
  227. *
  228. * The string must represent a valid year-month.
  229. * The format must be {@link yyyy-MM}.
  230. * Years outside the range 0000 to 9999 must be prefixed by the plus or minus symbol.
  231. *
  232. * @param {String} text the text to parse such as "2007-12", not null
  233. * @return {YearMonth} the parsed year-month, not null
  234. * @throws DateTimeParseException if the text cannot be parsed
  235. */
  236. static parseString(text) {
  237. return YearMonth.parseStringFormatter(text, PARSER);
  238. }
  239.  
  240. /**
  241. * Obtains an instance of {@link YearMonth} from a text string using a specific formatter.
  242. *
  243. * The text is parsed using the formatter, returning a year-month.
  244. *
  245. * @param {String} text the text to parse, not null
  246. * @param {DateTimeFormatter} formatter the formatter to use, not null
  247. * @return the parsed year-month, not null
  248. * @throws DateTimeParseException if the text cannot be parsed
  249. */
  250. static parseStringFormatter(text, formatter) {
  251. requireNonNull(formatter, 'formatter');
  252. return formatter.parse(text, YearMonth.FROM);
  253. }
  254.  
  255.  
  256. /**
  257. * Constructor.
  258. *
  259. * @param {number} year the year to represent, validated from MIN_YEAR to MAX_YEAR
  260. * @param {number} month the month-of-year to represent, validated from 1 (January) to 12 (December)
  261. * @private
  262. */
  263. constructor(year, month) {
  264. super();
  265. this._year = MathUtil.safeToInt(year);
  266. this._month = MathUtil.safeToInt(month);
  267. }
  268.  
  269. /**
  270. * function overloading for {@link YearMonth.isSupported}
  271. *
  272. * if called with 1 argument and first argument is an instance of TemporalField, then {@link YearMonth.isSupportedField} is executed,
  273. *
  274. * otherwise {@link YearMonth.isSupportedUnit} is executed
  275. *
  276. * @param {!(TemporalField|ChronoUnit)} fieldOrUnit
  277. * @returns {boolean}
  278. */
  279. isSupported(fieldOrUnit) {
  280. if (arguments.length === 1 && fieldOrUnit instanceof TemporalField) {
  281. return this.isSupportedField(fieldOrUnit);
  282. } else {
  283. return this.isSupportedUnit(fieldOrUnit);
  284. }
  285. }
  286.  
  287. /**
  288. * Checks if the specified field is supported.
  289. *
  290. * This checks if this year-month can be queried for the specified field.
  291. * If false, then calling {@link range} and {@link get} will throw an exception.
  292. *
  293. * If the field is a {@link ChronoField} then the query is implemented here.
  294. * The supported fields (see {@link isSupported}) will return valid
  295. * values based on this date-time.
  296. * The supported fields are:
  297. *
  298. * * {@link MONTH_OF_YEAR}
  299. * * {@link EPOCH_MONTH}
  300. * * {@link YEAR_OF_ERA}
  301. * * {@link YEAR}
  302. * * {@link ERA}
  303. *
  304. * All other {@link ChronoField} instances will return false.
  305. *
  306. * If the field is not a {@link ChronoField}, then the result of this method
  307. * is obtained by invoking {@link TemporalField.isSupportedBy}
  308. * passing `this` as the argument.
  309. * Whether the field is supported is determined by the field.
  310. *
  311. * @param {TemporalField} field the field to check, null returns false
  312. * @return {boolean} true if the field is supported on this year-month, false if not
  313. */
  314. isSupportedField(field) {
  315. if (field instanceof ChronoField) {
  316. return field === ChronoField.YEAR || field === ChronoField.MONTH_OF_YEAR ||
  317. field === ChronoField.PROLEPTIC_MONTH || field === ChronoField.YEAR_OF_ERA || field === ChronoField.ERA;
  318. }
  319. return field != null && field.isSupportedBy(this);
  320. }
  321.  
  322. isSupportedUnit(unit) {
  323. if (unit instanceof ChronoUnit) {
  324. return unit === ChronoUnit.MONTHS || unit === ChronoUnit.YEARS || unit === ChronoUnit.DECADES || unit === ChronoUnit.CENTURIES || unit === ChronoUnit.MILLENNIA || unit === ChronoUnit.ERAS;
  325. }
  326. return unit != null && unit.isSupportedBy(this);
  327. }
  328.  
  329. /**
  330. * Gets the range of valid values for the specified field.
  331. *
  332. * The range object expresses the minimum and maximum valid values for a field.
  333. * This year-month is used to enhance the accuracy of the returned range.
  334. * If it is not possible to return the range, because the field is not supported
  335. * or for some other reason, an exception is thrown.
  336. *
  337. * If the field is a {@link ChronoField} then the query is implemented here.
  338. * The supported fields (see {@link isSupported}) will return
  339. * appropriate range instances.
  340. * All other {@link ChronoField} instances will throw a {@link DateTimeException}.
  341. *
  342. * If the field is not a {@link ChronoField}, then the result of this method
  343. * is obtained by invoking {@link TemporalField.rangeRefinedBy}
  344. * passing `this` as the argument.
  345. * Whether the range can be obtained is determined by the field.
  346. *
  347. * @param {TemporalField} field the field to query the range for, not null
  348. * @return {ValueRange} the range of valid values for the field, not null
  349. * @throws DateTimeException if the range for the field cannot be obtained
  350. */
  351. range(field) {
  352. if (field === ChronoField.YEAR_OF_ERA) {
  353. return (this.year() <= 0 ? ValueRange.of(1, Year.MAX_VALUE + 1) : ValueRange.of(1, Year.MAX_VALUE));
  354. }
  355. return super.range(field);
  356. }
  357.  
  358. /**
  359. * Gets the value of the specified field from this year-month as an `int`.
  360. *
  361. * This queries this year-month for the value for the specified field.
  362. * The returned value will always be within the valid range of values for the field.
  363. * If it is not possible to return the value, because the field is not supported
  364. * or for some other reason, an exception is thrown.
  365. *
  366. * If the field is a {@link ChronoField} then the query is implemented here.
  367. * The supported fields (see {@link isSupported}) will return valid
  368. * values based on this year-month, except {@link EPOCH_MONTH} which is too
  369. * large to fit in an `int` and throw a {@link DateTimeException}.
  370. * All other {@link ChronoField} instances will throw a {@link DateTimeException}.
  371. *
  372. * If the field is not a {@link ChronoField}, then the result of this method
  373. * is obtained by invoking {@link TemporalField.getFrom}
  374. * passing `this` as the argument. Whether the value can be obtained,
  375. * and what the value represents, is determined by the field.
  376. *
  377. * @param {TemporalField} field the field to get, not null
  378. * @return {number} the value for the field
  379. * @throws DateTimeException if a value for the field cannot be obtained
  380. * @throws ArithmeticException if numeric overflow occurs
  381. */
  382. get(field) {
  383. requireNonNull(field, 'field');
  384. requireInstance(field, TemporalField, 'field');
  385. return this.range(field).checkValidIntValue(this.getLong(field), field);
  386. }
  387.  
  388. /**
  389. * Gets the value of the specified field from this year-month as a `long`.
  390. *
  391. * This queries this year-month for the value for the specified field.
  392. * If it is not possible to return the value, because the field is not supported
  393. * or for some other reason, an exception is thrown.
  394. *
  395. * If the field is a {@link ChronoField} then the query is implemented here.
  396. * The supported fields (see {@link isSupported}) will return valid
  397. * values based on this year-month.
  398. * All other {@link ChronoField} instances will throw a {@link DateTimeException}.
  399. *
  400. * If the field is not a {@link ChronoField}, then the result of this method
  401. * is obtained by invoking {@link TemporalField.getFrom}
  402. * passing `this` as the argument. Whether the value can be obtained,
  403. * and what the value represents, is determined by the field.
  404. *
  405. * @param {TemporalField} field the field to get, not null
  406. * @return {number} the value for the field
  407. * @throws DateTimeException if a value for the field cannot be obtained
  408. * @throws ArithmeticException if numeric overflow occurs
  409. */
  410. getLong(field) {
  411. requireNonNull(field, 'field');
  412. requireInstance(field, TemporalField, 'field');
  413. if (field instanceof ChronoField) {
  414. switch (field) {
  415. case ChronoField.MONTH_OF_YEAR: return this._month;
  416. case ChronoField.PROLEPTIC_MONTH: return this._getProlepticMonth();
  417. case ChronoField.YEAR_OF_ERA: return (this._year < 1 ? 1 - this._year : this._year);
  418. case ChronoField.YEAR: return this._year;
  419. case ChronoField.ERA: return (this._year < 1 ? 0 : 1);
  420. }
  421. throw new UnsupportedTemporalTypeException(`Unsupported field: ${field}`);
  422. }
  423. return field.getFrom(this);
  424. }
  425.  
  426. _getProlepticMonth() {
  427. return MathUtil.safeAdd(MathUtil.safeMultiply(this._year, 12), (this._month - 1));
  428. }
  429.  
  430. //-----------------------------------------------------------------------
  431. /**
  432. * Gets the year field.
  433. *
  434. * This method returns the primitive `int` value for the year.
  435. *
  436. * The year returned by this method is proleptic as per {@link get}.
  437. *
  438. * @return {number} the year, from MIN_YEAR to MAX_YEAR
  439. */
  440. year() {
  441. return this._year;
  442. }
  443.  
  444. /**
  445. * Gets the month-of-year field from 1 to 12.
  446. *
  447. * This method returns the month as an `int` from 1 to 12.
  448. * Application code is frequently clearer if the enum {@link Month}
  449. * is used by calling {@link getMonth}.
  450. *
  451. * @return {number} the month-of-year, from 1 to 12
  452. * @see #getMonth()
  453. */
  454. monthValue() {
  455. return this._month;
  456. }
  457.  
  458. /**
  459. * Gets the month-of-year field using the {@link Month} enum.
  460. *
  461. * This method returns the enum {@link Month} for the month.
  462. * This avoids confusion as to what `int` values mean.
  463. * If you need access to the primitive `int` value, use {@link Month#getValue}.
  464. *
  465. * @return {Month} the month-of-year, not null
  466. */
  467. month() {
  468. return Month.of(this._month);
  469. }
  470.  
  471. //-----------------------------------------------------------------------
  472. /**
  473. * Checks if the year is a leap year, according to the ISO proleptic
  474. * calendar system rules.
  475. *
  476. * This method applies the current rules for leap years across the whole time-line.
  477. * In general, a year is a leap year if it is divisible by four without
  478. * remainder. However, years divisible by 100, are not leap years, with
  479. * the exception of years divisible by 400 which are.
  480. *
  481. * For example, 1904 is a leap year it is divisible by 4.
  482. * 1900 was not a leap year as it is divisible by 100, however 2000 was a
  483. * leap year as it is divisible by 400.
  484. *
  485. * The calculation is proleptic - applying the same rules into the far future and far past.
  486. * This is historically inaccurate, but is correct for the ISO-8601 standard.
  487. *
  488. * @return {boolean} true if the year is leap, false otherwise
  489. */
  490. isLeapYear() {
  491. return IsoChronology.isLeapYear(this._year);
  492. }
  493.  
  494. /**
  495. * Checks if the day-of-month is valid for this year-month.
  496. *
  497. * This method checks whether this year and month and the input day form
  498. * a valid date.
  499. *
  500. * @param {number} dayOfMonth the day-of-month to validate, from 1 to 31, invalid value returns false
  501. * @return {boolean} true if the day is valid for this year-month
  502. */
  503. isValidDay(dayOfMonth) {
  504. return dayOfMonth >= 1 && dayOfMonth <= this.lengthOfMonth();
  505. }
  506.  
  507. /**
  508. * Returns the length of the month, taking account of the year.
  509. *
  510. * This returns the length of the month in days.
  511. * For example, a date in January would return 31.
  512. *
  513. * @return {number} the length of the month in days, from 28 to 31
  514. */
  515. lengthOfMonth() {
  516. return this.month().length(this.isLeapYear());
  517. }
  518.  
  519. /**
  520. * Returns the length of the year.
  521. *
  522. * This returns the length of the year in days, either 365 or 366.
  523. *
  524. * @return {number} 366 if the year is leap, 365 otherwise
  525. */
  526. lengthOfYear() {
  527. return (this.isLeapYear() ? 366 : 365);
  528. }
  529.  
  530. /**
  531. * function overloading for {@link YearMonth.with}
  532. *
  533. * if called with 1 argument, then {@link YearMonth.withAdjuster} is executed,
  534. * otherwise {@link YearMonth.withFieldValue} is executed.
  535. *
  536. * @param {!(TemporalAdjuster|TemporalField)} adjusterOrField
  537. * @param {?number} value nullable only of first argument is an instance of TemporalAdjuster
  538. * @returns {YearMonth}
  539. */
  540. with(adjusterOrField, value) {
  541. if (arguments.length === 1) {
  542. return this._withAdjuster(adjusterOrField);
  543. } else {
  544. return this._withField(adjusterOrField, value);
  545. }
  546. }
  547.  
  548. /**
  549. * Returns a copy of this year-month with the specified field set to a new value.
  550. *
  551. * This returns a new {@link YearMonth}, based on this one, with the value
  552. * for the specified field changed.
  553. * This can be used to change any supported field, such as the year or month.
  554. * If it is not possible to set the value, because the field is not supported or for
  555. * some other reason, an exception is thrown.
  556. *
  557. * If the field is a {@link ChronoField} then the adjustment is implemented here.
  558. * The supported fields behave as follows:
  559. *
  560. * * {@link MONTH_OF_YEAR} -
  561. * Returns a {@link YearMonth} with the specified month-of-year.
  562. * The year will be unchanged.
  563. * * {@link PROLEPTIC_MONTH} -
  564. * Returns a {@link YearMonth} with the specified proleptic-month.
  565. * This completely replaces the year and month of this object.
  566. * * {@link YEAR_OF_ERA} -
  567. * Returns a {@link YearMonth} with the specified year-of-era
  568. * The month and era will be unchanged.
  569. * * {@link YEAR} -
  570. * Returns a {@link YearMonth} with the specified year.
  571. * The month will be unchanged.
  572. * * {@link ERA} -
  573. * Returns a {@link YearMonth} with the specified era.
  574. * The month and year-of-era will be unchanged.
  575. *
  576. * In all cases, if the new value is outside the valid range of values for the field
  577. * then a {@link DateTimeException} will be thrown.
  578. *
  579. * All other {@link ChronoField} instances will throw a {@link DateTimeException}.
  580. *
  581. * If the field is not a {@link ChronoField}, then the result of this method
  582. * is obtained by invoking {@link TemporalField.adjustInto}
  583. * passing `this` as the argument. In this case, the field determines
  584. * whether and how to adjust the instant.
  585. *
  586. * This instance is immutable and unaffected by this method call.
  587. *
  588. * @param {TemporalField} field the field to set in the result, not null
  589. * @param {number} newValue the new value of the field in the result
  590. * @return a {@link YearMonth} based on `this` with the specified field set, not null
  591. * @throws DateTimeException if the field cannot be set
  592. * @throws ArithmeticException if numeric overflow occurs
  593. */
  594. _withField(field, newValue) {
  595. requireNonNull(field, 'field');
  596. requireInstance(field, TemporalField, 'field');
  597. if (field instanceof ChronoField) {
  598. const f = field;
  599. f.checkValidValue(newValue);
  600. switch (f) {
  601. case ChronoField.MONTH_OF_YEAR: return this.withMonth(newValue);
  602. case ChronoField.PROLEPTIC_MONTH: return this.plusMonths(newValue - this.getLong(ChronoField.PROLEPTIC_MONTH));
  603. case ChronoField.YEAR_OF_ERA: return this.withYear((this._year < 1 ? 1 - newValue : newValue));
  604. case ChronoField.YEAR: return this.withYear(newValue);
  605. case ChronoField.ERA: return (this.getLong(ChronoField.ERA) === newValue ? this : this.withYear(1 - this._year));
  606. }
  607. throw new UnsupportedTemporalTypeException(`Unsupported field: ${field}`);
  608. }
  609. return field.adjustInto(this, newValue);
  610. }
  611.  
  612. //-----------------------------------------------------------------------
  613. /**
  614. * Returns a copy of this {@link YearMonth} with the year altered.
  615. *
  616. * This instance is immutable and unaffected by this method call.
  617. *
  618. * @param {number} year the year to set in the returned year-month, from MIN_YEAR to MAX_YEAR
  619. * @return {YearMonth} based on this year-month with the requested year, not null
  620. * @throws DateTimeException if the year value is invalid
  621. */
  622. withYear(year) {
  623. ChronoField.YEAR.checkValidValue(year);
  624. return new YearMonth(year, this._month);
  625. }
  626.  
  627. /**
  628. * Returns a copy of this {@link YearMonth} with the month-of-year altered.
  629. *
  630. * This instance is immutable and unaffected by this method call.
  631. *
  632. * @param {number} month the month-of-year to set in the returned year-month, from 1 (January) to 12 (December)
  633. * @return {YearMonth} based on this year-month with the requested month, not null
  634. * @throws DateTimeException if the month-of-year value is invalid
  635. */
  636. withMonth(month) {
  637. ChronoField.MONTH_OF_YEAR.checkValidValue(month);
  638. return new YearMonth(this._year, month);
  639. }
  640.  
  641. //-----------------------------------------------------------------------
  642.  
  643. /**
  644. * @param {number} amountToAdd
  645. * @param {TemporalUnit} unit
  646. * @return {YearMonth} based on this year-month with the addition made, not null
  647. * @throws DateTimeException if the addition cannot be made
  648. * @throws ArithmeticException if numeric overflow occurs
  649. */
  650. _plusUnit(amountToAdd, unit) {
  651. requireNonNull(unit, 'unit');
  652. requireInstance(unit, TemporalUnit, 'unit');
  653. if (unit instanceof ChronoUnit) {
  654. switch (unit) {
  655. case ChronoUnit.MONTHS: return this.plusMonths(amountToAdd);
  656. case ChronoUnit.YEARS: return this.plusYears(amountToAdd);
  657. case ChronoUnit.DECADES: return this.plusYears(MathUtil.safeMultiply(amountToAdd, 10));
  658. case ChronoUnit.CENTURIES: return this.plusYears(MathUtil.safeMultiply(amountToAdd, 100));
  659. case ChronoUnit.MILLENNIA: return this.plusYears(MathUtil.safeMultiply(amountToAdd, 1000));
  660. case ChronoUnit.ERAS: return this.with(ChronoField.ERA, MathUtil.safeAdd(this.getLong(ChronoField.ERA), amountToAdd));
  661. }
  662. throw new UnsupportedTemporalTypeException(`Unsupported unit: ${unit}`);
  663. }
  664. return unit.addTo(this, amountToAdd);
  665. }
  666.  
  667. /**
  668. * Returns a copy of this year-month with the specified period in years added.
  669. *
  670. * This instance is immutable and unaffected by this method call.
  671. *
  672. * @param {number} yearsToAdd the years to add, may be negative
  673. * @return {YearMonth} based on this year-month with the years added, not null
  674. * @throws DateTimeException if the result exceeds the supported range
  675. */
  676. plusYears(yearsToAdd) {
  677. if (yearsToAdd === 0) {
  678. return this;
  679. }
  680. const newYear = ChronoField.YEAR.checkValidIntValue(this._year + yearsToAdd); // safe overflow
  681. return this.withYear(newYear);
  682. }
  683.  
  684. /**
  685. * Returns a copy of this year-month with the specified period in months added.
  686. *
  687. * This instance is immutable and unaffected by this method call.
  688. *
  689. * @param {number} monthsToAdd the months to add, may be negative
  690. * @return {YearMonth} based on this year-month with the months added, not null
  691. * @throws DateTimeException if the result exceeds the supported range
  692. */
  693. plusMonths(monthsToAdd) {
  694. if (monthsToAdd === 0) {
  695. return this;
  696. }
  697. const monthCount = (this._year * 12) + (this._month - 1);
  698. const calcMonths = monthCount + monthsToAdd;
  699. const newYear = ChronoField.YEAR.checkValidIntValue(MathUtil.floorDiv(calcMonths, 12));
  700. const newMonth = MathUtil.floorMod(calcMonths, 12) + 1;
  701. return new YearMonth(newYear, newMonth);
  702. }
  703.  
  704. //-----------------------------------------------------------------------
  705.  
  706. /**
  707. * Returns a copy of this year-month with the specified period in years subtracted.
  708. *
  709. * This instance is immutable and unaffected by this method call.
  710. *
  711. * @param {number} yearsToSubtract the years to subtract, may be negative
  712. * @return {YearMonth} based on this year-month with the years subtracted, not null
  713. * @throws DateTimeException if the result exceeds the supported range
  714. */
  715. minusYears(yearsToSubtract) {
  716. return (yearsToSubtract === MathUtil.MIN_SAFE_INTEGER ? this.plusYears(MathUtil.MIN_SAFE_INTEGER).plusYears(1) : this.plusYears(-yearsToSubtract));
  717. }
  718.  
  719. /**
  720. * Returns a copy of this year-month with the specified period in months subtracted.
  721. *
  722. * This instance is immutable and unaffected by this method call.
  723. *
  724. * @param {number} monthsToSubtract the months to subtract, may be negative
  725. * @return {YearMonth} based on this year-month with the months subtracted, not null
  726. * @throws DateTimeException if the result exceeds the supported range
  727. */
  728. minusMonths(monthsToSubtract) {
  729. return (monthsToSubtract === MathUtil.MIN_SAFE_INTEGER ? this.plusMonths(Math.MAX_SAFE_INTEGER).plusMonths(1) : this.plusMonths(-monthsToSubtract));
  730. }
  731.  
  732. //-----------------------------------------------------------------------
  733. /**
  734. * Queries this year-month using the specified query.
  735. *
  736. * This queries this year-month using the specified query strategy object.
  737. * The {@link TemporalQuery} object defines the logic to be used to
  738. * obtain the result. Read the documentation of the query to understand
  739. * what the result of this method will be.
  740. *
  741. * The result of this method is obtained by invoking the
  742. * {@link TemporalQuery#queryFrom} method on the
  743. * specified query passing `this` as the argument.
  744. *
  745. * @param {TemporalQuery} query the query to invoke, not null
  746. * @return {*} the query result, null may be returned (defined by the query)
  747. * @throws DateTimeException if unable to query (defined by the query)
  748. * @throws ArithmeticException if numeric overflow occurs (defined by the query)
  749. */
  750. query(query) {
  751. requireNonNull(query, 'query');
  752. requireInstance(query, TemporalQuery, 'query');
  753. if (query === TemporalQueries.chronology()) {
  754. return IsoChronology.INSTANCE;
  755. } else if (query === TemporalQueries.precision()) {
  756. return ChronoUnit.MONTHS;
  757. } else if (query === TemporalQueries.localDate() || query === TemporalQueries.localTime() ||
  758. query === TemporalQueries.zone() || query === TemporalQueries.zoneId() || query === TemporalQueries.offset()) {
  759. return null;
  760. }
  761. return super.query(query);
  762. }
  763.  
  764. /**
  765. * Adjusts the specified temporal object to have this year-month.
  766. *
  767. * This returns a temporal object of the same observable type as the input
  768. * with the year and month changed to be the same as this.
  769. *
  770. * The adjustment is equivalent to using {@link Temporal#with}
  771. * passing {@link ChronoField#PROLEPTIC_MONTH} as the field.
  772. * If the specified temporal object does not use the ISO calendar system then
  773. * a {@link DateTimeException} is thrown.
  774. *
  775. * In most cases, it is clearer to reverse the calling pattern by using
  776. * {@link Temporal#with}:
  777. * <pre>
  778. * // these two lines are equivalent, but the second approach is recommended
  779. * temporal = thisYearMonth.adjustInto(temporal);
  780. * temporal = temporal.with(thisYearMonth);
  781. * </pre>
  782. *
  783. * This instance is immutable and unaffected by this method call.
  784. *
  785. * @param {Temporal} temporal the target object to be adjusted, not null
  786. * @return {Temporal} the adjusted object, not null
  787. * @throws DateTimeException if unable to make the adjustment
  788. * @throws ArithmeticException if numeric overflow occurs
  789. */
  790. adjustInto(temporal) {
  791. requireNonNull(temporal, 'temporal');
  792. requireInstance(temporal, Temporal, 'temporal');
  793. /* TODO: only IsoChronology for now
  794. if (Chronology.from(temporal).equals(IsoChronology.INSTANCE) == false) {
  795. throw new DateTimeException("Adjustment only supported on ISO date-time");
  796. }*/
  797. return temporal.with(ChronoField.PROLEPTIC_MONTH, this._getProlepticMonth());
  798. }
  799.  
  800. /**
  801. * Calculates the period between this year-month and another year-month in
  802. * terms of the specified unit.
  803. *
  804. * This calculates the period between two year-months in terms of a single unit.
  805. * The start and end points are `this` and the specified year-month.
  806. * The result will be negative if the end is before the start.
  807. * The {@link Temporal} passed to this method must be a {@link YearMonth}.
  808. * For example, the period in years between two year-months can be calculated
  809. * using {@link startYearMonth.until}.
  810. *
  811. * The calculation returns a whole number, representing the number of
  812. * complete units between the two year-months.
  813. * For example, the period in decades between 2012-06 and 2032-05
  814. * will only be one decade as it is one month short of two decades.
  815. *
  816. * This method operates in association with {@link TemporalUnit#between}.
  817. * The result of this method is a `long` representing the amount of
  818. * the specified unit. By contrast, the result of {@link between} is an
  819. * object that can be used directly in addition/subtraction:
  820. * <pre>
  821. * long period = start.until(end, YEARS); // this method
  822. * dateTime.plus(YEARS.between(start, end)); // use in plus/minus
  823. * </pre>
  824. *
  825. * The calculation is implemented in this method for {@link ChronoUnit}.
  826. * The units {@link MONTHS}, {@link YEARS}, {@link DECADES},
  827. * {@link CENTURIES}, {@link MILLENNIA} and {@link ERAS} are supported.
  828. * Other {@link ChronoUnit} values will throw an exception.
  829. *
  830. * If the unit is not a {@link ChronoUnit}, then the result of this method
  831. * is obtained by invoking {@link TemporalUnit.between}
  832. * passing `this` as the first argument and the input temporal as
  833. * the second argument.
  834. *
  835. * This instance is immutable and unaffected by this method call.
  836. *
  837. * @param {Temporal} endExclusive the end year-month, which is converted to a {@link YearMonth}, not null
  838. * @param {TemporalUnit} unit the unit to measure the period in, not null
  839. * @return {number} the amount of the period between this year-month and the end year-month
  840. * @throws DateTimeException if the period cannot be calculated
  841. * @throws ArithmeticException if numeric overflow occurs
  842. */
  843. until(endExclusive, unit) {
  844. requireNonNull(endExclusive, 'endExclusive');
  845. requireNonNull(unit, 'unit');
  846. requireInstance(endExclusive, Temporal, 'endExclusive');
  847. requireInstance(unit, TemporalUnit, 'unit');
  848.  
  849. const end = YearMonth.from(endExclusive);
  850. if (unit instanceof ChronoUnit) {
  851. const monthsUntil = end._getProlepticMonth() - this._getProlepticMonth(); // no overflow
  852. switch (unit) {
  853. case ChronoUnit.MONTHS: return monthsUntil;
  854. case ChronoUnit.YEARS: return MathUtil.intDiv(monthsUntil, 12);
  855. case ChronoUnit.DECADES: return MathUtil.intDiv(monthsUntil, 120);
  856. case ChronoUnit.CENTURIES: return MathUtil.intDiv(monthsUntil, 1200);
  857. case ChronoUnit.MILLENNIA: return MathUtil.intDiv(monthsUntil, 12000);
  858. case ChronoUnit.ERAS: return end.getLong(ChronoField.ERA) - this.getLong(ChronoField.ERA);
  859. }
  860. throw new UnsupportedTemporalTypeException(`Unsupported unit: ${unit}`);
  861. }
  862. return unit.between(this, end);
  863. }
  864.  
  865. //-----------------------------------------------------------------------
  866. /**
  867. * Combines this year-month with a day-of-month to create a {@link LocalDate}.
  868. *
  869. * This returns a {@link LocalDate} formed from this year-month and the specified day-of-month.
  870. *
  871. * The day-of-month value must be valid for the year-month.
  872. *
  873. * This method can be used as part of a chain to produce a date:
  874. * <pre>
  875. * LocalDate date = year.atMonth(month).atDay(day);
  876. * </pre>
  877. *
  878. * @param {number} dayOfMonth the day-of-month to use, from 1 to 31
  879. * @return {LocalDate} the date formed from this year-month and the specified day, not null
  880. * @throws DateTimeException if the day is invalid for the year-month
  881. * @see #isValidDay(int)
  882. */
  883. atDay(dayOfMonth) {
  884. requireNonNull(dayOfMonth, 'dayOfMonth');
  885. return LocalDate.of(this._year, this._month, dayOfMonth);
  886. }
  887.  
  888. /**
  889. * Returns a {@link LocalDate} at the end of the month.
  890. *
  891. * This returns a {@link LocalDate} based on this year-month.
  892. * The day-of-month is set to the last valid day of the month, taking
  893. * into account leap years.
  894. *
  895. * This method can be used as part of a chain to produce a date:
  896. * <pre>
  897. * LocalDate date = year.atMonth(month).atEndOfMonth();
  898. * </pre>
  899. *
  900. * @return {LocalDate} the last valid date of this year-month, not null
  901. */
  902. atEndOfMonth() {
  903. return LocalDate.of(this._year, this._month, this.lengthOfMonth());
  904. }
  905.  
  906. //-----------------------------------------------------------------------
  907. /**
  908. * Compares this year-month to another year-month.
  909. *
  910. * The comparison is based first on the value of the year, then on the value of the month.
  911. * It is "consistent with equals", as defined by {@link Comparable}.
  912. *
  913. * @param {YearMonth} other the other year-month to compare to, not null
  914. * @return {number} the comparator value, negative if less, positive if greater
  915. */
  916. compareTo(other) {
  917. requireNonNull(other, 'other');
  918. requireInstance(other, YearMonth, 'other');
  919. let cmp = (this._year - other.year());
  920. if (cmp === 0) {
  921. cmp = (this._month - other.monthValue());
  922. }
  923. return cmp;
  924. }
  925.  
  926. /**
  927. * Is this year-month after the specified year-month.
  928. *
  929. * @param {YearMonth} other the other year-month to compare to, not null
  930. * @return {boolean} true if this is after the specified year-month
  931. */
  932. isAfter(other) {
  933. return this.compareTo(other) > 0;
  934. }
  935.  
  936. /**
  937. * Is this year-month before the specified year-month.
  938. *
  939. * @param {YearMonth} other the other year-month to compare to, not null
  940. * @return {boolean} true if this point is before the specified year-month
  941. */
  942. isBefore(other) {
  943. return this.compareTo(other) < 0;
  944. }
  945.  
  946. //-----------------------------------------------------------------------
  947. /**
  948. * Checks if this year-month is equal to another year-month.
  949. *
  950. * The comparison is based on the time-line position of the year-months.
  951. *
  952. * @param {*} obj the object to check, null returns false
  953. * @return {boolean} true if this is equal to the other year-month
  954. */
  955. equals(obj) {
  956. if (this === obj) {
  957. return true;
  958. }
  959. if (obj instanceof YearMonth) {
  960. const other = obj;
  961. return this.year() === other.year() && this.monthValue() === other.monthValue();
  962. }
  963. return false;
  964. }
  965.  
  966. //-----------------------------------------------------------------------
  967. /**
  968. * Outputs this year-month as a string, such as `2007-12`.
  969. *
  970. * The output will be in the format {@link yyyy-MM}:
  971. *
  972. * @return {String} a string representation of this year-month, not null
  973. */
  974. toString() {
  975. return PARSER.format(this);
  976. }
  977.  
  978. /**
  979. * toJSON() use by JSON.stringify
  980. * delegates to toString()
  981. *
  982. * @return {string}
  983. */
  984. toJSON() {
  985. return this.toString();
  986. }
  987.  
  988. /**
  989. * Outputs this year-month as a string using the formatter.
  990. *
  991. * @param {DateTimeFormatter} formatter the formatter to use, not null
  992. * @return {String} the formatted year-month string, not null
  993. * @throws DateTimeException if an error occurs during printing
  994. */
  995. format(formatter) {
  996. requireNonNull(formatter, 'formatter');
  997. return formatter.format(this);
  998. }
  999.  
  1000. }
  1001.  
  1002. let PARSER;
  1003.  
  1004. export function _init() {
  1005.  
  1006. PARSER = new DateTimeFormatterBuilder()
  1007. .appendValue(ChronoField.YEAR, 4, 10, SignStyle.EXCEEDS_PAD)
  1008. .appendLiteral('-')
  1009. .appendValue(ChronoField.MONTH_OF_YEAR, 2)
  1010. .toFormatter();
  1011.  
  1012. YearMonth.FROM = createTemporalQuery('YearMonth.FROM', (temporal) => {
  1013. return YearMonth.from(temporal);
  1014. });
  1015. }