1 /*
2  * Copyright (c) 1998, 2013, 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      */
ParagraphView(Element elem)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      */
setParent(View parent)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      */
getAttributes()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      */
setPropertiesFromAttributes()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 
getStyleSheet()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      */
calculateMinorAxisRequirements( int axis, SizeRequirements r)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      */
isVisible()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 delegate 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      */
paint(Graphics g, Shape a)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      */
getPreferredSpan(int axis)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      */
getMinimumSpan(int axis)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      */
getMaximumSpan(int axis)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 }
295