示例#1
0
 /// <summary>
 /// Sets standard value for Tube.AlgorithmOption
 /// </summary>
 public Tube(TubeSize size)
 {
     this.size = size;
     algorithmOption = AlgorithmOptions.Rectangle;
     calculateErrors = true;
     tubeSuccessful = false;
 }
示例#2
0
        /// <summary>
        /// Calculates a lower and an upper tube curve.
        /// </summary>
        /// <param name="reference">Reference curve with x and y values.</param>
        /// <param name="size">Size of tube.</param>
        /// <returns>Collection of return values.</returns>
        public override TubeReport Calculate(Curve reference, TubeSize size)
        {
            TubeReport report = new TubeReport();
            successful = false;

            if (reference != null && size != null)
            {
                report.Reference = reference;
                report.Size = size;
                report.Algorithm = AlgorithmOptions.Rectangle;
                report.Lower = CalculateLower(reference, size);
                report.Upper = CalculateUpper(reference, size);
                if (report.Lower.ImportSuccessful && report.Upper.ImportSuccessful)
                    successful = true;
            }
            return report;
        }
示例#3
0
        /// <summary>
        /// Calculates a lower and an upper tube curve.
        /// </summary>
        /// <param name="reference">Reference curve with x and y values.</param>
        /// <param name="size">Size of tube.</param>
        /// <returns>Collection of return values.</returns>
        public override TubeReport Calculate(Curve reference, TubeSize size)
        {
            TubeReport report = new TubeReport();

            successful = false;

            if (reference != null && size != null)
            {
                report.Reference = reference;
                report.Size      = size;
                report.Algorithm = AlgorithmOptions.Rectangle;
                report.Lower     = CalculateLower(reference, size);
                report.Upper     = CalculateUpper(reference, size);
                if (report.Lower.ImportSuccessful && report.Upper.ImportSuccessful)
                {
                    successful = true;
                }
            }
            return(report);
        }
示例#4
0
        /// <summary>
        /// Calculates the lower tube curve.
        /// </summary>
        /// <param name="reference">Reference curve.</param>
        /// <param name="size">Tube size.</param>
        /// <returns>Lower tube curve.</returns>
        private static Curve CalculateLower(Curve reference, TubeSize size)
        {
            Curve lower;                                            // lower tube curve
            List<double> LX = new List<double>(2 * reference.Count); // x-values of lower tube curve
            List<double> LY = new List<double>(2 * reference.Count); // y-values of lower tube curve

            // -------------------------------------------------------------------------------------------------------------
            // ---------------------------------- 1. Add corner points of the rectangle  -----------------------------------
            // -------------------------------------------------------------------------------------------------------------

            double m0, m1;                                           // slopes before and after point i of reference curve
            double s0, s1;                                           // signum of slopes of reference curve: 1 - increasing, 0 - constant, -1 - decreasing

            // -------------------------------------------------------------------------------------------------------------
            // 1.1. Start: Rectangle with center (x,y) = (reference.X[0], reference.Y[0])
            // -------------------------------------------------------------------------------------------------------------

            // slopes of reference curve (initialization)
            s0 = Math.Sign(reference.Y[1] - reference.Y[0]);
            if (reference.X[1] != reference.X[0])
                m0 = (reference.Y[1] - reference.Y[0]) / (reference.X[1] - reference.X[0]);
            else
                if (s0 > 0)
                    m0 = Double.PositiveInfinity;
                else
                    m0 = Double.NegativeInfinity;

            // add point down left
            LX.Add(reference.X[0] - size.X);
            LY.Add(reference.Y[0] - size.Y);

            if (s0 == 1)
            {
                // add point down right
                LX.Add(reference.X[0] + size.X);
                LY.Add(reference.Y[0] - size.Y);
            }

            // -------------------------------------------------------------------------------------------------------------
            // 1.2. Iteration: Rectangle with center (x,y) = (reference.X[i], reference.Y[i])
            // -------------------------------------------------------------------------------------------------------------
            for (int i = 1; i < reference.Count - 1; i++)
            {
                if (reference.X[i] == reference.X[i + 1] && reference.Y[i] == reference.Y[i + 1]) // ignore identical points
                    continue;

                // slopes of reference curve
                s1 = Math.Sign(reference.Y[i + 1] - reference.Y[i]);
                if (reference.X[i + 1] != reference.X[i])
                    m1 = (reference.Y[i + 1] - reference.Y[i]) / (reference.X[i + 1] - reference.X[i]);
                else
                    if (s1 > 0)
                        m1 = Double.PositiveInfinity;
                    else
                        m1 = Double.NegativeInfinity;

                // add no point for equal slopes of reference curve
                if (!(m0 == m1))
                {
                    if ((s0 != -1) && (s1 != -1))
                    {
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 != 1) && (s1 != 1))
                    {
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 == -1) && (s1 == 1))
                    {
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 == 1) && (s1 == -1))
                    {
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    // remove point for equal slopes equal 0 of tube curve
                    int last = LY.Count - 1;
                    if (reference.Y[i + 1] - size.Y == LY[last] && LY[last - 1] == LY[last])
                    {
                        LX.RemoveAt(last);
                        LY.RemoveAt(last);
                    }
                }
                s0 = s1;
                m0 = m1;
            }
            // -------------------------------------------------------------------------------------------------------------
            // 1.3. End: Rectangle with center (x,y) = (reference.X[reference.Count - 1], reference.Y[reference.Count - 1])
            // -------------------------------------------------------------------------------------------------------------
            if (s0 == -1)
            {
                // add point down left
                LX.Add(reference.X[reference.Count - 1] - size.X);
                LY.Add(reference.Y[reference.Count - 1] - size.Y);
            }
            // add point down right
            LX.Add(reference.X[reference.Count - 1] + size.X);
            LY.Add(reference.Y[reference.Count - 1] - size.Y);

            // -------------------------------------------------------------------------------------------------------------
            // -------------- 2. Remove points and add intersection points in case of backward order -----------------------
            // -------------------------------------------------------------------------------------------------------------

            removeLoop(LX, LY, true);

            lower = new Curve("Lower", LX.ToArray(), LY.ToArray());
            return lower;
        }
示例#5
0
        public Report CompareFiles(Log log, CsvFile csvBase, string sReportPath, ref Options options)
        {
            int iInvalids = 0;

            Report rep = new Report(sReportPath);
            log.WriteLine("Comparing \"{0}\" to \"{1}\"", _fileName, csvBase.ToString());

            rep.BaseFile = csvBase.ToString();
            rep.CompareFile = _fileName;

            Curve reference = new Curve();
            Curve compareCurve = new Curve();
            TubeReport tubeReport = new TubeReport();
            TubeSize size = null;
            Tube tube = new Tube(size);
            IOptions tubeOptions = new Options1(_dRangeDelta, Axes.X);

            foreach (KeyValuePair<string, List<double>> res in csvBase.Results)
            {
                if (!this.Results.ContainsKey(res.Key))
                    log.WriteLine(LogLevel.Warning, "{0} not found in \"{1}\", skipping checks.", res.Key, this._fileName);
                else
                {
                    compareCurve = new Curve(res.Key, this.XAxis.ToArray<double>(), this.Results[res.Key].ToArray<double>());

                    if (res.Value.Count == 0)
                    {
                        log.Error("{0} has no y-Values! Maybe error during parsing? Skipping", res.Key);
                        continue;
                    }
                    reference = new Curve("Reference ", csvBase.XAxis.ToArray(), csvBase.Results[res.Key].ToArray());
                    if (!reference.ImportSuccessful)
                    {
                        log.Error("Error in the calculation of the tubes. Skipping {0}", res.Key);
                        rep.Chart.Add(new Chart() { Title = res.Key, Errors = 1 });
                        continue;
                    }

                    if (reference.X.Length < compareCurve.X.Length)
                        log.WriteLine(LogLevel.Warning, "The resolution of the base x-axis is smaller than the compare x-axis. The better the base resolution is, the better the validation result will be!");
                    else
                        log.WriteLine(LogLevel.Debug, "The resolution of the base x-axis is good.");

                    size = new TubeSize(reference, true);
                    size.Calculate(_dRangeDelta, Axes.X, Relativity.Relative);
                    tube = new Tube(size);
                    tubeReport = tube.Calculate(reference);
                    tube.Validate(compareCurve);

                    if (tubeReport.Valid == Validity.Valid)
                        log.WriteLine(res.Key + " is valid");
                    else
                    {
                        log.WriteLine(LogLevel.Warning, "{0} is invalid! {1} errors have been found during validation.", res.Key,
                            (null != tube.Report.Errors && null != tube.Report.Errors.X) ? tube.Report.Errors.X.Length : 0);
                        iInvalids++;
                        Environment.ExitCode = 1;
                    }
                }
                if (null != tube.Report)//No charts for missing reports
                    PrepareCharts(reference, compareCurve, tube.Report.Errors, rep, tubeReport, res, options.UseBitmapPlots);
            }
            rep.Tolerance = _dRangeDelta;

            string sResult = "na";

            if (rep.TotalErrors == 0)
                sResult = "passed";
            else
                sResult = "failed";

            if (options.ComparisonFlag)
                using (TextWriter writer = File.CreateText(string.Format("{0}{1}compare_{2}.log", Path.GetDirectoryName(_fileName), Path.DirectorySeparatorChar, sResult)))
                {
                    //Content needs to be defined
                    writer.WriteLine("CSV Compare Version {0} ({1})", Info.AssemblyVersion, Assembly.GetExecutingAssembly().GetName().ProcessorArchitecture);
                    writer.WriteLine("Comparison result file for {0}", _fileName);
                    writer.WriteLine(". Time:        {0:o}", DateTime.Now);
                    writer.WriteLine(". Operation:   {0}", options.Mode);
                    writer.WriteLine(". Tolerance:   {0}", options.Tolerance);
                    writer.WriteLine(". Result:      {0}", sResult);

                    if (rep.TotalErrors > 0)
                    {
                        Chart pairMax = rep.Chart.Aggregate((l, r) => l.DeltaError > r.DeltaError ? l : r);
                        writer.WriteLine(". Biggest error: {0}=>{1}", pairMax.Title, pairMax.DeltaError);
                        writer.WriteLine(". Failed values:");

                        foreach (Chart c in (from r in rep.Chart where r.DeltaError > 0 select r).OrderByDescending(er => er.DeltaError))
                            writer.WriteLine("{0}=>{1}", c.Title, c.DeltaError);
                    }
                }

            rep.WriteReport(log, (!string.IsNullOrEmpty(options.ReportDir)) ? options.ReportDir : string.Empty, options);
            GC.Collect();//immediately forget big charts and data
            return rep;
        }
示例#6
0
        /// <summary>
        /// Calculates a lower and an upper tube curve.
        /// </summary>
        /// <param name="reference">Reference curve with x and y values..</param>
        /// <param name="size">Size of tube.</param>
        /// <returns>Collection of return values.</returns>
        public override TubeReport Calculate(Curve reference, TubeSize size)
        {
            /*
             * ///This method generates tubes around a given curve.
             * /// @param x The time values for the base curve
             * /// @param y The y values of the base curve
             * /// @param x_low An empty, initialized list that is to be filled with the time values for the lower tube curve
             * /// @param y_low An empty, initialized list that is to be filled with the y values for the lower tube curve
             * /// @param x_high An empty, initialized list that is to be filled with the time values for the upper tube curve
             * /// @param y_high An empty, initialized list that is to be filled with the y values for the upper tube curve
             * /// @param r relative tolerance argument
             * /// @param bAllIntervals used all intervals
             * */
            double[] x = reference.X.ToArray();
            double[] y = reference.Y.ToArray();
            _dDelta = size.X;
            List <double> xHigh = new List <double>();
            List <double> yHigh = new List <double>();
            List <double> xLow  = new List <double>();
            List <double> yLow  = new List <double>();

            TubeReport report = new TubeReport();

            successful = false;

            if (reference != null && size != null)
            {
                report.Reference = reference;
                report.Size      = size;
                report.Algorithm = AlgorithmOptions.Ellipse;

                /// SBR: Wir gehen erstmal prinzipiell davon aus das der Algorithmus korrekt verläuft.
                //_bAllIntervals = true;
                // set tStart and tStop
                _dtStart   = x[0];
                _dtStop    = x[x.Length - 1];
                _dXMinStep = ((_dtStop - _dtStart) + Math.Abs(_dtStart)) * _dXRelEps;

                // Initialize lists (upper tube)
                _lmh  = new ArrayList(x.Length); // slope in i
                _li0h = new ArrayList(x.Length); // i
                _li1h = new ArrayList(x.Length); // i - 1

                // Initialize lists (lower tube)
                _lml  = new ArrayList(x.Length);        // slope in i
                _li0l = new ArrayList(x.Length);        // i
                _li1l = new ArrayList(x.Length);        // i-1

                // calculate the tubes delta
                //_dDelta = r * (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart));

                // calculate S:
                double max = y[0];
                double min = y[0];

                xHigh.Clear();
                yHigh.Clear();
                xLow.Clear();
                yLow.Clear();

                if (xHigh.Capacity < x.Length)
                {
                    xHigh.Capacity = x.Length;
                }
                if (yHigh.Capacity < x.Length)
                {
                    yHigh.Capacity = x.Length;
                }
                if (xLow.Capacity < x.Length)
                {
                    xLow.Capacity = x.Length;
                }
                if (yLow.Capacity < x.Length)
                {
                    yLow.Capacity = x.Length;
                }

                for (int i = 1; i < y.Length; i++)
                {
                    try
                    {
                        double Y = y[i];
                        if (Y > max)
                        {
                            max = Y;
                        }
                        if (Y < min)
                        {
                            min = Y;
                        }
                    }
                    catch (IndexOutOfRangeException) { break; }
                }
                //            _dS = Math.Abs(4 * (max - min) / (Math.Abs(_dtStop - _dtStart)));
                _dS = (Math.Abs(max - min) + Math.Abs(min)) / (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart));

                if (_dS < 0.0004 / (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart)))
                {
                    _dS = 0.0004 / (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart));
                }

                bool bJump = false;

                // Begin calculation for the tubes
                for (int i = 1; i < x.Length; i++)
                {
                    try
                    {
                        // get current value
                        _dX1 = x[i];
                        _dY1 = y[i];

                        // get previous value
                        _dX2 = x[i - 1];
                        _dY2 = y[i - 1];
                    }
                    catch (IndexOutOfRangeException) { break; }

                    // catch jumps
                    bJump = false;
                    if ((_dX1 <= _dX2) && (_dY1 == _dY2) && (xHigh.Count == 0))
                    {
                        continue;
                    }
                    if ((_dX1 <= _dX2) && (_dY1 == _dY2))
                    {
                        _dX1 = Math.Max(x[(int)_li1l[_li1l.Count - 1]] + _dXMinStep, Math.Max(x[(int)_li1h[_li1h.Count - 1]] + _dXMinStep, _dX1));
                        x.SetValue(_dX1, i);
                        _dCurrentSlope = (double)_lmh[_lmh.Count - 1];
                    }
                    else
                    {
                        if (_dX1 <= _dX2)
                        {
                            bJump = true;
                            _dX1  = _dX2 + _dXMinStep;
                            x.SetValue(_dX1, i);
                        }
                        _dCurrentSlope = (_dY1 - _dY2) / (_dX1 - _dX2); // calculate current slope ( 3.2.6.1)
                    }

                    // fill lists with new values: values upper tube
                    _li0h.Add(i);
                    _li1h.Add(i - 1);
                    _lmh.Add(_dCurrentSlope);

                    // fill lists with new values: values lower tube
                    _li0l.Add(i);
                    _li1l.Add(i - 1);
                    if ((_dX1 <= _dX2) && (_dY1 == _dY2))
                    {
                        _dCurrentSlope = (double)_lml[_lml.Count - 1];
                    }
                    _lml.Add(_dCurrentSlope);

                    if (xHigh.Count == 0) // 1st interval (3.2.5)
                    {
                        if (bJump)
                        {
                            // initial values upper tube
                            _li0h[0] = i - 1;
                            _lmh[0]  = 0.0;
                            xHigh.Add(_dX2 - _dDelta - _dXMinStep);
                            yHigh.Add(_dY2 + _dDelta * _dS);
                            _li0h.Add(i);
                            _li1h.Add(i - 1);
                            _lmh.Add(_dCurrentSlope);
                            xHigh.Add(_dX2 - _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                            yHigh.Add(_dY2 + _dDelta * _dS);

                            // initial values lower tube
                            _li0l[0] = i - 1;
                            _lml[0]  = 0.0;
                            xLow.Add(_dX2 - _dDelta - _dXMinStep);
                            yLow.Add(_dY2 - _dDelta * _dS);
                            _li0l.Add(i);
                            _li1l.Add(i - 1);
                            _lml.Add(_dCurrentSlope);
                            xLow.Add(_dX2 + _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                            yLow.Add(_dY2 - _dDelta * _dS);
                        }
                        else
                        {       // initial values upper tube
                            xHigh.Add(_dX2 - _dDelta);
                            yHigh.Add(_dY2 - _dCurrentSlope * _dDelta + _dDelta * Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS)));

                            // initial values lower tube
                            xLow.Add(_dX2 - _dDelta);
                            yLow.Add(_dY2 - _dCurrentSlope * _dDelta - _dDelta * Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS)));
                        }
                    }
                    else        // if not 1st interval (3.2.6)
                    {
                        // fill lists with new values, set X and Y to arbitrary value (3.2.6.1)
                        xHigh.Add(1);
                        yHigh.Add(1);

                        xLow.Add(1);
                        yLow.Add(1);

                        // begin procedure for upper tube
                        GenerateHighTube(x, y, xHigh, yHigh);

                        // begin procedure for lower tube
                        GenerateLowTube(x, y, xLow, yLow);
                    }
                }

                // calculate terminal value
                _dX2 = (double)x.GetValue(x.Length - 1);
                if (bJump)
                {
                    _dY2 = (double)y.GetValue(y.Length - 1);
                    // upper tube
                    _dCurrentSlope = (double)_lmh[_lmh.Count - 1];

                    xHigh.Add(_dX2 - _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                    yHigh.Add(_dY2 + _dDelta * _dS);
                    xHigh.Add(_dX2 + _dDelta + _dXMinStep);
                    yHigh.Add(_dY2 + _dDelta * _dS);

                    // lower tube
                    _dCurrentSlope = (double)_lml[_lml.Count - 1];

                    xLow.Add(_dX2 + _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                    yLow.Add(_dY2 - _dDelta * _dS);
                    xLow.Add(_dX2 + _dDelta + _dXMinStep);
                    yLow.Add(_dY2 - _dDelta * _dS);
                }
                else
                {
                    // upper tube
                    _dX1 = (double)xHigh[xHigh.Count - 1];
                    _dY1 = (double)yHigh[yHigh.Count - 1];

                    _dCurrentSlope = (double)_lmh[_lmh.Count - 1];

                    xHigh.Add(_dX2 + _dDelta);
                    yHigh.Add(_dY1 + _dCurrentSlope * (_dX2 + _dDelta - _dX1));

                    // lower tube
                    _dX1 = (double)xLow[xLow.Count - 1];
                    _dY1 = (double)yLow[yLow.Count - 1];

                    _dCurrentSlope = (double)_lml[_lml.Count - 1];

                    xLow.Add(_dX2 + _dDelta);
                    yLow.Add(_dY1 + _dCurrentSlope * (_dX2 + _dDelta - _dX1));
                }
                report.Lower = new Curve("Lower", xLow.ToArray(), yLow.ToArray());
                report.Upper = new Curve("Upper", xHigh.ToArray(), yHigh.ToArray());
                if (report.Lower.ImportSuccessful && report.Upper.ImportSuccessful)
                {
                    successful = true;
                }
            }
            return(report);
        }
示例#7
0
        public Report CompareFiles(Log log, CsvFile csvBase, string sReportPath, ref Options options)
        {
            int iInvalids = 0;

            Report rep = new Report(sReportPath);

            log.WriteLine("Comparing \"{0}\" to \"{1}\"", _fileName, csvBase.ToString());

            rep.BaseFile    = csvBase.ToString();
            rep.CompareFile = _fileName;

            Curve      reference    = new Curve();
            Curve      compareCurve = new Curve();
            TubeReport tubeReport   = new TubeReport();
            TubeSize   size         = null;
            Tube       tube         = new Tube(size);

            switch (options.Direction)
            {
            case ToleranceDirection.X:
                userchoice = 1;     ///set to 1 for tolerenace in X axis
                break;

            case ToleranceDirection.Y:
                userchoice = 0;     ///set to 0 for tolerance in Y-axis
                break;

            default:    //Invalid mode
                Console.WriteLine(options.GetUsage());
                break;
            }

            if (userchoice == 1)
            {
                IOptions tubeOptions = new Options1(_dRangeDelta, Axes.X);
            }
            else if (userchoice == 0)
            {
                IOptions tubeOptions = new Options1(_dRangeDelta, Axes.Y);
            }
            else
            {
                Console.WriteLine("opted for wrong choice");
                Environment.ExitCode = 2;
            }


            foreach (KeyValuePair <string, List <double> > res in csvBase.Results)
            {
                if (!this.Results.ContainsKey(res.Key))
                {
                    log.Error("{0} not found in \"{1}\", skipping checks.", res.Key, this._fileName);
                    rep.Chart.Add(new Chart()
                    {
                        Title = res.Key, Errors = 1
                    });
                    Environment.ExitCode = 1;
                    continue;
                }
                else
                {
                    compareCurve = new Curve(res.Key, this.XAxis.ToArray <double>(), this.Results[res.Key].ToArray <double>());

                    if (res.Value.Count == 0)
                    {
                        log.Error("{0} has no y-Values! Maybe error during parsing? Skipping", res.Key);
                        continue;
                    }
                    reference = new Curve("Reference ", csvBase.XAxis.ToArray(), csvBase.Results[res.Key].ToArray());
                    if (!reference.ImportSuccessful)
                    {
                        log.Error("Error in the calculation of the tubes. Skipping {0}", res.Key);
                        rep.Chart.Add(new Chart()
                        {
                            Title = res.Key, Errors = 1
                        });
                        continue;
                    }

                    if (reference.X.Length < compareCurve.X.Length)
                    {
                        log.WriteLine(LogLevel.Warning, "The resolution of the base x-axis is smaller than the compare x-axis. The better the base resolution is, the better the validation result will be!");
                    }
                    else
                    {
                        log.WriteLine(LogLevel.Debug, "The resolution of the base x-axis is good.");
                    }

                    size = new TubeSize(reference, true);
                    switch (options.Method)
                    {
                    case ExecutionMethod.Relative:
                        if (userchoice == 1)
                        {
                            size.Calculate(_dRangeDelta, Axes.X, Relativity.Relative);
                        }
                        else
                        {
                            size.Calculate(_dRangeDelta, Axes.Y, Relativity.Relative);
                        }
                        break;

                    case ExecutionMethod.Absolute:
                        if (userchoice == 1)
                        {
                            size.Calculate(_dRangeDelta, Axes.X, Relativity.Absolute);
                        }
                        else
                        {
                            size.Calculate(_dRangeDelta, Axes.Y, Relativity.Absolute);
                        }
                        break;

                    default:    //Invalid mode
                        Console.WriteLine(options.GetUsage());
                        break;
                    }
                    tube       = new Tube(size);
                    tubeReport = tube.Calculate(reference);
                    tube.Validate(compareCurve);

                    if (tubeReport.Valid == Validity.Valid)
                    {
                        log.WriteLine(res.Key + " is valid");
                    }
                    else
                    {
                        log.WriteLine(LogLevel.Warning, "{0} is invalid! {1} errors have been found during validation.", res.Key,
                                      (null != tube.Report.Errors && null != tube.Report.Errors.X) ? tube.Report.Errors.X.Length : 0);
                        iInvalids++;
                        Environment.ExitCode = 1;
                    }
                }
                if (null != tube.Report)//No charts for missing reports
                {
                    PrepareCharts(reference, compareCurve, tube.Report.Errors, rep, tubeReport, res, options.UseBitmapPlots);
                }
            }
            rep.Tolerance = _dRangeDelta;

            string sResult = "na";

            if (rep.TotalErrors == 0)
            {
                sResult = "passed";
            }
            else
            {
                sResult = "failed";
            }

            if (options.ComparisonFlag)
            {
                using (TextWriter writer = File.CreateText(string.Format("{0}{1}compare_{2}.log", Path.GetDirectoryName(_fileName), Path.DirectorySeparatorChar, sResult)))
                {
                    //Content needs to be defined
                    writer.WriteLine("CSV Compare Version {0} ({1})", Info.AssemblyVersion, Assembly.GetExecutingAssembly().GetName().ProcessorArchitecture);
                    writer.WriteLine("Comparison result file for {0}", _fileName);
                    writer.WriteLine(". Time:        {0:o}", DateTime.Now);
                    writer.WriteLine(". Operation:   {0}", options.Mode);
                    writer.WriteLine(". Tolerance:   {0}", options.Tolerance);
                    writer.WriteLine(". Execution Method:   {0}", options.Method);
                    writer.WriteLine(". Direction of Tolerence:   {0}", options.Direction);
                    writer.WriteLine(". Result:      {0}", sResult);

                    if (rep.TotalErrors > 0)
                    {
                        Chart pairMax = rep.Chart.Aggregate((l, r) => l.DeltaError > r.DeltaError ? l : r);
                        if (pairMax.DeltaError > 0)
                        {
                            writer.WriteLine(". Largest error: {0}=>{1}", pairMax.Title, pairMax.DeltaError);
                            writer.WriteLine(". Failed values:");

                            foreach (Chart c in (from r in rep.Chart where r.DeltaError > 0 select r).OrderByDescending(er => er.DeltaError))
                            {
                                writer.WriteLine("{0}=>{1}", c.Title, c.DeltaError.ToString(CultureInfo.InvariantCulture));
                            }
                        }
                    }
                }
            }

            rep.WriteReport(log, (string.IsNullOrEmpty(options.ReportDir) || options.NoMetaReport) ? string.Empty : options.ReportDir, options);
            GC.Collect();//immediately forget big charts and data
            return(rep);
        }
示例#8
0
        /// <summary>
        /// Calculates a lower and an upper tube curve.
        /// </summary>
        /// <param name="reference">Reference curve with x and y values.</param>
        /// <param name="size">Size of tube.</param>
        /// <returns>Collection of return values.</returns>
        public virtual TubeReport Calculate(Curve reference, TubeSize size)
        {
            TubeReport report = new TubeReport();

            return(report);
        }
示例#9
0
 /// <summary>
 /// Calculates a lower and an upper tube curve.
 /// </summary>
 /// <param name="reference">Reference curve with x and y values.</param>
 /// <param name="size">Size of tube.</param>
 /// <returns>Collection of return values.</returns>
 public virtual TubeReport Calculate(Curve reference, TubeSize size)
 {
     TubeReport report = new TubeReport();
     return report;
 }
示例#10
0
        /// <summary>
        /// Calculates a lower and an upper tube curve.
        /// </summary>
        /// <param name="reference">Reference curve with x and y values..</param>
        /// <param name="size">Size of tube.</param>
        /// <returns>Collection of return values.</returns>
        public override TubeReport Calculate(Curve reference, TubeSize size)
        {
            /*
            ///This method generates tubes around a given curve.
            /// @param x The time values for the base curve
            /// @param y The y values of the base curve
            /// @param x_low An empty, initialized list that is to be filled with the time values for the lower tube curve
            /// @param y_low An empty, initialized list that is to be filled with the y values for the lower tube curve
            /// @param x_high An empty, initialized list that is to be filled with the time values for the upper tube curve
            /// @param y_high An empty, initialized list that is to be filled with the y values for the upper tube curve
            /// @param r relative tolerance argument
            /// @param bAllIntervals used all intervals
             * */
            double[] x = reference.X.ToArray();
            double[] y = reference.Y.ToArray();
            _dDelta = size.X;
            List<double> xHigh = new List<double>();
            List<double> yHigh = new List<double>();
            List<double> xLow = new List<double>();
            List<double> yLow = new List<double>();

            TubeReport report = new TubeReport();
            successful = false;

            if (reference != null && size != null)
            {
                report.Reference = reference;
                report.Size = size;
                report.Algorithm = AlgorithmOptions.Ellipse;

                /// SBR: Wir gehen erstmal prinzipiell davon aus das der Algorithmus korrekt verläuft.
                //_bAllIntervals = true;
                // set tStart and tStop
                _dtStart = x[0];
                _dtStop = x[x.Length - 1];
                _dXMinStep = ((_dtStop - _dtStart) + Math.Abs(_dtStart)) * _dXRelEps;

                // Initialize lists (upper tube)
                _lmh = new ArrayList(x.Length);	// slope in i
                _li0h = new ArrayList(x.Length);	// i
                _li1h = new ArrayList(x.Length);	// i - 1

                // Initialize lists (lower tube)
                _lml = new ArrayList(x.Length);		// slope in i
                _li0l = new ArrayList(x.Length);	// i
                _li1l = new ArrayList(x.Length);	// i-1

                // calculate the tubes delta
                //_dDelta = r * (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart));

                // calculate S:
                double max = y[0];
                double min = y[0];

                xHigh.Clear();
                yHigh.Clear();
                xLow.Clear();
                yLow.Clear();

                if (xHigh.Capacity < x.Length) xHigh.Capacity = x.Length;
                if (yHigh.Capacity < x.Length) yHigh.Capacity = x.Length;
                if (xLow.Capacity < x.Length) xLow.Capacity = x.Length;
                if (yLow.Capacity < x.Length) yLow.Capacity = x.Length;

                for (int i = 1; i < y.Length; i++)
                {
                    try
                    {
                        double Y = y[i];
                        if (Y > max)
                        {
                            max = Y;
                        }
                        if (Y < min)
                        {
                            min = Y;
                        }
                    }
                    catch (IndexOutOfRangeException) { break; }
                }
                //            _dS = Math.Abs(4 * (max - min) / (Math.Abs(_dtStop - _dtStart)));
                _dS = (Math.Abs(max - min) + Math.Abs(min)) / (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart));

                if (_dS < 0.0004 / (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart)))
                {
                    _dS = 0.0004 / (Math.Abs(_dtStop - _dtStart) + Math.Abs(_dtStart));
                }

                bool bJump = false;

                // Begin calculation for the tubes
                for (int i = 1; i < x.Length; i++)
                {
                    try
                    {
                        // get current value
                        _dX1 = x[i];
                        _dY1 = y[i];

                        // get previous value
                        _dX2 = x[i - 1];
                        _dY2 = y[i - 1];
                    }
                    catch (IndexOutOfRangeException) { break; }

                    // catch jumps
                    bJump = false;
                    if ((_dX1 <= _dX2) && (_dY1 == _dY2) && (xHigh.Count == 0))
                        continue;
                    if ((_dX1 <= _dX2) && (_dY1 == _dY2))
                    {
                        _dX1 = Math.Max(x[(int)_li1l[_li1l.Count - 1]] + _dXMinStep, Math.Max(x[(int)_li1h[_li1h.Count - 1]] + _dXMinStep, _dX1));
                        x.SetValue(_dX1, i);
                        _dCurrentSlope = (double)_lmh[_lmh.Count - 1];
                    }
                    else
                    {
                        if (_dX1 <= _dX2)
                        {
                            bJump = true;
                            _dX1 = _dX2 + _dXMinStep;
                            x.SetValue(_dX1, i);
                        }
                        _dCurrentSlope = (_dY1 - _dY2) / (_dX1 - _dX2); // calculate current slope ( 3.2.6.1)
                    }

                    // fill lists with new values: values upper tube
                    _li0h.Add(i);
                    _li1h.Add(i - 1);
                    _lmh.Add(_dCurrentSlope);

                    // fill lists with new values: values lower tube
                    _li0l.Add(i);
                    _li1l.Add(i - 1);
                    if ((_dX1 <= _dX2) && (_dY1 == _dY2))
                        _dCurrentSlope = (double)_lml[_lml.Count - 1];
                    _lml.Add(_dCurrentSlope);

                    if (xHigh.Count == 0) // 1st interval (3.2.5)
                    {
                        if (bJump)
                        {
                            // initial values upper tube
                            _li0h[0] = i - 1;
                            _lmh[0] = 0.0;
                            xHigh.Add(_dX2 - _dDelta - _dXMinStep);
                            yHigh.Add(_dY2 + _dDelta * _dS);
                            _li0h.Add(i);
                            _li1h.Add(i - 1);
                            _lmh.Add(_dCurrentSlope);
                            xHigh.Add(_dX2 - _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                            yHigh.Add(_dY2 + _dDelta * _dS);

                            // initial values lower tube
                            _li0l[0] = i - 1;
                            _lml[0] = 0.0;
                            xLow.Add(_dX2 - _dDelta - _dXMinStep);
                            yLow.Add(_dY2 - _dDelta * _dS);
                            _li0l.Add(i);
                            _li1l.Add(i - 1);
                            _lml.Add(_dCurrentSlope);
                            xLow.Add(_dX2 + _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                            yLow.Add(_dY2 - _dDelta * _dS);
                        }
                        else
                        {       // initial values upper tube
                            xHigh.Add(_dX2 - _dDelta);
                            yHigh.Add(_dY2 - _dCurrentSlope * _dDelta + _dDelta * Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS)));

                            // initial values lower tube
                            xLow.Add(_dX2 - _dDelta);
                            yLow.Add(_dY2 - _dCurrentSlope * _dDelta - _dDelta * Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS)));
                        }
                    }
                    else	// if not 1st interval (3.2.6)
                    {
                        // fill lists with new values, set X and Y to arbitrary value (3.2.6.1)
                        xHigh.Add(1);
                        yHigh.Add(1);

                        xLow.Add(1);
                        yLow.Add(1);

                        // begin procedure for upper tube
                        GenerateHighTube(x, y, xHigh, yHigh);

                        // begin procedure for lower tube
                        GenerateLowTube(x, y, xLow, yLow);
                    }

                }

                // calculate terminal value
                _dX2 = (double)x.GetValue(x.Length - 1);
                if (bJump)
                {
                    _dY2 = (double)y.GetValue(y.Length - 1);
                    // upper tube
                    _dCurrentSlope = (double)_lmh[_lmh.Count - 1];

                    xHigh.Add(_dX2 - _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                    yHigh.Add(_dY2 + _dDelta * _dS);
                    xHigh.Add(_dX2 + _dDelta + _dXMinStep);
                    yHigh.Add(_dY2 + _dDelta * _dS);

                    // lower tube
                    _dCurrentSlope = (double)_lml[_lml.Count - 1];

                    xLow.Add(_dX2 + _dDelta * _dCurrentSlope / (_dS + Math.Sqrt((_dCurrentSlope * _dCurrentSlope) + (_dS * _dS))));
                    yLow.Add(_dY2 - _dDelta * _dS);
                    xLow.Add(_dX2 + _dDelta + _dXMinStep);
                    yLow.Add(_dY2 - _dDelta * _dS);
                }
                else
                {
                    // upper tube
                    _dX1 = (double)xHigh[xHigh.Count - 1];
                    _dY1 = (double)yHigh[yHigh.Count - 1];

                    _dCurrentSlope = (double)_lmh[_lmh.Count - 1];

                    xHigh.Add(_dX2 + _dDelta);
                    yHigh.Add(_dY1 + _dCurrentSlope * (_dX2 + _dDelta - _dX1));

                    // lower tube
                    _dX1 = (double)xLow[xLow.Count - 1];
                    _dY1 = (double)yLow[yLow.Count - 1];

                    _dCurrentSlope = (double)_lml[_lml.Count - 1];

                    xLow.Add(_dX2 + _dDelta);
                    yLow.Add(_dY1 + _dCurrentSlope * (_dX2 + _dDelta - _dX1));
                }
                report.Lower = new Curve("Lower", xLow.ToArray(), yLow.ToArray());
                report.Upper = new Curve("Upper", xHigh.ToArray(), yHigh.ToArray());
                if (report.Lower.ImportSuccessful && report.Upper.ImportSuccessful)
                    successful = true;
            }
            return report;
        }
示例#11
0
        /// <summary>
        /// Compares a test curve with a reference curve. Calculates a tube, if test curve data == null.
        /// </summary>
        /// <param name="modelName">Model name.</param>
        /// <param name="resultName">Result name.</param>
        /// <param name="referenceX">x values of reference curve.</param>
        /// <param name="referenceY">y values of reference curve.</param>
        /// <param name="testX">x values of test curve.</param>
        /// <param name="testY">y values of test curve.</param>
        /// <param name="options">Options for calculation of tube size, chart and saving.</param>
        /// <returns>Tube report.</returns>
        public TubeReport Validate(string modelName, string resultName, double[] referenceX, double[] referenceY, double[] testX, double[] testY, IOptions options)
        {
            TubeReport report = new TubeReport();
            Curve refCurve, testCurve;
            bool testExists = (testX != null && testY != null && testX.Length > 0 && testY.Length > 0);
            #if GUI
            bool saveImage = (!String.IsNullOrWhiteSpace(options.ReportFolder) && Directory.Exists(options.ReportFolder));
            #endif
            string name = modelName + " - " + resultName;

            // write log file
            if (options.Log != null)
            {
                options.Log.WriteLine(LogLevel.Done, "----------------------------------------------");
                options.Log.WriteLine(LogLevel.Done, "Model: " + modelName);
                options.Log.WriteLine(LogLevel.Done, "Result: " + resultName);
            }

            if (referenceX != null && referenceY != null && referenceX.Length != 0 && referenceY.Length != 0)
            {
                // Data import: Prepare curve data
                refCurve = new Curve("Reference " + name, referenceX, referenceY);

                if (testExists)
                    testCurve = new Curve("Test " + name, testX, testY);
                else
                    testCurve = new Curve();

                if (refCurve.ImportSuccessful && (testCurve.ImportSuccessful || !testExists))
                {
                    // Calculate tube size
                    TubeSize size = new TubeSize(refCurve);

                    if (!Double.IsNaN(options.BaseX)) // overwrite BaseX just in case it has got a value
                        size.BaseX = options.BaseX;
                    if (!Double.IsNaN(options.BaseY))
                        size.BaseY = options.BaseY;
                    if (!Double.IsNaN(options.Ratio))
                        size.Ratio = options.Ratio;

                    if (options is Options1)
                        size.Calculate(((Options1)options).Value, ((Options1)options).Axes, options.Relativity);
                    else if (options is Options2)
                        size.Calculate(((Options2)options).X, ((Options2)options).Y, options.Relativity);

                    if (size.Successful)
                    {
                        // Calculate tube
                        Tube tube = new Tube(size);
                        tube.AlgorithmOption = Algorithms.AlgorithmOptions.Rectangle;
                        report = tube.Calculate(refCurve);

                        if (tube.TubeSuccessful)
                        {
                            if (testExists)
                            {
                                // Validation
                                report = tube.Validate(testCurve);

                                if (options.Log != null)
                                {
                                    options.Log.WriteLine(LogLevel.Done, "Test curve is " + report.Valid.ToString());
                                    options.Log.WriteLine(LogLevel.Done, "Errors (points of test curve outside tube): " + report.Errors.Count.ToString());
                                }

                                if (tube.ValidationSuccessful)
                                    report.ErrorStep = Step.None;

                                // Error: Validation not successful
                                else
                                {
                                    report.ErrorStep = Step.Validation;
                                    report.Valid = Validity.Undefined;
                                    if (options.Log != null)
                                        options.Log.WriteLine(LogLevel.Error, "Validation not successful.");
                                }
                            }
                            else
                            {
                                report.ErrorStep = Step.Validation;
                                report.Valid = Validity.Undefined;
                                if (options.Log != null)
                                    options.Log.WriteLine(LogLevel.Error, "no test curve data.");
                            }

            #if GUI
                            // Visualization
                            if (saveImage || options.ShowWindow)
                            {
                                ChartControl chartControl = new ChartControl(options.DrawFastAbove, options.DrawPointsBelow, options.DrawLabelNumber);
                                chartControl.addTitle("Tube size: (" + size.X + "; " + size.Y + ")");
                                chartControl.AddLine(refCurve.Name, refCurve, Color.FromKnownColor(KnownColor.OrangeRed));
                                chartControl.AddLine("Upper", report.Upper, Color.FromKnownColor(KnownColor.MediumSpringGreen));
                                chartControl.AddLine("Lower", report.Lower, Color.FromKnownColor(KnownColor.DeepSkyBlue));

                                if (testExists)
                                {
                                    string valid = "";
                                    if (report.Valid == Validity.Valid)
                                        valid = "valid";
                                    else if (report.Valid == Validity.Invalid)
                                        valid = "invalid";
                                    chartControl.AddLine("Test: " + valid, testCurve, Color.FromKnownColor(KnownColor.BlueViolet));
                                    chartControl.AddErrors("Errors " + name, report.Errors);
                                }

                                // Visualization: save image
                                if (saveImage)
                                    chartControl.saveAsImage(options.ReportFolder + resultName + ".png", System.Drawing.Imaging.ImageFormat.Png);

                                // Visualization: show window with MS Chart Control
                                if (options.ShowWindow)
                                    if (!testExists || (options.ShowValidity == Validity.All || (options.ShowValidity == Validity.Invalid && report.Valid == Validity.Invalid) || (options.ShowValidity == Validity.Valid && report.Valid == Validity.Valid)))
                                        Application.Run(chartControl);

                            }
            #endif
                        }
                        // Error: tube calculation not successful
                        else
                        {
                            report.ErrorStep = Step.Tube;
                            report.Valid = Validity.Undefined;

                            if (options.Log != null)
                                options.Log.WriteLine(LogLevel.Error, "Tube calculation not successful.");
                        }

                        if (options.Log != null)
                        {
                            options.Log.WriteLine(LogLevel.Done, "Tube calculation algorithm: " + tube.AlgorithmOption.ToString());
                        }
                    }
                    // Error: tube size calculation not successful
                    else
                    {
                        report.ErrorStep = Step.TubeSize;
                        report.Valid = Validity.Undefined;

                        if (options.Log != null)
                        {
                            options.Log.WriteLine(LogLevel.Error, "TubeSize calculation not successful.");
                            options.Log.WriteLine(LogLevel.Error, "TubeSize.Ratio: " + size.Ratio);
                            options.Log.WriteLine(LogLevel.Error, "TubeSize.BaseX: " + size.BaseX);
                            options.Log.WriteLine(LogLevel.Error, "TubeSize.BaseY: " + size.BaseY);
                        }
                    }
                    report.Size = size;
                }
                // Error: data import not successful
                else
                {
                    report.ErrorStep = Step.DataImport;
                    report.Valid = Validity.Undefined;

                    if (options.Log != null)
                    {
                        if (!refCurve.ImportSuccessful)
                            options.Log.WriteLine(LogLevel.Error, "Reference curve: Import not successful.");
                        if (testExists && !testCurve.ImportSuccessful)
                            options.Log.WriteLine(LogLevel.Error, "Test curve: Import not successful.");
                    }
                }
                report.Reference = refCurve;
                report.Test = testCurve;
            }
            // Error: no data
            else
            {
                report.ErrorStep = Step.DataImport;
                report.Valid = Validity.Undefined;

                if (options.Log != null)
                {
                    if (referenceX != null || referenceX.Length != 0)
                        options.Log.WriteLine(LogLevel.Error, "Reference curve: Missing data.");
                    if (referenceY!= null || referenceY.Length != 0)
                        options.Log.WriteLine(LogLevel.Error, "Test curve: Missing data.");
                }
            }

            report.ModelName = modelName;
            report.ResultName = resultName;
            return report;
        }
示例#12
0
        /// <summary>
        /// Calculates the lower tube curve.
        /// </summary>
        /// <param name="reference">Reference curve.</param>
        /// <param name="size">Tube size.</param>
        /// <returns>Lower tube curve.</returns>
        private static Curve CalculateLower(Curve reference, TubeSize size)
        {
            Curve         lower;                                       // lower tube curve
            List <double> LX = new List <double>(2 * reference.Count); // x-values of lower tube curve
            List <double> LY = new List <double>(2 * reference.Count); // y-values of lower tube curve

            // -------------------------------------------------------------------------------------------------------------
            // ---------------------------------- 1. Add corner points of the rectangle  -----------------------------------
            // -------------------------------------------------------------------------------------------------------------

            double m0, m1;                                           // slopes before and after point i of reference curve
            double s0, s1;                                           // signum of slopes of reference curve: 1 - increasing, 0 - constant, -1 - decreasing
            // machine accuracy, the least positive floating point number, with 1 + epsilon > 1, is ~ 2.2e-16
            double epsilon = 1e-15;
            int    b;

            // -------------------------------------------------------------------------------------------------------------
            // 1.1. Start: Rectangle with center (x,y) = (reference.X[0], reference.Y[0])
            // -------------------------------------------------------------------------------------------------------------

            // ignore identical point at the beginning
            b = 0;
            while ((reference.X[b] - reference.X[b + 1] == 0) && (reference.Y[b] - reference.Y[b + 1] == 0))
            {
                b++;
            }
            // slopes of reference curve (initialization)
            s0 = Math.Sign(reference.Y[b + 1] - reference.Y[b]);
            if (reference.X[b + 1] != reference.X[b])
            {
                m0 = (reference.Y[b + 1] - reference.Y[b]) / (reference.X[b + 1] - reference.X[b]);
            }
            else
            if (s0 > 0)
            {
                m0 = Double.PositiveInfinity;
            }
            else
            {
                m0 = Double.NegativeInfinity;
            }

            // add point down left
            LX.Add(reference.X[b] - size.X);
            LY.Add(reference.Y[b] - size.Y);

            if (s0 == 1)
            {
                // add point down right
                LX.Add(reference.X[b] + size.X);
                LY.Add(reference.Y[b] - size.Y);
            }

            // -------------------------------------------------------------------------------------------------------------
            // 1.2. Iteration: Rectangle with center (x,y) = (reference.X[i], reference.Y[i])
            // -------------------------------------------------------------------------------------------------------------
            for (int i = b + 1; i < reference.Count - 1; i++)
            {
                // ignore identical points
                if ((reference.X[i] - reference.X[i + 1] == 0) && (reference.Y[i] - reference.Y[i + 1] == 0))
                {
                    continue;
                }

                // slopes of reference curve
                s1 = Math.Sign(reference.Y[i + 1] - reference.Y[i]);
                if (reference.X[i + 1] - reference.X[i] != 0)
                {
                    m1 = (reference.Y[i + 1] - reference.Y[i]) / (reference.X[i + 1] - reference.X[i]);
                }
                else
                if (s1 > 0)
                {
                    m1 = Double.PositiveInfinity;
                }
                else
                {
                    m1 = Double.NegativeInfinity;
                }

                // add no point for equal slopes of reference curve
                if (!(m0 == m1))
                {
                    if ((s0 != -1) && (s1 != -1))
                    {
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 != 1) && (s1 != 1))
                    {
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 == -1) && (s1 == 1))
                    {
                        // add point down left
                        LX.Add(reference.X[i] - size.X);

                        LY.Add(reference.Y[i] - size.Y);
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 == 1) && (s1 == -1))
                    {
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    // remove the last added points in case of equal slopes equal 0 of tube curve
                    int last = LY.Count - 1;
                    if (reference.Y[i + 1] - size.Y == LY[last])
                    {
                        // remove two points, if two points were added at last
                        // (last - 2 >= 0, because start point + two added points)
                        if (s0 * s1 == -1 && LY[last - 2] == LY[last])
                        {
                            LX.RemoveAt(last);
                            LY.RemoveAt(last);
                            LX.RemoveAt(last - 1);
                            LY.RemoveAt(last - 1);
                        }
                        // remove one point, if one point was added at last
                        // (last - 1 >= 0, because start point + one added point)
                        else if (s0 * s1 != -1 && LY[last - 1] == LY[last])
                        {
                            LX.RemoveAt(last);
                            LY.RemoveAt(last);
                        }
                    }
                }
                s0 = s1;
                m0 = m1;
            }
            // -------------------------------------------------------------------------------------------------------------
            // 1.3. End: Rectangle with center (x,y) = (reference.X[reference.Count - 1], reference.Y[reference.Count - 1])
            // -------------------------------------------------------------------------------------------------------------
            if (s0 == -1)
            {
                // add point down left
                LX.Add(reference.X[reference.Count - 1] - size.X);
                LY.Add(reference.Y[reference.Count - 1] - size.Y);
            }
            // add point down right
            LX.Add(reference.X[reference.Count - 1] + size.X);
            LY.Add(reference.Y[reference.Count - 1] - size.Y);

            // -------------------------------------------------------------------------------------------------------------
            // -------------- 2. Remove points and add intersection points in case of backward order -----------------------
            // -------------------------------------------------------------------------------------------------------------

            removeLoop(LX, LY, true);

            lower = new Curve("Lower", LX.ToArray(), LY.ToArray());
            return(lower);
        }
示例#13
0
        /// <summary>
        /// Calculates the lower tube curve.
        /// </summary>
        /// <param name="reference">Reference curve.</param>
        /// <param name="size">Tube size.</param>
        /// <returns>Lower tube curve.</returns>
        private static Curve CalculateLower(Curve reference, TubeSize size)
        {
            Curve         lower;                                       // lower tube curve
            List <double> LX = new List <double>(2 * reference.Count); // x-values of lower tube curve
            List <double> LY = new List <double>(2 * reference.Count); // y-values of lower tube curve

            // -------------------------------------------------------------------------------------------------------------
            // ---------------------------------- 1. Add corner points of the rectangle  -----------------------------------
            // -------------------------------------------------------------------------------------------------------------

            double m0, m1;                                           // slopes before and after point i of reference curve
            double s0, s1;                                           // signum of slopes of reference curve: 1 - increasing, 0 - constant, -1 - decreasing

            // -------------------------------------------------------------------------------------------------------------
            // 1.1. Start: Rectangle with center (x,y) = (reference.X[0], reference.Y[0])
            // -------------------------------------------------------------------------------------------------------------

            // slopes of reference curve (initialization)
            s0 = Math.Sign(reference.Y[1] - reference.Y[0]);
            if (reference.X[1] != reference.X[0])
            {
                m0 = (reference.Y[1] - reference.Y[0]) / (reference.X[1] - reference.X[0]);
            }
            else
            if (s0 > 0)
            {
                m0 = Double.PositiveInfinity;
            }
            else
            {
                m0 = Double.NegativeInfinity;
            }

            // add point down left
            LX.Add(reference.X[0] - size.X);
            LY.Add(reference.Y[0] - size.Y);

            if (s0 == 1)
            {
                // add point down right
                LX.Add(reference.X[0] + size.X);
                LY.Add(reference.Y[0] - size.Y);
            }

            // -------------------------------------------------------------------------------------------------------------
            // 1.2. Iteration: Rectangle with center (x,y) = (reference.X[i], reference.Y[i])
            // -------------------------------------------------------------------------------------------------------------
            for (int i = 1; i < reference.Count - 1; i++)
            {
                if (reference.X[i] == reference.X[i + 1] && reference.Y[i] == reference.Y[i + 1]) // ignore identical points
                {
                    continue;
                }

                // slopes of reference curve
                s1 = Math.Sign(reference.Y[i + 1] - reference.Y[i]);
                if (reference.X[i + 1] != reference.X[i])
                {
                    m1 = (reference.Y[i + 1] - reference.Y[i]) / (reference.X[i + 1] - reference.X[i]);
                }
                else
                if (s1 > 0)
                {
                    m1 = Double.PositiveInfinity;
                }
                else
                {
                    m1 = Double.NegativeInfinity;
                }

                // add no point for equal slopes of reference curve
                if (!(m0 == m1))
                {
                    if ((s0 != -1) && (s1 != -1))
                    {
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 != 1) && (s1 != 1))
                    {
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 == -1) && (s1 == 1))
                    {
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    else if ((s0 == 1) && (s1 == -1))
                    {
                        // add point down right
                        LX.Add(reference.X[i] + size.X);
                        LY.Add(reference.Y[i] - size.Y);
                        // add point down left
                        LX.Add(reference.X[i] - size.X);
                        LY.Add(reference.Y[i] - size.Y);
                    }
                    // remove point for equal slopes equal 0 of tube curve
                    int last = LY.Count - 1;
                    if (reference.Y[i + 1] - size.Y == LY[last] && LY[last - 1] == LY[last])
                    {
                        LX.RemoveAt(last);
                        LY.RemoveAt(last);
                    }
                }
                s0 = s1;
                m0 = m1;
            }
            // -------------------------------------------------------------------------------------------------------------
            // 1.3. End: Rectangle with center (x,y) = (reference.X[reference.Count - 1], reference.Y[reference.Count - 1])
            // -------------------------------------------------------------------------------------------------------------
            if (s0 == -1)
            {
                // add point down left
                LX.Add(reference.X[reference.Count - 1] - size.X);
                LY.Add(reference.Y[reference.Count - 1] - size.Y);
            }
            // add point down right
            LX.Add(reference.X[reference.Count - 1] + size.X);
            LY.Add(reference.Y[reference.Count - 1] - size.Y);

            // -------------------------------------------------------------------------------------------------------------
            // -------------- 2. Remove points and add intersection points in case of backward order -----------------------
            // -------------------------------------------------------------------------------------------------------------

            removeLoop(LX, LY, true);

            lower = new Curve("Lower", LX.ToArray(), LY.ToArray());
            return(lower);
        }