Broken HPSF usage of POI's general little-endian classes fixed.
git-svn-id: https://svn.apache.org/repos/asf/jakarta/poi/trunk@352778 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
9102bc3f39
commit
b1fccf2aa9
@ -60,6 +60,7 @@ package org.apache.poi.contrib.poibrowser;
|
||||
|
||||
import java.io.*;
|
||||
import java.util.*;
|
||||
import org.apache.poi.hpsf.ClassID;
|
||||
|
||||
|
||||
|
||||
@ -141,6 +142,50 @@ public class Codec
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>Converts an int value (32-bit) into its hexadecimal
|
||||
* notation.</p>
|
||||
*/
|
||||
public static String hexEncode(final int i)
|
||||
{
|
||||
StringBuffer sb = new StringBuffer(8);
|
||||
sb.append((char) hexval[(i & 0xF0000000) >> 28]);
|
||||
sb.append((char) hexval[(i & 0x0F000000) >> 24]);
|
||||
sb.append((char) hexval[(i & 0x00F00000) >> 20]);
|
||||
sb.append((char) hexval[(i & 0x000F0000) >> 16]);
|
||||
sb.append((char) hexval[(i & 0x0000F000) >> 12]);
|
||||
sb.append((char) hexval[(i & 0x00000F00) >> 8]);
|
||||
sb.append((char) hexval[(i & 0x000000F0) >> 4]);
|
||||
sb.append((char) hexval[(i & 0x0000000F) >> 0]);
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>Converts a long value (64-bit) into its hexadecimal
|
||||
* notation.</p>
|
||||
*/
|
||||
public static String hexEncode(final long l)
|
||||
{
|
||||
StringBuffer sb = new StringBuffer(16);
|
||||
sb.append((l & 0xFFFFFFFF00000000L) >> 32);
|
||||
sb.append((l & 0x00000000FFFFFFFFL) >> 0);
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>Converts a class ID into its hexadecimal notation.</p>
|
||||
*/
|
||||
public static String hexEncode(final ClassID classID)
|
||||
{
|
||||
return hexEncode(classID.getBytes());
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>Decodes the hexadecimal representation of a sequence of
|
||||
* bytes into a byte array. Each character in the string
|
||||
|
@ -159,7 +159,7 @@ public class POIBrowser extends JFrame
|
||||
new PropertySetDescriptorRenderer());
|
||||
treeUI.setCellRenderer(etcr);
|
||||
setSize(600, 450);
|
||||
setTitle("POI Browser 0.06");
|
||||
setTitle("POI Browser 0.07");
|
||||
setVisible(true);
|
||||
}
|
||||
|
||||
|
@ -85,9 +85,6 @@ public class PropertySetDescriptorRenderer extends DocumentDescriptorRenderer
|
||||
final int row,
|
||||
final boolean hasFocus)
|
||||
{
|
||||
|
||||
throw new RuntimeException("THIS FUNCTION BROKEN -- FIX IT");
|
||||
/*
|
||||
final PropertySetDescriptor d = (PropertySetDescriptor)
|
||||
((DefaultMutableTreeNode) value).getUserObject();
|
||||
final PropertySet ps = d.getPropertySet();
|
||||
@ -110,9 +107,8 @@ public class PropertySetDescriptorRenderer extends DocumentDescriptorRenderer
|
||||
|
||||
if (ps instanceof SummaryInformation)
|
||||
{
|
||||
*/
|
||||
/* Use the convenience methods. */
|
||||
/* final SummaryInformation si = (SummaryInformation) ps;
|
||||
final SummaryInformation si = (SummaryInformation) ps;
|
||||
text.append("\n");
|
||||
text.append("\nTitle: " + si.getTitle());
|
||||
text.append("\nSubject: " + si.getSubject());
|
||||
@ -136,7 +132,7 @@ public class PropertySetDescriptorRenderer extends DocumentDescriptorRenderer
|
||||
|
||||
if (selected)
|
||||
Util.invert(text);
|
||||
return p;*/
|
||||
return p;
|
||||
}
|
||||
|
||||
|
||||
@ -164,12 +160,9 @@ public class PropertySetDescriptorRenderer extends DocumentDescriptorRenderer
|
||||
*/
|
||||
protected String toString(final Section s, final String name)
|
||||
{
|
||||
|
||||
throw new RuntimeException("THIS FUNCTION BROKEN -- FIX IT");
|
||||
/*
|
||||
final StringBuffer b = new StringBuffer();
|
||||
b.append("\n" + name + " Format ID: ");
|
||||
b.append(Integer.toHexString(s.getFormatID()));
|
||||
b.append(Codec.hexEncode(s.getFormatID()));
|
||||
b.append("\n" + name + " Offset: " + s.getOffset());
|
||||
b.append("\n" + name + " Section size: " + s.getSize());
|
||||
b.append("\n" + name + " Property count: " + s.getPropertyCount());
|
||||
@ -195,7 +188,6 @@ public class PropertySetDescriptorRenderer extends DocumentDescriptorRenderer
|
||||
b.append(value.toString());
|
||||
}
|
||||
return b.toString();
|
||||
*/
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -59,83 +59,93 @@ import java.io.*;
|
||||
import org.apache.poi.util.LittleEndian;
|
||||
|
||||
/**
|
||||
* REWRITE ME
|
||||
* <p>
|
||||
* Represents a class ID (16 bytes). Unlike other little-endian type the {@link
|
||||
* ClassID} is not just 16 bytes stored in the wrong order. Instead, it is a
|
||||
* double word (4 bytes) followed by two words (2 bytes each) followed by 8
|
||||
* bytes.</p>
|
||||
* <p>Represents a class ID (16 bytes). Unlike other little-endian
|
||||
* type the {@link ClassID} is not just 16 bytes stored in the wrong
|
||||
* order. Instead, it is a double word (4 bytes) followed by two
|
||||
* words (2 bytes each) followed by 8 bytes.</p>
|
||||
*
|
||||
*@author Rainer Klute (klute@rainer-klute.de)
|
||||
*@created May 10, 2002
|
||||
*@see LittleEndian
|
||||
*@version $Id$
|
||||
*@since 2002-02-09
|
||||
* @author Rainer Klute (klute@rainer-klute.de)
|
||||
* @version $Id$
|
||||
* @since 2002-02-09
|
||||
*/
|
||||
public class ClassID {
|
||||
public class ClassID
|
||||
{
|
||||
|
||||
/**
|
||||
* <p>
|
||||
*
|
||||
* Creates a {@link ClassID} and reads its value from a byte array.</p>
|
||||
*
|
||||
*@param src The byte array to read from.
|
||||
*@param offset The offset of the first byte to read.
|
||||
* <p>The bytes making out the class ID in correct order,
|
||||
* i.e. big-endian.</p>
|
||||
*/
|
||||
public ClassID(final byte[] src, final int offset) {
|
||||
// super(src, offset);
|
||||
protected byte[] bytes;
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>Creates a {@link ClassID} and reads its value from a byte
|
||||
* array.</p>
|
||||
*
|
||||
* @param src The byte array to read from.
|
||||
* @param offset The offset of the first byte to read.
|
||||
*/
|
||||
public ClassID(final byte[] src, final int offset)
|
||||
{
|
||||
read(src, offset);
|
||||
}
|
||||
|
||||
|
||||
|
||||
public final static int LENGTH = 16;
|
||||
|
||||
public int length() {
|
||||
public int length()
|
||||
{
|
||||
return LENGTH;
|
||||
}
|
||||
|
||||
public byte[] getBytes() {
|
||||
|
||||
throw new RuntimeException("This fucntion must be rewritten");
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Description of the Method - REWRITE ME REWRITE ME REWRITE ME
|
||||
* ISNT += offset a bug? -- doesn't the order of operations evaluate that
|
||||
* last?
|
||||
*
|
||||
*@param src Description of the Parameter
|
||||
*@param offset Description of the Parameter
|
||||
*@return Description of the Return Value
|
||||
* <p>Gets the bytes making out the class ID. They are returned in
|
||||
* correct order, i.e. big-endian.</p>
|
||||
*/
|
||||
public byte[] read(byte[] src, int offset) {
|
||||
byte[] retval = new byte[24];
|
||||
public byte[] getBytes()
|
||||
{
|
||||
return bytes;
|
||||
}
|
||||
|
||||
//throw new RuntimeException("This fucntion must be rewritten");
|
||||
|
||||
//Number[] b = new Number[11];
|
||||
|
||||
//b[0] = new Integer(LittleEndian.getInt(src, offset));
|
||||
//transfer the first Int from little to big endian
|
||||
retval[0] = src[3];
|
||||
retval[1] = src[2];
|
||||
retval[2] = src[1];
|
||||
retval[3] = src[0];
|
||||
/**
|
||||
* <p>Reads a class ID from a byte array by turning little-endian
|
||||
* into big-endian.</p>
|
||||
*
|
||||
* @param src The byte array to read from
|
||||
*
|
||||
* @param offset The offset within the <var>src</var> byte array
|
||||
*
|
||||
* @return A byte array containing the class ID.
|
||||
*/
|
||||
public byte[] read(final byte[] src, final int offset)
|
||||
{
|
||||
bytes = new byte[16];
|
||||
|
||||
//b[1] = new Short(LittleEndian.getInt(src, offset += LittleEndian.INT_SIZE));
|
||||
//transfer the second short from little to big endian
|
||||
retval[4] = src[5];
|
||||
retval[5] = src[4];
|
||||
/* Read double word. */
|
||||
bytes[0] = src[3 + offset];
|
||||
bytes[1] = src[2 + offset];
|
||||
bytes[2] = src[1 + offset];
|
||||
bytes[3] = src[0 + offset];
|
||||
|
||||
//b[2] = new Short(LittleEndian.getInt(src, offset += LittleEndian.SHORT_SIZE));
|
||||
//transfer the third short from little to big endian
|
||||
retval[6] = src[7];
|
||||
retval[7] = src[6];
|
||||
/* Read first word. */
|
||||
bytes[4] = src[5 + offset];
|
||||
bytes[5] = src[4 + offset];
|
||||
|
||||
System.arraycopy(src, 8, retval, 8, retval.length - 8);
|
||||
/* Read second word. */
|
||||
bytes[6] = src[7 + offset];
|
||||
bytes[7] = src[6 + offset];
|
||||
|
||||
return retval;
|
||||
/* Read 8 bytes. */
|
||||
for (int i = 8; i < 16; i++)
|
||||
bytes[i] = src[i + offset];
|
||||
|
||||
return bytes;
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -95,7 +95,7 @@ import org.apache.poi.poifs.filesystem.*;
|
||||
*/
|
||||
public class PropertySet {
|
||||
final static byte[] BYTE_ORDER_ASSERTION =
|
||||
new byte[]{(byte) 0xFF, (byte) 0xFE};
|
||||
new byte[]{(byte) 0xFE, (byte) 0xFF};
|
||||
final static byte[] FORMAT_ASSERTION =
|
||||
new byte[]{(byte) 0x00, (byte) 0x00};
|
||||
|
||||
|
@ -59,23 +59,19 @@ import org.apache.poi.util.LittleEndian;
|
||||
import org.apache.poi.hpsf.wellknown.*;
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Represents a section in a {@link PropertySet}.</p>
|
||||
*
|
||||
* Represents a section in a {@link PropertySet}.</p>
|
||||
*
|
||||
*@author Rainer Klute (klute@rainer-klute.de)
|
||||
*@author Drew Varner (Drew.Varner allUpIn sc.edu)
|
||||
*@created May 10, 2002
|
||||
*@version $Id$
|
||||
*@since 2002-02-09
|
||||
* @author Rainer Klute (klute@rainer-klute.de)
|
||||
* @author Drew Varner (Drew.Varner allUpIn sc.edu)
|
||||
* @version $Id$
|
||||
* @since 2002-02-09
|
||||
*/
|
||||
public class Section {
|
||||
public class Section
|
||||
{
|
||||
|
||||
/**
|
||||
* <p>
|
||||
*
|
||||
* Maps property IDs to section-private PID strings. These strings can be
|
||||
* found in the property with ID 0.</p>
|
||||
* <p>Maps property IDs to section-private PID strings. These
|
||||
* strings can be found in the property with ID 0.</p>
|
||||
*/
|
||||
protected Map dictionary;
|
||||
|
||||
@ -83,13 +79,13 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the format ID. The format ID is the "type" of the
|
||||
* section.</p>
|
||||
*
|
||||
* Returns the format ID. The format ID is the "type" of the section.</p>
|
||||
*
|
||||
*@return The formatID value
|
||||
* @return The format ID
|
||||
*/
|
||||
public ClassID getFormatID() {
|
||||
public ClassID getFormatID()
|
||||
{
|
||||
return formatID;
|
||||
}
|
||||
|
||||
@ -99,13 +95,12 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the offset of the section in the stream.</p>
|
||||
*
|
||||
* Returns the offset of the section in the stream.</p>
|
||||
*
|
||||
*@return The offset value
|
||||
* @return The offset of the section in the stream.
|
||||
*/
|
||||
public long getOffset() {
|
||||
public long getOffset()
|
||||
{
|
||||
return offset;
|
||||
}
|
||||
|
||||
@ -115,13 +110,12 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the section's size in bytes.</p>
|
||||
*
|
||||
* Returns the section's size in bytes.</p>
|
||||
*
|
||||
*@return The size value
|
||||
* @return The section's size in bytes.
|
||||
*/
|
||||
public int getSize() {
|
||||
public int getSize()
|
||||
{
|
||||
return size;
|
||||
}
|
||||
|
||||
@ -131,13 +125,12 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the number of properties in this section.</p>
|
||||
*
|
||||
* Returns the number of properties in this section.</p>
|
||||
*
|
||||
*@return The propertyCount value
|
||||
* @return The number of properties in this section.
|
||||
*/
|
||||
public int getPropertyCount() {
|
||||
public int getPropertyCount()
|
||||
{
|
||||
return propertyCount;
|
||||
}
|
||||
|
||||
@ -147,28 +140,26 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns this section's properties.</p>
|
||||
*
|
||||
* Returns this section's properties.</p>
|
||||
*
|
||||
*@return The properties value
|
||||
* @return This section's properties.
|
||||
*/
|
||||
public Property[] getProperties() {
|
||||
public Property[] getProperties()
|
||||
{
|
||||
return properties;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Creates a {@link Section} instance from a byte array.</p>
|
||||
*
|
||||
* Creates a {@link Section} instance from a byte array.</p>
|
||||
*
|
||||
*@param src Contains the complete property set stream.
|
||||
*@param offset The position in the stream that points to the section's
|
||||
* format ID.
|
||||
* @param src Contains the complete property set stream.
|
||||
* @param offset The position in the stream that points to the
|
||||
* section's format ID.
|
||||
*/
|
||||
public Section(final byte[] src, int offset) {
|
||||
public Section(final byte[] src, int offset)
|
||||
{
|
||||
/*
|
||||
* Read the format ID.
|
||||
*/
|
||||
@ -217,7 +208,8 @@ public class Section {
|
||||
length = (int)(src.length - this.offset - sOffset);
|
||||
} else {
|
||||
length = (int)
|
||||
LittleEndian.getUInt(src, offset + LittleEndian.INT_SIZE) - sOffset;
|
||||
LittleEndian.getUInt(src, offset + LittleEndian.INT_SIZE) -
|
||||
sOffset;
|
||||
}
|
||||
|
||||
/*
|
||||
@ -236,22 +228,21 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the value of the property with the specified ID. If
|
||||
* the property is not available, <code>null</code> is returned
|
||||
* and a subsequent call to {@link #wasNull} will return
|
||||
* <code>true</code>.</p>
|
||||
*
|
||||
* Returns the value of the property with the specified ID. If the property
|
||||
* is not available, <code>null</code> is returned and a subsequent call to
|
||||
* {@link #wasNull} will return <code>true</code>.</p>
|
||||
* @param id The property's ID
|
||||
*
|
||||
*@param id Description of the Parameter
|
||||
*@return The property value
|
||||
* @return The property's value
|
||||
*/
|
||||
protected Object getProperty(final int id) {
|
||||
protected Object getProperty(final int id)
|
||||
{
|
||||
wasNull = false;
|
||||
for (int i = 0; i < properties.length; i++) {
|
||||
if (id == properties[i].getID()) {
|
||||
for (int i = 0; i < properties.length; i++)
|
||||
if (id == properties[i].getID())
|
||||
return properties[i].getValue();
|
||||
}
|
||||
}
|
||||
wasNull = true;
|
||||
return null;
|
||||
}
|
||||
@ -259,47 +250,48 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the value of the numeric property with the specified
|
||||
* ID. If the property is not available, 0 is returned. A
|
||||
* subsequent call to {@link #wasNull} will return
|
||||
* <code>true</code> to let the caller distinguish that case from
|
||||
* a real property value of 0.</p>
|
||||
*
|
||||
* Returns the value of the numeric property with the specified ID. If the
|
||||
* property is not available, 0 is returned. A subsequent call to {@link
|
||||
* #wasNull} will return <code>true</code> to let the caller distinguish
|
||||
* that case from a real property value of 0.</p>
|
||||
* @param id The property's ID
|
||||
*
|
||||
*@param id Description of the Parameter
|
||||
*@return The propertyIntValue value
|
||||
* @return The property's value
|
||||
*/
|
||||
protected int getPropertyIntValue(final int id) {
|
||||
final Integer i = (Integer) getProperty(id);
|
||||
if (i != null) {
|
||||
protected int getPropertyIntValue(final int id)
|
||||
{
|
||||
/* FIXME: Find out why the following is a Long instead of an
|
||||
* Integer! */
|
||||
final Long i = (Long) getProperty(id);
|
||||
if (i != null)
|
||||
return i.intValue();
|
||||
} else {
|
||||
else
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the value of the boolean property with the specified
|
||||
* ID. If the property is not available, <code>false</code> is
|
||||
* returned. A subsequent call to {@link #wasNull} will return
|
||||
* <code>true</code> to let the caller distinguish that case from
|
||||
* a real property value of <code>false</code>.</p>
|
||||
*
|
||||
* Returns the value of the boolean property with the specified ID. If the
|
||||
* property is not available, <code>false</code> is returned. A subsequent
|
||||
* call to {@link #wasNull} will return <code>true</code> to let the caller
|
||||
* distinguish that case from a real property value of <code>false</code>.
|
||||
* </p>
|
||||
* @param id The property's ID
|
||||
*
|
||||
*@param id Description of the Parameter
|
||||
*@return The propertyBooleanValue value
|
||||
* @return The property's value
|
||||
*/
|
||||
protected boolean getPropertyBooleanValue(final int id) {
|
||||
protected boolean getPropertyBooleanValue(final int id)
|
||||
{
|
||||
final Boolean b = (Boolean) getProperty(id);
|
||||
if (b != null) {
|
||||
if (b != null)
|
||||
return b.booleanValue();
|
||||
} else {
|
||||
else
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
@ -307,46 +299,44 @@ public class Section {
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Checks whether the property which the last call to {@link
|
||||
* #getPropertyIntValue} or {@link #getProperty} tried to access
|
||||
* was available or not. This information might be important for
|
||||
* callers of {@link #getPropertyIntValue} since the latter
|
||||
* returns 0 if the property does not exist. Using {@link
|
||||
* #wasNull} the caller can distiguish this case from a property's
|
||||
* real value of 0.</p>
|
||||
*
|
||||
* Checks whether the property which the last call to {@link
|
||||
* #getPropertyIntValue} or {@link #getProperty} tried to access was
|
||||
* available or not. This information might be important for callers of
|
||||
* {@link #getPropertyIntValue} since the latter returns 0 if the property
|
||||
* does not exist. Using {@link #wasNull} the caller can distiguish this
|
||||
* case from a property's real value of 0.</p>
|
||||
*
|
||||
*@return <code>true</code> if the last call to {@link
|
||||
* #getPropertyIntValue} or {@link #getProperty} tried to access a
|
||||
* property that was not available, else <code>false</code>.
|
||||
* @return <code>true</code> if the last call to {@link
|
||||
* #getPropertyIntValue} or {@link #getProperty} tried to access a
|
||||
* property that was not available, else <code>false</code>.
|
||||
*/
|
||||
public boolean wasNull() {
|
||||
public boolean wasNull()
|
||||
{
|
||||
return wasNull;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* <p>
|
||||
* <p>Returns the PID string associated with a property ID. The ID
|
||||
* is first looked up in the {@link Section}'s private
|
||||
* dictionary. If it is not found there, the method calls {@link
|
||||
* SectionIDMap#getPIDString}.</p>
|
||||
*
|
||||
* Returns the PID string associated with a property ID. The ID is first
|
||||
* looked up in the {@link Section}'s private dictionary. If it is not
|
||||
* found there, the method calls {@link SectionIDMap#getPIDString}.</p>
|
||||
* @param pid The property ID
|
||||
*
|
||||
*@param pid Description of the Parameter
|
||||
*@return The pIDString value
|
||||
* @return The property ID's string value
|
||||
*/
|
||||
public String getPIDString(final int pid) {
|
||||
public String getPIDString(final int pid)
|
||||
{
|
||||
String s = null;
|
||||
if (dictionary != null) {
|
||||
if (dictionary != null)
|
||||
s = (String) dictionary.get(new Integer(pid));
|
||||
}
|
||||
if (s == null) {
|
||||
if (s == null)
|
||||
s = SectionIDMap.getPIDString(getFormatID().getBytes(), pid);
|
||||
}
|
||||
if (s == null) {
|
||||
if (s == null)
|
||||
s = SectionIDMap.UNDEFINED;
|
||||
}
|
||||
return s;
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user