mirror of https://github.com/apache/poi.git
fixed RecordFormatException when reading LbsDataSubRecord, see bugzilla 47701
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@886113 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
86d7916c2a
commit
b4da1cab04
|
@ -34,6 +34,7 @@
|
||||||
|
|
||||||
<changes>
|
<changes>
|
||||||
<release version="3.6-beta1" date="2009-??-??">
|
<release version="3.6-beta1" date="2009-??-??">
|
||||||
|
<action dev="POI-DEVELOPERS" type="fix">47701 - fixed RecordFormatException when reading list subrecords (LbsDataSubRecord)</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add"> memory usage optimization in XSSF - avoid creating parentless xml beans</action>
|
<action dev="POI-DEVELOPERS" type="add"> memory usage optimization in XSSF - avoid creating parentless xml beans</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">47188 - avoid corruption of workbook when adding cell comments </action>
|
<action dev="POI-DEVELOPERS" type="fix">47188 - avoid corruption of workbook when adding cell comments </action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">48106 - improved work with cell comments in XSSF</action>
|
<action dev="POI-DEVELOPERS" type="fix">48106 - improved work with cell comments in XSSF</action>
|
||||||
|
|
|
@ -46,6 +46,11 @@ public final class EndSubRecord extends SubRecord {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isTerminating(){
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
public String toString()
|
public String toString()
|
||||||
{
|
{
|
||||||
StringBuffer buffer = new StringBuffer();
|
StringBuffer buffer = new StringBuffer();
|
||||||
|
|
|
@ -0,0 +1,340 @@
|
||||||
|
/* ====================================================================
|
||||||
|
Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
contributor license agreements. See the NOTICE file distributed with
|
||||||
|
this work for additional information regarding copyright ownership.
|
||||||
|
The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
(the "License"); you may not use this file except in compliance with
|
||||||
|
the License. You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
==================================================================== */
|
||||||
|
package org.apache.poi.hssf.record;
|
||||||
|
|
||||||
|
import org.apache.poi.hssf.record.formula.*;
|
||||||
|
import org.apache.poi.util.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This structure specifies the properties of a list or drop-down list embedded object in a sheet.
|
||||||
|
*/
|
||||||
|
public class LbsDataSubRecord extends SubRecord {
|
||||||
|
|
||||||
|
public static final int sid = 0x0013;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* From [MS-XLS].pdf 2.5.147 FtLbsData:
|
||||||
|
*
|
||||||
|
* An unsigned integer that indirectly specifies whether
|
||||||
|
* some of the data in this structure appear in a subsequent Continue record.
|
||||||
|
* If _cbFContinued is 0x00, all of the fields in this structure except sid and _cbFContinued
|
||||||
|
* MUST NOT exist. If this entire structure is contained within the same record,
|
||||||
|
* then _cbFContinued MUST be greater than or equal to the size, in bytes,
|
||||||
|
* of this structure, not including the four bytes for the ft and _cbFContinued fields
|
||||||
|
*/
|
||||||
|
private int _cbFContinued;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* a formula that specifies the range of cell values that are the items in this list.
|
||||||
|
*/
|
||||||
|
private int _unknownPreFormulaInt;
|
||||||
|
private Ptg _linkPtg;
|
||||||
|
private Byte _unknownPostFormulaByte;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An unsigned integer that specifies the number of items in the list.
|
||||||
|
*/
|
||||||
|
private int _cLines;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An unsigned integer that specifies the one-based index of the first selected item in this list.
|
||||||
|
* A value of 0x00 specifies there is no currently selected item.
|
||||||
|
*/
|
||||||
|
private int _iSel;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* flags that tell what data follows
|
||||||
|
*/
|
||||||
|
private int _flags;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An ObjId that specifies the edit box associated with this list.
|
||||||
|
* A value of 0x00 specifies that there is no edit box associated with this list.
|
||||||
|
*/
|
||||||
|
private int _idEdit;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An optional LbsDropData that specifies properties for this dropdown control.
|
||||||
|
* This field MUST exist if and only if the containing Obj?s cmo.ot is equal to 0x14.
|
||||||
|
*/
|
||||||
|
private LbsDropData _dropData;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An optional array of strings where each string specifies an item in the list.
|
||||||
|
* The number of elements in this array, if it exists, MUST be {@link #_cLines}
|
||||||
|
*/
|
||||||
|
private String[] _rgLines;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An optional array of booleans that specifies
|
||||||
|
* which items in the list are part of a multiple selection
|
||||||
|
*/
|
||||||
|
private boolean[] _bsels;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param in the stream to read data from
|
||||||
|
* @param cbFContinued the seconf short in the record header
|
||||||
|
* @param cmoOt the containing Obj's {@link CommonObjectDataSubRecord#field_1_objectType}
|
||||||
|
*/
|
||||||
|
public LbsDataSubRecord(LittleEndianInput in, int cbFContinued, int cmoOt) {
|
||||||
|
_cbFContinued = cbFContinued;
|
||||||
|
|
||||||
|
int encodedTokenLen = in.readUShort();
|
||||||
|
if (encodedTokenLen > 0) {
|
||||||
|
int formulaSize = in.readUShort();
|
||||||
|
_unknownPreFormulaInt = in.readInt();
|
||||||
|
|
||||||
|
Ptg[] ptgs = Ptg.readTokens(formulaSize, in);
|
||||||
|
if (ptgs.length != 1) {
|
||||||
|
throw new RecordFormatException("Read " + ptgs.length
|
||||||
|
+ " tokens but expected exactly 1");
|
||||||
|
}
|
||||||
|
_linkPtg = ptgs[0];
|
||||||
|
switch (encodedTokenLen - formulaSize - 6) {
|
||||||
|
case 1:
|
||||||
|
_unknownPostFormulaByte = in.readByte();
|
||||||
|
break;
|
||||||
|
case 0:
|
||||||
|
_unknownPostFormulaByte = null;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new RecordFormatException("Unexpected leftover bytes");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_cLines = in.readUShort();
|
||||||
|
_iSel = in.readUShort();
|
||||||
|
_flags = in.readUShort();
|
||||||
|
_idEdit = in.readUShort();
|
||||||
|
|
||||||
|
// From [MS-XLS].pdf 2.5.147 FtLbsData:
|
||||||
|
// This field MUST exist if and only if the containing Obj?s cmo.ot is equal to 0x14.
|
||||||
|
if(cmoOt == 0x14) {
|
||||||
|
_dropData = new LbsDropData(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
// From [MS-XLS].pdf 2.5.147 FtLbsData:
|
||||||
|
// This array MUST exist if and only if the fValidPlex flag (0x2) is set
|
||||||
|
if((_flags & 0x2) != 0) {
|
||||||
|
_rgLines = new String[_cLines];
|
||||||
|
for(int i=0; i < _cLines; i++) {
|
||||||
|
_rgLines[i] = StringUtil.readUnicodeString(in);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// bits 5-6 in the _flags specify the type
|
||||||
|
// of selection behavior this list control is expected to support
|
||||||
|
|
||||||
|
// From [MS-XLS].pdf 2.5.147 FtLbsData:
|
||||||
|
// This array MUST exist if and only if the wListType field is not equal to 0.
|
||||||
|
if(((_flags >> 4) & 0x2) != 0) {
|
||||||
|
_bsels = new boolean[_cLines];
|
||||||
|
for(int i=0; i < _cLines; i++) {
|
||||||
|
_bsels[i] = in.readByte() == 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return true as LbsDataSubRecord is always the last sub-record
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean isTerminating(){
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected int getDataSize() {
|
||||||
|
int result = 2; // 2 initial shorts
|
||||||
|
|
||||||
|
// optional link formula
|
||||||
|
if (_linkPtg != null) {
|
||||||
|
result += 2; // encoded Ptg size
|
||||||
|
result += 4; // unknown int
|
||||||
|
result += _linkPtg.getSize();
|
||||||
|
if (_unknownPostFormulaByte != null) {
|
||||||
|
result += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result += 4 * 2; // 4 shorts
|
||||||
|
if(_dropData != null) {
|
||||||
|
result += _dropData.getDataSize();
|
||||||
|
}
|
||||||
|
if(_rgLines != null) {
|
||||||
|
for(String str : _rgLines){
|
||||||
|
result += StringUtil.getEncodedSize(str);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(_bsels != null) {
|
||||||
|
result += _bsels.length;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void serialize(LittleEndianOutput out) {
|
||||||
|
out.writeShort(sid);
|
||||||
|
out.writeShort(_cbFContinued);
|
||||||
|
|
||||||
|
if (_linkPtg == null) {
|
||||||
|
out.writeShort(0);
|
||||||
|
} else {
|
||||||
|
int formulaSize = _linkPtg.getSize();
|
||||||
|
int linkSize = formulaSize + 6;
|
||||||
|
if (_unknownPostFormulaByte != null) {
|
||||||
|
linkSize++;
|
||||||
|
}
|
||||||
|
out.writeShort(linkSize);
|
||||||
|
out.writeShort(formulaSize);
|
||||||
|
out.writeInt(_unknownPreFormulaInt);
|
||||||
|
_linkPtg.write(out);
|
||||||
|
if (_unknownPostFormulaByte != null) {
|
||||||
|
out.writeByte(_unknownPostFormulaByte.intValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
out.writeShort(_cLines);
|
||||||
|
out.writeShort(_iSel);
|
||||||
|
out.writeShort(_flags);
|
||||||
|
out.writeShort(_idEdit);
|
||||||
|
|
||||||
|
if(_dropData != null) {
|
||||||
|
_dropData.serialize(out);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(_rgLines != null) {
|
||||||
|
for(String str : _rgLines){
|
||||||
|
StringUtil.writeUnicodeString(out, str);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(_bsels != null) {
|
||||||
|
for(boolean val : _bsels){
|
||||||
|
out.writeByte(val ? 1 : 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object clone() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
StringBuffer sb = new StringBuffer(256);
|
||||||
|
|
||||||
|
sb.append("[ftLbsData]\n");
|
||||||
|
sb.append(" .unknownShort1 =").append(HexDump.shortToHex(_cbFContinued)).append("\n");
|
||||||
|
sb.append(" .formula = ").append('\n');
|
||||||
|
sb.append(_linkPtg.toString()).append(_linkPtg.getRVAType()).append('\n');
|
||||||
|
sb.append(" .nEntryCount =").append(HexDump.shortToHex(_cLines)).append("\n");
|
||||||
|
sb.append(" .selEntryIx =").append(HexDump.shortToHex(_iSel)).append("\n");
|
||||||
|
sb.append(" .style =").append(HexDump.shortToHex(_flags)).append("\n");
|
||||||
|
sb.append(" .unknownShort10=").append(HexDump.shortToHex(_idEdit)).append("\n");
|
||||||
|
if(_dropData != null) sb.append('\n').append(_dropData.toString());
|
||||||
|
sb.append("[/ftLbsData]\n");
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @return the formula that specifies the range of cell values that are the items in this list.
|
||||||
|
*/
|
||||||
|
public Ptg getFormula(){
|
||||||
|
return _linkPtg;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the number of items in the list
|
||||||
|
*/
|
||||||
|
public int getNumberOfItems(){
|
||||||
|
return _cLines;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This structure specifies properties of the dropdown list control
|
||||||
|
*/
|
||||||
|
public static class LbsDropData {
|
||||||
|
/**
|
||||||
|
* An unsigned integer that specifies the style of this dropdown.
|
||||||
|
*/
|
||||||
|
private int _wStyle;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An unsigned integer that specifies the number of lines to be displayed in the dropdown.
|
||||||
|
*/
|
||||||
|
private int _cLine;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An unsigned integer that specifies the smallest width in pixels allowed for the dropdown window
|
||||||
|
*/
|
||||||
|
private int _dxMin;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* a string that specifies the current string value in the dropdown
|
||||||
|
*/
|
||||||
|
private String _str;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optional, undefined and MUST be ignored.
|
||||||
|
* This field MUST exist if and only if the size of str in bytes is an odd number
|
||||||
|
*/
|
||||||
|
private Byte _unused;
|
||||||
|
|
||||||
|
public LbsDropData(LittleEndianInput in){
|
||||||
|
_wStyle = in.readUShort();
|
||||||
|
_cLine = in.readUShort();
|
||||||
|
_dxMin = in.readUShort();
|
||||||
|
_str = StringUtil.readUnicodeString(in);
|
||||||
|
if(StringUtil.getEncodedSize(_str) % 2 != 0){
|
||||||
|
_unused = in.readByte();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void serialize(LittleEndianOutput out) {
|
||||||
|
out.writeShort(_wStyle);
|
||||||
|
out.writeShort(_cLine);
|
||||||
|
out.writeShort(_dxMin);
|
||||||
|
StringUtil.writeUnicodeString(out, _str);
|
||||||
|
if(_unused != null) out.writeByte(_unused);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getDataSize() {
|
||||||
|
int size = 6;
|
||||||
|
size += StringUtil.getEncodedSize(_str);
|
||||||
|
size += _unused;
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString(){
|
||||||
|
StringBuffer sb = new StringBuffer();
|
||||||
|
sb.append("[LbsDropData]\n");
|
||||||
|
sb.append(" ._wStyle: ").append(_wStyle).append('\n');
|
||||||
|
sb.append(" ._cLine: ").append(_cLine).append('\n');
|
||||||
|
sb.append(" ._dxMin: ").append(_dxMin).append('\n');
|
||||||
|
sb.append(" ._str: ").append(_str).append('\n');
|
||||||
|
if(_unused != null) sb.append(" ._unused: ").append(_unused).append('\n');
|
||||||
|
sb.append("[/LbsDropData]\n");
|
||||||
|
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -78,20 +78,24 @@ public final class ObjRecord extends Record {
|
||||||
subrecords = null;
|
subrecords = null;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//YK: files produced by OO violate the condition below
|
||||||
|
/*
|
||||||
if (subRecordData.length % 2 != 0) {
|
if (subRecordData.length % 2 != 0) {
|
||||||
String msg = "Unexpected length of subRecordData : " + HexDump.toHex(subRecordData);
|
String msg = "Unexpected length of subRecordData : " + HexDump.toHex(subRecordData);
|
||||||
throw new RecordFormatException(msg);
|
throw new RecordFormatException(msg);
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
// System.out.println(HexDump.toHex(subRecordData));
|
|
||||||
|
|
||||||
subrecords = new ArrayList<SubRecord>();
|
subrecords = new ArrayList<SubRecord>();
|
||||||
ByteArrayInputStream bais = new ByteArrayInputStream(subRecordData);
|
ByteArrayInputStream bais = new ByteArrayInputStream(subRecordData);
|
||||||
LittleEndianInputStream subRecStream = new LittleEndianInputStream(bais);
|
LittleEndianInputStream subRecStream = new LittleEndianInputStream(bais);
|
||||||
|
CommonObjectDataSubRecord cmo = (CommonObjectDataSubRecord)SubRecord.createSubRecord(subRecStream, 0);
|
||||||
|
subrecords.add(cmo);
|
||||||
while (true) {
|
while (true) {
|
||||||
SubRecord subRecord = SubRecord.createSubRecord(subRecStream);
|
SubRecord subRecord = SubRecord.createSubRecord(subRecStream, cmo.getObjectType());
|
||||||
subrecords.add(subRecord);
|
subrecords.add(subRecord);
|
||||||
if (subRecord instanceof EndSubRecord) {
|
if (subRecord.isTerminating()) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -107,6 +111,7 @@ public final class ObjRecord extends Record {
|
||||||
}
|
}
|
||||||
_isPaddedToQuadByteMultiple = false;
|
_isPaddedToQuadByteMultiple = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
_isPaddedToQuadByteMultiple = false;
|
_isPaddedToQuadByteMultiple = false;
|
||||||
}
|
}
|
||||||
|
@ -123,10 +128,6 @@ public final class ObjRecord extends Record {
|
||||||
* to the its proper size. POI does the same.
|
* to the its proper size. POI does the same.
|
||||||
*/
|
*/
|
||||||
private static boolean canPaddingBeDiscarded(byte[] data, int nRemainingBytes) {
|
private static boolean canPaddingBeDiscarded(byte[] data, int nRemainingBytes) {
|
||||||
if (data.length < 0x1FEE) {
|
|
||||||
// this looks like a different problem
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
// make sure none of the padding looks important
|
// make sure none of the padding looks important
|
||||||
for(int i=data.length-nRemainingBytes; i<data.length; i++) {
|
for(int i=data.length-nRemainingBytes; i<data.length; i++) {
|
||||||
if (data[i] != 0x00) {
|
if (data[i] != 0x00) {
|
||||||
|
|
|
@ -17,19 +17,13 @@
|
||||||
|
|
||||||
package org.apache.poi.hssf.record;
|
package org.apache.poi.hssf.record;
|
||||||
|
|
||||||
import java.io.ByteArrayOutputStream;
|
|
||||||
|
|
||||||
import org.apache.poi.hssf.record.formula.Area3DPtg;
|
|
||||||
import org.apache.poi.hssf.record.formula.AreaPtg;
|
|
||||||
import org.apache.poi.hssf.record.formula.Ptg;
|
|
||||||
import org.apache.poi.hssf.record.formula.Ref3DPtg;
|
|
||||||
import org.apache.poi.hssf.record.formula.RefPtg;
|
|
||||||
import org.apache.poi.util.HexDump;
|
import org.apache.poi.util.HexDump;
|
||||||
import org.apache.poi.util.LittleEndianByteArrayInputStream;
|
|
||||||
import org.apache.poi.util.LittleEndianInput;
|
import org.apache.poi.util.LittleEndianInput;
|
||||||
import org.apache.poi.util.LittleEndianOutput;
|
import org.apache.poi.util.LittleEndianOutput;
|
||||||
import org.apache.poi.util.LittleEndianOutputStream;
|
import org.apache.poi.util.LittleEndianOutputStream;
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Subrecords are part of the OBJ class.
|
* Subrecords are part of the OBJ class.
|
||||||
*/
|
*/
|
||||||
|
@ -38,7 +32,15 @@ public abstract class SubRecord {
|
||||||
// no fields to initialise
|
// no fields to initialise
|
||||||
}
|
}
|
||||||
|
|
||||||
public static SubRecord createSubRecord(LittleEndianInput in) {
|
/**
|
||||||
|
* read a sub-record from the supplied stream
|
||||||
|
*
|
||||||
|
* @param in the stream to read from
|
||||||
|
* @param cmoOt the objectType field of the containing CommonObjectDataSubRecord,
|
||||||
|
* we need it to propagate to next sub-records as it defines what data follows
|
||||||
|
* @return the created sub-record
|
||||||
|
*/
|
||||||
|
public static SubRecord createSubRecord(LittleEndianInput in, int cmoOt) {
|
||||||
int sid = in.readUShort();
|
int sid = in.readUShort();
|
||||||
int secondUShort = in.readUShort(); // Often (but not always) the datasize for the sub-record
|
int secondUShort = in.readUShort(); // Often (but not always) the datasize for the sub-record
|
||||||
|
|
||||||
|
@ -54,7 +56,7 @@ public abstract class SubRecord {
|
||||||
case NoteStructureSubRecord.sid:
|
case NoteStructureSubRecord.sid:
|
||||||
return new NoteStructureSubRecord(in, secondUShort);
|
return new NoteStructureSubRecord(in, secondUShort);
|
||||||
case LbsDataSubRecord.sid:
|
case LbsDataSubRecord.sid:
|
||||||
return new LbsDataSubRecord(in, secondUShort);
|
return new LbsDataSubRecord(in, secondUShort, cmoOt);
|
||||||
}
|
}
|
||||||
return new UnknownSubRecord(in, sid, secondUShort);
|
return new UnknownSubRecord(in, sid, secondUShort);
|
||||||
}
|
}
|
||||||
|
@ -78,6 +80,17 @@ public abstract class SubRecord {
|
||||||
public abstract void serialize(LittleEndianOutput out);
|
public abstract void serialize(LittleEndianOutput out);
|
||||||
public abstract Object clone();
|
public abstract Object clone();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wether this record terminates the sub-record stream.
|
||||||
|
* There are two cases when this method must be overridden and return <code>true</code>
|
||||||
|
* - EndSubRecord (sid = 0x00)
|
||||||
|
* - LbsDataSubRecord (sid = 0x12)
|
||||||
|
*
|
||||||
|
* @return whether this record is the last in the sub-record stream
|
||||||
|
*/
|
||||||
|
public boolean isTerminating(){
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
private static final class UnknownSubRecord extends SubRecord {
|
private static final class UnknownSubRecord extends SubRecord {
|
||||||
|
|
||||||
|
@ -111,140 +124,4 @@ public abstract class SubRecord {
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO make into a top level class
|
|
||||||
// perhaps all SubRecord sublcasses could go to their own package
|
|
||||||
private static final class LbsDataSubRecord extends SubRecord {
|
|
||||||
|
|
||||||
public static final int sid = 0x0013;
|
|
||||||
|
|
||||||
private int _unknownShort1;
|
|
||||||
private int _unknownInt4;
|
|
||||||
private Ptg _linkPtg;
|
|
||||||
private Byte _unknownByte6;
|
|
||||||
private int _nEntryCount;
|
|
||||||
private int _selectedEntryIndex;
|
|
||||||
private int _style;
|
|
||||||
private int _unknownShort10;
|
|
||||||
private int _comboStyle;
|
|
||||||
private int _lineCount;
|
|
||||||
private int _unknownShort13;
|
|
||||||
|
|
||||||
public LbsDataSubRecord(LittleEndianInput in, int unknownShort1) {
|
|
||||||
_unknownShort1 = unknownShort1;
|
|
||||||
int linkSize = in.readUShort();
|
|
||||||
if (linkSize > 0) {
|
|
||||||
int formulaSize = in.readUShort();
|
|
||||||
_unknownInt4 = in.readInt();
|
|
||||||
|
|
||||||
|
|
||||||
byte[] buf = new byte[formulaSize];
|
|
||||||
in.readFully(buf);
|
|
||||||
_linkPtg = readRefPtg(buf);
|
|
||||||
switch (linkSize - formulaSize - 6) {
|
|
||||||
case 1:
|
|
||||||
_unknownByte6 = Byte.valueOf(in.readByte());
|
|
||||||
break;
|
|
||||||
case 0:
|
|
||||||
_unknownByte6 = null;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new RecordFormatException("Unexpected leftover bytes");
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
|
||||||
_unknownInt4 = 0;
|
|
||||||
_linkPtg = null;
|
|
||||||
_unknownByte6 = null;
|
|
||||||
}
|
|
||||||
_nEntryCount = in.readUShort();
|
|
||||||
_selectedEntryIndex = in.readUShort();
|
|
||||||
_style = in.readUShort();
|
|
||||||
_unknownShort10 = in.readUShort();
|
|
||||||
_comboStyle = in.readUShort();
|
|
||||||
_lineCount = in.readUShort();
|
|
||||||
_unknownShort13 = in.readUShort();
|
|
||||||
|
|
||||||
}
|
|
||||||
protected int getDataSize() {
|
|
||||||
int result = 2; // 2 initial shorts
|
|
||||||
|
|
||||||
// optional link formula
|
|
||||||
if (_linkPtg != null) {
|
|
||||||
result += 2; // encoded Ptg size
|
|
||||||
result += 4; // unknown int
|
|
||||||
result += _linkPtg.getSize();
|
|
||||||
if (_unknownByte6 != null) {
|
|
||||||
result += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
result += 7 * 2; // 7 shorts
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
public void serialize(LittleEndianOutput out) {
|
|
||||||
out.writeShort(sid);
|
|
||||||
out.writeShort(_unknownShort1); // note - this is *not* the size
|
|
||||||
if (_linkPtg == null) {
|
|
||||||
out.writeShort(0);
|
|
||||||
} else {
|
|
||||||
int formulaSize = _linkPtg.getSize();
|
|
||||||
int linkSize = formulaSize + 6;
|
|
||||||
if (_unknownByte6 != null) {
|
|
||||||
linkSize++;
|
|
||||||
}
|
|
||||||
out.writeShort(linkSize);
|
|
||||||
out.writeShort(formulaSize);
|
|
||||||
out.writeInt(_unknownInt4);
|
|
||||||
_linkPtg.write(out);
|
|
||||||
if (_unknownByte6 != null) {
|
|
||||||
out.writeByte(_unknownByte6.intValue());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
out.writeShort(_nEntryCount);
|
|
||||||
out.writeShort(_selectedEntryIndex);
|
|
||||||
out.writeShort(_style);
|
|
||||||
out.writeShort(_unknownShort10);
|
|
||||||
out.writeShort(_comboStyle);
|
|
||||||
out.writeShort(_lineCount);
|
|
||||||
out.writeShort(_unknownShort13);
|
|
||||||
}
|
|
||||||
private static Ptg readRefPtg(byte[] formulaRawBytes) {
|
|
||||||
LittleEndianInput in = new LittleEndianByteArrayInputStream(formulaRawBytes);
|
|
||||||
byte ptgSid = in.readByte();
|
|
||||||
switch(ptgSid) {
|
|
||||||
case AreaPtg.sid: return new AreaPtg(in);
|
|
||||||
case Area3DPtg.sid: return new Area3DPtg(in);
|
|
||||||
case RefPtg.sid: return new RefPtg(in);
|
|
||||||
case Ref3DPtg.sid: return new Ref3DPtg(in);
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
public Object clone() {
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
public String toString() {
|
|
||||||
StringBuffer sb = new StringBuffer(256);
|
|
||||||
|
|
||||||
sb.append("[ftLbsData]\n");
|
|
||||||
sb.append(" .unknownShort1 =").append(HexDump.shortToHex(_unknownShort1)).append("\n");
|
|
||||||
if (_linkPtg == null) {
|
|
||||||
sb.append(" <no link formula>\n");
|
|
||||||
} else {
|
|
||||||
sb.append(" .unknownInt4 =").append(HexDump.intToHex(_unknownInt4)).append("\n");
|
|
||||||
sb.append(" .linkPtg =").append(_linkPtg.toFormulaString()).append(" (").append(_linkPtg.getRVAType()).append(")").append("\n");
|
|
||||||
if (_unknownByte6 != null) {
|
|
||||||
sb.append(" .unknownByte6 =").append(HexDump.byteToHex(_unknownByte6.byteValue())).append("\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
sb.append(" .nEntryCount =").append(HexDump.shortToHex(_nEntryCount)).append("\n");
|
|
||||||
sb.append(" .selEntryIx =").append(HexDump.shortToHex(_selectedEntryIndex)).append("\n");
|
|
||||||
sb.append(" .style =").append(HexDump.shortToHex(_style)).append("\n");
|
|
||||||
sb.append(" .unknownShort10=").append(HexDump.shortToHex(_unknownShort10)).append("\n");
|
|
||||||
sb.append(" .comboStyle =").append(HexDump.shortToHex(_comboStyle)).append("\n");
|
|
||||||
sb.append(" .lineCount =").append(HexDump.shortToHex(_lineCount)).append("\n");
|
|
||||||
sb.append(" .unknownShort13=").append(HexDump.shortToHex(_unknownShort13)).append("\n");
|
|
||||||
sb.append("[/ftLbsData]\n");
|
|
||||||
return sb.toString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,170 @@
|
||||||
|
/* ====================================================================
|
||||||
|
Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
contributor license agreements. See the NOTICE file distributed with
|
||||||
|
this work for additional information regarding copyright ownership.
|
||||||
|
The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
(the "License"); you may not use this file except in compliance with
|
||||||
|
the License. You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
==================================================================== */
|
||||||
|
|
||||||
|
package org.apache.poi.hssf.record;
|
||||||
|
|
||||||
|
|
||||||
|
import junit.framework.TestCase;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
|
||||||
|
import org.apache.poi.util.HexRead;
|
||||||
|
import org.apache.poi.util.HexDump;
|
||||||
|
import org.apache.poi.util.LittleEndianInputStream;
|
||||||
|
import org.apache.poi.util.LittleEndianOutputStream;
|
||||||
|
import org.apache.poi.hssf.record.formula.Ptg;
|
||||||
|
import org.apache.poi.hssf.record.formula.AreaPtg;
|
||||||
|
import org.apache.poi.ss.util.CellRangeAddress;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the serialization and deserialization of the LbsDataSubRecord class works correctly.
|
||||||
|
*
|
||||||
|
* @author Yegor Kozlov
|
||||||
|
*/
|
||||||
|
public final class TestLbsDataSubRecord extends TestCase {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test read-write round trip
|
||||||
|
* test data was taken from 47701.xls
|
||||||
|
*/
|
||||||
|
public void test_47701(){
|
||||||
|
byte[] data = HexRead.readFromString(
|
||||||
|
"15, 00, 12, 00, 12, 00, 02, 00, 11, 20, " +
|
||||||
|
"00, 00, 00, 00, 80, 3D, 03, 05, 00, 00, " +
|
||||||
|
"00, 00, 0C, 00, 14, 00, 00, 00, 00, 00, " +
|
||||||
|
"00, 00, 00, 00, 00, 00, 01, 00, 0A, 00, " +
|
||||||
|
"00, 00, 10, 00, 01, 00, 13, 00, EE, 1F, " +
|
||||||
|
"10, 00, 09, 00, 40, 9F, 74, 01, 25, 09, " +
|
||||||
|
"00, 0C, 00, 07, 00, 07, 00, 07, 04, 00, " +
|
||||||
|
"00, 00, 08, 00, 00, 00");
|
||||||
|
RecordInputStream in = TestcaseRecordInputStream.create(ObjRecord.sid, data);
|
||||||
|
// check read OK
|
||||||
|
ObjRecord record = new ObjRecord(in);
|
||||||
|
assertEquals(3, record.getSubRecords().size());
|
||||||
|
SubRecord sr = record.getSubRecords().get(2);
|
||||||
|
assertTrue(sr instanceof LbsDataSubRecord);
|
||||||
|
LbsDataSubRecord lbs = (LbsDataSubRecord)sr;
|
||||||
|
assertEquals(4, lbs.getNumberOfItems());
|
||||||
|
|
||||||
|
assertTrue(lbs.getFormula() instanceof AreaPtg);
|
||||||
|
AreaPtg ptg = (AreaPtg)lbs.getFormula();
|
||||||
|
CellRangeAddress range = new CellRangeAddress(
|
||||||
|
ptg.getFirstRow(), ptg.getLastRow(), ptg.getFirstColumn(), ptg.getLastColumn());
|
||||||
|
assertEquals("H10:H13", range.formatAsString());
|
||||||
|
|
||||||
|
// check that it re-serializes to the same data
|
||||||
|
byte[] ser = record.serialize();
|
||||||
|
TestcaseRecordInputStream.confirmRecordEncoding(ObjRecord.sid, data, ser);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* test data was taken from the file attached to Bugzilla 45778
|
||||||
|
*/
|
||||||
|
public void test_45778(){
|
||||||
|
byte[] data = HexRead.readFromString(
|
||||||
|
"15, 00, 12, 00, 14, 00, 01, 00, 01, 00, " +
|
||||||
|
"01, 21, 00, 00, 3C, 13, F4, 03, 00, 00, " +
|
||||||
|
"00, 00, 0C, 00, 14, 00, 00, 00, 00, 00, " +
|
||||||
|
"00, 00, 00, 00, 00, 00, 01, 00, 08, 00, 00, " +
|
||||||
|
"00, 10, 00, 00, 00, " +
|
||||||
|
"13, 00, EE, 1F, " +
|
||||||
|
"00, 00, " +
|
||||||
|
"08, 00, " + //number of items
|
||||||
|
"08, 00, " + //selected item
|
||||||
|
"01, 03, " + //flags
|
||||||
|
"00, 00, " + //objId
|
||||||
|
//LbsDropData
|
||||||
|
"0A, 00, " + //flags
|
||||||
|
"14, 00, " + //the number of lines to be displayed in the dropdown
|
||||||
|
"6C, 00, " + //the smallest width in pixels allowed for the dropdown window
|
||||||
|
"00, 00, " + //num chars
|
||||||
|
"00, 00");
|
||||||
|
RecordInputStream in = TestcaseRecordInputStream.create(ObjRecord.sid, data);
|
||||||
|
// check read OK
|
||||||
|
ObjRecord record = new ObjRecord(in);
|
||||||
|
|
||||||
|
SubRecord sr = record.getSubRecords().get(2);
|
||||||
|
assertTrue(sr instanceof LbsDataSubRecord);
|
||||||
|
LbsDataSubRecord lbs = (LbsDataSubRecord)sr;
|
||||||
|
assertEquals(8, lbs.getNumberOfItems());
|
||||||
|
assertNull(lbs.getFormula());
|
||||||
|
|
||||||
|
// check that it re-serializes to the same data
|
||||||
|
byte[] ser = record.serialize();
|
||||||
|
TestcaseRecordInputStream.confirmRecordEncoding(ObjRecord.sid, data, ser);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test data produced by OpenOffice 3.1 by opening and saving 47701.xls
|
||||||
|
* There are 5 padding bytes that are removed by POI
|
||||||
|
*/
|
||||||
|
public void test_remove_padding(){
|
||||||
|
byte[] data = HexRead.readFromString(
|
||||||
|
"5D, 00, 4C, 00, " +
|
||||||
|
"15, 00, 12, 00, 12, 00, 01, 00, 11, 00, " +
|
||||||
|
"00, 00, 00, 00, 00, 00, 00, 00, 00, 00, " +
|
||||||
|
"00, 00, 0C, 00, 14, 00, 00, 00, 00, 00, " +
|
||||||
|
"00, 00, 00, 00, 00, 00, 01, 00, 09, 00, " +
|
||||||
|
"00, 00, 0F, 00, 01, 00, " +
|
||||||
|
"13, 00, 1B, 00, " +
|
||||||
|
"10, 00, " + //next 16 bytes is a ptg aray
|
||||||
|
"09, 00, 00, 00, 00, 00, 25, 09, 00, 0C, 00, 07, 00, 07, 00, 00, " +
|
||||||
|
"01, 00, " + //num lines
|
||||||
|
"00, 00, " + //selected
|
||||||
|
"08, 00, " +
|
||||||
|
"00, 00, " +
|
||||||
|
"00, 00, 00, 00, 00"); //padding bytes
|
||||||
|
|
||||||
|
RecordInputStream in = TestcaseRecordInputStream.create(data);
|
||||||
|
// check read OK
|
||||||
|
ObjRecord record = new ObjRecord(in);
|
||||||
|
// check that it re-serializes to the same data
|
||||||
|
byte[] ser = record.serialize();
|
||||||
|
|
||||||
|
assertEquals(data.length-5, ser.length);
|
||||||
|
for(int i=0; i < ser.length; i++) assertEquals(data[i],ser[i]);
|
||||||
|
|
||||||
|
//check we can read the trimmed record
|
||||||
|
RecordInputStream in2 = TestcaseRecordInputStream.create(ser);
|
||||||
|
ObjRecord record2 = new ObjRecord(in2);
|
||||||
|
byte[] ser2 = record2.serialize();
|
||||||
|
assertTrue(Arrays.equals(ser, ser2));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void test_LbsDropData(){
|
||||||
|
byte[] data = HexRead.readFromString(
|
||||||
|
//LbsDropData
|
||||||
|
"0A, 00, " + //flags
|
||||||
|
"14, 00, " + //the number of lines to be displayed in the dropdown
|
||||||
|
"6C, 00, " + //the smallest width in pixels allowed for the dropdown window
|
||||||
|
"00, 00, " + //num chars
|
||||||
|
"00, " + //compression flag
|
||||||
|
"00"); //padding byte
|
||||||
|
|
||||||
|
LittleEndianInputStream in = new LittleEndianInputStream(new ByteArrayInputStream(data));
|
||||||
|
|
||||||
|
LbsDataSubRecord.LbsDropData lbs = new LbsDataSubRecord.LbsDropData(in);
|
||||||
|
|
||||||
|
ByteArrayOutputStream baos = new ByteArrayOutputStream();
|
||||||
|
lbs.serialize(new LittleEndianOutputStream(baos));
|
||||||
|
|
||||||
|
assertTrue(Arrays.equals(data, baos.toByteArray()));
|
||||||
|
}
|
||||||
|
}
|
|
@ -1522,4 +1522,12 @@ public final class TestBugs extends BaseTestBugzillaIssues {
|
||||||
HSSFCell cell2 = s.getRow(0).getCell(1);
|
HSSFCell cell2 = s.getRow(0).getCell(1);
|
||||||
assertEquals(1.0, cell2.getNumericCellValue());
|
assertEquals(1.0, cell2.getNumericCellValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POI 3.5 beta 7 can not read excel file contain list box (Form Control)
|
||||||
|
*/
|
||||||
|
public void test47701() {
|
||||||
|
openSample("47701.xls");
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Binary file not shown.
Loading…
Reference in New Issue