mirror of https://github.com/apache/poi.git
Fix bug #45623 - Support stripping HSSF header and footer fields (eg page number) out of header and footer text if required
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@686046 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
5595a24adf
commit
bb270804b9
|
@ -37,7 +37,8 @@
|
||||||
|
|
||||||
<!-- Don't forget to update status.xml too! -->
|
<!-- Don't forget to update status.xml too! -->
|
||||||
<release version="3.1.1-alpha1" date="2008-??-??">
|
<release version="3.1.1-alpha1" date="2008-??-??">
|
||||||
<action dev="POI-DEVELOPERS" type="fix">45622 - Support stripping HWPF fields (eg macros) out of text, via Range.stripFields(text)</action>
|
<action dev="POI-DEVELOPERS" type="add">45623 - Support stripping HSSF header and footer fields (eg page number) out of header and footer text if required</action>
|
||||||
|
<action dev="POI-DEVELOPERS" type="add">45622 - Support stripping HWPF fields (eg macros) out of text, via Range.stripFields(text)</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">New HPSF based TextExtractor for document metadata, org.apache.poi.hpsf.extractor.HPSFPropertiesExtractor</action>
|
<action dev="POI-DEVELOPERS" type="add">New HPSF based TextExtractor for document metadata, org.apache.poi.hpsf.extractor.HPSFPropertiesExtractor</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">Properly update the array of Slide's text runs in HSLF when new text shapes are added</action>
|
<action dev="POI-DEVELOPERS" type="fix">Properly update the array of Slide's text runs in HSLF when new text shapes are added</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">45590 - Fix for Header/footer extraction for .ppt files saved in Office 2007</action>
|
<action dev="POI-DEVELOPERS" type="fix">45590 - Fix for Header/footer extraction for .ppt files saved in Office 2007</action>
|
||||||
|
|
|
@ -34,7 +34,8 @@
|
||||||
<!-- Don't forget to update changes.xml too! -->
|
<!-- Don't forget to update changes.xml too! -->
|
||||||
<changes>
|
<changes>
|
||||||
<release version="3.1.1-alpha1" date="2008-??-??">
|
<release version="3.1.1-alpha1" date="2008-??-??">
|
||||||
<action dev="POI-DEVELOPERS" type="fix">45622 - Support stripping HWPF fields (eg macros) out of text, via Range.stripFields(text)</action>
|
<action dev="POI-DEVELOPERS" type="add">45623 - Support stripping HSSF header and footer fields (eg page number) out of header and footer text if required</action>
|
||||||
|
<action dev="POI-DEVELOPERS" type="add">45622 - Support stripping HWPF fields (eg macros) out of text, via Range.stripFields(text)</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">New HPSF based TextExtractor for document metadata, org.apache.poi.hpsf.extractor.HPSFPropertiesExtractor</action>
|
<action dev="POI-DEVELOPERS" type="add">New HPSF based TextExtractor for document metadata, org.apache.poi.hpsf.extractor.HPSFPropertiesExtractor</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">Properly update the array of Slide's text runs in HSLF when new text shapes are added</action>
|
<action dev="POI-DEVELOPERS" type="fix">Properly update the array of Slide's text runs in HSLF when new text shapes are added</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">45590 - Fix for Header/footer extraction for .ppt files saved in Office 2007</action>
|
<action dev="POI-DEVELOPERS" type="fix">45590 - Fix for Header/footer extraction for .ppt files saved in Office 2007</action>
|
||||||
|
|
|
@ -16,6 +16,8 @@
|
||||||
==================================================================== */
|
==================================================================== */
|
||||||
package org.apache.poi.hssf.usermodel;
|
package org.apache.poi.hssf.usermodel;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Common class for {@link HSSFHeader} and
|
* Common class for {@link HSSFHeader} and
|
||||||
* {@link HSSFFooter}.
|
* {@link HSSFFooter}.
|
||||||
|
@ -25,6 +27,8 @@ public abstract class HeaderFooter {
|
||||||
protected String center;
|
protected String center;
|
||||||
protected String right;
|
protected String right;
|
||||||
|
|
||||||
|
private boolean stripFields = false;
|
||||||
|
|
||||||
protected HeaderFooter(String text) {
|
protected HeaderFooter(String text) {
|
||||||
while (text != null && text.length() > 1) {
|
while (text != null && text.length() > 1) {
|
||||||
int pos = text.length();
|
int pos = text.length();
|
||||||
|
@ -70,6 +74,8 @@ public abstract class HeaderFooter {
|
||||||
* @return The string representing the left side.
|
* @return The string representing the left side.
|
||||||
*/
|
*/
|
||||||
public String getLeft() {
|
public String getLeft() {
|
||||||
|
if(stripFields)
|
||||||
|
return stripFields(left);
|
||||||
return left;
|
return left;
|
||||||
}
|
}
|
||||||
public abstract void setLeft( String newLeft );
|
public abstract void setLeft( String newLeft );
|
||||||
|
@ -79,6 +85,8 @@ public abstract class HeaderFooter {
|
||||||
* @return The string representing the center.
|
* @return The string representing the center.
|
||||||
*/
|
*/
|
||||||
public String getCenter() {
|
public String getCenter() {
|
||||||
|
if(stripFields)
|
||||||
|
return stripFields(center);
|
||||||
return center;
|
return center;
|
||||||
}
|
}
|
||||||
public abstract void setCenter( String newCenter );
|
public abstract void setCenter( String newCenter );
|
||||||
|
@ -88,12 +96,15 @@ public abstract class HeaderFooter {
|
||||||
* @return The string representing the right side.
|
* @return The string representing the right side.
|
||||||
*/
|
*/
|
||||||
public String getRight() {
|
public String getRight() {
|
||||||
|
if(stripFields)
|
||||||
|
return stripFields(right);
|
||||||
return right;
|
return right;
|
||||||
}
|
}
|
||||||
public abstract void setRight( String newRight );
|
public abstract void setRight( String newRight );
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the string that represents the change in font size.
|
* Returns the string that represents the change in font size.
|
||||||
*
|
*
|
||||||
|
@ -122,9 +133,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for page number
|
* @return The special string for page number
|
||||||
*/
|
*/
|
||||||
public static String page()
|
public static String page() {
|
||||||
{
|
return PAGE_FIELD.sequence;
|
||||||
return "&P";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -132,9 +142,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for the number of pages
|
* @return The special string for the number of pages
|
||||||
*/
|
*/
|
||||||
public static String numPages()
|
public static String numPages() {
|
||||||
{
|
return NUM_PAGES_FIELD.sequence;
|
||||||
return "&N";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -142,9 +151,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for the date
|
* @return The special string for the date
|
||||||
*/
|
*/
|
||||||
public static String date()
|
public static String date() {
|
||||||
{
|
return DATE_FIELD.sequence;
|
||||||
return "&D";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -152,9 +160,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for the time
|
* @return The special string for the time
|
||||||
*/
|
*/
|
||||||
public static String time()
|
public static String time() {
|
||||||
{
|
return TIME_FIELD.sequence;
|
||||||
return "&T";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -162,9 +169,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for the file name
|
* @return The special string for the file name
|
||||||
*/
|
*/
|
||||||
public static String file()
|
public static String file() {
|
||||||
{
|
return FILE_FIELD.sequence;
|
||||||
return "&F";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -172,9 +178,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for tab name
|
* @return The special string for tab name
|
||||||
*/
|
*/
|
||||||
public static String tab()
|
public static String tab() {
|
||||||
{
|
return TAB_FIELD.sequence;
|
||||||
return "&A";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -182,9 +187,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for start underline
|
* @return The special string for start underline
|
||||||
*/
|
*/
|
||||||
public static String startUnderline()
|
public static String startUnderline() {
|
||||||
{
|
return UNDERLINE_FIELD.sequence;
|
||||||
return "&U";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -192,9 +196,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for end underline
|
* @return The special string for end underline
|
||||||
*/
|
*/
|
||||||
public static String endUnderline()
|
public static String endUnderline() {
|
||||||
{
|
return UNDERLINE_FIELD.sequence;
|
||||||
return "&U";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -202,9 +205,8 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for start double underline
|
* @return The special string for start double underline
|
||||||
*/
|
*/
|
||||||
public static String startDoubleUnderline()
|
public static String startDoubleUnderline() {
|
||||||
{
|
return DOUBLE_UNDERLINE_FIELD.sequence;
|
||||||
return "&E";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -212,8 +214,78 @@ public abstract class HeaderFooter {
|
||||||
*
|
*
|
||||||
* @return The special string for end double underline
|
* @return The special string for end double underline
|
||||||
*/
|
*/
|
||||||
public static String endDoubleUnderline()
|
public static String endDoubleUnderline() {
|
||||||
{
|
return DOUBLE_UNDERLINE_FIELD.sequence;
|
||||||
return "&E";
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes any fields (eg macros, page markers etc)
|
||||||
|
* from the string.
|
||||||
|
* Normally used to make some text suitable for showing
|
||||||
|
* to humans, and the resultant text should not normally
|
||||||
|
* be saved back into the document!
|
||||||
|
*/
|
||||||
|
public static String stripFields(String text) {
|
||||||
|
int pos;
|
||||||
|
|
||||||
|
// Firstly, do the easy ones which are static
|
||||||
|
for(int i=0; i<Field.ALL_FIELDS.size(); i++) {
|
||||||
|
String seq = ((Field)Field.ALL_FIELDS.get(i)).sequence;
|
||||||
|
while((pos = text.indexOf(seq)) > -1) {
|
||||||
|
text = text.substring(0, pos) +
|
||||||
|
text.substring(pos+seq.length());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now do the tricky, dynamic ones
|
||||||
|
text = text.replaceAll("\\&\\d+", "");
|
||||||
|
text = text.replaceAll("\\&\".*?,.*?\"", "");
|
||||||
|
|
||||||
|
// All done
|
||||||
|
return text;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Are fields currently being stripped from
|
||||||
|
* the text that this {@link HeaderStories} returns?
|
||||||
|
* Default is false, but can be changed
|
||||||
|
*/
|
||||||
|
public boolean areFieldsStripped() {
|
||||||
|
return stripFields;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Should fields (eg macros) be stripped from
|
||||||
|
* the text that this class returns?
|
||||||
|
* Default is not to strip.
|
||||||
|
* @param stripFields
|
||||||
|
*/
|
||||||
|
public void setAreFieldsStripped(boolean stripFields) {
|
||||||
|
this.stripFields = stripFields;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static final Field TAB_FIELD = new Field("&A");
|
||||||
|
public static final Field DATE_FIELD = new Field("&D");
|
||||||
|
public static final Field FILE_FIELD = new Field("&F");
|
||||||
|
public static final Field PAGE_FIELD = new Field("&P");
|
||||||
|
public static final Field TIME_FIELD = new Field("&T");
|
||||||
|
public static final Field NUM_PAGES_FIELD = new Field("&N");
|
||||||
|
public static final Field UNDERLINE_FIELD = new Field("&U");
|
||||||
|
public static final Field DOUBLE_UNDERLINE_FIELD = new Field("&E");
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a special field in a header or footer,
|
||||||
|
* eg the page number
|
||||||
|
*/
|
||||||
|
public static class Field {
|
||||||
|
private static ArrayList ALL_FIELDS = new ArrayList();
|
||||||
|
/** The character sequence that marks this field */
|
||||||
|
public final String sequence;
|
||||||
|
private Field(String sequence) {
|
||||||
|
this.sequence = sequence;
|
||||||
|
ALL_FIELDS.add(this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,11 +17,6 @@
|
||||||
|
|
||||||
package org.apache.poi.hssf.usermodel;
|
package org.apache.poi.hssf.usermodel;
|
||||||
|
|
||||||
import java.io.ByteArrayInputStream;
|
|
||||||
import java.io.ByteArrayOutputStream;
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileInputStream;
|
|
||||||
|
|
||||||
import junit.framework.TestCase;
|
import junit.framework.TestCase;
|
||||||
|
|
||||||
import org.apache.poi.hssf.HSSFTestDataSamples;
|
import org.apache.poi.hssf.HSSFTestDataSamples;
|
||||||
|
@ -50,6 +45,46 @@ public final class TestHSSFHeaderFooter extends TestCase {
|
||||||
assertEquals("Top Right", head.getRight());
|
assertEquals("Top Right", head.getRight());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testSpecialChars() {
|
||||||
|
assertEquals("&U", HSSFHeader.startUnderline());
|
||||||
|
assertEquals("&U", HSSFHeader.endUnderline());
|
||||||
|
assertEquals("&P", HSSFHeader.page());
|
||||||
|
|
||||||
|
assertEquals("&22", HSSFFooter.fontSize((short)22));
|
||||||
|
assertEquals("&\"Arial,bold\"", HSSFFooter.font("Arial", "bold"));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testStripFields() {
|
||||||
|
String simple = "I am a test header";
|
||||||
|
String withPage = "I am a&P test header";
|
||||||
|
String withLots = "I&A am&N a&P test&T header&U";
|
||||||
|
String withFont = "I&22 am a&\"Arial,bold\" test header";
|
||||||
|
String withOtherAnds = "I am a&P test header&Z";
|
||||||
|
|
||||||
|
assertEquals(simple, HSSFHeader.stripFields(simple));
|
||||||
|
assertEquals(simple, HSSFHeader.stripFields(withPage));
|
||||||
|
assertEquals(simple, HSSFHeader.stripFields(withLots));
|
||||||
|
assertEquals(simple, HSSFHeader.stripFields(withFont));
|
||||||
|
assertEquals(simple + "&Z", HSSFHeader.stripFields(withOtherAnds));
|
||||||
|
|
||||||
|
// Now test the default strip flag
|
||||||
|
HSSFWorkbook wb = HSSFTestDataSamples.openSampleWorkbook("EmbeddedChartHeaderTest.xls");
|
||||||
|
HSSFSheet s = wb.getSheetAt( 0 );
|
||||||
|
HSSFHeader head = s.getHeader();
|
||||||
|
|
||||||
|
assertEquals("Top Left", head.getLeft());
|
||||||
|
assertEquals("Top Center", head.getCenter());
|
||||||
|
assertEquals("Top Right", head.getRight());
|
||||||
|
|
||||||
|
head.setLeft("Top &P&F&D Left");
|
||||||
|
assertEquals("Top &P&F&D Left", head.getLeft());
|
||||||
|
assertFalse(head.areFieldsStripped());
|
||||||
|
|
||||||
|
head.setAreFieldsStripped(true);
|
||||||
|
assertEquals("Top Left", head.getLeft());
|
||||||
|
assertTrue(head.areFieldsStripped());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests that get header retreives the proper values.
|
* Tests that get header retreives the proper values.
|
||||||
*
|
*
|
||||||
|
|
Loading…
Reference in New Issue