1 // Permission is hereby granted, free of charge, to any person obtaining
2 // a copy of this software and associated documentation files (the
3 // "Software"), to deal in the Software without restriction, including
4 // without limitation the rights to use, copy, modify, merge, publish,
5 // distribute, sublicense, and/or sell copies of the Software, and to
6 // permit persons to whom the Software is furnished to do so, subject to
7 // the following conditions:
9 // The above copyright notice and this permission notice shall be
10 // included in all copies or substantial portions of the Software.
12 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
13 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
14 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
15 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
16 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
17 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
18 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
20 // Copyright (c) 2007 Novell, Inc. (http://www.novell.com)
23 // Chris Toshok (toshok@novell.com)
26 using System.ComponentModel;
27 using System.Globalization;
28 using System.Windows.Converters;
29 using System.Windows.Markup;
30 using System.Windows.Media;
32 namespace System.Windows {
35 [ValueSerializer (typeof (VectorValueSerializer))]
36 [TypeConverter (typeof (VectorConverter))]
37 public struct Vector : IFormattable
39 public Vector (double x, double y)
45 public bool Equals (Vector value)
47 return _x == value.X && _y == value.Y;
50 public override bool Equals (object o)
55 return Equals ((Vector)o);
58 public override int GetHashCode ()
62 return (_x.GetHashCode () * 397) ^ _y.GetHashCode ();
66 public static bool Equals (Vector vector1, Vector vector2)
68 return vector1.Equals (vector2);
71 public static Point Add (Vector vector, Point point)
73 return new Point (vector.X + point.X, vector.Y + point.Y);
76 public static Vector Add (Vector vector1, Vector vector2)
78 return new Vector (vector1.X + vector2.X,
79 vector1.Y + vector2.Y);
82 public static double AngleBetween (Vector vector1, Vector vector2)
84 double cos_theta = (vector1.X * vector2.X + vector1.Y * vector2.Y) / (vector1.Length * vector2.Length);
86 return Math.Acos (cos_theta) / Math.PI * 180;
89 public static double CrossProduct (Vector vector1, Vector vector2)
91 // ... what operation is this exactly?
92 return vector1.X * vector2.Y - vector1.Y * vector2.X;
95 public static double Determinant (Vector vector1, Vector vector2)
97 // same as CrossProduct, it appears.
98 return vector1.X * vector2.Y - vector1.Y * vector2.X;
101 public static Vector Divide (Vector vector, double scalar)
103 return new Vector (vector.X / scalar, vector.Y / scalar);
106 public static double Multiply (Vector vector1, Vector vector2)
108 return vector1.X * vector2.X + vector1.Y * vector2.Y;
111 public static Vector Multiply (Vector vector, Matrix matrix)
113 return new Vector (vector.X * matrix.M11 + vector.Y * matrix.M21,
114 vector.X * matrix.M12 + vector.Y * matrix.M22);
117 public static Vector Multiply (double scalar, Vector vector)
119 return new Vector (scalar * vector.X, scalar * vector.Y);
122 public static Vector Multiply (Vector vector, double scalar)
124 return new Vector (scalar * vector.X, scalar * vector.Y);
127 public void Negate ()
133 public void Normalize ()
135 double ls = LengthSquared;
139 double l = Math.Sqrt (ls);
144 public static Vector Subtract (Vector vector1, Vector vector2)
146 return new Vector (vector1.X - vector2.X, vector1.Y - vector2.Y);
149 public static Vector Parse (string source)
152 throw new ArgumentNullException ("source");
153 var tokenizer = new NumericListTokenizer (source, CultureInfo.InvariantCulture);
156 if (!double.TryParse (tokenizer.GetNextToken (), NumberStyles.Float, CultureInfo.InvariantCulture, out x) ||
157 !double.TryParse (tokenizer.GetNextToken (), NumberStyles.Float, CultureInfo.InvariantCulture, out y))
159 throw new FormatException (string.Format ("Invalid Vector format: {0}", source));
161 if (!tokenizer.HasNoMoreTokens ())
163 throw new InvalidOperationException("Invalid Vector format: " + source);
165 return new Vector(x, y);
168 public override string ToString ()
170 return ToString(null);
173 public string ToString (IFormatProvider provider)
175 return ToString (null, provider);
178 string IFormattable.ToString (string format, IFormatProvider provider)
180 return ToString (format, provider);
183 private string ToString(string format,IFormatProvider formatProvider)
185 if (formatProvider == null)
186 formatProvider = CultureInfo.CurrentCulture;
188 format = string.Empty;
189 var separator = NumericListTokenizer.GetSeparator (formatProvider);
190 var vectorFormat = string.Format ("{{0:{0}}}{1}{{1:{0}}}", format, separator);
191 return string.Format (formatProvider, vectorFormat, _x, _y);
194 public double Length {
195 get { return Math.Sqrt (LengthSquared); }
198 public double LengthSquared {
199 get { return _x * _x + _y * _y; }
213 public static explicit operator Point (Vector vector)
215 return new Point (vector.X, vector.Y);
218 public static explicit operator Size (Vector vector)
220 return new Size (vector.X, vector.Y);
223 public static Vector operator - (Vector vector1, Vector vector2)
225 return Subtract (vector1, vector2);
228 public static Vector operator - (Vector vector)
230 Vector result = vector;
235 public static bool operator != (Vector vector1, Vector vector2)
237 return !Equals (vector1, vector2);
240 public static bool operator == (Vector vector1, Vector vector2)
242 return Equals (vector1, vector2);
245 public static double operator * (Vector vector1, Vector vector2)
247 return Multiply (vector1, vector2);
250 public static Vector operator * (Vector vector, Matrix matrix)
252 return Multiply (vector, matrix);
255 public static Vector operator * (double scalar, Vector vector)
257 return Multiply (scalar, vector);
260 public static Vector operator * (Vector vector, double scalar)
262 return Multiply (vector, scalar);
265 public static Vector operator / (Vector vector, double scalar)
267 return Divide (vector, scalar);
270 public static Point operator + (Vector vector, Point point)
272 return Add (vector, point);
275 public static Vector operator + (Vector vector1, Vector vector2)
277 return Add (vector1, vector2);