view dwtx/text/edits/TextEdit.d @ 153:f70d9508c95c

Fix java Collection imports
author Frank Benoit <benoit@tionex.de>
date Mon, 25 Aug 2008 00:27:31 +0200
parents 5cf141e43417
children a9566845f1cb
line wrap: on
line source

/*******************************************************************************
 * Copyright (c) 2000, 2008 IBM Corporation and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 *
 * Contributors:
 *     IBM Corporation - initial API and implementation
 * Port to the D programming language:
 *     Frank Benoit <benoit@tionex.de>
 *******************************************************************************/
module dwtx.text.edits.TextEdit;

import dwtx.text.edits.MultiTextEdit; // packageimport
import dwtx.text.edits.CopySourceEdit; // packageimport
import dwtx.text.edits.MoveSourceEdit; // packageimport
import dwtx.text.edits.CopyingRangeMarker; // packageimport
import dwtx.text.edits.ReplaceEdit; // packageimport
import dwtx.text.edits.EditDocument; // packageimport
import dwtx.text.edits.UndoCollector; // packageimport
import dwtx.text.edits.DeleteEdit; // packageimport
import dwtx.text.edits.MoveTargetEdit; // packageimport
import dwtx.text.edits.CopyTargetEdit; // packageimport
import dwtx.text.edits.TextEditCopier; // packageimport
import dwtx.text.edits.ISourceModifier; // packageimport
import dwtx.text.edits.TextEditMessages; // packageimport
import dwtx.text.edits.TextEditProcessor; // packageimport
import dwtx.text.edits.MalformedTreeException; // packageimport
import dwtx.text.edits.TreeIterationInfo; // packageimport
import dwtx.text.edits.TextEditVisitor; // packageimport
import dwtx.text.edits.TextEditGroup; // packageimport
import dwtx.text.edits.RangeMarker; // packageimport
import dwtx.text.edits.UndoEdit; // packageimport
import dwtx.text.edits.InsertEdit; // packageimport


import dwt.dwthelper.utils;

import dwtx.dwtxhelper.Collection;





import dwtx.core.runtime.Assert;
import dwtx.jface.text.BadLocationException;
import dwtx.jface.text.IDocument;
import dwtx.jface.text.IRegion;
import dwtx.jface.text.Region;


/**
 * A text edit describes an elementary text manipulation operation. Edits are
 * executed by applying them to a document (e.g. an instance of <code>IDocument
 * </code>).
 * <p>
 * Text edits form a tree. Clients can navigate the tree upwards, from child to
 * parent, as well as downwards. Newly created edits are un-parented. New edits
 * are added to the tree by calling one of the <code>add</code> methods on a parent
 * edit.
 * </p>
 * <p>
 * An edit tree is well formed in the following sense:
 * <ul>
 *   <li>a parent edit covers all its children</li>
 *   <li>children don't overlap</li>
 *   <li>an edit with length 0 can't have any children</li>
 * </ul>
 * Any manipulation of the tree that violates one of the above requirements results
 * in a <code>MalformedTreeException</code>.
 * </p>
 * <p>
 * Insert edits are represented by an edit of length 0. If more than one insert
 * edit exists at the same offset then the edits are executed in the order in which
 * they have been added to a parent. The following code example:
 * <pre>
 *    IDocument document= new Document("org");
 *    MultiTextEdit edit= new MultiTextEdit();
 *    edit.addChild(new InsertEdit(0, "www."));
 *    edit.addChild(new InsertEdit(0, "eclipse."));
 *    edit.apply(document);
 * </pre>
 * therefore results in string: "www.eclipse.org".
 * </p>
 * <p>
 * Text edits can be executed in a mode where the edit's region is updated to
 * reflect the edit's position in the changed document. Region updating is enabled
 * by default or can be requested by passing <code>UPDATE_REGIONS</code> to the
 * {@link #apply(IDocument, int) apply(IDocument, int)} method. In the above example
 * the region of the <code>InsertEdit(0, "eclipse.")</code> edit after executing
 * the root edit is <code>[3, 8]</code>. If the region of an edit got deleted during
 * change execution the region is set to <code>[-1, -1]</code> and the method {@link
 * #isDeleted() isDeleted} returns <code>true</code>.
 * </p>
 * This class isn't intended to be subclassed outside of the edit framework. Clients
 * are only allowed to subclass <code>MultiTextEdit</code>.
 *
 * @since 3.0
 * @noextend This class is not intended to be subclassed by clients.
 */
public abstract class TextEdit {

    /**
     * Flags indicating that neither <code>CREATE_UNDO</code> nor
     * <code>UPDATE_REGIONS</code> is set.
     */
    public static const int NONE= 0;

    /**
     * Flags indicating that applying an edit tree to a document
     * is supposed to create a corresponding undo edit. If not
     * specified <code>null</code> is returned from method <code>
     * apply</code>.
     */
    public static const int CREATE_UNDO= 1 << 0;

    /**
     * Flag indicating that the edit's region will be updated to
     * reflect its position in the changed document. If not specified
     * when applying an edit tree to a document the edit's region will
     * be arbitrary. It is even not guaranteed that the tree is still
     * well formed.
     */
    public static const int UPDATE_REGIONS= 1 << 1;

    private static class InsertionComparator : Comparator {
        public int compare(Object o1, Object o2)  {
            TextEdit edit1= cast(TextEdit)o1;
            TextEdit edit2= cast(TextEdit)o2;

            int offset1= edit1.getOffset();
            int length1= edit1.getLength();

            int offset2= edit2.getOffset();
            int length2= edit2.getLength();

            if (offset1 is offset2 && length1 is 0 && length2 is 0) {
                return 0;
            }
            if (offset1 + length1 <= offset2) {
                return -1;
            }
            if (offset2 + length2 <= offset1) {
                return 1;
            }
            throw new MalformedTreeException(
                    null, edit1,
                    TextEditMessages.getString("TextEdit.overlapping")); //$NON-NLS-1$
        }
    }

    private static const TextEdit[] EMPTY_ARRAY= new TextEdit[0];
    private static const InsertionComparator INSERTION_COMPARATOR= new InsertionComparator();

    private static const int DELETED_VALUE= -1;

    private int fOffset;
    private int fLength;

    private TextEdit fParent;
    private List fChildren;

    int fDelta;

    /**
     * Create a new text edit. Parent is initialized to <code>
     * null<code> and the edit doesn't have any children.
     *
     * @param offset the edit's offset
     * @param length the edit's length
     */
    protected this(int offset, int length) {
        Assert.isTrue(offset >= 0 && length >= 0);
        fOffset= offset;
        fLength= length;
        fDelta= 0;
    }

    /**
     * Copy constructor
     *
     * @param source the source to copy form
     */
    protected this(TextEdit source) {
        fOffset= source.fOffset;
        fLength= source.fLength;
        fDelta= 0;
    }

    //---- Region management -----------------------------------------------

    /**
     * Returns the range that this edit is manipulating. The returned
     * <code>IRegion</code> contains the edit's offset and length at
     * the point in time when this call is made. Any subsequent changes
     * to the edit's offset and length aren't reflected in the returned
     * region object.
     * <p>
     * Creating a region for a deleted edit will result in an assertion
     * failure.
     *
     * @return the manipulated region
     */
    public final IRegion getRegion() {
        return new Region(getOffset(), getLength());
    }

    /**
     * Returns the offset of the edit. An offset is a 0-based
     * character index. Returns <code>-1</code> if the edit
     * is marked as deleted.
     *
     * @return the offset of the edit
     */
    public int getOffset() {
        return fOffset;
    }

    /**
     * Returns the length of the edit. Returns <code>-1</code>
     * if the edit is marked as deleted.
     *
     * @return the length of the edit
     */
    public int getLength() {
        return fLength;
    }

    /**
     * Returns the inclusive end position of this edit. The inclusive end
     * position denotes the last character of the region manipulated by
     * this edit. The returned value is the result of the following
     * calculation:
     * <pre>
     *   getOffset() + getLength() - 1;
     * <pre>
     *
     * @return the inclusive end position
     */
    public final int getInclusiveEnd() {
        return getOffset() + getLength() - 1;
    }

    /**
     * Returns the exclusive end position of this edit. The exclusive end
     * position denotes the next character of the region manipulated by
     * this edit. The returned value is the result of the following
     * calculation:
     * <pre>
     *   getOffset() + getLength();
     * </pre>
     *
     * @return the exclusive end position
     */
    public final int getExclusiveEnd() {
        return getOffset() + getLength();
    }

    /**
     * Returns whether this edit has been deleted or not.
     *
     * @return <code>true</code> if the edit has been
     *  deleted; otherwise <code>false</code> is returned.
     */
    public final bool isDeleted() {
        return fOffset is DELETED_VALUE && fLength is DELETED_VALUE;
    }

    /**
     * Move all offsets in the tree by the given delta. This node must be a
     * root node. The resulting offsets must be greater or equal to zero.
     *
     * @param delta the delta
     * @since 3.1
     */
    public final void moveTree(int delta) {
        Assert.isTrue(fParent is null);
        Assert.isTrue(getOffset() + delta >= 0);
        internalMoveTree(delta);
    }

    /**
     * Returns <code>true</code> if the edit covers the given edit
     * <code>other</code>. It is up to the concrete text edit to
     * decide if a edit of length zero can cover another edit.
     *
     * @param other the other edit
     * @return <code>true<code> if the edit covers the other edit;
     *  otherwise <code>false</code> is returned.
     */
    public bool covers(TextEdit other) {
        if (getLength() is 0 && !canZeroLengthCover())
            return false;

        if (!other.isDefined())
            return true;

        int thisOffset= getOffset();
        int otherOffset= other.getOffset();
        return thisOffset <= otherOffset && otherOffset + other.getLength() <= thisOffset + getLength();
    }

    /**
     * Returns <code>true</code> if an edit with length zero can cover
     * another edit. Returns <code>false</code> otherwise.
     *
     * @return whether an edit of length zero can cover another edit
     */
    protected bool canZeroLengthCover() {
        return false;
    }

    /**
     * Returns whether the region of this edit is defined or not.
     *
     * @return whether the region of this edit is defined or not
     *
     * @since 3.1
     */
    bool isDefined() {
        return true;
    }

    //---- parent and children management -----------------------------

    /**
     * Returns the edit's parent. The method returns <code>null</code>
     * if this edit hasn't been add to another edit.
     *
     * @return the edit's parent
     */
    public final TextEdit getParent() {
        return fParent;
    }

    /**
     * Returns the root edit of the edit tree.
     *
     * @return the root edit of the edit tree
     * @since 3.1
     */
    public final TextEdit getRoot() {
        TextEdit result= this;
        while (result.fParent !is null) {
            result= result.fParent;
        }
        return result;
    }

    /**
     * Adds the given edit <code>child</code> to this edit.
     *
     * @param child the child edit to add
     * @exception MalformedTreeException is thrown if the child
     *  edit can't be added to this edit. This is the case if the child
     *  overlaps with one of its siblings or if the child edit's region
     *  isn't fully covered by this edit.
     */
    public final void addChild(TextEdit child)  {
        internalAdd(child);
    }

    /**
     * Adds all edits in <code>edits</code> to this edit.
     *
     * @param edits the text edits to add
     * @exception MalformedTreeException is thrown if one of
     *  the given edits can't be added to this edit.
     *
     * @see #addChild(TextEdit)
     */
    public final void addChildren(TextEdit[] edits)  {
        for (int i= 0; i < edits.length; i++) {
            internalAdd(edits[i]);
        }
    }

    /**
     * Removes the edit specified by the given index from the list
     * of children. Returns the child edit that was removed from
     * the list of children. The parent of the returned edit is
     * set to <code>null</code>.
     *
     * @param index the index of the edit to remove
     * @return the removed edit
     * @exception IndexOutOfBoundsException if the index
     *  is out of range
     */
    public final TextEdit removeChild(int index) {
        if (fChildren is null)
            throw new IndexOutOfBoundsException("Index: " + index + " Size: 0");  //$NON-NLS-1$//$NON-NLS-2$
        TextEdit result= cast(TextEdit)fChildren.remove(index);
        result.internalSetParent(null);
        if (fChildren.isEmpty())
            fChildren= null;
        return result;
    }

    /**
     * Removes the first occurrence of the given child from the list
     * of children.
     *
     * @param child the child to be removed
     * @return <code>true</code> if the edit contained the given
     *  child; otherwise <code>false</code> is returned
     */
    public final bool removeChild(TextEdit child) {
        Assert.isNotNull(child);
        if (fChildren is null)
            return false;
        bool result= fChildren.remove(child);
        if (result) {
            child.internalSetParent(null);
            if (fChildren.isEmpty())
                fChildren= null;
        }
        return result;
    }

    /**
     * Removes all child edits from and returns them. The parent
     * of the removed edits is set to <code>null</code>.
     *
     * @return an array of the removed edits
     */
    public final TextEdit[] removeChildren() {
        if (fChildren is null)
            return EMPTY_ARRAY;
        int size= fChildren.size();
        TextEdit[] result= new TextEdit[size];
        for (int i= 0; i < size; i++) {
            result[i]= cast(TextEdit)fChildren.get(i);
            result[i].internalSetParent(null);
        }
        fChildren= null;
        return result;
    }

    /**
     * Returns <code>true</code> if this edit has children. Otherwise
     * <code>false</code> is returned.
     *
     * @return <code>true</code> if this edit has children; otherwise
     *  <code>false</code> is returned
     */
    public final bool hasChildren() {
        return fChildren !is null && ! fChildren.isEmpty();
    }

    /**
     * Returns the edit's children. If the edit doesn't have any
     * children an empty array is returned.
     *
     * @return the edit's children
     */
    public final TextEdit[] getChildren() {
        if (fChildren is null)
            return EMPTY_ARRAY;
        return (TextEdit[])fChildren.toArray(new TextEdit[fChildren.size()]);
    }

    /**
     * Returns the size of the managed children.
     *
     * @return the size of the children
     */
    public final int getChildrenSize() {
        if (fChildren is null)
            return 0;
        return fChildren.size();
    }

    /**
     * Returns the text range spawned by the given array of text edits.
     * The method requires that the given array contains at least one
     * edit. If all edits passed are deleted the method returns <code>
     * null</code>.
     *
     * @param edits an array of edits
     * @return the text range spawned by the given array of edits or
     *  <code>null</code> if all edits are marked as deleted
     */
    public static IRegion getCoverage(TextEdit[] edits) {
        Assert.isTrue(edits !is null && edits.length > 0);

        int offset= Integer.MAX_VALUE;
        int end= Integer.MIN_VALUE;
        int deleted= 0;
        for (int i= 0; i < edits.length; i++) {
            TextEdit edit= edits[i];
            if (edit.isDeleted()) {
                deleted++;
            } else {
                offset= Math.min(offset, edit.getOffset());
                end= Math.max(end, edit.getExclusiveEnd());
            }
        }
        if (edits.length is deleted)
            return null;

        return new Region(offset, end - offset);
    }

    /*
     * Hook called before this edit gets added to the passed
     * parent.
     */
    void aboutToBeAdded(TextEdit parent) {
    }

    //---- Object methods ------------------------------------------------------

    /**
     * The <code>Edit</code> implementation of this <code>Object</code>
     * method uses object identity (is).
     *
     * @param obj the other object
     * @return <code>true</code> iff <code>this is obj</code>; otherwise
     *  <code>false</code> is returned
     *
     * @see Object#equals(java.lang.Object)
     */
    public final bool equals(Object obj) {
        return this is obj; // equivalent to Object.equals
    }

    /**
     * The <code>Edit</code> implementation of this <code>Object</code>
     * method calls uses <code>Object#hashCode()</code> to compute its
     * hash code.
     *
     * @return the object's hash code value
     *
     * @see Object#hashCode()
     */
    public final int hashCode() {
        return super.hashCode();
    }

    /*
     * @see java.lang.Object#toString()
     */
    public String toString() {
        StringBuffer buffer= new StringBuffer();
        toStringWithChildren(buffer, 0);
        return buffer.toString();
    }

    /**
     * Adds the string representation of this text edit without
     * children information to the given string buffer.
     *
     * @param buffer    the string buffer
     * @param indent    the indent level in number of spaces
     * @since 3.3
     */
    void internalToString(StringBuffer buffer, int indent) {
        for (int i= indent; i > 0; i--) {
            buffer.append("  "); //$NON-NLS-1$
        }
        buffer.append("{"); //$NON-NLS-1$
        String name= getClass().getName();
        int index= name.lastIndexOf('.');
        if (index !is -1) {
            buffer.append(name.substring(index + 1));
        } else {
            buffer.append(name);
        }
        buffer.append("} "); //$NON-NLS-1$
        if (isDeleted()) {
            buffer.append("[deleted]"); //$NON-NLS-1$
        } else {
            buffer.append("["); //$NON-NLS-1$
            buffer.append(getOffset());
            buffer.append(","); //$NON-NLS-1$
            buffer.append(getLength());
            buffer.append("]"); //$NON-NLS-1$
        }
    }

    /**
     * Adds the string representation for this text edit
     * and its children to the given string buffer.
     *
     * @param buffer    the string buffer
     * @param indent    the indent level in number of spaces
     * @since 3.3
     */
    private void toStringWithChildren(StringBuffer buffer, int indent) {
        internalToString(buffer, indent);
        if (fChildren !is null) {
            for (Iterator iterator= fChildren.iterator(); iterator.hasNext();) {
                TextEdit child= cast(TextEdit) iterator.next();
                buffer.append('\n');
                child.toStringWithChildren(buffer, indent + 1);
            }
        }
    }

    //---- Copying -------------------------------------------------------------

    /**
     * Creates a deep copy of the edit tree rooted at this
     * edit.
     *
     * @return a deep copy of the edit tree
     * @see #doCopy()
     */
    public final TextEdit copy() {
        TextEditCopier copier= new TextEditCopier(this);
        return copier.perform();
    }

    /**
     * Creates and returns a copy of this edit. The copy method should be
     * implemented in a way so that the copy can executed without causing
     * any harm to the original edit. Implementors of this method are
     * responsible for creating deep or shallow copies of referenced
     * object to fulfill this requirement.
     * <p>
     * Implementers of this method should use the copy constructor <code>
     * Edit#Edit(Edit source) to initialize the edit part of the copy.
     * Implementors aren't responsible to actually copy the children or
     * to set the right parent.
     * <p>
     * This method <b>should not be called</b> from outside the framework.
     * Please use <code>copy</code> to create a copy of a edit tree.
     *
     * @return a copy of this edit.
     * @see #copy()
     * @see #postProcessCopy(TextEditCopier)
     * @see TextEditCopier
     */
    protected abstract TextEdit doCopy();

    /**
     * This method is called on every edit of the copied tree to do some
     * post-processing like connected an edit to a different edit in the tree.
     * <p>
     * This default implementation does nothing
     *
     * @param copier the copier that manages a map between original and
     *  copied edit.
     * @see TextEditCopier
     */
    protected void postProcessCopy(TextEditCopier copier) {
    }

    //---- Visitor support -------------------------------------------------

    /**
     * Accepts the given visitor on a visit of the current edit.
     *
     * @param visitor the visitor object
     * @exception IllegalArgumentException if the visitor is null
     */
    public final void accept(TextEditVisitor visitor) {
        Assert.isNotNull(visitor);
        // begin with the generic pre-visit
        visitor.preVisit(this);
        // dynamic dispatch to internal method for type-specific visit/endVisit
        accept0(visitor);
        // end with the generic post-visit
        visitor.postVisit(this);
    }

    /**
     * Accepts the given visitor on a type-specific visit of the current edit.
     * This method must be implemented in all concrete text edits.
     * <p>
     * General template for implementation on each concrete TextEdit class:
     * <pre>
     * <code>
     * bool visitChildren= visitor.visit(this);
     * if (visitChildren) {
     *    acceptChildren(visitor);
     * }
     * </code>
     * </pre>
     * Note that the caller (<code>accept</code>) takes care of invoking
     * <code>visitor.preVisit(this)</code> and <code>visitor.postVisit(this)</code>.
     * </p>
     *
     * @param visitor the visitor object
     */
    protected abstract void accept0(TextEditVisitor visitor);


    /**
     * Accepts the given visitor on the edits children.
     * <p>
     * This method must be used by the concrete implementations of
     * <code>accept</code> to traverse list-values properties; it
     * encapsulates the proper handling of on-the-fly changes to the list.
     * </p>
     *
     * @param visitor the visitor object
     */
    protected final void acceptChildren(TextEditVisitor visitor) {
        if (fChildren is null)
            return;
        Iterator iterator= fChildren.iterator();
        while (iterator.hasNext()) {
            TextEdit curr= cast(TextEdit) iterator.next();
            curr.accept(visitor);
        }
    }

    //---- Execution -------------------------------------------------------

    /**
     * Applies the edit tree rooted by this edit to the given document. To check
     * if the edit tree can be applied to the document either catch <code>
     * MalformedTreeException</code> or use <code>TextEditProcessor</code> to
     * execute an edit tree.
     *
     * @param document the document to be manipulated
     * @param style flags controlling the execution of the edit tree. Valid
     *  flags are: <code>CREATE_UNDO</code> and </code>UPDATE_REGIONS</code>.
     * @return a undo edit, if <code>CREATE_UNDO</code> is specified. Otherwise
     *  <code>null</code> is returned.
     *
     * @exception MalformedTreeException is thrown if the tree isn't
     *  in a valid state. This exception is thrown before any edit
     *  is executed. So the document is still in its original state.
     * @exception BadLocationException is thrown if one of the edits
     *  in the tree can't be executed. The state of the document is
     *  undefined if this exception is thrown.
     *
     * @see TextEditProcessor#performEdits()
     */
    public final UndoEdit apply(IDocument document, int style)  {
        try {
            TextEditProcessor processor= new TextEditProcessor(document, this, style);
            return processor.performEdits();
        } finally {
            // disconnect from text edit processor
            fParent= null;
        }
    }

    /**
     * Applies the edit tree rooted by this edit to the given document. This
     * method is a convenience method for <code>apply(document, CREATE_UNDO | UPDATE_REGIONS)
     * </code>
     *
     * @param document the document to which to apply this edit
     * @return a undo edit, if <code>CREATE_UNDO</code> is specified. Otherwise
     *  <code>null</code> is returned.
     * @exception MalformedTreeException is thrown if the tree isn't
     *  in a valid state. This exception is thrown before any edit
     *  is executed. So the document is still in its original state.
     * @exception BadLocationException is thrown if one of the edits
     *  in the tree can't be executed. The state of the document is
     *  undefined if this exception is thrown.
     * @see #apply(IDocument, int)
     */
    public final UndoEdit apply(IDocument document)  {
        return apply(document, CREATE_UNDO | UPDATE_REGIONS);
    }

    UndoEdit dispatchPerformEdits(TextEditProcessor processor)  {
        return processor.executeDo();
    }

    void dispatchCheckIntegrity(TextEditProcessor processor)  {
        processor.checkIntegrityDo();
    }

    //---- internal state accessors ----------------------------------------------------------

    void internalSetParent(TextEdit parent) {
        if (parent !is null)
            Assert.isTrue(fParent is null);
        fParent= parent;
    }

    void internalSetOffset(int offset) {
        Assert.isTrue(offset >= 0);
        fOffset= offset;
    }

    void internalSetLength(int length) {
        Assert.isTrue(length >= 0);
        fLength= length;
    }

    List internalGetChildren() {
        return fChildren;
    }

    void internalSetChildren(List children) {
        fChildren= children;
    }

    void internalAdd(TextEdit child)  {
        child.aboutToBeAdded(this);
        if (child.isDeleted())
            throw new MalformedTreeException(this, child, TextEditMessages.getString("TextEdit.deleted_edit")); //$NON-NLS-1$
        if (!covers(child))
            throw new MalformedTreeException(this, child, TextEditMessages.getString("TextEdit.range_outside")); //$NON-NLS-1$
        if (fChildren is null) {
            fChildren= new ArrayList(2);
        }
        int index= computeInsertionIndex(child);
        fChildren.add(index, child);
        child.internalSetParent(this);
    }

    private int computeInsertionIndex(TextEdit edit)  {
        int size= fChildren.size();
        if (size is 0)
            return 0;
        int lastIndex= size - 1;
        TextEdit last= cast(TextEdit)fChildren.get(lastIndex);
        if (last.getExclusiveEnd() <= edit.getOffset())
            return size;
        try {

            int index= Collections.binarySearch(fChildren, edit, INSERTION_COMPARATOR);

            if (index < 0)
                // edit is not in fChildren
                return -index - 1;

            // edit is already in fChildren
            // make sure that multiple insertion points at the same offset are inserted last.
            while (index < lastIndex && INSERTION_COMPARATOR.compare(fChildren.get(index), fChildren.get(index + 1)) is 0)
                index++;

            return index + 1;

        } catch(MalformedTreeException e) {
            e.setParent(this);
            throw e;
        }
    }

    //---- Offset & Length updating -------------------------------------------------

    /**
     * Adjusts the edits offset according to the given
     * delta. This method doesn't update any children.
     *
     * @param delta the delta of the text replace operation
     */
    void adjustOffset(int delta) {
        if (isDeleted())
            return;
        fOffset+= delta;
        Assert.isTrue(fOffset >= 0);
    }

    /**
     * Adjusts the edits length according to the given
     * delta. This method doesn't update any children.
     *
     * @param delta the delta of the text replace operation
     */
    void adjustLength(int delta) {
        if (isDeleted())
            return;
        fLength+= delta;
        Assert.isTrue(fLength >= 0);
    }

    /**
     * Marks the edit as deleted. This method doesn't update
     * any children.
     */
    void markAsDeleted() {
        fOffset= DELETED_VALUE;
        fLength= DELETED_VALUE;
    }

    //---- Edit processing ----------------------------------------------

    /**
     * Traverses the edit tree to perform the consistency check.
     *
     * @param processor the text edit processor
     * @param document the document to be manipulated
     * @param sourceEdits the list of source edits to be performed before
     *  the actual tree is applied to the document
     *
     * @return the number of indirect move or copy target edit children
     */
    int traverseConsistencyCheck(TextEditProcessor processor, IDocument document, List sourceEdits) {
        int result= 0;
        if (fChildren !is null) {
            for (int i= fChildren.size() - 1; i >= 0; i--) {
                TextEdit child= cast(TextEdit)fChildren.get(i);
                result= Math.max(result, child.traverseConsistencyCheck(processor, document, sourceEdits));
            }
        }
        if (processor.considerEdit(this)) {
            performConsistencyCheck(processor, document);
        }
        return result;
    }

    void performConsistencyCheck(TextEditProcessor processor, IDocument document) {
    }

    void traverseSourceComputation(TextEditProcessor processor, IDocument document) {
    }

    void performSourceComputation(TextEditProcessor processor, IDocument document) {
    }

    int traverseDocumentUpdating(TextEditProcessor processor, IDocument document)  {
        int delta= 0;
        if (fChildren !is null) {
            for (int i= fChildren.size() - 1; i >= 0; i--) {
                TextEdit child= cast(TextEdit)fChildren.get(i);
                delta+= child.traverseDocumentUpdating(processor, document);
                childDocumentUpdated();
            }
        }
        if (processor.considerEdit(this)) {
            if (delta !is 0)
                adjustLength(delta);
            int r= performDocumentUpdating(document);
            if (r !is 0)
                adjustLength(r);
            delta+= r;
        }
        return delta;
    }

    /**
     * Hook method called when the document updating of a child edit has been
     * completed. When a client calls {@link #apply(IDocument)} or
     * {@link #apply(IDocument, int)} this method is called
     * {@link #getChildrenSize()} times.
     * <p>
     * May be overridden by subclasses of {@link MultiTextEdit}.
     *
     * @since 3.1
     */
    protected void childDocumentUpdated() {
    }

    abstract int performDocumentUpdating(IDocument document) ;

    int traverseRegionUpdating(TextEditProcessor processor, IDocument document, int accumulatedDelta, bool delete_) {
        performRegionUpdating(accumulatedDelta, delete_);
        if (fChildren !is null) {
            bool childDelete= delete_ || deleteChildren();
            for (Iterator iter= fChildren.iterator(); iter.hasNext();) {
                TextEdit child= cast(TextEdit)iter.next();
                accumulatedDelta= child.traverseRegionUpdating(processor, document, accumulatedDelta, childDelete);
                childRegionUpdated();
            }
        }
        return accumulatedDelta + fDelta;
    }

    /**
     * Hook method called when the region updating of a child edit has been
     * completed. When a client calls {@link #apply(IDocument)} this method is
     * called {@link #getChildrenSize()} times. When calling
     * {@link #apply(IDocument, int)} this method is called
     * {@link #getChildrenSize()} times, when the style parameter contains the
     * {@link #UPDATE_REGIONS} flag.
     * <p>
     * May be overridden by subclasses of {@link MultiTextEdit}.
     *
     * @since 3.1
     */
    protected void childRegionUpdated() {
    }

    void performRegionUpdating(int accumulatedDelta, bool delete_) {
        if (delete_)
            markAsDeleted();
        else
            adjustOffset(accumulatedDelta);
    }

    abstract bool deleteChildren();

    void internalMoveTree(int delta) {
        adjustOffset(delta);
        if (fChildren !is null) {
            for (Iterator iter= fChildren.iterator(); iter.hasNext();) {
                (cast(TextEdit)iter.next()).internalMoveTree(delta);
            }
        }
    }

    void deleteTree() {
        markAsDeleted();
        if (fChildren !is null) {
            for (Iterator iter= fChildren.iterator(); iter.hasNext();) {
                TextEdit child= cast(TextEdit)iter.next();
                child.deleteTree();
            }
        }
    }
}