diff options
author | Yegor Kozlov <yegor@apache.org> | 2010-09-04 12:33:09 +0000 |
---|---|---|
committer | Yegor Kozlov <yegor@apache.org> | 2010-09-04 12:33:09 +0000 |
commit | 6f9470cc897dd3247ccd147c116f7151f4fd007f (patch) | |
tree | d7e6cdf10d9d980668a1dd04e643581e1328f523 | |
parent | 5fc4b570caeff1c689881d001628ebeeab3c4a41 (diff) | |
download | poi-6f9470cc897dd3247ccd147c116f7151f4fd007f.tar.gz poi-6f9470cc897dd3247ccd147c116f7151f4fd007f.zip |
change cell type to error when setting Double.NaN or Infinities, see Bugzilla 49761
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@992591 13f79535-47bb-0310-9956-ffa450edef68
5 files changed, 91 insertions, 19 deletions
diff --git a/src/documentation/content/xdocs/status.xml b/src/documentation/content/xdocs/status.xml index ecd28631be..a4ffb04682 100644 --- a/src/documentation/content/xdocs/status.xml +++ b/src/documentation/content/xdocs/status.xml @@ -34,6 +34,7 @@ <changes> <release version="3.7-beta3" date="2010-??-??"> + <action dev="poi-developers" type="fix">49761 - change cell type to error when setting Double.NaN or Infinities</action> <action dev="poi-developers" type="fix">49833 - ensure that CTNumPr is included in poi-ooxml-schemas.jar</action> <action dev="POI-DEVELOPERS" type="fix">49841 - fixed LEFT and RIGHT to return #VALUE! when called with a negative operand </action> <action dev="POI-DEVELOPERS" type="fix">49783 - fixed evaluation of XSSF workbooks containing formulas with reference errors (#REF!)</action> diff --git a/src/java/org/apache/poi/hssf/usermodel/HSSFCell.java b/src/java/org/apache/poi/hssf/usermodel/HSSFCell.java index 35d09c6b6d..2862a0d032 100644 --- a/src/java/org/apache/poi/hssf/usermodel/HSSFCell.java +++ b/src/java/org/apache/poi/hssf/usermodel/HSSFCell.java @@ -46,11 +46,7 @@ import org.apache.poi.hssf.record.common.UnicodeString; import org.apache.poi.hssf.record.formula.ExpPtg; import org.apache.poi.hssf.record.formula.Ptg; import org.apache.poi.hssf.record.formula.eval.ErrorEval; -import org.apache.poi.ss.usermodel.Cell; -import org.apache.poi.ss.usermodel.CellStyle; -import org.apache.poi.ss.usermodel.Comment; -import org.apache.poi.ss.usermodel.Hyperlink; -import org.apache.poi.ss.usermodel.RichTextString; +import org.apache.poi.ss.usermodel.*; import org.apache.poi.ss.util.CellRangeAddress; import org.apache.poi.ss.util.CellReference; import org.apache.poi.ss.util.NumberToTextConverter; @@ -464,20 +460,31 @@ public class HSSFCell implements Cell { * will change the cell to a numeric cell and set its value. */ public void setCellValue(double value) { - int row=_record.getRow(); - short col=_record.getColumn(); - short styleIndex=_record.getXFIndex(); - - switch (_cellType) { - default: - setCellType(CELL_TYPE_NUMERIC, false, row, col, styleIndex); - case CELL_TYPE_NUMERIC: - (( NumberRecord ) _record).setValue(value); - break; - case CELL_TYPE_FORMULA: - ((FormulaRecordAggregate)_record).setCachedDoubleResult(value); - break; + if(Double.isInfinite(value)) { + // Excel does not support positive/negative infinities, + // rather, it gives a #DIV/0! error in these cases. + setCellErrorValue(FormulaError.DIV0.getCode()); + } else if (Double.isNaN(value)){ + // Excel does not support Not-a-Number (NaN), + // instead it immediately generates a #NUM! error. + setCellErrorValue(FormulaError.NUM.getCode()); + } else { + int row=_record.getRow(); + short col=_record.getColumn(); + short styleIndex=_record.getXFIndex(); + + switch (_cellType) { + default: + setCellType(CELL_TYPE_NUMERIC, false, row, col, styleIndex); + case CELL_TYPE_NUMERIC: + (( NumberRecord ) _record).setValue(value); + break; + case CELL_TYPE_FORMULA: + ((FormulaRecordAggregate)_record).setCachedDoubleResult(value); + break; + } } + } /** diff --git a/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFCell.java b/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFCell.java index 374c6b00d2..cee581daf8 100644 --- a/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFCell.java +++ b/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFCell.java @@ -218,7 +218,14 @@ public final class XSSFCell implements Cell { * will change the cell to a numeric cell and set its value. */ public void setCellValue(double value) { - if(Double.isInfinite(value) || Double.isNaN(value)) { + if(Double.isInfinite(value)) { + // Excel does not support positive/negative infinities, + // rather, it gives a #DIV/0! error in these cases. + _cell.setT(STCellType.E); + _cell.setV(FormulaError.DIV0.getString()); + } else if (Double.isNaN(value)){ + // Excel does not support Not-a-Number (NaN), + // instead it immediately generates an #NUM! error. _cell.setT(STCellType.E); _cell.setV(FormulaError.NUM.getString()); } else { diff --git a/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFCell.java b/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFCell.java index 77e95c622a..76f53c9c37 100644 --- a/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFCell.java +++ b/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFCell.java @@ -320,4 +320,6 @@ public final class TestHSSFCell extends BaseTestCell { assertEquals("The maximum length of cell contents (text) is 32,767 characters", e.getMessage()); } } + + } diff --git a/src/testcases/org/apache/poi/ss/usermodel/BaseTestCell.java b/src/testcases/org/apache/poi/ss/usermodel/BaseTestCell.java index 39d21dd85f..dbbad380d4 100644 --- a/src/testcases/org/apache/poi/ss/usermodel/BaseTestCell.java +++ b/src/testcases/org/apache/poi/ss/usermodel/BaseTestCell.java @@ -489,4 +489,59 @@ public abstract class BaseTestCell extends TestCase { int i2 = cell.getCellStyle().getIndex(); assertEquals(i1, i2); } + + /** + * Excel's implementation of floating number arithmetic does not fully adhere to IEEE 754: + * + * From http://support.microsoft.com/kb/78113: + * + * <ul> + * <li> Positive/Negative Infinities: + * Infinities occur when you divide by 0. Excel does not support infinities, rather, + * it gives a #DIV/0! error in these cases. + * </li> + * <li> + * Not-a-Number (NaN): + * NaN is used to represent invalid operations (such as infinity/infinity, + * infinity-infinity, or the square root of -1). NaNs allow a program to + * continue past an invalid operation. Excel instead immediately generates + * an error such as #NUM! or #DIV/0!. + * </li> + * </ul> + */ + public void testNanAndInfinity() { + Workbook wb = _testDataProvider.createWorkbook(); + Sheet workSheet = wb.createSheet("Sheet1"); + Row row = workSheet.createRow(0); + + Cell cell0 = row.createCell(0); + cell0.setCellValue(Double.NaN); + assertEquals("Double.NaN should change cell type to CELL_TYPE_ERROR", Cell.CELL_TYPE_ERROR, cell0.getCellType()); + assertEquals("Double.NaN should change cell value to #NUM!", ErrorConstants.ERROR_NUM, cell0.getErrorCellValue()); + + Cell cell1 = row.createCell(1); + cell1.setCellValue(Double.POSITIVE_INFINITY); + assertEquals("Double.POSITIVE_INFINITY should change cell type to CELL_TYPE_ERROR", Cell.CELL_TYPE_ERROR, cell1.getCellType()); + assertEquals("Double.POSITIVE_INFINITY should change cell value to #DIV/0!", ErrorConstants.ERROR_DIV_0, cell1.getErrorCellValue()); + + Cell cell2 = row.createCell(2); + cell2.setCellValue(Double.NEGATIVE_INFINITY); + assertEquals("Double.NEGATIVE_INFINITY should change cell type to CELL_TYPE_ERROR", Cell.CELL_TYPE_ERROR, cell2.getCellType()); + assertEquals("Double.NEGATIVE_INFINITY should change cell value to #DIV/0!", ErrorConstants.ERROR_DIV_0, cell2.getErrorCellValue()); + + wb = _testDataProvider.writeOutAndReadBack(wb); + row = wb.getSheetAt(0).getRow(0); + + cell0 = row.getCell(0); + assertEquals(Cell.CELL_TYPE_ERROR, cell0.getCellType()); + assertEquals(ErrorConstants.ERROR_NUM, cell0.getErrorCellValue()); + + cell1 = row.getCell(1); + assertEquals(Cell.CELL_TYPE_ERROR, cell1.getCellType()); + assertEquals(ErrorConstants.ERROR_DIV_0, cell1.getErrorCellValue()); + + cell2 = row.getCell(2); + assertEquals(Cell.CELL_TYPE_ERROR, cell2.getCellType()); + assertEquals(ErrorConstants.ERROR_DIV_0, cell2.getErrorCellValue()); + } } |