CalendarUtils.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */

  17. package org.apache.commons.lang3.time;

  18. import java.time.LocalDateTime;
  19. import java.time.OffsetDateTime;
  20. import java.time.ZoneId;
  21. import java.time.ZonedDateTime;
  22. import java.util.Calendar;
  23. import java.util.Locale;
  24. import java.util.Locale.Category;
  25. import java.util.Map;
  26. import java.util.Objects;

  27. /**
  28.  * Helps use {@link Calendar}s.
  29.  *
  30.  * @since 3.10
  31.  */
  32. public class CalendarUtils {

  33.     /**
  34.      * The singleton instance for {@link Calendar#getInstance()}. The instance is created when the class is initialized and is based on the current time in the
  35.      * default time zone with the default {@link Category#FORMAT} locale.
  36.      *
  37.      * @see CalendarUtils#getInstance()
  38.      */
  39.     public static final CalendarUtils INSTANCE = getInstance();

  40.     /**
  41.      * Creates a new instance based on the current time in the default time zone with the default {@link Category#FORMAT} locale.
  42.      *
  43.      * @return a new instance.
  44.      * @since 3.14.0
  45.      */
  46.     public static CalendarUtils getInstance() {
  47.         return new CalendarUtils(Calendar.getInstance());
  48.     }

  49.     /**
  50.      * Gets a CalendarUtils using the default time zone and specified locale. The {@code CalendarUtils} returned is based on the current time in the
  51.      * default time zone with the given locale.
  52.      *
  53.      * @param locale the locale for the week data
  54.      * @return a Calendar.
  55.      */
  56.     static CalendarUtils getInstance(final Locale locale) {
  57.         return new CalendarUtils(Calendar.getInstance(locale), locale);
  58.     }

  59.     /**
  60.      * Converts a Calendar to a LocalDateTime.
  61.      *
  62.      * @param calendar the Calendar to convert.
  63.      * @return a LocalDateTime.
  64.      * @since 3.17.0
  65.      */
  66.     public static LocalDateTime toLocalDateTime(final Calendar calendar) {
  67.         return LocalDateTime.ofInstant(calendar.toInstant(), toZoneId(calendar));
  68.     }

  69.     /**
  70.      * Converts a Calendar to a OffsetDateTime.
  71.      *
  72.      * @param calendar the Calendar to convert.
  73.      * @return a OffsetDateTime.
  74.      * @since 3.17.0
  75.      */
  76.     public static OffsetDateTime toOffsetDateTime(final Calendar calendar) {
  77.         return OffsetDateTime.ofInstant(calendar.toInstant(), toZoneId(calendar));
  78.     }

  79.     /**
  80.      * Converts a Calendar to a ZonedDateTime.
  81.      *
  82.      * @param calendar the Calendar to convert.
  83.      * @return a ZonedDateTime.
  84.      * @since 3.17.0
  85.      */
  86.     public static ZonedDateTime toZonedDateTime(final Calendar calendar) {
  87.         return ZonedDateTime.ofInstant(calendar.toInstant(), toZoneId(calendar));
  88.     }

  89.     private static ZoneId toZoneId(final Calendar calendar) {
  90.         return calendar.getTimeZone().toZoneId();
  91.     }

  92.     private final Calendar calendar;

  93.     private final Locale locale;

  94.     /**
  95.      * Creates an instance for the given Calendar.
  96.      *
  97.      * @param calendar A Calendar.
  98.      */
  99.     public CalendarUtils(final Calendar calendar) {
  100.         this(calendar, Locale.getDefault());
  101.     }
  102.     /**
  103.      * Creates an instance for the given Calendar.
  104.      *
  105.      * @param calendar A Calendar.
  106.      * @param locale A Locale.
  107.      */
  108.     CalendarUtils(final Calendar calendar, final Locale locale) {
  109.         this.calendar = Objects.requireNonNull(calendar, "calendar");
  110.         this.locale = Objects.requireNonNull(locale, "locale");
  111.     }

  112.     /**
  113.      * Gets the current day of month.
  114.      *
  115.      * @return the current day of month.
  116.      */
  117.     public int getDayOfMonth() {
  118.         return calendar.get(Calendar.DAY_OF_MONTH);
  119.     }

  120.     /**
  121.      * Gets the current day of year.
  122.      *
  123.      * @return the current day of year.
  124.      * @since 3.13.0
  125.      */
  126.     public int getDayOfYear() {
  127.         return calendar.get(Calendar.DAY_OF_YEAR);
  128.     }

  129.     /**
  130.      * Gets the current month.
  131.      *
  132.      * @return the current month.
  133.      */
  134.     public int getMonth() {
  135.         return calendar.get(Calendar.MONTH);
  136.     }

  137.     /**
  138.      * Gets month names in the requested style.
  139.      * @param style Must be a valid {@link Calendar#getDisplayNames(int, int, Locale)} month style.
  140.      * @return Styled names of months
  141.      */
  142.     String[] getMonthDisplayNames(final int style) {
  143.         // Unfortunately standalone month names are not available in DateFormatSymbols,
  144.         // so we have to extract them.
  145.         final Map<String, Integer> displayNames = calendar.getDisplayNames(Calendar.MONTH, style, locale);
  146.         if (displayNames == null) {
  147.             return null;
  148.         }
  149.         final String[] monthNames = new String[displayNames.size()];
  150.         displayNames.forEach((k, v) -> monthNames[v] = k);
  151.         return monthNames;
  152.     }

  153.     /**
  154.      * Gets full standalone month names as used in "LLLL" date formatting.
  155.      * @return Long names of months
  156.      */
  157.     String[] getStandaloneLongMonthNames() {
  158.         return getMonthDisplayNames(Calendar.LONG_STANDALONE);
  159.     }

  160.     /**
  161.      * Gets short standalone month names as used in "LLLL" date formatting.
  162.      * @return Short names of months
  163.      */
  164.     String[] getStandaloneShortMonthNames() {
  165.         return getMonthDisplayNames(Calendar.SHORT_STANDALONE);
  166.     }

  167.     /**
  168.      * Gets the current year.
  169.      *
  170.      * @return the current year.
  171.      */
  172.     public int getYear() {
  173.         return calendar.get(Calendar.YEAR);
  174.     }

  175.     /**
  176.      * Converts this instance to a {@link LocalDateTime}.
  177.      *
  178.      * @return a LocalDateTime.
  179.      * @since 3.17.0
  180.      */
  181.     public LocalDateTime toLocalDateTime() {
  182.         return toLocalDateTime(calendar);
  183.     }

  184.     /**
  185.      * Converts this instance to a {@link OffsetDateTime}.
  186.      *
  187.      * @return a OffsetDateTime.
  188.      * @since 3.17.0
  189.      */
  190.     public OffsetDateTime toOffsetDateTime() {
  191.         return toOffsetDateTime(calendar);
  192.     }

  193.     /**
  194.      * Converts this instance to a {@link ZonedDateTime}.
  195.      *
  196.      * @return a ZonedDateTime.
  197.      * @since 3.17.0
  198.      */
  199.     public ZonedDateTime toZonedDateTime() {
  200.         return toZonedDateTime(calendar);
  201.     }

  202. }