mirror of
https://github.com/jkriege2/JKQtPlotter.git
synced 2025-01-12 08:50:31 +08:00
f2004a6f66
IMPROVED/REWORKED: legend/key positioning as combination of 3 values, e.g. \c JKQTPKeyOutsideTop|JKQTPKeyTop|JKQTPKeyRight or \c JKQTPKeyInside|JKQTPKeyTopJKQTPKeyRight
249 lines
11 KiB
C++
249 lines
11 KiB
C++
/*
|
|
Copyright (c) 2020-2022 Jan W. Krieger (<jan@jkrieger.de>)
|
|
|
|
|
|
|
|
This software is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License (LGPL) as published by
|
|
the Free Software Foundation, either version 2.1 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License (LGPL) for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License (LGPL)
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
#ifndef jkqtpevaluatedparametriccurve_H
|
|
#define jkqtpevaluatedparametriccurve_H
|
|
|
|
|
|
#include <QString>
|
|
#include <QPainter>
|
|
#include <QPair>
|
|
#include "jkqtplotter/graphs/jkqtpscatter.h"
|
|
#include "jkqtplotter/jkqtpgraphsbasestylingmixins.h"
|
|
#include "jkqtplotter/jkqtplotter_imexport.h"
|
|
#include "jkqtcommon/jkqtpgeometrytools.h"
|
|
#include "jkqtplotter/graphs/jkqtpevaluatedfunctionbase.h"
|
|
#include <functional>
|
|
|
|
|
|
/** \brief Base class for line plots where the data is taken from a user supplied function \f$ [x,y]=f(t) \f$
|
|
* The function is evaluated on a user-specified range \f$ t \in \left[t_\text{min}, t_\text{max}\right] \f$
|
|
* \ingroup jkqtplotter_functiongraphs
|
|
*
|
|
* This class uses the intelligent plotting algorithm for functions, implemented in JKQTPAdaptiveFunctionGraphEvaluator.
|
|
* It starts by sampling the function at minSamples positions. Then each function interval is bisected recursively if
|
|
* necessary. To do so the function is evaluated at the mid point and the slopes \f$ \alpha_{\mbox{left}} \f$
|
|
* and \f$ \alpha_{\mbox{right}} \f$ of the two linear segments are compared. the midpoint is added
|
|
* to the graph if \f[ \left|\alpha_{\mbox{right}}-\alpha_{\mbox{left}}\right|>\mbox{slopeTolerance} \f]
|
|
* In addition all sampling points except minimum and maximum are beeing shifted by a random fraction their
|
|
* distance to the other points. This helps to prevent beats when sampling periodic functions.
|
|
*
|
|
* Finally the obtained data is cleaned up to reduce the amount of points, by deleting a point, when it leads to an
|
|
* angle between consecutive line-segments of less than dataCleanupMaxAllowedAngleDegree.
|
|
*
|
|
*
|
|
* \see JKQTPXYFunctionLineGraphBase for a concrete implementation with C++-functors as functions
|
|
*/
|
|
class JKQTPLOTTER_LIB_EXPORT JKQTPXYFunctionLineGraphBase: public JKQTPEvaluatedFunctionGraphBase, public JKQTPGraphLineStyleMixin {
|
|
Q_OBJECT
|
|
public:
|
|
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraphBase(JKQTBasePlotter* parent=nullptr);
|
|
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraphBase(JKQTPlotter* parent);
|
|
|
|
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraphBase(double tmin_, double tmax_, JKQTBasePlotter* parent);
|
|
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraphBase(double tmin_, double tmax_, JKQTPlotter* parent);
|
|
|
|
/** \brief class destructor */
|
|
virtual ~JKQTPXYFunctionLineGraphBase() override;
|
|
|
|
/** \brief plots the graph to the plotter object specified as parent */
|
|
virtual void draw(JKQTPEnhancedPainter& painter) override;
|
|
/** \brief plots a key marker inside the specified rectangle \a rect */
|
|
virtual void drawKeyMarker(JKQTPEnhancedPainter& painter, const QRectF& rect) override;
|
|
/** \brief returns the color to be used for the key label */
|
|
virtual QColor getKeyLabelColor() const override;
|
|
|
|
|
|
/** \brief returns the t-value range for \f$ [x,y]=f(t), t \in \left[t_\text{min}, t_\text{max}\right] \f$ */
|
|
QPair<double,double> getTRange() const;
|
|
/** \copydoc tmin */
|
|
double getTMin() const;
|
|
/** \copydoc tmax */
|
|
double getTMax() const;
|
|
|
|
public Q_SLOTS:
|
|
/** \copydoc tmin */
|
|
void setTMin(double val);
|
|
/** \copydoc tmax */
|
|
void setTMax(double val);
|
|
/** \brief set the t-value range for \f$ [x,y]=f(t), t \in \left[t_\text{min}, t_\text{max}\right] \f$ */
|
|
void setTRange(double tmin_, double tmax_);
|
|
/** \brief set the t-value range for \f$ [x,y]=f(t), t \in \left[t_\text{min}, t_\text{max}\right] \f$ */
|
|
void setTRange(const QPair<double,double>& range);
|
|
protected:
|
|
|
|
/** \brief lower bound of t-value range for \f$ [x,y]=f(t), t \in \left[t_\text{min}, t_\text{max}\right] \f$ , i.e. \f$ t_\text{min} \f$ , default is 0
|
|
*
|
|
* \see getTMin(), getTMax(), setTMin(), setTMax(), setTRange(), getTRange() */
|
|
double tmin;
|
|
/** \brief upper bound of t-value range for \f$ [x,y]=f(t), t \in \left[t_\text{min}, t_\text{max}\right] \f$ , i.e. \f$ t_\text{min} \f$ , default is 1
|
|
*
|
|
* \see getTMin(), getTMax(), setTMin(), setTMax(), setTRange(), getTRange() */
|
|
double tmax;
|
|
|
|
};
|
|
|
|
|
|
|
|
/** \brief type of functions that may be plotted by JKQTPXYFunctionLineGraph
|
|
* \ingroup jkqtplotter_functiongraphs
|
|
*
|
|
* This is the type of functions \f$ [x,y]=f(t, \vec{p}) \f$ that may be plottet by JKQTPXYFunctionLineGraph.
|
|
* It is possible to supply parameters \f$ \vec{p} \f$ to the function that
|
|
* influence its result. Parameters are given as a pointer to some memory location. The function has to
|
|
* know on its own how to interpret these.
|
|
*/
|
|
typedef std::function<QPointF(double, const QVector<double>)> jkqtpParametricCurveFunctionType;
|
|
|
|
/** \brief simplified type of functions (without parameters) that may be plotted by JKQTPXYFunctionLineGraph
|
|
* \ingroup jkqtplotter_functiongraphs
|
|
*
|
|
* This is the type of functions \f$ [x,y]=f(t) \f$ that may be plottet by JKQTPXYFunctionLineGraph
|
|
* and JKQTPYFunctionLineGraph.
|
|
*/
|
|
typedef std::function<QPointF(double)> jkqtpSimpleParametricCurveFunctionType;
|
|
|
|
|
|
/** \brief This implements line plots where the data is taken from a user supplied function \f$ [x,y]=f(t) \f$
|
|
* The function is evaluated on a user-specified range \f$ t \in \left[t_\text{min}, t_\text{max}\right] \f$
|
|
* \ingroup jkqtplotter_functiongraphs
|
|
*
|
|
* \see JKQTPXYFunctionLineGraphBase for details on the used plotting algorithm
|
|
*
|
|
* The following image shows a Lissajou's fugure drawn with this function
|
|
*
|
|
* \image html JKQTPXYFunctionLineGraph.png
|
|
*
|
|
* The source code for this example is:
|
|
* \code
|
|
* JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
|
* // here we define the C++-functor for [x,y]=f(t)
|
|
* func1->setPlotFunctionFunctor([](double t) -> QPointF {
|
|
* const double a=5;
|
|
* const double b=4;
|
|
* const double delta=JKQTPSTATISTICS_PI/4.0;
|
|
* return QPointF(sin(a*t+delta), sin(b*t));
|
|
* });
|
|
* // and define the range over which to evaluate
|
|
* func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
|
* \endcode
|
|
*
|
|
* The source code for the same example, but using functions that also get a parameter vector:
|
|
* \code
|
|
* JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
|
* // here we define the C++-functor for [x,y]=f(t)
|
|
* func1->setPlotFunctionFunctor([](double t, const QVector<double>& params) -> QPointF {
|
|
* return QPointF(3.0*sin(params[0]*t+params[2])+8.0, 3.0*sin(params[1]*t));
|
|
* });
|
|
* // now we define the 3 parameters of the function
|
|
* // parameters are a, b, delta, as in the example above (in that order)
|
|
* func1->setParamsV(5, 4, JKQTPSTATISTICS_PI/4.0);
|
|
* // and define the range over which to evaluate
|
|
* func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
|
* \endcode
|
|
*
|
|
* \see \ref JKQTPlotterEvalCurves , JKQTPAdaptiveFunctionGraphEvaluator, JKQTPXFunctionLineGraph, JKQTPYFunctionLineGraph
|
|
*/
|
|
class JKQTPLOTTER_LIB_EXPORT JKQTPXYFunctionLineGraph: public JKQTPXYFunctionLineGraphBase {
|
|
Q_OBJECT
|
|
public:
|
|
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(JKQTBasePlotter* parent=nullptr);
|
|
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(JKQTPlotter* parent);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(const jkqtpSimpleParametricCurveFunctionType & f, const QString& title, double tmin_=0, double tmax_=1, JKQTBasePlotter* parent=nullptr);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(const jkqtpSimpleParametricCurveFunctionType & f, const QString& title, double tmin_, double tmax_, JKQTPlotter* parent);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(jkqtpSimpleParametricCurveFunctionType && f, const QString& title, double tmin_=0, double tmax_=1, JKQTBasePlotter* parent=nullptr);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(jkqtpSimpleParametricCurveFunctionType && f, const QString& title, double tmin_, double tmax_, JKQTPlotter* parent);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(const jkqtpParametricCurveFunctionType & f, const QString& title, double tmin_=0, double tmax_=1, JKQTBasePlotter* parent=nullptr);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(const jkqtpParametricCurveFunctionType & f, const QString& title, double tmin_, double tmax_, JKQTPlotter* parent);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(jkqtpParametricCurveFunctionType && f, const QString& title, double tmin_=0, double tmax_=1, JKQTBasePlotter* parent=nullptr);
|
|
/** \brief class constructor */
|
|
JKQTPXYFunctionLineGraph(jkqtpParametricCurveFunctionType && f, const QString& title, double tmin_, double tmax_, JKQTPlotter* parent);
|
|
|
|
/** \brief class destructor */
|
|
virtual ~JKQTPXYFunctionLineGraph() override;
|
|
|
|
|
|
/** \brief sets a functor to be plotted
|
|
*
|
|
* \see plotFunction
|
|
*/
|
|
void setPlotFunctionFunctor (jkqtpParametricCurveFunctionType && __value);
|
|
/** \brief sets a functor to be plotted
|
|
*
|
|
* \see plotFunction
|
|
*/
|
|
void setPlotFunctionFunctor (const jkqtpParametricCurveFunctionType & __value);
|
|
/** \brief sets a functor to be plotted
|
|
*
|
|
* \see simplePlotFunction
|
|
*/
|
|
void setPlotFunctionFunctor (jkqtpSimpleParametricCurveFunctionType && __value);
|
|
/** \brief sets a functor to be plotted
|
|
*
|
|
* \see simplePlotFunction
|
|
*/
|
|
void setPlotFunctionFunctor (const jkqtpSimpleParametricCurveFunctionType & __value);
|
|
/** \copydoc plotFunction
|
|
*
|
|
* \see isSimplePlotFunction() */
|
|
jkqtpParametricCurveFunctionType getPlotFunctionFunctor () const;
|
|
/** \copydoc simplePlotFunction
|
|
*
|
|
* \see isSimplePlotFunction() */
|
|
jkqtpSimpleParametricCurveFunctionType getSimplePlotFunction () const;
|
|
|
|
/** \brief returns whether the plot function was defined as a jkqtpSimpleParametricCurveFunctionType (\c true ) or
|
|
* a jkqtpParametricCurveFunctionType (\c false ) */
|
|
bool isSimplePlotFunction() const;
|
|
|
|
|
|
protected:
|
|
/** \copydoc JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec() */
|
|
virtual PlotFunctorSpec buildPlotFunctorSpec() override;
|
|
|
|
|
|
/** \brief the function to be plotted */
|
|
jkqtpParametricCurveFunctionType plotFunction;
|
|
/** \brief a simple function to be plotted, simplified form without parameters */
|
|
jkqtpSimpleParametricCurveFunctionType simplePlotFunction;
|
|
|
|
};
|
|
|
|
#endif // jkqtpevaluatedparametriccurve_H
|