diff --git a/src/ooxml/java/org/apache/poi/openxml4j/opc/PackageRelationshipTypes.java b/src/ooxml/java/org/apache/poi/openxml4j/opc/PackageRelationshipTypes.java index 9344a64abf..134e6a9fa3 100644 --- a/src/ooxml/java/org/apache/poi/openxml4j/opc/PackageRelationshipTypes.java +++ b/src/ooxml/java/org/apache/poi/openxml4j/opc/PackageRelationshipTypes.java @@ -99,4 +99,9 @@ public interface PackageRelationshipTypes { * Style type. */ String STYLE_PART = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/styles"; + + /** + * External Link to another Document + */ + String EXTERNAL_LINK_PATH = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/externalLinkPath"; } diff --git a/src/ooxml/java/org/apache/poi/xssf/model/ExternalLinksTable.java b/src/ooxml/java/org/apache/poi/xssf/model/ExternalLinksTable.java index 22a0de9c45..9d66b6fa48 100644 --- a/src/ooxml/java/org/apache/poi/xssf/model/ExternalLinksTable.java +++ b/src/ooxml/java/org/apache/poi/xssf/model/ExternalLinksTable.java @@ -25,6 +25,8 @@ import java.util.List; import org.apache.poi.POIXMLDocumentPart; import org.apache.poi.openxml4j.opc.PackagePart; import org.apache.poi.openxml4j.opc.PackageRelationship; +import org.apache.poi.openxml4j.opc.PackageRelationshipTypes; +import org.apache.poi.openxml4j.opc.TargetMode; import org.apache.poi.ss.usermodel.Name; import org.apache.xmlbeans.XmlException; import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTExternalDefinedName; @@ -78,6 +80,38 @@ public class ExternalLinksTable extends POIXMLDocumentPart { public CTExternalLink getCTExternalLink(){ return link; } + + /** + * Returns the last recorded name of the file that this + * is linked to + */ + public String getLinkedFileName() { + String rId = link.getExternalBook().getId(); + PackageRelationship rel = getPackagePart().getRelationship(rId); + if (rel != null && rel.getTargetMode() == TargetMode.EXTERNAL) { + return rel.getTargetURI().toString(); + } else { + return null; + } + } + /** + * Updates the last recorded name for the file that this links to + */ + public void setLinkedFileName(String target) { + String rId = link.getExternalBook().getId(); + + if (rId == null || rId.isEmpty()) { + // We're a new External Link Table, so nothing to remove + } else { + // Relationships can't be changed, so remove the old one + getPackagePart().removeRelationship(rId); + } + + // Have a new one added + PackageRelationship newRel = getPackagePart().addExternalRelationship( + target, PackageRelationshipTypes.EXTERNAL_LINK_PATH); + link.getExternalBook().setId(newRel.getId()); + } @SuppressWarnings("deprecation") public List getSheetNames() { diff --git a/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFWorkbook.java b/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFWorkbook.java index f1025bce74..84e118233c 100644 --- a/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFWorkbook.java +++ b/src/ooxml/java/org/apache/poi/xssf/usermodel/XSSFWorkbook.java @@ -79,6 +79,7 @@ import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTCalcPr; import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTDefinedName; import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTDefinedNames; import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTDialogsheet; +import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTExternalReference; import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTSheet; import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTSheets; import org.openxmlformats.schemas.spreadsheetml.x2006.main.CTWorkbook; @@ -157,9 +158,9 @@ public class XSSFWorkbook extends POIXMLDocument implements Workbook, Iterable externalLinks; /** * A collection of custom XML mappings @@ -284,16 +285,19 @@ public class XSSFWorkbook extends POIXMLDocument implements Workbook, Iterable shIdMap = new HashMap(); + Map elIdMap = new HashMap(); for(POIXMLDocumentPart p : getRelations()){ if(p instanceof SharedStringsTable) sharedStringSource = (SharedStringsTable)p; else if(p instanceof StylesTable) stylesSource = (StylesTable)p; else if(p instanceof ThemesTable) theme = (ThemesTable)p; else if(p instanceof CalculationChain) calcChain = (CalculationChain)p; - else if(p instanceof ExternalLinksTable) externalLinks = (ExternalLinksTable)p; else if(p instanceof MapInfo) mapInfo = (MapInfo)p; else if (p instanceof XSSFSheet) { shIdMap.put(p.getPackageRelationship().getId(), (XSSFSheet)p); } + else if (p instanceof ExternalLinksTable) { + elIdMap.put(p.getPackageRelationship().getId(), (ExternalLinksTable)p); + } } if (stylesSource == null) { @@ -307,7 +311,8 @@ public class XSSFWorkbook extends POIXMLDocument implements Workbook, Iterable(shIdMap.size()); for (CTSheet ctSheet : this.workbook.getSheets().getSheetArray()) { XSSFSheet sh = shIdMap.get(ctSheet.getId()); @@ -319,6 +324,20 @@ public class XSSFWorkbook extends POIXMLDocument implements Workbook, Iterable(elIdMap.size()); + if (this.workbook.isSetExternalReferences()) { + for (CTExternalReference er : this.workbook.getExternalReferences().getExternalReferenceArray()) { + ExternalLinksTable el = elIdMap.get(er.getId()); + if(el == null) { + logger.log(POILogger.WARN, "ExternalLinksTable with r:id " + er.getId()+ " was defined, but didn't exist in package, skipping"); + continue; + } + externalLinks.add(el); + } + } // Process the named ranges reprocessNamedRanges(); @@ -1611,16 +1630,20 @@ public class XSSFWorkbook extends POIXMLDocument implements Workbook, IterableThe external links table specifies details of named ranges etc * that are referenced from other workbooks, along with the last seen * values of what they point to.

* - * @return the ExternalLinksTable object or null if not defined + *

Note that Excel uses index 0 for the current workbook, so the first + * External Links in a formula would be '[1]Foo' which corresponds to + * entry 0 in this list.

+ + * @return the ExternalLinksTable list, which may be empty */ @Internal - public ExternalLinksTable getExternalLinksTable() { + public List getExternalLinksTable() { return externalLinks; } diff --git a/src/ooxml/testcases/org/apache/poi/xssf/model/TestExternalLinksTable.java b/src/ooxml/testcases/org/apache/poi/xssf/model/TestExternalLinksTable.java index 612924ad5f..1fbf86a335 100644 --- a/src/ooxml/testcases/org/apache/poi/xssf/model/TestExternalLinksTable.java +++ b/src/ooxml/testcases/org/apache/poi/xssf/model/TestExternalLinksTable.java @@ -29,7 +29,8 @@ public final class TestExternalLinksTable { @Test public void none() { XSSFWorkbook wb = XSSFTestDataSamples.openSampleWorkbook("SampleSS.xlsx"); - assertEquals(null, wb.getExternalLinksTable()); + assertNotNull(wb.getExternalLinksTable()); + assertEquals(0, wb.getExternalLinksTable().size()); } @Test @@ -37,8 +38,10 @@ public final class TestExternalLinksTable { XSSFWorkbook wb = XSSFTestDataSamples.openSampleWorkbook("ref-56737.xlsx"); assertNotNull(wb.getExternalLinksTable()); Name name = null; - - ExternalLinksTable links = wb.getExternalLinksTable(); + + assertEquals(1, wb.getExternalLinksTable().size()); + + ExternalLinksTable links = wb.getExternalLinksTable().get(0); assertEquals(3, links.getSheetNames().size()); assertEquals(2, links.getDefinedNames().size()); @@ -57,17 +60,20 @@ public final class TestExternalLinksTable { assertEquals(1, name.getSheetIndex()); assertEquals("Defines", name.getSheetName()); assertEquals("'Defines'!$A$1", name.getRefersToFormula()); + + assertEquals("56737.xlsx", links.getLinkedFileName()); } @Test public void basicReadWriteRead() { XSSFWorkbook wb = XSSFTestDataSamples.openSampleWorkbook("ref-56737.xlsx"); - Name name = wb.getExternalLinksTable().getDefinedNames().get(1); + Name name = wb.getExternalLinksTable().get(0).getDefinedNames().get(1); name.setNameName("Testing"); name.setRefersToFormula("$A$1"); wb = XSSFTestDataSamples.writeOutAndReadBack(wb); - ExternalLinksTable links = wb.getExternalLinksTable(); + assertEquals(1, wb.getExternalLinksTable().size()); + ExternalLinksTable links = wb.getExternalLinksTable().get(0); name = links.getDefinedNames().get(0); assertEquals("NR_Global_B2", name.getNameName()); @@ -86,6 +92,53 @@ public final class TestExternalLinksTable { public void readWithReferencesToTwoExternalBooks() { XSSFWorkbook wb = XSSFTestDataSamples.openSampleWorkbook("ref2-56737.xlsx"); - // TODO Fix so we can see both of them... + assertNotNull(wb.getExternalLinksTable()); + Name name = null; + + assertEquals(2, wb.getExternalLinksTable().size()); + + // Check the first one, links to 56737.xlsx + ExternalLinksTable links = wb.getExternalLinksTable().get(0); + assertEquals("56737.xlsx", links.getLinkedFileName()); + assertEquals(3, links.getSheetNames().size()); + assertEquals(2, links.getDefinedNames().size()); + + assertEquals("Uses", links.getSheetNames().get(0)); + assertEquals("Defines", links.getSheetNames().get(1)); + assertEquals("56737", links.getSheetNames().get(2)); + + name = links.getDefinedNames().get(0); + assertEquals("NR_Global_B2", name.getNameName()); + assertEquals(-1, name.getSheetIndex()); + assertEquals(null, name.getSheetName()); + assertEquals("'Defines'!$B$2", name.getRefersToFormula()); + + name = links.getDefinedNames().get(1); + assertEquals("NR_To_A1", name.getNameName()); + assertEquals(1, name.getSheetIndex()); + assertEquals("Defines", name.getSheetName()); + assertEquals("'Defines'!$A$1", name.getRefersToFormula()); + + + // Check the second one, links to 56737.xls, slightly differently + links = wb.getExternalLinksTable().get(1); + assertEquals("56737.xls", links.getLinkedFileName()); + assertEquals(2, links.getSheetNames().size()); + assertEquals(2, links.getDefinedNames().size()); + + assertEquals("Uses", links.getSheetNames().get(0)); + assertEquals("Defines", links.getSheetNames().get(1)); + + name = links.getDefinedNames().get(0); + assertEquals("NR_Global_B2", name.getNameName()); + assertEquals(-1, name.getSheetIndex()); + assertEquals(null, name.getSheetName()); + assertEquals("'Defines'!$B$2", name.getRefersToFormula()); + + name = links.getDefinedNames().get(1); + assertEquals("NR_To_A1", name.getNameName()); + assertEquals(1, name.getSheetIndex()); + assertEquals("Defines", name.getSheetName()); + assertEquals("'Defines'!$A$1", name.getRefersToFormula()); } }