#develop (short for SharpDevelop) is a free IDE for .NET programming languages.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

211 lines
5.4 KiB

// <file>
// <copyright see="prj:///doc/copyright.txt"/>
// <license see="prj:///doc/license.txt"/>
// <owner name="Daniel Grunwald"/>
// <version>$Revision$</version>
// </file>
using System;
using System.Diagnostics;
namespace ICSharpCode.AvalonEdit.Document
{
/// <summary>
/// A segment that can be put into a TextSegmentCollection.
/// </summary>
/// <remarks>
/// On insertions at the start or end offset of the text segment, a TextSegmentCollection handling the document
/// changes will keep the TextSegment small;
/// i.e. use AfterInsertion for the start position and BeforeInsertion for the end position.
/// </remarks>
public class TextSegment : ISegment
{
internal ISegmentTree ownerTree;
internal TextSegment left, right, parent;
/// <summary>
/// The color of the segment in the red/black tree.
/// </summary>
internal bool color;
/// <summary>
/// The "length" of the node (distance to previous node)
/// </summary>
internal int nodeLength;
/// <summary>
/// The total "length" of this subtree.
/// </summary>
internal int totalNodeLength; // totalNodeLength = nodeLength + left.totalNodeLength + right.totalNodeLength
/// <summary>
/// The length of the segment (do not confuse with nodeLength).
/// </summary>
internal int segmentLength;
/// <summary>
/// distanceToMaxEnd = Max(segmentLength,
/// left.distanceToMaxEnd + left.Offset - Offset,
/// left.distanceToMaxEnd + right.Offset - Offset)
/// </summary>
internal int distanceToMaxEnd;
int ISegment.Offset {
get { return StartOffset; }
}
/// <summary>
/// Gets whether this segment is connected to a TextSegmentCollection and will automatically
/// update its offsets.
/// </summary>
protected bool IsConnectedToCollection {
get {
return ownerTree != null;
}
}
/// <summary>
/// Gets/Sets the start offset of the segment.
/// </summary>
public int StartOffset {
get {
// If the segment is not connected to a tree, we store the offset in "nodeLength".
// Otherwise, "nodeLength" contains the distance to the start offset of the previous node
Debug.Assert(!(ownerTree == null && parent != null));
Debug.Assert(!(ownerTree == null && left != null));
TextSegment n = this;
int offset = n.nodeLength;
if (n.left != null)
offset += n.left.totalNodeLength;
while (n.parent != null) {
if (n == n.parent.right) {
if (n.parent.left != null)
offset += n.parent.left.totalNodeLength;
offset += n.parent.nodeLength;
}
n = n.parent;
}
return offset;
}
set {
if (value < 0)
throw new ArgumentOutOfRangeException("value", "Offset must not be negative");
if (this.StartOffset != value) {
// need a copy of the variable because ownerTree.Remove() sets this.ownerTree to null
ISegmentTree ownerTree = this.ownerTree;
if (ownerTree != null) {
ownerTree.Remove(this);
nodeLength = value;
ownerTree.Add(this);
} else {
nodeLength = value;
}
}
}
}
/// <summary>
/// Gets the end offset of the segment.
/// </summary>
public int EndOffset {
get {
return StartOffset + Length;
}
set {
int newLength = value - StartOffset;
if (newLength < 0)
throw new ArgumentOutOfRangeException("value", "EndOffset must be greater or equal to StartOffset");
Length = newLength;
}
}
/// <summary>
/// Gets/Sets the length of the segment.
/// </summary>
public int Length {
get {
return segmentLength;
}
set {
if (value < 0)
throw new ArgumentOutOfRangeException("value", "Length must not be negative");
segmentLength = value;
if (ownerTree != null)
ownerTree.UpdateAugmentedData(this);
}
}
internal TextSegment LeftMost {
get {
TextSegment node = this;
while (node.left != null)
node = node.left;
return node;
}
}
internal TextSegment RightMost {
get {
TextSegment node = this;
while (node.right != null)
node = node.right;
return node;
}
}
/// <summary>
/// Gets the inorder successor of the node.
/// </summary>
internal TextSegment Successor {
get {
if (right != null) {
return right.LeftMost;
} else {
TextSegment node = this;
TextSegment oldNode;
do {
oldNode = node;
node = node.parent;
// go up until we are coming out of a left subtree
} while (node != null && node.right == oldNode);
return node;
}
}
}
/// <summary>
/// Gets the inorder predecessor of the node.
/// </summary>
internal TextSegment Predecessor {
get {
if (left != null) {
return left.RightMost;
} else {
TextSegment node = this;
TextSegment oldNode;
do {
oldNode = node;
node = node.parent;
// go up until we are coming out of a right subtree
} while (node != null && node.left == oldNode);
return node;
}
}
}
#if DEBUG
internal string ToDebugString()
{
return "[nodeLength=" + nodeLength + " totalNodeLength=" + totalNodeLength
+ " distanceToMaxEnd=" + distanceToMaxEnd + " MaxEndOffset=" + (StartOffset + distanceToMaxEnd) + "]";
}
#endif
/// <inheritdoc/>
public override string ToString()
{
return "[" + GetType().Name + " Offset=" + StartOffset + " Length=" + Length + " EndOffset=" + EndOffset + "]";
}
}
}