mirror of https://github.com/apache/poi.git
Bugzilla 46973 - fixed defined names to behave better when refersToFormula is unset
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@762479 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
7873f0d8cf
commit
b302384e4d
|
@ -37,6 +37,7 @@
|
||||||
|
|
||||||
<!-- Don't forget to update status.xml too! -->
|
<!-- Don't forget to update status.xml too! -->
|
||||||
<release version="3.5-beta6" date="2009-??-??">
|
<release version="3.5-beta6" date="2009-??-??">
|
||||||
|
<action dev="POI-DEVELOPERS" type="fix">46973 - Fixed defined names to behave better when refersToFormula is unset</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action>
|
<action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action>
|
<action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46948 - Fixed evaluation of range operator to allow for area-ref operands</action>
|
<action dev="POI-DEVELOPERS" type="fix">46948 - Fixed evaluation of range operator to allow for area-ref operands</action>
|
||||||
|
|
|
@ -34,6 +34,7 @@
|
||||||
<!-- Don't forget to update changes.xml too! -->
|
<!-- Don't forget to update changes.xml too! -->
|
||||||
<changes>
|
<changes>
|
||||||
<release version="3.5-beta6" date="2009-??-??">
|
<release version="3.5-beta6" date="2009-??-??">
|
||||||
|
<action dev="POI-DEVELOPERS" type="fix">46973 - Fixed defined names to behave better when refersToFormula is unset</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action>
|
<action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action>
|
<action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46948 - Fixed evaluation of range operator to allow for area-ref operands</action>
|
<action dev="POI-DEVELOPERS" type="fix">46948 - Fixed evaluation of range operator to allow for area-ref operands</action>
|
||||||
|
|
|
@ -321,4 +321,32 @@ public abstract class Ptg implements Cloneable {
|
||||||
* @return <code>false</code> if this token is classified as 'reference', 'value', or 'array'
|
* @return <code>false</code> if this token is classified as 'reference', 'value', or 'array'
|
||||||
*/
|
*/
|
||||||
public abstract boolean isBaseToken();
|
public abstract boolean isBaseToken();
|
||||||
|
|
||||||
|
public static boolean doesFormulaReferToDeletedCell(Ptg[] ptgs) {
|
||||||
|
for (int i = 0; i < ptgs.length; i++) {
|
||||||
|
if (isDeletedCellRef(ptgs[i])) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
private static boolean isDeletedCellRef(Ptg ptg) {
|
||||||
|
if (ptg == ErrPtg.REF_INVALID) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (ptg instanceof DeletedArea3DPtg) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (ptg instanceof DeletedRef3DPtg) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (ptg instanceof AreaErrPtg) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (ptg instanceof RefErrorPtg) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,8 +21,8 @@ import org.apache.poi.hssf.model.HSSFFormulaParser;
|
||||||
import org.apache.poi.hssf.model.Workbook;
|
import org.apache.poi.hssf.model.Workbook;
|
||||||
import org.apache.poi.hssf.record.NameRecord;
|
import org.apache.poi.hssf.record.NameRecord;
|
||||||
import org.apache.poi.hssf.record.formula.Ptg;
|
import org.apache.poi.hssf.record.formula.Ptg;
|
||||||
import org.apache.poi.ss.usermodel.Name;
|
|
||||||
import org.apache.poi.ss.formula.FormulaType;
|
import org.apache.poi.ss.formula.FormulaType;
|
||||||
|
import org.apache.poi.ss.usermodel.Name;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* High Level Representation of a 'defined name' which could be a 'built-in' name,
|
* High Level Representation of a 'defined name' which could be a 'built-in' name,
|
||||||
|
@ -160,46 +160,26 @@ public final class HSSFName implements Name {
|
||||||
setRefersToFormula(ref);
|
setRefersToFormula(ref);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets the formula that the name is defined to refer to. The following are representative examples:
|
|
||||||
*
|
|
||||||
* <ul>
|
|
||||||
* <li><code>'My Sheet'!$A$3</code></li>
|
|
||||||
* <li><code>8.3</code></li>
|
|
||||||
* <li><code>HR!$A$1:$Z$345</code></li>
|
|
||||||
* <li><code>SUM(Sheet1!A1,Sheet2!B2)</li>
|
|
||||||
* <li><code>-PMT(Interest_Rate/12,Number_of_Payments,Loan_Amount)</li>
|
|
||||||
* </ul>
|
|
||||||
*
|
|
||||||
* @param formulaText the reference for this name
|
|
||||||
* @throws IllegalArgumentException if the specified reference is unparsable
|
|
||||||
*/
|
|
||||||
public void setRefersToFormula(String formulaText) {
|
public void setRefersToFormula(String formulaText) {
|
||||||
Ptg[] ptgs = HSSFFormulaParser.parse(formulaText, _book, FormulaType.NAMEDRANGE, getSheetIndex());
|
Ptg[] ptgs = HSSFFormulaParser.parse(formulaText, _book, FormulaType.NAMEDRANGE, getSheetIndex());
|
||||||
_definedNameRec.setNameDefinition(ptgs);
|
_definedNameRec.setNameDefinition(ptgs);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the formula that the name is defined to refer to. The following are representative examples:
|
|
||||||
*
|
|
||||||
* @return the reference for this name
|
|
||||||
* @see #setRefersToFormula(String)
|
|
||||||
*/
|
|
||||||
public String getRefersToFormula() {
|
public String getRefersToFormula() {
|
||||||
if (_definedNameRec.isFunctionName()) {
|
if (_definedNameRec.isFunctionName()) {
|
||||||
throw new IllegalStateException("Only applicable to named ranges");
|
throw new IllegalStateException("Only applicable to named ranges");
|
||||||
}
|
}
|
||||||
return HSSFFormulaParser.toFormulaString(_book, _definedNameRec.getNameDefinition());
|
Ptg[] ptgs = _definedNameRec.getNameDefinition();
|
||||||
|
if (ptgs.length < 1) {
|
||||||
|
// 'refersToFormula' has not been set yet
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return HSSFFormulaParser.toFormulaString(_book, ptgs);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests if this name points to a cell that no longer exists
|
|
||||||
*
|
|
||||||
* @return true if the name refers to a deleted cell, false otherwise
|
|
||||||
*/
|
|
||||||
public boolean isDeleted(){
|
public boolean isDeleted(){
|
||||||
String formulaText = getRefersToFormula();
|
Ptg[] ptgs = _definedNameRec.getNameDefinition();
|
||||||
return formulaText.indexOf("#REF!") != -1;
|
return Ptg.doesFormulaReferToDeletedCell(ptgs);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -118,7 +118,7 @@ public interface Name {
|
||||||
/**
|
/**
|
||||||
* Returns the formula that the name is defined to refer to.
|
* Returns the formula that the name is defined to refer to.
|
||||||
*
|
*
|
||||||
* @return the reference for this name
|
* @return the reference for this name, <code>null</code> if it has not been set yet. Never empty string
|
||||||
* @see #setRefersToFormula(String)
|
* @see #setRefersToFormula(String)
|
||||||
*/
|
*/
|
||||||
String getRefersToFormula();
|
String getRefersToFormula();
|
||||||
|
@ -134,10 +134,10 @@ public interface Name {
|
||||||
* <li><code>-PMT(Interest_Rate/12,Number_of_Payments,Loan_Amount)</li>
|
* <li><code>-PMT(Interest_Rate/12,Number_of_Payments,Loan_Amount)</li>
|
||||||
* </ul>
|
* </ul>
|
||||||
*
|
*
|
||||||
* @param ref the reference for this name
|
* @param formulaText the reference for this name
|
||||||
* @throws IllegalArgumentException if the specified reference is unparsable
|
* @throws IllegalArgumentException if the specified formulaText is unparsable
|
||||||
*/
|
*/
|
||||||
void setRefersToFormula(String ref);
|
void setRefersToFormula(String formulaText);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if this name is a function name
|
* Checks if this name is a function name
|
||||||
|
@ -149,7 +149,7 @@ public interface Name {
|
||||||
/**
|
/**
|
||||||
* Checks if this name points to a cell that no longer exists
|
* Checks if this name points to a cell that no longer exists
|
||||||
*
|
*
|
||||||
* @return true if the name refers to a deleted cell, false otherwise
|
* @return <code>true</code> if the name refers to a deleted cell, <code>false</code> otherwise
|
||||||
*/
|
*/
|
||||||
boolean isDeleted();
|
boolean isDeleted();
|
||||||
|
|
||||||
|
|
|
@ -181,25 +181,18 @@ public final class XSSFName implements Name {
|
||||||
ctName.setName(name);
|
ctName.setName(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the reference of this named range, such as Sales!C20:C30.
|
|
||||||
*
|
|
||||||
* @return the reference of this named range
|
|
||||||
*/
|
|
||||||
public String getRefersToFormula() {
|
public String getRefersToFormula() {
|
||||||
return ctName.getStringValue();
|
String result = ctName.getStringValue();
|
||||||
|
if (result == null || result.length() < 1) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets the reference of this named range, such as Sales!C20:C30.
|
|
||||||
*
|
|
||||||
* @param formulaText the reference to set
|
|
||||||
* @throws IllegalArgumentException if the specified reference is unparsable
|
|
||||||
*/
|
|
||||||
public void setRefersToFormula(String formulaText) {
|
public void setRefersToFormula(String formulaText) {
|
||||||
XSSFEvaluationWorkbook fpb = XSSFEvaluationWorkbook.create(workbook);
|
XSSFEvaluationWorkbook fpb = XSSFEvaluationWorkbook.create(workbook);
|
||||||
try {
|
try {
|
||||||
Ptg[] ptgs = FormulaParser.parse(formulaText, fpb, FormulaType.NAMEDRANGE, getSheetIndex());
|
FormulaParser.parse(formulaText, fpb, FormulaType.NAMEDRANGE, getSheetIndex());
|
||||||
} catch (RuntimeException e) {
|
} catch (RuntimeException e) {
|
||||||
if (e.getClass().getName().startsWith(FormulaParser.class.getName())) {
|
if (e.getClass().getName().startsWith(FormulaParser.class.getName())) {
|
||||||
throw new IllegalArgumentException("Unparsable formula '" + formulaText + "'", e);
|
throw new IllegalArgumentException("Unparsable formula '" + formulaText + "'", e);
|
||||||
|
@ -209,14 +202,14 @@ public final class XSSFName implements Name {
|
||||||
ctName.setStringValue(formulaText);
|
ctName.setStringValue(formulaText);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests if this name points to a cell that no longer exists
|
|
||||||
*
|
|
||||||
* @return true if the name refers to a deleted cell, false otherwise
|
|
||||||
*/
|
|
||||||
public boolean isDeleted(){
|
public boolean isDeleted(){
|
||||||
String ref = getRefersToFormula();
|
String formulaText = getRefersToFormula();
|
||||||
return ref != null && ref.indexOf("#REF!") != -1;
|
if (formulaText == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
XSSFEvaluationWorkbook fpb = XSSFEvaluationWorkbook.create(workbook);
|
||||||
|
Ptg[] ptgs = FormulaParser.parse(formulaText, fpb, FormulaType.NAMEDRANGE, getSheetIndex());
|
||||||
|
return Ptg.doesFormulaReferToDeletedCell(ptgs);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -17,13 +17,13 @@
|
||||||
|
|
||||||
package org.apache.poi.ss.usermodel;
|
package org.apache.poi.ss.usermodel;
|
||||||
|
|
||||||
|
import junit.framework.AssertionFailedError;
|
||||||
import junit.framework.TestCase;
|
import junit.framework.TestCase;
|
||||||
|
|
||||||
|
import org.apache.poi.hssf.usermodel.HSSFName;
|
||||||
import org.apache.poi.ss.ITestDataProvider;
|
import org.apache.poi.ss.ITestDataProvider;
|
||||||
import org.apache.poi.ss.formula.FormulaParser;
|
|
||||||
import org.apache.poi.ss.formula.FormulaType;
|
|
||||||
import org.apache.poi.ss.util.CellReference;
|
|
||||||
import org.apache.poi.ss.util.AreaReference;
|
import org.apache.poi.ss.util.AreaReference;
|
||||||
import org.apache.poi.hssf.record.formula.Ptg;
|
import org.apache.poi.ss.util.CellReference;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests of implementations of {@link org.apache.poi.ss.usermodel.Name}.
|
* Tests of implementations of {@link org.apache.poi.ss.usermodel.Name}.
|
||||||
|
@ -395,7 +395,7 @@ public abstract class BaseTestNamedRange extends TestCase {
|
||||||
* Test that multiple named ranges can be added written and read
|
* Test that multiple named ranges can be added written and read
|
||||||
*/
|
*/
|
||||||
public void testMultipleNamedWrite() {
|
public void testMultipleNamedWrite() {
|
||||||
Workbook wb = getTestDataProvider().createWorkbook();
|
Workbook wb = getTestDataProvider().createWorkbook();
|
||||||
|
|
||||||
|
|
||||||
wb.createSheet("testSheet1");
|
wb.createSheet("testSheet1");
|
||||||
|
@ -498,4 +498,50 @@ public abstract class BaseTestNamedRange extends TestCase {
|
||||||
assertEquals("Contents of cell retrieved by its named reference", contents, cvalue);
|
assertEquals("Contents of cell retrieved by its named reference", contents, cvalue);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bugzilla attachment 23444 (from bug 46973) has a NAME record with the following encoding:
|
||||||
|
* <pre>
|
||||||
|
* 00000000 | 18 00 17 00 00 00 00 08 00 00 00 00 00 00 00 00 | ................
|
||||||
|
* 00000010 | 00 00 00 55 50 53 53 74 61 74 65 | ...UPSState
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
|
* This caused trouble for anything that requires {@link HSSFName#getRefersToFormula()}
|
||||||
|
* It is easy enough to re-create the the same data (by not setting the formula). Excel
|
||||||
|
* seems to gracefully remove this uninitialized name record. It would be nice if POI
|
||||||
|
* could do the same, but that would involve adjusting subsequent name indexes across
|
||||||
|
* all formulas. <p/>
|
||||||
|
*
|
||||||
|
* For the moment, POI has been made to behave more sensibly with uninitialised name
|
||||||
|
* records.
|
||||||
|
*/
|
||||||
|
public final void testUninitialisedNameGetRefersToFormula_bug46973() {
|
||||||
|
Workbook wb = getTestDataProvider().createWorkbook();
|
||||||
|
Name n = wb.createName();
|
||||||
|
n.setNameName("UPSState");
|
||||||
|
String formula;
|
||||||
|
try {
|
||||||
|
formula = n.getRefersToFormula();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
if (e.getMessage().equals("ptgs must not be null")) {
|
||||||
|
throw new AssertionFailedError("Identified bug 46973");
|
||||||
|
}
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
assertNull(formula);
|
||||||
|
assertFalse(n.isDeleted()); // according to exact definition of isDeleted()
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testDeletedCell() {
|
||||||
|
Workbook wb = getTestDataProvider().createWorkbook();
|
||||||
|
Name n = wb.createName();
|
||||||
|
n.setNameName("MyName");
|
||||||
|
// contrived example to expose bug:
|
||||||
|
n.setRefersToFormula("if(A1,\"#REF!\", \"\")");
|
||||||
|
|
||||||
|
if (n.isDeleted()) {
|
||||||
|
throw new AssertionFailedError("Identified bug in recoginising formulas referring to deleted cells");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue