1 /*
   2  * Copyright (c) 1998, 2011, Oracle and/or its affiliates. All rights reserved.
   3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   4  *
   5  * This code is free software; you can redistribute it and/or modify it
   6  * under the terms of the GNU General Public License version 2 only, as
   7  * published by the Free Software Foundation.  Oracle designates this
   8  * particular file as subject to the "Classpath" exception as provided
   9  * by Oracle in the LICENSE file that accompanied this code.
  10  *
  11  * This code is distributed in the hope that it will be useful, but WITHOUT
  12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  14  * version 2 for more details (a copy is included in the LICENSE file that
  15  * accompanied this code).
  16  *
  17  * You should have received a copy of the GNU General Public License version
  18  * 2 along with this work; if not, write to the Free Software Foundation,
  19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  20  *
  21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  22  * or visit www.oracle.com if you need additional information or have any
  23  * questions.
  24  */
  25 package javax.swing.text.html;
  26 
  27 import java.awt.*;
  28 import javax.swing.SizeRequirements;
  29 import javax.swing.event.DocumentEvent;
  30 import javax.swing.text.Document;
  31 import javax.swing.text.Element;
  32 import javax.swing.text.AttributeSet;
  33 import javax.swing.text.StyleConstants;
  34 import javax.swing.text.View;
  35 import javax.swing.text.ViewFactory;
  36 import javax.swing.text.BadLocationException;
  37 import javax.swing.text.JTextComponent;
  38 
  39 /**
  40  * Displays the a paragraph, and uses css attributes for its
  41  * configuration.
  42  *
  43  * @author  Timothy Prinzing
  44  */
  45 
  46 public class ParagraphView extends javax.swing.text.ParagraphView {
  47 
  48     /**
  49      * Constructs a ParagraphView for the given element.
  50      *
  51      * @param elem the element that this view is responsible for
  52      */
  53     public ParagraphView(Element elem) {
  54         super(elem);
  55     }
  56 
  57     /**
  58      * Establishes the parent view for this view.  This is
  59      * guaranteed to be called before any other methods if the
  60      * parent view is functioning properly.
  61      * <p>
  62      * This is implemented
  63      * to forward to the superclass as well as call the
  64      * {@link #setPropertiesFromAttributes setPropertiesFromAttributes}
  65      * method to set the paragraph properties from the css
  66      * attributes.  The call is made at this time to ensure
  67      * the ability to resolve upward through the parents
  68      * view attributes.
  69      *
  70      * @param parent the new parent, or null if the view is
  71      *  being removed from a parent it was previously added
  72      *  to
  73      */
  74     public void setParent(View parent) {
  75         super.setParent(parent);
  76         if (parent != null) {
  77             setPropertiesFromAttributes();
  78         }
  79     }
  80 
  81     /**
  82      * Fetches the attributes to use when rendering.  This is
  83      * implemented to multiplex the attributes specified in the
  84      * model with a StyleSheet.
  85      */
  86     public AttributeSet getAttributes() {
  87         if (attr == null) {
  88             StyleSheet sheet = getStyleSheet();
  89             attr = sheet.getViewAttributes(this);
  90         }
  91         return attr;
  92     }
  93 
  94     /**
  95      * Sets up the paragraph from css attributes instead of
  96      * the values found in StyleConstants (i.e. which are used
  97      * by the superclass).  Since
  98      */
  99     protected void setPropertiesFromAttributes() {
 100         StyleSheet sheet = getStyleSheet();
 101         attr = sheet.getViewAttributes(this);
 102         painter = sheet.getBoxPainter(attr);
 103         if (attr != null) {
 104             super.setPropertiesFromAttributes();
 105             setInsets((short) painter.getInset(TOP, this),
 106                       (short) painter.getInset(LEFT, this),
 107                       (short) painter.getInset(BOTTOM, this),
 108                       (short) painter.getInset(RIGHT, this));
 109             Object o = attr.getAttribute(CSS.Attribute.TEXT_ALIGN);
 110             if (o != null) {
 111                 // set horizontal alignment
 112                 String ta = o.toString();
 113                 if (ta.equals("left")) {
 114                     setJustification(StyleConstants.ALIGN_LEFT);
 115                 } else if (ta.equals("center")) {
 116                     setJustification(StyleConstants.ALIGN_CENTER);
 117                 } else if (ta.equals("right")) {
 118                     setJustification(StyleConstants.ALIGN_RIGHT);
 119                 } else if (ta.equals("justify")) {
 120                     setJustification(StyleConstants.ALIGN_JUSTIFIED);
 121                 }
 122             }
 123             // Get the width/height
 124             cssWidth = (CSS.LengthValue)attr.getAttribute(
 125                                         CSS.Attribute.WIDTH);
 126             cssHeight = (CSS.LengthValue)attr.getAttribute(
 127                                          CSS.Attribute.HEIGHT);
 128         }
 129     }
 130 
 131     protected StyleSheet getStyleSheet() {
 132         HTMLDocument doc = (HTMLDocument) getDocument();
 133         return doc.getStyleSheet();
 134     }
 135 
 136 
 137     /**
 138      * Calculate the needs for the paragraph along the minor axis.
 139      *
 140      * <p>If size requirements are explicitly specified for the paragraph,
 141      * use that requirements.  Otherwise, use the requirements of the
 142      * superclass {@link javax.swing.text.ParagraphView}.</p>
 143      *
 144      * <p>If the {@code axis} parameter is neither {@code View.X_AXIS} nor
 145      * {@code View.Y_AXIS}, {@link IllegalArgumentException} is thrown.  If the
 146      * {@code r} parameter is {@code null,} a new {@code SizeRequirements}
 147      * object is created, otherwise the supplied {@code SizeRequirements}
 148      * object is returned.</p>
 149      *
 150      * @param axis  the minor axis
 151      * @param r     the input {@code SizeRequirements} object
 152      * @return      the new or adjusted {@code SizeRequirements} object
 153      * @throws IllegalArgumentException  if the {@code axis} parameter is invalid
 154      */
 155     protected SizeRequirements calculateMinorAxisRequirements(
 156                                                 int axis, SizeRequirements r) {
 157         r = super.calculateMinorAxisRequirements(axis, r);
 158 
 159         if (BlockView.spanSetFromAttributes(axis, r, cssWidth, cssHeight)) {
 160             // Offset by the margins so that pref/min/max return the
 161             // right value.
 162             int margin = (axis == X_AXIS) ? getLeftInset() + getRightInset() :
 163                                             getTopInset() + getBottomInset();
 164             r.minimum -= margin;
 165             r.preferred -= margin;
 166             r.maximum -= margin;
 167         }
 168         return r;
 169     }
 170 
 171 
 172     /**
 173      * Indicates whether or not this view should be
 174      * displayed.  If none of the children wish to be
 175      * displayed and the only visible child is the
 176      * break that ends the paragraph, the paragraph
 177      * will not be considered visible.  Otherwise,
 178      * it will be considered visible and return true.
 179      *
 180      * @return true if the paragraph should be displayed
 181      */
 182     public boolean isVisible() {
 183 
 184         int n = getLayoutViewCount() - 1;
 185         for (int i = 0; i < n; i++) {
 186             View v = getLayoutView(i);
 187             if (v.isVisible()) {
 188                 return true;
 189             }
 190         }
 191         if (n > 0) {
 192             View v = getLayoutView(n);
 193             if ((v.getEndOffset() - v.getStartOffset()) == 1) {
 194                 return false;
 195             }
 196         }
 197         // If it's the last paragraph and not editable, it shouldn't
 198         // be visible.
 199         if (getStartOffset() == getDocument().getLength()) {
 200             boolean editable = false;
 201             Component c = getContainer();
 202             if (c instanceof JTextComponent) {
 203                 editable = ((JTextComponent)c).isEditable();
 204             }
 205             if (!editable) {
 206                 return false;
 207             }
 208         }
 209         return true;
 210     }
 211 
 212     /**
 213      * Renders using the given rendering surface and area on that
 214      * surface.  This is implemented to delgate to the superclass
 215      * after stashing the base coordinate for tab calculations.
 216      *
 217      * @param g the rendering surface to use
 218      * @param a the allocated region to render into
 219      * @see View#paint
 220      */
 221     public void paint(Graphics g, Shape a) {
 222         if (a == null) {
 223             return;
 224         }
 225 
 226         Rectangle r;
 227         if (a instanceof Rectangle) {
 228             r = (Rectangle) a;
 229         } else {
 230             r = a.getBounds();
 231         }
 232         painter.paint(g, r.x, r.y, r.width, r.height, this);
 233         super.paint(g, a);
 234     }
 235 
 236     /**
 237      * Determines the preferred span for this view.  Returns
 238      * 0 if the view is not visible, otherwise it calls the
 239      * superclass method to get the preferred span.
 240      * axis.
 241      *
 242      * @param axis may be either View.X_AXIS or View.Y_AXIS
 243      * @return   the span the view would like to be rendered into;
 244      *           typically the view is told to render into the span
 245      *           that is returned, although there is no guarantee;
 246      *           the parent may choose to resize or break the view
 247      * @see javax.swing.text.ParagraphView#getPreferredSpan
 248      */
 249     public float getPreferredSpan(int axis) {
 250         if (!isVisible()) {
 251             return 0;
 252         }
 253         return super.getPreferredSpan(axis);
 254     }
 255 
 256     /**
 257      * Determines the minimum span for this view along an
 258      * axis.  Returns 0 if the view is not visible, otherwise
 259      * it calls the superclass method to get the minimum span.
 260      *
 261      * @param axis may be either <code>View.X_AXIS</code> or
 262      *  <code>View.Y_AXIS</code>
 263      * @return  the minimum span the view can be rendered into
 264      * @see javax.swing.text.ParagraphView#getMinimumSpan
 265      */
 266     public float getMinimumSpan(int axis) {
 267         if (!isVisible()) {
 268             return 0;
 269         }
 270         return super.getMinimumSpan(axis);
 271     }
 272 
 273     /**
 274      * Determines the maximum span for this view along an
 275      * axis.  Returns 0 if the view is not visible, otherwise
 276      * it calls the superclass method ot get the maximum span.
 277      *
 278      * @param axis may be either <code>View.X_AXIS</code> or
 279      *  <code>View.Y_AXIS</code>
 280      * @return  the maximum span the view can be rendered into
 281      * @see javax.swing.text.ParagraphView#getMaximumSpan
 282      */
 283     public float getMaximumSpan(int axis) {
 284         if (!isVisible()) {
 285             return 0;
 286         }
 287         return super.getMaximumSpan(axis);
 288     }
 289 
 290     private AttributeSet attr;
 291     private StyleSheet.BoxPainter painter;
 292     private CSS.LengthValue cssWidth;
 293     private CSS.LengthValue cssHeight;
 294 }