// Copyright (c) 2010-2013 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.ComponentModel; using System.Globalization; namespace ICSharpCode.AvalonEdit.Document { #if !NREFACTORY /// /// A line/column position. /// Text editor lines/columns are counted started from one. /// /// /// The document provides the methods and /// to convert between offsets and TextLocations. /// [Serializable] [TypeConverter(typeof(TextLocationConverter))] public struct TextLocation : IComparable, IEquatable { /// /// Represents no text location (0, 0). /// public static readonly TextLocation Empty = new TextLocation(0, 0); /// /// Creates a TextLocation instance. /// public TextLocation(int line, int column) { this.line = line; this.column = column; } readonly int column, line; /// /// Gets the line number. /// public int Line { get { return line; } } /// /// Gets the column number. /// public int Column { get { return column; } } /// /// Gets whether the TextLocation instance is empty. /// public bool IsEmpty { get { return column <= 0 && line <= 0; } } /// /// Gets a string representation for debugging purposes. /// public override string ToString() { return string.Format(CultureInfo.InvariantCulture, "(Line {1}, Col {0})", this.column, this.line); } /// /// Gets a hash code. /// public override int GetHashCode() { return unchecked (191 * column.GetHashCode() ^ line.GetHashCode()); } /// /// Equality test. /// public override bool Equals(object obj) { if (!(obj is TextLocation)) return false; return (TextLocation)obj == this; } /// /// Equality test. /// public bool Equals(TextLocation other) { return this == other; } /// /// Equality test. /// public static bool operator ==(TextLocation left, TextLocation right) { return left.column == right.column && left.line == right.line; } /// /// Inequality test. /// public static bool operator !=(TextLocation left, TextLocation right) { return left.column != right.column || left.line != right.line; } /// /// Compares two text locations. /// public static bool operator <(TextLocation left, TextLocation right) { if (left.line < right.line) return true; else if (left.line == right.line) return left.column < right.column; else return false; } /// /// Compares two text locations. /// public static bool operator >(TextLocation left, TextLocation right) { if (left.line > right.line) return true; else if (left.line == right.line) return left.column > right.column; else return false; } /// /// Compares two text locations. /// public static bool operator <=(TextLocation left, TextLocation right) { return !(left > right); } /// /// Compares two text locations. /// public static bool operator >=(TextLocation left, TextLocation right) { return !(left < right); } /// /// Compares two text locations. /// public int CompareTo(TextLocation other) { if (this == other) return 0; if (this < other) return -1; else return 1; } } /// /// Converts strings of the form '0+[;,]0+' to a . /// public class TextLocationConverter : TypeConverter { /// public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType) { return sourceType == typeof(string) || base.CanConvertFrom(context, sourceType); } /// public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType) { return destinationType == typeof(TextLocation) || base.CanConvertTo(context, destinationType); } /// public override object ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, object value) { if (value is string) { string[] parts = ((string)value).Split(';', ','); if (parts.Length == 2) { return new TextLocation(int.Parse(parts[0], culture), int.Parse(parts[1], culture)); } } return base.ConvertFrom(context, culture, value); } /// public override object ConvertTo(ITypeDescriptorContext context, CultureInfo culture, object value, Type destinationType) { if (value is TextLocation && destinationType == typeof(string)) { var loc = (TextLocation)value; return loc.Line.ToString(culture) + ";" + loc.Column.ToString(culture); } return base.ConvertTo(context, culture, value, destinationType); } } /// /// An (Offset,Length)-pair. /// public interface ISegment { /// /// Gets the start offset of the segment. /// int Offset { get; } /// /// Gets the length of the segment. /// /// For line segments (IDocumentLine), the length does not include the line delimeter. int Length { get; } /// /// Gets the end offset of the segment. /// /// EndOffset = Offset + Length; int EndOffset { get; } } /// /// Extension methods for . /// public static class ISegmentExtensions { /// /// Gets whether fully contains the specified segment. /// /// /// Use segment.Contains(offset, 0) to detect whether a segment (end inclusive) contains offset; /// use segment.Contains(offset, 1) to detect whether a segment (end exclusive) contains offset. /// public static bool Contains (this ISegment segment, int offset, int length) { return segment.Offset <= offset && offset + length <= segment.EndOffset; } /// /// Gets whether fully contains the specified segment. /// public static bool Contains (this ISegment thisSegment, ISegment segment) { return segment != null && thisSegment.Offset <= segment.Offset && segment.EndOffset <= thisSegment.EndOffset; } } #endif }