/*
 * 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: PositionIterator.java 1805173 2017-08-16 10:50:04Z ssteiner $ */

package org.apache.fop.layoutmgr;

import java.util.Iterator;
import java.util.NoSuchElementException;

An iterator over Position instances, that is wrapped around another 'parent' Iterator. The parent can be either another PositionIterator, or an iterator over KnuthElements, for example.
The next() method always returns a Position. The getPos(Object) method can be overridden in subclasses to take care of obtaining the LayoutManager or Position from the object returned by the parent iterator's next() method.
/** * An iterator over {@link Position} instances, that is wrapped around * another 'parent' {@link Iterator}. The parent can be either another * {@code PositionIterator}, or an iterator over {@link KnuthElement}s, * for example.<br> * The {@link #next()} method always returns a {@link Position}. The * {@link #getPos(Object)} method can be overridden in subclasses * to take care of obtaining the {@link LayoutManager} or {@link Position} * from the object returned by the parent iterator's {@code next()} method. */
public class PositionIterator implements Iterator<Position> { private Iterator parentIter; private Object nextObj; private LayoutManager childLM; private boolean hasNext;
Construct position iterator.
Params:
  • parentIter – an iterator to use as parent
/** * Construct position iterator. * @param parentIter an iterator to use as parent */
public PositionIterator(Iterator parentIter) { this.parentIter = parentIter; lookAhead(); //checkNext(); }
Returns:layout manager of next child layout manager or null
/** @return layout manager of next child layout manager or null */
public LayoutManager getNextChildLM() { // Move to next "segment" of iterator, ie: new childLM if (childLM == null && nextObj != null) { childLM = getLM(nextObj); hasNext = true; } return childLM; }
Params:
  • nextObj – next object from which to obtain position
Returns:layout manager
/** * @param nextObj next object from which to obtain position * @return layout manager */
protected LayoutManager getLM(Object nextObj) { return getPos(nextObj).getLM(); }
Default implementation assumes that the passed nextObj is itself a Position, and just returns it. Subclasses for which this is not the case, must provide a suitable override this method.
Params:
  • nextObj – next object from which to obtain position
Returns:position of next object.
/** * Default implementation assumes that the passed * {@code nextObj} is itself a {@link Position}, and just returns it. * Subclasses for which this is not the case, <em>must</em> provide a * suitable override this method. * @param nextObj next object from which to obtain position * @return position of next object. */
protected Position getPos(Object nextObj) { if (nextObj instanceof Position) { return (Position)nextObj; } throw new IllegalArgumentException( "Cannot obtain Position from the given object."); } private void lookAhead() { if (parentIter.hasNext()) { hasNext = true; nextObj = parentIter.next(); } else { endIter(); } }
Returns:true if not at end of sub-sequence with same child layout manager
/** @return true if not at end of sub-sequence with same child layout manager */
protected boolean checkNext() { LayoutManager lm = getLM(nextObj); if (childLM == null) { childLM = lm; } else if (childLM != lm && lm != null) { // End of this sub-sequence with same child LM hasNext = false; childLM = null; return false; } return true; }
end (reset) iterator
/** end (reset) iterator */
protected void endIter() { hasNext = false; nextObj = null; childLM = null; }
{@inheritDoc}
/** {@inheritDoc} */
public boolean hasNext() { return (hasNext && checkNext()); }
{@inheritDoc}
/** {@inheritDoc} */
public Position next() throws NoSuchElementException { if (hasNext) { Position retPos = getPos(nextObj); lookAhead(); return retPos; } else { throw new NoSuchElementException("PosIter"); } }
Returns:peek at next object
/** @return peek at next object */
public Object peekNext() { return nextObj; }
{@inheritDoc}
/** {@inheritDoc} */
public void remove() throws UnsupportedOperationException { throw new UnsupportedOperationException("PositionIterator doesn't support remove"); } }