flowview.java
来自「linux下建立JAVA虚拟机的源码KAFFE」· Java 代码 · 共 606 行 · 第 1/2 页
JAVA
606 行
/* FlowView.java -- A composite View Copyright (C) 2005 Free Software Foundation, Inc.This file is part of GNU Classpath.GNU Classpath is free software; you can redistribute it and/or modifyit under the terms of the GNU General Public License as published bythe 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, butWITHOUT ANY WARRANTY; without even the implied warranty ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNUGeneral Public License for more details.You should have received a copy of the GNU General Public Licensealong with GNU Classpath; see the file COPYING. If not, write to theFree Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA02110-1301 USA.Linking this library statically or dynamically with other modules ismaking a combined work based on this library. Thus, the terms andconditions of the GNU General Public License cover the wholecombination.As a special exception, the copyright holders of this library give youpermission to link this library with independent modules to produce anexecutable, regardless of the license terms of these independentmodules, and to copy and distribute the resulting executable underterms of your choice, provided that you also meet, for each linkedindependent module, the terms and conditions of the license of thatmodule. An independent module is a module which is not derived fromor based on this library. If you modify this library, you may extendthis exception to your version of the library, but you are notobligated to do so. If you do not wish to do so, delete thisexception statement from your version. */package javax.swing.text;import java.awt.Rectangle;import java.awt.Shape;import javax.swing.SizeRequirements;import javax.swing.event.DocumentEvent;/** * A <code>View</code> that can flows it's children into it's layout space. * * The <code>FlowView</code> manages a set of logical views (that are * the children of the {@link #layoutPool} field). These are translated * at layout time into a set of physical views. These are the views that * are managed as the real child views. Each of these child views represents * a row and are laid out within a box using the superclasses behaviour. * The concrete implementation of the rows must be provided by subclasses. * * @author Roman Kennke (roman@kennke.org) */public abstract class FlowView extends BoxView{ /** * A strategy for translating the logical views of a <code>FlowView</code> * into the real views. */ public static class FlowStrategy { /** * Creates a new instance of <code>FlowStragegy</code>. */ public FlowStrategy() { // Nothing to do here. } /** * Receives notification from a <code>FlowView</code> that some content * has been inserted into the document at a location that the * <code>FlowView</code> is responsible for. * * The default implementation simply calls {@link #layout}. * * @param fv the flow view that sends the notification * @param e the document event describing the change * @param alloc the current allocation of the flow view */ public void insertUpdate(FlowView fv, DocumentEvent e, Rectangle alloc) { // The default implementation does nothing. } /** * Receives notification from a <code>FlowView</code> that some content * has been removed from the document at a location that the * <code>FlowView</code> is responsible for. * * The default implementation simply calls {@link #layout}. * * @param fv the flow view that sends the notification * @param e the document event describing the change * @param alloc the current allocation of the flow view */ public void removeUpdate(FlowView fv, DocumentEvent e, Rectangle alloc) { // The default implementation does nothing. } /** * Receives notification from a <code>FlowView</code> that some attributes * have changed in the document at a location that the * <code>FlowView</code> is responsible for. * * The default implementation simply calls {@link #layout}. * * @param fv the flow view that sends the notification * @param e the document event describing the change * @param alloc the current allocation of the flow view */ public void changedUpdate(FlowView fv, DocumentEvent e, Rectangle alloc) { // The default implementation does nothing. } /** * Returns the logical view of the managed <code>FlowView</code>. * * @param fv the flow view for which to return the logical view * * @return the logical view of the managed <code>FlowView</code> */ protected View getLogicalView(FlowView fv) { return fv.layoutPool; } /** * Performs the layout for the whole view. By default this rebuilds * all the physical views from the logical views of the managed FlowView. * * This is called by {@link FlowView#layout} to update the layout of * the view. * * @param fv the flow view for which we perform the layout */ public void layout(FlowView fv) { fv.removeAll(); Element el = fv.getElement(); int rowStart = el.getStartOffset(); int end = el.getEndOffset(); int rowIndex = 0; while (rowStart >= 0 && rowStart < end) { View row = fv.createRow(); fv.append(row); rowStart = layoutRow(fv, rowIndex, rowStart); rowIndex++; } } /** * Lays out one row of the flow view. This is called by {@link #layout} * to fill one row with child views until the available span is exhausted. * * The default implementation fills the row by calling * {@link #createView(FlowView, int, int, int)} until the available space * is exhausted, a forced break is encountered or there are no more views * in the logical view. If the available space is exhausted, * {@link #adjustRow(FlowView, int, int, int)} is called to fit the row * into the available span. * * @param fv the flow view for which we perform the layout * @param rowIndex the index of the row * @param pos the model position for the beginning of the row * * @return the start position of the next row */ protected int layoutRow(FlowView fv, int rowIndex, int pos) { View row = fv.getView(rowIndex); int axis = fv.getFlowAxis(); int span = fv.getFlowSpan(rowIndex); int x = fv.getFlowStart(rowIndex); int offset = pos; View logicalView = getLogicalView(fv); // Special case when span == 0. We need to layout the row as if it had // a span of Integer.MAX_VALUE. if (span == 0) span = Integer.MAX_VALUE; while (span > 0) { if (logicalView.getViewIndex(offset, Position.Bias.Forward) == -1) break; View view = createView(fv, offset, span, rowIndex); if (view == null) break; int viewSpan = (int) view.getPreferredSpan(axis); row.append(view); int breakWeight = view.getBreakWeight(axis, x, span); if (breakWeight >= View.ForcedBreakWeight) break; x += viewSpan; span -= viewSpan; offset += (view.getEndOffset() - view.getStartOffset()); } if (span < 0) { int flowStart = fv.getFlowStart(axis); int flowSpan = fv.getFlowSpan(axis); adjustRow(fv, rowIndex, flowSpan, flowStart); int rowViewCount = row.getViewCount(); if (rowViewCount > 0) offset = row.getView(rowViewCount - 1).getEndOffset(); else offset = -1; } return offset != pos ? offset : -1; } /** * Creates physical views that form the rows of the flow view. This * can be an entire view from the logical view (if it fits within the * available span), a fragment of such a view (if it doesn't fit in the * available span and can be broken down) or <code>null</code> (if it does * not fit in the available span and also cannot be broken down). * * The default implementation fetches the logical view at the specified * <code>startOffset</code>. If that view has a different startOffset than * specified in the argument, a fragment is created using * {@link View#createFragment(int, int)} that has the correct startOffset * and the logical view's endOffset. * * @param fv the flow view * @param startOffset the start offset for the view to be created * @param spanLeft the available span * @param rowIndex the index of the row * * @return a view to fill the row with, or <code>null</code> if there * is no view or view fragment that fits in the available span */ protected View createView(FlowView fv, int startOffset, int spanLeft, int rowIndex) { View logicalView = getLogicalView(fv); // FIXME: Handle the bias thing correctly. int index = logicalView.getViewIndex(startOffset, Position.Bias.Forward); View retVal = null; if (index >= 0) { retVal = logicalView.getView(index); if (retVal.getStartOffset() != startOffset) retVal = retVal.createFragment(startOffset, retVal.getEndOffset()); } return retVal; } /** * Tries to adjust the specified row to fit within the desired span. The * default implementation iterates through the children of the specified * row to find the view that has the highest break weight and - if there * is more than one view with such a break weight - which is nearest to * the end of the row. If there is such a view that has a break weight > * {@link View#BadBreakWeight}, this view is broken using the * {@link View#breakView(int, int, float, float)} method and this view and * all views after the now broken view are replaced by the broken view. * * @param fv the flow view * @param rowIndex the index of the row to be adjusted * @param desiredSpan the layout span * @param x the X location at which the row starts */ protected void adjustRow(FlowView fv, int rowIndex, int desiredSpan, int x) { // Determine the last view that has the highest break weight. int axis = fv.getFlowAxis(); View row = fv.getView(rowIndex); int count = row.getViewCount(); int breakIndex = -1; int maxBreakWeight = View.BadBreakWeight; int breakX = x; int breakSpan = desiredSpan; int currentX = x; int currentSpan = desiredSpan; for (int i = 0; i < count; ++i) { View view = row.getView(i); int weight = view.getBreakWeight(axis, currentX, currentSpan); if (weight >= maxBreakWeight) { breakIndex = i; breakX = currentX; breakSpan = currentSpan; maxBreakWeight = weight; } int size = (int) view.getPreferredSpan(axis); currentX += size; currentSpan -= size; } // If there is a potential break location found, break the row at // this location. if (breakIndex > -1) {
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?