/* * @(#)SpringLayout.java 1.19 03/12/19 * * Copyright 2004 Sun Microsystems, Inc. All rights reserved. * SUN PROPRIETARY/CONFIDENTIAL. Use is subject to license terms. */ package javax.swing; import java.awt.*; import java.util.*; /** * A SpringLayout lays out the children of its associated container * according to a set of constraints. * See How to Use SpringLayout * in The Java Tutorial for examples of using * SpringLayout. * *

* Each constraint, * represented by a Spring object, * controls the vertical or horizontal distance * between two component edges. * The edges can belong to * any child of the container, * or to the container itself. * For example, * the allowable width of a component * can be expressed using a constraint * that controls the distance between the west (left) and east (right) * edges of the component. * The allowable y coordinates for a component * can be expressed by constraining the distance between * the north (top) edge of the component * and the north edge of its container. * *

* Every child of a SpringLayout-controlled container, * as well as the container itself, * has exactly one set of constraints * associated with it. * These constraints are represented by * a SpringLayout.Constraints object. * By default, * SpringLayout creates constraints * that make their associated component * have the minimum, preferred, and maximum sizes * returned by the component's * {@link java.awt.Component#getMinimumSize}, * {@link java.awt.Component#getPreferredSize}, and * {@link java.awt.Component#getMaximumSize} * methods. The x and y positions are initially not * constrained, so that until you constrain them the Component * will be positioned at 0,0 relative to the Insets of the * parent Container. * *

* You can change * a component's constraints in several ways. * You can * use one of the * {@link #putConstraint putConstraint} * methods * to establish a spring * linking the edges of two components within the same container. * Or you can get the appropriate SpringLayout.Constraints * object using * {@link #getConstraints getConstraints} * and then modify one or more of its springs. * Or you can get the spring for a particular edge of a component * using {@link #getConstraint getConstraint}, * and modify it. * You can also associate * your own SpringLayout.Constraints object * with a component by specifying the constraints object * when you add the component to its container * (using * {@link Container#add(Component, Object)}). * *

* The Spring object representing each constraint * has a minimum, preferred, maximum, and current value. * The current value of the spring * is somewhere between the minimum and maximum values, * according to the formula given in the * {@link Spring#sum} method description. * When the minimum, preferred, and maximum values are the same, * the current value is always equal to them; * this inflexible spring is called a strut. * You can create struts using the factory method * {@link Spring#constant(int)}. * The Spring class also provides factory methods * for creating other kinds of springs, * including springs that depend on other springs. * *

* In a SpringLayout, the position of each edge is dependent on * the position of just one other edge. If a constraint is subsequently added * to create a new binding for an edge, the previous binding is discarded * and the edge remains dependent on a single edge. * Springs should only be attached * between edges of the container and its immediate children; the behavior * of the SpringLayout when presented with constraints linking * the edges of components from different containers (either internal or * external) is undefined. * *

* SpringLayout vs. Other Layout Managers *

* *
*
* Note: * Unlike many layout managers, * SpringLayout doesn't automatically set the location of * the components it manages. * If you hand-code a GUI that uses SpringLayout, * remember to initialize component locations by constraining the west/east * and north/south locations. *

* Depending on the constraints you use, * you may also need to set the size of the container explicitly. *


*
* *

* Despite the simplicity of SpringLayout, * it can emulate the behavior of most other layout managers. * For some features, * such as the line breaking provided by FlowLayout, * you'll need to * create a special-purpose subclass of the Spring class. * *

* SpringLayout also provides a way to solve * many of the difficult layout * problems that cannot be solved by nesting combinations * of Boxes. That said, SpringLayout honors the * LayoutManager2 contract correctly and so can be nested with * other layout managers -- a technique that can be preferable to * creating the constraints implied by the other layout managers. *

* The asymptotic complexity of the layout operation of a SpringLayout * is linear in the number of constraints (and/or components). *

* Warning: * Serialized objects of this class will not be compatible with * future Swing releases. The current serialization support is * appropriate for short term storage or RMI between applications running * the same version of Swing. As of 1.4, support for long term storage * of all JavaBeansTM * has been added to the java.beans package. * Please see {@link java.beans.XMLEncoder}. * * @see Spring * @see SpringLayout.Constraints * * @version 1.19 12/19/03 * @author Philip Milne * @author Joe Winchester * @since 1.4 */ public class SpringLayout implements LayoutManager2 { private Map componentConstraints = new HashMap(); private Spring cyclicReference = Spring.constant(Spring.UNSET); private Set cyclicSprings; private Set acyclicSprings; /** * Specifies the top edge of a component's bounding rectangle. */ public static final String NORTH = "North"; /** * Specifies the bottom edge of a component's bounding rectangle. */ public static final String SOUTH = "South"; /** * Specifies the right edge of a component's bounding rectangle. */ public static final String EAST = "East"; /** * Specifies the left edge of a component's bounding rectangle. */ public static final String WEST = "West"; /** * A Constraints object holds the * constraints that govern the way a component's size and position * change in a container controlled by a SpringLayout. * A Constraints object is * like a Rectangle, in that it * has x, y, * width, and height properties. * In the Constraints object, however, * these properties have * Spring values instead of integers. * In addition, * a Constraints object * can be manipulated as four edges * -- north, south, east, and west -- * using the constraint property. * *

* The following formulas are always true * for a Constraints object: * *

     *       west = x
     *      north = y
     *       east = x + width
     *      south = y + height
* * Note: In this document, * operators represent methods * in the Spring class. * For example, "a + b" is equal to * Spring.sum(a, b), * and "a - b" is equal to * Spring.sum(a, Spring.minus(b)). * See the * {@link Spring Spring API documentation} * for further details * of spring arithmetic. * *

* * Because a Constraints object's properties -- * representing its edges, size, and location -- can all be set * independently and yet are interrelated, * the object can become over-constrained. * For example, * if both the x and width * properties are set * and then the east edge is set, * the object is over-constrained horizontally. * When this happens, one of the values * (in this case, the x property) * automatically changes so * that the formulas still hold. * *

* The following table shows which value changes * when a Constraints object * is over-constrained horizontally. * *

* * * * * * * * * * * * * * * * * * *
Value Being Set
(method used)
Result When Over-Constrained Horizontally
* (x, width, and the east edge are all non-null)
x or the west edge
(setX or setConstraint)
width value is automatically set to east - x.
width
(setWidth)
east edge's value is automatically set to x + width.
east edge
(setConstraint)
x value is automatically set to east - width.
* *

* The rules for the vertical properties are similar: *

* * * * * * * * * * * * * * * * * * *
Value Being Set
(method used)
Result When Over-Constrained Vertically
(y, height, and the south edge are all non-null)
y or the north edge
(setY or setConstraint)
height value is automatically set to south - y.
height
(setHeight)
south edge's value is automatically set to y + height.
south edge
(setConstraint)
y value is automatically set to south - height.
* */ public static class Constraints { private Spring x; private Spring y; private Spring width; private Spring height; private Spring east; private Spring south; private Spring verticalDerived = null; private Spring horizontalDerived = null; /** * Creates an empty Constraints object. */ public Constraints() { this(null, null, null, null); } /** * Creates a Constraints object with the * specified values for its * x and y properties. * The height and width springs * have null values. * * @param x the spring controlling the component's x value * @param y the spring controlling the component's y value */ public Constraints(Spring x, Spring y) { this(x, y, null, null); } /** * Creates a Constraints object with the * specified values for its * x, y, width, * and height properties. * Note: If the SpringLayout class * encounters null values in the * Constraints object of a given component, * it replaces them with suitable defaults. * * @param x the spring value for the x property * @param y the spring value for the y property * @param width the spring value for the width property * @param height the spring value for the height property */ public Constraints(Spring x, Spring y, Spring width, Spring height) { this.x = x; this.y = y; this.width = width; this.height = height; } /** * Creates a Constraints object with * suitable x, y, width and * height springs for component, c. * The x and y springs are constant * springs initialised with the component's location at * the time this method is called. The width and * height springs are special springs, created by * the Spring.width() and Spring.height() * methods, which track the size characteristics of the component * when they change. * * @param c the component whose characteristics will be reflected by this Constraints object * @throws NullPointerException if c is null. * @since 1.5 */ public Constraints(Component c) { this.x = Spring.constant(c.getX()); this.y = Spring.constant(c.getY()); this.width = Spring.width(c); this.height = Spring.height(c); } private boolean overConstrainedHorizontally() { return (x != null) && (width != null) && (east != null); } private boolean overConstrainedVertically() { return (y != null) && (height != null) && (south != null); } private Spring sum(Spring s1, Spring s2) { return (s1 == null || s2 == null) ? null : Spring.sum(s1, s2); } private Spring difference(Spring s1, Spring s2) { return (s1 == null || s2 == null) ? null : Spring.difference(s1, s2); } /** * Sets the x property, * which controls the x value * of a component's location. * * @param x the spring controlling the x value * of a component's location * * @see #getX * @see SpringLayout.Constraints */ public void setX(Spring x) { this.x = x; horizontalDerived = null; if (overConstrainedHorizontally()) { width = null; } } /** * Returns the value of the x property. * * @return the spring controlling the x value * of a component's location * * @see #setX * @see SpringLayout.Constraints */ public Spring getX() { if (x != null) { return x; } if (horizontalDerived == null) { horizontalDerived = difference(east, width); } return horizontalDerived; } /** * Sets the y property, * which controls the y value * of a component's location. * * @param y the spring controlling the y value * of a component's location * * @see #getY * @see SpringLayout.Constraints */ public void setY(Spring y) { this.y = y; verticalDerived = null; if (overConstrainedVertically()) { height = null; } } /** * Returns the value of the y property. * * @return the spring controlling the y value * of a component's location * * @see #setY * @see SpringLayout.Constraints */ public Spring getY() { if (y != null) { return y; } if (verticalDerived == null) { verticalDerived = difference(south, height); } return verticalDerived; } /** * Sets the width property, * which controls the width of a component. * * @param width the spring controlling the width of this * Constraints object * * @see #getWidth * @see SpringLayout.Constraints */ public void setWidth(Spring width) { this.width = width; horizontalDerived = null; if (overConstrainedHorizontally()) { east = null; } } /** * Returns the value of the width property. * * @return the spring controlling the width of a component * * @see #setWidth * @see SpringLayout.Constraints */ public Spring getWidth() { if (width != null) { return width; } if (horizontalDerived == null) { horizontalDerived = difference(east, x); } return horizontalDerived; } /** * Sets the height property, * which controls the height of a component. * * @param height the spring controlling the height of this Constraints * object * * @see #getHeight * @see SpringLayout.Constraints */ public void setHeight(Spring height) { this.height = height; verticalDerived = null; if (overConstrainedVertically()) { south = null; } } /** * Returns the value of the height property. * * @return the spring controlling the height of a component * * @see #setHeight * @see SpringLayout.Constraints */ public Spring getHeight() { if (height != null) { return height; } if (verticalDerived == null) { verticalDerived = difference(south, y); } return verticalDerived; } private void setEast(Spring east) { this.east = east; horizontalDerived = null; if (overConstrainedHorizontally()) { x = null; } } private Spring getEast() { if (east != null) { return east; } if (horizontalDerived == null) { horizontalDerived = sum(x, width); } return horizontalDerived; } private void setSouth(Spring south) { this.south = south; verticalDerived = null; if (overConstrainedVertically()) { y = null; } } private Spring getSouth() { if (south != null) { return south; } if (verticalDerived == null) { verticalDerived = sum(y, height); } return verticalDerived; } /** * Sets the spring controlling the specified edge. * The edge must have one of the following values: * SpringLayout.NORTH, SpringLayout.SOUTH, * SpringLayout.EAST, SpringLayout.WEST. * * @param edgeName the edge to be set * @param s the spring controlling the specified edge * * @see #getConstraint * @see #NORTH * @see #SOUTH * @see #EAST * @see #WEST * @see SpringLayout.Constraints */ public void setConstraint(String edgeName, Spring s) { edgeName = edgeName.intern(); if (edgeName == "West") { setX(s); } else if (edgeName == "North") { setY(s); } else if (edgeName == "East") { setEast(s); } else if (edgeName == "South") { setSouth(s); } } /** * Returns the value of the specified edge. * The edge must have one of the following values: * SpringLayout.NORTH, SpringLayout.SOUTH, * SpringLayout.EAST, SpringLayout.WEST. * * @param edgeName the edge whose value * is to be returned * * @return the spring controlling the specified edge * * @see #setConstraint * @see #NORTH * @see #SOUTH * @see #EAST * @see #WEST * @see SpringLayout.Constraints */ public Spring getConstraint(String edgeName) { edgeName = edgeName.intern(); return (edgeName == "West") ? getX() : (edgeName == "North") ? getY() : (edgeName == "East") ? getEast() : (edgeName == "South") ? getSouth() : null; } /*pp*/ void reset() { if (x != null) x.setValue(Spring.UNSET); if (y != null) y.setValue(Spring.UNSET); if (width != null) width.setValue(Spring.UNSET); if (height != null) height.setValue(Spring.UNSET); if (east != null) east.setValue(Spring.UNSET); if (south != null) south.setValue(Spring.UNSET); if (horizontalDerived != null) horizontalDerived.setValue(Spring.UNSET); if (verticalDerived != null) verticalDerived.setValue(Spring.UNSET); } } private static class SpringProxy extends Spring { private String edgeName; private Component c; private SpringLayout l; public SpringProxy(String edgeName, Component c, SpringLayout l) { this.edgeName = edgeName; this.c = c; this.l = l; } private Spring getConstraint() { return l.getConstraints(c).getConstraint(edgeName); } public int getMinimumValue() { return getConstraint().getMinimumValue(); } public int getPreferredValue() { return getConstraint().getPreferredValue(); } public int getMaximumValue() { return getConstraint().getMaximumValue(); } public int getValue() { return getConstraint().getValue(); } public void setValue(int size) { getConstraint().setValue(size); } /*pp*/ boolean isCyclic(SpringLayout l) { return l.isCyclic(getConstraint()); } public String toString() { return "SpringProxy for " + edgeName + " edge of " + c.getName() + "."; } } /** * Constructs a new SpringLayout. */ public SpringLayout() {} private void resetCyclicStatuses() { cyclicSprings = new HashSet(); acyclicSprings = new HashSet(); } private void setParent(Container p) { resetCyclicStatuses(); Constraints pc = getConstraints(p); pc.setX(Spring.constant(0)); pc.setY(Spring.constant(0)); // The applyDefaults() method automatically adds width and // height springs that delegate their calculations to the // getMinimumSize(), getPreferredSize() and getMaximumSize() // methods of the relevant component. In the case of the // parent this will cause an infinite loop since these // methods, in turn, delegate their calculations to the // layout manager. Check for this case and replace the // the springs that would cause this problem with a // constant springs that supply default values. Spring width = pc.getWidth(); if (width instanceof Spring.WidthSpring && ((Spring.WidthSpring)width).c == p) { pc.setWidth(Spring.constant(0, 0, Integer.MAX_VALUE)); } Spring height = pc.getHeight(); if (height instanceof Spring.HeightSpring && ((Spring.HeightSpring)height).c == p) { pc.setHeight(Spring.constant(0, 0, Integer.MAX_VALUE)); } } /*pp*/ boolean isCyclic(Spring s) { if (s == null) { return false; } if (cyclicSprings.contains(s)) { return true; } if (acyclicSprings.contains(s)) { return false; } cyclicSprings.add(s); boolean result = s.isCyclic(this); if (!result) { acyclicSprings.add(s); cyclicSprings.remove(s); } else { System.err.println(s + " is cyclic. "); } return result; } private Spring abandonCycles(Spring s) { return isCyclic(s) ? cyclicReference : s; } // LayoutManager methods. /** * Has no effect, * since this layout manager does not * use a per-component string. */ public void addLayoutComponent(String name, Component c) {} /** * Removes the constraints associated with the specified component. * * @param c the component being removed from the container */ public void removeLayoutComponent(Component c) { componentConstraints.remove(c); } private static Dimension addInsets(int width, int height, Container p) { Insets i = p.getInsets(); return new Dimension(width + i.left + i.right, height + i.top + i.bottom); } public Dimension minimumLayoutSize(Container parent) { setParent(parent); Constraints pc = getConstraints(parent); return addInsets(abandonCycles(pc.getWidth()).getMinimumValue(), abandonCycles(pc.getHeight()).getMinimumValue(), parent); } public Dimension preferredLayoutSize(Container parent) { setParent(parent); Constraints pc = getConstraints(parent); return addInsets(abandonCycles(pc.getWidth()).getPreferredValue(), abandonCycles(pc.getHeight()).getPreferredValue(), parent); } // LayoutManager2 methods. public Dimension maximumLayoutSize(Container parent) { setParent(parent); Constraints pc = getConstraints(parent); return addInsets(abandonCycles(pc.getWidth()).getMaximumValue(), abandonCycles(pc.getHeight()).getMaximumValue(), parent); } /** * If constraints is an instance of * SpringLayout.Constraints, * associates the constraints with the specified component. *

* @param component the component being added * @param constraints the component's constraints * * @see SpringLayout.Constraints */ public void addLayoutComponent(Component component, Object constraints) { if (constraints instanceof Constraints) { putConstraints(component, (Constraints)constraints); } } /** * Returns 0.5f (centered). */ public float getLayoutAlignmentX(Container p) { return 0.5f; } /** * Returns 0.5f (centered). */ public float getLayoutAlignmentY(Container p) { return 0.5f; } public void invalidateLayout(Container p) {} // End of LayoutManger2 methods /** * Links edge e1 of component c1 to * edge e2 of component c2, * with a fixed distance between the edges. This * constraint will cause the assignment *

     *     value(e1, c1) = value(e2, c2) + pad
* to take place during all subsequent layout operations. *

* @param e1 the edge of the dependent * @param c1 the component of the dependent * @param pad the fixed distance between dependent and anchor * @param e2 the edge of the anchor * @param c2 the component of the anchor * * @see #putConstraint(String, Component, Spring, String, Component) */ public void putConstraint(String e1, Component c1, int pad, String e2, Component c2) { putConstraint(e1, c1, Spring.constant(pad), e2, c2); } /** * Links edge e1 of component c1 to * edge e2 of component c2. As edge * (e2, c2) changes value, edge (e1, c1) will * be calculated by taking the (spring) sum of (e2, c2) * and s. Each edge must have one of the following values: * SpringLayout.NORTH, SpringLayout.SOUTH, * SpringLayout.EAST, SpringLayout.WEST. *

* @param e1 the edge of the dependent * @param c1 the component of the dependent * @param s the spring linking dependent and anchor * @param e2 the edge of the anchor * @param c2 the component of the anchor * * @see #putConstraint(String, Component, int, String, Component) * @see #NORTH * @see #SOUTH * @see #EAST * @see #WEST */ public void putConstraint(String e1, Component c1, Spring s, String e2, Component c2) { putConstraint(e1, c1, Spring.sum(s, getConstraint(e2, c2))); } private void putConstraint(String e, Component c, Spring s) { if (s != null) { getConstraints(c).setConstraint(e, s); } } private Constraints applyDefaults(Component c, Constraints constraints) { if (constraints == null) { constraints = new Constraints(); } if (constraints.getWidth() == null) { constraints.setWidth(new Spring.WidthSpring(c)); } if (constraints.getHeight() == null) { constraints.setHeight(new Spring.HeightSpring(c)); } if (constraints.getX() == null) { constraints.setX(Spring.constant(0)); } if (constraints.getY() == null) { constraints.setY(Spring.constant(0)); } return constraints; } private void putConstraints(Component component, Constraints constraints) { componentConstraints.put(component, applyDefaults(component, constraints)); } /** * Returns the constraints for the specified component. * Note that, * unlike the GridBagLayout * getConstraints method, * this method does not clone constraints. * If no constraints * have been associated with this component, * this method * returns a default constraints object positioned at * 0,0 relative to the parent's Insets and its width/height * constrained to the minimum, maximum, and preferred sizes of the * component. The size characteristics * are not frozen at the time this method is called; * instead this method returns a constraints object * whose characteristics track the characteristics * of the component as they change. * * @param c the component whose constraints will be returned * * @return the constraints for the specified component */ public Constraints getConstraints(Component c) { Constraints result = (Constraints)componentConstraints.get(c); if (result == null) { if (c instanceof javax.swing.JComponent) { Object cp = ((javax.swing.JComponent)c).getClientProperty(SpringLayout.class); if (cp instanceof Constraints) { return applyDefaults(c, (Constraints)cp); } } result = new Constraints(); putConstraints(c, result); } return result; } /** * Returns the spring controlling the distance between * the specified edge of * the component and the top or left edge of its parent. This * method, instead of returning the current binding for the * edge, returns a proxy that tracks the characteristics * of the edge even if the edge is subsequently rebound. * Proxies are intended to be used in builder envonments * where it is useful to allow the user to define the * constraints for a layout in any order. Proxies do, however, * provide the means to create cyclic dependencies amongst * the constraints of a layout. Such cycles are detected * internally by SpringLayout so that * the layout operation always terminates. * * @param edgeName must be * SpringLayout.NORTH, * SpringLayout.SOUTH, * SpringLayout.EAST, or * SpringLayout.WEST * @param c the component whose edge spring is desired * * @return a proxy for the spring controlling the distance between the * specified edge and the top or left edge of its parent * * @see #NORTH * @see #SOUTH * @see #EAST * @see #WEST */ public Spring getConstraint(String edgeName, Component c) { // The interning here is unnecessary; it was added for efficiency. edgeName = edgeName.intern(); return new SpringProxy(edgeName, c, this); } public void layoutContainer(Container parent) { setParent(parent); int n = parent.getComponentCount(); getConstraints(parent).reset(); for (int i = 0 ; i < n ; i++) { getConstraints(parent.getComponent(i)).reset(); } Insets insets = parent.getInsets(); Constraints pc = getConstraints(parent); abandonCycles(pc.getX()).setValue(0); abandonCycles(pc.getY()).setValue(0); abandonCycles(pc.getWidth()).setValue(parent.getWidth() - insets.left - insets.right); abandonCycles(pc.getHeight()).setValue(parent.getHeight() - insets.top - insets.bottom); for (int i = 0 ; i < n ; i++) { Component c = parent.getComponent(i); Constraints cc = getConstraints(c); int x = abandonCycles(cc.getX()).getValue(); int y = abandonCycles(cc.getY()).getValue(); int width = abandonCycles(cc.getWidth()).getValue(); int height = abandonCycles(cc.getHeight()).getValue(); c.setBounds(insets.left + x, insets.top + y, width, height); } } }