All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.poi.hwmf.record.HwmfBitmapDib Maven / Gradle / Ivy

There is a newer version: 5.2.5
Show newest version
/* ====================================================================
   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.hwmf.record;

import javax.imageio.ImageIO;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;

import org.apache.poi.util.IOUtils;
import org.apache.poi.util.LittleEndian;
import org.apache.poi.util.LittleEndianConsts;
import org.apache.poi.util.LittleEndianInputStream;
import org.apache.poi.util.POILogFactory;
import org.apache.poi.util.POILogger;
import org.apache.poi.util.RecordFormatException;

/**
 * The DeviceIndependentBitmap Object defines an image in device-independent bitmap (DIB) format.
 */
public class HwmfBitmapDib {

    private static final int MAX_RECORD_LENGTH = 10000000;

    public static enum BitCount {
        /**
         * The image SHOULD be in either JPEG or PNG format. <6> Neither of these formats includes
         *  a color table, so this value specifies that no color table is present. See [JFIF] and [RFC2083]
         *  for more information concerning JPEG and PNG compression formats.
         */
        BI_BITCOUNT_0(0x0000),
        /**
         * Each pixel in the bitmap is represented by a single bit. If the bit is clear, the pixel is displayed
         *  with the color of the first entry in the color table; if the bit is set, the pixel has the color of the
         *  second entry in the table.
         */
        BI_BITCOUNT_1(0x0001),
        /**
         * Each pixel in the bitmap is represented by a 4-bit index into the color table, and each byte
         *  contains 2 pixels.
         */
        BI_BITCOUNT_2(0x0004),
        /**
         * Each pixel in the bitmap is represented by an 8-bit index into the color table, and each byte
         *  contains 1 pixel.
         */
        BI_BITCOUNT_3(0x0008),
        /**
         * Each pixel in the bitmap is represented by a 16-bit value.
         * 
* If the Compression field of the BitmapInfoHeader Object is BI_RGB, the Colors field of DIB * is NULL. Each WORD in the bitmap array represents a single pixel. The relative intensities of * red, green, and blue are represented with 5 bits for each color component. The value for blue * is in the least significant 5 bits, followed by 5 bits each for green and red. The most significant * bit is not used. The color table is used for optimizing colors on palette-based devices, and * contains the number of entries specified by the ColorUsed field of the BitmapInfoHeader * Object. *
* If the Compression field of the BitmapInfoHeader Object is BI_BITFIELDS, the Colors field * contains three DWORD color masks that specify the red, green, and blue components, * respectively, of each pixel. Each WORD in the bitmap array represents a single pixel. *
* When the Compression field is set to BI_BITFIELDS, bits set in each DWORD mask MUST be * contiguous and SHOULD NOT overlap the bits of another mask. */ BI_BITCOUNT_4(0x0010), /** * The bitmap has a maximum of 2^24 colors, and the Colors field of DIB is * NULL. Each 3-byte triplet in the bitmap array represents the relative intensities of blue, green, * and red, respectively, for a pixel. The Colors color table is used for optimizing colors used on * palette-based devices, and MUST contain the number of entries specified by the ColorUsed * field of the BitmapInfoHeader Object. */ BI_BITCOUNT_5(0x0018), /** * The bitmap has a maximum of 2^24 colors. *
* If the Compression field of the BitmapInfoHeader Object is set to BI_RGB, the Colors field * of DIB is set to NULL. Each DWORD in the bitmap array represents the relative intensities of * blue, green, and red, respectively, for a pixel. The high byte in each DWORD is not used. The * Colors color table is used for optimizing colors used on palette-based devices, and MUST * contain the number of entries specified by the ColorUsed field of the BitmapInfoHeader * Object. *
* If the Compression field of the BitmapInfoHeader Object is set to BI_BITFIELDS, the Colors * field contains three DWORD color masks that specify the red, green, and blue components, * respectively, of each pixel. Each DWORD in the bitmap array represents a single pixel. *
* When the Compression field is set to BI_BITFIELDS, bits set in each DWORD mask must be * contiguous and should not overlap the bits of another mask. All the bits in the pixel do not * need to be used. */ BI_BITCOUNT_6(0x0020); int flag; BitCount(int flag) { this.flag = flag; } static BitCount valueOf(int flag) { for (BitCount bc : values()) { if (bc.flag == flag) return bc; } return null; } } public static enum Compression { /** * The bitmap is in uncompressed red green blue (RGB) format that is not compressed * and does not use color masks. */ BI_RGB(0x0000), /** * An RGB format that uses run-length encoding (RLE) compression for bitmaps * with 8 bits per pixel. The compression uses a 2-byte format consisting of a count byte * followed by a byte containing a color index. */ BI_RLE8(0x0001), /** * An RGB format that uses RLE compression for bitmaps with 4 bits per pixel. The * compression uses a 2-byte format consisting of a count byte followed by two word-length * color indexes. */ BI_RLE4(0x0002), /** * The bitmap is not compressed and the color table consists of three DWORD * color masks that specify the red, green, and blue components, respectively, of each pixel. * This is valid when used with 16 and 32-bits per pixel bitmaps. */ BI_BITFIELDS(0x0003), /** * The image is a JPEG image, as specified in [JFIF]. This value SHOULD only be used in * certain bitmap operations, such as JPEG pass-through. The application MUST query for the * pass-through support, since not all devices support JPEG pass-through. Using non-RGB * bitmaps MAY limit the portability of the metafile to other devices. For instance, display device * contexts generally do not support this pass-through. */ BI_JPEG(0x0004), /** * The image is a PNG image, as specified in [RFC2083]. This value SHOULD only be * used certain bitmap operations, such as JPEG/PNG pass-through. The application MUST query * for the pass-through support, because not all devices support JPEG/PNG pass-through. Using * non-RGB bitmaps MAY limit the portability of the metafile to other devices. For instance, * display device contexts generally do not support this pass-through. */ BI_PNG(0x0005), /** * The image is an uncompressed CMYK format. */ BI_CMYK(0x000B), /** * A CMYK format that uses RLE compression for bitmaps with 8 bits per pixel. * The compression uses a 2-byte format consisting of a count byte followed by a byte containing * a color index. */ BI_CMYKRLE8(0x000C), /** * A CMYK format that uses RLE compression for bitmaps with 4 bits per pixel. * The compression uses a 2-byte format consisting of a count byte followed by two word-length * color indexes. */ BI_CMYKRLE4(0x000D); int flag; Compression(int flag) { this.flag = flag; } static Compression valueOf(int flag) { for (Compression c : values()) { if (c.flag == flag) return c; } return null; } } private final static POILogger logger = POILogFactory.getLogger(HwmfBitmapDib.class); private static final int BMP_HEADER_SIZE = 14; private int headerSize; private int headerWidth; private int headerHeight; private int headerPlanes; private BitCount headerBitCount; private Compression headerCompression; private long headerImageSize = -1; @SuppressWarnings("unused") private int headerXPelsPerMeter = -1; @SuppressWarnings("unused") private int headerYPelsPerMeter = -1; private long headerColorUsed = -1; @SuppressWarnings("unused") private long headerColorImportant = -1; private Color colorTable[]; @SuppressWarnings("unused") private int colorMaskR,colorMaskG,colorMaskB; // size of header and color table, for start of image data calculation private int introSize; private byte imageData[]; public int init(LittleEndianInputStream leis, int recordSize) throws IOException { leis.mark(10000); // need to read the header to calculate start of bitmap data correct introSize = readHeader(leis); assert(introSize == headerSize); introSize += readColors(leis); assert(introSize < 10000); int fileSize = (headerImageSize < headerSize) ? recordSize : (int)Math.min(introSize+headerImageSize,recordSize); leis.reset(); imageData = IOUtils.toByteArray(leis, fileSize); assert( headerSize != 0x0C || ((((headerWidth * headerPlanes * headerBitCount.flag + 31) & ~31) / 8) * Math.abs(headerHeight)) == headerImageSize); return fileSize; } protected int readHeader(LittleEndianInputStream leis) throws IOException { int size = 0; /** * DIBHeaderInfo (variable): Either a BitmapCoreHeader Object or a * BitmapInfoHeader Object that specifies information about the image. * * The first 32 bits of this field is the HeaderSize value. * If it is 0x0000000C, then this is a BitmapCoreHeader; otherwise, this is a BitmapInfoHeader. */ headerSize = leis.readInt(); size += LittleEndianConsts.INT_SIZE; if (headerSize == 0x0C) { // BitmapCoreHeader // A 16-bit unsigned integer that defines the width of the DIB, in pixels. headerWidth = leis.readUShort(); // A 16-bit unsigned integer that defines the height of the DIB, in pixels. headerHeight = leis.readUShort(); // A 16-bit unsigned integer that defines the number of planes for the target // device. This value MUST be 0x0001. headerPlanes = leis.readUShort(); // A 16-bit unsigned integer that defines the format of each pixel, and the // maximum number of colors in the DIB. headerBitCount = BitCount.valueOf(leis.readUShort()); size += 4*LittleEndianConsts.SHORT_SIZE; } else { // BitmapInfoHeader // A 32-bit signed integer that defines the width of the DIB, in pixels. // This value MUST be positive. // This field SHOULD specify the width of the decompressed image file, // if the Compression value specifies JPEG or PNG format. headerWidth = leis.readInt(); // A 32-bit signed integer that defines the height of the DIB, in pixels. // This value MUST NOT be zero. // - If this value is positive, the DIB is a bottom-up bitmap, // and its origin is the lower-left corner. // This field SHOULD specify the height of the decompressed image file, // if the Compression value specifies JPEG or PNG format. // - If this value is negative, the DIB is a top-down bitmap, // and its origin is the upper-left corner. Top-down bitmaps do not support compression. headerHeight = leis.readInt(); // A 16-bit unsigned integer that defines the number of planes for the target // device. This value MUST be 0x0001. headerPlanes = leis.readUShort(); // A 16-bit unsigned integer that defines the format of each pixel, and the // maximum number of colors in the DIB. headerBitCount = BitCount.valueOf(leis.readUShort()); // A 32-bit unsigned integer that defines the compression mode of the DIB. // This value MUST NOT specify a compressed format if the DIB is a top-down bitmap, // as indicated by the Height value. headerCompression = Compression.valueOf((int)leis.readUInt()); // A 32-bit unsigned integer that defines the size, in bytes, of the image. // If the Compression value is BI_RGB, this value SHOULD be zero and MUST be ignored. // If the Compression value is BI_JPEG or BI_PNG, this value MUST specify the size of the JPEG // or PNG image buffer, respectively. headerImageSize = leis.readUInt(); // A 32-bit signed integer that defines the horizontal resolution, // in pixels-per-meter, of the target device for the DIB. headerXPelsPerMeter = leis.readInt(); // A 32-bit signed integer that defines the vertical resolution, headerYPelsPerMeter = leis.readInt(); // A 32-bit unsigned integer that specifies the number of indexes in the // color table used by the DIB // in pixelsper-meter, of the target device for the DIB. headerColorUsed = leis.readUInt(); // A 32-bit unsigned integer that defines the number of color indexes that are // required for displaying the DIB. If this value is zero, all color indexes are required. headerColorImportant = leis.readUInt(); size += 8*LittleEndianConsts.INT_SIZE+2*LittleEndianConsts.SHORT_SIZE; } assert(size == headerSize); return size; } protected int readColors(LittleEndianInputStream leis) throws IOException { switch (headerBitCount) { default: case BI_BITCOUNT_0: // no table return 0; case BI_BITCOUNT_1: // 2 colors return readRGBQuad(leis, (int)(headerColorUsed == 0 ? 2 : Math.min(headerColorUsed,2))); case BI_BITCOUNT_2: // 16 colors return readRGBQuad(leis, (int)(headerColorUsed == 0 ? 16 : Math.min(headerColorUsed,16))); case BI_BITCOUNT_3: // 256 colors return readRGBQuad(leis, (int)(headerColorUsed == 0 ? 256 : Math.min(headerColorUsed,256))); case BI_BITCOUNT_4: switch (headerCompression) { case BI_RGB: colorMaskB = 0x1F; colorMaskG = 0x1F<<5; colorMaskR = 0x1F<<10; return 0; case BI_BITFIELDS: colorMaskB = leis.readInt(); colorMaskG = leis.readInt(); colorMaskR = leis.readInt(); return 3*LittleEndianConsts.INT_SIZE; default: throw new IOException("Invalid compression option ("+headerCompression+") for bitcount ("+headerBitCount+")."); } case BI_BITCOUNT_5: case BI_BITCOUNT_6: switch (headerCompression) { case BI_RGB: colorMaskR=0xFF; colorMaskG=0xFF; colorMaskB=0xFF; return 0; case BI_BITFIELDS: colorMaskB = leis.readInt(); colorMaskG = leis.readInt(); colorMaskR = leis.readInt(); return 3*LittleEndianConsts.INT_SIZE; default: throw new IOException("Invalid compression option ("+headerCompression+") for bitcount ("+headerBitCount+")."); } } } protected int readRGBQuad(LittleEndianInputStream leis, int count) throws IOException { int size = 0; colorTable = new Color[count]; for (int i=0; i




© 2015 - 2024 Weber Informatics LLC | Privacy Policy