org.threeten.bp.chrono.HijrahDate Maven / Gradle / Ivy
Show all versions of threetenbp Show documentation
/*
* Copyright (c) 2007-present, Stephen Colebourne & Michael Nascimento Santos
*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* * Neither the name of JSR-310 nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.threeten.bp.chrono;
import static org.threeten.bp.temporal.ChronoField.ALIGNED_DAY_OF_WEEK_IN_MONTH;
import static org.threeten.bp.temporal.ChronoField.ALIGNED_DAY_OF_WEEK_IN_YEAR;
import static org.threeten.bp.temporal.ChronoField.ALIGNED_WEEK_OF_MONTH;
import static org.threeten.bp.temporal.ChronoField.ALIGNED_WEEK_OF_YEAR;
import static org.threeten.bp.temporal.ChronoField.DAY_OF_MONTH;
import static org.threeten.bp.temporal.ChronoField.MONTH_OF_YEAR;
import static org.threeten.bp.temporal.ChronoField.YEAR;
import java.io.BufferedReader;
import java.io.DataInput;
import java.io.DataOutput;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Serializable;
import java.text.ParseException;
import java.util.HashMap;
import java.util.StringTokenizer;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import org.threeten.bp.Clock;
import org.threeten.bp.DateTimeException;
import org.threeten.bp.DayOfWeek;
import org.threeten.bp.LocalDate;
import org.threeten.bp.LocalTime;
import org.threeten.bp.ZoneId;
import org.threeten.bp.jdk8.Jdk8Methods;
import org.threeten.bp.temporal.ChronoField;
import org.threeten.bp.temporal.TemporalAccessor;
import org.threeten.bp.temporal.TemporalAdjuster;
import org.threeten.bp.temporal.TemporalAmount;
import org.threeten.bp.temporal.TemporalField;
import org.threeten.bp.temporal.TemporalQuery;
import org.threeten.bp.temporal.TemporalUnit;
import org.threeten.bp.temporal.UnsupportedTemporalTypeException;
import org.threeten.bp.temporal.ValueRange;
/**
* A date in the Hijrah calendar system.
*
* This implements {@code ChronoLocalDate} for the {@link HijrahChronology Hijrah calendar}.
*
* The Hijrah calendar has a different total of days in a year than
* Gregorian calendar, and a month is based on the period of a complete
* revolution of the moon around the earth (as between successive new moons).
* The calendar cycles becomes longer and unstable, and sometimes a manual
* adjustment (for entering deviation) is necessary for correctness
* because of the complex algorithm.
*
* HijrahDate supports the manual adjustment feature by providing a configuration
* file. The configuration file contains the adjustment (deviation) data with following format.
*
* StartYear/StartMonth(0-based)-EndYear/EndMonth(0-based):Deviation day (1, 2, -1, or -2)
* Line separator or ";" is used for the separator of each deviation data.
* Here is the example.
*
* 1429/0-1429/1:1
* 1429/2-1429/7:1;1429/6-1429/11:1
* 1429/11-9999/11:1
* The default location of the configuration file is:
*
* $CLASSPATH/org/threeten/bp/chrono
* And the default file name is:
*
* hijrah_deviation.cfg
* The default location and file name can be overriden by setting
* following two Java's system property.
*
* Location: org.threeten.bp.i18n.HijrahDate.deviationConfigDir
* File name: org.threeten.bp.i18n.HijrahDate.deviationConfigFile
*
* Specification for implementors
* This class is immutable and thread-safe.
*/
public final class HijrahDate
extends ChronoDateImpl
implements Serializable {
// this class is package-scoped so that future conversion to public
// would not change serialization
/**
* Serialization version.
*/
private static final long serialVersionUID = -5207853542612002020L;
/**
* The minimum valid year-of-era.
*/
public static final int MIN_VALUE_OF_ERA = 1;
/**
* The maximum valid year-of-era.
* This is currently set to 9999 but may be changed to increase the valid range
* in a future version of the specification.
*/
public static final int MAX_VALUE_OF_ERA = 9999;
/**
* 0-based, for number of day-of-year in the beginning of month in normal
* year.
*/
private static final int NUM_DAYS[] =
{0, 30, 59, 89, 118, 148, 177, 207, 236, 266, 295, 325};
/**
* 0-based, for number of day-of-year in the beginning of month in leap year.
*/
private static final int LEAP_NUM_DAYS[] =
{0, 30, 59, 89, 118, 148, 177, 207, 236, 266, 295, 325};
/**
* 0-based, for day-of-month in normal year.
*/
private static final int MONTH_LENGTH[] =
{30, 29, 30, 29, 30, 29, 30, 29, 30, 29, 30, 29};
/**
* 0-based, for day-of-month in leap year.
*/
private static final int LEAP_MONTH_LENGTH[] =
{30, 29, 30, 29, 30, 29, 30, 29, 30, 29, 30, 30};
/**
*
* Greatest Least
* Field name Minimum Minimum Maximum Maximum
* ---------- ------- ------- ------- -------
* ERA 0 0 1 1
* YEAR_OF_ERA 1 1 9999 9999
* MONTH_OF_YEAR 1 1 12 12
* DAY_OF_MONTH 1 1 29 30
* DAY_OF_YEAR 1 1 354 355
*
*
* Minimum values.
*/
private static final int MIN_VALUES[] =
{
0,
MIN_VALUE_OF_ERA,
0,
1,
0,
1,
1
};
/**
* Least maximum values.
*/
private static final int LEAST_MAX_VALUES[] =
{
1,
MAX_VALUE_OF_ERA,
11,
51,
5,
29,
354
};
/**
* Maximum values.
*/
private static final int MAX_VALUES[] =
{
1,
MAX_VALUE_OF_ERA,
11,
52,
6,
30,
355
};
/**
* Position of day-of-month. This value is used to get the min/max value
* from an array.
*/
private static final int POSITION_DAY_OF_MONTH = 5;
/**
* Position of day-of-year. This value is used to get the min/max value from
* an array.
*/
private static final int POSITION_DAY_OF_YEAR = 6;
/**
* Zero-based start date of cycle year.
*/
private static final int CYCLEYEAR_START_DATE[] =
{
0,
354,
709,
1063,
1417,
1772,
2126,
2481,
2835,
3189,
3544,
3898,
4252,
4607,
4961,
5315,
5670,
6024,
6379,
6733,
7087,
7442,
7796,
8150,
8505,
8859,
9214,
9568,
9922,
10277
};
/**
* File separator.
*/
private static final char FILE_SEP = File.separatorChar;
/**
* Path separator.
*/
private static final String PATH_SEP = File.pathSeparator;
/**
* Default config file name.
*/
private static final String DEFAULT_CONFIG_FILENAME = "hijrah_deviation.cfg";
/**
* Default path to the config file.
*/
private static final String DEFAULT_CONFIG_PATH = "org" + FILE_SEP + "threeten" + FILE_SEP + "bp" + FILE_SEP + "chrono";
/**
* Holding the adjusted month days in year. The key is a year (Integer) and
* the value is the all the month days in year (Integer[]).
*/
private static final HashMap ADJUSTED_MONTH_DAYS = new HashMap();
/**
* Holding the adjusted month length in year. The key is a year (Integer)
* and the value is the all the month length in year (Integer[]).
*/
private static final HashMap ADJUSTED_MONTH_LENGTHS = new HashMap();
/**
* Holding the adjusted days in the 30 year cycle. The key is a cycle number
* (Integer) and the value is the all the starting days of the year in the
* cycle (Integer[]).
*/
private static final HashMap ADJUSTED_CYCLE_YEARS = new HashMap();
/**
* Holding the adjusted cycle in the 1 - 30000 year. The key is the cycle
* number (Integer) and the value is the starting days in the cycle in the
* term.
*/
private static final Long[] ADJUSTED_CYCLES;
/**
* Holding the adjusted min values.
*/
private static final Integer[] ADJUSTED_MIN_VALUES;
/**
* Holding the adjusted max least max values.
*/
private static final Integer[] ADJUSTED_LEAST_MAX_VALUES;
/**
* Holding adjusted max values.
*/
private static final Integer[] ADJUSTED_MAX_VALUES;
/**
* Holding the non-adjusted month days in year for non leap year.
*/
private static final Integer[] DEFAULT_MONTH_DAYS;
/**
* Holding the non-adjusted month days in year for leap year.
*/
private static final Integer[] DEFAULT_LEAP_MONTH_DAYS;
/**
* Holding the non-adjusted month length for non leap year.
*/
private static final Integer[] DEFAULT_MONTH_LENGTHS;
/**
* Holding the non-adjusted month length for leap year.
*/
private static final Integer[] DEFAULT_LEAP_MONTH_LENGTHS;
/**
* Holding the non-adjusted 30 year cycle starting day.
*/
private static final Integer[] DEFAULT_CYCLE_YEARS;
/**
* number of 30-year cycles to hold the deviation data.
*/
private static final int MAX_ADJUSTED_CYCLE = 334; // to support year 9999
static { // Initialize the static integer array;
DEFAULT_MONTH_DAYS = new Integer[NUM_DAYS.length];
for (int i = 0; i < NUM_DAYS.length; i++) {
DEFAULT_MONTH_DAYS[i] = new Integer(NUM_DAYS[i]);
}
DEFAULT_LEAP_MONTH_DAYS = new Integer[LEAP_NUM_DAYS.length];
for (int i = 0; i < LEAP_NUM_DAYS.length; i++) {
DEFAULT_LEAP_MONTH_DAYS[i] = new Integer(LEAP_NUM_DAYS[i]);
}
DEFAULT_MONTH_LENGTHS = new Integer[MONTH_LENGTH.length];
for (int i = 0; i < MONTH_LENGTH.length; i++) {
DEFAULT_MONTH_LENGTHS[i] = new Integer(MONTH_LENGTH[i]);
}
DEFAULT_LEAP_MONTH_LENGTHS = new Integer[LEAP_MONTH_LENGTH.length];
for (int i = 0; i < LEAP_MONTH_LENGTH.length; i++) {
DEFAULT_LEAP_MONTH_LENGTHS[i] = new Integer(LEAP_MONTH_LENGTH[i]);
}
DEFAULT_CYCLE_YEARS = new Integer[CYCLEYEAR_START_DATE.length];
for (int i = 0; i < CYCLEYEAR_START_DATE.length; i++) {
DEFAULT_CYCLE_YEARS[i] = new Integer(CYCLEYEAR_START_DATE[i]);
}
ADJUSTED_CYCLES = new Long[MAX_ADJUSTED_CYCLE];
for (int i = 0; i < ADJUSTED_CYCLES.length; i++) {
ADJUSTED_CYCLES[i] = new Long(10631 * i);
}
// Initialize min values, least max values and max values.
ADJUSTED_MIN_VALUES = new Integer[MIN_VALUES.length];
for (int i = 0; i < MIN_VALUES.length; i++) {
ADJUSTED_MIN_VALUES[i] = new Integer(MIN_VALUES[i]);
}
ADJUSTED_LEAST_MAX_VALUES = new Integer[LEAST_MAX_VALUES.length];
for (int i = 0; i < LEAST_MAX_VALUES.length; i++) {
ADJUSTED_LEAST_MAX_VALUES[i] = new Integer(LEAST_MAX_VALUES[i]);
}
ADJUSTED_MAX_VALUES = new Integer[MAX_VALUES.length];
for (int i = 0; i < MAX_VALUES.length; i++) {
ADJUSTED_MAX_VALUES[i] = new Integer(MAX_VALUES[i]);
}
try {
readDeviationConfig();
} catch (IOException e) {
// do nothing. Ignore deviation config.
// e.printStackTrace();
} catch (ParseException e) {
// do nothing. Ignore deviation config.
// e.printStackTrace();
}
}
/**
* Number of Gregorian day of July 19, year 622 (Gregorian), which is epoch day
* of Hijrah calendar.
*/
private static final int HIJRAH_JAN_1_1_GREGORIAN_DAY = -492148;
/**
* The era.
*/
private final transient HijrahEra era;
/**
* The year.
*/
private final transient int yearOfEra;
/**
* The month-of-year.
*/
private final transient int monthOfYear;
/**
* The day-of-month.
*/
private final transient int dayOfMonth;
/**
* The day-of-year.
*/
private final transient int dayOfYear;
/**
* The day-of-week.
*/
private final transient DayOfWeek dayOfWeek;
/**
* Gregorian days for this object. Holding number of days since 1970/01/01.
* The number of days are calculated with pure Gregorian calendar
* based.
*/
private final long gregorianEpochDay;
/**
* True if year is leap year.
*/
private final transient boolean isLeapYear;
//-----------------------------------------------------------------------
/**
* Obtains the current {@code HijrahDate} of the Islamic Umm Al-Qura calendar
* in the default time-zone.
*
* This will query the {@link Clock#systemDefaultZone() system clock} in the default
* time-zone to obtain the current date.
*
* Using this method will prevent the ability to use an alternate clock for testing
* because the clock is hard-coded.
*
* @return the current date using the system clock and default time-zone, not null
*/
public static HijrahDate now() {
return now(Clock.systemDefaultZone());
}
/**
* Obtains the current {@code HijrahDate} of the Islamic Umm Al-Qura calendar
* in the specified time-zone.
*
* This will query the {@link Clock#system(ZoneId) system clock} to obtain the current date.
* Specifying the time-zone avoids dependence on the default time-zone.
*
* Using this method will prevent the ability to use an alternate clock for testing
* because the clock is hard-coded.
*
* @param zone the zone ID to use, not null
* @return the current date using the system clock, not null
*/
public static HijrahDate now(ZoneId zone) {
return now(Clock.system(zone));
}
/**
* Obtains the current {@code HijrahDate} of the Islamic Umm Al-Qura calendar
* from the specified clock.
*
* This will query the specified clock to obtain the current date - today.
* Using this method allows the use of an alternate clock for testing.
* The alternate clock may be introduced using {@linkplain Clock dependency injection}.
*
* @param clock the clock to use, not null
* @return the current date, not null
* @throws DateTimeException if the current date cannot be obtained
*/
public static HijrahDate now(Clock clock) {
return HijrahChronology.INSTANCE.dateNow(clock);
}
//-------------------------------------------------------------------------
/**
* Obtains an instance of {@code HijrahDate} from the Hijrah era year,
* month-of-year and day-of-month. This uses the Hijrah era.
*
* @param prolepticYear the proleptic year to represent in the Hijrah
* @param monthOfYear the month-of-year to represent, from 1 to 12
* @param dayOfMonth the day-of-month to represent, from 1 to 30
* @return the Hijrah date, never null
* @throws IllegalCalendarFieldValueException if the value of any field is out of range
* @throws InvalidCalendarFieldException if the day-of-month is invalid for the month-year
*/
public static HijrahDate of(int prolepticYear, int monthOfYear, int dayOfMonth) {
return (prolepticYear >= 1) ?
HijrahDate.of(HijrahEra.AH, prolepticYear, monthOfYear, dayOfMonth) :
HijrahDate.of(HijrahEra.BEFORE_AH, 1 - prolepticYear, monthOfYear, dayOfMonth);
}
/**
* Obtains an instance of {@code HijrahDate} from the era, year-of-era
* month-of-year and day-of-month.
*
* @param era the era to represent, not null
* @param yearOfEra the year-of-era to represent, from 1 to 9999
* @param monthOfYear the month-of-year to represent, from 1 to 12
* @param dayOfMonth the day-of-month to represent, from 1 to 31
* @return the Hijrah date, never null
* @throws IllegalCalendarFieldValueException if the value of any field is out of range
* @throws InvalidCalendarFieldException if the day-of-month is invalid for the month-year
*/
static HijrahDate of(HijrahEra era, int yearOfEra, int monthOfYear, int dayOfMonth) {
Jdk8Methods.requireNonNull(era, "era");
checkValidYearOfEra(yearOfEra);
checkValidMonth(monthOfYear);
checkValidDayOfMonth(dayOfMonth);
long gregorianDays = getGregorianEpochDay(era.prolepticYear(yearOfEra), monthOfYear, dayOfMonth);
return new HijrahDate(gregorianDays);
}
/**
* Check the validity of a yearOfEra.
* @param yearOfEra the year to check
*/
private static void checkValidYearOfEra(int yearOfEra) {
if (yearOfEra < MIN_VALUE_OF_ERA ||
yearOfEra > MAX_VALUE_OF_ERA) {
throw new DateTimeException("Invalid year of Hijrah Era");
}
}
private static void checkValidDayOfYear(int dayOfYear) {
if (dayOfYear < 1 ||
dayOfYear > getMaximumDayOfYear()) {
throw new DateTimeException("Invalid day of year of Hijrah date");
}
}
private static void checkValidMonth(int month) {
if (month < 1 || month > 12) {
throw new DateTimeException("Invalid month of Hijrah date");
}
}
private static void checkValidDayOfMonth(int dayOfMonth) {
if (dayOfMonth < 1 ||
dayOfMonth > getMaximumDayOfMonth()) {
throw new DateTimeException("Invalid day of month of Hijrah date, day "
+ dayOfMonth + " greater than " + getMaximumDayOfMonth() + " or less than 1");
}
}
/**
* Obtains an instance of {@code HijrahDate} from a date.
*
* @param date the date to use, not null
* @return the Hijrah date, never null
* @throws IllegalCalendarFieldValueException if the year is invalid
*/
static HijrahDate of(LocalDate date) {
long gregorianDays = date.toEpochDay();
return new HijrahDate(gregorianDays);
}
static HijrahDate ofEpochDay(long epochDay) {
return new HijrahDate(epochDay);
}
/**
* Obtains a {@code HijrahDate} of the Islamic Umm Al-Qura calendar from a temporal object.
*
* This obtains a date in the Hijrah calendar system based on the specified temporal.
* A {@code TemporalAccessor} represents an arbitrary set of date and time information,
* which this factory converts to an instance of {@code HijrahDate}.
*
* The conversion typically uses the {@link ChronoField#EPOCH_DAY EPOCH_DAY}
* field, which is standardized across calendar systems.
*
* This method matches the signature of the functional interface {@link TemporalQuery}
* allowing it to be used as a query via method reference, {@code HijrahDate::from}.
*
* @param temporal the temporal object to convert, not null
* @return the date in Hijrah calendar system, not null
* @throws DateTimeException if unable to convert to a {@code HijrahDate}
*/
public static HijrahDate from(TemporalAccessor temporal) {
return HijrahChronology.INSTANCE.date(temporal);
}
//-------------------------------------------------------------------------
/**
* Constructs an instance with the specified date.
*
* @param gregorianDay the number of days from 0001/01/01 (Gregorian), caller calculated
*/
private HijrahDate(long gregorianDay) {
int[] dateInfo = getHijrahDateInfo(gregorianDay);
checkValidYearOfEra(dateInfo[1]);
checkValidMonth(dateInfo[2]);
checkValidDayOfMonth(dateInfo[3]);
checkValidDayOfYear(dateInfo[4]);
this.era = HijrahEra.of(dateInfo[0]);
this.yearOfEra = dateInfo[1];
this.monthOfYear = dateInfo[2];
this.dayOfMonth = dateInfo[3];
this.dayOfYear = dateInfo[4];
this.dayOfWeek = DayOfWeek.of(dateInfo[5]);
this.gregorianEpochDay = gregorianDay;
this.isLeapYear = isLeapYear(this.yearOfEra);
}
/**
* Replaces the date instance from the stream with a valid one.
*
* @return the resolved date, never null
*/
private Object readResolve() {
return new HijrahDate(this.gregorianEpochDay);
}
//-----------------------------------------------------------------------
@Override
public HijrahChronology getChronology() {
return HijrahChronology.INSTANCE;
}
@Override
public HijrahEra getEra() {
return this.era;
}
@Override
public ValueRange range(TemporalField field) {
if (field instanceof ChronoField) {
if (isSupported(field)) {
ChronoField f = (ChronoField) field;
switch (f) {
case DAY_OF_MONTH: return ValueRange.of(1, lengthOfMonth());
case DAY_OF_YEAR: return ValueRange.of(1, lengthOfYear());
case ALIGNED_WEEK_OF_MONTH: return ValueRange.of(1, 5); // TODO
case YEAR_OF_ERA: return ValueRange.of(1, 1000); // TODO
}
return getChronology().range(f);
}
throw new UnsupportedTemporalTypeException("Unsupported field: " + field);
}
return field.rangeRefinedBy(this);
}
@Override
public long getLong(TemporalField field) {
if (field instanceof ChronoField) {
switch ((ChronoField) field) {
case DAY_OF_WEEK: return dayOfWeek.getValue();
case ALIGNED_DAY_OF_WEEK_IN_MONTH: return ((dayOfMonth - 1) % 7) + 1;
case ALIGNED_DAY_OF_WEEK_IN_YEAR: return ((dayOfYear - 1) % 7) + 1;
case DAY_OF_MONTH: return this.dayOfMonth;
case DAY_OF_YEAR: return this.dayOfYear;
case EPOCH_DAY: return toEpochDay();
case ALIGNED_WEEK_OF_MONTH: return ((dayOfMonth - 1) / 7) + 1;
case ALIGNED_WEEK_OF_YEAR: return ((dayOfYear - 1) / 7) + 1;
case MONTH_OF_YEAR: return monthOfYear;
case YEAR_OF_ERA: return yearOfEra;
case YEAR: return yearOfEra;
case ERA: return era.getValue();
}
throw new UnsupportedTemporalTypeException("Unsupported field: " + field);
}
return field.getFrom(this);
}
//-------------------------------------------------------------------------
@Override
public HijrahDate with(TemporalAdjuster adjuster) {
return (HijrahDate) super.with(adjuster);
}
@Override
public HijrahDate with(TemporalField field, long newValue) {
if (field instanceof ChronoField) {
ChronoField f = (ChronoField) field;
f.checkValidValue(newValue); // TODO: validate value
int nvalue = (int) newValue;
switch (f) {
case DAY_OF_WEEK: return plusDays(newValue - dayOfWeek.getValue());
case ALIGNED_DAY_OF_WEEK_IN_MONTH: return plusDays(newValue - getLong(ALIGNED_DAY_OF_WEEK_IN_MONTH));
case ALIGNED_DAY_OF_WEEK_IN_YEAR: return plusDays(newValue - getLong(ALIGNED_DAY_OF_WEEK_IN_YEAR));
case DAY_OF_MONTH: return resolvePreviousValid(yearOfEra, monthOfYear, nvalue);
case DAY_OF_YEAR: return resolvePreviousValid(yearOfEra, ((nvalue - 1) / 30) + 1, ((nvalue - 1) % 30) + 1);
case EPOCH_DAY: return new HijrahDate(nvalue);
case ALIGNED_WEEK_OF_MONTH: return plusDays((newValue - getLong(ALIGNED_WEEK_OF_MONTH)) * 7);
case ALIGNED_WEEK_OF_YEAR: return plusDays((newValue - getLong(ALIGNED_WEEK_OF_YEAR)) * 7);
case MONTH_OF_YEAR: return resolvePreviousValid(yearOfEra, nvalue, dayOfMonth);
case YEAR_OF_ERA: return resolvePreviousValid(yearOfEra >= 1 ? nvalue : 1 - nvalue, monthOfYear, dayOfMonth);
case YEAR: return resolvePreviousValid(nvalue, monthOfYear, dayOfMonth);
case ERA: return resolvePreviousValid(1 - yearOfEra, monthOfYear, dayOfMonth);
}
throw new UnsupportedTemporalTypeException("Unsupported field: " + field);
}
return field.adjustInto(this, newValue);
}
private static HijrahDate resolvePreviousValid(int yearOfEra, int month, int day) {
int monthDays = getMonthDays(month - 1, yearOfEra);
if (day > monthDays) {
day = monthDays;
}
return HijrahDate.of(yearOfEra, month, day);
}
@Override
public HijrahDate plus(TemporalAmount amount) {
return (HijrahDate) super.plus(amount);
}
@Override
public HijrahDate plus(long amountToAdd, TemporalUnit unit) {
return (HijrahDate) super.plus(amountToAdd, unit);
}
@Override
public HijrahDate minus(TemporalAmount amount) {
return (HijrahDate) super.minus(amount);
}
@Override
public HijrahDate minus(long amountToAdd, TemporalUnit unit) {
return (HijrahDate) super.minus(amountToAdd, unit);
}
//-------------------------------------------------------------------------
@Override
@SuppressWarnings("unchecked")
public final ChronoLocalDateTime atTime(LocalTime localTime) {
return (ChronoLocalDateTime) super.atTime(localTime);
}
@Override
public long toEpochDay() {
return getGregorianEpochDay(yearOfEra, monthOfYear, dayOfMonth);
}
//-----------------------------------------------------------------------
/**
* Checks if the year is a leap year, according to the Hijrah calendar system rules.
*
* @return true if this date is in a leap year
*/
@Override
public boolean isLeapYear() {
return this.isLeapYear;
}
//-----------------------------------------------------------------------
@Override
HijrahDate plusYears(long years) {
if (years == 0) {
return this;
}
int newYear = Jdk8Methods.safeAdd(this.yearOfEra, (int)years);
return HijrahDate.of(this.era, newYear, this.monthOfYear, this.dayOfMonth);
}
@Override
HijrahDate plusMonths(long months) {
if (months == 0) {
return this;
}
int newMonth = this.monthOfYear - 1;
newMonth = newMonth + (int)months;
int years = newMonth / 12;
newMonth = newMonth % 12;
while (newMonth < 0) {
newMonth += 12;
years = Jdk8Methods.safeSubtract(years, 1);
}
int newYear = Jdk8Methods.safeAdd(this.yearOfEra, years);
return HijrahDate.of(this.era, newYear, newMonth + 1, this.dayOfMonth);
}
@Override
HijrahDate plusDays(long days) {
return new HijrahDate(this.gregorianEpochDay + days);
}
//-----------------------------------------------------------------------
/**
* Returns the int array containing the following field from the julian day.
*
* int[0] = ERA
* int[1] = YEAR
* int[2] = MONTH
* int[3] = DATE
* int[4] = DAY_OF_YEAR
* int[5] = DAY_OF_WEEK
*
* @param julianDay a julian day.
*/
private static int[] getHijrahDateInfo(long gregorianDays) {
int era, year, month, date, dayOfWeek, dayOfYear;
int cycleNumber, yearInCycle, dayOfCycle;
long epochDay = gregorianDays - HIJRAH_JAN_1_1_GREGORIAN_DAY;
if (epochDay >= 0) {
cycleNumber = getCycleNumber(epochDay); // 0 - 99.
dayOfCycle = getDayOfCycle(epochDay, cycleNumber); // 0 - 10631.
yearInCycle = getYearInCycle(cycleNumber, dayOfCycle); // 0 - 29.
dayOfYear = getDayOfYear(cycleNumber, dayOfCycle, yearInCycle);
// 0 - 354/355
year = cycleNumber * 30 + yearInCycle + 1; // 1-based year.
month = getMonthOfYear(dayOfYear, year); // 0-based month-of-year
date = getDayOfMonth(dayOfYear, month, year); // 0-based date
++date; // Convert from 0-based to 1-based
era = HijrahEra.AH.getValue();
} else {
cycleNumber = (int) epochDay / 10631; // 0 or negative number.
dayOfCycle = (int) epochDay % 10631; // -10630 - 0.
if (dayOfCycle == 0) {
dayOfCycle = -10631;
cycleNumber++;
}
yearInCycle = getYearInCycle(cycleNumber, dayOfCycle); // 0 - 29.
dayOfYear = getDayOfYear(cycleNumber, dayOfCycle, yearInCycle);
year = cycleNumber * 30 - yearInCycle; // negative number.
year = 1 - year;
dayOfYear = (isLeapYear(year) ? (dayOfYear + 355)
: (dayOfYear + 354));
month = getMonthOfYear(dayOfYear, year);
date = getDayOfMonth(dayOfYear, month, year);
++date; // Convert from 0-based to 1-based
era = HijrahEra.BEFORE_AH.getValue();
}
// Hijrah day zero is a Friday
dayOfWeek = (int) ((epochDay + 5) % 7);
dayOfWeek += (dayOfWeek <= 0) ? 7 : 0;
int dateInfo[] = new int[6];
dateInfo[0] = era;
dateInfo[1] = year;
dateInfo[2] = month + 1; // change to 1-based.
dateInfo[3] = date;
dateInfo[4] = dayOfYear + 1; // change to 1-based.
dateInfo[5] = dayOfWeek;
return dateInfo;
}
/**
* Return Gregorian epoch day from Hijrah year, month, and day.
*
* @param prolepticYear the year to represent, caller calculated
* @param monthOfYear the month-of-year to represent, caller calculated
* @param dayOfMonth the day-of-month to represent, caller calculated
* @return a julian day
*/
private static long getGregorianEpochDay(int prolepticYear, int monthOfYear, int dayOfMonth) {
long day = yearToGregorianEpochDay(prolepticYear);
day += getMonthDays(monthOfYear - 1, prolepticYear);
day += dayOfMonth;
return day;
}
/**
* Returns the Gregorian epoch day from the proleptic year
* @param prolepticYear the proleptic year
* @return the Epoch day
*/
private static long yearToGregorianEpochDay(int prolepticYear) {
int cycleNumber = (prolepticYear - 1) / 30; // 0-based.
int yearInCycle = (prolepticYear - 1) % 30; // 0-based.
int dayInCycle = getAdjustedCycle(cycleNumber)[Math.abs(yearInCycle)]
.intValue();
if (yearInCycle < 0) {
dayInCycle = -dayInCycle;
}
Long cycleDays;
try {
cycleDays = ADJUSTED_CYCLES[cycleNumber];
} catch (ArrayIndexOutOfBoundsException e) {
cycleDays = null;
}
if (cycleDays == null) {
cycleDays = new Long(cycleNumber * 10631);
}
return (cycleDays.longValue() + dayInCycle + HIJRAH_JAN_1_1_GREGORIAN_DAY - 1);
}
/**
* Returns the 30 year cycle number from the epoch day.
*
* @param epochDay an epoch day
* @return a cycle number
*/
private static int getCycleNumber(long epochDay) {
Long[] days = ADJUSTED_CYCLES;
int cycleNumber;
try {
for (int i = 0; i < days.length; i++) {
if (epochDay < days[i].longValue()) {
return i - 1;
}
}
cycleNumber = (int) epochDay / 10631;
} catch (ArrayIndexOutOfBoundsException e) {
cycleNumber = (int) epochDay / 10631;
}
return cycleNumber;
}
/**
* Returns day of cycle from the epoch day and cycle number.
*
* @param epochDay an epoch day
* @param cycleNumber a cycle number
* @return a day of cycle
*/
private static int getDayOfCycle(long epochDay, int cycleNumber) {
Long day;
try {
day = ADJUSTED_CYCLES[cycleNumber];
} catch (ArrayIndexOutOfBoundsException e) {
day = null;
}
if (day == null) {
day = new Long(cycleNumber * 10631);
}
return (int) (epochDay - day.longValue());
}
/**
* Returns the year in cycle from the cycle number and day of cycle.
*
* @param cycleNumber a cycle number
* @param dayOfCycle day of cycle
* @return a year in cycle
*/
private static int getYearInCycle(int cycleNumber, long dayOfCycle) {
Integer[] cycles = getAdjustedCycle(cycleNumber);
if (dayOfCycle == 0) {
return 0;
}
if (dayOfCycle > 0) {
for (int i = 0; i < cycles.length; i++) {
if (dayOfCycle < cycles[i].intValue()) {
return i - 1;
}
}
return 29;
} else {
dayOfCycle = -dayOfCycle;
for (int i = 0; i < cycles.length; i++) {
if (dayOfCycle <= cycles[i].intValue()) {
return i - 1;
}
}
return 29;
}
}
/**
* Returns adjusted 30 year cycle startind day as Integer array from the
* cycle number specified.
*
* @param cycleNumber a cycle number
* @return an Integer array
*/
private static Integer[] getAdjustedCycle(int cycleNumber) {
Integer[] cycles;
try {
cycles = ADJUSTED_CYCLE_YEARS.get(new Integer(cycleNumber));
} catch (ArrayIndexOutOfBoundsException e) {
cycles = null;
}
if (cycles == null) {
cycles = DEFAULT_CYCLE_YEARS;
}
return cycles;
}
/**
* Returns adjusted month days as Integer array form the year specified.
*
* @param year a year
* @return an Integer array
*/
private static Integer[] getAdjustedMonthDays(int year) {
Integer[] newMonths;
try {
newMonths = ADJUSTED_MONTH_DAYS.get(new Integer(year));
} catch (ArrayIndexOutOfBoundsException e) {
newMonths = null;
}
if (newMonths == null) {
if (isLeapYear(year)) {
newMonths = DEFAULT_LEAP_MONTH_DAYS;
} else {
newMonths = DEFAULT_MONTH_DAYS;
}
}
return newMonths;
}
/**
* Returns adjusted month length as Integer array form the year specified.
*
* @param year a year
* @return an Integer array
*/
private static Integer[] getAdjustedMonthLength(int year) {
Integer[] newMonths;
try {
newMonths = ADJUSTED_MONTH_LENGTHS.get(new Integer(year));
} catch (ArrayIndexOutOfBoundsException e) {
newMonths = null;
}
if (newMonths == null) {
if (isLeapYear(year)) {
newMonths = DEFAULT_LEAP_MONTH_LENGTHS;
} else {
newMonths = DEFAULT_MONTH_LENGTHS;
}
}
return newMonths;
}
/**
* Returns day-of-year.
*
* @param cycleNumber a cycle number
* @param dayOfCycle day of cycle
* @param yearInCycle year in cycle
* @return day-of-year
*/
private static int getDayOfYear(int cycleNumber, int dayOfCycle, int yearInCycle) {
Integer[] cycles = getAdjustedCycle(cycleNumber);
if (dayOfCycle > 0) {
return dayOfCycle - cycles[yearInCycle].intValue();
} else {
return cycles[yearInCycle].intValue() + dayOfCycle;
}
}
/**
* Returns month-of-year. 0-based.
*
* @param dayOfYear day-of-year
* @param year a year
* @return month-of-year
*/
private static int getMonthOfYear(int dayOfYear, int year) {
Integer[] newMonths = getAdjustedMonthDays(year);
if (dayOfYear >= 0) {
for (int i = 0; i < newMonths.length; i++) {
if (dayOfYear < newMonths[i].intValue()) {
return i - 1;
}
}
return 11;
} else {
dayOfYear = (isLeapYear(year) ? (dayOfYear + 355)
: (dayOfYear + 354));
for (int i = 0; i < newMonths.length; i++) {
if (dayOfYear < newMonths[i].intValue()) {
return i - 1;
}
}
return 11;
}
}
/**
* Returns day-of-month.
*
* @param dayOfYear day of year
* @param month month
* @param year year
* @return day-of-month
*/
private static int getDayOfMonth(int dayOfYear, int month, int year) {
Integer[] newMonths = getAdjustedMonthDays(year);
if (dayOfYear >= 0) {
if (month > 0) {
return dayOfYear - newMonths[month].intValue();
} else {
return dayOfYear;
}
} else {
dayOfYear = (isLeapYear(year) ? (dayOfYear + 355)
: (dayOfYear + 354));
if (month > 0) {
return dayOfYear - newMonths[month].intValue();
} else {
return dayOfYear;
}
}
}
/**
* Determines if the given year is a leap year.
*
* @param year year
* @return true if leap year
*/
static boolean isLeapYear(long year) {
return (14 + 11 * (year > 0 ? year : -year)) % 30 < 11;
}
/**
* Returns month days from the beginning of year.
*
* @param month month (0-based)
* @parma year year
* @return month days from the beginning of year
*/
private static int getMonthDays(int month, int year) {
Integer[] newMonths = getAdjustedMonthDays(year);
return newMonths[month].intValue();
}
/**
* Returns month length.
*
* @param month month (0-based)
* @param year year
* @return month length
*/
static int getMonthLength(int month, int year) {
Integer[] newMonths = getAdjustedMonthLength(year);
return newMonths[month].intValue();
}
@Override
public int lengthOfMonth() {
return getMonthLength(monthOfYear - 1, yearOfEra);
}
/**
* Returns year length.
*
* @param year year
* @return year length
*/
static int getYearLength(int year) {
int cycleNumber = (year - 1) / 30;
Integer[] cycleYears;
try {
cycleYears = ADJUSTED_CYCLE_YEARS.get(cycleNumber);
} catch (ArrayIndexOutOfBoundsException e) {
cycleYears = null;
}
if (cycleYears != null) {
int yearInCycle = (year - 1) % 30;
if (yearInCycle == 29) {
return ADJUSTED_CYCLES[cycleNumber + 1].intValue()
- ADJUSTED_CYCLES[cycleNumber].intValue()
- cycleYears[yearInCycle].intValue();
}
return cycleYears[yearInCycle + 1].intValue()
- cycleYears[yearInCycle].intValue();
} else {
return isLeapYear(year) ? 355 : 354;
}
}
@Override
public int lengthOfYear() {
return getYearLength(yearOfEra); // TODO: proleptic year
}
/**
* Returns maximum day-of-month.
*
* @return maximum day-of-month
*/
static int getMaximumDayOfMonth() {
return ADJUSTED_MAX_VALUES[POSITION_DAY_OF_MONTH];
}
/**
* Returns smallest maximum day-of-month.
*
* @return smallest maximum day-of-month
*/
static int getSmallestMaximumDayOfMonth() {
return ADJUSTED_LEAST_MAX_VALUES[POSITION_DAY_OF_MONTH];
}
/**
* Returns maximum day-of-year.
*
* @return maximum day-of-year
*/
static int getMaximumDayOfYear() {
return ADJUSTED_MAX_VALUES[POSITION_DAY_OF_YEAR];
}
/**
* Returns smallest maximum day-of-year.
*
* @return smallest maximum day-of-year
*/
static int getSmallestMaximumDayOfYear() {
return ADJUSTED_LEAST_MAX_VALUES[POSITION_DAY_OF_YEAR];
}
// ----- Deviation handling -----//
/**
* Adds deviation definition. The year and month sepcifed should be the
* caluculated Hijrah year and month. The month is 0 based. e.g. 8 for
* Ramadan (9th month) Addition of anything minus deviation days is
* calculated negatively in the case the user wants to subtract days from
* the calendar. For example, adding -1 days will subtract one day from the
* current date. Please note that this behavior is different from the
* addDeviaiton method.
*
* @param startYear start year
* @param startMonth start month
* @param endYear end year
* @param endMonth end month
* @param offset offset
*/
private static void addDeviationAsHijrah(int startYear,
int startMonth, int endYear, int endMonth, int offset) {
if (startYear < 1) {
throw new IllegalArgumentException("startYear < 1");
}
if (endYear < 1) {
throw new IllegalArgumentException("endYear < 1");
}
if (startMonth < 0 || startMonth > 11) {
throw new IllegalArgumentException(
"startMonth < 0 || startMonth > 11");
}
if (endMonth < 0 || endMonth > 11) {
throw new IllegalArgumentException("endMonth < 0 || endMonth > 11");
}
if (endYear > 9999) {
throw new IllegalArgumentException("endYear > 9999");
}
if (endYear < startYear) {
throw new IllegalArgumentException("startYear > endYear");
}
if (endYear == startYear && endMonth < startMonth) {
throw new IllegalArgumentException(
"startYear == endYear && endMonth < startMonth");
}
// Adjusting start year.
boolean isStartYLeap = isLeapYear(startYear);
// Adjusting the number of month.
Integer[] orgStartMonthNums = ADJUSTED_MONTH_DAYS.get(new Integer(
startYear));
if (orgStartMonthNums == null) {
if (isStartYLeap) {
orgStartMonthNums = new Integer[LEAP_NUM_DAYS.length];
for (int l = 0; l < LEAP_NUM_DAYS.length; l++) {
orgStartMonthNums[l] = new Integer(LEAP_NUM_DAYS[l]);
}
} else {
orgStartMonthNums = new Integer[NUM_DAYS.length];
for (int l = 0; l < NUM_DAYS.length; l++) {
orgStartMonthNums[l] = new Integer(NUM_DAYS[l]);
}
}
}
Integer[] newStartMonthNums = new Integer[orgStartMonthNums.length];
for (int month = 0; month < 12; month++) {
if (month > startMonth) {
newStartMonthNums[month] = new Integer(orgStartMonthNums[month]
.intValue()
- offset);
} else {
newStartMonthNums[month] = new Integer(orgStartMonthNums[month]
.intValue());
}
}
ADJUSTED_MONTH_DAYS.put(new Integer(startYear), newStartMonthNums);
// Adjusting the days of month.
Integer[] orgStartMonthLengths = ADJUSTED_MONTH_LENGTHS.get(new Integer(
startYear));
if (orgStartMonthLengths == null) {
if (isStartYLeap) {
orgStartMonthLengths = new Integer[LEAP_MONTH_LENGTH.length];
for (int l = 0; l < LEAP_MONTH_LENGTH.length; l++) {
orgStartMonthLengths[l] = new Integer(LEAP_MONTH_LENGTH[l]);
}
} else {
orgStartMonthLengths = new Integer[MONTH_LENGTH.length];
for (int l = 0; l < MONTH_LENGTH.length; l++) {
orgStartMonthLengths[l] = new Integer(MONTH_LENGTH[l]);
}
}
}
Integer[] newStartMonthLengths = new Integer[orgStartMonthLengths.length];
for (int month = 0; month < 12; month++) {
if (month == startMonth) {
newStartMonthLengths[month] = new Integer(
orgStartMonthLengths[month].intValue() - offset);
} else {
newStartMonthLengths[month] = new Integer(
orgStartMonthLengths[month].intValue());
}
}
ADJUSTED_MONTH_LENGTHS.put(new Integer(startYear), newStartMonthLengths);
if (startYear != endYear) {
// System.out.println("over year");
// Adjusting starting 30 year cycle.
int sCycleNumber = (startYear - 1) / 30;
int sYearInCycle = (startYear - 1) % 30; // 0-based.
Integer[] startCycles = ADJUSTED_CYCLE_YEARS.get(new Integer(
sCycleNumber));
if (startCycles == null) {
startCycles = new Integer[CYCLEYEAR_START_DATE.length];
for (int j = 0; j < startCycles.length; j++) {
startCycles[j] = new Integer(CYCLEYEAR_START_DATE[j]);
}
}
for (int j = sYearInCycle + 1; j < CYCLEYEAR_START_DATE.length; j++) {
startCycles[j] = new Integer(startCycles[j].intValue() - offset);
}
// System.out.println(sCycleNumber + ":" + sYearInCycle);
ADJUSTED_CYCLE_YEARS.put(new Integer(sCycleNumber), startCycles);
int sYearInMaxY = (startYear - 1) / 30;
int sEndInMaxY = (endYear - 1) / 30;
if (sYearInMaxY != sEndInMaxY) {
// System.out.println("over 30");
// Adjusting starting 30 * MAX_ADJUSTED_CYCLE year cycle.
// System.out.println(sYearInMaxY);
for (int j = sYearInMaxY + 1; j < ADJUSTED_CYCLES.length; j++) {
ADJUSTED_CYCLES[j] = new Long(ADJUSTED_CYCLES[j].longValue()
- offset);
}
// Adjusting ending 30 * MAX_ADJUSTED_CYCLE year cycles.
for (int j = sEndInMaxY + 1; j < ADJUSTED_CYCLES.length; j++) {
ADJUSTED_CYCLES[j] = new Long(ADJUSTED_CYCLES[j].longValue()
+ offset);
}
}
// Adjusting ending 30 year cycle.
int eCycleNumber = (endYear - 1) / 30;
int sEndInCycle = (endYear - 1) % 30; // 0-based.
Integer[] endCycles = ADJUSTED_CYCLE_YEARS.get(new Integer(
eCycleNumber));
if (endCycles == null) {
endCycles = new Integer[CYCLEYEAR_START_DATE.length];
for (int j = 0; j < endCycles.length; j++) {
endCycles[j] = new Integer(CYCLEYEAR_START_DATE[j]);
}
}
for (int j = sEndInCycle + 1; j < CYCLEYEAR_START_DATE.length; j++) {
endCycles[j] = new Integer(endCycles[j].intValue() + offset);
}
ADJUSTED_CYCLE_YEARS.put(new Integer(eCycleNumber), endCycles);
}
// Adjusting ending year.
boolean isEndYLeap = isLeapYear(endYear);
Integer[] orgEndMonthDays = ADJUSTED_MONTH_DAYS.get(new Integer(endYear));
if (orgEndMonthDays == null) {
if (isEndYLeap) {
orgEndMonthDays = new Integer[LEAP_NUM_DAYS.length];
for (int l = 0; l < LEAP_NUM_DAYS.length; l++) {
orgEndMonthDays[l] = new Integer(LEAP_NUM_DAYS[l]);
}
} else {
orgEndMonthDays = new Integer[NUM_DAYS.length];
for (int l = 0; l < NUM_DAYS.length; l++) {
orgEndMonthDays[l] = new Integer(NUM_DAYS[l]);
}
}
}
Integer[] newEndMonthDays = new Integer[orgEndMonthDays.length];
for (int month = 0; month < 12; month++) {
if (month > endMonth) {
newEndMonthDays[month] = new Integer(orgEndMonthDays[month]
.intValue()
+ offset);
} else {
newEndMonthDays[month] = new Integer(orgEndMonthDays[month]
.intValue());
}
}
ADJUSTED_MONTH_DAYS.put(new Integer(endYear), newEndMonthDays);
// Adjusting the days of month.
Integer[] orgEndMonthLengths = ADJUSTED_MONTH_LENGTHS.get(new Integer(
endYear));
if (orgEndMonthLengths == null) {
if (isEndYLeap) {
orgEndMonthLengths = new Integer[LEAP_MONTH_LENGTH.length];
for (int l = 0; l < LEAP_MONTH_LENGTH.length; l++) {
orgEndMonthLengths[l] = new Integer(LEAP_MONTH_LENGTH[l]);
}
} else {
orgEndMonthLengths = new Integer[MONTH_LENGTH.length];
for (int l = 0; l < MONTH_LENGTH.length; l++) {
orgEndMonthLengths[l] = new Integer(MONTH_LENGTH[l]);
}
}
}
Integer[] newEndMonthLengths = new Integer[orgEndMonthLengths.length];
for (int month = 0; month < 12; month++) {
if (month == endMonth) {
newEndMonthLengths[month] = new Integer(
orgEndMonthLengths[month].intValue() + offset);
} else {
newEndMonthLengths[month] = new Integer(
orgEndMonthLengths[month].intValue());
}
}
ADJUSTED_MONTH_LENGTHS.put(new Integer(endYear), newEndMonthLengths);
Integer[] startMonthLengths = ADJUSTED_MONTH_LENGTHS.get(new Integer(
startYear));
Integer[] endMonthLengths = ADJUSTED_MONTH_LENGTHS.get(new Integer(
endYear));
Integer[] startMonthDays = ADJUSTED_MONTH_DAYS
.get(new Integer(startYear));
Integer[] endMonthDays = ADJUSTED_MONTH_DAYS.get(new Integer(endYear));
int startMonthLength = startMonthLengths[startMonth].intValue();
int endMonthLength = endMonthLengths[endMonth].intValue();
int startMonthDay = startMonthDays[11].intValue()
+ startMonthLengths[11].intValue();
int endMonthDay = endMonthDays[11].intValue()
+ endMonthLengths[11].intValue();
int maxMonthLength = ADJUSTED_MAX_VALUES[POSITION_DAY_OF_MONTH]
.intValue();
int leastMaxMonthLength = ADJUSTED_LEAST_MAX_VALUES[POSITION_DAY_OF_MONTH]
.intValue();
if (maxMonthLength < startMonthLength) {
maxMonthLength = startMonthLength;
}
if (maxMonthLength < endMonthLength) {
maxMonthLength = endMonthLength;
}
ADJUSTED_MAX_VALUES[POSITION_DAY_OF_MONTH] = new Integer(maxMonthLength);
if (leastMaxMonthLength > startMonthLength) {
leastMaxMonthLength = startMonthLength;
}
if (leastMaxMonthLength > endMonthLength) {
leastMaxMonthLength = endMonthLength;
}
ADJUSTED_LEAST_MAX_VALUES[POSITION_DAY_OF_MONTH] = new Integer(
leastMaxMonthLength);
int maxMonthDay = ADJUSTED_MAX_VALUES[POSITION_DAY_OF_YEAR].intValue();
int leastMaxMonthDay = ADJUSTED_LEAST_MAX_VALUES[POSITION_DAY_OF_YEAR]
.intValue();
if (maxMonthDay < startMonthDay) {
maxMonthDay = startMonthDay;
}
if (maxMonthDay < endMonthDay) {
maxMonthDay = endMonthDay;
}
ADJUSTED_MAX_VALUES[POSITION_DAY_OF_YEAR] = new Integer(maxMonthDay);
if (leastMaxMonthDay > startMonthDay) {
leastMaxMonthDay = startMonthDay;
}
if (leastMaxMonthDay > endMonthDay) {
leastMaxMonthDay = endMonthDay;
}
ADJUSTED_LEAST_MAX_VALUES[POSITION_DAY_OF_YEAR] = new Integer(
leastMaxMonthDay);
}
/**
* Read hijrah_deviation.cfg file. The config file contains the deviation data with
* following format.
*
* StartYear/StartMonth(0-based)-EndYear/EndMonth(0-based):Deviation day (1,
* 2, -1, or -2)
*
* Line separator or ";" is used for the separator of each deviation data.
*
* Here is the example.
*
* 1429/0-1429/1:1
* 1429/2-1429/7:1;1429/6-1429/11:1
* 1429/11-9999/11:1
*
* @throws IOException for zip/jar file handling exception.
* @throws ParseException if the format of the configuration file is wrong.
*/
private static void readDeviationConfig() throws IOException, ParseException {
InputStream is = getConfigFileInputStream();
if (is != null) {
BufferedReader br = null;
try {
br = new BufferedReader(new InputStreamReader(is));
String line = "";
int num = 0;
while ((line = br.readLine()) != null) {
num++;
line = line.trim();
parseLine(line, num);
}
} finally {
if (br != null) {
br.close();
}
}
}
}
/**
* Parse each deviation element.
*
* @param line a line to parse
* @param num line number
* @throws ParseException if line has incorrect format.
*/
private static void parseLine(String line, int num) throws ParseException {
StringTokenizer st = new StringTokenizer(line, ";");
while (st.hasMoreTokens()) {
String deviationElement = st.nextToken();
int offsetIndex = deviationElement.indexOf(':');
if (offsetIndex != -1) {
String offsetString = deviationElement.substring(
offsetIndex + 1, deviationElement.length());
int offset;
try {
offset = Integer.parseInt(offsetString);
} catch (NumberFormatException ex) {
throw new ParseException(
"Offset is not properly set at line " + num + ".",
num);
}
int separatorIndex = deviationElement.indexOf('-');
if (separatorIndex != -1) {
String startDateStg = deviationElement.substring(0,
separatorIndex);
String endDateStg = deviationElement.substring(
separatorIndex + 1, offsetIndex);
int startDateYearSepIndex = startDateStg.indexOf('/');
int endDateYearSepIndex = endDateStg.indexOf('/');
int startYear = -1;
int endYear = -1;
int startMonth = -1;
int endMonth = -1;
if (startDateYearSepIndex != -1) {
String startYearStg = startDateStg.substring(0,
startDateYearSepIndex);
String startMonthStg = startDateStg.substring(
startDateYearSepIndex + 1, startDateStg
.length());
try {
startYear = Integer.parseInt(startYearStg);
} catch (NumberFormatException ex) {
throw new ParseException(
"Start year is not properly set at line "
+ num + ".", num);
}
try {
startMonth = Integer.parseInt(startMonthStg);
} catch (NumberFormatException ex) {
throw new ParseException(
"Start month is not properly set at line "
+ num + ".", num);
}
} else {
throw new ParseException(
"Start year/month has incorrect format at line "
+ num + ".", num);
}
if (endDateYearSepIndex != -1) {
String endYearStg = endDateStg.substring(0,
endDateYearSepIndex);
String endMonthStg = endDateStg.substring(
endDateYearSepIndex + 1, endDateStg.length());
try {
endYear = Integer.parseInt(endYearStg);
} catch (NumberFormatException ex) {
throw new ParseException(
"End year is not properly set at line "
+ num + ".", num);
}
try {
endMonth = Integer.parseInt(endMonthStg);
} catch (NumberFormatException ex) {
throw new ParseException(
"End month is not properly set at line "
+ num + ".", num);
}
} else {
throw new ParseException(
"End year/month has incorrect format at line "
+ num + ".", num);
}
if (startYear != -1 && startMonth != -1 && endYear != -1
&& endMonth != -1) {
addDeviationAsHijrah(startYear, startMonth, endYear,
endMonth, offset);
} else {
throw new ParseException("Unknown error at line " + num
+ ".", num);
}
} else {
throw new ParseException(
"Start and end year/month has incorrect format at line "
+ num + ".", num);
}
} else {
throw new ParseException("Offset has incorrect format at line "
+ num + ".", num);
}
}
}
/**
* Return InputStream for deviation configuration file.
* The default location of the deviation file is:
*
* $CLASSPATH/org/threeten/bp/chrono
*
* And the default file name is:
*
* hijrah_deviation.cfg
*
* The default location and file name can be overriden by setting
* following two Java's system property.
*
* Location: org.threeten.bp.i18n.HijrahDate.deviationConfigDir
* File name: org.threeten.bp.i18n.HijrahDate.deviationConfigFile
*
* Regarding the file format, see readDeviationConfig() method for details.
*
* @return InputStream for file reading exception.
* @throws IOException for zip/jar file handling exception.
*/
private static InputStream getConfigFileInputStream() throws IOException {
String fileName = System
.getProperty("org.threeten.bp.i18n.HijrahDate.deviationConfigFile");
if (fileName == null) {
fileName = DEFAULT_CONFIG_FILENAME;
}
String dir = System
.getProperty("org.threeten.bp.i18n.HijrahDate.deviationConfigDir");
if (dir != null) {
if (!(dir.length() == 0 && dir.endsWith(System
.getProperty("file.separator")))) {
dir = dir + System.getProperty("file.separator");
}
File file = new File(dir + FILE_SEP + fileName);
if (file.exists()) {
try {
return new FileInputStream(file);
} catch (IOException ioe) {
throw ioe;
}
} else {
return null;
}
} else {
String classPath = System.getProperty("java.class.path");
StringTokenizer st = new StringTokenizer(classPath, PATH_SEP);
while (st.hasMoreTokens()) {
String path = st.nextToken();
File file = new File(path);
if (file.exists()) {
if (file.isDirectory()) {
File f = new File(
path + FILE_SEP + DEFAULT_CONFIG_PATH, fileName);
if (f.exists()) {
try {
return new FileInputStream(path + FILE_SEP
+ DEFAULT_CONFIG_PATH + FILE_SEP
+ fileName);
} catch (IOException ioe) {
throw ioe;
}
}
} else {
ZipFile zip;
try {
zip = new ZipFile(file);
} catch (IOException ioe) {
zip = null;
}
if (zip != null) {
String targetFile = DEFAULT_CONFIG_PATH + FILE_SEP
+ fileName;
ZipEntry entry = zip.getEntry(targetFile);
if (entry == null) {
if (FILE_SEP == '/') {
targetFile = targetFile.replace('/', '\\');
} else if (FILE_SEP == '\\') {
targetFile = targetFile.replace('\\', '/');
}
entry = zip.getEntry(targetFile);
}
if (entry != null) {
try {
return zip.getInputStream(entry);
} catch (IOException ioe) {
throw ioe;
}
}
}
}
}
}
return null;
}
}
//-----------------------------------------------------------------------
private Object writeReplace() {
return new Ser(Ser.HIJRAH_DATE_TYPE, this);
}
void writeExternal(DataOutput out) throws IOException {
// HijrahChrono is implicit in the Hijrah_DATE_TYPE
out.writeInt(get(YEAR));
out.writeByte(get(MONTH_OF_YEAR));
out.writeByte(get(DAY_OF_MONTH));
}
static ChronoLocalDate readExternal(DataInput in) throws IOException {
int year = in.readInt();
int month = in.readByte();
int dayOfMonth = in.readByte();
return HijrahChronology.INSTANCE.date(year, month, dayOfMonth);
}
}