/* | 
|
 * Copyright (c) 1997, 2013, 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™ | 
|
     * 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™ | 
|
     * 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™ | 
|
     * 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™ | 
|
     * 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™ | 
|
     * 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™ | 
|
     * 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™ | 
|
     * 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™ | 
|
     * 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");  | 
|
}  | 
|
}  | 
|
}  | 
|
}  | 
|
}  |