You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			2240 lines
		
	
	
		
			68 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			2240 lines
		
	
	
		
			68 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-------------------------------------------------------------
 | ||
| // <copyright company=<3D>Microsoft Corporation<6F>>
 | ||
| //   Copyright <20> Microsoft Corporation. All Rights Reserved.
 | ||
| // </copyright>
 | ||
| //-------------------------------------------------------------
 | ||
| // @owner=alexgor, deliant
 | ||
| //=================================================================
 | ||
| //  File:		ChartArea3D.cs
 | ||
| //
 | ||
| //  Namespace:	System.Web.UI.WebControls[Windows.Forms].Charting
 | ||
| //
 | ||
| //	Classes:	ChartArea3DStyle, ChartArea3D
 | ||
| //
 | ||
| //  Purpose:	ChartArea3D class represents 3D chart area. It contains
 | ||
| //              methods for coordinates transformation, drawing the 3D
 | ||
| //              scene and many 3D related helper methods.
 | ||
| //
 | ||
| //	Reviewed:	AG - Microsoft 16, 2007
 | ||
| //
 | ||
| //===================================================================
 | ||
| 
 | ||
| #region Used namespaces
 | ||
| using System;
 | ||
| using System.Drawing;
 | ||
| using System.Drawing.Drawing2D;
 | ||
| using System.ComponentModel;
 | ||
| using System.ComponentModel.Design;
 | ||
| using System.Collections;
 | ||
| using System.Globalization;
 | ||
| using System.Collections.Generic;
 | ||
| 
 | ||
| #if WINFORMS_CONTROL
 | ||
| 	using System.Windows.Forms.DataVisualization.Charting;
 | ||
| 	using System.Windows.Forms.DataVisualization.Charting.Data;
 | ||
| 	using System.Windows.Forms.DataVisualization.Charting.ChartTypes;
 | ||
| 	using System.Windows.Forms.DataVisualization.Charting.Utilities;
 | ||
| 	using System.Windows.Forms.DataVisualization.Charting.Borders3D;
 | ||
| #else
 | ||
| 	using System.Web.UI.DataVisualization.Charting;
 | ||
| 	using System.Web.UI.DataVisualization.Charting.ChartTypes;
 | ||
|     using System.Web.UI.DataVisualization.Charting.Utilities;
 | ||
| 	using System.Web.UI;
 | ||
| #endif
 | ||
| 
 | ||
| 
 | ||
| #endregion
 | ||
| 
 | ||
| #if WINFORMS_CONTROL
 | ||
| 	namespace System.Windows.Forms.DataVisualization.Charting
 | ||
| #else
 | ||
| namespace System.Web.UI.DataVisualization.Charting
 | ||
| 
 | ||
| #endif
 | ||
| {
 | ||
| 	#region 3D lightStyle style enumerations
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// A lighting style for a 3D chart area.
 | ||
| 		/// </summary>
 | ||
| 		public enum LightStyle
 | ||
| 		{
 | ||
| 			/// <summary>
 | ||
| 			/// No lighting.
 | ||
| 			/// </summary>
 | ||
| 			None,
 | ||
|             /// <summary>
 | ||
|             /// Simplistic lighting.
 | ||
|             /// </summary>
 | ||
| 			Simplistic,
 | ||
|             /// <summary>
 | ||
|             /// Realistic lighting.
 | ||
|             /// </summary>
 | ||
| 			Realistic
 | ||
| 		}
 | ||
| 
 | ||
| 	#endregion
 | ||
| 
 | ||
| 	#region 3D Center of Projetion coordinates enumeration
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Coordinates of the Center Of Projection
 | ||
| 		/// </summary>
 | ||
| 		[Flags]
 | ||
| 		internal enum COPCoordinates
 | ||
| 		{
 | ||
| 			/// <summary>
 | ||
| 			/// Check X coordinate.
 | ||
| 			/// </summary>
 | ||
| 			X = 1,
 | ||
| 			/// <summary>
 | ||
| 			/// Check Y coordinate.
 | ||
| 			/// </summary>
 | ||
| 			Y = 2,
 | ||
| 			/// <summary>
 | ||
| 			/// Check Z coordinate.
 | ||
| 			/// </summary>
 | ||
| 			Z = 4
 | ||
| 		}
 | ||
| 
 | ||
| 	#endregion
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// The ChartArea3DStyleClass class provides the functionality for 3D attributes of chart areas,
 | ||
|         /// such as rotation angles and perspective.
 | ||
|         /// </summary>
 | ||
| #if ASPPERM_35
 | ||
| 	[AspNetHostingPermission(System.Security.Permissions.SecurityAction.InheritanceDemand, Level = AspNetHostingPermissionLevel.Minimal)]
 | ||
|     [AspNetHostingPermission(System.Security.Permissions.SecurityAction.LinkDemand, Level = AspNetHostingPermissionLevel.Minimal)]
 | ||
| #endif
 | ||
|     public class ChartArea3DStyle
 | ||
| 	{
 | ||
| 		#region Constructor and Initialization
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// ChartArea3DStyle constructor.
 | ||
| 		/// </summary>
 | ||
| 		public ChartArea3DStyle()
 | ||
| 		{
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// ChartArea3DStyle constructor.
 | ||
| 		/// </summary>
 | ||
| 		public ChartArea3DStyle(ChartArea chartArea)
 | ||
| 		{
 | ||
|             this._chartArea = chartArea;
 | ||
| 		}
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Initialize Chart area and axes
 | ||
|         /// </summary>
 | ||
|         /// <param name="chartArea">Chart area object.</param>
 | ||
|         internal void Initialize(ChartArea chartArea)
 | ||
|         {
 | ||
|             this._chartArea = chartArea;
 | ||
|         }
 | ||
| 
 | ||
|         #endregion
 | ||
| 		
 | ||
| 		#region Fields
 | ||
| 
 | ||
| 		// Reference to the chart area object
 | ||
| 		private	ChartArea	_chartArea = null;
 | ||
| 
 | ||
| 		// Enables/disables 3D chart types in the area.
 | ||
| 		private	bool		_enable3D	= false;
 | ||
| 
 | ||
| 		// Indicates that axes are set at the right angle independent of the rotation.
 | ||
| 		private	bool		_isRightAngleAxes	= true;
 | ||
| 
 | ||
| 		// Indicates that series should be drawn as isClustered.
 | ||
| 		private	bool		_isClustered	= false;
 | ||
| 
 | ||
| 		// 3D area lightStyle style.
 | ||
| 		private LightStyle	_lightStyle = LightStyle.Simplistic;
 | ||
| 
 | ||
| 		// 3D area perspective which controls the scaleView of the chart depth.
 | ||
| 		private int			_perspective = 0;
 | ||
| 
 | ||
| 		// Chart area rotation angle around the X axis.
 | ||
| 		private int			_inclination = 30;
 | ||
| 
 | ||
| 		// Chart area rotation angle around the Y axis.
 | ||
| 		private int			_rotation = 30;
 | ||
| 
 | ||
| 		// Chart area walls width.
 | ||
| 		private int			_wallWidth = 7;
 | ||
| 
 | ||
| 		// Series points depth in percentages
 | ||
| 		private int			_pointDepth = 100;
 | ||
| 
 | ||
| 		// Series points gap depth in percentages
 | ||
| 		private int			_pointGapDepth = 100;
 | ||
| 
 | ||
| 		#endregion
 | ||
| 
 | ||
| 		#region Properties
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Gets or sets a Boolean value that toggles 3D for a chart area on and off.
 | ||
|         /// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(false),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_Enable3D"),
 | ||
| 		ParenthesizePropertyNameAttribute(true)
 | ||
| 		]
 | ||
| 		public bool Enable3D
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return this._enable3D;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
|                 if (this._enable3D != value)
 | ||
| 				{
 | ||
|                     this._enable3D = value;
 | ||
| 
 | ||
|                     if (this._chartArea != null)
 | ||
| 					{
 | ||
| #if SUBAXES
 | ||
| 						// If one of the axes has sub axis the scales has to be recalculated
 | ||
| 						foreach(Axis axis in this._chartArea.Axes)
 | ||
| 						{
 | ||
| 							if(axis.SubAxes.Count > 0)
 | ||
| 							{
 | ||
| 								this._chartArea.ResetAutoValues();
 | ||
| 								break;
 | ||
| 							}
 | ||
| 						}
 | ||
| #endif // SUBAXES
 | ||
| 
 | ||
|                         this._chartArea.Invalidate();
 | ||
| 					}
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Gets or sets a Boolean that determines if a chart area  is displayed using an isometric projection.  
 | ||
|         /// </summary>
 | ||
| 		[
 | ||
|         SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(true),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_RightAngleAxes"),
 | ||
| 		RefreshPropertiesAttribute(RefreshProperties.All)
 | ||
| 		]
 | ||
| 		public bool IsRightAngleAxes
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _isRightAngleAxes;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
|                 _isRightAngleAxes = value;
 | ||
| 
 | ||
| 				// Adjust 3D properties values
 | ||
|                 if (_isRightAngleAxes)
 | ||
| 				{
 | ||
| 					// Disable perspective if right angle axis are used
 | ||
|                     this._perspective = 0;
 | ||
| 				}
 | ||
| 
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Gets or sets a Boolean value that determines if bar chart or column 
 | ||
|         /// chart data series are clustered (displayed along distinct rows).  
 | ||
|         /// </summary>
 | ||
| 		[
 | ||
|         SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(false),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_Clustered"),
 | ||
| 		]
 | ||
| 		public bool IsClustered
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _isClustered;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
|                 _isClustered = value;
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets the style of lighting for a 3D chart area.  
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
|         SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(typeof(LightStyle), "Simplistic"),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_Light"),
 | ||
| 		]
 | ||
| 		public LightStyle LightStyle
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _lightStyle;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
|                 _lightStyle = value;
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets the percent of perspective for a 3D chart area.
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(0),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_Perspective"),
 | ||
| 		RefreshPropertiesAttribute(RefreshProperties.All)
 | ||
| 		]
 | ||
| 		public int Perspective
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _perspective;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
| 				if(value < 0 || value > 100)
 | ||
| 				{
 | ||
|                     throw (new ArgumentOutOfRangeException("value", SR.ExceptionChartArea3DPerspectiveInvalid));
 | ||
| 				}
 | ||
| 
 | ||
|                 _perspective = value;
 | ||
| 
 | ||
| 				// Adjust 3D properties values
 | ||
|                 if (_perspective != 0)
 | ||
| 				{
 | ||
| 					// Disable right angle axes
 | ||
|                     this._isRightAngleAxes = false;
 | ||
| 				}
 | ||
| 
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets the inclination for a 3D chart area.
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(30),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_Inclination"),
 | ||
| 		RefreshPropertiesAttribute(RefreshProperties.All)
 | ||
| 		]
 | ||
| 		public int Inclination
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _inclination;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
| 				if(value < -90 || value > 90)
 | ||
| 				{
 | ||
|                     throw (new ArgumentOutOfRangeException("value", SR.ExceptionChartArea3DInclinationInvalid));
 | ||
| 				}
 | ||
|                 _inclination = value;
 | ||
| 
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets the rotation angle for a 3D chart area.
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(30),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_Rotation"),
 | ||
| 		RefreshPropertiesAttribute(RefreshProperties.All)
 | ||
| 		]
 | ||
| 		public int Rotation
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _rotation;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
| 				if(value < -180 || value > 180)
 | ||
| 				{
 | ||
|                     throw (new ArgumentOutOfRangeException("value", SR.ExceptionChartArea3DRotationInvalid));
 | ||
| 				}
 | ||
|                 _rotation = value;
 | ||
| 
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets the width of the walls displayed in 3D chart areas.  
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(7),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_WallWidth"),
 | ||
| 		RefreshPropertiesAttribute(RefreshProperties.All)
 | ||
| 		]
 | ||
| 		public int WallWidth
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _wallWidth;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
| 				if(value < 0 || value > 30)
 | ||
| 				{
 | ||
|                     throw (new ArgumentOutOfRangeException("value", SR.ExceptionChartArea3DWallWidthInvalid));
 | ||
| 				}
 | ||
| 
 | ||
|                 _wallWidth = value;
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets the depth of data points displayed in 3D chart areas (0-1000%).  
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(100),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_PointDepth"),
 | ||
| 		RefreshPropertiesAttribute(RefreshProperties.All)
 | ||
| 		]
 | ||
| 		public int PointDepth
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _pointDepth;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
| 				if(value < 0 || value > 1000)
 | ||
| 				{
 | ||
|                     throw (new ArgumentOutOfRangeException("value", SR.ExceptionChartArea3DPointsDepthInvalid));
 | ||
| 				}
 | ||
| 
 | ||
|                 _pointDepth = value;
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets the distance between series rows in 3D chart areas (0-1000%).
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(100),
 | ||
| 		SRDescription("DescriptionAttributeChartArea3DStyle_PointGapDepth"),
 | ||
| 		RefreshPropertiesAttribute(RefreshProperties.All)
 | ||
| 		]
 | ||
| 		public int PointGapDepth
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _pointGapDepth;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
| 				if(value < 0 || value > 1000)
 | ||
| 				{
 | ||
|                     throw (new ArgumentOutOfRangeException("value", SR.ExceptionChartArea3DPointsGapInvalid));
 | ||
| 				}
 | ||
| 
 | ||
|                 _pointGapDepth = value;
 | ||
|                 if (this._chartArea != null)
 | ||
| 				{
 | ||
|                     this._chartArea.Invalidate();
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		#endregion
 | ||
| 	}
 | ||
| 
 | ||
| 	/// <summary>
 | ||
|     /// ChartArea3D class represents 3D chart area. It contains all the 3D 
 | ||
|     /// scene settings and methods for drawing the 3D plotting area, and calculating 
 | ||
|     /// the depth of chart elements.
 | ||
| 	/// </summary>
 | ||
| 	public partial class ChartArea
 | ||
| 	{
 | ||
| 		#region Fields
 | ||
| 
 | ||
| 		// Chart area 3D style attribuytes
 | ||
| 		private		ChartArea3DStyle	_area3DStyle = new ChartArea3DStyle();
 | ||
| 
 | ||
| 		// Coordinate convertion matrix
 | ||
| 		internal	Matrix3D			matrix3D = new Matrix3D();
 | ||
| 
 | ||
| 		// Chart area scene wall width in relative coordinates
 | ||
| 		internal	SizeF				areaSceneWallWidth = SizeF.Empty;
 | ||
| 
 | ||
| 		// Chart area scene depth
 | ||
| 		internal	float				areaSceneDepth = 0;
 | ||
| 
 | ||
| 		// Visible surfaces in plotting area
 | ||
| 		private		SurfaceNames			_visibleSurfaces;
 | ||
| 
 | ||
| 		// Z axis depth of series points
 | ||
| 		private		double				_pointsDepth = 0;
 | ||
| 
 | ||
| 		// Z axis depth of the gap between isClustered series
 | ||
| 		private		double				_pointsGapDepth = 0;
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Indicates that series order should be reversed to simulate Y axis rotation.
 | ||
| 		/// </summary>
 | ||
| 		private	bool				_reverseSeriesOrder = false;
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Old X axis reversed flag
 | ||
| 		/// </summary>
 | ||
| 		internal	bool				oldReverseX = false;
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Old Y axis reversed flag
 | ||
| 		/// </summary>
 | ||
| 		internal	bool				oldReverseY = false;
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Old Y axis rotation angle
 | ||
| 		/// </summary>
 | ||
| 		internal	int					oldYAngle = 30;
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// List of all stack group names
 | ||
| 		/// </summary>
 | ||
| 		private	ArrayList			_stackGroupNames = null;
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// This list contains an array of series names for each 3D cluster
 | ||
|         /// </summary>
 | ||
| 		internal	List<List<string>>	seriesClusters = null;
 | ||
| 
 | ||
|         #endregion
 | ||
| 
 | ||
| 		#region 3D Style properties
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Gets or sets a ChartArea3DStyle object, used to draw all series in a chart area in 3D.
 | ||
| 		/// </summary>
 | ||
| 		[
 | ||
| 		SRCategory("CategoryAttribute3D"),
 | ||
| 		Bindable(true),
 | ||
| 		DefaultValue(null),
 | ||
| 		SRDescription("DescriptionAttributeArea3DStyle"),
 | ||
| 		DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
 | ||
| 		TypeConverter(typeof(NoNameExpandableObjectConverter)),
 | ||
| #if !WINFORMS_CONTROL
 | ||
| 		PersistenceMode(PersistenceMode.InnerProperty),
 | ||
| #endif
 | ||
| 		]
 | ||
| 		public ChartArea3DStyle Area3DStyle
 | ||
| 		{
 | ||
| 			get
 | ||
| 			{
 | ||
|                 return _area3DStyle;
 | ||
| 			}
 | ||
| 			set
 | ||
| 			{
 | ||
|                 _area3DStyle = value;
 | ||
| 
 | ||
| 				// Initialize style object
 | ||
|                 _area3DStyle.Initialize((ChartArea)this);
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Indicates that series order should be reversed to simulate Y axis rotation.
 | ||
|         /// </summary>
 | ||
|         internal bool ReverseSeriesOrder
 | ||
|         {
 | ||
|             get { return _reverseSeriesOrder; }
 | ||
|         }
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Gets the list of all stack group names
 | ||
|         /// </summary>
 | ||
|         internal ArrayList StackGroupNames
 | ||
|         {
 | ||
|             get { return _stackGroupNames; }
 | ||
|         }
 | ||
| 
 | ||
| 		#endregion
 | ||
| 
 | ||
| 		#region 3D Coordinates transfotmation methods
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Call this method to apply 3D transformations on an array of 3D points (must be done before calling GDI+ drawing methods).
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="points">3D Points array.</param>
 | ||
| 		public void TransformPoints( Point3D[] points )
 | ||
| 		{
 | ||
| 			// Convert Z coordinates from 0-100% to axis values
 | ||
| 			foreach(Point3D pt in points)
 | ||
| 			{
 | ||
| 				pt.Z = (pt.Z / 100f) * this.areaSceneDepth;
 | ||
| 			}
 | ||
| 
 | ||
| 			// Transform points
 | ||
| 			this.matrix3D.TransformPoints( points );
 | ||
| 		}
 | ||
| 
 | ||
| 		#endregion
 | ||
| 
 | ||
| 		#region 3D Scene drawing methods
 | ||
| 		
 | ||
| 		/// <summary>
 | ||
| 		/// Draws chart area 3D scene, which consists of 3 or 2 walls.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="graph">Chart graphics object.</param>
 | ||
| 		/// <param name="position">Chart area 2D position.</param>
 | ||
| 		internal void DrawArea3DScene(ChartGraphics graph, RectangleF position)
 | ||
| 		{
 | ||
| 			// Reference to the chart area class
 | ||
| 			ChartArea chartArea = (ChartArea)this;
 | ||
| 
 | ||
| 			// Calculate relative size of the wall
 | ||
| 			areaSceneWallWidth = graph.GetRelativeSize( new SizeF(this.Area3DStyle.WallWidth, this.Area3DStyle.WallWidth));
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Calculate the depth of the chart area scene
 | ||
| 			//***********************************************************
 | ||
| 			areaSceneDepth = GetArea3DSceneDepth();
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Initialize coordinate transformation matrix
 | ||
| 			//***********************************************************
 | ||
| 			this.matrix3D.Initialize(
 | ||
| 				position, 
 | ||
| 				areaSceneDepth, 
 | ||
| 				this.Area3DStyle.Inclination,
 | ||
| 				this.Area3DStyle.Rotation,
 | ||
| 				this.Area3DStyle.Perspective,
 | ||
| 				this.Area3DStyle.IsRightAngleAxes);
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Initialize Lighting
 | ||
| 			//***********************************************************
 | ||
| 			this.matrix3D.InitLight( 
 | ||
| 				this.Area3DStyle.LightStyle
 | ||
| 			);
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Find chart area visible surfaces
 | ||
| 			//***********************************************************
 | ||
|             _visibleSurfaces = graph.GetVisibleSurfaces(
 | ||
| 				position, 
 | ||
| 				0, 
 | ||
| 				areaSceneDepth, 
 | ||
| 				this.matrix3D);
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Chech if area scene should be drawn
 | ||
| 			//***********************************************************
 | ||
| 			Color	sceneBackColor = chartArea.BackColor;
 | ||
| 
 | ||
| 			// Do not draw the transparent walls
 | ||
| 			if(sceneBackColor == Color.Transparent)
 | ||
| 			{
 | ||
| 				// Area wall is not visible
 | ||
| 				areaSceneWallWidth = SizeF.Empty;
 | ||
| 				return;
 | ||
| 			}
 | ||
| 
 | ||
| 			// If color is not set (default) - use LightGray
 | ||
| 			if(sceneBackColor == Color.Empty)
 | ||
| 			{
 | ||
| 				sceneBackColor = Color.LightGray;
 | ||
| 			}
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Adjust scene 2D rectangle so that wall are drawn
 | ||
| 			//** outside plotting area.
 | ||
| 			//***********************************************************
 | ||
| 			// If bottom wall is visible
 | ||
| 			if(IsBottomSceneWallVisible())
 | ||
| 			{
 | ||
| 				position.Height += areaSceneWallWidth.Height;
 | ||
| 			}
 | ||
| 
 | ||
| 			// Adjust for the left/right wall
 | ||
| 			position.Width += areaSceneWallWidth.Width;
 | ||
| 			if(this.Area3DStyle.Rotation > 0)
 | ||
| 			{
 | ||
| 				position.X -= areaSceneWallWidth.Width;
 | ||
| 			}
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Draw scene walls
 | ||
| 			//***********************************************************
 | ||
| 
 | ||
| 			// Draw back wall
 | ||
| 			RectangleF	wallRect2D = new RectangleF(position.Location, position.Size);
 | ||
| 			float		wallDepth = areaSceneWallWidth.Width;
 | ||
| 			float		wallZPosition = -wallDepth;
 | ||
| 
 | ||
| 			// For isometric projection Front wall should be visible sometimes
 | ||
| 			if( IsMainSceneWallOnFront())
 | ||
| 			{
 | ||
| 				wallZPosition = areaSceneDepth;
 | ||
| 			}
 | ||
| 
 | ||
| 			graph.Fill3DRectangle( 
 | ||
| 				wallRect2D,
 | ||
| 				wallZPosition,
 | ||
| 				wallDepth,
 | ||
| 				this.matrix3D,
 | ||
| 				chartArea.Area3DStyle.LightStyle,
 | ||
| 				sceneBackColor, 
 | ||
| 				chartArea.BorderColor, 
 | ||
| 				chartArea.BorderWidth, 
 | ||
| 				chartArea.BorderDashStyle, 
 | ||
| 				DrawingOperationTypes.DrawElement );
 | ||
| 
 | ||
| 			// Draw side wall on the left or right side
 | ||
| 			wallRect2D = new RectangleF(position.Location, position.Size);
 | ||
| 			wallRect2D.Width = areaSceneWallWidth.Width;
 | ||
| 			if(!IsSideSceneWallOnLeft())
 | ||
| 			{
 | ||
| 				// Wall is on the right side
 | ||
| 				wallRect2D.X = position.Right - areaSceneWallWidth.Width;
 | ||
| 			}
 | ||
| 			graph.Fill3DRectangle( 
 | ||
| 				wallRect2D,
 | ||
| 				0f,
 | ||
| 				areaSceneDepth,
 | ||
| 				this.matrix3D,
 | ||
| 				chartArea.Area3DStyle.LightStyle,
 | ||
| 				sceneBackColor, 
 | ||
| 				chartArea.BorderColor, 
 | ||
| 				chartArea.BorderWidth, 
 | ||
| 				chartArea.BorderDashStyle, 
 | ||
| 				DrawingOperationTypes.DrawElement);
 | ||
| 
 | ||
| 			// Draw bottom wall
 | ||
| 			if(IsBottomSceneWallVisible())
 | ||
| 			{
 | ||
| 				wallRect2D = new RectangleF(position.Location, position.Size);
 | ||
| 				wallRect2D.Height = areaSceneWallWidth.Height;
 | ||
| 				wallRect2D.Y = position.Bottom - areaSceneWallWidth.Height;
 | ||
| 				wallRect2D.Width -= areaSceneWallWidth.Width;
 | ||
| 				if(IsSideSceneWallOnLeft())
 | ||
| 				{
 | ||
| 					wallRect2D.X += areaSceneWallWidth.Width;
 | ||
| 				}
 | ||
| 
 | ||
| 				wallZPosition = 0;
 | ||
| 				graph.Fill3DRectangle( 
 | ||
| 					wallRect2D,
 | ||
| 					0f,
 | ||
| 					areaSceneDepth,
 | ||
| 					this.matrix3D,
 | ||
| 					chartArea.Area3DStyle.LightStyle,
 | ||
| 					sceneBackColor, 
 | ||
| 					chartArea.BorderColor, 
 | ||
| 					chartArea.BorderWidth, 
 | ||
| 					chartArea.BorderDashStyle, 
 | ||
| 					DrawingOperationTypes.DrawElement );
 | ||
| 			}
 | ||
| 
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Helper method which return True if bottom wall of the 
 | ||
| 		/// chart area scene is visible.
 | ||
| 		/// </summary>
 | ||
| 		/// <returns>True if bottom wall is visible.</returns>
 | ||
| 		internal bool IsBottomSceneWallVisible()
 | ||
| 		{
 | ||
| 			return (this.Area3DStyle.Inclination >= 0);
 | ||
| 		}
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Helper method which return True if main wall of the 
 | ||
|         /// chart area scene is displayed on the front side.
 | ||
|         /// </summary>
 | ||
|         /// <returns>True if front wall is visible.</returns>
 | ||
| 		internal bool IsMainSceneWallOnFront()
 | ||
| 		{
 | ||
| 			// Note: Not used in this version!
 | ||
| 			return false;
 | ||
| 		}
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Helper method which return True if side wall of the 
 | ||
|         /// chart area scene is displayed on the left side.
 | ||
|         /// </summary>
 | ||
|         /// <returns>True if bottom wall is visible.</returns>
 | ||
| 		internal bool IsSideSceneWallOnLeft()
 | ||
| 		{
 | ||
| 			return (this.Area3DStyle.Rotation > 0);
 | ||
| 		}
 | ||
| 
 | ||
| 		#endregion
 | ||
| 
 | ||
| 		#region 3D Scene depth claculation methods
 | ||
| 
 | ||
| 		/// <summary>
 | ||
|         /// Call this method to get the Z position of a series (useful for custom drawing).
 | ||
| 		/// </summary>
 | ||
|         /// <param name="series">The series to retrieve the Z position for.</param>
 | ||
| 		/// <returns>The Z position of the specified series. Measured as a percentage of the chart area's depth.</returns>
 | ||
| 		public float GetSeriesZPosition(Series series)
 | ||
| 		{
 | ||
| 			float	positionZ, depth;
 | ||
| 			GetSeriesZPositionAndDepth(series, out depth, out positionZ);
 | ||
| 			return ((positionZ + depth/2f) / this.areaSceneDepth) * 100f;
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Call this method to get the depth of a series in a chart area.
 | ||
| 		/// </summary>
 | ||
|         /// <param name="series">The series to retrieve the depth for.</param>
 | ||
| 		/// <returns>The depth of the specified series. Measured as a percentage of the chart area's depth.</returns>
 | ||
| 		public float GetSeriesDepth(Series series)
 | ||
| 		{
 | ||
| 			float	positionZ, depth;
 | ||
| 			GetSeriesZPositionAndDepth(series, out depth, out positionZ);
 | ||
| 			return (depth / this.areaSceneDepth) * 100f;
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Calculates area 3D scene depth depending on the number of isClustered 
 | ||
| 		/// series and interval between points.
 | ||
| 		/// </summary>
 | ||
| 		/// <returns>Returns the depth of the chart area scene.</returns>
 | ||
| 		private float GetArea3DSceneDepth()
 | ||
| 		{
 | ||
| 			//***********************************************************
 | ||
| 			//** Calculate the smallest interval between points
 | ||
| 			//***********************************************************
 | ||
| 
 | ||
| 			// Check if any series attached to the area is indexed
 | ||
|             bool indexedSeries = ChartHelper.IndexedSeries(this.Common, this._series.ToArray());
 | ||
| 
 | ||
| 			// Smallest interval series
 | ||
| 			Series	smallestIntervalSeries = null;
 | ||
| 			if(this._series.Count > 0)
 | ||
| 			{
 | ||
| 				smallestIntervalSeries = this.Common.DataManager.Series[(string)this._series[0]];
 | ||
| 			}
 | ||
| 
 | ||
| 			// Get X axis
 | ||
| 			Axis	xAxis = ((ChartArea)this).AxisX;
 | ||
| 			if(this._series.Count > 0)
 | ||
| 			{
 | ||
| 				Series	firstSeries = this.Common.DataManager.Series[this._series[0]];
 | ||
| 				if(firstSeries != null && firstSeries.XAxisType == AxisType.Secondary)
 | ||
| 				{
 | ||
| 					xAxis = ((ChartArea)this).AxisX2;
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			// Get smallest interval between points (use interval 1 for indexed series)
 | ||
| 			double clusteredInterval = 1;
 | ||
| 			if(!indexedSeries)
 | ||
| 			{
 | ||
| 				bool sameInterval;
 | ||
| 				clusteredInterval = this.GetPointsInterval(this._series, xAxis.IsLogarithmic, xAxis.logarithmBase, false, out sameInterval, out smallestIntervalSeries);
 | ||
| 			}
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Check if "DrawSideBySide" attribute is set.
 | ||
| 			//***********************************************************
 | ||
| 			bool	drawSideBySide = false;
 | ||
| 			if(smallestIntervalSeries != null)
 | ||
| 			{
 | ||
| 				drawSideBySide = Common.ChartTypeRegistry.GetChartType(smallestIntervalSeries.ChartTypeName).SideBySideSeries;
 | ||
| 				foreach(string seriesName in this._series)
 | ||
| 				{
 | ||
| 					if(this.Common.DataManager.Series[seriesName].IsCustomPropertySet(CustomPropertyName.DrawSideBySide))
 | ||
| 					{
 | ||
| 						string attribValue = this.Common.DataManager.Series[seriesName][CustomPropertyName.DrawSideBySide];
 | ||
| 						if(String.Compare(attribValue, "False", StringComparison.OrdinalIgnoreCase) == 0)
 | ||
| 						{
 | ||
| 							drawSideBySide = false;
 | ||
| 						}
 | ||
| 						else if(String.Compare(attribValue, "True", StringComparison.OrdinalIgnoreCase) == 0)
 | ||
| 						{
 | ||
| 							drawSideBySide = true;
 | ||
| 						}
 | ||
|                         else if (String.Compare(attribValue, "Auto", StringComparison.OrdinalIgnoreCase) == 0)
 | ||
| 						{
 | ||
| 							// Do nothing
 | ||
| 						}
 | ||
| 						else
 | ||
| 						{
 | ||
|                             throw (new InvalidOperationException(SR.ExceptionAttributeDrawSideBySideInvalid));
 | ||
| 						}
 | ||
| 					}
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			// Get smallest interval cate----cal axis
 | ||
| 			Axis	categoricalAxis = ((ChartArea)this).AxisX;
 | ||
| 			if(smallestIntervalSeries != null && smallestIntervalSeries.XAxisType == AxisType.Secondary)
 | ||
| 			{
 | ||
| 				categoricalAxis = ((ChartArea)this).AxisX2;
 | ||
| 			}
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** If series with the smallest interval is displayed
 | ||
| 			//** side-by-side - devide the interval by number of series
 | ||
| 			//** of the same chart type.
 | ||
| 			//***********************************************************
 | ||
| 			double pointWidthSize = 0.8;
 | ||
| 			int	seriesNumber = 1;
 | ||
| 			if(smallestIntervalSeries != null)
 | ||
| 			{
 | ||
| 				// Check if series is side-by-side
 | ||
| 				if(this.Area3DStyle.IsClustered && drawSideBySide)
 | ||
| 				{
 | ||
| 					// Count number of side-by-side series
 | ||
| 					seriesNumber = 0;
 | ||
| 					foreach(string seriesName in this._series)
 | ||
| 					{
 | ||
| 						// Get series object from name
 | ||
| 						Series	curSeries = this.Common.DataManager.Series[seriesName];
 | ||
| 						if(String.Compare(curSeries.ChartTypeName, smallestIntervalSeries.ChartTypeName, StringComparison.OrdinalIgnoreCase) == 0 )
 | ||
| 						{
 | ||
| 							++seriesNumber;
 | ||
| 						}
 | ||
| 					}
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Stacked column and bar charts can be drawn side-by-side
 | ||
| 			//** using the StackGroupName custom properties. The code 
 | ||
| 			//** checks if multiple groups are used how many of these
 | ||
| 			//** groups exsist.
 | ||
| 			//**
 | ||
| 			//** If isClustered mode enabled each stack group is drawn 
 | ||
| 			//** using it's own cluster.
 | ||
| 			//***********************************************************
 | ||
| 			if(smallestIntervalSeries != null && this.Area3DStyle.IsClustered)
 | ||
| 			{
 | ||
| 				// Check series support stack groups
 | ||
| 				if(Common.ChartTypeRegistry.GetChartType(smallestIntervalSeries.ChartTypeName).SupportStackedGroups)
 | ||
| 				{
 | ||
| 					// Calculate how many stack groups exsist
 | ||
| 					seriesNumber = 0;
 | ||
| 					ArrayList stackGroupNames = new ArrayList();
 | ||
| 					foreach(string seriesName in this._series)
 | ||
| 					{
 | ||
| 						// Get series object from name
 | ||
| 						Series	curSeries = this.Common.DataManager.Series[seriesName];
 | ||
| 						if(String.Compare(curSeries.ChartTypeName, smallestIntervalSeries.ChartTypeName, StringComparison.OrdinalIgnoreCase) == 0 )
 | ||
| 						{
 | ||
| 							string seriesStackGroupName = string.Empty;
 | ||
| 							if(curSeries.IsCustomPropertySet(CustomPropertyName.StackedGroupName))
 | ||
| 							{
 | ||
| 								seriesStackGroupName = curSeries[CustomPropertyName.StackedGroupName];
 | ||
| 							}
 | ||
| 
 | ||
| 							// Add group name if it do not already exsist
 | ||
| 							if(!stackGroupNames.Contains(seriesStackGroupName))
 | ||
| 							{
 | ||
| 								stackGroupNames.Add(seriesStackGroupName);
 | ||
| 							}
 | ||
| 						}
 | ||
| 					}
 | ||
| 					seriesNumber = stackGroupNames.Count;
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Check if series provide custom value for point\gap depth
 | ||
| 			//***********************************************************
 | ||
|             _pointsDepth = clusteredInterval * pointWidthSize * this.Area3DStyle.PointDepth / 100.0;
 | ||
|             _pointsDepth = categoricalAxis.GetPixelInterval(_pointsDepth);
 | ||
| 			if(smallestIntervalSeries != null)
 | ||
| 			{
 | ||
|                 _pointsDepth = smallestIntervalSeries.GetPointWidth(
 | ||
| 					this.Common.graph, 
 | ||
| 					categoricalAxis, 
 | ||
| 					clusteredInterval, 
 | ||
| 					0.8) / seriesNumber;
 | ||
|                 _pointsDepth *= this.Area3DStyle.PointDepth / 100.0;
 | ||
| 			}
 | ||
|             _pointsGapDepth = (_pointsDepth * 0.8) * this.Area3DStyle.PointGapDepth / 100.0;
 | ||
| 
 | ||
| 			// Get point depth and gap from series
 | ||
| 			if(smallestIntervalSeries != null)
 | ||
| 			{
 | ||
|                 smallestIntervalSeries.GetPointDepthAndGap(
 | ||
| 					this.Common.graph, 
 | ||
| 					categoricalAxis,
 | ||
|                     ref _pointsDepth,
 | ||
|                     ref _pointsGapDepth);
 | ||
| 			}
 | ||
| 
 | ||
| 
 | ||
| 			//***********************************************************
 | ||
| 			//** Calculate scene depth
 | ||
| 			//***********************************************************
 | ||
|             return (float)((_pointsGapDepth + _pointsDepth) * GetNumberOfClusters());
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Calculates the depth and Z position for specified series.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="series">Series object.</param>
 | ||
| 		/// <param name="depth">Returns series depth.</param>
 | ||
| 		/// <param name="positionZ">Returns series Z position.</param>
 | ||
| 		internal void GetSeriesZPositionAndDepth(Series series, out float depth, out float positionZ)
 | ||
| 		{
 | ||
|             // Check arguments
 | ||
|             if (series == null)
 | ||
|                 throw new ArgumentNullException("series");
 | ||
| 
 | ||
| 			// Get series cluster index
 | ||
| 			int seriesIndex = GetSeriesClusterIndex(series);
 | ||
| 
 | ||
| 			// Initialize the output parameters
 | ||
|             depth = (float)_pointsDepth;
 | ||
|             positionZ = (float)(_pointsGapDepth / 2.0 + (_pointsDepth + _pointsGapDepth) * seriesIndex);
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Returns number of clusters on the Z axis.
 | ||
| 		/// </summary>
 | ||
| 		/// <returns>Number of clusters on the Z axis.</returns>
 | ||
| 		internal int GetNumberOfClusters()
 | ||
| 		{
 | ||
| 			if(this.seriesClusters == null)
 | ||
| 			{
 | ||
| 				// Lists that hold processed chart types and stacked groups
 | ||
| 				ArrayList	processedChartTypes = new ArrayList();
 | ||
| 				ArrayList	processedStackedGroups = new ArrayList();
 | ||
| 
 | ||
| 				// Reset series cluster list
 | ||
| 				this.seriesClusters = new List<List<string>>();
 | ||
| 			
 | ||
| 				// Iterate through all series that belong to this chart area
 | ||
| 				int clusterIndex = -1;
 | ||
| 				foreach(string seriesName in this._series)
 | ||
| 				{
 | ||
| 					// Get series object by name
 | ||
| 					Series	curSeries = this.Common.DataManager.Series[seriesName];
 | ||
| 
 | ||
| 					// Check if stacked chart type is using multiple groups that 
 | ||
| 					// can be displayed in individual clusters
 | ||
| 					if(!this.Area3DStyle.IsClustered &&
 | ||
| 						Common.ChartTypeRegistry.GetChartType(curSeries.ChartTypeName).SupportStackedGroups)
 | ||
| 					{
 | ||
| 						// Get group name
 | ||
| 						string stackGroupName = StackedColumnChart.GetSeriesStackGroupName(curSeries);
 | ||
| 
 | ||
| 						// Check if group was already counted
 | ||
| 						if(processedStackedGroups.Contains(stackGroupName))
 | ||
| 						{
 | ||
| 							// Find in which cluster this stacked group is located
 | ||
| 							bool found = false;
 | ||
| 							for(int index = 0; !found && index < this.seriesClusters.Count; index++)
 | ||
| 							{
 | ||
| 								foreach(string name in this.seriesClusters[index])
 | ||
| 								{
 | ||
| 									// Get series object by name
 | ||
| 									Series	ser = this.Common.DataManager.Series[name];
 | ||
| 									if(stackGroupName == StackedColumnChart.GetSeriesStackGroupName(ser))
 | ||
| 									{
 | ||
| 										clusterIndex = index;
 | ||
| 										found = true;
 | ||
| 									}
 | ||
| 								}
 | ||
| 							}
 | ||
| 						}
 | ||
| 						else
 | ||
| 						{
 | ||
| 							// Increase cluster index
 | ||
| 							clusterIndex = this.seriesClusters.Count;
 | ||
| 
 | ||
| 							// Add processed group name
 | ||
| 							processedStackedGroups.Add(stackGroupName);
 | ||
| 						}
 | ||
| 					}
 | ||
| 
 | ||
| 
 | ||
| 						// Chech if series is displayed in the same cluster than other series
 | ||
| 					else if( Common.ChartTypeRegistry.GetChartType(curSeries.ChartTypeName).Stacked ||
 | ||
| 						(this.Area3DStyle.IsClustered && Common.ChartTypeRegistry.GetChartType(curSeries.ChartTypeName).SideBySideSeries) )
 | ||
| 					{
 | ||
| 						// Check if this chart type is already in the list
 | ||
| 						if(processedChartTypes.Contains(curSeries.ChartTypeName.ToUpper(System.Globalization.CultureInfo.InvariantCulture)))
 | ||
| 						{
 | ||
| 							// Find in which cluster this chart type is located
 | ||
| 							bool found = false;
 | ||
| 							for(int index = 0; !found && index < this.seriesClusters.Count; index++)
 | ||
| 							{
 | ||
| 								foreach(string name in this.seriesClusters[index])
 | ||
| 								{
 | ||
| 									// Get series object by name
 | ||
| 									Series	ser = this.Common.DataManager.Series[name];
 | ||
| 									if(ser.ChartTypeName.ToUpper(System.Globalization.CultureInfo.InvariantCulture) == 
 | ||
| 										curSeries.ChartTypeName.ToUpper(System.Globalization.CultureInfo.InvariantCulture))
 | ||
| 									{
 | ||
| 										clusterIndex = index;
 | ||
| 										found = true;
 | ||
| 									}
 | ||
| 								}
 | ||
| 							}
 | ||
| 						}
 | ||
| 						else
 | ||
| 						{
 | ||
| 							// Increase cluster index
 | ||
| 							clusterIndex = this.seriesClusters.Count;
 | ||
| 
 | ||
| 							// Add new chart type into the collection
 | ||
| 							processedChartTypes.Add(curSeries.ChartTypeName.ToUpper(System.Globalization.CultureInfo.InvariantCulture));
 | ||
| 						}
 | ||
| 					}
 | ||
| 					else
 | ||
| 					{
 | ||
| 						// Create New cluster
 | ||
| 						clusterIndex = this.seriesClusters.Count;
 | ||
| 					}
 | ||
| 
 | ||
| 					// Create an item in the cluster list that will hold all series names
 | ||
| 					if(this.seriesClusters.Count <= clusterIndex)
 | ||
| 					{
 | ||
| 						this.seriesClusters.Add(new List<string>());
 | ||
| 					}
 | ||
| 
 | ||
| 					// Add series name into the current cluster
 | ||
| 					this.seriesClusters[clusterIndex].Add(seriesName);
 | ||
| 				}
 | ||
| 			}
 | ||
| 				
 | ||
| 			return this.seriesClusters.Count;
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Get series cluster index.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="series">Series object.</param>
 | ||
| 		/// <returns>Series cluster index.</returns>
 | ||
| 		internal int GetSeriesClusterIndex(Series series)
 | ||
| 		{
 | ||
| 			// Fill list of clusters
 | ||
| 			if(this.seriesClusters == null)
 | ||
| 			{
 | ||
| 				this.GetNumberOfClusters();
 | ||
| 			}
 | ||
| 
 | ||
| 			// Iterate through all clusters
 | ||
| 			for(int clusterIndex = 0; clusterIndex < this.seriesClusters.Count; clusterIndex++)
 | ||
| 			{
 | ||
| 				List<string> seriesNames = this.seriesClusters[clusterIndex];
 | ||
| 
 | ||
| 				// Iterate through all series names
 | ||
| 				foreach(string seriesName in seriesNames)
 | ||
| 				{
 | ||
| 					if(seriesName == series.Name)
 | ||
| 					{
 | ||
| 						// Check if series are drawn in reversed order
 | ||
| 						if(this._reverseSeriesOrder)
 | ||
| 						{
 | ||
| 							clusterIndex = (this.seriesClusters.Count - 1) - clusterIndex;
 | ||
| 						}
 | ||
| 						return clusterIndex;
 | ||
| 					}
 | ||
| 				}
 | ||
| 			}
 | ||
| 			return 0;
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
| 		#endregion
 | ||
| 
 | ||
| 		#region 3D Scene helper methods
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// This method is used to calculate estimated scene 
 | ||
| 		/// depth. Regular scene depth method can not be used 
 | ||
| 		/// because Plot area position is zero. Instead, Chart 
 | ||
| 		/// area position is used to find depth of the scene. 
 | ||
| 		/// Algorithm which draws axis labels will decide what 
 | ||
| 		/// should be size and position of plotting area.
 | ||
| 		/// </summary>
 | ||
| 		/// <returns>Returns estimated scene depth</returns>
 | ||
| 		private float GetEstimatedSceneDepth()
 | ||
| 		{
 | ||
| 			float sceneDepth;
 | ||
| 
 | ||
| 			ChartArea area = (ChartArea) this;
 | ||
| 
 | ||
| 
 | ||
| 			// Reset current list of clusters
 | ||
| 			this.seriesClusters = null;
 | ||
| 
 | ||
| 
 | ||
| 			ElementPosition plottingAreaRect = area.InnerPlotPosition;
 | ||
| 
 | ||
| 			area.AxisX.PlotAreaPosition = area.Position;
 | ||
| 			area.AxisY.PlotAreaPosition = area.Position;
 | ||
| 			area.AxisX2.PlotAreaPosition = area.Position;
 | ||
| 			area.AxisY2.PlotAreaPosition = area.Position;
 | ||
| 
 | ||
| 			sceneDepth = GetArea3DSceneDepth();
 | ||
| 
 | ||
| 			area.AxisX.PlotAreaPosition = plottingAreaRect;
 | ||
| 			area.AxisY.PlotAreaPosition = plottingAreaRect;
 | ||
| 			area.AxisX2.PlotAreaPosition = plottingAreaRect;
 | ||
| 			area.AxisY2.PlotAreaPosition = plottingAreaRect;
 | ||
| 
 | ||
| 			return sceneDepth;
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Estimate Interval for 3D Charts. When scene is rotated the 
 | ||
| 		/// number of labels should be changed.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="graph">Chart graphics object.</param>
 | ||
| 		internal void Estimate3DInterval(ChartGraphics graph )
 | ||
| 		{
 | ||
| 			// Reference to the chart area class
 | ||
|             ChartArea area = (ChartArea)this;
 | ||
| 
 | ||
| 			// Calculate relative size of the wall
 | ||
| 			areaSceneWallWidth = graph.GetRelativeSize( new SizeF(this.Area3DStyle.WallWidth, this.Area3DStyle.WallWidth));
 | ||
| 			
 | ||
| 			//***********************************************************
 | ||
| 			//** Calculate the depth of the chart area scene
 | ||
| 			//***********************************************************
 | ||
| 			areaSceneDepth = GetEstimatedSceneDepth();
 | ||
| 
 | ||
| 			RectangleF plottingRect = area.Position.ToRectangleF();
 | ||
| 
 | ||
| 			// Check if plot area position was recalculated.
 | ||
| 			// If not and non-auto InnerPlotPosition & Position were
 | ||
| 			// specified - do all needed calculations
 | ||
| 			if(PlotAreaPosition.Width == 0 && 
 | ||
| 				PlotAreaPosition.Height == 0 &&
 | ||
| 				!area.InnerPlotPosition.Auto 
 | ||
| 				&& !area.Position.Auto)
 | ||
| 			{
 | ||
| 				// Initialize plotting area position
 | ||
| 				
 | ||
| 				if(!area.InnerPlotPosition.Auto)
 | ||
| 				{
 | ||
| 					plottingRect.X += (area.Position.Width / 100F) * area.InnerPlotPosition.X;
 | ||
| 					plottingRect.Y += (area.Position.Height / 100F) * area.InnerPlotPosition.Y;
 | ||
| 					plottingRect.Width = (area.Position.Width / 100F) * area.InnerPlotPosition.Width;
 | ||
| 					plottingRect.Height = (area.Position.Height / 100F) * area.InnerPlotPosition.Height;
 | ||
| 				}
 | ||
| 
 | ||
| 			}
 | ||
| 
 | ||
| 			int yAngle = GetRealYAngle( );
 | ||
| 			
 | ||
| 			//***********************************************************
 | ||
| 			//** Initialize coordinate transformation matrix
 | ||
| 			//***********************************************************
 | ||
| 			Matrix3D intervalMatrix3D = new Matrix3D();
 | ||
| 			intervalMatrix3D.Initialize(
 | ||
| 				plottingRect, 
 | ||
| 				areaSceneDepth, 
 | ||
| 				this.Area3DStyle.Inclination,
 | ||
| 				yAngle,
 | ||
| 				this.Area3DStyle.Perspective,
 | ||
| 				this.Area3DStyle.IsRightAngleAxes);
 | ||
| 			bool notUsed;
 | ||
| 			float zPosition;
 | ||
| 			double size;
 | ||
| 			
 | ||
| 			Point3D [] points = new Point3D[8];
 | ||
| 			
 | ||
| 			if( area.switchValueAxes )
 | ||
| 			{
 | ||
| 				
 | ||
| 				// X Axis
 | ||
| 				zPosition = axisX.GetMarksZPosition( out notUsed );
 | ||
| 			
 | ||
| 				points[0] = new Point3D( plottingRect.X, plottingRect.Y, zPosition );
 | ||
| 				points[1] = new Point3D( plottingRect.X, plottingRect.Bottom, zPosition );
 | ||
| 
 | ||
| 				// Y Axis
 | ||
| 				zPosition = axisY.GetMarksZPosition( out notUsed );
 | ||
| 
 | ||
| 				points[2] = new Point3D( plottingRect.X, plottingRect.Bottom, zPosition );
 | ||
| 				points[3] = new Point3D( plottingRect.Right, plottingRect.Bottom, zPosition );
 | ||
| 
 | ||
| 				// X2 Axis
 | ||
| 				zPosition = axisX2.GetMarksZPosition( out notUsed );
 | ||
| 			
 | ||
| 				points[4] = new Point3D( plottingRect.X, plottingRect.Y, zPosition );
 | ||
| 				points[5] = new Point3D( plottingRect.X, plottingRect.Bottom, zPosition );
 | ||
| 
 | ||
| 				// Y2 Axis
 | ||
| 				zPosition = axisY2.GetMarksZPosition( out notUsed );
 | ||
| 
 | ||
| 				points[6] = new Point3D( plottingRect.X, plottingRect.Y, zPosition );
 | ||
| 				points[7] = new Point3D( plottingRect.Right, plottingRect.Y, zPosition );
 | ||
| 			}
 | ||
| 			else
 | ||
| 			{
 | ||
| 				// X Axis
 | ||
| 				zPosition = axisX.GetMarksZPosition( out notUsed );
 | ||
| 			
 | ||
| 				points[0] = new Point3D( plottingRect.X, plottingRect.Bottom, zPosition );
 | ||
| 				points[1] = new Point3D( plottingRect.Right, plottingRect.Bottom, zPosition );
 | ||
| 
 | ||
| 				// Y Axis
 | ||
| 				zPosition = axisY.GetMarksZPosition( out notUsed );
 | ||
| 
 | ||
| 				points[2] = new Point3D( plottingRect.X, plottingRect.Y, zPosition );
 | ||
| 				points[3] = new Point3D( plottingRect.X, plottingRect.Bottom, zPosition );
 | ||
| 
 | ||
| 				// X2 Axis
 | ||
| 				zPosition = axisX2.GetMarksZPosition( out notUsed );
 | ||
| 			
 | ||
| 				points[4] = new Point3D( plottingRect.X, plottingRect.Y, zPosition );
 | ||
| 				points[5] = new Point3D( plottingRect.Right, plottingRect.Y, zPosition );
 | ||
| 
 | ||
| 				// Y2 Axis
 | ||
| 				zPosition = axisY2.GetMarksZPosition( out notUsed );
 | ||
| 
 | ||
| 				points[6] = new Point3D( plottingRect.X, plottingRect.Y, zPosition );
 | ||
| 				points[7] = new Point3D( plottingRect.X, plottingRect.Bottom, zPosition );
 | ||
| 			}
 | ||
| 
 | ||
| 			// Crossing has to be reset because interval and 
 | ||
| 			// sometimes minimum and maximum are changed.
 | ||
| 			foreach( Axis axis in area.Axes )
 | ||
| 			{
 | ||
| 				axis.crossing = axis.tempCrossing;
 | ||
| 			}
 | ||
| 
 | ||
| 			// Transform all points
 | ||
| 			intervalMatrix3D.TransformPoints( points );
 | ||
| 
 | ||
| 			int axisIndx = 0;
 | ||
| 			foreach( Axis axis in area.Axes )
 | ||
| 			{
 | ||
| 				// Find size of projected axis
 | ||
| 				size = Math.Sqrt( 
 | ||
| 					( points[axisIndx].X - points[axisIndx+1].X ) * ( points[axisIndx].X - points[axisIndx+1].X ) + 
 | ||
| 					( points[axisIndx].Y - points[axisIndx+1].Y ) * ( points[axisIndx].Y - points[axisIndx+1].Y ) );
 | ||
| 
 | ||
| 				// At the beginning plotting area chart is not calculated because 
 | ||
| 				// algorithm for labels calculates plotting area position. To 
 | ||
| 				// calculate labels position we need interval and interval 
 | ||
| 				// need this correction. Because of that Chart area is used 
 | ||
| 				// instead of plotting area position. If secondary label is 
 | ||
| 				// enabled error for using chart area position instead of 
 | ||
| 				// plotting area position is much bigger. This value 
 | ||
| 				// corrects this error.
 | ||
| 				float plottingChartAreaCorrection = 1;
 | ||
| 				if( !area.switchValueAxes )
 | ||
| 				{
 | ||
| 					plottingChartAreaCorrection = 0.5F;
 | ||
| 				}
 | ||
| 
 | ||
| 				// Set correction for axis size
 | ||
| 				if( axis.AxisName == AxisName.X || axis.AxisName == AxisName.X2 )
 | ||
| 				{
 | ||
| 					if( area.switchValueAxes )
 | ||
| 						axis.interval3DCorrection =  size / plottingRect.Height;
 | ||
| 					else
 | ||
| 						axis.interval3DCorrection =  size / plottingRect.Width;
 | ||
| 				}
 | ||
| 				else
 | ||
| 				{
 | ||
| 					if( area.switchValueAxes )
 | ||
| 						axis.interval3DCorrection =  size / plottingRect.Width;
 | ||
| 					else
 | ||
| 						axis.interval3DCorrection =  size / plottingRect.Height * plottingChartAreaCorrection;
 | ||
| 				}
 | ||
| 
 | ||
| 				// There is a limit for correction
 | ||
| 				if( axis.interval3DCorrection < 0.15 )
 | ||
| 					axis.interval3DCorrection = 0.15;
 | ||
| 
 | ||
| 				// There is a limit for correction
 | ||
| 				if( axis.interval3DCorrection > 0.8 )
 | ||
| 					axis.interval3DCorrection = 1.0;
 | ||
| 
 | ||
| 				axisIndx += 2;
 | ||
| 
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Calculates real Y angle from Y angle and reverseSeriesOrder field
 | ||
| 		/// </summary>
 | ||
| 		/// <returns>Real Y angle</returns>
 | ||
| 		internal int GetRealYAngle( )
 | ||
| 		{
 | ||
| 			int yAngle;
 | ||
| 
 | ||
| 			// Case from -90 to 90
 | ||
| 			yAngle = this.Area3DStyle.Rotation;
 | ||
| 
 | ||
| 			// Case from 90 to 180
 | ||
| 			if( this._reverseSeriesOrder && this.Area3DStyle.Rotation >= 0 )
 | ||
| 				yAngle = this.Area3DStyle.Rotation - 180;
 | ||
| 
 | ||
| 			// Case from -90 to -180
 | ||
| 			if( this._reverseSeriesOrder && this.Area3DStyle.Rotation <= 0 )
 | ||
| 				yAngle = this.Area3DStyle.Rotation + 180;
 | ||
| 
 | ||
| 			return yAngle;
 | ||
| 		}
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Check if surface element should be drawn on the Back or Front layer.
 | ||
|         /// </summary>
 | ||
|         /// <param name="surfaceName">Surface name.</param>
 | ||
|         /// <param name="backLayer">Back/front layer.</param>
 | ||
|         /// <param name="onEdge">Indicates that element is on the edge (drawn on the back layer).</param>
 | ||
|         /// <returns>True if element should be drawn.</returns>
 | ||
| 		internal bool ShouldDrawOnSurface(SurfaceNames surfaceName, bool backLayer, bool onEdge)
 | ||
| 		{
 | ||
| 			// Check if surface element should be drawn on the Back or Front layer.
 | ||
|             bool isVisible = ((this._visibleSurfaces & surfaceName) == surfaceName);
 | ||
| 
 | ||
| 			// Elements on the edge are drawn on the back layer
 | ||
| 			if(onEdge)
 | ||
| 			{
 | ||
| 				return backLayer;
 | ||
| 			}
 | ||
| 
 | ||
| 			return (backLayer == (!isVisible) );
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Indicates that data points in all series of this 
 | ||
| 		/// chart area should be drawn in reversed order.
 | ||
| 		/// </summary>
 | ||
| 		/// <returns>True if series points should be drawn in reversed order.</returns>
 | ||
| 		internal bool DrawPointsInReverseOrder()
 | ||
| 		{
 | ||
| 			return (this.Area3DStyle.Rotation <= 0);
 | ||
| 		}
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Checks if points should be drawn from sides to center.
 | ||
|         /// </summary>
 | ||
|         /// <param name="coord">Which coordinate of COP (X, Y or Z) to test for surface overlapping</param>
 | ||
|         /// <returns>True if points should be drawn from sides to center.</returns>
 | ||
| 		internal bool DrawPointsToCenter(ref COPCoordinates coord)
 | ||
| 		{
 | ||
| 			bool			result = false;
 | ||
| 			COPCoordinates	resultCoordinates = 0;
 | ||
| 
 | ||
| 			// Check only if perspective is set
 | ||
| 			if(this.Area3DStyle.Perspective != 0)
 | ||
| 			{
 | ||
| 				if( (coord & COPCoordinates.X) == COPCoordinates.X )
 | ||
| 				{
 | ||
| 					// Only when Left & Right sides of plotting area are invisible
 | ||
|                     if ((this._visibleSurfaces & SurfaceNames.Left) == 0 &&
 | ||
|                         (this._visibleSurfaces & SurfaceNames.Right) == 0)
 | ||
| 					{
 | ||
| 						result = true;
 | ||
| 					}
 | ||
| 					resultCoordinates = resultCoordinates | COPCoordinates.X;
 | ||
| 				}
 | ||
| 				if( (coord & COPCoordinates.Y) == COPCoordinates.Y )
 | ||
| 				{
 | ||
| 					// Only when Top & Bottom sides of plotting area are invisible
 | ||
|                     if ((this._visibleSurfaces & SurfaceNames.Top) == 0 &&
 | ||
|                         (this._visibleSurfaces & SurfaceNames.Bottom) == 0)
 | ||
| 					{
 | ||
| 						result = true;
 | ||
| 					}
 | ||
| 					resultCoordinates = resultCoordinates | COPCoordinates.Y;
 | ||
| 				}
 | ||
| 				if( (coord & COPCoordinates.Z) == COPCoordinates.Z )
 | ||
| 				{
 | ||
| 					// Only when Front & Back sides of plotting area are invisible
 | ||
|                     if ((this._visibleSurfaces & SurfaceNames.Front) == 0 &&
 | ||
|                         (this._visibleSurfaces & SurfaceNames.Back) == 0)
 | ||
| 					{
 | ||
| 						result = true;
 | ||
| 					}
 | ||
| 					resultCoordinates = resultCoordinates | COPCoordinates.Z;
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			return result;
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Checks if series should be drawn from sides to center.
 | ||
| 		/// </summary>
 | ||
| 		/// <returns>True if series should be drawn from sides to center.</returns>
 | ||
| 		internal bool DrawSeriesToCenter()
 | ||
| 		{
 | ||
| 			// Check only if perspective is set
 | ||
| 			if(this.Area3DStyle.Perspective != 0)
 | ||
| 			{
 | ||
| 				// Only when Left & Right sides of plotting area are invisible
 | ||
|                 if ((this._visibleSurfaces & SurfaceNames.Front) == 0 &&
 | ||
|                     (this._visibleSurfaces & SurfaceNames.Back) == 0)
 | ||
| 				{
 | ||
| 					return true;
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			return false;
 | ||
| 		}
 | ||
| 
 | ||
| 		#endregion
 | ||
| 
 | ||
| 		#region 3D Series drawing and selection methods
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Draws 3D series in the chart area.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="graph">Chart graphics object.</param>
 | ||
| 		internal void PaintChartSeries3D( ChartGraphics graph )
 | ||
| 		{
 | ||
| 			// Reference to the chart area object
 | ||
| 			ChartArea	area = (ChartArea)this;
 | ||
| 
 | ||
| 			// Get order of series drawing
 | ||
| 			List<Series>	seriesDrawingOrder = GetSeriesDrawingOrder(_reverseSeriesOrder);
 | ||
| 
 | ||
| 			// Loop through all series in the order of drawing
 | ||
| 			IChartType type;
 | ||
| 			foreach( object seriesObj in seriesDrawingOrder)
 | ||
| 			{
 | ||
| 				Series series = (Series)seriesObj;
 | ||
| 				type = Common.ChartTypeRegistry.GetChartType(series.ChartTypeName);
 | ||
| 				type.Paint( graph, Common, area, series );
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		#endregion
 | ||
| 
 | ||
| 		#region 3D Series & Points drawing order methods
 | ||
|                 
 | ||
| 		/// <summary>
 | ||
| 		/// Gets a list of series names that belong to the same 3D cluster.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="seriesName">One of the series names that belongs to the cluster.</param>
 | ||
| 		/// <returns>List of all series names that belong to the same cluster as specified series.</returns>
 | ||
| 		internal List<string> GetClusterSeriesNames(string seriesName)
 | ||
| 		{
 | ||
| 			// Iterate through all clusters
 | ||
| 			foreach(List<string> seriesNames in this.seriesClusters)
 | ||
| 			{
 | ||
| 				if(seriesNames.Contains(seriesName))
 | ||
| 				{
 | ||
| 					return seriesNames;
 | ||
| 				}
 | ||
| 			}
 | ||
| 			return new List<string>();
 | ||
| 		}
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Gets the series list in drawing order.
 | ||
|         /// </summary>
 | ||
|         /// <param name="reverseSeriesOrder">Series order should be reversed because of the Y axis angle.</param>
 | ||
|         /// <returns>Gets the series list in drawing order.</returns>
 | ||
| 		private List<Series> GetSeriesDrawingOrder(bool reverseSeriesOrder)
 | ||
| 		{
 | ||
| 			// Create list of series
 | ||
|             List<Series> seriesList = new List<Series>();
 | ||
| 
 | ||
| 			// Iterate through all clusters
 | ||
| 			foreach(List<string> seriesNames in this.seriesClusters)
 | ||
| 			{
 | ||
| 				// Make sure there is at least one series
 | ||
| 				if(seriesNames.Count > 0)
 | ||
| 				{
 | ||
| 					// Get first series object in the current cluster
 | ||
| 					Series series = Common.DataManager.Series[seriesNames[0]];
 | ||
| 
 | ||
| 					// Add series into the drawing list
 | ||
| 					seriesList.Add(series);
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			// Reversed series list
 | ||
| 			if(reverseSeriesOrder)
 | ||
| 			{
 | ||
| 				seriesList.Reverse();
 | ||
| 			}
 | ||
| 
 | ||
| 			// Check if series should be drawn from sides into the center
 | ||
| 			if(DrawSeriesToCenter() &&
 | ||
| 				this.matrix3D.IsInitialized())
 | ||
| 			{
 | ||
| 				// Get Z coordinate of Center Of Projection
 | ||
| 				Point3D		areaProjectionCenter = new Point3D(float.NaN, float.NaN, float.NaN);
 | ||
| 				areaProjectionCenter = this.GetCenterOfProjection(COPCoordinates.Z);
 | ||
| 				if(!float.IsNaN(areaProjectionCenter.Z))
 | ||
| 				{
 | ||
| 					// Loop through all series
 | ||
| 					for(int seriesIndex = 0; seriesIndex < seriesList.Count; seriesIndex++)
 | ||
| 					{
 | ||
| 						// Check if series is not empty
 | ||
| 						if(((Series)seriesList[seriesIndex]).Points.Count == 0)
 | ||
| 						{
 | ||
| 							continue;
 | ||
| 						}
 | ||
| 
 | ||
| 						// Get series Z position
 | ||
| 						float seriesDepth, seriesZPosition;
 | ||
| 						this.GetSeriesZPositionAndDepth((Series)seriesList[seriesIndex], out seriesDepth, out seriesZPosition);
 | ||
| 
 | ||
| 						// Check if series passes the Z coordinate of Center of Projection
 | ||
| 						if(seriesZPosition >= areaProjectionCenter.Z)
 | ||
| 						{
 | ||
| 							// Reversed all series order staring from previous series
 | ||
| 							--seriesIndex;
 | ||
| 							if(seriesIndex < 0)
 | ||
| 								seriesIndex = 0;
 | ||
| 							seriesList.Reverse(seriesIndex, seriesList.Count - seriesIndex);
 | ||
| 							break;
 | ||
| 						}
 | ||
| 					}
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			return seriesList;
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Gets number of stack groups in specified array of series names.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="seriesNamesList">Array of series names.</param>
 | ||
| 		/// <returns>Number of stack groups. One by default.</returns>
 | ||
| 		private int GetNumberOfStackGroups(IList<string> seriesNamesList)
 | ||
| 		{
 | ||
| 			this._stackGroupNames = new ArrayList();
 | ||
| 			foreach( object seriesName in seriesNamesList )
 | ||
| 			{
 | ||
| 				// Get series object
 | ||
| 				Series	ser = this.Common.DataManager.Series[(string)seriesName];
 | ||
| 
 | ||
| 				// Get stack group name from the series
 | ||
| 				string stackGroupName = string.Empty;
 | ||
| 				if(ser.IsCustomPropertySet(CustomPropertyName.StackedGroupName))
 | ||
| 				{
 | ||
| 					stackGroupName = ser[CustomPropertyName.StackedGroupName];
 | ||
| 				}
 | ||
| 
 | ||
| 				// Add group name if it do not already exsist
 | ||
| 				if(!this._stackGroupNames.Contains(stackGroupName))
 | ||
| 				{
 | ||
| 					this._stackGroupNames.Add(stackGroupName);
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			return this._stackGroupNames.Count;
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Gets index of the series stack group.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="series">Series to get the index for.</param>
 | ||
| 		/// <param name="stackGroupName">Group name this series belongs to.</param>
 | ||
| 		/// <returns>Index of series stack group.</returns>
 | ||
| 		internal int GetSeriesStackGroupIndex(Series series, ref string stackGroupName)
 | ||
| 		{
 | ||
| 			stackGroupName = string.Empty;
 | ||
| 			if(this._stackGroupNames != null)
 | ||
| 			{
 | ||
| 				// Get stack group name from the series
 | ||
| 				if(series.IsCustomPropertySet(CustomPropertyName.StackedGroupName))
 | ||
| 				{
 | ||
| 					stackGroupName = series[CustomPropertyName.StackedGroupName];
 | ||
| 				}
 | ||
| 				return this._stackGroupNames.IndexOf(stackGroupName);
 | ||
| 			}
 | ||
| 			return 0;
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
|         /// <summary>
 | ||
|         /// Determine the order of points drawing from one or several series of the same type.
 | ||
|         /// </summary>
 | ||
|         /// <param name="seriesNamesList">List of series names.</param>
 | ||
|         /// <param name="chartType">Chart type.</param>
 | ||
|         /// <param name="selection">If True selection mode is active (points order should be reversed).</param>
 | ||
|         /// <param name="coord">Which coordinate of COP (X, Y or Z) to test for surface overlapping</param>
 | ||
|         /// <param name="comparer">Points comparer class. Can be Null.</param>
 | ||
|         /// <param name="mainYValueIndex">Index of the main Y value.</param>
 | ||
|         /// <param name="sideBySide">Series should be drawn side by side.</param>
 | ||
|         /// <returns>Array list of points in drawing order.</returns>
 | ||
| 		internal ArrayList GetDataPointDrawingOrder(
 | ||
| 			List<string> seriesNamesList, 
 | ||
| 			IChartType chartType, 
 | ||
| 			bool selection, 
 | ||
| 			COPCoordinates coord, 
 | ||
| 			IComparer comparer,
 | ||
| 			int mainYValueIndex,
 | ||
| 			bool sideBySide)
 | ||
| 		{
 | ||
| 			ChartArea area = (ChartArea)this;
 | ||
| 
 | ||
| 			// Array of points in all series
 | ||
| 			ArrayList pointsList = new ArrayList();
 | ||
| 
 | ||
| 			//************************************************************
 | ||
| 			//** Analyse input series
 | ||
| 			//************************************************************
 | ||
| 
 | ||
| 			// Find the number of data series for side-by-side drawing
 | ||
| 			double	numOfSeries = 1;
 | ||
| 			if(area.Area3DStyle.IsClustered && !chartType.Stacked && sideBySide)
 | ||
| 			{
 | ||
| 				numOfSeries = seriesNamesList.Count;
 | ||
| 			}
 | ||
| 
 | ||
| 
 | ||
| 			// Check stacked series group names
 | ||
| 			if(chartType.SupportStackedGroups)
 | ||
| 			{
 | ||
| 				// Fill the list of group names and get the number of unique groups
 | ||
| 				int numberOfGroups = this.GetNumberOfStackGroups(seriesNamesList);
 | ||
| 
 | ||
| 				// If series are not isClustered set series number to the stacked group number
 | ||
| 				if(this.Area3DStyle.IsClustered &&
 | ||
| 					seriesNamesList.Count > 0)
 | ||
| 				{
 | ||
| 					numOfSeries = numberOfGroups;
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 
 | ||
| 			// Check if chart series are indexed
 | ||
|             bool indexedSeries = ChartHelper.IndexedSeries(this.Common, seriesNamesList.ToArray());
 | ||
| 
 | ||
| 			//************************************************************
 | ||
| 			//** Loop through all series and fill array of points
 | ||
| 			//************************************************************
 | ||
| 			int	seriesIndx = 0;
 | ||
| 			foreach( object seriesName in seriesNamesList )
 | ||
| 			{
 | ||
| 				// Get series object
 | ||
| 				Series	ser = this.Common.DataManager.Series[(string)seriesName];
 | ||
| 
 | ||
| 
 | ||
| 				// Check if stacked groups present
 | ||
| 				if(chartType.SupportStackedGroups && 
 | ||
| 					this._stackGroupNames != null)
 | ||
| 				{
 | ||
| 					// Get index of the series using stacked group
 | ||
| 					string groupName = string.Empty;
 | ||
| 					seriesIndx = this.GetSeriesStackGroupIndex(ser, ref groupName);
 | ||
| 
 | ||
| 					// Set current group name
 | ||
|                     StackedColumnChart stackedColumnChart = chartType as StackedColumnChart;
 | ||
|                     if (stackedColumnChart != null)
 | ||
| 					{
 | ||
|                         stackedColumnChart.currentStackGroup = groupName;
 | ||
| 					}
 | ||
| 					else
 | ||
| 					{
 | ||
|                         StackedBarChart stackedBarChart = chartType as StackedBarChart;
 | ||
|                         if (stackedBarChart != null)
 | ||
|                         {
 | ||
|                             stackedBarChart.currentStackGroup = groupName;
 | ||
|                         }
 | ||
| 					}
 | ||
| 				}
 | ||
| 
 | ||
| 
 | ||
| 				// Set active vertical/horizontal axis and their max/min values
 | ||
| 				Axis	vAxis = (ser.YAxisType == AxisType.Primary) ? area.AxisY : area.AxisY2;
 | ||
| 				Axis	hAxis = (ser.XAxisType == AxisType.Primary) ? area.AxisX : area.AxisX2;
 | ||
| 
 | ||
| 				// Get points interval:
 | ||
| 				//  - set interval to 1 for indexed series
 | ||
| 				//  - if points are not equaly spaced, the minimum interval between points is selected.
 | ||
| 				//  - if points have same interval bars do not overlap each other.
 | ||
| 				bool	sameInterval = true;
 | ||
| 				double	interval = 1;
 | ||
| 				if(!indexedSeries)
 | ||
| 				{
 | ||
| 					interval = area.GetPointsInterval( seriesNamesList, hAxis.IsLogarithmic, hAxis.logarithmBase, true, out sameInterval );
 | ||
| 				}
 | ||
| 
 | ||
| 				// Get column width
 | ||
| 				double	width = ser.GetPointWidth(area.Common.graph, hAxis, interval, 0.8) / numOfSeries;
 | ||
| 					
 | ||
| 				// Get series depth and Z position
 | ||
| 				float seriesDepth, seriesZPosition;
 | ||
| 				this.GetSeriesZPositionAndDepth(ser, out seriesDepth, out seriesZPosition);
 | ||
| 
 | ||
| 				//************************************************************
 | ||
| 				//** Loop through all points in series
 | ||
| 				//************************************************************
 | ||
| 				int	index = 0;
 | ||
| 				foreach( DataPoint point in ser.Points )
 | ||
| 				{
 | ||
| 					// Increase point index
 | ||
| 					index++;
 | ||
| 						
 | ||
| 					// Set x position
 | ||
| 					double	xCenterVal;
 | ||
| 					double	xPosition;
 | ||
| 					if( indexedSeries )
 | ||
| 					{
 | ||
| 						// The formula for position is based on a distance 
 | ||
| 						//from the grid line or nPoints position.
 | ||
| 						xPosition = hAxis.GetPosition( (double)index ) - width * ((double) numOfSeries) / 2.0 + width/2 + seriesIndx * width;
 | ||
| 						xCenterVal = hAxis.GetPosition( (double)index );
 | ||
| 
 | ||
| 					}
 | ||
| 					else if( sameInterval )
 | ||
| 					{
 | ||
| 						xPosition = hAxis.GetPosition( point.XValue ) - width * ((double) numOfSeries) / 2.0 + width/2 + seriesIndx * width;
 | ||
| 						xCenterVal = hAxis.GetPosition( point.XValue );
 | ||
| 					}
 | ||
| 					else
 | ||
| 					{
 | ||
| 						xPosition = hAxis.GetPosition( point.XValue );
 | ||
| 						xCenterVal = hAxis.GetPosition( point.XValue );
 | ||
| 					}
 | ||
| 
 | ||
| 	
 | ||
| 					//************************************************************
 | ||
| 					//** Create and add new DataPoint3D object
 | ||
| 					//************************************************************
 | ||
| 					DataPoint3D pointEx = new DataPoint3D();
 | ||
| 					pointEx.indexedSeries = indexedSeries;
 | ||
| 					pointEx.dataPoint = point;
 | ||
| 					pointEx.index = index;
 | ||
| 					pointEx.xPosition = xPosition;
 | ||
| 					pointEx.xCenterVal = xCenterVal;
 | ||
| 					pointEx.width = ser.GetPointWidth(area.Common.graph, hAxis, interval, 0.8) / numOfSeries;
 | ||
| 					pointEx.depth = seriesDepth;
 | ||
| 					pointEx.zPosition = seriesZPosition;
 | ||
| 
 | ||
| 					// Set Y value and height
 | ||
|                     double yValue = chartType.GetYValue(Common, area, ser, point, index - 1, mainYValueIndex);
 | ||
|                     if (point.IsEmpty && Double.IsNaN(yValue))
 | ||
|                     {
 | ||
|                         yValue = 0.0;
 | ||
|                     }
 | ||
| 					pointEx.yPosition = vAxis.GetPosition(yValue);
 | ||
| 					pointEx.height = vAxis.GetPosition(yValue - chartType.GetYValue(Common, area, ser, point, index - 1, -1));
 | ||
| 
 | ||
| 
 | ||
| 					pointsList.Add(pointEx);
 | ||
| 				}
 | ||
| 
 | ||
| 				// Data series index
 | ||
| 				if(numOfSeries > 1 && sideBySide)
 | ||
| 				{
 | ||
| 					seriesIndx++;
 | ||
| 				}
 | ||
| 			}
 | ||
| 		
 | ||
| 			//************************************************************
 | ||
| 			//** Sort points in drawing order
 | ||
| 			//************************************************************
 | ||
| 			if(comparer == null)
 | ||
| 			{
 | ||
| 				comparer = new PointsDrawingOrderComparer((ChartArea)this, selection, coord);
 | ||
| 			}
 | ||
| 			pointsList.Sort(comparer);
 | ||
| 
 | ||
| 			return pointsList;
 | ||
| 		}
 | ||
| 		
 | ||
| 		#endregion	
 | ||
| 
 | ||
| 		#region Points drawing order comparer class
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Used to compare points in array and sort them by drawing order.
 | ||
| 		/// </summary>
 | ||
| 		internal class PointsDrawingOrderComparer : IComparer
 | ||
| 		{
 | ||
| 			/// <summary>
 | ||
| 			/// Chart area object reference.
 | ||
| 			/// </summary>
 | ||
| 			private	ChartArea	_area = null;
 | ||
| 
 | ||
| 			/// <summary>
 | ||
| 			/// Area X position where visible sides are switched.
 | ||
| 			/// </summary>
 | ||
| 			private	Point3D		_areaProjectionCenter = new Point3D(float.NaN, float.NaN, float.NaN);
 | ||
| 
 | ||
| 			/// <summary>
 | ||
| 			/// Selection mode. Points order should be reversed.
 | ||
| 			/// </summary>
 | ||
| 			private bool		_selection = false;
 | ||
| 
 | ||
|             /// <summary>
 | ||
|             /// Public constructor.
 | ||
|             /// </summary>
 | ||
|             /// <param name="area">Chart area.</param>
 | ||
|             /// <param name="selection">Selection indicator.</param>
 | ||
|             /// <param name="coord">Which coordinate of COP (X, Y or Z) to test for surface overlapping</param>
 | ||
| 			public PointsDrawingOrderComparer(ChartArea	area, bool selection, COPCoordinates coord)
 | ||
| 			{
 | ||
| 				this._area = area;
 | ||
| 				this._selection = selection;
 | ||
| 
 | ||
| 				// Get center of projection
 | ||
| 				if(area.DrawPointsToCenter(ref coord))
 | ||
| 				{
 | ||
| 					_areaProjectionCenter = area.GetCenterOfProjection(coord);
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
|             /// <summary>
 | ||
|             /// Comparer method.
 | ||
|             /// </summary>
 | ||
|             /// <param name="o1">First object.</param>
 | ||
|             /// <param name="o2">Second object.</param>
 | ||
|             /// <returns>Comparison result.</returns>
 | ||
| 			public int Compare(object o1, object o2)
 | ||
| 			{
 | ||
| 				DataPoint3D point1 = (DataPoint3D) o1;
 | ||
| 				DataPoint3D point2 = (DataPoint3D) o2;
 | ||
| 
 | ||
| 				int	result = 0;
 | ||
| 				if(point1.xPosition < point2.xPosition)
 | ||
| 				{
 | ||
| 					result = -1;
 | ||
| 				}
 | ||
| 				else if(point1.xPosition > point2.xPosition)
 | ||
| 				{
 | ||
| 					result = 1;
 | ||
| 				}
 | ||
| 				else
 | ||
| 				{
 | ||
| 					
 | ||
| 					// If X coordinate is the same - filter by Y coordinate
 | ||
| 					if(point1.yPosition < point2.yPosition)
 | ||
| 					{
 | ||
| 						result = 1;
 | ||
| 					}
 | ||
| 					else if(point1.yPosition > point2.yPosition)
 | ||
| 					{
 | ||
| 						result = -1;
 | ||
| 					}
 | ||
| 
 | ||
| 					// Order points from sides to center
 | ||
| 					if(!float.IsNaN(_areaProjectionCenter.Y))
 | ||
| 					{
 | ||
| 						double yMin1 = Math.Min(point1.yPosition, point1.height);
 | ||
| 						double yMax1 = Math.Max(point1.yPosition, point1.height);
 | ||
| 						double yMin2 = Math.Min(point2.yPosition, point2.height);
 | ||
| 						double yMax2 = Math.Max(point2.yPosition, point2.height);
 | ||
| 
 | ||
| 						if(_area.IsBottomSceneWallVisible())
 | ||
| 						{
 | ||
| 							if( yMin1 <= _areaProjectionCenter.Y && yMin2 <= _areaProjectionCenter.Y )
 | ||
| 							{
 | ||
| 								result *= -1;
 | ||
| 							}
 | ||
| 							else if( yMin1 <= _areaProjectionCenter.Y)
 | ||
| 							{
 | ||
| 								result = 1;
 | ||
| 							}
 | ||
| 
 | ||
| 						}
 | ||
| 						else
 | ||
| 						{
 | ||
| 							
 | ||
| 							if( yMax1 >= _areaProjectionCenter.Y && yMax2 >= _areaProjectionCenter.Y )
 | ||
| 							{
 | ||
| 								result *= 1;
 | ||
| 							}
 | ||
| 							else if( yMax1 >= _areaProjectionCenter.Y)
 | ||
| 							{
 | ||
| 								result = 1;
 | ||
| 							}
 | ||
| 							else
 | ||
| 							{
 | ||
| 								result *= -1;
 | ||
| 							}
 | ||
| 						}
 | ||
| 					}
 | ||
| 
 | ||
| 						// Reversed order if looking from the bottom
 | ||
| 					else if(!_area.IsBottomSceneWallVisible())
 | ||
| 					{
 | ||
| 						result *= -1;
 | ||
| 					}
 | ||
| 			
 | ||
| 				}
 | ||
| 
 | ||
| 				if(point1.xPosition != point2.xPosition)
 | ||
| 				{
 | ||
| 					// Order points from sides to center
 | ||
| 					if (!float.IsNaN(_areaProjectionCenter.X))
 | ||
| 					{
 | ||
|                         if ((point1.xPosition + point1.width / 2f) >= _areaProjectionCenter.X &&
 | ||
|                             (point2.xPosition + point2.width / 2f) >= _areaProjectionCenter.X)
 | ||
| 						{
 | ||
| 							result *= -1;
 | ||
| 						}
 | ||
| 					}
 | ||
| 
 | ||
| 					// Reversed order of points by X value
 | ||
|                     else if (_area.DrawPointsInReverseOrder())
 | ||
| 					{
 | ||
| 						result *= -1;
 | ||
| 					}
 | ||
| 				}
 | ||
| 
 | ||
|                 return (_selection) ? -result : result;
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| #endregion
 | ||
| 
 | ||
| 		#region Center of Projection calculation methods
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Returns one or many (X, Y, Z) coordinates of the center of projection.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="coord">Defines coordinates to return.</param>
 | ||
| 		/// <returns>Center of projection. Value can be set to float.NaN if not defined or outside plotting area.</returns>
 | ||
| 		internal Point3D GetCenterOfProjection(COPCoordinates coord)
 | ||
| 		{
 | ||
| 			// Define 2 points in the opposite corners of the plotting area
 | ||
| 			Point3D[]	points = new Point3D[2];
 | ||
| 			points[0] = new Point3D(this.PlotAreaPosition.X, this.PlotAreaPosition.Bottom, 0f);
 | ||
| 			points[1] = new Point3D(this.PlotAreaPosition.Right, this.PlotAreaPosition.Y, this.areaSceneDepth);
 | ||
| 
 | ||
| 			// Check if surfaces (points 1 & 2) has same orientation
 | ||
| 			bool	xSameOrientation, ySameOrientation, zSameOrientation;
 | ||
| 			CheckSurfaceOrientation(
 | ||
| 				coord,
 | ||
| 				points[0],
 | ||
| 				points[1],
 | ||
| 				out xSameOrientation, 
 | ||
| 				out ySameOrientation, 
 | ||
| 				out zSameOrientation);
 | ||
| 
 | ||
| 			// If orientation of all surfaces is the same - no futher processing is required (COP is outside of plotting area)
 | ||
| 			Point3D		resultPoint = new Point3D(
 | ||
| 				(xSameOrientation) ? float.NaN : 0f,
 | ||
| 				(ySameOrientation) ? float.NaN : 0f,
 | ||
| 				(zSameOrientation) ? float.NaN : 0f);
 | ||
| 			if( ( ((coord & COPCoordinates.X) != COPCoordinates.X) || xSameOrientation ) &&
 | ||
| 				( ((coord & COPCoordinates.Y) != COPCoordinates.Y) || ySameOrientation ) &&
 | ||
| 				( ((coord & COPCoordinates.Z) != COPCoordinates.Z) || zSameOrientation ) )
 | ||
| 			{
 | ||
| 				return resultPoint;
 | ||
| 			}
 | ||
| 
 | ||
| 			// Calculate the smallest interval (0.5 pixels) in relative coordinates
 | ||
| 			SizeF	interval = new SizeF(0.5f, 0.5f);
 | ||
| #if WINFORMS_CONTROL
 | ||
| 			interval.Width = interval.Width * 100F / ((float)(this.Common.Chart.Width - 1)); 
 | ||
| 			interval.Height = interval.Height * 100F / ((float)(this.Common.Chart.Height - 1)); 
 | ||
| #else
 | ||
| 			interval.Width = interval.Width * 100F / ((float)(this.Common.Chart.Width.Value - 1)); 
 | ||
| 			interval.Height = interval.Height * 100F / ((float)(this.Common.Chart.Height.Value - 1)); 
 | ||
| #endif	//#if WINFORMS_CONTROL
 | ||
| 
 | ||
| 			// Find middle point and check it's surface orientation
 | ||
| 			bool	doneFlag = false;
 | ||
| 			while(!doneFlag)
 | ||
| 			{
 | ||
| 				// Find middle point
 | ||
| 				Point3D	middlePoint = new Point3D( 
 | ||
| 					(points[0].X + points[1].X) / 2f,
 | ||
| 					(points[0].Y + points[1].Y) / 2f,
 | ||
| 					(points[0].Z + points[1].Z) / 2f);
 | ||
| 
 | ||
| 				// Check if surfaces (points 1 & middle) has same orientation
 | ||
| 				CheckSurfaceOrientation(
 | ||
| 					coord,
 | ||
| 					points[0],
 | ||
| 					middlePoint,
 | ||
| 					out xSameOrientation, 
 | ||
| 					out ySameOrientation, 
 | ||
| 					out zSameOrientation);
 | ||
| 
 | ||
| 				// Calculate points 1 & 2 depending on surface orientation of the middle point
 | ||
| 				points[(xSameOrientation) ? 0 : 1].X = middlePoint.X;
 | ||
| 				points[(ySameOrientation) ? 0 : 1].Y = middlePoint.Y;
 | ||
| 				points[(zSameOrientation) ? 0 : 1].Z = middlePoint.Z;
 | ||
| 
 | ||
| 				// Check if no more calculations required
 | ||
| 				doneFlag = true;
 | ||
| 				if( (coord & COPCoordinates.X) == COPCoordinates.X && 
 | ||
| 					Math.Abs(points[1].X - points[0].X) >= interval.Width)
 | ||
| 				{
 | ||
| 					doneFlag = false;
 | ||
| 				}
 | ||
| 				if( (coord & COPCoordinates.Y) == COPCoordinates.Y && 
 | ||
| 					Math.Abs(points[1].Y - points[0].Y) >= interval.Height)
 | ||
| 				{
 | ||
| 					doneFlag = false;
 | ||
| 				}
 | ||
| 				if( (coord & COPCoordinates.Z) == COPCoordinates.Z && 
 | ||
| 					Math.Abs(points[1].Z - points[0].Z) >= interval.Width)
 | ||
| 				{
 | ||
| 					doneFlag = false;
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			// Calculate result point
 | ||
| 			if(!float.IsNaN(resultPoint.X))
 | ||
| 				resultPoint.X = (points[0].X + points[1].X) / 2f;
 | ||
| 			if(!float.IsNaN(resultPoint.Y))
 | ||
| 				resultPoint.Y = (points[0].Y + points[1].Y) / 2f;
 | ||
| 			if(!float.IsNaN(resultPoint.Z))
 | ||
| 				resultPoint.Z = (points[0].Z + points[1].Z) / 2f;
 | ||
| 			return resultPoint;
 | ||
| 		}
 | ||
| 
 | ||
| 		/// <summary>
 | ||
| 		/// Checks orientations of two normal surfaces for each coordinate X, Y and Z.
 | ||
| 		/// </summary>
 | ||
| 		/// <param name="coord">Defines coordinates to return.</param>
 | ||
| 		/// <param name="point1">First point.</param>
 | ||
| 		/// <param name="point2">Second point.</param>
 | ||
| 		/// <param name="xSameOrientation">X surfaces orientation is the same.</param>
 | ||
| 		/// <param name="ySameOrientation">Y surfaces orientation is the same.</param>
 | ||
| 		/// <param name="zSameOrientation">Z surfaces orientation is the same.</param>
 | ||
| 		private void CheckSurfaceOrientation(
 | ||
| 			COPCoordinates coord,
 | ||
| 			Point3D point1, 
 | ||
| 			Point3D point2, 
 | ||
| 			out bool xSameOrientation, 
 | ||
| 			out bool ySameOrientation, 
 | ||
| 			out bool zSameOrientation)
 | ||
| 		{
 | ||
| 			Point3D[]	pointsSurface = new Point3D[3];
 | ||
| 			bool		surf1, surf2;
 | ||
| 
 | ||
| 			// Initialize returned values
 | ||
| 			xSameOrientation = true;
 | ||
| 			ySameOrientation = true;
 | ||
| 			zSameOrientation = true;
 | ||
| 
 | ||
| 			// Check X axis coordinates (ledt & right surfaces)
 | ||
| 			if( (coord & COPCoordinates.X) == COPCoordinates.X )
 | ||
| 			{
 | ||
| 				// Define Left surface coordinates, transform them and check visibility
 | ||
| 				pointsSurface[0] = new Point3D(point1.X, this.PlotAreaPosition.Y, 0f);
 | ||
| 				pointsSurface[1] = new Point3D(point1.X, this.PlotAreaPosition.Bottom, 0f);
 | ||
| 				pointsSurface[2] = new Point3D(point1.X, this.PlotAreaPosition.Bottom, this.areaSceneDepth);
 | ||
| 				this.matrix3D.TransformPoints( pointsSurface );
 | ||
| 				surf1 = ChartGraphics.IsSurfaceVisible(pointsSurface[0], pointsSurface[1], pointsSurface[2]);
 | ||
| 
 | ||
| 				// Define Right surface coordinates, transform them and check visibility
 | ||
| 				pointsSurface[0] = new Point3D(point2.X, this.PlotAreaPosition.Y, 0f);
 | ||
| 				pointsSurface[1] = new Point3D(point2.X, this.PlotAreaPosition.Bottom, 0f);
 | ||
| 				pointsSurface[2] = new Point3D(point2.X, this.PlotAreaPosition.Bottom, this.areaSceneDepth);
 | ||
| 				this.matrix3D.TransformPoints( pointsSurface );
 | ||
| 				surf2 = ChartGraphics.IsSurfaceVisible(pointsSurface[0], pointsSurface[1], pointsSurface[2]);
 | ||
| 
 | ||
| 				// Check if surfaces have same visibility
 | ||
| 				xSameOrientation = (surf1 == surf2);
 | ||
| 			}
 | ||
| 
 | ||
| 			// Check Y axis coordinates (top & bottom surfaces)
 | ||
| 			if( (coord & COPCoordinates.Y) == COPCoordinates.Y )
 | ||
| 			{
 | ||
| 				// Define Bottom surface coordinates, transform them and check visibility
 | ||
| 				pointsSurface[0] = new Point3D(this.PlotAreaPosition.X, point1.Y, this.areaSceneDepth);
 | ||
| 				pointsSurface[1] = new Point3D(this.PlotAreaPosition.X, point1.Y, 0f);
 | ||
| 				pointsSurface[2] = new Point3D(this.PlotAreaPosition.Right, point1.Y, 0f);
 | ||
| 				this.matrix3D.TransformPoints( pointsSurface );
 | ||
| 				surf1 = ChartGraphics.IsSurfaceVisible(pointsSurface[0], pointsSurface[1], pointsSurface[2]);
 | ||
| 
 | ||
| 				// Define Top surface coordinates, transform them and check visibility
 | ||
| 				pointsSurface[0] = new Point3D(this.PlotAreaPosition.X, point2.Y, this.areaSceneDepth);
 | ||
| 				pointsSurface[1] = new Point3D(this.PlotAreaPosition.X, point2.Y, 0f);
 | ||
| 				pointsSurface[2] = new Point3D(this.PlotAreaPosition.Right, point2.Y, 0f);
 | ||
| 				this.matrix3D.TransformPoints( pointsSurface );
 | ||
| 				surf2 = ChartGraphics.IsSurfaceVisible(pointsSurface[0], pointsSurface[1], pointsSurface[2]);
 | ||
| 
 | ||
| 				// Check if surfaces have same visibility
 | ||
| 				ySameOrientation = (surf1 == surf2);
 | ||
| 			}
 | ||
| 
 | ||
| 			// Check Y axis coordinates (front & back surfaces)
 | ||
| 			if( (coord & COPCoordinates.Z) == COPCoordinates.Z )
 | ||
| 			{
 | ||
| 				// Define Front surface coordinates, transform them and check visibility
 | ||
| 				pointsSurface[0] = new Point3D(this.PlotAreaPosition.X, this.PlotAreaPosition.Y, point1.Z);
 | ||
| 				pointsSurface[1] = new Point3D(this.PlotAreaPosition.X, this.PlotAreaPosition.Bottom, point1.Z);
 | ||
| 				pointsSurface[2] = new Point3D(this.PlotAreaPosition.Right, this.PlotAreaPosition.Bottom, point1.Z);
 | ||
| 				this.matrix3D.TransformPoints( pointsSurface );
 | ||
| 				surf1 = ChartGraphics.IsSurfaceVisible(pointsSurface[0], pointsSurface[1], pointsSurface[2]);
 | ||
| 
 | ||
| 				// Define Back surface coordinates, transform them and check visibility
 | ||
| 				pointsSurface[0] = new Point3D(this.PlotAreaPosition.X, this.PlotAreaPosition.Y, point2.Z);
 | ||
| 				pointsSurface[1] = new Point3D(this.PlotAreaPosition.X, this.PlotAreaPosition.Bottom, point2.Z);
 | ||
| 				pointsSurface[2] = new Point3D(this.PlotAreaPosition.Right, this.PlotAreaPosition.Bottom, point2.Z);
 | ||
| 				this.matrix3D.TransformPoints( pointsSurface );
 | ||
| 				surf2 = ChartGraphics.IsSurfaceVisible(pointsSurface[0], pointsSurface[1], pointsSurface[2]);
 | ||
| 
 | ||
| 				// Check if surfaces have same visibility
 | ||
| 				zSameOrientation = (surf1 == surf2);
 | ||
| 			}
 | ||
| 		}
 | ||
| #endregion
 | ||
| 	}
 | ||
| }
 |