Package org.apache.poi.hssf.record.aggregates

Source Code of org.apache.poi.hssf.record.aggregates.TestPageSettingsBlock

/* ====================================================================
   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.aggregates;

import java.util.Arrays;

import junit.framework.AssertionFailedError;
import junit.framework.TestCase;

import org.apache.poi.hssf.HSSFTestDataSamples;
import org.apache.poi.hssf.model.RecordStream;
import org.apache.poi.hssf.model.InternalSheet;
import org.apache.poi.hssf.record.*;
import org.apache.poi.hssf.usermodel.HSSFPrintSetup;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.hssf.usermodel.RecordInspector.RecordCollector;
import org.apache.poi.util.HexRead;

/**
* Tess for {@link PageSettingsBlock}
*
* @author Dmitriy Kumshayev
*/
public final class TestPageSettingsBlock extends TestCase {

  public void testPrintSetup_bug46548() {

    // PageSettingBlock in this file contains PLS (sid=x004D) record
    // followed by ContinueRecord (sid=x003C)
    HSSFWorkbook wb = HSSFTestDataSamples.openSampleWorkbook("ex46548-23133.xls");
    HSSFSheet sheet = wb.getSheetAt(0);
    HSSFPrintSetup ps = sheet.getPrintSetup();

    try {
      ps.getCopies();
    } catch (NullPointerException e) {
      e.printStackTrace();
      throw new AssertionFailedError("Identified bug 46548: PageSettingBlock missing PrintSetupRecord record");
    }
  }

  /**
   * Bug 46840 occurred because POI failed to recognise HEADERFOOTER as part of the
   * {@link PageSettingsBlock}.
   */
  public void testHeaderFooter_bug46840() {

    int rowIx = 5;
    int colIx = 6;
    NumberRecord nr = new NumberRecord();
    nr.setRow(rowIx);
    nr.setColumn((short) colIx);
    nr.setValue(3.0);

    Record[] recs = {
        BOFRecord.createSheetBOF(),
        new HeaderRecord("&LSales Figures"),
        new FooterRecord("&LJanuary"),
        new HeaderFooterRecord(HexRead.readFromString("9C 08 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 C4 60 00 00 00 00 00 00 00 00")),
        new DimensionsRecord(),
        new WindowTwoRecord(),
        new UserSViewBegin(HexRead.readFromString("ED 77 3B 86 BC 3F 37 4C A9 58 60 23 43 68 54 4B 01 00 00 00 64 00 00 00 40 00 00 00 02 00 00 00 3D 80 04 00 00 00 00 00 00 00 0C 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 F0 3F FF FF 01 00")),
        new HeaderRecord("&LSales Figures"),
        new FooterRecord("&LJanuary"),
        new HeaderFooterRecord(HexRead.readFromString("9C 08 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 C4 60 00 00 00 00 00 00 00 00")),
        new UserSViewEnd(HexRead.readFromString("01, 00")),

        EOFRecord.instance,
    };
    RecordStream rs = new RecordStream(Arrays.asList(recs), 0);
    InternalSheet sheet;
    try {
      sheet = InternalSheet.createSheet(rs);
    } catch (RuntimeException e) {
      if (e.getMessage().equals("two Page Settings Blocks found in the same sheet")) {
        throw new AssertionFailedError("Identified bug 46480");
      }
      throw e;
    }

    RecordCollector rv = new RecordCollector();
    sheet.visitContainedRecords(rv, rowIx);
    Record[] outRecs = rv.getRecords();
    assertEquals(13, outRecs.length);
  }

  /**
   * Bug 46953 occurred because POI didn't handle late PSB records properly.
   */
  public void testLateHeaderFooter_bug46953() {

    int rowIx = 5;
    int colIx = 6;
    NumberRecord nr = new NumberRecord();
    nr.setRow(rowIx);
    nr.setColumn((short) colIx);
    nr.setValue(3.0);

    Record[] recs = {
        BOFRecord.createSheetBOF(),
        new HeaderRecord("&LSales Figures"),
        new FooterRecord("&LJanuary"),
        new DimensionsRecord(),
        new WindowTwoRecord(),
        new HeaderFooterRecord(HexRead.readFromString("9C 08 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 C4 60 00 00 00 00 00 00 00 00")),
        EOFRecord.instance,
    };
    RecordStream rs = new RecordStream(Arrays.asList(recs), 0);
    InternalSheet sheet = InternalSheet.createSheet(rs);

    RecordCollector rv = new RecordCollector();
    sheet.visitContainedRecords(rv, 0);
    Record[] outRecs = rv.getRecords();
    if (outRecs[4] == EOFRecord.instance) {
      throw new AssertionFailedError("Identified bug 46953 - EOF incorrectly appended to PSB");
    }
    assertEquals(recs.length+1, outRecs.length); // +1 for index record

    assertEquals(BOFRecord.class, outRecs[0].getClass());
    assertEquals(IndexRecord.class, outRecs[1].getClass());
    assertEquals(HeaderRecord.class, outRecs[2].getClass());
    assertEquals(FooterRecord.class, outRecs[3].getClass());
    assertEquals(HeaderFooterRecord.class, outRecs[4].getClass());
    assertEquals(DimensionsRecord.class, outRecs[5].getClass());
    assertEquals(WindowTwoRecord.class, outRecs[6].getClass());
    assertEquals(EOFRecord.instance, outRecs[7]);
  }
  /**
   * Bug 47199 was due to the margin records being located well after the initial PSB records.
   * The example file supplied (attachment 23710) had three non-PSB record types
   * between the PRINTSETUP record and first MARGIN record:
   * <ul>
   * <li>PRINTSETUP(0x00A1)</li>
   * <li>DEFAULTCOLWIDTH(0x0055)</li>
   * <li>COLINFO(0x007D)</li>
   * <li>DIMENSIONS(0x0200)</li>
   * <li>BottomMargin(0x0029)</li>
   * </ul>
   */
  public void testLateMargins_bug47199() {

    Record[] recs = {
        BOFRecord.createSheetBOF(),
        new HeaderRecord("&LSales Figures"),
        new FooterRecord("&LJanuary"),
        new DimensionsRecord(),
        createBottomMargin(0.787F),
        new WindowTwoRecord(),
        EOFRecord.instance,
    };
    RecordStream rs = new RecordStream(Arrays.asList(recs), 0);

    InternalSheet sheet;
    try {
      sheet = InternalSheet.createSheet(rs);
    } catch (RuntimeException e) {
      if (e.getMessage().equals("two Page Settings Blocks found in the same sheet")) {
        throw new AssertionFailedError("Identified bug 47199a - failed to process late margings records");
      }
      throw e;
    }

    RecordCollector rv = new RecordCollector();
    sheet.visitContainedRecords(rv, 0);
    Record[] outRecs = rv.getRecords();
    assertEquals(recs.length+1, outRecs.length); // +1 for index record

    assertEquals(BOFRecord.class, outRecs[0].getClass());
    assertEquals(IndexRecord.class, outRecs[1].getClass());
    assertEquals(HeaderRecord.class, outRecs[2].getClass());
    assertEquals(FooterRecord.class, outRecs[3].getClass());
    assertEquals(DimensionsRecord.class, outRecs[5].getClass());
    assertEquals(WindowTwoRecord.class, outRecs[6].getClass());
    assertEquals(EOFRecord.instance, outRecs[7]);
  }

  private Record createBottomMargin(float value) {
    BottomMarginRecord result = new BottomMarginRecord();
    result.setMargin(value);
    return result;
  }

  /**
   * The PageSettingsBlock should not allow multiple copies of the same record.  This extra assertion
   * was added while fixing bug 47199.  All existing POI test samples comply with this requirement.
   */
  public void testDuplicatePSBRecord_bug47199() {

    // Hypothetical setup of PSB records which should cause POI to crash
    Record[] recs = {
        new HeaderRecord("&LSales Figures"),
        new HeaderRecord("&LInventory"),
    };
    RecordStream rs = new RecordStream(Arrays.asList(recs), 0);

    try {
      new PageSettingsBlock(rs);
      throw new AssertionFailedError("Identified bug 47199b - duplicate PSB records should not be allowed");
    } catch (RecordFormatException e) {
      if (e.getMessage().equals("Duplicate PageSettingsBlock record (sid=0x14)")) {
        // expected during successful test
      } else {
        throw new AssertionFailedError("Expected RecordFormatException due to duplicate PSB record");
      }
    }
  }

  private static UnknownRecord ur(int sid, String hexData) {
    return new UnknownRecord(sid, HexRead.readFromString(hexData));
  }

  /**
   * Excel tolerates missing header / footer records, but adds them (empty) in when re-saving.
   * This is not critical functionality but it has been decided to keep POI consistent with
   * Excel in this regard.
   */
  public void testMissingHeaderFooter() {
    // initialise PSB with some records, but not the header / footer
    Record[] recs = {
        new HCenterRecord(),
        new VCenterRecord(),
    };
    RecordStream rs = new RecordStream(Arrays.asList(recs), 0);
    PageSettingsBlock psb = new PageSettingsBlock(rs);

    // serialize the PSB to see what records come out
    RecordCollector rc = new RecordCollector();
    psb.visitContainedRecords(rc);
    Record[] outRecs = rc.getRecords();

    if (outRecs.length == 2) {
      throw new AssertionFailedError("PageSettingsBlock didn't add missing header/footer records");
    }
    assertEquals(4, outRecs.length);
    assertEquals(HeaderRecord.class, outRecs[0].getClass());
    assertEquals(FooterRecord.class, outRecs[1].getClass());
    assertEquals(HCenterRecord.class, outRecs[2].getClass());
    assertEquals(VCenterRecord.class, outRecs[3].getClass());

    // make sure the added header / footer records are empty
    HeaderRecord hr = (HeaderRecord) outRecs[0];
    assertEquals("", hr.getText());
    FooterRecord fr = (FooterRecord) outRecs[1];
    assertEquals("", fr.getText());
  }

  /**
   * Apparently it's OK to have more than one PLS record.
   * Attachment 23866 from bug 47415 had a PageSettingsBlock with two PLS records.  This file
   * seems to open OK in Excel(2007) but both PLS records are removed (perhaps because the
   * specified printers were not available on the testing machine).  Since the example file does
   * not upset Excel, POI will preserve multiple PLS records.</p>
   *
   * As of June 2009, PLS is still uninterpreted by POI
   */
  public void testDuplicatePLS_bug47415() {
    Record plsA = ur(UnknownRecord.PLS_004D, "BA AD F0 0D");
    Record plsB = ur(UnknownRecord.PLS_004D, "DE AD BE EF");
    Record contB1 = new ContinueRecord(HexRead.readFromString("FE ED"));
    Record contB2 = new ContinueRecord(HexRead.readFromString("FA CE"));
    Record[] recs = {
        new HeaderRecord("&LSales Figures"),
        new FooterRecord("&LInventory"),
        new HCenterRecord(),
        new VCenterRecord(),
        plsA,
        plsB, contB1, contB2, // make sure continuing PLS is still OK
    };
    RecordStream rs = new RecordStream(Arrays.asList(recs), 0);
    PageSettingsBlock psb;
    try {
      psb = new PageSettingsBlock(rs);
    } catch (RecordFormatException e) {
      if ("Duplicate PageSettingsBlock record (sid=0x4d)".equals(e.getMessage())) {
        throw new AssertionFailedError("Identified bug 47415");
      }
      throw e;
    }

    // serialize the PSB to see what records come out
    RecordCollector rc = new RecordCollector();
    psb.visitContainedRecords(rc);
    Record[] outRecs = rc.getRecords();

    // records were assembled in standard order, so this simple check is OK
    assertTrue(Arrays.equals(recs, outRecs));
  }

    public void testDuplicateHeaderFooter_bug48026() {

        Record[] recs = {
                BOFRecord.createSheetBOF(),
                new IndexRecord(),

                //PageSettingsBlock
                new HeaderRecord("&LDecember"),
                new FooterRecord("&LJanuary"),
                new DimensionsRecord(),

                new WindowTwoRecord(),

                //CustomViewSettingsRecordAggregate
                new UserSViewBegin(HexRead.readFromString("53 CE BD CC DE 38 44 45 97 C1 5C 89 F9 37 32 1B 01 00 00 00 64 00 00 00 40 00 00 00 03 00 00 00 7D 00 00 20 00 00 34 00 00 00 18 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 FF FF FF FF")),
                new SelectionRecord(0, 0),
                new UserSViewEnd(HexRead.readFromString("01 00")),

                // two HeaderFooterRecord records, the first one has zero GUID (16 bytes at offset 12) and belongs to the PSB,
                // the other is matched with a CustomViewSettingsRecordAggregate having UserSViewBegin with the same GUID
                new HeaderFooterRecord(HexRead.readFromString("9C 08 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 34 33 00 00 00 00 00 00 00 00")),
                new HeaderFooterRecord(HexRead.readFromString("9C 08 00 00 00 00 00 00 00 00 00 00 53 CE BD CC DE 38 44 45 97 C1 5C 89 F9 37 32 1B 34 33 00 00 00 00 00 00 00 00")),

                EOFRecord.instance,
        };
        RecordStream rs = new RecordStream(Arrays.asList(recs), 0);
        InternalSheet sheet;
        try {
            sheet = InternalSheet.createSheet(rs);
        } catch (RuntimeException e) {
            if (e.getMessage().equals("Duplicate PageSettingsBlock record (sid=0x89c)")) {
                throw new AssertionFailedError("Identified bug 48026");
            }
            throw e;
        }

        RecordCollector rv = new RecordCollector();
        sheet.visitContainedRecords(rv, 0);
        Record[] outRecs = rv.getRecords();

        assertEquals(recs.length, outRecs.length);
        //expected order of records:
        Record[] expectedRecs = {
                recs[0]//BOFRecord
                recs[1]//IndexRecord

                //PageSettingsBlock
                recs[2]//HeaderRecord
                recs[3]//FooterRecord
                recs[9]//HeaderFooterRecord
                recs[4]// DimensionsRecord
                recs[5]// WindowTwoRecord

                //CustomViewSettingsRecordAggregate
                recs[6]// UserSViewBegin
                recs[7]// SelectionRecord
                recs[10], // HeaderFooterRecord
                recs[8]// UserSViewEnd

                recs[11]//EOFRecord
        };
        for(int i=0; i < expectedRecs.length; i++){
            assertEquals("Record mismatch at index " + i,  expectedRecs[i].getClass(), outRecs[i].getClass());
        }
        HeaderFooterRecord hd1 = (HeaderFooterRecord)expectedRecs[4];
        //GUID is zero
        assertTrue(Arrays.equals(new byte[16], hd1.getGuid()));
        assertTrue(hd1.isCurrentSheet());

        UserSViewBegin svb = (UserSViewBegin)expectedRecs[7];
        HeaderFooterRecord hd2 = (HeaderFooterRecord)expectedRecs[9];
        assertFalse(hd2.isCurrentSheet());
        //GUIDs of HeaderFooterRecord and UserSViewBegin must be the same
        assertTrue(Arrays.equals(svb.getGuid(), hd2.getGuid()));
    }

}
TOP

Related Classes of org.apache.poi.hssf.record.aggregates.TestPageSettingsBlock

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.