mirror of git://gcc.gnu.org/git/gcc.git
826 lines
18 KiB
Java
826 lines
18 KiB
Java
/* AbstractDocument.java --
|
|
Copyright (C) 2002, 2004 Free Software Foundation, Inc.
|
|
|
|
This file is part of GNU Classpath.
|
|
|
|
GNU Classpath is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the 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, but
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with GNU Classpath; see the file COPYING. If not, write to the
|
|
Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
|
02111-1307 USA.
|
|
|
|
Linking this library statically or dynamically with other modules is
|
|
making a combined work based on this library. Thus, the terms and
|
|
conditions of the GNU General Public License cover the whole
|
|
combination.
|
|
|
|
As a special exception, the copyright holders of this library give you
|
|
permission to link this library with independent modules to produce an
|
|
executable, regardless of the license terms of these independent
|
|
modules, and to copy and distribute the resulting executable under
|
|
terms of your choice, provided that you also meet, for each linked
|
|
independent module, the terms and conditions of the license of that
|
|
module. An independent module is a module which is not derived from
|
|
or based on this library. If you modify this library, you may extend
|
|
this exception to your version of the library, but you are not
|
|
obligated to do so. If you do not wish to do so, delete this
|
|
exception statement from your version. */
|
|
|
|
|
|
package javax.swing.text;
|
|
|
|
import java.io.Serializable;
|
|
import java.util.Collections;
|
|
import java.util.Dictionary;
|
|
import java.util.Enumeration;
|
|
import java.util.EventListener;
|
|
import java.util.Vector;
|
|
|
|
import javax.swing.event.DocumentEvent;
|
|
import javax.swing.event.DocumentListener;
|
|
import javax.swing.event.EventListenerList;
|
|
import javax.swing.event.UndoableEditEvent;
|
|
import javax.swing.event.UndoableEditListener;
|
|
import javax.swing.tree.TreeNode;
|
|
import javax.swing.undo.AbstractUndoableEdit;
|
|
import javax.swing.undo.CompoundEdit;
|
|
import javax.swing.undo.UndoableEdit;
|
|
|
|
public abstract class AbstractDocument
|
|
implements Document, Serializable
|
|
{
|
|
private static final long serialVersionUID = -116069779446114664L;
|
|
protected static final String BAD_LOCATION = "document location failure";
|
|
public static final String BidiElementName = "bidi level";
|
|
public static final String ContentElementName = "content";
|
|
public static final String ParagraphElementName = "paragraph";
|
|
public static final String SectionElementName = "section";
|
|
public static final String ElementNameAttribute = "$ename";
|
|
|
|
Content content;
|
|
AttributeContext context;
|
|
protected EventListenerList listenerList = new EventListenerList();
|
|
|
|
protected AbstractDocument(Content doc)
|
|
{
|
|
this(doc, StyleContext.getDefaultStyleContext());
|
|
}
|
|
|
|
protected AbstractDocument(Content doc, AttributeContext ctx)
|
|
{
|
|
content = doc;
|
|
context = ctx;
|
|
}
|
|
|
|
// These still need to be implemented by a derived class:
|
|
public abstract Element getParagraphElement(int pos);
|
|
|
|
public abstract Element getDefaultRootElement();
|
|
|
|
protected Element createBranchElement(Element parent,
|
|
AttributeSet attributes)
|
|
{
|
|
return new BranchElement(parent, attributes);
|
|
}
|
|
|
|
protected Element createLeafElement(Element parent, AttributeSet attributes,
|
|
int start, int end)
|
|
{
|
|
return new LeafElement(parent, attributes, start, end);
|
|
}
|
|
|
|
public Position createPosition(final int offset) throws BadLocationException
|
|
{
|
|
if (offset < 0 || offset > getLength())
|
|
throw new BadLocationException(getText(0, getLength()), offset);
|
|
|
|
return new Position()
|
|
{
|
|
public int getOffset()
|
|
{
|
|
return offset;
|
|
}
|
|
};
|
|
}
|
|
|
|
protected void fireChangedUpdate(DocumentEvent event)
|
|
{
|
|
DocumentListener[] listeners = getDocumentListeners();
|
|
|
|
for (int index = 0; index < listeners.length; ++index)
|
|
listeners[index].changedUpdate(event);
|
|
}
|
|
|
|
protected void fireInsertUpdate(DocumentEvent event)
|
|
{
|
|
DocumentListener[] listeners = getDocumentListeners();
|
|
|
|
for (int index = 0; index < listeners.length; ++index)
|
|
listeners[index].insertUpdate(event);
|
|
}
|
|
|
|
protected void fireRemoveUpdate(DocumentEvent event)
|
|
{
|
|
DocumentListener[] listeners = getDocumentListeners();
|
|
|
|
for (int index = 0; index < listeners.length; ++index)
|
|
listeners[index].removeUpdate(event);
|
|
}
|
|
|
|
protected void fireUndoableEditUpdate(UndoableEditEvent event)
|
|
{
|
|
UndoableEditListener[] listeners = getUndoableEditListeners();
|
|
|
|
for (int index = 0; index < listeners.length; ++index)
|
|
listeners[index].undoableEditHappened(event);
|
|
}
|
|
|
|
public int getAsynchronousLoadPriority()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
protected AttributeContext getAttributeContext()
|
|
{
|
|
return context;
|
|
}
|
|
|
|
public Element getBidiRootElement()
|
|
{
|
|
return null;
|
|
}
|
|
|
|
protected Content getContent()
|
|
{
|
|
return content;
|
|
}
|
|
|
|
protected Thread getCurrentWriter()
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public Dictionary getDocumentProperties()
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public Position getEndPosition()
|
|
{
|
|
return new Position()
|
|
{
|
|
public int getOffset()
|
|
{
|
|
return getLength();
|
|
}
|
|
};
|
|
}
|
|
|
|
public int getLength()
|
|
{
|
|
return content.length();
|
|
}
|
|
|
|
public EventListener[] getListeners(Class listenerType)
|
|
{
|
|
return listenerList.getListeners(listenerType);
|
|
}
|
|
|
|
public Object getProperty(Object key)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public Element[] getRootElements()
|
|
{
|
|
Element[] elements = new Element[1];
|
|
elements[0] = getDefaultRootElement();
|
|
return elements;
|
|
}
|
|
|
|
public Position getStartPosition()
|
|
{
|
|
return new Position()
|
|
{
|
|
public int getOffset()
|
|
{
|
|
return 0;
|
|
}
|
|
};
|
|
}
|
|
|
|
public String getText(int offset, int length) throws BadLocationException
|
|
{
|
|
return content.getString(offset, length);
|
|
}
|
|
|
|
public void getText(int offset, int length, Segment segment)
|
|
throws BadLocationException
|
|
{
|
|
content.getChars(offset, length, segment);
|
|
}
|
|
|
|
public void insertString(int offset, String text, AttributeSet attributes)
|
|
throws BadLocationException
|
|
{
|
|
// Just return when no text to insert was given.
|
|
if (text == null || text.length() == 0)
|
|
return;
|
|
|
|
DefaultDocumentEvent event =
|
|
new DefaultDocumentEvent(offset, text.length(),
|
|
DocumentEvent.EventType.INSERT);
|
|
content.insertString(offset, text);
|
|
insertUpdate(event, attributes);
|
|
fireInsertUpdate(event);
|
|
}
|
|
|
|
protected void insertUpdate(DefaultDocumentEvent chng, AttributeSet attr)
|
|
{
|
|
}
|
|
|
|
protected void postRemoveUpdate(DefaultDocumentEvent chng)
|
|
{
|
|
}
|
|
|
|
public void putProperty(Object key, Object value)
|
|
{
|
|
}
|
|
|
|
public void readLock()
|
|
{
|
|
}
|
|
|
|
public void readUnlock()
|
|
{
|
|
}
|
|
|
|
public void remove(int offset, int length) throws BadLocationException
|
|
{
|
|
DefaultDocumentEvent event =
|
|
new DefaultDocumentEvent(offset, length,
|
|
DocumentEvent.EventType.REMOVE);
|
|
removeUpdate(event);
|
|
content.remove(offset, length);
|
|
postRemoveUpdate(event);
|
|
fireRemoveUpdate(event);
|
|
}
|
|
|
|
/**
|
|
* Replaces some text in the document.
|
|
*
|
|
* @since 1.4
|
|
*/
|
|
public void replace(int offset, int length, String text,
|
|
AttributeSet attributes)
|
|
throws BadLocationException
|
|
{
|
|
remove(offset, length);
|
|
insertString(offset, text, attributes);
|
|
}
|
|
|
|
/**
|
|
* Adds a <code>DocumentListener</code> object to this document.
|
|
*
|
|
* @param listener the listener to add
|
|
*/
|
|
public void addDocumentListener(DocumentListener listener)
|
|
{
|
|
listenerList.add(DocumentListener.class, listener);
|
|
}
|
|
|
|
/**
|
|
* Removes a <code>DocumentListener</code> object from this document.
|
|
*
|
|
* @param listener the listener to remove
|
|
*/
|
|
public void removeDocumentListener(DocumentListener listener)
|
|
{
|
|
listenerList.remove(DocumentListener.class, listener);
|
|
}
|
|
|
|
/**
|
|
* Returns add added <code>DocumentListener</code> objects.
|
|
*
|
|
* @return an array of listeners
|
|
*/
|
|
public DocumentListener[] getDocumentListeners()
|
|
{
|
|
return (DocumentListener[]) getListeners(DocumentListener.class);
|
|
}
|
|
|
|
/**
|
|
* Adds a <code>UndoableEditListener</code> object to this document.
|
|
*
|
|
* @param listener the listener to add
|
|
*/
|
|
public void addUndoableEditListener(UndoableEditListener listener)
|
|
{
|
|
listenerList.add(UndoableEditListener.class, listener);
|
|
}
|
|
|
|
/**
|
|
* Removes a <code>UndoableEditListener</code> object from this document.
|
|
*
|
|
* @param listener the listener to remove
|
|
*/
|
|
public void removeUndoableEditListener(UndoableEditListener listener)
|
|
{
|
|
listenerList.remove(UndoableEditListener.class, listener);
|
|
}
|
|
|
|
/**
|
|
* Returns add added <code>UndoableEditListener</code> objects.
|
|
*
|
|
* @return an array of listeners
|
|
*/
|
|
public UndoableEditListener[] getUndoableEditListeners()
|
|
{
|
|
return (UndoableEditListener[]) getListeners(UndoableEditListener.class);
|
|
}
|
|
|
|
protected void removeUpdate(DefaultDocumentEvent chng)
|
|
{
|
|
}
|
|
|
|
public void render(Runnable r)
|
|
{
|
|
}
|
|
|
|
public void setAsynchronousLoadPriority(int p)
|
|
{
|
|
}
|
|
|
|
public void setDocumentProperties(Dictionary x)
|
|
{
|
|
}
|
|
|
|
protected void writeLock()
|
|
{
|
|
}
|
|
|
|
protected void writeUnlock()
|
|
{
|
|
}
|
|
|
|
public interface AttributeContext
|
|
{
|
|
AttributeSet addAttribute(AttributeSet old, Object name, Object value);
|
|
|
|
AttributeSet addAttributes(AttributeSet old, AttributeSet attributes);
|
|
|
|
AttributeSet getEmptySet();
|
|
|
|
void reclaim(AttributeSet attributes);
|
|
|
|
AttributeSet removeAttribute(AttributeSet old, Object name);
|
|
|
|
AttributeSet removeAttributes(AttributeSet old, AttributeSet attributes);
|
|
|
|
AttributeSet removeAttributes(AttributeSet old, Enumeration names);
|
|
}
|
|
|
|
public interface Content
|
|
{
|
|
Position createPosition(int offset) throws BadLocationException;
|
|
|
|
int length();
|
|
|
|
UndoableEdit insertString(int where, String str)
|
|
throws BadLocationException;
|
|
|
|
UndoableEdit remove(int where, int nitems) throws BadLocationException;
|
|
|
|
String getString(int where, int len) throws BadLocationException;
|
|
|
|
void getChars(int where, int len, Segment txt) throws BadLocationException;
|
|
}
|
|
|
|
public abstract class AbstractElement
|
|
implements Element, MutableAttributeSet, TreeNode, Serializable
|
|
{
|
|
private static final long serialVersionUID = 1265312733007397733L;
|
|
int count;
|
|
int offset;
|
|
|
|
AttributeSet attributes;
|
|
|
|
Element element_parent;
|
|
Vector element_children;
|
|
|
|
TreeNode tree_parent;
|
|
Vector tree_children;
|
|
|
|
public AbstractElement(Element p, AttributeSet s)
|
|
{
|
|
element_parent = p;
|
|
attributes = s;
|
|
}
|
|
|
|
// TreeNode implementation
|
|
|
|
public Enumeration children()
|
|
{
|
|
return Collections.enumeration(tree_children);
|
|
}
|
|
|
|
public boolean getAllowsChildren()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
public TreeNode getChildAt(int index)
|
|
{
|
|
return (TreeNode) tree_children.get(index);
|
|
}
|
|
|
|
public int getChildCount()
|
|
{
|
|
return tree_children.size();
|
|
}
|
|
|
|
public int getIndex(TreeNode node)
|
|
{
|
|
return tree_children.indexOf(node);
|
|
}
|
|
|
|
public TreeNode getParent()
|
|
{
|
|
return tree_parent;
|
|
}
|
|
|
|
public abstract boolean isLeaf();
|
|
|
|
|
|
// MutableAttributeSet support
|
|
|
|
public void addAttribute(Object name, Object value)
|
|
{
|
|
attributes = getAttributeContext().addAttribute(attributes, name, value);
|
|
}
|
|
|
|
public void addAttributes(AttributeSet attrs)
|
|
{
|
|
attributes = getAttributeContext().addAttributes(attributes, attrs);
|
|
}
|
|
|
|
public void removeAttribute(Object name)
|
|
{
|
|
attributes = getAttributeContext().removeAttribute(attributes, name);
|
|
}
|
|
|
|
public void removeAttributes(AttributeSet attrs)
|
|
{
|
|
attributes = getAttributeContext().removeAttributes(attributes, attrs);
|
|
}
|
|
|
|
public void removeAttributes(Enumeration names)
|
|
{
|
|
attributes = getAttributeContext().removeAttributes(attributes, names);
|
|
}
|
|
|
|
public void setResolveParent(AttributeSet parent)
|
|
{
|
|
attributes = getAttributeContext().addAttribute(attributes, ResolveAttribute, parent);
|
|
}
|
|
|
|
|
|
// AttributeSet interface support
|
|
|
|
public boolean containsAttribute(Object name, Object value)
|
|
{
|
|
return attributes.containsAttribute(name, value);
|
|
}
|
|
|
|
public boolean containsAttributes(AttributeSet attrs)
|
|
{
|
|
return attributes.containsAttributes(attrs);
|
|
}
|
|
|
|
public AttributeSet copyAttributes()
|
|
{
|
|
return attributes.copyAttributes();
|
|
}
|
|
|
|
public Object getAttribute(Object key)
|
|
{
|
|
return attributes.getAttribute(key);
|
|
}
|
|
|
|
public int getAttributeCount()
|
|
{
|
|
return attributes.getAttributeCount();
|
|
}
|
|
|
|
public Enumeration getAttributeNames()
|
|
{
|
|
return attributes.getAttributeNames();
|
|
}
|
|
|
|
public AttributeSet getResolveParent()
|
|
{
|
|
return attributes.getResolveParent();
|
|
}
|
|
|
|
public boolean isDefined(Object attrName)
|
|
{
|
|
return attributes.isDefined(attrName);
|
|
}
|
|
|
|
public boolean isEqual(AttributeSet attrs)
|
|
{
|
|
return attributes.isEqual(attrs);
|
|
}
|
|
|
|
// Element interface support
|
|
|
|
public AttributeSet getAttributes()
|
|
{
|
|
return attributes;
|
|
}
|
|
|
|
public Document getDocument()
|
|
{
|
|
return AbstractDocument.this;
|
|
}
|
|
|
|
public Element getElement(int index)
|
|
{
|
|
return (Element) element_children.get(index);
|
|
}
|
|
|
|
public String getName()
|
|
{
|
|
return (String) getAttribute(NameAttribute);
|
|
}
|
|
|
|
public Element getParentElement()
|
|
{
|
|
return element_parent;
|
|
}
|
|
|
|
public abstract int getEndOffset();
|
|
|
|
public abstract int getElementCount();
|
|
|
|
public abstract int getElementIndex(int offset);
|
|
|
|
public abstract int getStartOffset();
|
|
}
|
|
|
|
public class BranchElement extends AbstractElement
|
|
{
|
|
private static final long serialVersionUID = -8595176318868717313L;
|
|
|
|
private Vector children = new Vector();
|
|
|
|
public BranchElement(Element parent, AttributeSet attributes)
|
|
{
|
|
super(parent, attributes);
|
|
}
|
|
|
|
public Enumeration children()
|
|
{
|
|
return children.elements();
|
|
}
|
|
|
|
public boolean getAllowsChildren()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
public Element getElement(int index)
|
|
{
|
|
if (index < 0 || index >= children.size())
|
|
return null;
|
|
|
|
return (Element) children.get(index);
|
|
}
|
|
|
|
public int getElementCount()
|
|
{
|
|
return children.size();
|
|
}
|
|
|
|
public int getElementIndex(int offset)
|
|
{
|
|
if (children.size() == 0)
|
|
return 0;
|
|
|
|
Element element = positionToElement(offset);
|
|
|
|
if (element == null)
|
|
return 0;
|
|
|
|
return children.indexOf(element);
|
|
}
|
|
|
|
public int getEndOffset()
|
|
{
|
|
return ((Element) children.lastElement()).getEndOffset();
|
|
}
|
|
|
|
public String getName()
|
|
{
|
|
return "AbstractDocument.BranchElement";
|
|
}
|
|
|
|
public int getStartOffset()
|
|
{
|
|
return ((Element) children.firstElement()).getStartOffset();
|
|
}
|
|
|
|
public boolean isLeaf()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
public Element positionToElement(int position)
|
|
{
|
|
// XXX: There is surely a better algorithm
|
|
// as beginning from first element each time.
|
|
for (int index = 0; index < children.size(); ++index)
|
|
{
|
|
Element elem = (Element) children.get(index);
|
|
|
|
if ((elem.getStartOffset() <= position)
|
|
&& (position < elem.getEndOffset()))
|
|
return elem;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public void replace(int offset, int length, Element[] elems)
|
|
{
|
|
for (int index = 0; index < length; ++index)
|
|
children.removeElementAt(offset);
|
|
|
|
for (int index = 0; index < elems.length; ++index)
|
|
children.add(offset + index, elems[index]);
|
|
}
|
|
|
|
public String toString()
|
|
{
|
|
return getName() + ": " + "content";
|
|
}
|
|
}
|
|
|
|
public class DefaultDocumentEvent extends CompoundEdit
|
|
implements DocumentEvent
|
|
{
|
|
private static final long serialVersionUID = -7406103236022413522L;
|
|
|
|
private int offset;
|
|
private int length;
|
|
private DocumentEvent.EventType type;
|
|
|
|
public DefaultDocumentEvent(int offset, int length,
|
|
DocumentEvent.EventType type)
|
|
{
|
|
this.offset = offset;
|
|
this.length = length;
|
|
this.type = type;
|
|
}
|
|
|
|
public Document getDocument()
|
|
{
|
|
return AbstractDocument.this;
|
|
}
|
|
|
|
public int getLength()
|
|
{
|
|
return length;
|
|
}
|
|
|
|
public int getOffset()
|
|
{
|
|
return offset;
|
|
}
|
|
|
|
public DocumentEvent.EventType getType()
|
|
{
|
|
return type;
|
|
}
|
|
|
|
public DocumentEvent.ElementChange getChange(Element elem)
|
|
{
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public static class ElementEdit extends AbstractUndoableEdit
|
|
implements DocumentEvent.ElementChange
|
|
{
|
|
private static final long serialVersionUID = -1216620962142928304L;
|
|
|
|
private Element elem;
|
|
private int index;
|
|
private Element[] removed;
|
|
private Element[] added;
|
|
|
|
public ElementEdit(Element elem, int index,
|
|
Element[] removed, Element[] added)
|
|
{
|
|
this.elem = elem;
|
|
this.index = index;
|
|
this.removed = removed;
|
|
this.added = added;
|
|
}
|
|
|
|
public Element[] getChildrenAdded()
|
|
{
|
|
return added;
|
|
}
|
|
|
|
public Element[] getChildrenRemoved()
|
|
{
|
|
return removed;
|
|
}
|
|
|
|
public Element getElement()
|
|
{
|
|
return elem;
|
|
}
|
|
|
|
public int getIndex()
|
|
{
|
|
return index;
|
|
}
|
|
}
|
|
|
|
public class LeafElement extends AbstractElement
|
|
{
|
|
private static final long serialVersionUID = 5115368706941283802L;
|
|
private int start;
|
|
private int end;
|
|
|
|
public LeafElement(Element parent, AttributeSet attributes, int start,
|
|
int end)
|
|
{
|
|
super(parent, attributes);
|
|
this.start = start;
|
|
this.end = end;
|
|
}
|
|
|
|
public Enumeration children()
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public boolean getAllowsChildren()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
public Element getElement()
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public int getElementCount()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
public int getElementIndex(int offset)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
public int getEndOffset()
|
|
{
|
|
return end;
|
|
}
|
|
|
|
public String getName()
|
|
{
|
|
return "AbstractDocument.LeafElement";
|
|
}
|
|
|
|
public int getStartOffset()
|
|
{
|
|
return start;
|
|
}
|
|
|
|
public boolean isLeaf()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
public String toString()
|
|
{
|
|
return getName() + ": " + "content";
|
|
}
|
|
}
|
|
}
|