diff --git a/src/documentation/content/xdocs/status.xml b/src/documentation/content/xdocs/status.xml index b1a1825ad..538273a20 100644 --- a/src/documentation/content/xdocs/status.xml +++ b/src/documentation/content/xdocs/status.xml @@ -34,6 +34,7 @@ + 51780 - support replacement of content types in OPC packages 52784 - replace ISO control characters with question marks in SXSSF to be consistent with XSSF 52057 - updated formula test framework to be aware of recently added Functions 52574 - support setting header / footer page margins in HSSF diff --git a/src/ooxml/java/org/apache/poi/openxml4j/opc/OPCPackage.java b/src/ooxml/java/org/apache/poi/openxml4j/opc/OPCPackage.java index f384cc650..7a0da815b 100644 --- a/src/ooxml/java/org/apache/poi/openxml4j/opc/OPCPackage.java +++ b/src/ooxml/java/org/apache/poi/openxml4j/opc/OPCPackage.java @@ -1439,4 +1439,50 @@ public abstract class OPCPackage implements RelationshipSource, Closeable { */ protected abstract PackagePart[] getPartsImpl() throws InvalidFormatException; + + /** + * Replace a content type in this package. + * + *

+ * A typical scneario to call this method is to rename a template file to the main format, e.g. + * ".dotx" to ".docx" + * ".dotm" to ".docm" + * ".xltx" to ".xlsx" + * ".xltm" to ".xlsm" + * ".potx" to ".pptx" + * ".potm" to ".pptm" + *

+ * For example, a code converting a .xlsm macro workbook to .xlsx would look as follows: + *

+ *


+     *
+     *     OPCPackage pkg = OPCPackage.open(new FileInputStream("macro-workbook.xlsm"));
+     *     pkg.replaceContentType(
+     *         "application/vnd.ms-excel.sheet.macroEnabled.main+xml",
+     *         "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml");
+     *
+     *     FileOutputStream out = new FileOutputStream("workbook.xlsx");
+     *     pkg.save(out);
+     *     out.close();
+     *
+     *    
+ *

+ * + * @param oldContentType the content type to be replaced + * @param newContentType the replacement + * @return whether replacement was succesfull + * @since POI-3.8 + */ + public boolean replaceContentType(String oldContentType, String newContentType){ + boolean success = false; + ArrayList list = getPartsByContentType(oldContentType); + for (PackagePart packagePart : list) { + if (packagePart.getContentType().equals(oldContentType)) { + PackagePartName partName = packagePart.getPartName(); + contentTypeManager.addContentType(partName, newContentType); + success = true; + } + } + return success; + } } diff --git a/src/ooxml/testcases/org/apache/poi/openxml4j/opc/TestPackage.java b/src/ooxml/testcases/org/apache/poi/openxml4j/opc/TestPackage.java index 35e87ec00..b6bc97c18 100644 --- a/src/ooxml/testcases/org/apache/poi/openxml4j/opc/TestPackage.java +++ b/src/ooxml/testcases/org/apache/poi/openxml4j/opc/TestPackage.java @@ -17,12 +17,7 @@ package org.apache.poi.openxml4j.opc; -import java.io.ByteArrayOutputStream; -import java.io.File; -import java.io.FileInputStream; -import java.io.FileOutputStream; -import java.io.IOException; -import java.io.OutputStream; +import java.io.*; import java.lang.reflect.Field; import java.net.URI; import java.util.*; @@ -528,4 +523,23 @@ public final class TestPackage extends TestCase { assertTrue(selected.containsKey("/word/theme/theme1.xml")); assertTrue(selected.containsKey("/word/webSettings.xml")); } + + public void testReplaceContentType() throws Exception { + InputStream is = OpenXML4JTestDataSamples.openSampleStream("sample.xlsx"); + OPCPackage p = OPCPackage.open(is); + + ContentTypeManager mgr = getContentTypeManager(p); + + assertTrue(mgr.isContentTypeRegister("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml")); + assertFalse(mgr.isContentTypeRegister("application/vnd.ms-excel.sheet.macroEnabled.main+xml")); + + assertTrue( + p.replaceContentType( + "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml", + "application/vnd.ms-excel.sheet.macroEnabled.main+xml") + ); + + assertFalse(mgr.isContentTypeRegister("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml")); + assertTrue(mgr.isContentTypeRegister("application/vnd.ms-excel.sheet.macroEnabled.main+xml")); + } }