mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			1113 lines
		
	
	
		
			35 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			1113 lines
		
	
	
		
			35 KiB
		
	
	
	
		
			Java
		
	
	
	
/* BoxView.java -- An composite view
 | 
						|
   Copyright (C) 2005, 2006  Free Software Foundation, Inc.
 | 
						|
 | 
						|
This file is part of GNU Classpath.
 | 
						|
 | 
						|
GNU Classpath is free software; you can redistribute it and/or modify
 | 
						|
it under the terms of the GNU General Public License as published by
 | 
						|
the Free Software Foundation; either version 2, or (at your option)
 | 
						|
any later version.
 | 
						|
 | 
						|
GNU Classpath is distributed in the hope that it will be useful, but
 | 
						|
WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | 
						|
General Public License for more details.
 | 
						|
 | 
						|
You should have received a copy of the GNU General Public License
 | 
						|
along with GNU Classpath; see the file COPYING.  If not, write to the
 | 
						|
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 | 
						|
02110-1301 USA.
 | 
						|
 | 
						|
Linking this library statically or dynamically with other modules is
 | 
						|
making a combined work based on this library.  Thus, the terms and
 | 
						|
conditions of the GNU General Public License cover the whole
 | 
						|
combination.
 | 
						|
 | 
						|
As a special exception, the copyright holders of this library give you
 | 
						|
permission to link this library with independent modules to produce an
 | 
						|
executable, regardless of the license terms of these independent
 | 
						|
modules, and to copy and distribute the resulting executable under
 | 
						|
terms of your choice, provided that you also meet, for each linked
 | 
						|
independent module, the terms and conditions of the license of that
 | 
						|
module.  An independent module is a module which is not derived from
 | 
						|
or based on this library.  If you modify this library, you may extend
 | 
						|
this exception to your version of the library, but you are not
 | 
						|
obligated to do so.  If you do not wish to do so, delete this
 | 
						|
exception statement from your version. */
 | 
						|
 | 
						|
 | 
						|
package javax.swing.text;
 | 
						|
 | 
						|
import java.awt.Container;
 | 
						|
import java.awt.Graphics;
 | 
						|
import java.awt.Rectangle;
 | 
						|
import java.awt.Shape;
 | 
						|
 | 
						|
import javax.swing.SizeRequirements;
 | 
						|
import javax.swing.event.DocumentEvent;
 | 
						|
 | 
						|
/**
 | 
						|
 * An implementation of {@link CompositeView} that arranges its children in
 | 
						|
 * a box along one axis. This is comparable to how the <code>BoxLayout</code>
 | 
						|
 * works, but for <code>View</code> children.
 | 
						|
 *
 | 
						|
 * @author Roman Kennke (roman@kennke.org)
 | 
						|
 */
 | 
						|
public class BoxView
 | 
						|
  extends CompositeView
 | 
						|
{
 | 
						|
 | 
						|
  /**
 | 
						|
   * The axis along which this <code>BoxView</code> is laid out.
 | 
						|
   */
 | 
						|
  private int myAxis;
 | 
						|
 | 
						|
  /**
 | 
						|
   * Indicates if the layout is valid along X_AXIS or Y_AXIS.
 | 
						|
   */
 | 
						|
  private boolean[] layoutValid = new boolean[2];
 | 
						|
 | 
						|
  /**
 | 
						|
   * Indicates if the requirements for an axis are valid.
 | 
						|
   */
 | 
						|
  private boolean[] requirementsValid = new boolean[2];
 | 
						|
 | 
						|
  /**
 | 
						|
   * The spans along the X_AXIS and Y_AXIS.
 | 
						|
   */
 | 
						|
  private int[][] spans = new int[2][];
 | 
						|
 | 
						|
  /**
 | 
						|
   * The offsets of the children along the X_AXIS and Y_AXIS.
 | 
						|
   */
 | 
						|
  private int[][] offsets = new int[2][];
 | 
						|
 | 
						|
  /**
 | 
						|
   * The size requirements along the X_AXIS and Y_AXIS.
 | 
						|
   */
 | 
						|
  private SizeRequirements[] requirements = new SizeRequirements[2];
 | 
						|
 | 
						|
  /**
 | 
						|
   * The current span along X_AXIS or Y_AXIS.
 | 
						|
   */
 | 
						|
  private int[] span = new int[2];
 | 
						|
 | 
						|
  /**
 | 
						|
   * Creates a new <code>BoxView</code> for the given
 | 
						|
   * <code>Element</code> and axis. Valid values for the axis are
 | 
						|
   * {@link View#X_AXIS} and {@link View#Y_AXIS}.
 | 
						|
   *
 | 
						|
   * @param element the element that is rendered by this BoxView
 | 
						|
   * @param axis the axis along which the box is laid out
 | 
						|
   */
 | 
						|
  public BoxView(Element element, int axis)
 | 
						|
  {
 | 
						|
    super(element);
 | 
						|
    myAxis = axis;
 | 
						|
    layoutValid[0] = false;
 | 
						|
    layoutValid[1] = false;
 | 
						|
    requirementsValid[X_AXIS] = false;
 | 
						|
    requirementsValid[Y_AXIS] = false;
 | 
						|
    span[0] = 0;
 | 
						|
    span[1] = 0;
 | 
						|
    requirements[0] = new SizeRequirements();
 | 
						|
    requirements[1] = new SizeRequirements();
 | 
						|
 | 
						|
    // Initialize the cache arrays.
 | 
						|
    spans[0] = new int[0];
 | 
						|
    spans[1] = new int[0];
 | 
						|
    offsets[0] = new int[0];
 | 
						|
    offsets[1] = new int[0];
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the axis along which this <code>BoxView</code> is laid out.
 | 
						|
   *
 | 
						|
   * @return the axis along which this <code>BoxView</code> is laid out
 | 
						|
   *
 | 
						|
   * @since 1.3
 | 
						|
   */
 | 
						|
  public int getAxis()
 | 
						|
  {
 | 
						|
    return myAxis;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Sets the axis along which this <code>BoxView</code> is laid out.
 | 
						|
   *
 | 
						|
   * Valid values for the axis are {@link View#X_AXIS} and
 | 
						|
   * {@link View#Y_AXIS}.
 | 
						|
   *
 | 
						|
   * @param axis the axis along which this <code>BoxView</code> is laid out
 | 
						|
   *
 | 
						|
   * @since 1.3
 | 
						|
   */
 | 
						|
  public void setAxis(int axis)
 | 
						|
  {
 | 
						|
    boolean changed = axis != myAxis;
 | 
						|
    myAxis = axis;
 | 
						|
    if (changed)
 | 
						|
      preferenceChanged(null, true, true);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Marks the layout along the specified axis as invalid. This is triggered
 | 
						|
   * automatically when any of the child view changes its preferences
 | 
						|
   * via {@link #preferenceChanged(View, boolean, boolean)}.
 | 
						|
   *
 | 
						|
   * The layout will be updated the next time when
 | 
						|
   * {@link #setSize(float, float)} is called, typically from within the
 | 
						|
   * {@link #paint(Graphics, Shape)} method.
 | 
						|
   *
 | 
						|
   * Valid values for the axis are {@link View#X_AXIS} and
 | 
						|
   * {@link View#Y_AXIS}.
 | 
						|
   *
 | 
						|
   * @param axis an <code>int</code> value
 | 
						|
   *
 | 
						|
   * @since 1.3
 | 
						|
   */
 | 
						|
  public void layoutChanged(int axis)
 | 
						|
  {
 | 
						|
    if (axis != X_AXIS && axis != Y_AXIS)
 | 
						|
      throw new IllegalArgumentException("Invalid axis parameter.");
 | 
						|
    layoutValid[axis] = false;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns <code>true</code> if the layout along the specified
 | 
						|
   * <code>axis</code> is valid, <code>false</code> otherwise.
 | 
						|
   *
 | 
						|
   * Valid values for the axis are {@link View#X_AXIS} and
 | 
						|
   * {@link View#Y_AXIS}.
 | 
						|
   *
 | 
						|
   * @param axis the axis
 | 
						|
   *
 | 
						|
   * @return <code>true</code> if the layout along the specified
 | 
						|
   *         <code>axis</code> is valid, <code>false</code> otherwise
 | 
						|
   *
 | 
						|
   * @since 1.4
 | 
						|
   */
 | 
						|
  protected boolean isLayoutValid(int axis)
 | 
						|
  {
 | 
						|
    if (axis != X_AXIS && axis != Y_AXIS)
 | 
						|
      throw new IllegalArgumentException("Invalid axis parameter.");
 | 
						|
    return layoutValid[axis];
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Paints the child <code>View</code> at the specified <code>index</code>.
 | 
						|
   * This method modifies the actual values in <code>alloc</code> so make
 | 
						|
   * sure you have a copy of the original values if you need them.
 | 
						|
   *
 | 
						|
   * @param g the <code>Graphics</code> context to paint to
 | 
						|
   * @param alloc the allocated region for the child to paint into
 | 
						|
   * @param index the index of the child to be painted
 | 
						|
   *
 | 
						|
   * @see #childAllocation(int, Rectangle)
 | 
						|
   */
 | 
						|
  protected void paintChild(Graphics g, Rectangle alloc, int index)
 | 
						|
  {
 | 
						|
    View child = getView(index);
 | 
						|
    child.paint(g, alloc);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Replaces child views by some other child views. If there are no views to
 | 
						|
   * remove (<code>length == 0</code>), the result is a simple insert, if
 | 
						|
   * there are no children to add (<code>view == null</code>) the result
 | 
						|
   * is a simple removal.
 | 
						|
   *
 | 
						|
   * In addition this invalidates the layout and resizes the internal cache
 | 
						|
   * for the child allocations. The old children's cached allocations can
 | 
						|
   * still be accessed (although they are not guaranteed to be valid), and
 | 
						|
   * the new children will have an initial offset and span of 0.
 | 
						|
   *
 | 
						|
   * @param offset the start offset from where to remove children
 | 
						|
   * @param length the number of children to remove
 | 
						|
   * @param views the views that replace the removed children
 | 
						|
   */
 | 
						|
  public void replace(int offset, int length, View[] views)
 | 
						|
  {
 | 
						|
    // Actually perform the replace.
 | 
						|
    super.replace(offset, length, views);
 | 
						|
 | 
						|
    // Resize and copy data for cache arrays.
 | 
						|
    int newItems = views != null ? views.length : 0;
 | 
						|
    int minor = 1 - myAxis;
 | 
						|
    offsets[myAxis] = replaceLayoutArray(offsets[myAxis], offset, newItems);
 | 
						|
    spans[myAxis] = replaceLayoutArray(spans[myAxis], offset, newItems);
 | 
						|
    layoutValid[myAxis] = false;
 | 
						|
    requirementsValid[myAxis] = false;
 | 
						|
    offsets[minor] = replaceLayoutArray(offsets[minor], offset, newItems);
 | 
						|
    spans[minor] = replaceLayoutArray(spans[minor], offset, newItems);
 | 
						|
    layoutValid[minor] = false;
 | 
						|
    requirementsValid[minor] = false;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Helper method. This updates the layout cache arrays in response
 | 
						|
   * to a call to {@link #replace(int, int, View[])}.
 | 
						|
   *
 | 
						|
   * @param oldArray the old array
 | 
						|
   *
 | 
						|
   * @return the replaced array
 | 
						|
   */
 | 
						|
  private int[] replaceLayoutArray(int[] oldArray, int offset, int newItems)
 | 
						|
 | 
						|
  {
 | 
						|
    int num = getViewCount();
 | 
						|
    int[] newArray = new int[num];
 | 
						|
    System.arraycopy(oldArray, 0, newArray, 0, offset);
 | 
						|
    System.arraycopy(oldArray, offset, newArray, offset + newItems,
 | 
						|
                     num - newItems - offset);
 | 
						|
    return newArray;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * A Rectangle instance to be reused in the paint() method below.
 | 
						|
   */
 | 
						|
  private final Rectangle tmpRect = new Rectangle();
 | 
						|
 | 
						|
  private Rectangle clipRect = new Rectangle();
 | 
						|
 | 
						|
  /**
 | 
						|
   * Renders the <code>Element</code> that is associated with this
 | 
						|
   * <code>View</code>.
 | 
						|
   *
 | 
						|
   * @param g the <code>Graphics</code> context to render to
 | 
						|
   * @param a the allocated region for the <code>Element</code>
 | 
						|
   */
 | 
						|
  public void paint(Graphics g, Shape a)
 | 
						|
  {
 | 
						|
    // Try to avoid allocation if possible (almost all cases).
 | 
						|
    Rectangle alloc = a instanceof Rectangle ? (Rectangle) a : a.getBounds();
 | 
						|
 | 
						|
    // This returns a cached instance.
 | 
						|
    alloc = getInsideAllocation(alloc);
 | 
						|
 | 
						|
    int count = getViewCount();
 | 
						|
    for (int i = 0; i < count; i++)
 | 
						|
      {
 | 
						|
        View child = getView(i);
 | 
						|
        tmpRect.setBounds(alloc);
 | 
						|
        childAllocation(i, tmpRect);
 | 
						|
        if (g.hitClip(tmpRect.x, tmpRect.y, tmpRect.width, tmpRect.height))
 | 
						|
          paintChild(g, tmpRect, i);
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the preferred span of the content managed by this
 | 
						|
   * <code>View</code> along the specified <code>axis</code>.
 | 
						|
   *
 | 
						|
   * @param axis the axis
 | 
						|
   *
 | 
						|
   * @return the preferred span of this <code>View</code>.
 | 
						|
   */
 | 
						|
  public float getPreferredSpan(int axis)
 | 
						|
  {
 | 
						|
    updateRequirements(axis);
 | 
						|
    // Add margin.
 | 
						|
    float margin;
 | 
						|
    if (axis == X_AXIS)
 | 
						|
      margin = getLeftInset() + getRightInset();
 | 
						|
    else
 | 
						|
      margin = getTopInset() + getBottomInset();
 | 
						|
    return requirements[axis].preferred + margin;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the maximum span of this view along the specified axis.
 | 
						|
   * This returns <code>Integer.MAX_VALUE</code> for the minor axis
 | 
						|
   * and the preferred span for the major axis.
 | 
						|
   *
 | 
						|
   * @param axis the axis
 | 
						|
   *
 | 
						|
   * @return the maximum span of this view along the specified axis
 | 
						|
   */
 | 
						|
  public float getMaximumSpan(int axis)
 | 
						|
  {
 | 
						|
    updateRequirements(axis);
 | 
						|
    // Add margin.
 | 
						|
    float margin;
 | 
						|
    if (axis == X_AXIS)
 | 
						|
      margin = getLeftInset() + getRightInset();
 | 
						|
    else
 | 
						|
      margin = getTopInset() + getBottomInset();
 | 
						|
    return requirements[axis].maximum + margin;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the minimum span of this view along the specified axis.
 | 
						|
   * This calculates the minimum span using
 | 
						|
   * {@link #calculateMajorAxisRequirements} or
 | 
						|
   * {@link #calculateMinorAxisRequirements} (depending on the axis) and
 | 
						|
   * returns the resulting minimum span.
 | 
						|
   *
 | 
						|
   * @param axis the axis
 | 
						|
   *
 | 
						|
   * @return the minimum span of this view along the specified axis
 | 
						|
   */
 | 
						|
  public float getMinimumSpan(int axis)
 | 
						|
  {
 | 
						|
    updateRequirements(axis);
 | 
						|
    // Add margin.
 | 
						|
    float margin;
 | 
						|
    if (axis == X_AXIS)
 | 
						|
      margin = getLeftInset() + getRightInset();
 | 
						|
    else
 | 
						|
      margin = getTopInset() + getBottomInset();
 | 
						|
    return requirements[axis].minimum + margin;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Calculates size requirements for a baseline layout. This is not
 | 
						|
   * used by the BoxView itself, but by subclasses that wish to perform
 | 
						|
   * a baseline layout, like the FlowView's rows.
 | 
						|
   *
 | 
						|
   * @param axis the axis that is examined
 | 
						|
   * @param sr the <code>SizeRequirements</code> object to hold the result,
 | 
						|
   *        if <code>null</code>, a new one is created
 | 
						|
   *
 | 
						|
   * @return the size requirements for this <code>BoxView</code> along
 | 
						|
   *         the specified axis
 | 
						|
   */
 | 
						|
  protected SizeRequirements baselineRequirements(int axis,
 | 
						|
                                                  SizeRequirements sr)
 | 
						|
  {
 | 
						|
    // Create new instance if sr == null.
 | 
						|
    if (sr == null)
 | 
						|
      sr = new SizeRequirements();
 | 
						|
    sr.alignment = 0.5F;
 | 
						|
 | 
						|
    // Calculate overall ascent and descent.
 | 
						|
    int totalAscentMin = 0;
 | 
						|
    int totalAscentPref = 0;
 | 
						|
    int totalAscentMax = 0;
 | 
						|
    int totalDescentMin = 0;
 | 
						|
    int totalDescentPref = 0;
 | 
						|
    int totalDescentMax = 0;
 | 
						|
 | 
						|
    int count = getViewCount();
 | 
						|
    for (int i = 0; i < count; i++)
 | 
						|
      {
 | 
						|
        View v = getView(i);
 | 
						|
        float align = v.getAlignment(axis);
 | 
						|
        int span = (int) v.getPreferredSpan(axis);
 | 
						|
        int ascent = (int) (align * span);
 | 
						|
        int descent = span - ascent;
 | 
						|
 | 
						|
        totalAscentPref = Math.max(ascent, totalAscentPref);
 | 
						|
        totalDescentPref = Math.max(descent, totalDescentPref);
 | 
						|
        if (v.getResizeWeight(axis) > 0)
 | 
						|
          {
 | 
						|
            // If the view is resizable, then use the min and max size
 | 
						|
            // of the view.
 | 
						|
            span = (int) v.getMinimumSpan(axis);
 | 
						|
            ascent = (int) (align * span);
 | 
						|
            descent = span - ascent;
 | 
						|
            totalAscentMin = Math.max(ascent, totalAscentMin);
 | 
						|
            totalDescentMin = Math.max(descent, totalDescentMin);
 | 
						|
 | 
						|
            span = (int) v.getMaximumSpan(axis);
 | 
						|
            ascent = (int) (align * span);
 | 
						|
            descent = span - ascent;
 | 
						|
            totalAscentMax = Math.max(ascent, totalAscentMax);
 | 
						|
            totalDescentMax = Math.max(descent, totalDescentMax);
 | 
						|
          }
 | 
						|
        else
 | 
						|
          {
 | 
						|
            // If the view is not resizable, use the preferred span.
 | 
						|
            totalAscentMin = Math.max(ascent, totalAscentMin);
 | 
						|
            totalDescentMin = Math.max(descent, totalDescentMin);
 | 
						|
            totalAscentMax = Math.max(ascent, totalAscentMax);
 | 
						|
            totalDescentMax = Math.max(descent, totalDescentMax);
 | 
						|
          }
 | 
						|
      }
 | 
						|
 | 
						|
    // Preferred overall span is the sum of the preferred ascent and descent.
 | 
						|
    // With overflow check.
 | 
						|
    sr.preferred = (int) Math.min((long) totalAscentPref
 | 
						|
                                  + (long) totalDescentPref,
 | 
						|
                                  Integer.MAX_VALUE);
 | 
						|
 | 
						|
    // Align along the baseline.
 | 
						|
    if (sr.preferred > 0)
 | 
						|
      sr.alignment = (float) totalAscentPref / sr.preferred;
 | 
						|
 | 
						|
    if (sr.alignment == 0)
 | 
						|
      {
 | 
						|
        // Nothing above the baseline, use the descent.
 | 
						|
        sr.minimum = totalDescentMin;
 | 
						|
        sr.maximum = totalDescentMax;
 | 
						|
      }
 | 
						|
    else if (sr.alignment == 1.0F)
 | 
						|
      {
 | 
						|
        // Nothing below the baseline, use the descent.
 | 
						|
        sr.minimum = totalAscentMin;
 | 
						|
        sr.maximum = totalAscentMax;
 | 
						|
      }
 | 
						|
    else
 | 
						|
      {
 | 
						|
        sr.minimum = Math.max((int) (totalAscentMin / sr.alignment),
 | 
						|
                              (int) (totalDescentMin / (1.0F - sr.alignment)));
 | 
						|
        sr.maximum = Math.min((int) (totalAscentMax / sr.alignment),
 | 
						|
                              (int) (totalDescentMax / (1.0F - sr.alignment)));
 | 
						|
      }
 | 
						|
    return sr;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Calculates the baseline layout of the children of this
 | 
						|
   * <code>BoxView</code> along the specified axis.
 | 
						|
   *
 | 
						|
   * This is not used by the BoxView itself, but by subclasses that wish to
 | 
						|
   * perform a baseline layout, like the FlowView's rows.
 | 
						|
   *
 | 
						|
   * @param span the target span
 | 
						|
   * @param axis the axis that is examined
 | 
						|
   * @param offsets an empty array, filled with the offsets of the children
 | 
						|
   * @param spans an empty array, filled with the spans of the children
 | 
						|
   */
 | 
						|
  protected void baselineLayout(int span, int axis, int[] offsets,
 | 
						|
                                int[] spans)
 | 
						|
  {
 | 
						|
    int totalAscent = (int) (span * getAlignment(axis));
 | 
						|
    int totalDescent = span - totalAscent;
 | 
						|
 | 
						|
    int count = getViewCount();
 | 
						|
    for (int i = 0; i < count; i++)
 | 
						|
      {
 | 
						|
        View v = getView(i);
 | 
						|
        float align = v.getAlignment(axis);
 | 
						|
        int viewSpan;
 | 
						|
        if (v.getResizeWeight(axis) > 0)
 | 
						|
          {
 | 
						|
            // If possible, then resize for best fit.
 | 
						|
            int min = (int) v.getMinimumSpan(axis);
 | 
						|
            int max = (int) v.getMaximumSpan(axis);
 | 
						|
            if (align == 0.0F)
 | 
						|
              viewSpan = Math.max(Math.min(max, totalDescent), min);
 | 
						|
            else if (align == 1.0F)
 | 
						|
              viewSpan = Math.max(Math.min(max, totalAscent), min);
 | 
						|
            else
 | 
						|
              {
 | 
						|
                int fit = (int) Math.min(totalAscent / align,
 | 
						|
                                         totalDescent / (1.0F - align));
 | 
						|
                viewSpan = Math.max(Math.min(max, fit), min);
 | 
						|
              }
 | 
						|
          }
 | 
						|
        else
 | 
						|
          viewSpan = (int) v.getPreferredSpan(axis);
 | 
						|
        offsets[i] = totalAscent - (int) (viewSpan * align);
 | 
						|
        spans[i] = viewSpan;
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Calculates the size requirements of this <code>BoxView</code> along
 | 
						|
   * its major axis, that is the axis specified in the constructor.
 | 
						|
   *
 | 
						|
   * @param axis the axis that is examined
 | 
						|
   * @param sr the <code>SizeRequirements</code> object to hold the result,
 | 
						|
   *        if <code>null</code>, a new one is created
 | 
						|
   *
 | 
						|
   * @return the size requirements for this <code>BoxView</code> along
 | 
						|
   *         the specified axis
 | 
						|
   */
 | 
						|
  protected SizeRequirements calculateMajorAxisRequirements(int axis,
 | 
						|
                                                           SizeRequirements sr)
 | 
						|
  {
 | 
						|
    SizeRequirements res = sr;
 | 
						|
    if (res == null)
 | 
						|
      res = new SizeRequirements();
 | 
						|
 | 
						|
    float min = 0;
 | 
						|
    float pref = 0;
 | 
						|
    float max = 0;
 | 
						|
 | 
						|
    int n = getViewCount();
 | 
						|
    for (int i = 0; i < n; i++)
 | 
						|
      {
 | 
						|
        View child = getView(i);
 | 
						|
        min += child.getMinimumSpan(axis);
 | 
						|
        pref += child.getPreferredSpan(axis);
 | 
						|
        max += child.getMaximumSpan(axis);
 | 
						|
      }
 | 
						|
 | 
						|
    res.minimum = (int) min;
 | 
						|
    res.preferred = (int) pref;
 | 
						|
    res.maximum = (int) max;
 | 
						|
    res.alignment = 0.5F;
 | 
						|
 | 
						|
    return res;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Calculates the size requirements of this <code>BoxView</code> along
 | 
						|
   * its minor axis, that is the axis opposite to the axis specified in the
 | 
						|
   * constructor.
 | 
						|
   *
 | 
						|
   * @param axis the axis that is examined
 | 
						|
   * @param sr the <code>SizeRequirements</code> object to hold the result,
 | 
						|
   *        if <code>null</code>, a new one is created
 | 
						|
   *
 | 
						|
   * @return the size requirements for this <code>BoxView</code> along
 | 
						|
   *         the specified axis
 | 
						|
   */
 | 
						|
  protected SizeRequirements calculateMinorAxisRequirements(int axis,
 | 
						|
                                                            SizeRequirements sr)
 | 
						|
  {
 | 
						|
    SizeRequirements res = sr;
 | 
						|
    if (res == null)
 | 
						|
      res = new SizeRequirements();
 | 
						|
 | 
						|
    res.minimum = 0;
 | 
						|
    res.preferred = 0;
 | 
						|
    res.maximum = Integer.MAX_VALUE;
 | 
						|
    res.alignment = 0.5F;
 | 
						|
    int n = getViewCount();
 | 
						|
    for (int i = 0; i < n; i++)
 | 
						|
      {
 | 
						|
        View child = getView(i);
 | 
						|
        res.minimum = Math.max((int) child.getMinimumSpan(axis), res.minimum);
 | 
						|
        res.preferred = Math.max((int) child.getPreferredSpan(axis),
 | 
						|
                                 res.preferred);
 | 
						|
        res.maximum = Math.max((int) child.getMaximumSpan(axis), res.maximum);
 | 
						|
      }
 | 
						|
 | 
						|
    return res;
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns <code>true</code> if the specified point lies before the
 | 
						|
   * given <code>Rectangle</code>, <code>false</code> otherwise.
 | 
						|
   *
 | 
						|
   * "Before" is typically defined as being to the left or above.
 | 
						|
   *
 | 
						|
   * @param x the X coordinate of the point
 | 
						|
   * @param y the Y coordinate of the point
 | 
						|
   * @param r the rectangle to test the point against
 | 
						|
   *
 | 
						|
   * @return <code>true</code> if the specified point lies before the
 | 
						|
   *         given <code>Rectangle</code>, <code>false</code> otherwise
 | 
						|
   */
 | 
						|
  protected boolean isBefore(int x, int y, Rectangle r)
 | 
						|
  {
 | 
						|
    boolean result = false;
 | 
						|
 | 
						|
    if (myAxis == X_AXIS)
 | 
						|
      result = x < r.x;
 | 
						|
    else
 | 
						|
      result = y < r.y;
 | 
						|
 | 
						|
    return result;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns <code>true</code> if the specified point lies after the
 | 
						|
   * given <code>Rectangle</code>, <code>false</code> otherwise.
 | 
						|
   *
 | 
						|
   * "After" is typically defined as being to the right or below.
 | 
						|
   *
 | 
						|
   * @param x the X coordinate of the point
 | 
						|
   * @param y the Y coordinate of the point
 | 
						|
   * @param r the rectangle to test the point against
 | 
						|
   *
 | 
						|
   * @return <code>true</code> if the specified point lies after the
 | 
						|
   *         given <code>Rectangle</code>, <code>false</code> otherwise
 | 
						|
   */
 | 
						|
  protected boolean isAfter(int x, int y, Rectangle r)
 | 
						|
  {
 | 
						|
    boolean result = false;
 | 
						|
 | 
						|
    if (myAxis == X_AXIS)
 | 
						|
      result = x > r.x + r.width;
 | 
						|
    else
 | 
						|
      result = y > r.y + r.height;
 | 
						|
 | 
						|
    return result;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the child <code>View</code> at the specified location.
 | 
						|
   *
 | 
						|
   * @param x the X coordinate
 | 
						|
   * @param y the Y coordinate
 | 
						|
   * @param r the inner allocation of this <code>BoxView</code> on entry,
 | 
						|
   *        the allocation of the found child on exit
 | 
						|
   *
 | 
						|
   * @return the child <code>View</code> at the specified location
 | 
						|
   */
 | 
						|
  protected View getViewAtPoint(int x, int y, Rectangle r)
 | 
						|
  {
 | 
						|
    View result = null;
 | 
						|
    int count = getViewCount();
 | 
						|
    if (myAxis == X_AXIS)
 | 
						|
      {
 | 
						|
        // Border case. Requested point is left from the box.
 | 
						|
        if (x < r.x + offsets[X_AXIS][0])
 | 
						|
          {
 | 
						|
            childAllocation(0, r);
 | 
						|
            result = getView(0);
 | 
						|
          }
 | 
						|
        else
 | 
						|
          {
 | 
						|
            // Search views inside box.
 | 
						|
            for (int i = 0; i < count && result == null; i++)
 | 
						|
              {
 | 
						|
                if (x < r.x + offsets[X_AXIS][i])
 | 
						|
                  {
 | 
						|
                    childAllocation(i - 1, r);
 | 
						|
                    result = getView(i - 1);
 | 
						|
                  }
 | 
						|
              }
 | 
						|
          }
 | 
						|
      }
 | 
						|
    else // Same algorithm for Y_AXIS.
 | 
						|
      {
 | 
						|
        // Border case. Requested point is above the box.
 | 
						|
        if (y < r.y + offsets[Y_AXIS][0])
 | 
						|
          {
 | 
						|
            childAllocation(0, r);
 | 
						|
            result = getView(0);
 | 
						|
          }
 | 
						|
        else
 | 
						|
          {
 | 
						|
            // Search views inside box.
 | 
						|
            for (int i = 0; i < count && result == null; i++)
 | 
						|
              {
 | 
						|
                if (y < r.y + offsets[Y_AXIS][i])
 | 
						|
                  {
 | 
						|
                    childAllocation(i - 1, r);
 | 
						|
                    result = getView(i - 1);
 | 
						|
                  }
 | 
						|
              }
 | 
						|
          }
 | 
						|
      }
 | 
						|
    // Not found, other border case: point is right from or below the box.
 | 
						|
    if (result == null)
 | 
						|
      {
 | 
						|
        childAllocation(count - 1, r);
 | 
						|
        result = getView(count - 1);
 | 
						|
      }
 | 
						|
    return result;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Computes the allocation for a child <code>View</code>. The parameter
 | 
						|
   * <code>a</code> stores the allocation of this <code>CompositeView</code>
 | 
						|
   * and is then adjusted to hold the allocation of the child view.
 | 
						|
   *
 | 
						|
   * @param index
 | 
						|
   *          the index of the child <code>View</code>
 | 
						|
   * @param a
 | 
						|
   *          the allocation of this <code>CompositeView</code> before the
 | 
						|
   *          call, the allocation of the child on exit
 | 
						|
   */
 | 
						|
  protected void childAllocation(int index, Rectangle a)
 | 
						|
  {
 | 
						|
    a.x += offsets[X_AXIS][index];
 | 
						|
    a.y += offsets[Y_AXIS][index];
 | 
						|
    a.width = spans[X_AXIS][index];
 | 
						|
    a.height = spans[Y_AXIS][index];
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Lays out the children of this <code>BoxView</code> with the specified
 | 
						|
   * bounds.
 | 
						|
   *
 | 
						|
   * @param width the width of the allocated region for the children (that
 | 
						|
   *        is the inner allocation of this <code>BoxView</code>
 | 
						|
   * @param height the height of the allocated region for the children (that
 | 
						|
   *        is the inner allocation of this <code>BoxView</code>
 | 
						|
   */
 | 
						|
  protected void layout(int width, int height)
 | 
						|
  {
 | 
						|
    layoutAxis(X_AXIS, width);
 | 
						|
    layoutAxis(Y_AXIS, height);
 | 
						|
  }
 | 
						|
 | 
						|
  private void layoutAxis(int axis, int s)
 | 
						|
  {
 | 
						|
    if (span[axis] != s)
 | 
						|
      layoutValid[axis] = false;
 | 
						|
    if (! layoutValid[axis])
 | 
						|
      {
 | 
						|
        span[axis] = s;
 | 
						|
        updateRequirements(axis);
 | 
						|
        if (axis == myAxis)
 | 
						|
          layoutMajorAxis(span[axis], axis, offsets[axis], spans[axis]);
 | 
						|
        else
 | 
						|
          layoutMinorAxis(span[axis], axis, offsets[axis], spans[axis]);
 | 
						|
        layoutValid[axis] = true;
 | 
						|
 | 
						|
        // Push out child layout.
 | 
						|
        int viewCount = getViewCount();
 | 
						|
        for (int i = 0; i < viewCount; i++)
 | 
						|
          {
 | 
						|
            View v = getView(i);
 | 
						|
            v.setSize(spans[X_AXIS][i], spans[Y_AXIS][i]);
 | 
						|
          }
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Performs the layout along the major axis of a <code>BoxView</code>.
 | 
						|
   *
 | 
						|
   * @param targetSpan the (inner) span of the <code>BoxView</code> in which
 | 
						|
   *        to layout the children
 | 
						|
   * @param axis the axis along which the layout is performed
 | 
						|
   * @param offsets the array that holds the offsets of the children on exit
 | 
						|
   * @param spans the array that holds the spans of the children on exit
 | 
						|
   */
 | 
						|
  protected void layoutMajorAxis(int targetSpan, int axis, int[] offsets,
 | 
						|
                                 int[] spans)
 | 
						|
  {
 | 
						|
    // Set the spans to the preferred sizes. Determine the space
 | 
						|
    // that we have to adjust the sizes afterwards.
 | 
						|
    long sumPref = 0;
 | 
						|
    int n = getViewCount();
 | 
						|
    for (int i = 0; i < n; i++)
 | 
						|
      {
 | 
						|
        View child = getView(i);
 | 
						|
        spans[i] = (int) child.getPreferredSpan(axis);
 | 
						|
        sumPref += spans[i];
 | 
						|
      }
 | 
						|
 | 
						|
    // Try to adjust the spans so that we fill the targetSpan.
 | 
						|
    long diff = targetSpan - sumPref;
 | 
						|
    float factor = 0.0F;
 | 
						|
    int[] diffs = null;
 | 
						|
    if (diff != 0)
 | 
						|
      {
 | 
						|
        long total = 0;
 | 
						|
        diffs = new int[n];
 | 
						|
        for (int i = 0; i < n; i++)
 | 
						|
          {
 | 
						|
            View child = getView(i);
 | 
						|
            int span;
 | 
						|
            if (diff < 0)
 | 
						|
              {
 | 
						|
                span = (int) child.getMinimumSpan(axis);
 | 
						|
                diffs[i] = spans[i] - span;
 | 
						|
              }
 | 
						|
            else
 | 
						|
              {
 | 
						|
                span = (int) child.getMaximumSpan(axis);
 | 
						|
                diffs[i] = span - spans[i];
 | 
						|
              }
 | 
						|
            total += span;
 | 
						|
          }
 | 
						|
 | 
						|
        float maxAdjust = Math.abs(total - sumPref);
 | 
						|
        factor = diff / maxAdjust;
 | 
						|
        factor = Math.min(factor, 1.0F);
 | 
						|
        factor = Math.max(factor, -1.0F);
 | 
						|
      }
 | 
						|
 | 
						|
    // Actually perform adjustments.
 | 
						|
    int totalOffs = 0;
 | 
						|
    for (int i = 0; i < n; i++)
 | 
						|
      {
 | 
						|
        offsets[i] = totalOffs;
 | 
						|
        if (diff != 0)
 | 
						|
          {
 | 
						|
            float adjust = factor * diffs[i];
 | 
						|
            spans[i] += Math.round(adjust);
 | 
						|
          }
 | 
						|
        // Avoid overflow here.
 | 
						|
        totalOffs = (int) Math.min((long) totalOffs + (long) spans[i],
 | 
						|
                                    Integer.MAX_VALUE);
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Performs the layout along the minor axis of a <code>BoxView</code>.
 | 
						|
   *
 | 
						|
   * @param targetSpan the (inner) span of the <code>BoxView</code> in which
 | 
						|
   *        to layout the children
 | 
						|
   * @param axis the axis along which the layout is performed
 | 
						|
   * @param offsets the array that holds the offsets of the children on exit
 | 
						|
   * @param spans the array that holds the spans of the children on exit
 | 
						|
   */
 | 
						|
  protected void layoutMinorAxis(int targetSpan, int axis, int[] offsets,
 | 
						|
                                 int[] spans)
 | 
						|
  {
 | 
						|
    int count = getViewCount();
 | 
						|
    for (int i = 0; i < count; i++)
 | 
						|
      {
 | 
						|
        View child = getView(i);
 | 
						|
        int max = (int) child.getMaximumSpan(axis);
 | 
						|
        if (max < targetSpan)
 | 
						|
          {
 | 
						|
            // Align child when it can't be made as wide as the target span.
 | 
						|
            float align = child.getAlignment(axis);
 | 
						|
            offsets[i] = (int) ((targetSpan - max) * align);
 | 
						|
            spans[i] = max;
 | 
						|
          }
 | 
						|
        else
 | 
						|
          {
 | 
						|
            // Expand child to target width if possible.
 | 
						|
            int min = (int) child.getMinimumSpan(axis);
 | 
						|
            offsets[i] = 0;
 | 
						|
            spans[i] = Math.max(min, targetSpan);
 | 
						|
          }
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns <code>true</code> if the cached allocations for the children
 | 
						|
   * are still valid, <code>false</code> otherwise.
 | 
						|
   *
 | 
						|
   * @return <code>true</code> if the cached allocations for the children
 | 
						|
   *         are still valid, <code>false</code> otherwise
 | 
						|
   */
 | 
						|
  protected boolean isAllocationValid()
 | 
						|
  {
 | 
						|
    return isLayoutValid(X_AXIS) && isLayoutValid(Y_AXIS);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Return the current width of the box. This is the last allocated width.
 | 
						|
   *
 | 
						|
   * @return the current width of the box
 | 
						|
   */
 | 
						|
  public int getWidth()
 | 
						|
  {
 | 
						|
    // The RI returns the following here, however, I'd think that is a bug.
 | 
						|
    // return span[X_AXIS] + getLeftInset() - getRightInset();
 | 
						|
    return span[X_AXIS] + getLeftInset() + getRightInset();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Return the current height of the box. This is the last allocated height.
 | 
						|
   *
 | 
						|
   * @return the current height of the box
 | 
						|
   */
 | 
						|
  public int getHeight()
 | 
						|
  {
 | 
						|
    // The RI returns the following here, however, I'd think that is a bug.
 | 
						|
    // return span[Y_AXIS] + getTopInset() - getBottomInset();
 | 
						|
    return span[Y_AXIS] + getTopInset() + getBottomInset();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Sets the size of the view. If the actual size has changed, the layout
 | 
						|
   * is updated accordingly.
 | 
						|
   *
 | 
						|
   * @param width the new width
 | 
						|
   * @param height the new height
 | 
						|
   */
 | 
						|
  public void setSize(float width, float height)
 | 
						|
  {
 | 
						|
    layout((int) (width - getLeftInset() - getRightInset()),
 | 
						|
           (int) (height - getTopInset() - getBottomInset()));
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the span for the child view with the given index for the specified
 | 
						|
   * axis.
 | 
						|
   *
 | 
						|
   * @param axis the axis to examine, either <code>X_AXIS</code> or
 | 
						|
   *        <code>Y_AXIS</code>
 | 
						|
   * @param childIndex the index of the child for for which to return the span
 | 
						|
   *
 | 
						|
   * @return the span for the child view with the given index for the specified
 | 
						|
   *         axis
 | 
						|
   */
 | 
						|
  protected int getSpan(int axis, int childIndex)
 | 
						|
  {
 | 
						|
    if (axis != X_AXIS && axis != Y_AXIS)
 | 
						|
      throw new IllegalArgumentException("Illegal axis argument");
 | 
						|
    return spans[axis][childIndex];
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the offset for the child view with the given index for the
 | 
						|
   * specified axis.
 | 
						|
   *
 | 
						|
   * @param axis the axis to examine, either <code>X_AXIS</code> or
 | 
						|
   *        <code>Y_AXIS</code>
 | 
						|
   * @param childIndex the index of the child for for which to return the span
 | 
						|
   *
 | 
						|
   * @return the offset for the child view with the given index for the
 | 
						|
   *         specified axis
 | 
						|
   */
 | 
						|
  protected int getOffset(int axis, int childIndex)
 | 
						|
  {
 | 
						|
    if (axis != X_AXIS && axis != Y_AXIS)
 | 
						|
      throw new IllegalArgumentException("Illegal axis argument");
 | 
						|
    return offsets[axis][childIndex];
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the alignment for this box view for the specified axis. The
 | 
						|
   * axis that is tiled (the major axis) will be requested to be aligned
 | 
						|
   * centered (0.5F). The minor axis alignment depends on the child view's
 | 
						|
   * total alignment.
 | 
						|
   *
 | 
						|
   * @param axis the axis which is examined
 | 
						|
   *
 | 
						|
   * @return the alignment for this box view for the specified axis
 | 
						|
   */
 | 
						|
  public float getAlignment(int axis)
 | 
						|
  {
 | 
						|
     updateRequirements(axis);
 | 
						|
     return requirements[axis].alignment;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Called by a child View when its preferred span has changed.
 | 
						|
   *
 | 
						|
   * @param width indicates that the preferred width of the child changed.
 | 
						|
   * @param height indicates that the preferred height of the child changed.
 | 
						|
   * @param child the child View.
 | 
						|
   */
 | 
						|
  public void preferenceChanged(View child, boolean width, boolean height)
 | 
						|
  {
 | 
						|
    if (width)
 | 
						|
      {
 | 
						|
        layoutValid[X_AXIS] = false;
 | 
						|
        requirementsValid[X_AXIS] = false;
 | 
						|
      }
 | 
						|
    if (height)
 | 
						|
      {
 | 
						|
        layoutValid[Y_AXIS] = false;
 | 
						|
        requirementsValid[Y_AXIS] = false;
 | 
						|
      }
 | 
						|
    super.preferenceChanged(child, width, height);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Maps the document model position <code>pos</code> to a Shape
 | 
						|
   * in the view coordinate space.  This method overrides CompositeView's
 | 
						|
   * method to make sure the children are allocated properly before
 | 
						|
   * calling the super's behaviour.
 | 
						|
   */
 | 
						|
  public Shape modelToView(int pos, Shape a, Position.Bias bias)
 | 
						|
      throws BadLocationException
 | 
						|
  {
 | 
						|
    // Make sure everything is allocated properly and then call super
 | 
						|
    if (! isAllocationValid())
 | 
						|
      {
 | 
						|
        Rectangle bounds = a.getBounds();
 | 
						|
        setSize(bounds.width, bounds.height);
 | 
						|
      }
 | 
						|
    return super.modelToView(pos, a, bias);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the resize weight of this view. A value of <code>0</code> or less
 | 
						|
   * means this view is not resizeable. Positive values make the view
 | 
						|
   * resizeable. This implementation returns <code>0</code> for the major
 | 
						|
   * axis and <code>1</code> for the minor axis of this box view.
 | 
						|
   *
 | 
						|
   * @param axis the axis
 | 
						|
   *
 | 
						|
   * @return the resizability of this view along the specified axis
 | 
						|
   *
 | 
						|
   * @throws IllegalArgumentException if <code>axis</code> is invalid
 | 
						|
   */
 | 
						|
  public int getResizeWeight(int axis)
 | 
						|
  {
 | 
						|
    if (axis != X_AXIS && axis != Y_AXIS)
 | 
						|
      throw new IllegalArgumentException("Illegal axis argument");
 | 
						|
    updateRequirements(axis);
 | 
						|
    int weight = 0;
 | 
						|
    if ((requirements[axis].preferred != requirements[axis].minimum)
 | 
						|
        || (requirements[axis].preferred != requirements[axis].maximum))
 | 
						|
      weight = 1;
 | 
						|
    return weight;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns the child allocation for the child view with the specified
 | 
						|
   * <code>index</code>. If the layout is invalid, this returns
 | 
						|
   * <code>null</code>.
 | 
						|
   *
 | 
						|
   * @param index the child view index
 | 
						|
   * @param a the allocation to this view
 | 
						|
   *
 | 
						|
   * @return the child allocation for the child view with the specified
 | 
						|
   *         <code>index</code> or <code>null</code> if the layout is invalid
 | 
						|
   *         or <code>a</code> is null
 | 
						|
   */
 | 
						|
  public Shape getChildAllocation(int index, Shape a)
 | 
						|
  {
 | 
						|
    Shape ret = null;
 | 
						|
    if (isAllocationValid() && a != null)
 | 
						|
      ret = super.getChildAllocation(index, a);
 | 
						|
    return ret;
 | 
						|
  }
 | 
						|
 | 
						|
  protected void forwardUpdate(DocumentEvent.ElementChange ec, DocumentEvent e,
 | 
						|
                               Shape a, ViewFactory vf)
 | 
						|
  {
 | 
						|
    boolean wasValid = isLayoutValid(myAxis);
 | 
						|
    super.forwardUpdate(ec, e, a, vf);
 | 
						|
    // Trigger repaint when one of the children changed the major axis.
 | 
						|
    if (wasValid && ! isLayoutValid(myAxis))
 | 
						|
      {
 | 
						|
        Container c = getContainer();
 | 
						|
        if (a != null && c != null)
 | 
						|
          {
 | 
						|
            int pos = e.getOffset();
 | 
						|
            int index = getViewIndexAtPosition(pos);
 | 
						|
            Rectangle r = getInsideAllocation(a);
 | 
						|
            if (myAxis == X_AXIS)
 | 
						|
              {
 | 
						|
                r.x += offsets[myAxis][index];
 | 
						|
                r.width -= offsets[myAxis][index];
 | 
						|
              }
 | 
						|
            else
 | 
						|
              {
 | 
						|
                r.y += offsets[myAxis][index];
 | 
						|
                r.height -= offsets[myAxis][index];
 | 
						|
              }
 | 
						|
            c.repaint(r.x, r.y, r.width, r.height);
 | 
						|
          }
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  public int viewToModel(float x, float y, Shape a, Position.Bias[] bias)
 | 
						|
  {
 | 
						|
    if (! isAllocationValid())
 | 
						|
      {
 | 
						|
        Rectangle r = a instanceof Rectangle ? (Rectangle) a : a.getBounds();
 | 
						|
        setSize(r.width, r.height);
 | 
						|
      }
 | 
						|
    return super.viewToModel(x, y, a, bias);
 | 
						|
  }
 | 
						|
 | 
						|
  protected boolean flipEastAndWestAtEnds(int position, Position.Bias bias)
 | 
						|
  {
 | 
						|
    // FIXME: What to do here?
 | 
						|
    return super.flipEastAndWestAtEnds(position, bias);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Updates the view's cached requirements along the specified axis if
 | 
						|
   * necessary. The requirements are only updated if the layout for the
 | 
						|
   * specified axis is marked as invalid.
 | 
						|
   *
 | 
						|
   * @param axis the axis
 | 
						|
   */
 | 
						|
  private void updateRequirements(int axis)
 | 
						|
  {
 | 
						|
    if (axis != Y_AXIS && axis != X_AXIS)
 | 
						|
      throw new IllegalArgumentException("Illegal axis: " + axis);
 | 
						|
    if (! requirementsValid[axis])
 | 
						|
      {
 | 
						|
        if (axis == myAxis)
 | 
						|
          requirements[axis] = calculateMajorAxisRequirements(axis,
 | 
						|
                                                           requirements[axis]);
 | 
						|
        else
 | 
						|
          requirements[axis] = calculateMinorAxisRequirements(axis,
 | 
						|
                                                           requirements[axis]);
 | 
						|
        requirementsValid[axis] = true;
 | 
						|
      }
 | 
						|
  }
 | 
						|
}
 |