| /* |
| * 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. |
| */ |
| |
| /* $Id$ */ |
| |
| package org.apache.fop.fo.pagination; |
| |
| // Java |
| import java.util.Map; |
| |
| import org.xml.sax.Locator; |
| |
| import org.apache.fop.apps.FOPException; |
| import org.apache.fop.fo.FONode; |
| import org.apache.fop.fo.FObj; |
| import org.apache.fop.fo.PropertyList; |
| import org.apache.fop.fo.ValidationException; |
| |
| /** |
| * Class modelling the <a href="http://www.w3.org/TR/xsl/#fo_layout-master-set"> |
| * <code>fo:layout-master-set</code></a> object. |
| * |
| * This class maintains the set of simple page master and |
| * page sequence masters. |
| * The masters are stored so that the page sequence can obtain |
| * the required page master to create a page. |
| * The page sequence masters can be reset as they hold state |
| * information for a page sequence. |
| */ |
| public class LayoutMasterSet extends FObj { |
| |
| private Map<String, SimplePageMaster> simplePageMasters; |
| private Map<String, PageSequenceMaster> pageSequenceMasters; |
| |
| /** |
| * Create a LayoutMasterSet instance that is a child of the given |
| * parent {@link FONode}. |
| * |
| * @param parent {@link FONode} that is the parent of this object |
| */ |
| public LayoutMasterSet(FONode parent) { |
| super(parent); |
| } |
| |
| /** {@inheritDoc} */ |
| public void bind(PropertyList pList) throws FOPException { |
| // No properties in layout-master-set. |
| } |
| |
| /** {@inheritDoc} */ |
| public void startOfNode() throws FOPException { |
| getRoot().setLayoutMasterSet(this); |
| simplePageMasters = new java.util.HashMap<String, SimplePageMaster>(); |
| pageSequenceMasters = new java.util.HashMap<String, PageSequenceMaster>(); |
| } |
| |
| /** {@inheritDoc} */ |
| public void endOfNode() throws FOPException { |
| if (firstChild == null) { |
| missingChildElementError("(simple-page-master|page-sequence-master)+"); |
| } |
| checkRegionNames(); |
| resolveSubSequenceReferences(); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * <br>XSL/FOP: (simple-page-master|page-sequence-master)+ |
| */ |
| protected void validateChildNode(Locator loc, String nsURI, String localName) |
| throws ValidationException { |
| if (FO_URI.equals(nsURI)) { |
| if (!localName.equals("simple-page-master") |
| && !localName.equals("page-sequence-master")) { |
| invalidChildError(loc, nsURI, localName); |
| } |
| } |
| } |
| |
| /** |
| * Section 7.25.7: check to see that if a region-name is a |
| * duplicate, that it maps to the same fo region-class. |
| * @throws ValidationException if there's a name duplication |
| */ |
| private void checkRegionNames() throws ValidationException { |
| // (user-entered) region-name to default region map. |
| Map<String, String> allRegions = new java.util.HashMap<String, String>(); |
| for (SimplePageMaster simplePageMaster : simplePageMasters.values()) { |
| Map<String, Region> spmRegions = simplePageMaster.getRegions(); |
| for (Region region : spmRegions.values()) { |
| if (allRegions.containsKey(region.getRegionName())) { |
| String defaultRegionName |
| = allRegions.get(region.getRegionName()); |
| if (!defaultRegionName.equals(region.getDefaultRegionName())) { |
| getFOValidationEventProducer().regionNameMappedToMultipleRegionClasses(this, |
| region.getRegionName(), |
| defaultRegionName, |
| region.getDefaultRegionName(), getLocator()); |
| } |
| } |
| allRegions.put(region.getRegionName(), |
| region.getDefaultRegionName()); |
| } |
| } |
| } |
| |
| private void resolveSubSequenceReferences() throws ValidationException { |
| for (PageSequenceMaster psm : pageSequenceMasters.values()) { |
| for (SubSequenceSpecifier subSequenceSpecifier : psm.getSubSequenceSpecifier()) { |
| subSequenceSpecifier.resolveReferences(this); |
| } |
| } |
| } |
| |
| /** |
| * Add a simple page master. |
| * The name is checked to throw an error if already added. |
| * @param sPM simple-page-master to add |
| * @throws ValidationException if there's a problem with name uniqueness |
| */ |
| protected void addSimplePageMaster(SimplePageMaster sPM) |
| throws ValidationException { |
| |
| // check for duplication of master-name |
| String masterName = sPM.getMasterName(); |
| if (existsName(masterName)) { |
| getFOValidationEventProducer().masterNameNotUnique(this, |
| getName(), |
| masterName, sPM.getLocator()); |
| } |
| this.simplePageMasters.put(masterName, sPM); |
| } |
| |
| private boolean existsName(String masterName) { |
| return (simplePageMasters.containsKey(masterName) |
| || pageSequenceMasters.containsKey(masterName)); |
| } |
| |
| /** |
| * Get a simple page master by name. |
| * This is used by the page sequence to get a page master for |
| * creating pages. |
| * @param masterName the name of the page master |
| * @return the requested simple-page-master |
| */ |
| public SimplePageMaster getSimplePageMaster(String masterName) { |
| return simplePageMasters.get(masterName); |
| } |
| |
| /** |
| * Add a page sequence master. |
| * The name is checked to throw an error if already added. |
| * @param masterName name for the master |
| * @param pSM PageSequenceMaster instance |
| * @throws ValidationException if there's a problem with name uniqueness |
| */ |
| protected void addPageSequenceMaster(String masterName, |
| PageSequenceMaster pSM) |
| throws ValidationException { |
| // check against duplication of master-name |
| if (existsName(masterName)) { |
| getFOValidationEventProducer().masterNameNotUnique(this, |
| getName(), |
| masterName, pSM.getLocator()); |
| } |
| this.pageSequenceMasters.put(masterName, pSM); |
| } |
| |
| /** |
| * Get a page sequence master by name. |
| * This is used by the page sequence to get a page master for |
| * creating pages. |
| * @param masterName name of the master |
| * @return the requested PageSequenceMaster instance |
| */ |
| public PageSequenceMaster getPageSequenceMaster(String masterName) { |
| return this.pageSequenceMasters.get(masterName); |
| } |
| |
| /** |
| * Checks whether or not a region name exists in this master set. |
| * @param regionName name of the region |
| * @return true when the region name specified has a region in this LayoutMasterSet |
| */ |
| public boolean regionNameExists(String regionName) { |
| for (SimplePageMaster spm : simplePageMasters.values()) { |
| if (spm.regionNameExists(regionName)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| /** {@inheritDoc} */ |
| public String getLocalName() { |
| return "layout-master-set"; |
| } |
| |
| /** |
| * {@inheritDoc} |
| * @return {@link org.apache.fop.fo.Constants#FO_LAYOUT_MASTER_SET} |
| */ |
| public int getNameId() { |
| return FO_LAYOUT_MASTER_SET; |
| } |
| |
| /** |
| * Returns the default name of the region to which the flow or static-content having |
| * the given flow-name is assigned. |
| * |
| * @param flowName the value of the flow-name property |
| * @return the default region name ("xsl-region-body", "xsl-region-before", etc.) |
| */ |
| public String getDefaultRegionNameFor(String flowName) { |
| for (SimplePageMaster spm : simplePageMasters.values()) { |
| for (Region region : spm.getRegions().values()) { |
| if (region.getRegionName().equals(flowName)) { |
| return region.getDefaultRegionName(); |
| } |
| } |
| } |
| assert flowName.equals("xsl-before-float-separator") || flowName.equals("xsl-footnote-separator"); |
| return flowName; |
| } |
| } |
| |