aboutsummaryrefslogtreecommitdiffstats
path: root/Software/Visual_Studio/SideChains/RealTimeGraphX/Renderers/GraphEraseRenderer.cs
blob: ae8aeb5e3775c71865ddef4698a1011ef2460d45 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace RealTimeGraphX.Renderers
{
    /// <summary>
    /// Represents an heart beat monitor style graph renderer.
    /// </summary>
    /// <typeparam name="TDataSeries">The type of the graph data series.</typeparam>
    /// <typeparam name="XDataPoint">The type of the x-axis data point.</typeparam>
    /// <typeparam name="YDataPoint">The type of the y-axis data point.</typeparam>
    /// <seealso cref="RealTimeGraphX.GraphRendererBase{TDataSeries, XDataPoint, YDataPoint}" />
    public class GraphEraseRenderer<TDataSeries, XDataPoint, YDataPoint> : GraphRendererBase<TDataSeries, XDataPoint, YDataPoint> where XDataPoint : GraphDataPointBase where YDataPoint : GraphDataPointBase where TDataSeries : class, IGraphDataSeries
    {
        private Dictionary<TDataSeries, int> _series_replace_index_dictionary = new Dictionary<TDataSeries, int>();

        /// <summary>
        /// Returns an array of absolute graph data points to render.
        /// This method is called per data series.
        /// </summary>
        /// <param name="seriesCollection">A collection of all data series that is currently in the rendering pass.</param>
        /// <param name="series">The current data series to render.</param>
        /// <param name="toRender">Pending data series object to render.</param>
        /// <returns></returns>
        protected override List<GraphPoint> OnRender(IEnumerable<PendingSeries> seriesCollection, TDataSeries series, PendingSeries toRender)
        {
            int replace_index = 0;
            int new_items = toRender.NewItemsCount;

            if (_series_replace_index_dictionary.ContainsKey(series))
            {
                replace_index = _series_replace_index_dictionary[series];
            }
            else
            {
                _series_replace_index_dictionary.Add(series, 0);
            }

            XDataPoint min_x = toRender.XX.Min();
            XDataPoint max_x = toRender.XX.Max();

            YDataPoint min_y = Input.Range.MinimumY;
            YDataPoint max_y = Input.Range.MaximumY;

            if (Input.Range.AutoY)
            {
                min_y = seriesCollection.SelectMany(x => x.YY).Min();
                max_y = seriesCollection.SelectMany(x => x.YY).Max();
            }

            var min_x_erase = Input.Range.MaximumX > max_x ? min_x : max_x - Input.Range.MaximumX;

            OnEffectiveRangeXChanged(min_x_erase, max_x);
            OnEffectiveRangeYChanged(min_y, max_y);

            if (max_x - min_x > Input.Range.MaximumX)
            {
                for (int i = 0; i < new_items; i++)
                {
                    int value_index = toRender.XX.Count - new_items + i;

                    toRender.YY[replace_index++] = toRender.YY[value_index];
                    toRender.XX.RemoveAt(value_index);
                    toRender.YY.RemoveAt(value_index);

                    if (replace_index >= toRender.XX.Count)
                    {
                        replace_index = 0;

                        if (series == _series_replace_index_dictionary.Last().Key)
                        {
                            foreach (var s in _series_replace_index_dictionary.Keys.ToList())
                            {
                                _series_replace_index_dictionary[s] = replace_index;
                            }
                        }
                    }
                }
            }

            min_x = toRender.XX.Min();
            max_x = toRender.XX.Max();

            var dxList = toRender.XX.Select(x => x.ComputeRelativePosition(min_x, max_x)).ToList();
            var dyList = toRender.YY.Select(x => x.ComputeRelativePosition(min_y, max_y)).ToList();

            List<GraphPoint> points = new List<GraphPoint>();

            for (int i = 0; i < dxList.Count; i++)
            {
                float image_x = ConvertXValueToRendererValue(dxList[i]);
                float image_y = ConvertYValueToRendererValue(dyList[i]);

                GraphPoint point = new GraphPoint(image_x, image_y);
                points.Add(point);
            }

            CurrentXPosition = points[replace_index].X;

            _series_replace_index_dictionary[series] = replace_index;

            return points;
        }
    }
}