2 // System.Drawing.Drawing2D.LinearGradientBrush.cs
5 // Dennis Hayes (dennish@Raytek.com)
6 // Ravindra (rkumar@novell.com)
8 // Copyright (C) 2002/3 Ximian, Inc. http://www.ximian.com
9 // Copyright (C) 2004,2006 Novell, Inc (http://www.novell.com)
11 // Permission is hereby granted, free of charge, to any person obtaining
12 // a copy of this software and associated documentation files (the
13 // "Software"), to deal in the Software without restriction, including
14 // without limitation the rights to use, copy, modify, merge, publish,
15 // distribute, sublicense, and/or sell copies of the Software, and to
16 // permit persons to whom the Software is furnished to do so, subject to
17 // the following conditions:
19 // The above copyright notice and this permission notice shall be
20 // included in all copies or substantial portions of the Software.
22 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
23 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
24 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
25 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
26 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
27 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
28 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
31 using System.ComponentModel;
32 using System.Runtime.InteropServices;
34 namespace System.Drawing.Drawing2D {
36 public sealed class LinearGradientBrush : Brush
40 internal LinearGradientBrush (IntPtr native)
42 Status status = GDIPlus.GdipGetLineRect (native, out rectangle);
43 SetNativeBrush (native);
44 GDIPlus.CheckStatus (status);
47 public LinearGradientBrush (Point point1, Point point2, Color color1, Color color2)
50 Status status = GDIPlus.GdipCreateLineBrushI (ref point1, ref point2, color1.ToArgb (), color2.ToArgb (), WrapMode.Tile, out nativeObject);
51 GDIPlus.CheckStatus (status);
52 SetNativeBrush (nativeObject);
54 status = GDIPlus.GdipGetLineRect (nativeObject, out rectangle);
55 GDIPlus.CheckStatus (status);
58 public LinearGradientBrush (PointF point1, PointF point2, Color color1, Color color2)
61 Status status = GDIPlus.GdipCreateLineBrush (ref point1, ref point2, color1.ToArgb (), color2.ToArgb (), WrapMode.Tile, out nativeObject);
62 GDIPlus.CheckStatus (status);
63 SetNativeBrush (nativeObject);
65 status = GDIPlus.GdipGetLineRect (nativeObject, out rectangle);
66 GDIPlus.CheckStatus (status);
69 public LinearGradientBrush (Rectangle rect, Color color1, Color color2, LinearGradientMode linearGradientMode)
71 if (linearGradientMode < LinearGradientMode.Horizontal || linearGradientMode > LinearGradientMode.BackwardDiagonal) {
72 throw new InvalidEnumArgumentException (nameof (linearGradientMode), unchecked ((int)linearGradientMode), typeof (LinearGradientMode));
75 if (rect.Width == 0 || rect.Height == 0) {
76 throw new ArgumentException( string.Format ("Rectangle '{0}' cannot have a width or height equal to 0.", rect.ToString ()));
80 Status status = GDIPlus.GdipCreateLineBrushFromRectI (ref rect, color1.ToArgb (), color2.ToArgb (), linearGradientMode, WrapMode.Tile, out nativeObject);
81 GDIPlus.CheckStatus (status);
82 SetNativeBrush (nativeObject);
84 rectangle = (RectangleF) rect;
87 public LinearGradientBrush (Rectangle rect, Color color1, Color color2, float angle) : this (rect, color1, color2, angle, false)
91 public LinearGradientBrush (RectangleF rect, Color color1, Color color2, LinearGradientMode linearGradientMode)
93 if (linearGradientMode < LinearGradientMode.Horizontal || linearGradientMode > LinearGradientMode.BackwardDiagonal) {
94 throw new InvalidEnumArgumentException (nameof (linearGradientMode), unchecked ((int)linearGradientMode), typeof (LinearGradientMode));
97 if (rect.Width == 0.0 || rect.Height == 0.0) {
98 throw new ArgumentException (string.Format ("Rectangle '{0}' cannot have a width or height equal to 0.", rect.ToString ()));
102 Status status = GDIPlus.GdipCreateLineBrushFromRect (ref rect, color1.ToArgb (), color2.ToArgb (), linearGradientMode, WrapMode.Tile, out nativeObject);
103 GDIPlus.CheckStatus (status);
104 SetNativeBrush (nativeObject);
109 public LinearGradientBrush (RectangleF rect, Color color1, Color color2, float angle) : this (rect, color1, color2, angle, false)
113 public LinearGradientBrush (Rectangle rect, Color color1, Color color2, float angle, bool isAngleScaleable)
115 if (rect.Width == 0 || rect.Height == 0) {
116 throw new ArgumentException (string.Format ("Rectangle '{0}' cannot have a width or height equal to 0.", rect.ToString ()));
120 Status status = GDIPlus.GdipCreateLineBrushFromRectWithAngleI (ref rect, color1.ToArgb (), color2.ToArgb (), angle, isAngleScaleable, WrapMode.Tile, out nativeObject);
121 GDIPlus.CheckStatus (status);
122 SetNativeBrush (nativeObject);
124 rectangle = (RectangleF) rect;
127 public LinearGradientBrush (RectangleF rect, Color color1, Color color2, float angle, bool isAngleScaleable)
129 if (rect.Width == 0 || rect.Height == 0) {
130 throw new ArgumentException (string.Format ("Rectangle '{0}' cannot have a width or height equal to 0.", rect.ToString ()));
134 Status status = GDIPlus.GdipCreateLineBrushFromRectWithAngle (ref rect, color1.ToArgb (), color2.ToArgb (), angle, isAngleScaleable, WrapMode.Tile, out nativeObject);
135 GDIPlus.CheckStatus (status);
136 SetNativeBrush (nativeObject);
146 Status status = GDIPlus.GdipGetLineBlendCount (NativeBrush, out count);
147 GDIPlus.CheckStatus (status);
148 float [] factors = new float [count];
149 float [] positions = new float [count];
150 status = GDIPlus.GdipGetLineBlend (NativeBrush, factors, positions, count);
151 GDIPlus.CheckStatus (status);
153 Blend blend = new Blend ();
154 blend.Factors = factors;
155 blend.Positions = positions;
160 // no null check, MS throws a NullReferenceException here
162 float [] factors = value.Factors;
163 float [] positions = value.Positions;
164 count = factors.Length;
166 if (count == 0 || positions.Length == 0)
167 throw new ArgumentException ("Invalid Blend object. It should have at least 2 elements in each of the factors and positions arrays.");
169 if (count != positions.Length)
170 throw new ArgumentException ("Invalid Blend object. It should contain the same number of factors and positions values.");
172 if (positions [0] != 0.0F)
173 throw new ArgumentException ("Invalid Blend object. The positions array must have 0.0 as its first element.");
175 if (positions [count - 1] != 1.0F)
176 throw new ArgumentException ("Invalid Blend object. The positions array must have 1.0 as its last element.");
178 Status status = GDIPlus.GdipSetLineBlend (NativeBrush, factors, positions, count);
179 GDIPlus.CheckStatus (status);
183 [MonoTODO ("The GammaCorrection value is ignored when using libgdiplus.")]
184 public bool GammaCorrection {
186 bool gammaCorrection;
187 Status status = GDIPlus.GdipGetLineGammaCorrection (NativeBrush, out gammaCorrection);
188 GDIPlus.CheckStatus (status);
189 return gammaCorrection;
192 Status status = GDIPlus.GdipSetLineGammaCorrection (NativeBrush, value);
193 GDIPlus.CheckStatus (status);
197 public ColorBlend InterpolationColors {
200 Status status = GDIPlus.GdipGetLinePresetBlendCount (NativeBrush, out count);
201 GDIPlus.CheckStatus (status);
202 int [] intcolors = new int [count];
203 float [] positions = new float [count];
204 status = GDIPlus.GdipGetLinePresetBlend (NativeBrush, intcolors, positions, count);
205 GDIPlus.CheckStatus (status);
207 ColorBlend interpolationColors = new ColorBlend ();
208 Color [] colors = new Color [count];
209 for (int i = 0; i < count; i++)
210 colors [i] = Color.FromArgb (intcolors [i]);
211 interpolationColors.Colors = colors;
212 interpolationColors.Positions = positions;
214 return interpolationColors;
218 throw new ArgumentException ("InterpolationColors is null");
220 Color [] colors = value.Colors;
221 float [] positions = value.Positions;
222 count = colors.Length;
224 if (count == 0 || positions.Length == 0)
225 throw new ArgumentException ("Invalid ColorBlend object. It should have at least 2 elements in each of the colors and positions arrays.");
227 if (count != positions.Length)
228 throw new ArgumentException ("Invalid ColorBlend object. It should contain the same number of positions and color values.");
230 if (positions [0] != 0.0F)
231 throw new ArgumentException ("Invalid ColorBlend object. The positions array must have 0.0 as its first element.");
233 if (positions [count - 1] != 1.0F)
234 throw new ArgumentException ("Invalid ColorBlend object. The positions array must have 1.0 as its last element.");
236 int [] blend = new int [colors.Length];
237 for (int i = 0; i < colors.Length; i++)
238 blend [i] = colors [i].ToArgb ();
240 Status status = GDIPlus.GdipSetLinePresetBlend (NativeBrush, blend, positions, count);
241 GDIPlus.CheckStatus (status);
245 public Color [] LinearColors {
247 int [] colors = new int [2];
248 Status status = GDIPlus.GdipGetLineColors (NativeBrush, colors);
249 GDIPlus.CheckStatus (status);
250 Color [] linearColors = new Color [2];
251 linearColors [0] = Color.FromArgb (colors [0]);
252 linearColors [1] = Color.FromArgb (colors [1]);
257 // no null check, MS throws a NullReferenceException here
258 Status status = GDIPlus.GdipSetLineColors (NativeBrush, value [0].ToArgb (), value [1].ToArgb ());
259 GDIPlus.CheckStatus (status);
263 public RectangleF Rectangle {
269 public Matrix Transform {
271 Matrix matrix = new Matrix ();
272 Status status = GDIPlus.GdipGetLineTransform (NativeBrush, matrix.nativeMatrix);
273 GDIPlus.CheckStatus (status);
279 throw new ArgumentNullException ("Transform");
281 Status status = GDIPlus.GdipSetLineTransform (NativeBrush, value.nativeMatrix);
282 GDIPlus.CheckStatus (status);
286 public WrapMode WrapMode {
289 Status status = GDIPlus.GdipGetLineWrapMode (NativeBrush, out wrapMode);
290 GDIPlus.CheckStatus (status);
295 // note: Clamp isn't valid (context wise) but it is checked in libgdiplus
296 if ((value < WrapMode.Tile) || (value > WrapMode.Clamp))
297 throw new InvalidEnumArgumentException ("WrapMode");
299 Status status = GDIPlus.GdipSetLineWrapMode (NativeBrush, value);
300 GDIPlus.CheckStatus (status);
306 public void MultiplyTransform (Matrix matrix)
308 MultiplyTransform (matrix, MatrixOrder.Prepend);
311 public void MultiplyTransform (Matrix matrix, MatrixOrder order)
314 throw new ArgumentNullException ("matrix");
316 Status status = GDIPlus.GdipMultiplyLineTransform (NativeBrush, matrix.nativeMatrix, order);
317 GDIPlus.CheckStatus (status);
320 public void ResetTransform ()
322 Status status = GDIPlus.GdipResetLineTransform (NativeBrush);
323 GDIPlus.CheckStatus (status);
326 public void RotateTransform (float angle)
328 RotateTransform (angle, MatrixOrder.Prepend);
331 public void RotateTransform (float angle, MatrixOrder order)
333 Status status = GDIPlus.GdipRotateLineTransform (NativeBrush, angle, order);
334 GDIPlus.CheckStatus (status);
337 public void ScaleTransform (float sx, float sy)
339 ScaleTransform (sx, sy, MatrixOrder.Prepend);
342 public void ScaleTransform (float sx, float sy, MatrixOrder order)
344 Status status = GDIPlus.GdipScaleLineTransform (NativeBrush, sx, sy, order);
345 GDIPlus.CheckStatus (status);
348 public void SetBlendTriangularShape (float focus)
350 SetBlendTriangularShape (focus, 1.0F);
353 public void SetBlendTriangularShape (float focus, float scale)
355 if (focus < 0 || focus > 1 || scale < 0 || scale > 1)
356 throw new ArgumentException ("Invalid parameter passed.");
358 Status status = GDIPlus.GdipSetLineLinearBlend (NativeBrush, focus, scale);
359 GDIPlus.CheckStatus (status);
362 public void SetSigmaBellShape (float focus)
364 SetSigmaBellShape (focus, 1.0F);
367 public void SetSigmaBellShape (float focus, float scale)
369 if (focus < 0 || focus > 1 || scale < 0 || scale > 1)
370 throw new ArgumentException ("Invalid parameter passed.");
372 Status status = GDIPlus.GdipSetLineSigmaBlend (NativeBrush, focus, scale);
373 GDIPlus.CheckStatus (status);
376 public void TranslateTransform (float dx, float dy)
378 TranslateTransform (dx, dy, MatrixOrder.Prepend);
381 public void TranslateTransform (float dx, float dy, MatrixOrder order)
383 Status status = GDIPlus.GdipTranslateLineTransform (NativeBrush, dx, dy, order);
384 GDIPlus.CheckStatus (status);
387 public override object Clone ()
390 Status status = (Status) GDIPlus.GdipCloneBrush (new HandleRef (this, NativeBrush), out clonePtr);
391 GDIPlus.CheckStatus (status);
393 return new LinearGradientBrush (clonePtr);