mirror of https://github.com/apache/poi.git
Handle timezones better with cell.setCellValue(Calendar), so now 20:00-03:00, 20:00+00:00 and 20:00+03:00 will all be recorded as 20:00, and not 17:00 / 20:00 / 23:00 (pass a Date not a Calendar for old behaviour) - patch from bug #38641
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@619502 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
212604f51e
commit
d543f9034a
|
@ -36,6 +36,7 @@
|
||||||
|
|
||||||
<!-- Don't forget to update status.xml too! -->
|
<!-- Don't forget to update status.xml too! -->
|
||||||
<release version="3.1-beta1" date="2008-??-??">
|
<release version="3.1-beta1" date="2008-??-??">
|
||||||
|
<action dev="POI-DEVELOPERS" type="fix">38641 - Handle timezones better with cell.setCellValue(Calendar), so now 20:00-03:00, 20:00+00:00 and 20:00+03:00 will all be recorded as 20:00, and not 17:00 / 20:00 / 23:00 (pass a Date not a Calendar for old behaviour)</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">44373 - Have HSSFDateUtil.isADateFormat recognize more formats as being dates</action>
|
<action dev="POI-DEVELOPERS" type="fix">44373 - Have HSSFDateUtil.isADateFormat recognize more formats as being dates</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">37923 - Support for Excel hyperlinks</action>
|
<action dev="POI-DEVELOPERS" type="add">37923 - Support for Excel hyperlinks</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">Implement hashCode() and equals(obj) on HSSFFont and HSSFCellStyle</action>
|
<action dev="POI-DEVELOPERS" type="add">Implement hashCode() and equals(obj) on HSSFFont and HSSFCellStyle</action>
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
<!-- Don't forget to update changes.xml too! -->
|
<!-- Don't forget to update changes.xml too! -->
|
||||||
<changes>
|
<changes>
|
||||||
<release version="3.1-beta1" date="2008-??-??">
|
<release version="3.1-beta1" date="2008-??-??">
|
||||||
|
<action dev="POI-DEVELOPERS" type="fix">38641 - Handle timezones better with cell.setCellValue(Calendar), so now 20:00-03:00, 20:00+00:00 and 20:00+03:00 will all be recorded as 20:00, and not 17:00 / 20:00 / 23:00 (pass a Date not a Calendar for old behaviour)</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">44373 - Have HSSFDateUtil.isADateFormat recognize more formats as being dates</action>
|
<action dev="POI-DEVELOPERS" type="fix">44373 - Have HSSFDateUtil.isADateFormat recognize more formats as being dates</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">37923 - Support for Excel hyperlinks</action>
|
<action dev="POI-DEVELOPERS" type="add">37923 - Support for Excel hyperlinks</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">Implement hashCode() and equals(obj) on HSSFFont and HSSFCellStyle</action>
|
<action dev="POI-DEVELOPERS" type="add">Implement hashCode() and equals(obj) on HSSFFont and HSSFCellStyle</action>
|
||||||
|
|
|
@ -547,13 +547,20 @@ public class HSSFCell
|
||||||
* set a date value for the cell. Excel treats dates as numeric so you will need to format the cell as
|
* set a date value for the cell. Excel treats dates as numeric so you will need to format the cell as
|
||||||
* a date.
|
* a date.
|
||||||
*
|
*
|
||||||
|
* This will set the cell value based on the Calendar's timezone. As Excel
|
||||||
|
* does not support timezones this means that both 20:00+03:00 and
|
||||||
|
* 20:00-03:00 will be reported as the same value (20:00) even that there
|
||||||
|
* are 6 hours difference between the two times. This difference can be
|
||||||
|
* preserved by using <code>setCellValue(value.getTime())</code> which will
|
||||||
|
* automatically shift the times to the default timezone.
|
||||||
|
*
|
||||||
* @param value the date value to set this cell to. For formulas we'll set the
|
* @param value the date value to set this cell to. For formulas we'll set the
|
||||||
* precalculated value, for numerics we'll set its value. For othertypes we
|
* precalculated value, for numerics we'll set its value. For othertypes we
|
||||||
* will change the cell to a numeric cell and set its value.
|
* will change the cell to a numeric cell and set its value.
|
||||||
*/
|
*/
|
||||||
public void setCellValue(Calendar value)
|
public void setCellValue(Calendar value)
|
||||||
{
|
{
|
||||||
setCellValue(value.getTime());
|
setCellValue( HSSFDateUtil.getExcelDate(value, this.book.isUsing1904DateWindowing()) );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -70,9 +70,25 @@ public class HSSFDateUtil
|
||||||
public static double getExcelDate(Date date, boolean use1904windowing) {
|
public static double getExcelDate(Date date, boolean use1904windowing) {
|
||||||
Calendar calStart = new GregorianCalendar();
|
Calendar calStart = new GregorianCalendar();
|
||||||
calStart.setTime(date); // If date includes hours, minutes, and seconds, set them to 0
|
calStart.setTime(date); // If date includes hours, minutes, and seconds, set them to 0
|
||||||
|
return internalGetExcelDate(calStart, use1904windowing);
|
||||||
if ((!use1904windowing && calStart.get(Calendar.YEAR) < 1900) ||
|
}
|
||||||
(use1904windowing && calStart.get(Calendar.YEAR) < 1904))
|
/**
|
||||||
|
* Given a Date in the form of a Calendar, converts it into a double
|
||||||
|
* representing its internal Excel representation, which is the
|
||||||
|
* number of days since 1/1/1900. Fractional days represent hours,
|
||||||
|
* minutes, and seconds.
|
||||||
|
*
|
||||||
|
* @return Excel representation of Date (-1 if error - test for error by checking for less than 0.1)
|
||||||
|
* @param date the Calendar holding the date to convert
|
||||||
|
* @param use1904windowing Should 1900 or 1904 date windowing be used?
|
||||||
|
*/
|
||||||
|
public static double getExcelDate(Calendar date, boolean use1904windowing) {
|
||||||
|
// Don't alter the supplied Calendar as we do our work
|
||||||
|
return internalGetExcelDate( (Calendar)date.clone(), use1904windowing );
|
||||||
|
}
|
||||||
|
private static double internalGetExcelDate(Calendar date, boolean use1904windowing) {
|
||||||
|
if ((!use1904windowing && date.get(Calendar.YEAR) < 1900) ||
|
||||||
|
(use1904windowing && date.get(Calendar.YEAR) < 1904))
|
||||||
{
|
{
|
||||||
return BAD_DATE;
|
return BAD_DATE;
|
||||||
} else {
|
} else {
|
||||||
|
@ -83,12 +99,12 @@ public class HSSFDateUtil
|
||||||
// be 4 hours.
|
// be 4 hours.
|
||||||
// E.g. 2004-03-28 04:00 CEST - 2004-03-28 00:00 CET is 3 hours
|
// E.g. 2004-03-28 04:00 CEST - 2004-03-28 00:00 CET is 3 hours
|
||||||
// and 2004-10-31 04:00 CET - 2004-10-31 00:00 CEST is 5 hours
|
// and 2004-10-31 04:00 CET - 2004-10-31 00:00 CEST is 5 hours
|
||||||
double fraction = (((calStart.get(Calendar.HOUR_OF_DAY) * 60
|
double fraction = (((date.get(Calendar.HOUR_OF_DAY) * 60
|
||||||
+ calStart.get(Calendar.MINUTE)
|
+ date.get(Calendar.MINUTE)
|
||||||
) * 60 + calStart.get(Calendar.SECOND)
|
) * 60 + date.get(Calendar.SECOND)
|
||||||
) * 1000 + calStart.get(Calendar.MILLISECOND)
|
) * 1000 + date.get(Calendar.MILLISECOND)
|
||||||
) / ( double ) DAY_MILLISECONDS;
|
) / ( double ) DAY_MILLISECONDS;
|
||||||
calStart = dayStart(calStart);
|
Calendar calStart = dayStart(date);
|
||||||
|
|
||||||
double value = fraction + absoluteDay(calStart, use1904windowing);
|
double value = fraction + absoluteDay(calStart, use1904windowing);
|
||||||
|
|
||||||
|
|
|
@ -196,6 +196,29 @@ public class TestHSSFDateUtil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that we deal with timezones properly
|
||||||
|
*/
|
||||||
|
public void testCalendarConversion() {
|
||||||
|
GregorianCalendar date = new GregorianCalendar(2002, 0, 1, 12, 1, 1);
|
||||||
|
Date expected = date.getTime();
|
||||||
|
double expectedExcel = HSSFDateUtil.getExcelDate(expected);
|
||||||
|
|
||||||
|
// Iteratating over the hours exposes any rounding issues.
|
||||||
|
for (int hour = -12; hour <= 12; hour++)
|
||||||
|
{
|
||||||
|
String id = "GMT" + (hour < 0 ? "" : "+") + hour + ":00";
|
||||||
|
date.setTimeZone(TimeZone.getTimeZone(id));
|
||||||
|
date.set(Calendar.HOUR_OF_DAY, 12);
|
||||||
|
double excelDate = HSSFDateUtil.getExcelDate(date, false);
|
||||||
|
Date javaDate = HSSFDateUtil.getJavaDate(excelDate);
|
||||||
|
|
||||||
|
// Should match despite timezone
|
||||||
|
assertEquals("Checking timezone " + id, expected.getTime(), javaDate.getTime());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests that we correctly detect date formats as such
|
* Tests that we correctly detect date formats as such
|
||||||
*/
|
*/
|
||||||
|
|
Loading…
Reference in New Issue