Package org.docx4j.model.structure

Source Code of org.docx4j.model.structure.HeaderFooterPolicy

package org.docx4j.model.structure;

/*
* Inspired/converted from org.apache.poi.xwpf.model.XWPFHeaderFooterPolicy,
*  which
*  ====================================================================
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.
==================================================================== */


import java.util.List;

import org.docx4j.openpackaging.exceptions.InvalidFormatException;
import org.docx4j.openpackaging.parts.Part;
import org.docx4j.openpackaging.parts.PartName;
import org.docx4j.openpackaging.parts.WordprocessingML.FooterPart;
import org.docx4j.openpackaging.parts.WordprocessingML.HeaderPart;
import org.docx4j.openpackaging.parts.relationships.RelationshipsPart;
import org.docx4j.wml.BooleanDefaultTrue;
import org.docx4j.wml.CTRel;
import org.docx4j.wml.FooterReference;
import org.docx4j.wml.HdrFtrRef;
import org.docx4j.wml.HeaderReference;
import org.docx4j.wml.ObjectFactory;
import org.docx4j.wml.SectPr;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class HeaderFooterPolicy {

  protected static Logger log = LoggerFactory.getLogger(HeaderFooterPolicy.class)
 
  private HeaderPart firstHeaderActive;
  private HeaderPart firstHeader;  // Need this so it can be copied in next section, even if not used in this one
  private FooterPart firstFooterActive;
  private FooterPart firstFooter;
 
  private HeaderPart evenHeader;
  private FooterPart evenFooter;
 
  private HeaderPart defaultHeader;
  private FooterPart defaultFooter;

  //dummyHeader and dummyFooter are only used when evenAndOddHeaders == true.
  //they should only be used in "read-mode" and never changed
  //they are not added to the rels
  private static Object dummyHeaderFooterMutex = new Object();
  private static HeaderPart dummyHeader;
  private static FooterPart dummyFooter;
 
  /* "same as previous" functionality:
   *
   * When "same as previous" is set, the sectPr Word emits contains
   * no w:headerReference at all. Absence of such implies settings
   * are inherited from the previous sectPr.
   *
   * In the Word UI, you can change headers in either section, and
   * see them reflected in the other.
   *
   * This does not apply to a first page header.  That remains
   * independent!
   *
   * If you say you want no header in a sectPr, Word inserts something like:
   *
   *   <w:sectPr><w:headerReference w:type="default" r:id="rId8"/>
   *  
   * with that w:hdr containing an empty paragraph.
   *
   * Note that "same as previous" in header behaves independently from
   * footer.
   *
   *
   */
 
  private HeaderFooterPolicy() {}
 
  /**
   * Figures out the policy for the given section,
   *  and creates any header and footer objects
   *  as required.
   */
  public HeaderFooterPolicy(SectPr sectPr, HeaderFooterPolicy previousHF,
      RelationshipsPart rels, BooleanDefaultTrue evenAndOddHeaders)
//    throws Exception
    {
    // Grab what headers and footers have been defined   
    if (sectPr == null) {
      log.error("Passed null sectPr?!");
      return;
    }

    List<CTRel> hdrFtrRefs = null;
    BooleanDefaultTrue titlePage = null;
   
    if (sectPr.getType()!=null
        && "continuous".equals(sectPr.getType().getVal())) {
      // If this is a continuous section, use the headers/footers from the previous section!
      log.debug("this is a continuous section");

      if (previousHF!=null) {
       
        // for a continuous sectPr, ignore the stuff in this sectPr,
        // by taking our settings from previousHF

        firstHeaderActive=previousHF.firstHeaderActive;
        firstHeader=previousHF.firstHeader; 
        firstFooterActive=previousHF.firstFooterActive;
        firstFooter=previousHF.firstFooter;
       
        evenHeader=previousHF.evenHeader;
        evenFooter=previousHF.evenFooter;
       
        defaultHeader=previousHF.defaultHeader;
        defaultFooter=previousHF.defaultFooter;       
       
        return;
      }
    }

    // The usual non-continuous case
    // (or first sectPr in docx is continuous - maybe the docx starts with columns?)

    if (previousHF==null) {
      log.debug("previousHF==null");
      previousHF= new HeaderFooterPolicy();
     
    }    
    hdrFtrRefs = sectPr.getEGHdrFtrReferences();
    titlePage = sectPr.getTitlePg();
   
    // Headers.
    // Init from previousHF
    firstHeader   = previousHF.firstHeader;
    if (titlePage!=null && titlePage.isVal() ) {
      firstHeaderActive   = previousHF.firstHeader;
    }
   
    defaultHeader = previousHF.defaultHeader;
    evenHeader    =  previousHF.evenHeader;
    // and overwrite with whatever we have
    // specific to this sectPr
    setHeaderReferences(hdrFtrRefs, rels, titlePage );
   
    // Now, same for Footers.
    // Init from previousHF
    firstFooter   = previousHF.firstFooter;
    if (titlePage!=null && titlePage.isVal() ) {
      firstFooterActive   = previousHF.firstFooter;
    }
    defaultFooter = previousHF.defaultFooter;
    evenFooter    =  previousHF.evenFooter;
    // and overwrite with whatever we have
    // specific to this sectPr
    setFooterReferences(hdrFtrRefs, rels, titlePage );
   
    if ((titlePage != null) && (titlePage.isVal())) {
      if (firstHeaderActive == null) {
        firstHeaderActive = getDummyHeader();
      }
      if (firstFooterActive == null) {
        firstFooterActive = getDummyFooter();
      }
    }
   
    if (evenAndOddHeaders == null) {
     
      log.debug("evenAndOddHeader setting missing; defaults to false");
     
      // per spec, assume false, so use odd only:-
     
      //Any even header/footer will be ignored
      //As the setting is on the document level it is not necessary to
      //keep any defined header/footer for inheritance
      evenHeader = null;
      evenFooter = null;
     
    } else {
     
      log.debug("evenAndOddHeader: "+ evenAndOddHeaders.isVal());
     
      if (evenAndOddHeaders.isVal()) {
       
        // If true, per the spec, use both even and odd
       
        //If there is only a default/odd header/footer present, then
        //the even header/footer is always a dummy empty header/footer
        if (evenHeader == null) {
          evenHeader = getDummyHeader();
        }
        if (evenFooter == null) {
          evenFooter = getDummyFooter();
        }
       
        // Experimental: 2014 08 01; required for FOPAreaTreeHelper;
        // (without this, where no header is defined in the docx,
        //  xsl-region-before-default is not created)
        if (defaultHeader == null) {
          defaultHeader = getDummyHeader();
        }
        if (defaultFooter == null) {
          defaultFooter = getDummyFooter();
        }
      }
      else {
        // Per spec, use odd only.  Any even header/footer will be ignored
        //As the setting is on the document level it is not necessary to
        //keep any defined header/footer for inheritance
        evenHeader = null;
        evenFooter = null;
      }
    }
  }

  private HeaderPart getDummyHeader() {
    if (dummyHeader == null) {
      createDummyHeaderFooter();
    }
    return dummyHeader;
  }

  private FooterPart getDummyFooter() {
    if (dummyFooter == null) {
      createDummyHeaderFooter();
    }
    return dummyFooter;
  }
 
 

  private void createDummyHeaderFooter() {
    synchronized (dummyHeaderFooterMutex) {
      if (dummyHeader == null) {
        ObjectFactory factory = new ObjectFactory();
        try {
          dummyHeader = new HeaderPart(new PartName("/word/dummyheader.xml"));
          dummyFooter = new FooterPart(new PartName("/word/dummyfooter.xml"));
        } catch (InvalidFormatException e) {
          //should not happen
        }
        dummyHeader.setJaxbElement(factory.createHdr());
        dummyHeader.getJaxbElement().getContent().add(factory.createP());
        dummyFooter.setJaxbElement(factory.createFtr());
        dummyFooter.getJaxbElement().getContent().add(factory.createP());
      }
    }
  }

//  private boolean hasHdrRef(List<CTRel> hdrFtrRefs) {
//   
//    if (hdrFtrRefs==null) return false;
//    for (CTRel rel : hdrFtrRefs) {
//      if (rel instanceof HeaderReference ) return true;
//    }
//    return false;
//  }
//  private boolean hasFtrRef(List<CTRel> hdrFtrRefs) {
//   
//    if (hdrFtrRefs==null) return false;
//    for (CTRel rel : hdrFtrRefs) {
//      if (rel instanceof FooterReference ) return true;
//    }
//    return false;
//  }

  private void setHeaderReferences(List<CTRel> hdrFtrRefs, RelationshipsPart rels,
      BooleanDefaultTrue titlePage) {
   
    for (CTRel rel : hdrFtrRefs) {
      String relId = rel.getId();
      log.debug("for h|f relId: " + relId);
     
      Part part = rels.getPart(relId);
      if (rel instanceof HeaderReference  ) {
       
        HeaderReference headerReference = (HeaderReference)rel;
       
        if (headerReference.getType() == HdrFtrRef.FIRST) {
          firstHeader = (HeaderPart)part;
          if (titlePage!=null && titlePage.isVal()) {
            log.debug("setting first page header");
            firstHeaderActive = (HeaderPart)part;
          }
        } else if (headerReference.getType() == HdrFtrRef.EVEN) {
          log.debug("setting even page header");
          evenHeader =  (HeaderPart)part;
        else {
          log.debug("setting default page header");
          defaultHeader = (HeaderPart)part;
        }
      }
    }
  }
 
  private void setFooterReferences(List<CTRel> hdrFtrRefs, RelationshipsPart rels,
      BooleanDefaultTrue titlePage) {
   
    for (CTRel rel : hdrFtrRefs) {
      String relId = rel.getId();
      log.debug("for h|f relId: " + relId);
     
      Part part = rels.getPart(relId);
      if (rel instanceof FooterReference  ) {
       
        FooterReference footerReference = (FooterReference)rel;
       
        if (footerReference.getType() == HdrFtrRef.FIRST) {
          firstFooter = (FooterPart)part;
          if (titlePage!=null && titlePage.isVal()) {               
            log.debug("setting first page footer");
            firstFooterActive = (FooterPart)part;
          }
        } else if (footerReference.getType() == HdrFtrRef.EVEN) {
          log.debug("setting even page footer");
          evenFooter =  (FooterPart)part;
        else {
          log.debug("setting default page footer");
          defaultFooter = (FooterPart)part;
        }
      }
    }
  }
 
  public HeaderPart getFirstHeader() {
    return firstHeaderActive;
  }
  public FooterPart getFirstFooter() {
    return firstFooterActive;
  }
  public HeaderPart getEvenHeader() {
    return evenHeader;
  }
  public FooterPart getEvenFooter() {
    return evenFooter;
  }
  /** If an even header is present this is the odd header
   *  otherwise it is both, even and odd header
   */
  public HeaderPart getDefaultHeader() {
    return defaultHeader;
  }
  /** If an even footer is present this is the odd footer
   *  otherwise it is both, even and odd footer
   */
  public FooterPart getDefaultFooter() {
    return defaultFooter;
  }

  /**
   * Get the header that applies to the given
   *  (1 based) page.
   * @param pageNumber The one based page number
   */
  public HeaderPart getHeader(int pageNumber) {
    if(pageNumber == 1 && firstHeaderActive != null) {
      return firstHeaderActive;
    }
    if(pageNumber % 2 == 0 && evenHeader != null) {
      return evenHeader;
    }
    return defaultHeader;
  }
  /**
   * Get the footer that applies to the given
   *  (1 based) page.
   * @param pageNumber The one based page number
   */
  public FooterPart getFooter(int pageNumber) {
    if(pageNumber == 1 && firstFooterActive != null) {
      return firstFooterActive;
    }
    if(pageNumber % 2 == 0 && evenFooter != null) {
      return evenFooter;
    }
    return defaultFooter;
  }
 
}
TOP

Related Classes of org.docx4j.model.structure.HeaderFooterPolicy

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.