Use Tree Navigation
public class

StyledEditorKit

extends DefaultEditorKit
/*
 * Copyright (c) 1997, 2004, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code 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
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package javax.swing.text;

import java.io.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.event.*;
import javax.swing.Action;
import javax.swing.JEditorPane;
import javax.swing.KeyStroke;
import javax.swing.UIManager;

/**
 * This is the set of things needed by a text component
 * to be a reasonably functioning editor for some <em>type</em>
 * of text document.  This implementation provides a default
 * implementation which treats text as styled text and
 * provides a minimal set of actions for editing styled text.
 *
 * @author  Timothy Prinzing
 */

public class StyledEditorKit extends DefaultEditorKit {

   
/**
     * Creates a new EditorKit used for styled documents.
     */

   
public StyledEditorKit() {
        createInputAttributeUpdated
();
        createInputAttributes
();
   
}

   
/**
     * Gets the input attributes for the pane.  When
     * the caret moves and there is no selection, the
     * input attributes are automatically mutated to
     * reflect the character attributes of the current
     * caret location.  The styled editing actions
     * use the input attributes to carry out their
     * actions.
     *
     * @return the attribute set
     */

   
public MutableAttributeSet getInputAttributes() {
       
return inputAttributes;
   
}

   
/**
     * Fetches the element representing the current
     * run of character attributes for the caret.
     *
     * @return the element
     */

   
public Element getCharacterAttributeRun() {
       
return currentRun;
   
}

   
// --- EditorKit methods ---------------------------

   
/**
     * Fetches the command list for the editor.  This is
     * the list of commands supported by the superclass
     * augmented by the collection of commands defined
     * locally for style operations.
     *
     * @return the command list
     */

   
public Action[] getActions() {
       
return TextAction.augmentList(super.getActions(), this.defaultActions);
   
}

   
/**
     * Creates an uninitialized text storage model
     * that is appropriate for this type of editor.
     *
     * @return the model
     */

   
public Document createDefaultDocument() {
       
return new DefaultStyledDocument();
   
}

   
/**
     * Called when the kit is being installed into
     * a JEditorPane.
     *
     * @param c the JEditorPane
     */

   
public void install(JEditorPane c) {
        c
.addCaretListener(inputAttributeUpdater);
        c
.addPropertyChangeListener(inputAttributeUpdater);
       
Caret caret = c.getCaret();
       
if (caret != null) {
            inputAttributeUpdater
.updateInputAttributes
                                 
(caret.getDot(), caret.getMark(), c);
       
}
   
}

   
/**
     * Called when the kit is being removed from the
     * JEditorPane.  This is used to unregister any
     * listeners that were attached.
     *
     * @param c the JEditorPane
     */

   
public void deinstall(JEditorPane c) {
        c
.removeCaretListener(inputAttributeUpdater);
        c
.removePropertyChangeListener(inputAttributeUpdater);

       
// remove references to current document so it can be collected.
        currentRun
= null;
        currentParagraph
= null;
   
}

   
/**
     * Fetches a factory that is suitable for producing
     * views of any models that are produced by this
     * kit.  This is implemented to return View implementations
     * for the following kinds of elements:
     * <ul>
     * <li>AbstractDocument.ContentElementName
     * <li>AbstractDocument.ParagraphElementName
     * <li>AbstractDocument.SectionElementName
     * <li>StyleConstants.ComponentElementName
     * <li>StyleConstants.IconElementName
     * </ul>
     *
     * @return the factory
     */

   
public ViewFactory getViewFactory() {
       
return defaultFactory;
   
}

   
/**
     * Creates a copy of the editor kit.
     *
     * @return the copy
     */

   
public Object clone() {
       
StyledEditorKit o = (StyledEditorKit)super.clone();
        o
.currentRun = o.currentParagraph = null;
        o
.createInputAttributeUpdated();
        o
.createInputAttributes();
       
return o;
   
}

   
/**
     * Creates the AttributeSet used for the selection.
     */

   
private void createInputAttributes() {
        inputAttributes
= new SimpleAttributeSet() {
           
public AttributeSet getResolveParent() {
               
return (currentParagraph != null) ?
                           currentParagraph
.getAttributes() : null;
           
}

           
public Object clone() {
               
return new SimpleAttributeSet(this);
           
}
       
};
   
}

   
/**
     * Creates a new <code>AttributeTracker</code>.
     */

   
private void createInputAttributeUpdated() {
        inputAttributeUpdater
= new AttributeTracker();
   
}


   
private static final ViewFactory defaultFactory = new StyledViewFactory();

   
Element currentRun;
   
Element currentParagraph;

   
/**
     * This is the set of attributes used to store the
     * input attributes.
     */

   
MutableAttributeSet inputAttributes;

   
/**
     * This listener will be attached to the caret of
     * the text component that the EditorKit gets installed
     * into.  This should keep the input attributes updated
     * for use by the styled actions.
     */

   
private AttributeTracker inputAttributeUpdater;

   
/**
     * Tracks caret movement and keeps the input attributes set
     * to reflect the current set of attribute definitions at the
     * caret position.
     * <p>This implements PropertyChangeListener to update the
     * input attributes when the Document changes, as if the Document
     * changes the attributes will almost certainly change.
     */

   
class AttributeTracker implements CaretListener, PropertyChangeListener, Serializable {

       
/**
         * Updates the attributes. <code>dot</code> and <code>mark</code>
         * mark give the positions of the selection in <code>c</code>.
         */

       
void updateInputAttributes(int dot, int mark, JTextComponent c) {
           
// EditorKit might not have installed the StyledDocument yet.
           
Document aDoc = c.getDocument();
           
if (!(aDoc instanceof StyledDocument)) {
               
return ;
           
}
           
int start = Math.min(dot, mark);
           
// record current character attributes.
           
StyledDocument doc = (StyledDocument)aDoc;
           
// If nothing is selected, get the attributes from the character
           
// before the start of the selection, otherwise get the attributes
           
// from the character element at the start of the selection.
           
Element run;
            currentParagraph
= doc.getParagraphElement(start);
           
if (currentParagraph.getStartOffset() == start || dot != mark) {
               
// Get the attributes from the character at the selection
               
// if in a different paragrah!
                run
= doc.getCharacterElement(start);
           
}
           
else {
                run
= doc.getCharacterElement(Math.max(start-1, 0));
           
}
           
if (run != currentRun) {
                   
/*
                     * PENDING(prinz) All attributes that represent a single
                     * glyph position and can't be inserted into should be
                     * removed from the input attributes... this requires
                     * mixing in an interface to indicate that condition.
                     * When we can add things again this logic needs to be
                     * improved!!
                     */

                currentRun
= run;
                createInputAttributes
(currentRun, getInputAttributes());
           
}
       
}

       
public void propertyChange(PropertyChangeEvent evt) {
           
Object newValue = evt.getNewValue();
           
Object source = evt.getSource();

           
if ((source instanceof JTextComponent) &&
               
(newValue instanceof Document)) {
               
// New document will have changed selection to 0,0.
                updateInputAttributes
(0, 0, (JTextComponent)source);
           
}
       
}

       
public void caretUpdate(CaretEvent e) {
            updateInputAttributes
(e.getDot(), e.getMark(),
                                 
(JTextComponent)e.getSource());
       
}
   
}

   
/**
     * Copies the key/values in <code>element</code>s AttributeSet into
     * <code>set</code>. This does not copy component, icon, or element
     * names attributes. Subclasses may wish to refine what is and what
     * isn't copied here. But be sure to first remove all the attributes that
     * are in <code>set</code>.<p>
     * This is called anytime the caret moves over a different location.
     *
     */

   
protected void createInputAttributes(Element element,
                                         
MutableAttributeSet set) {
       
if (element.getAttributes().getAttributeCount() > 0
           
|| element.getEndOffset() - element.getStartOffset() > 1
           
|| element.getEndOffset() < element.getDocument().getLength()) {
           
set.removeAttributes(set);
           
set.addAttributes(element.getAttributes());
           
set.removeAttribute(StyleConstants.ComponentAttribute);
           
set.removeAttribute(StyleConstants.IconAttribute);
           
set.removeAttribute(AbstractDocument.ElementNameAttribute);
           
set.removeAttribute(StyleConstants.ComposedTextAttribute);
       
}
   
}

   
// ---- default ViewFactory implementation ---------------------

   
static class StyledViewFactory implements ViewFactory {

       
public View create(Element elem) {
           
String kind = elem.getName();
           
if (kind != null) {
               
if (kind.equals(AbstractDocument.ContentElementName)) {
                   
return new LabelView(elem);
               
} else if (kind.equals(AbstractDocument.ParagraphElementName)) {
                   
return new ParagraphView(elem);
               
} else if (kind.equals(AbstractDocument.SectionElementName)) {
                   
return new BoxView(elem, View.Y_AXIS);
               
} else if (kind.equals(StyleConstants.ComponentElementName)) {
                   
return new ComponentView(elem);
               
} else if (kind.equals(StyleConstants.IconElementName)) {
                   
return new IconView(elem);
               
}
           
}

           
// default to text display
           
return new LabelView(elem);
       
}

   
}

   
// --- Action implementations ---------------------------------

   
private static final Action[] defaultActions = {
       
new FontFamilyAction("font-family-SansSerif", "SansSerif"),
       
new FontFamilyAction("font-family-Monospaced", "Monospaced"),
       
new FontFamilyAction("font-family-Serif", "Serif"),
       
new FontSizeAction("font-size-8", 8),
       
new FontSizeAction("font-size-10", 10),
       
new FontSizeAction("font-size-12", 12),
       
new FontSizeAction("font-size-14", 14),
       
new FontSizeAction("font-size-16", 16),
       
new FontSizeAction("font-size-18", 18),
       
new FontSizeAction("font-size-24", 24),
       
new FontSizeAction("font-size-36", 36),
       
new FontSizeAction("font-size-48", 48),
       
new AlignmentAction("left-justify", StyleConstants.ALIGN_LEFT),
       
new AlignmentAction("center-justify", StyleConstants.ALIGN_CENTER),
       
new AlignmentAction("right-justify", StyleConstants.ALIGN_RIGHT),
       
new BoldAction(),
       
new ItalicAction(),
       
new StyledInsertBreakAction(),
       
new UnderlineAction()
   
};

   
/**
     * An action that assumes it's being fired on a JEditorPane
     * with a StyledEditorKit (or subclass) installed.  This has
     * some convenience methods for causing character or paragraph
     * level attribute changes.  The convenience methods will
     * throw an IllegalArgumentException if the assumption of
     * a StyledDocument, a JEditorPane, or a StyledEditorKit
     * fail to be true.
     * <p>
     * The component that gets acted upon by the action
     * will be the source of the ActionEvent if the source
     * can be narrowed to a JEditorPane type.  If the source
     * can't be narrowed, the most recently focused text
     * component is changed.  If neither of these are the
     * case, the action cannot be performed.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public abstract static class StyledTextAction extends TextAction {

       
/**
         * Creates a new StyledTextAction from a string action name.
         *
         * @param nm the name of the action
         */

       
public StyledTextAction(String nm) {
           
super(nm);
       
}

       
/**
         * Gets the target editor for an action.
         *
         * @param e the action event
         * @return the editor
         */

       
protected final JEditorPane getEditor(ActionEvent e) {
           
JTextComponent tcomp = getTextComponent(e);
           
if (tcomp instanceof JEditorPane) {
               
return (JEditorPane) tcomp;
           
}
           
return null;
       
}

       
/**
         * Gets the document associated with an editor pane.
         *
         * @param e the editor
         * @return the document
         * @exception IllegalArgumentException for the wrong document type
         */

       
protected final StyledDocument getStyledDocument(JEditorPane e) {
           
Document d = e.getDocument();
           
if (d instanceof StyledDocument) {
               
return (StyledDocument) d;
           
}
           
throw new IllegalArgumentException("document must be StyledDocument");
       
}

       
/**
         * Gets the editor kit associated with an editor pane.
         *
         * @param e the editor pane
         * @return the kit
         * @exception IllegalArgumentException for the wrong document type
         */

       
protected final StyledEditorKit getStyledEditorKit(JEditorPane e) {
           
EditorKit k = e.getEditorKit();
           
if (k instanceof StyledEditorKit) {
               
return (StyledEditorKit) k;
           
}
           
throw new IllegalArgumentException("EditorKit must be StyledEditorKit");
       
}

       
/**
         * Applies the given attributes to character
         * content.  If there is a selection, the attributes
         * are applied to the selection range.  If there
         * is no selection, the attributes are applied to
         * the input attribute set which defines the attributes
         * for any new text that gets inserted.
         *
         * @param editor the editor
         * @param attr the attributes
         * @param replace   if true, then replace the existing attributes first
         */

       
protected final void setCharacterAttributes(JEditorPane editor,
                                             
AttributeSet attr, boolean replace) {
           
int p0 = editor.getSelectionStart();
           
int p1 = editor.getSelectionEnd();
           
if (p0 != p1) {
               
StyledDocument doc = getStyledDocument(editor);
                doc
.setCharacterAttributes(p0, p1 - p0, attr, replace);
           
}
           
StyledEditorKit k = getStyledEditorKit(editor);
           
MutableAttributeSet inputAttributes = k.getInputAttributes();
           
if (replace) {
                inputAttributes
.removeAttributes(inputAttributes);
           
}
            inputAttributes
.addAttributes(attr);
       
}

       
/**
         * Applies the given attributes to paragraphs.  If
         * there is a selection, the attributes are applied
         * to the paragraphs that intersect the selection.
         * if there is no selection, the attributes are applied
         * to the paragraph at the current caret position.
         *
         * @param editor the editor
         * @param attr the attributes
         * @param replace   if true, replace the existing attributes first
         */

       
protected final void setParagraphAttributes(JEditorPane editor,
                                           
AttributeSet attr, boolean replace) {
           
int p0 = editor.getSelectionStart();
           
int p1 = editor.getSelectionEnd();
           
StyledDocument doc = getStyledDocument(editor);
            doc
.setParagraphAttributes(p0, p1 - p0, attr, replace);
       
}

   
}

   
/**
     * An action to set the font family in the associated
     * JEditorPane.  This will use the family specified as
     * the command string on the ActionEvent if there is one,
     * otherwise the family that was initialized with will be used.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public static class FontFamilyAction extends StyledTextAction {

       
/**
         * Creates a new FontFamilyAction.
         *
         * @param nm the action name
         * @param family the font family
         */

       
public FontFamilyAction(String nm, String family) {
           
super(nm);
           
this.family = family;
       
}

       
/**
         * Sets the font family.
         *
         * @param e the event
         */

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane editor = getEditor(e);
           
if (editor != null) {
               
String family = this.family;
               
if ((e != null) && (e.getSource() == editor)) {
                   
String s = e.getActionCommand();
                   
if (s != null) {
                        family
= s;
                   
}
               
}
               
if (family != null) {
                   
MutableAttributeSet attr = new SimpleAttributeSet();
                   
StyleConstants.setFontFamily(attr, family);
                    setCharacterAttributes
(editor, attr, false);
               
} else {
                   
UIManager.getLookAndFeel().provideErrorFeedback(editor);
               
}
           
}
       
}

       
private String family;
   
}

   
/**
     * An action to set the font size in the associated
     * JEditorPane.  This will use the size specified as
     * the command string on the ActionEvent if there is one,
     * otherwise the size that was initialized with will be used.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public static class FontSizeAction extends StyledTextAction {

       
/**
         * Creates a new FontSizeAction.
         *
         * @param nm the action name
         * @param size the font size
         */

       
public FontSizeAction(String nm, int size) {
           
super(nm);
           
this.size = size;
       
}

       
/**
         * Sets the font size.
         *
         * @param e the action event
         */

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane editor = getEditor(e);
           
if (editor != null) {
               
int size = this.size;
               
if ((e != null) && (e.getSource() == editor)) {
                   
String s = e.getActionCommand();
                   
try {
                        size
= Integer.parseInt(s, 10);
                   
} catch (NumberFormatException nfe) {
                   
}
               
}
               
if (size != 0) {
                   
MutableAttributeSet attr = new SimpleAttributeSet();
                   
StyleConstants.setFontSize(attr, size);
                    setCharacterAttributes
(editor, attr, false);
               
} else {
                   
UIManager.getLookAndFeel().provideErrorFeedback(editor);
               
}
           
}
       
}

       
private int size;
   
}

   
/**
     * An action to set foreground color.  This sets the
     * <code>StyleConstants.Foreground</code> attribute for the
     * currently selected range of the target JEditorPane.
     * This is done by calling
     * <code>StyledDocument.setCharacterAttributes</code>
     * on the styled document associated with the target
     * JEditorPane.
     * <p>
     * If the target text component is specified as the
     * source of the ActionEvent and there is a command string,
     * the command string will be interpreted as the foreground
     * color.  It will be interpreted by called
     * <code>Color.decode</code>, and should therefore be
     * legal input for that method.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public static class ForegroundAction extends StyledTextAction {

       
/**
         * Creates a new ForegroundAction.
         *
         * @param nm the action name
         * @param fg the foreground color
         */

       
public ForegroundAction(String nm, Color fg) {
           
super(nm);
           
this.fg = fg;
       
}

       
/**
         * Sets the foreground color.
         *
         * @param e the action event
         */

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane editor = getEditor(e);
           
if (editor != null) {
               
Color fg = this.fg;
               
if ((e != null) && (e.getSource() == editor)) {
                   
String s = e.getActionCommand();
                   
try {
                        fg
= Color.decode(s);
                   
} catch (NumberFormatException nfe) {
                   
}
               
}
               
if (fg != null) {
                   
MutableAttributeSet attr = new SimpleAttributeSet();
                   
StyleConstants.setForeground(attr, fg);
                    setCharacterAttributes
(editor, attr, false);
               
} else {
                   
UIManager.getLookAndFeel().provideErrorFeedback(editor);
               
}
           
}
       
}

       
private Color fg;
   
}

   
/**
     * An action to set paragraph alignment.  This sets the
     * <code>StyleConstants.Alignment</code> attribute for the
     * currently selected range of the target JEditorPane.
     * This is done by calling
     * <code>StyledDocument.setParagraphAttributes</code>
     * on the styled document associated with the target
     * JEditorPane.
     * <p>
     * If the target text component is specified as the
     * source of the ActionEvent and there is a command string,
     * the command string will be interpreted as an integer
     * that should be one of the legal values for the
     * <code>StyleConstants.Alignment</code> attribute.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public static class AlignmentAction extends StyledTextAction {

       
/**
         * Creates a new AlignmentAction.
         *
         * @param nm the action name
         * @param a the alignment >= 0
         */

       
public AlignmentAction(String nm, int a) {
           
super(nm);
           
this.a = a;
       
}

       
/**
         * Sets the alignment.
         *
         * @param e the action event
         */

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane editor = getEditor(e);
           
if (editor != null) {
               
int a = this.a;
               
if ((e != null) && (e.getSource() == editor)) {
                   
String s = e.getActionCommand();
                   
try {
                        a
= Integer.parseInt(s, 10);
                   
} catch (NumberFormatException nfe) {
                   
}
               
}
               
MutableAttributeSet attr = new SimpleAttributeSet();
               
StyleConstants.setAlignment(attr, a);
                setParagraphAttributes
(editor, attr, false);
           
}
       
}

       
private int a;
   
}

   
/**
     * An action to toggle the bold attribute.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public static class BoldAction extends StyledTextAction {

       
/**
         * Constructs a new BoldAction.
         */

       
public BoldAction() {
           
super("font-bold");
       
}

       
/**
         * Toggles the bold attribute.
         *
         * @param e the action event
         */

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane editor = getEditor(e);
           
if (editor != null) {
               
StyledEditorKit kit = getStyledEditorKit(editor);
               
MutableAttributeSet attr = kit.getInputAttributes();
               
boolean bold = (StyleConstants.isBold(attr)) ? false : true;
               
SimpleAttributeSet sas = new SimpleAttributeSet();
               
StyleConstants.setBold(sas, bold);
                setCharacterAttributes
(editor, sas, false);
           
}
       
}
   
}

   
/**
     * An action to toggle the italic attribute.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public static class ItalicAction extends StyledTextAction {

       
/**
         * Constructs a new ItalicAction.
         */

       
public ItalicAction() {
           
super("font-italic");
       
}

       
/**
         * Toggles the italic attribute.
         *
         * @param e the action event
         */

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane editor = getEditor(e);
           
if (editor != null) {
               
StyledEditorKit kit = getStyledEditorKit(editor);
               
MutableAttributeSet attr = kit.getInputAttributes();
               
boolean italic = (StyleConstants.isItalic(attr)) ? false : true;
               
SimpleAttributeSet sas = new SimpleAttributeSet();
               
StyleConstants.setItalic(sas, italic);
                setCharacterAttributes
(editor, sas, false);
           
}
       
}
   
}

   
/**
     * An action to toggle the underline attribute.
     * <p>
     * <strong>Warning:</strong>
     * 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 JavaBeans<sup><font size="-2">TM</font></sup>
     * has been added to the <code>java.beans</code> package.
     * Please see {@link java.beans.XMLEncoder}.
     */

   
public static class UnderlineAction extends StyledTextAction {

       
/**
         * Constructs a new UnderlineAction.
         */

       
public UnderlineAction() {
           
super("font-underline");
       
}

       
/**
         * Toggles the Underline attribute.
         *
         * @param e the action event
         */

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane editor = getEditor(e);
           
if (editor != null) {
               
StyledEditorKit kit = getStyledEditorKit(editor);
               
MutableAttributeSet attr = kit.getInputAttributes();
               
boolean underline = (StyleConstants.isUnderline(attr)) ? false : true;
               
SimpleAttributeSet sas = new SimpleAttributeSet();
               
StyleConstants.setUnderline(sas, underline);
                setCharacterAttributes
(editor, sas, false);
           
}
       
}
   
}


   
/**
     * StyledInsertBreakAction has similar behavior to that of
     * <code>DefaultEditorKit.InsertBreakAction</code>. That is when
     * its <code>actionPerformed</code> method is invoked, a newline
     * is inserted. Beyond that, this will reset the input attributes to
     * what they were before the newline was inserted.
     */

   
static class StyledInsertBreakAction extends StyledTextAction {
       
private SimpleAttributeSet tempSet;

       
StyledInsertBreakAction() {
           
super(insertBreakAction);
       
}

       
public void actionPerformed(ActionEvent e) {
           
JEditorPane target = getEditor(e);

           
if (target != null) {
               
if ((!target.isEditable()) || (!target.isEnabled())) {
                   
UIManager.getLookAndFeel().provideErrorFeedback(target);
                   
return;
               
}
               
StyledEditorKit sek = getStyledEditorKit(target);

               
if (tempSet != null) {
                    tempSet
.removeAttributes(tempSet);
               
}
               
else {
                    tempSet
= new SimpleAttributeSet();
               
}
                tempSet
.addAttributes(sek.getInputAttributes());
                target
.replaceSelection("\n");

               
MutableAttributeSet ia = sek.getInputAttributes();

                ia
.removeAttributes(ia);
                ia
.addAttributes(tempSet);
                tempSet
.removeAttributes(tempSet);
           
}
           
else {
               
// See if we are in a JTextComponent.
               
JTextComponent text = getTextComponent(e);

               
if (text != null) {
                   
if ((!text.isEditable()) || (!text.isEnabled())) {
                       
UIManager.getLookAndFeel().provideErrorFeedback(target);
                       
return;
                   
}
                    text
.replaceSelection("\n");
               
}
           
}
       
}
   
}
}