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

import junit.framework.*;
import org.apache.poi.hwpf.*;

import org.apache.poi.util.LittleEndian;

public final class TestPlexOfCps
  extends TestCase
{
  private PlexOfCps _plexOfCps = null;
  private HWPFDocFixture _hWPFDocFixture;

  public void testWriteRead() {
    _plexOfCps = new PlexOfCps(4);

    int last = 0;
    for (int x = 0; x < 110; x++)
    {
      byte[] intHolder = new byte[4];
      int span = (int)(110.0f * Math.random());
      LittleEndian.putInt(intHolder, 0, span);
      _plexOfCps.addProperty(new GenericPropertyNode(last, last + span, intHolder));
      last += span;
    }

    byte[] output = _plexOfCps.toByteArray();
    _plexOfCps = new PlexOfCps(output, 0, output.length, 4);
    int len = _plexOfCps.length();
    assertEquals(len, 110);

    last = 0;
    for (int x = 0; x < len; x++)
    {
      GenericPropertyNode node = _plexOfCps.getProperty(x);
      assertEquals(node.getStart(), last);
      last = node.getEnd();
      int span = LittleEndian.getInt(node.getBytes());
      assertEquals(node.getEnd()-node.getStart(), span);
    }
  }
  @Override
protected void setUp()
    throws Exception
  {
    super.setUp();
    /**@todo verify the constructors*/
    _hWPFDocFixture = new HWPFDocFixture(this, HWPFDocFixture.DEFAULT_TEST_FILE);

    _hWPFDocFixture.setUp();
  }

  @Override
protected void tearDown()
    throws Exception
  {
    _plexOfCps = null;
    _hWPFDocFixture.tearDown();

    _hWPFDocFixture = null;
    super.tearDown();
  }

}
