// Copyright (c) 2014 AlphaSierraPapa for the SharpDevelop Team // // Permission is hereby granted, free of charge, to any person obtaining a copy of this // software and associated documentation files (the "Software"), to deal in the Software // without restriction, including without limitation the rights to use, copy, modify, merge, // publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons // to whom the Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all copies or // substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, // INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR // PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE // FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. using System; using System.Diagnostics; using System.Text; #if NREFACTORY using ICSharpCode.NRefactory.Editor; #endif using ICSharpCode.AvalonEdit.Document; using ICSharpCode.AvalonEdit.Rendering; using ICSharpCode.AvalonEdit.Utils; namespace ICSharpCode.AvalonEdit.Folding { /// /// A section that can be folded. /// public sealed class FoldingSection : TextSegment { readonly FoldingManager manager; bool isFolded; internal CollapsedLineSection[] collapsedSections; string title; /// /// Gets/sets if the section is folded. /// public bool IsFolded { get { return isFolded; } set { if (isFolded != value) { isFolded = value; ValidateCollapsedLineSections(); // create/destroy CollapsedLineSection manager.Redraw(this); } } } internal void ValidateCollapsedLineSections() { if (!isFolded) { RemoveCollapsedLineSection(); return; } // It is possible that StartOffset/EndOffset get set to invalid values via the property setters in TextSegment, // so we coerce those values into the valid range. DocumentLine startLine = manager.document.GetLineByOffset(StartOffset.CoerceValue(0, manager.document.TextLength)); DocumentLine endLine = manager.document.GetLineByOffset(EndOffset.CoerceValue(0, manager.document.TextLength)); if (startLine == endLine) { RemoveCollapsedLineSection(); } else { if (collapsedSections == null) collapsedSections = new CollapsedLineSection[manager.textViews.Count]; // Validate collapsed line sections DocumentLine startLinePlusOne = startLine.NextLine; for (int i = 0; i < collapsedSections.Length; i++) { var collapsedSection = collapsedSections[i]; if (collapsedSection == null || collapsedSection.Start != startLinePlusOne || collapsedSection.End != endLine) { // recreate this collapsed section if (collapsedSection != null) { Debug.WriteLine("CollapsedLineSection validation - recreate collapsed section from " + startLinePlusOne + " to " + endLine); collapsedSection.Uncollapse(); } collapsedSections[i] = manager.textViews[i].CollapseLines(startLinePlusOne, endLine); } } } } /// protected override void OnSegmentChanged() { ValidateCollapsedLineSections(); base.OnSegmentChanged(); // don't redraw if the FoldingSection wasn't added to the FoldingManager's collection yet if (IsConnectedToCollection) manager.Redraw(this); } /// /// Gets/Sets the text used to display the collapsed version of the folding section. /// public string Title { get { return title; } set { if (title != value) { title = value; if (this.IsFolded) manager.Redraw(this); } } } /// /// Gets the content of the collapsed lines as text. /// public string TextContent { get { return manager.document.GetText(StartOffset, EndOffset - StartOffset); } } /// /// Gets the content of the collapsed lines as tooltip text. /// [Obsolete] public string TooltipText { get { // This fixes SD-1394: // Each line is checked for leading indentation whitespaces. If // a line has the same or more indentation than the first line, // it is reduced. If a line is less indented than the first line // the indentation is removed completely. // // See the following example: // line 1 // line 2 // line 3 // line 4 // // is reduced to: // line 1 // line 2 // line 3 // line 4 var startLine = manager.document.GetLineByOffset(StartOffset); var endLine = manager.document.GetLineByOffset(EndOffset); var builder = new StringBuilder(); var current = startLine; ISegment startIndent = TextUtilities.GetLeadingWhitespace(manager.document, startLine); while (current != endLine.NextLine) { ISegment currentIndent = TextUtilities.GetLeadingWhitespace(manager.document, current); if (current == startLine && current == endLine) builder.Append(manager.document.GetText(StartOffset, EndOffset - StartOffset)); else if (current == startLine) { if (current.EndOffset - StartOffset > 0) builder.AppendLine(manager.document.GetText(StartOffset, current.EndOffset - StartOffset).TrimStart()); } else if (current == endLine) { if (startIndent.Length <= currentIndent.Length) builder.Append(manager.document.GetText(current.Offset + startIndent.Length, EndOffset - current.Offset - startIndent.Length)); else builder.Append(manager.document.GetText(current.Offset + currentIndent.Length, EndOffset - current.Offset - currentIndent.Length)); } else { if (startIndent.Length <= currentIndent.Length) builder.AppendLine(manager.document.GetText(current.Offset + startIndent.Length, current.Length - startIndent.Length)); else builder.AppendLine(manager.document.GetText(current.Offset + currentIndent.Length, current.Length - currentIndent.Length)); } current = current.NextLine; } return builder.ToString(); } } /// /// Gets/Sets an additional object associated with this folding section. /// public object Tag { get; set; } internal FoldingSection(FoldingManager manager, int startOffset, int endOffset) { Debug.Assert(manager != null); this.manager = manager; this.StartOffset = startOffset; this.Length = endOffset - startOffset; } void RemoveCollapsedLineSection() { if (collapsedSections != null) { foreach (var collapsedSection in collapsedSections) { if (collapsedSection != null && collapsedSection.Start != null) collapsedSection.Uncollapse(); } collapsedSections = null; } } } }