// 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.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); /// /// Constant of the minimum line. /// public const int MinLine = 1; /// /// Constant of the minimum column. /// public const int MinColumn = 1; /// /// Creates a TextLocation instance. /// public TextLocation(int line, int column) { this.line = line; this.column = column; } 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 < MinLine && line < MinColumn; } } /// /// 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; } } 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]), int.Parse(parts[1])); } } return base.ConvertFrom(context, culture, value); } public override object ConvertTo(ITypeDescriptorContext context, CultureInfo culture, object value, Type destinationType) { if (value is TextLocation) { var loc = (TextLocation)value; return loc.Line + ";" + loc.Column; } return base.ConvertTo(context, culture, value, destinationType); } } }