blob: 836ed32bae3d85661ac3aa43bd3f7d61a5cb6f54 [file] [log] [blame]
/* ====================================================================
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You under the Apache License, Version 2.0
(the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
==================================================================== */
package org.apache.poi.hssf.usermodel;
import static org.apache.poi.hssf.HSSFTestDataSamples.writeOutAndReadBack;
import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertSame;
import static org.junit.jupiter.api.Assertions.assertTrue;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Calendar;
import org.apache.poi.hssf.HSSFTestDataSamples;
import org.apache.poi.hssf.model.HSSFFormulaParser;
import org.apache.poi.ss.formula.FormulaType;
import org.apache.poi.ss.formula.ptg.NamePtg;
import org.apache.poi.ss.formula.ptg.Ptg;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.DateUtil;
import org.apache.poi.ss.usermodel.Name;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.util.CellReference;
import org.apache.poi.util.LocaleUtil;
import org.apache.poi.util.TempFile;
import org.junit.jupiter.api.Test;
final class TestFormulas {
private static HSSFWorkbook openSample(String sampleFileName) {
return HSSFTestDataSamples.openSampleWorkbook(sampleFileName);
}
/**
* Add 1+1 -- WHoohoo!
*/
@Test
void testBasicAddIntegers() throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
//get our minimum values
HSSFRow r = s.createRow(1);
HSSFCell c = r.createCell(1);
c.setCellFormula(1 + "+" + 1);
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
r = s.getRow(1);
c = r.getCell(1);
assertEquals("1+1", c.getCellFormula(), "Formula is as expected");
}
}
}
/**
* Add various integers
*/
@Test
void testAddIntegers() throws IOException {
binomialOperator("+");
}
/**
* Multiply various integers
*/
@Test
void testMultplyIntegers() throws IOException {
binomialOperator("*");
}
/**
* Subtract various integers
*/
@Test
void testSubtractIntegers() throws IOException {
binomialOperator("-");
}
/**
* Subtract various integers
*/
@Test
void testDivideIntegers() throws IOException {
binomialOperator("/");
}
/**
* Exponentialize various integers;
*/
@Test
void testPowerIntegers() throws IOException {
binomialOperator("^");
}
/**
* Concatenate two numbers 1&2 = 12
*/
@Test
void testConcatIntegers() throws IOException {
binomialOperator("&");
}
/**
* tests 1*2+3*4
*/
@Test
void testOrderOfOperationsMultiply() throws IOException {
orderTest("1*2+3*4");
}
/**
* tests 1*2+3^4
*/
@Test
void testOrderOfOperationsPower() throws IOException {
orderTest("1*2+3^4");
}
/**
* Tests that parenthesis are obeyed
*/
@Test
void testParenthesis() throws IOException {
orderTest("(1*3)+2+(1+2)*(3^4)^5");
}
@Test
void testReferencesOpr() throws IOException {
String[] operation = new String[] {
"+", "-", "*", "/", "^", "&"
};
for (final String op : operation) {
operationRefTest(op);
}
}
/**
* Tests creating a file with floating point in a formula.
*
*/
@Test
void testFloat() throws IOException {
floatTest("*");
floatTest("/");
}
private static void floatTest(String operator) throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
//get our minimum values
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(1);
c.setCellFormula("" + Float.MIN_VALUE + operator + Float.MIN_VALUE);
for (int x = 1; x < Short.MAX_VALUE && x > 0; x = (short) (x * 2)) {
r = s.createRow(x);
for (int y = 1; y < 256 && y > 0; y = (short) (y + 2)) {
c = r.createCell(y);
c.setCellFormula("" + x + "." + y + operator + y + "." + x);
}
}
if (s.getLastRowNum() < Short.MAX_VALUE) {
r = s.createRow(0);
c = r.createCell(0);
c.setCellFormula("" + Float.MAX_VALUE + operator + Float.MAX_VALUE);
}
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
floatVerify(operator, wb2);
}
}
}
private static void floatVerify(String operator, HSSFWorkbook wb) {
HSSFSheet s = wb.getSheetAt(0);
// don't know how to check correct result .. for the moment, we just verify that the file can be read.
for (int x = 1; x < Short.MAX_VALUE && x > 0; x=(short)(x*2)) {
HSSFRow r = s.getRow(x);
for (int y = 1; y < 256 && y > 0; y=(short)(y+2)) {
HSSFCell c = r.getCell(y);
assertNotNull(c.getCellFormula(), "got a formula");
assertEquals(("" + x + "." + y + operator + y + "." + x), c.getCellFormula(),
"loop Formula is as expected " + x + "." + y + operator + y + "." + x + "!=" + c.getCellFormula());
}
}
}
@Test
void testAreaSum() throws IOException {
areaFunctionTest("SUM");
}
@Test
void testAreaAverage() throws IOException {
areaFunctionTest("AVERAGE");
}
@Test
void testRefArraySum() throws IOException {
refArrayFunctionTest("SUM");
}
@Test
void testAreaArraySum() throws IOException {
refAreaArrayFunctionTest("SUM");
}
private static void operationRefTest(String operator) throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
//get our minimum values
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(1);
c.setCellFormula("A2" + operator + "A3");
for (int x = 1; x < Short.MAX_VALUE && x > 0; x = (short) (x * 2)) {
r = s.createRow(x);
for (int y = 1; y < 256 && y > 0; y++) {
short refx1 = (short)(x + (x + 50 < Short.MAX_VALUE ? 50 : -4));
short refx2 = (short)(x + (x + 50 < Short.MAX_VALUE ? 46 : -3));
short refy1 = (short)(y + (y + 50 < 255 ? 50 : -4));
short refy2 = (short)(y + (y + 50 < 255 ? 49 : -3));
CellReference cr = new CellReference(refx1, refy1, false, false);
String ref = cr.formatAsString();
cr = new CellReference(refx2, refy2, false, false);
String ref2 = cr.formatAsString();
c = r.createCell(y);
c.setCellFormula("" + ref + operator + ref2);
}
}
//make sure we do the maximum value of the Int operator
if (s.getLastRowNum() < Short.MAX_VALUE) {
r = s.getRow(0);
c = r.createCell(0);
c.setCellFormula("" + "B1" + operator + "IV255");
}
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
operationalRefVerify(operator, wb2);
}
}
}
/**
* Opens the sheet we wrote out by binomialOperator and makes sure the formulas
* all match what we expect (x operator y)
*/
private static void operationalRefVerify(String operator, HSSFWorkbook wb) {
HSSFSheet s = wb.getSheetAt(0);
//get our minimum values
HSSFRow r = s.getRow(0);
HSSFCell c = r.getCell(1);
//get our minimum values
assertEquals(("A2" + operator + "A3"), c.getCellFormula(),
"minval Formula is as expected A2" + operator + "A3 != " + c.getCellFormula());
for (int x = 1; x < Short.MAX_VALUE && x > 0; x=(short)(x*2)) {
r = s.getRow(x);
for (int y = 1; y < 256 && y > 0; y++) {
int refx1 = x + (x+50 < Short.MAX_VALUE ? 50 : -4);
int refx2 = x + (x+50 < Short.MAX_VALUE ? 46 : -3);
int refy1 = y + (y+50 < 255 ? 50 : -4);
int refy2 = y + (y+50 < 255 ? 49 : -3);
c = r.getCell(y);
CellReference cr= new CellReference(refx1, refy1, false, false);
String ref=cr.formatAsString();
cr=new CellReference(refx2,refy2, false, false);
String ref2=cr.formatAsString();
assertEquals(("" + ref + operator + ref2), c.getCellFormula(),
"loop Formula is as expected " + ref + operator + ref2 + "!=" + c.getCellFormula());
}
}
//test our maximum values
r = s.getRow(0);
c = r.getCell(0);
assertEquals("B1"+operator+"IV255", c.getCellFormula());
}
/**
* tests order wrting out == order writing in for a given formula
*/
private static void orderTest(String formula) throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
//get our minimum values
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(1);
c.setCellFormula(formula);
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
//get our minimum values
r = s.getRow(0);
c = r.getCell(1);
assertEquals(formula, c.getCellFormula(), "minval Formula is as expected");
}
}
}
/**
* All multi-binomial operator tests use this to create a worksheet with a
* huge set of x operator y formulas. Next we call binomialVerify and verify
* that they are all how we expect.
*/
private static void binomialOperator(String operator) throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
//get our minimum values
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(1);
c.setCellFormula(1 + operator + 1);
for (int x = 1; x < Short.MAX_VALUE && x > 0; x = (short) (x * 2)) {
r = s.createRow(x);
for (int y = 1; y < 256 && y > 0; y++) {
c = r.createCell(y);
c.setCellFormula("" + x + operator + y);
}
}
//make sure we do the maximum value of the Int operator
if (s.getLastRowNum() < Short.MAX_VALUE) {
r = s.getRow(0);
c = r.createCell(0);
c.setCellFormula("" + Short.MAX_VALUE + operator + Short.MAX_VALUE);
}
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
binomialVerify(operator, wb2);
}
}
}
/**
* Opens the sheet we wrote out by binomialOperator and makes sure the formulas
* all match what we expect (x operator y)
*/
private static void binomialVerify(String operator, HSSFWorkbook wb) {
HSSFSheet s = wb.getSheetAt(0);
//get our minimum values
HSSFRow r = s.getRow(0);
HSSFCell c = r.getCell(1);
assertEquals(("1" + operator + "1"), c.getCellFormula(),
"minval Formula is as expected 1" + operator + "1 != " + c.getCellFormula());
for (int x = 1; x < Short.MAX_VALUE && x > 0; x=(short)(x*2)) {
r = s.getRow(x);
for (int y = 1; y < 256 && y > 0; y++) {
c = r.getCell(y);
assertEquals(("" + x + operator + y), c.getCellFormula(),
"loop Formula is as expected " + x + operator + y + "!=" + c.getCellFormula());
}
}
//test our maximum values
r = s.getRow(0);
c = r.getCell(0);
assertEquals(("" + Short.MAX_VALUE + operator + Short.MAX_VALUE), c.getCellFormula(),
"maxval Formula is as expected");
}
/**
* Writes a function then tests to see if its correct
*/
public static void areaFunctionTest(String function) throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellFormula(function + "(A2:A3)");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
r = s.getRow(0);
c = r.getCell(0);
assertEquals(function + "(A2:A3)", c.getCellFormula(),
"function =" + function + "(A2:A3)");
}
}
}
/**
* Writes a function then tests to see if its correct
*/
void refArrayFunctionTest(String function) throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellFormula(function + "(A2,A3)");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
r = s.getRow(0);
c = r.getCell(0);
assertEquals((function + "(A2,A3)"), c.getCellFormula(), "function =" + function + "(A2,A3)");
}
}
}
/**
* Writes a function then tests to see if its correct
*
*/
void refAreaArrayFunctionTest(String function) throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellFormula(function + "(A2:A4,B2:B4)");
c = r.createCell(1);
c.setCellFormula(function + "($A$2:$A4,B$2:B4)");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
r = s.getRow(0);
c = r.getCell(0);
assertEquals((function + "(A2:A4,B2:B4)"), c.getCellFormula(), "function =" + function + "(A2:A4,B2:B4)");
c = r.getCell(1);
assertEquals((function + "($A$2:$A4,B$2:B4)"), c.getCellFormula(), "function =" + function + "($A$2:$A4,B$2:B4)");
}
}
}
@Test
void testAbsRefs() throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet();
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellFormula("A3+A2");
c = r.createCell(1);
c.setCellFormula("$A3+$A2");
c = r.createCell(2);
c.setCellFormula("A$3+A$2");
c = r.createCell(3);
c.setCellFormula("$A$3+$A$2");
c = r.createCell(4);
c.setCellFormula("SUM($A$3,$A$2)");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
r = s.getRow(0);
c = r.getCell(0);
assertEquals("A3+A2", c.getCellFormula());
c = r.getCell(1);
assertEquals("$A3+$A2", c.getCellFormula());
c = r.getCell(2);
assertEquals("A$3+A$2", c.getCellFormula());
c = r.getCell(3);
assertEquals("$A$3+$A$2", c.getCellFormula());
c = r.getCell(4);
assertEquals("SUM($A$3,$A$2)", c.getCellFormula());
}
}
}
@Test
void testSheetFunctions() throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet("A");
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellValue(1);
c = r.createCell(1);
c.setCellValue(2);
s = wb1.createSheet("B");
r = s.createRow(0);
c = r.createCell(0);
c.setCellFormula("AVERAGE(A!A1:B1)");
c = r.createCell(1);
c.setCellFormula("A!A1+A!B1");
c = r.createCell(2);
c.setCellFormula("A!$A$1+A!$B1");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheet("B");
r = s.getRow(0);
c = r.getCell(0);
assertEquals("AVERAGE(A!A1:B1)", c.getCellFormula(),
"expected: AVERAGE(A!A1:B1) got: " + c.getCellFormula());
c = r.getCell(1);
assertEquals("A!A1+A!B1", c.getCellFormula(),
"expected: A!A1+A!B1 got: " + c.getCellFormula());
}
}
}
@Test
void testRVAoperands() throws IOException {
File file = TempFile.createTempFile("testFormulaRVA",".xls");
try (HSSFWorkbook wb = new HSSFWorkbook()) {
HSSFSheet s = wb.createSheet();
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellFormula("A3+A2");
c = r.createCell(1);
c.setCellFormula("AVERAGE(A3,A2)");
c = r.createCell(2);
c.setCellFormula("ROW(A3)");
c = r.createCell(3);
c.setCellFormula("AVERAGE(A2:A3)");
c = r.createCell(4);
c.setCellFormula("POWER(A2,A3)");
c = r.createCell(5);
c.setCellFormula("SIN(A2)");
c = r.createCell(6);
c.setCellFormula("SUM(A2:A3)");
c = r.createCell(7);
c.setCellFormula("SUM(A2,A3)");
r = s.createRow(1);
c = r.createCell(0);
c.setCellValue(2.0);
r = s.createRow(2);
c = r.createCell(0);
c.setCellValue(3.0);
try (FileOutputStream out = new FileOutputStream(file)) {
wb.write(out);
}
assertTrue(file.exists());
}
}
@Test
void testStringFormulas() throws IOException {
try (HSSFWorkbook wb = new HSSFWorkbook()) {
HSSFSheet s = wb.createSheet("A");
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(1);
c.setCellFormula("UPPER(\"abc\")");
c = r.createCell(2);
c.setCellFormula("LOWER(\"ABC\")");
c = r.createCell(3);
c.setCellFormula("CONCATENATE(\" my \",\" name \")");
writeOutAndReadBack(wb).close();
}
try (HSSFWorkbook wb = openSample("StringFormulas.xls")) {
HSSFSheet s = wb.getSheetAt(0);
HSSFRow r = s.getRow(0);
HSSFCell c = r.getCell(0);
assertEquals("UPPER(\"xyz\")", c.getCellFormula());
}
}
@Test
void testLogicalFormulas() throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet("A");
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(1);
c.setCellFormula("IF(A1<A2,B1,B2)");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
r = s.getRow(0);
c = r.getCell(1);
assertEquals("IF(A1<A2,B1,B2)", c.getCellFormula(), "Formula in cell 1 ");
}
}
}
@Test
void testDateFormulas() throws IOException {
try (HSSFWorkbook wb = new HSSFWorkbook()) {
HSSFSheet s = wb.createSheet("testSheet1");
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
Calendar cal = LocaleUtil.getLocaleCalendar();
HSSFCellStyle cellStyle = wb.createCellStyle();
cellStyle.setDataFormat(HSSFDataFormat.getBuiltinFormat("m/d/yy h:mm"));
c.setCellValue(cal.getTime());
c.setCellStyle(cellStyle);
int hour = cal.get(Calendar.HOUR_OF_DAY);
double excelDate = c.getNumericCellValue();
assertEquals(cal.getTime().getTime(), DateUtil.getJavaDate(excelDate).getTime());
assertEquals(cal.getTime(), c.getDateCellValue());
for (int k = 1; k < 100; k++) {
r = s.createRow(k);
c = r.createCell(0);
c.setCellFormula("A" + (k) + "+1");
c.setCellStyle(cellStyle);
}
writeOutAndReadBack(wb).close();
}
}
@Test
void testIfFormulas() throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet("testSheet1");
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(1);
c.setCellValue(1);
c = r.createCell(2);
c.setCellValue(2);
c = r.createCell(3);
c.setCellFormula("MAX(A1:B1)");
c = r.createCell(4);
c.setCellFormula("IF(A1=D1,\"A1\",\"B1\")");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s = wb2.getSheetAt(0);
r = s.getRow(0);
c = r.getCell(4);
assertEquals("IF(A1=D1,\"A1\",\"B1\")", c.getCellFormula(),
"expected: IF(A1=D1,\"A1\",\"B1\") got " + c.getCellFormula());
}
}
try (HSSFWorkbook wb1 = openSample("IfFormulaTest.xls")) {
HSSFSheet s = wb1.getSheetAt(0);
HSSFRow r = s.getRow(3);
HSSFCell c = r.getCell(0);
assertEquals("IF(A3=A1,\"A1\",\"A2\")", c.getCellFormula(),
"expected: IF(A3=A1,\"A1\",\"A2\") got " + c.getCellFormula());
//c = r.getCell((short)1);
//assertTrue("expected: A!A1+A!B1 got: "+c.getCellFormula(), ("A!A1+A!B1").equals(c.getCellFormula()));
}
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet("testSheet1");
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellFormula("IF(1=1,0,1)");
writeOutAndReadBack(wb1).close();
}
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s = wb1.createSheet("testSheet1");
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellValue(1);
c = r.createCell(1);
c.setCellValue(3);
HSSFCell formulaCell = r.createCell(3);
r = s.createRow(1);
c = r.createCell(0);
c.setCellValue(3);
c = r.createCell(1);
c.setCellValue(7);
formulaCell.setCellFormula("IF(A1=B1,AVERAGE(A1:B1),AVERAGE(A2:B2))");
writeOutAndReadBack(wb1).close();
}
}
@Test
void testSumIf() throws IOException {
String function ="SUMIF(A1:A5,\">4000\",B1:B5)";
try (HSSFWorkbook wb = openSample("sumifformula.xls")) {
HSSFSheet s = wb.getSheetAt(0);
HSSFRow r = s.getRow(0);
HSSFCell c = r.getCell(2);
assertEquals(function, c.getCellFormula());
}
try (HSSFWorkbook wb = new HSSFWorkbook()) {
HSSFSheet s = wb.createSheet();
HSSFRow r = s.createRow(0);
HSSFCell c = r.createCell(0);
c.setCellValue(1000);
c = r.createCell(1);
c.setCellValue(1);
r = s.createRow(1);
c = r.createCell(0);
c.setCellValue(2000);
c = r.createCell(1);
c.setCellValue(2);
r = s.createRow(2);
c = r.createCell(0);
c.setCellValue(3000);
c = r.createCell(1);
c.setCellValue(3);
r = s.createRow(3);
c = r.createCell(0);
c.setCellValue(4000);
c = r.createCell(1);
c.setCellValue(4);
r = s.createRow(4);
c = r.createCell(0);
c.setCellValue(5000);
c = r.createCell(1);
c.setCellValue(5);
r = s.getRow(0);
c = r.createCell(2);
c.setCellFormula(function);
writeOutAndReadBack(wb).close();
}
}
@Test
void testSquareMacro() throws IOException {
try (HSSFWorkbook w = openSample("SquareMacro.xls")) {
HSSFSheet s0 = w.getSheetAt(0);
HSSFRow[] r = {s0.getRow(0), s0.getRow(1)};
HSSFCell a1 = r[0].getCell(0);
assertEquals("square(1)", a1.getCellFormula());
assertEquals(1d, a1.getNumericCellValue(), 1e-9);
HSSFCell a2 = r[1].getCell(0);
assertEquals("square(2)", a2.getCellFormula());
assertEquals(4d, a2.getNumericCellValue(), 1e-9);
HSSFCell b1 = r[0].getCell(1);
assertEquals("IF(TRUE,square(1))", b1.getCellFormula());
assertEquals(1d, b1.getNumericCellValue(), 1e-9);
HSSFCell b2 = r[1].getCell(1);
assertEquals("IF(TRUE,square(2))", b2.getCellFormula());
assertEquals(4d, b2.getNumericCellValue(), 1e-9);
HSSFCell c1 = r[0].getCell(2);
assertEquals("square(square(1))", c1.getCellFormula());
assertEquals(1d, c1.getNumericCellValue(), 1e-9);
HSSFCell c2 = r[1].getCell(2);
assertEquals("square(square(2))", c2.getCellFormula());
assertEquals(16d, c2.getNumericCellValue(), 1e-9);
HSSFCell d1 = r[0].getCell(3);
assertEquals("square(one())", d1.getCellFormula());
assertEquals(1d, d1.getNumericCellValue(), 1e-9);
HSSFCell d2 = r[1].getCell(3);
assertEquals("square(two())", d2.getCellFormula());
assertEquals(4d, d2.getNumericCellValue(), 1e-9);
}
}
@Test
void testStringFormulaRead() throws IOException {
try (HSSFWorkbook w = openSample("StringFormulas.xls")) {
HSSFCell c = w.getSheetAt(0).getRow(0).getCell(0);
assertEquals("XYZ", c.getRichStringCellValue().getString(), "String Cell value");
}
}
/** test for bug 34021*/
@Test
void testComplexSheetRefs() throws IOException {
try (HSSFWorkbook wb1 = new HSSFWorkbook()) {
HSSFSheet s1 = wb1.createSheet("Sheet a.1");
HSSFSheet s2 = wb1.createSheet("Sheet.A");
s2.createRow(1).createCell(2).setCellFormula("'Sheet a.1'!A1");
s1.createRow(1).createCell(2).setCellFormula("'Sheet.A'!A1");
try (HSSFWorkbook wb2 = writeOutAndReadBack(wb1)) {
s1 = wb2.getSheet("Sheet a.1");
assertNotNull(s1);
assertEquals("Sheet.A!A1", s1.getRow(1).getCell(2).getCellFormula());
s2 = wb2.getSheet("Sheet.A");
assertNotNull(s2);
assertEquals("'Sheet a.1'!A1", s2.getRow(1).getCell(2).getCellFormula());
}
}
}
/** Unknown Ptg 3C*/
@Test
void test27272_1() throws IOException {
try (HSSFWorkbook wb = openSample("27272_1.xls")) {
wb.getSheetAt(0);
assertEquals("Compliance!#REF!", wb.getNameAt(0).getRefersToFormula(), "Reference for named range");
writeOutAndReadBack(wb).close();
}
}
/** Unknown Ptg 3D*/
@Test
void test27272_2() throws IOException {
try (HSSFWorkbook wb = openSample("27272_2.xls")) {
assertEquals("LOAD.POD_HISTORIES!#REF!", wb.getNameAt(0).getRefersToFormula(), "Reference for named range");
writeOutAndReadBack(wb).close();
}
}
/** MissingArgPtg */
@Test
void testMissingArgPtg() throws IOException {
try (HSSFWorkbook wb = new HSSFWorkbook()) {
HSSFCell cell = wb.createSheet("Sheet1").createRow(4).createCell(0);
assertDoesNotThrow(() -> cell.setCellFormula("IF(A1=\"A\",1,)"));
}
}
@Test
void testSharedFormula() throws IOException {
try (HSSFWorkbook wb = openSample("SharedFormulaTest.xls")) {
assertEquals("A$1*2", wb.getSheetAt(0).getRow(1).getCell(1).toString());
assertEquals("$A11*2", wb.getSheetAt(0).getRow(11).getCell(1).toString());
assertEquals("DZ2*2", wb.getSheetAt(0).getRow(1).getCell(128).toString());
assertEquals("B32770*2", wb.getSheetAt(0).getRow(32768).getCell(1).toString());
}
}
/**
* Test creation / evaluation of formulas with sheet-level names
*/
@Test
void testSheetLevelFormulas() throws IOException {
try (HSSFWorkbook wb = new HSSFWorkbook()) {
HSSFSheet sh1 = wb.createSheet("Sheet1");
HSSFName nm1 = wb.createName();
nm1.setNameName("sales_1");
nm1.setSheetIndex(0);
nm1.setRefersToFormula("Sheet1!$A$1");
HSSFRow row = sh1.createRow(0);
row.createCell(0).setCellValue(3);
row.createCell(1).setCellFormula("sales_1");
row.createCell(2).setCellFormula("sales_1*2");
HSSFSheet sh2 = wb.createSheet("Sheet2");
HSSFName nm2 = wb.createName();
nm2.setNameName("sales_1");
nm2.setSheetIndex(1);
nm2.setRefersToFormula("Sheet2!$A$1");
row = sh2.createRow(0);
row.createCell(0).setCellValue(5);
row.createCell(1).setCellFormula("sales_1");
row.createCell(2).setCellFormula("sales_1*3");
//check that NamePtg refers to the correct NameRecord
Ptg[] ptgs1 = HSSFFormulaParser.parse("sales_1", wb, FormulaType.CELL, 0);
NamePtg nPtg1 = (NamePtg) ptgs1[0];
assertSame(nm1, wb.getNameAt(nPtg1.getIndex()));
Ptg[] ptgs2 = HSSFFormulaParser.parse("sales_1", wb, FormulaType.CELL, 1);
NamePtg nPtg2 = (NamePtg) ptgs2[0];
assertSame(nm2, wb.getNameAt(nPtg2.getIndex()));
//check that the formula evaluator returns the correct result
HSSFFormulaEvaluator evaluator = new HSSFFormulaEvaluator(wb);
assertEquals(3.0, evaluator.evaluate(sh1.getRow(0).getCell(1)).getNumberValue(), 0.0);
assertEquals(6.0, evaluator.evaluate(sh1.getRow(0).getCell(2)).getNumberValue(), 0.0);
assertEquals(5.0, evaluator.evaluate(sh2.getRow(0).getCell(1)).getNumberValue(), 0.0);
assertEquals(15.0, evaluator.evaluate(sh2.getRow(0).getCell(2)).getNumberValue(), 0.0);
}
}
/**
* Verify that FormulaParser handles defined names beginning with underscores,
* see Bug #49640
*/
@Test
void testFormulasWithUnderscore() throws IOException{
try (HSSFWorkbook wb = new HSSFWorkbook()) {
Name nm1 = wb.createName();
nm1.setNameName("_score1");
nm1.setRefersToFormula("A1");
Name nm2 = wb.createName();
nm2.setNameName("_score2");
nm2.setRefersToFormula("A2");
Sheet sheet = wb.createSheet();
Cell cell = sheet.createRow(0).createCell(2);
cell.setCellFormula("_score1*SUM(_score1+_score2)");
assertEquals("_score1*SUM(_score1+_score2)", cell.getCellFormula());
}
}
}