// 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.Collections.Generic;
using ICSharpCode.AvalonEdit.Document;
#if NREFACTORY
using ICSharpCode.NRefactory.Editor;
#endif
namespace ICSharpCode.AvalonEdit.Editing
{
///
/// Implementation for that stores the segments
/// in a .
///
public class TextSegmentReadOnlySectionProvider : IReadOnlySectionProvider where T : TextSegment
{
readonly TextSegmentCollection segments;
///
/// Gets the collection storing the read-only segments.
///
public TextSegmentCollection Segments {
get { return segments; }
}
///
/// Creates a new TextSegmentReadOnlySectionProvider instance for the specified document.
///
public TextSegmentReadOnlySectionProvider(TextDocument textDocument)
{
segments = new TextSegmentCollection(textDocument);
}
///
/// Creates a new TextSegmentReadOnlySectionProvider instance using the specified TextSegmentCollection.
///
public TextSegmentReadOnlySectionProvider(TextSegmentCollection segments)
{
if (segments == null)
throw new ArgumentNullException("segments");
this.segments = segments;
}
///
/// Gets whether insertion is possible at the specified offset.
///
public virtual bool CanInsert(int offset)
{
foreach (TextSegment segment in segments.FindSegmentsContaining(offset)) {
if (segment.StartOffset < offset && offset < segment.EndOffset)
return false;
}
return true;
}
///
/// Gets the deletable segments inside the given segment.
///
public virtual IEnumerable GetDeletableSegments(ISegment segment)
{
if (segment == null)
throw new ArgumentNullException("segment");
if (segment.Length == 0 && CanInsert(segment.Offset)) {
yield return segment;
yield break;
}
int readonlyUntil = segment.Offset;
foreach (TextSegment ts in segments.FindOverlappingSegments(segment)) {
int start = ts.StartOffset;
int end = start + ts.Length;
if (start > readonlyUntil) {
yield return new SimpleSegment(readonlyUntil, start - readonlyUntil);
}
if (end > readonlyUntil) {
readonlyUntil = end;
}
}
int endOffset = segment.EndOffset;
if (readonlyUntil < endOffset) {
yield return new SimpleSegment(readonlyUntil, endOffset - readonlyUntil);
}
}
}
}