/*******************************************************************************
* Copyright (c) 2006-2012
* Software Technology Group, Dresden University of Technology
* DevBoost GmbH, Berlin, Amtsgericht Charlottenburg, HRB 140026
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Software Technology Group - TU Dresden, Germany;
* DevBoost GmbH - Berlin, Germany
* - initial API and implementation
******************************************************************************/
/*
* @(#)VerticalLayouter.java 3.0 2008-05-28
*
* Copyright (c) 1996-2008 by the original authors of JHotDraw
* and all its contributors.
* All rights reserved.
*
* The copyright of this software is owned by the authors and
* contributors of the JHotDraw project ("the copyright holders").
* You may not use, copy or modify this software, except in
* accordance with the license agreement you entered into with
* the copyright holders. For details see accompanying license terms.
*/
package org.jhotdraw.draw;
import org.jhotdraw.util.*;
import java.awt.*;
import java.awt.geom.*;
import java.util.*;
import org.jhotdraw.geom.*;
import static org.jhotdraw.draw.AttributeKeys.*;
/**
* A layouter which lays out all children of a CompositeFigure in vertical
* direction. The preferredSize of the figures is used to determine the layout.
* This may cause some figures to resize.
* <p>
* The VerticalLayouter honors the LAYOUT_INSETS and the COMPOSITE_ALIGNMENT
* AttributeKey when laying out a CompositeFigure.
* <p>
* If COMPOSITE_ALIGNMENT is not set on the composite figure,
* the layout assigns the same width to all figures.
*
*
* @author Werner Randelshofer
* @version 3.0 2008-05-28 Added support for alignment.
* <br>2.0 2006-01-14 Changed to support double precision coordinates.
* <br>1.0 1. Dezember 2003 Created.
*/
public class VerticalLayouter extends AbstractLayouter {
/**
* This alignment is used, when
*/
private Alignment defaultAlignment = Alignment.BLOCK;
public Rectangle2D.Double calculateLayout(CompositeFigure layoutable, Point2D.Double anchor, Point2D.Double lead) {
Insets2D.Double layoutInsets = LAYOUT_INSETS.get(layoutable);
if (layoutInsets == null) {
layoutInsets = new Insets2D.Double(0, 0, 0, 0);
}
Rectangle2D.Double layoutBounds = new Rectangle2D.Double(anchor.x, anchor.y, 0, 0);
for (Figure child : layoutable.getChildren()) {
if (child.isVisible()) {
Dimension2DDouble preferredSize = child.getPreferredSize();
Insets2D.Double ins = getInsets(child);
layoutBounds.width = Math.max(layoutBounds.width, preferredSize.width + ins.left + ins.right);
layoutBounds.height += preferredSize.height + ins.top + ins.bottom;
}
}
layoutBounds.width += layoutInsets.left + layoutInsets.right;
layoutBounds.height += layoutInsets.top + layoutInsets.bottom;
return layoutBounds;
}
public Rectangle2D.Double layout(CompositeFigure layoutable, Point2D.Double anchor, Point2D.Double lead) {
Insets2D.Double layoutInsets = LAYOUT_INSETS.get(layoutable);
Alignment compositeAlignment = COMPOSITE_ALIGNMENT.get(layoutable);
if (layoutInsets == null) {
layoutInsets = new Insets2D.Double();
}
Rectangle2D.Double layoutBounds = calculateLayout(layoutable, anchor, lead);
double y = layoutBounds.y + layoutInsets.top;
for (Figure child : layoutable.getChildren()) {
if (child.isVisible()) {
Insets2D.Double insets = getInsets(child);
double height = child.getPreferredSize().height;
double width = child.getPreferredSize().width;
switch (compositeAlignment) {
case LEADING:
child.setBounds(
new Point2D.Double(
layoutBounds.x + layoutInsets.left + insets.left,
y + insets.top),
new Point2D.Double(
layoutBounds.x + + layoutInsets.left + insets.left + width,
y + insets.top + height));
break;
case TRAILING:
child.setBounds(
new Point2D.Double(
layoutBounds.x + layoutBounds.width - layoutInsets.right - insets.right - width,
y + insets.top),
new Point2D.Double(
layoutBounds.x + layoutBounds.width - layoutInsets.right - insets.right,
y + insets.top + height));
break;
case CENTER:
child.setBounds(
new Point2D.Double(
layoutBounds.x + (layoutBounds.width - width) / 2d,
y + insets.top),
new Point2D.Double(
layoutBounds.x + (layoutBounds.width + width) / 2d,
y + insets.top + height));
break;
case BLOCK:
default:
child.setBounds(
new Point2D.Double(
layoutBounds.x + layoutInsets.left + insets.left,
y + insets.top),
new Point2D.Double(
layoutBounds.x + layoutBounds.width - layoutInsets.right - insets.right,
y + insets.top + height));
break;
}
y += height + insets.top + insets.bottom;
}
}
return layoutBounds;
}
}