diff --git a/src/documentation/content/xdocs/status.xml b/src/documentation/content/xdocs/status.xml
index 99942aaaeb..6ba397d182 100644
--- a/src/documentation/content/xdocs/status.xml
+++ b/src/documentation/content/xdocs/status.xml
@@ -35,6 +35,7 @@
+ 46776 - Added clone() method to MulBlankRecord to fix crash in Sheet.cloneSheet()
47244 - Fixed HSSFSheet to handle missing header / footer records
47312 - Fixed formula parser to properly reject cell references with a '0' row component
47199 - Fixed PageSettingsBlock/Sheet to tolerate margin records after other non-PSB records
diff --git a/src/java/org/apache/poi/hssf/record/MulBlankRecord.java b/src/java/org/apache/poi/hssf/record/MulBlankRecord.java
index 20a061a026..87d4eafeac 100644
--- a/src/java/org/apache/poi/hssf/record/MulBlankRecord.java
+++ b/src/java/org/apache/poi/hssf/record/MulBlankRecord.java
@@ -22,128 +22,114 @@ import org.apache.poi.util.LittleEndianOutput;
/**
* Title: Multiple Blank cell record(0x00BE)
* Description: Represents a set of columns in a row with no value but with styling.
- * In this release we have read-only support for this record type.
- * The RecordFactory converts this to a set of BlankRecord objects.
+ *
* REFERENCE: PG 329 Microsoft Excel 97 Developer's Kit (ISBN: 1-57231-498-2)
* @author Andrew C. Oliver (acoliver at apache dot org)
* @author Glen Stampoultzis (glens at apache.org)
* @see BlankRecord
*/
public final class MulBlankRecord extends StandardRecord {
- public final static short sid = 0x00BE;
-
- private int field_1_row;
- private short field_2_first_col;
- private short[] field_3_xfs;
- private short field_4_last_col;
+ public final static short sid = 0x00BE;
- public MulBlankRecord(int row, int firstCol, short[] xfs) {
- field_1_row = row;
- field_2_first_col = (short)firstCol;
- field_3_xfs = xfs;
- field_4_last_col = (short) (firstCol + xfs.length - 1);
- }
+ private final int _row;
+ private final int _firstCol;
+ private final short[] _xfs;
+ private final int _lastCol;
- /**
- * get the row number of the cells this represents
- *
- * @return row number
- */
- public int getRow()
- {
- return field_1_row;
- }
+ public MulBlankRecord(int row, int firstCol, short[] xfs) {
+ _row = row;
+ _firstCol = firstCol;
+ _xfs = xfs;
+ _lastCol = firstCol + xfs.length - 1;
+ }
- /**
- * starting column (first cell this holds in the row)
- * @return first column number
- */
- public short getFirstColumn()
- {
- return field_2_first_col;
- }
+ /**
+ * @return the row number of the cells this represents
+ */
+ public int getRow() {
+ return _row;
+ }
- /**
- * ending column (last cell this holds in the row)
- * @return first column number
- */
- public short getLastColumn()
- {
- return field_4_last_col;
- }
+ /**
+ * @return starting column (first cell this holds in the row). Zero based
+ */
+ public int getFirstColumn() {
+ return _firstCol;
+ }
- /**
- * get the number of columns this contains (last-first +1)
- * @return number of columns (last - first +1)
- */
- public int getNumColumns()
- {
- return field_4_last_col - field_2_first_col + 1;
- }
+ /**
+ * get the number of columns this contains (last-first +1)
+ * @return number of columns (last - first +1)
+ */
+ public int getNumColumns() {
+ return _lastCol - _firstCol + 1;
+ }
- /**
- * returns the xf index for column (coffset = column - field_2_first_col)
- * @param coffset the column (coffset = column - field_2_first_col)
- * @return the XF index for the column
- */
- public short getXFAt(int coffset)
- {
- return field_3_xfs[ coffset ];
- }
+ /**
+ * returns the xf index for column (coffset = column - field_2_first_col)
+ * @param coffset the column (coffset = column - field_2_first_col)
+ * @return the XF index for the column
+ */
+ public short getXFAt(int coffset) {
+ return _xfs[coffset];
+ }
- /**
- * @param in the RecordInputstream to read the record from
- */
- public MulBlankRecord(RecordInputStream in) {
- field_1_row = in.readUShort();
- field_2_first_col = in.readShort();
- field_3_xfs = parseXFs(in);
- field_4_last_col = in.readShort();
- }
+ /**
+ * @param in the RecordInputstream to read the record from
+ */
+ public MulBlankRecord(RecordInputStream in) {
+ _row = in.readUShort();
+ _firstCol = in.readShort();
+ _xfs = parseXFs(in);
+ _lastCol = in.readShort();
+ }
- private static short [] parseXFs(RecordInputStream in)
- {
- short[] retval = new short[ (in.remaining() - 2) / 2 ];
+ private static short [] parseXFs(RecordInputStream in) {
+ short[] retval = new short[(in.remaining() - 2) / 2];
- for (int idx = 0; idx < retval.length;idx++)
- {
- retval[idx] = in.readShort();
- }
- return retval;
- }
+ for (int idx = 0; idx < retval.length;idx++) {
+ retval[idx] = in.readShort();
+ }
+ return retval;
+ }
- public String toString() {
- StringBuffer buffer = new StringBuffer();
+ public String toString() {
+ StringBuffer buffer = new StringBuffer();
- buffer.append("[MULBLANK]\n");
- buffer.append("row = ").append(Integer.toHexString(getRow())).append("\n");
- buffer.append("firstcol = ").append(Integer.toHexString(getFirstColumn())).append("\n");
- buffer.append(" lastcol = ").append(Integer.toHexString(getLastColumn())).append("\n");
- for (int k = 0; k < getNumColumns(); k++) {
- buffer.append("xf").append(k).append(" = ").append(
- Integer.toHexString(getXFAt(k))).append("\n");
- }
- buffer.append("[/MULBLANK]\n");
- return buffer.toString();
- }
+ buffer.append("[MULBLANK]\n");
+ buffer.append("row = ").append(Integer.toHexString(getRow())).append("\n");
+ buffer.append("firstcol = ").append(Integer.toHexString(getFirstColumn())).append("\n");
+ buffer.append(" lastcol = ").append(Integer.toHexString(_lastCol)).append("\n");
+ for (int k = 0; k < getNumColumns(); k++) {
+ buffer.append("xf").append(k).append(" = ").append(
+ Integer.toHexString(getXFAt(k))).append("\n");
+ }
+ buffer.append("[/MULBLANK]\n");
+ return buffer.toString();
+ }
- public short getSid()
- {
- return sid;
- }
+ public short getSid() {
+ return sid;
+ }
- public void serialize(LittleEndianOutput out) {
- out.writeShort(field_1_row);
- out.writeShort(field_2_first_col);
- int nItems = field_3_xfs.length;
- for (int i = 0; i < nItems; i++) {
- out.writeShort(field_3_xfs[i]);
- }
- out.writeShort(field_4_last_col);
- }
+ public void serialize(LittleEndianOutput out) {
+ out.writeShort(_row);
+ out.writeShort(_firstCol);
+ int nItems = _xfs.length;
+ for (int i = 0; i < nItems; i++) {
+ out.writeShort(_xfs[i]);
+ }
+ out.writeShort(_lastCol);
+ }
- protected int getDataSize() {
- // 3 short fields + array of shorts
- return 6 + field_3_xfs.length * 2;
- }
+ protected int getDataSize() {
+ // 3 short fields + array of shorts
+ return 6 + _xfs.length * 2;
+ }
+
+ @Override
+ public Object clone() {
+ // immutable - so OK to return this
+ return this;
+ }
}
diff --git a/src/testcases/org/apache/poi/hssf/model/TestSheet.java b/src/testcases/org/apache/poi/hssf/model/TestSheet.java
index 68ff2f6599..a16c278e5a 100644
--- a/src/testcases/org/apache/poi/hssf/model/TestSheet.java
+++ b/src/testcases/org/apache/poi/hssf/model/TestSheet.java
@@ -18,6 +18,7 @@
package org.apache.poi.hssf.model;
import java.util.ArrayList;
+import java.util.Arrays;
import java.util.List;
import junit.framework.AssertionFailedError;
@@ -34,6 +35,7 @@ import org.apache.poi.hssf.record.FormulaRecord;
import org.apache.poi.hssf.record.GutsRecord;
import org.apache.poi.hssf.record.IndexRecord;
import org.apache.poi.hssf.record.MergeCellsRecord;
+import org.apache.poi.hssf.record.MulBlankRecord;
import org.apache.poi.hssf.record.NumberRecord;
import org.apache.poi.hssf.record.Record;
import org.apache.poi.hssf.record.RecordBase;
@@ -629,11 +631,11 @@ public final class TestSheet extends TestCase {
assertEquals(colIx, dims.getFirstCol());
assertEquals(colIx, dims.getLastCol());
}
-
+
/**
- * Prior to the fix for bug 46547, shifting formulas would have the side-effect
+ * Prior to the fix for bug 46547, shifting formulas would have the side-effect
* of creating a {@link ConditionalFormattingTable}. There was no impairment to
- * functionality since empty record aggregates are equivalent to missing record
+ * functionality since empty record aggregates are equivalent to missing record
* aggregates. However, since this unnecessary creation helped expose bug 46547b,
* and since there is a slight performance hit the fix was made to avoid it.
*/
@@ -643,14 +645,14 @@ public final class TestSheet extends TestCase {
List sheetRecs = sheet.getRecords();
assertEquals(22, sheetRecs.size());
-
+
FormulaShifter shifter = FormulaShifter.createForRowShift(0, 0, 0, 1);
sheet.updateFormulasAfterCellShift(shifter, 0);
if (sheetRecs.size() == 23 && sheetRecs.get(21) instanceof ConditionalFormattingTable) {
throw new AssertionFailedError("Identified bug 46547a");
}
assertEquals(22, sheetRecs.size());
-
+
}
/**
* Bug 46547 happened when attempting to add conditional formatting to a sheet
@@ -671,4 +673,33 @@ public final class TestSheet extends TestCase {
}
assertNotNull(cft);
}
+
+ public void testCloneMulBlank_bug46776() {
+ Record[] recs = {
+ Sheet.createBOF(),
+ new DimensionsRecord(),
+ new RowRecord(1),
+ new MulBlankRecord(1, 3, new short[] { 0x0F, 0x0F, 0x0F, } ),
+ new RowRecord(2),
+ createWindow2Record(),
+ EOFRecord.instance,
+ };
+
+ Sheet sheet = createSheet(Arrays.asList(recs));
+
+ Sheet sheet2;
+ try {
+ sheet2 = sheet.cloneSheet();
+ } catch (RuntimeException e) {
+ if (e.getMessage().equals("The class org.apache.poi.hssf.record.MulBlankRecord needs to define a clone method")) {
+ throw new AssertionFailedError("Identified bug 46776");
+ }
+ throw e;
+ }
+
+ RecordCollector rc = new RecordCollector();
+ sheet2.visitContainedRecords(rc, 0);
+ Record[] clonedRecs = rc.getRecords();
+ assertEquals(recs.length+2, clonedRecs.length); // +2 for INDEX and DBCELL
+ }
}