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.
236 lines
7.6 KiB
236 lines
7.6 KiB
// <file>
|
|
// <copyright see="prj:///doc/copyright.txt"/>
|
|
// <license see="prj:///doc/license.txt"/>
|
|
// <owner name="Mike Krüger" email="mike@icsharpcode.net"/>
|
|
// <version>$Revision$</version>
|
|
// </file>
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.Drawing;
|
|
using ICSharpCode.TextEditor.Util;
|
|
|
|
namespace ICSharpCode.TextEditor.Document
|
|
{
|
|
public sealed class LineSegment : ISegment
|
|
{
|
|
internal LineSegmentTree.Enumerator treeEntry;
|
|
|
|
public EolMarker EolMarker { get; set; }
|
|
|
|
public bool IsDeleted => !treeEntry.IsValid;
|
|
|
|
public int LineNumber => treeEntry.CurrentIndex;
|
|
|
|
public int Offset => treeEntry.CurrentOffset;
|
|
|
|
public int Length => TotalLength - DelimiterLength;
|
|
|
|
public int TotalLength { get; internal set; }
|
|
|
|
public int DelimiterLength { get; internal set; }
|
|
|
|
// highlighting information
|
|
public List<TextWord> Words { get; set; }
|
|
|
|
public SpanStack HighlightSpanStack { get; set; }
|
|
|
|
int ISegment.Offset
|
|
{
|
|
get => Offset;
|
|
set => throw new NotSupportedException();
|
|
}
|
|
|
|
int ISegment.Length
|
|
{
|
|
get => Length;
|
|
set => throw new NotSupportedException();
|
|
}
|
|
|
|
public TextWord GetWord(int column)
|
|
{
|
|
var curColumn = 0;
|
|
foreach (var word in Words)
|
|
{
|
|
if (column < curColumn + word.Length)
|
|
return word;
|
|
curColumn += word.Length;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public HighlightColor GetColorForPosition(int x)
|
|
{
|
|
if (Words != null)
|
|
{
|
|
var xPos = 0;
|
|
foreach (var word in Words)
|
|
{
|
|
if (x < xPos + word.Length)
|
|
return word.SyntaxColor;
|
|
xPos += word.Length;
|
|
}
|
|
}
|
|
|
|
return new HighlightColor(nameof(SystemColors.WindowText), bold: false, italic: false);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts a <see cref="LineSegment" /> instance to string (for debug purposes)
|
|
/// </summary>
|
|
public override string ToString()
|
|
{
|
|
if (IsDeleted)
|
|
return "[LineSegment: (deleted) Length = " + Length + ", TotalLength = " + TotalLength + ", DelimiterLength = " + DelimiterLength + "]";
|
|
return "[LineSegment: LineNumber=" + LineNumber + ", Offset = " + Offset + ", Length = " + Length + ", TotalLength = " + TotalLength + ", DelimiterLength = " + DelimiterLength + "]";
|
|
}
|
|
|
|
#region Anchor management
|
|
|
|
private WeakCollection<TextAnchor> anchors;
|
|
|
|
public TextAnchor CreateAnchor(int column)
|
|
{
|
|
if (column < 0 || column > Length)
|
|
throw new ArgumentOutOfRangeException(nameof(column));
|
|
var anchor = new TextAnchor(this, column);
|
|
AddAnchor(anchor);
|
|
return anchor;
|
|
}
|
|
|
|
private void AddAnchor(TextAnchor anchor)
|
|
{
|
|
Debug.Assert(anchor.Line == this);
|
|
|
|
if (anchors == null)
|
|
anchors = new WeakCollection<TextAnchor>();
|
|
|
|
anchors.Add(anchor);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Is called when the LineSegment is deleted.
|
|
/// </summary>
|
|
internal void Deleted(ref DeferredEventList deferredEventList)
|
|
{
|
|
//Console.WriteLine("Deleted");
|
|
treeEntry = LineSegmentTree.Enumerator.Invalid;
|
|
if (anchors != null)
|
|
{
|
|
foreach (var a in anchors)
|
|
a.Delete(ref deferredEventList);
|
|
anchors = null;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Is called when a part of the line is removed.
|
|
/// </summary>
|
|
internal void RemovedLinePart(ref DeferredEventList deferredEventList, int startColumn, int length)
|
|
{
|
|
if (length == 0)
|
|
return;
|
|
Debug.Assert(length > 0);
|
|
|
|
//Console.WriteLine("RemovedLinePart " + startColumn + ", " + length);
|
|
if (anchors != null)
|
|
{
|
|
List<TextAnchor> deletedAnchors = null;
|
|
foreach (var a in anchors)
|
|
if (a.ColumnNumber > startColumn)
|
|
{
|
|
if (a.ColumnNumber >= startColumn + length)
|
|
{
|
|
a.ColumnNumber -= length;
|
|
}
|
|
else
|
|
{
|
|
if (deletedAnchors == null)
|
|
deletedAnchors = new List<TextAnchor>();
|
|
a.Delete(ref deferredEventList);
|
|
deletedAnchors.Add(a);
|
|
}
|
|
}
|
|
|
|
if (deletedAnchors != null)
|
|
foreach (var a in deletedAnchors)
|
|
anchors.Remove(a);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Is called when a part of the line is inserted.
|
|
/// </summary>
|
|
internal void InsertedLinePart(int startColumn, int length)
|
|
{
|
|
if (length == 0)
|
|
return;
|
|
Debug.Assert(length > 0);
|
|
|
|
//Console.WriteLine("InsertedLinePart " + startColumn + ", " + length);
|
|
if (anchors != null)
|
|
foreach (var a in anchors)
|
|
if (a.MovementType == AnchorMovementType.BeforeInsertion
|
|
? a.ColumnNumber > startColumn
|
|
: a.ColumnNumber >= startColumn)
|
|
a.ColumnNumber += length;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Is called after another line's content is appended to this line because the newline in between
|
|
/// was deleted.
|
|
/// The DefaultLineManager will call Deleted() on the deletedLine after the MergedWith call.
|
|
/// firstLineLength: the length of the line before the merge.
|
|
/// </summary>
|
|
internal void MergedWith(LineSegment deletedLine, int firstLineLength)
|
|
{
|
|
//Console.WriteLine("MergedWith");
|
|
|
|
if (deletedLine.anchors != null)
|
|
{
|
|
foreach (var a in deletedLine.anchors)
|
|
{
|
|
a.Line = this;
|
|
AddAnchor(a);
|
|
a.ColumnNumber += firstLineLength;
|
|
}
|
|
|
|
deletedLine.anchors = null;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Is called after a newline was inserted into this line, splitting it into this and followingLine.
|
|
/// </summary>
|
|
internal void SplitTo(LineSegment followingLine)
|
|
{
|
|
//Console.WriteLine("SplitTo");
|
|
|
|
if (anchors != null)
|
|
{
|
|
List<TextAnchor> movedAnchors = null;
|
|
foreach (var a in anchors)
|
|
if (a.MovementType == AnchorMovementType.BeforeInsertion
|
|
? a.ColumnNumber > Length
|
|
: a.ColumnNumber >= Length)
|
|
{
|
|
a.Line = followingLine;
|
|
followingLine.AddAnchor(a);
|
|
a.ColumnNumber -= Length;
|
|
|
|
if (movedAnchors == null)
|
|
movedAnchors = new List<TextAnchor>();
|
|
movedAnchors.Add(a);
|
|
}
|
|
|
|
if (movedAnchors != null)
|
|
foreach (var a in movedAnchors)
|
|
anchors.Remove(a);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|