mirror of
https://github.com/jkriege2/JKQtPlotter.git
synced 2025-01-12 17:00:32 +08:00
- cleanup of class-hierarchy of evaluated functions
- added slots for several properties of evaluated functions - updated examples
This commit is contained in:
parent
414e12920e
commit
f77bc97ca5
@ -23,6 +23,7 @@ Changes, compared to \ref page_whatsnew_V2019_11 "v2019.11" include:
|
|||||||
<li>removed the usage of some deprecated functions and objects (e.g. QMatrix)</li>
|
<li>removed the usage of some deprecated functions and objects (e.g. QMatrix)</li>
|
||||||
<li>improved: geometric objects now use an adaptive drawing algorithm to represent curves (before e.g. ellipses were always separated into a fixed number of line-segments)</li>
|
<li>improved: geometric objects now use an adaptive drawing algorithm to represent curves (before e.g. ellipses were always separated into a fixed number of line-segments)</li>
|
||||||
<li>improved: constructors and access functions for several geometric objects (e.g. more constructors, additional functions to retrieve parameters in diferent forms, iterators for polygons, ...)</li>
|
<li>improved: constructors and access functions for several geometric objects (e.g. more constructors, additional functions to retrieve parameters in diferent forms, iterators for polygons, ...)</li>
|
||||||
|
<li>improved: reworked class hirarchy of parsed function plots and declared several setters as slots.</li>
|
||||||
<li>new: added geometric plot objects JKQTPGeoArrow to draw arrows (aka lines with added line-end decorators, also extended JKQTPGeoLine, JKQTPGeoInfiniteLine, JKQTPGeoPolyLines to draw line-end decorator (aka arrows)</li>
|
<li>new: added geometric plot objects JKQTPGeoArrow to draw arrows (aka lines with added line-end decorators, also extended JKQTPGeoLine, JKQTPGeoInfiniteLine, JKQTPGeoPolyLines to draw line-end decorator (aka arrows)</li>
|
||||||
<li>new: all geometric objects can either be drawn as graphic element (i.e. lines are straight line, even on non-linear axes), or as mathematical curve (i.e. on non-linear axes, lines become the appropriate curve representing the linear function, connecting the given start/end-points). The only exceptions are ellipses (and the derived arcs,pies,chords), which are always drawn as mathematical curves</li>
|
<li>new: all geometric objects can either be drawn as graphic element (i.e. lines are straight line, even on non-linear axes), or as mathematical curve (i.e. on non-linear axes, lines become the appropriate curve representing the linear function, connecting the given start/end-points). The only exceptions are ellipses (and the derived arcs,pies,chords), which are always drawn as mathematical curves</li>
|
||||||
<li>new: a new graph class JKQTPXYFunctionLineGraph draws parametric 2D curves ( \f$ [x,y] = f(t) \f$ ), see \ref JKQTPlotterEvalCurves for an example</li>
|
<li>new: a new graph class JKQTPXYFunctionLineGraph draws parametric 2D curves ( \f$ [x,y] = f(t) \f$ ), see \ref JKQTPlotterEvalCurves for an example</li>
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
# Example (JKQTPlotter): Plotting Parametric Mathematical Curves as Line Graphs {#JKQTPlotterEvalCurves}
|
# Example (JKQTPlotter): Plotting Parametric Mathematical Curves as Line Graphs {#JKQTPlotterEvalCurves}
|
||||||
## Basics
|
## Basics
|
||||||
This project (see `./examples/evalcurve/`) demonstrates how to plot mathematical functions as line graphs. The functions may be defined as static C functions, C++ functors or c++ inline functions.
|
This project (see `./examples/evalcurve/`) demonstrates how to plot mathematical functions as line graphs. The functions may be defined as static C functions, C++ functors or c++ inline functions. The functions may simply depend on the parameter `t`, or on `t` and a vector of parameters.
|
||||||
|
|
||||||
|
The class uses an adaptive algorithm, which determines by the local slope, at how many points (or how close points) the functor has to be evaluated.
|
||||||
|
|
||||||
[TOC]
|
[TOC]
|
||||||
|
|
||||||
@ -9,24 +11,55 @@ The example shows how to plot a simple C++ inline function:
|
|||||||
|
|
||||||
```.cpp
|
```.cpp
|
||||||
JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
||||||
func1->setPlotFunctionFunctor([](double t) -> QPointF {
|
func1->setPlotFunctionFunctor([](double t) ->QPointF {
|
||||||
const double a=5;
|
return QPointF(
|
||||||
const double b=4;
|
sin(t)*(exp(cos(t))-2.0*cos(4.0*t)-jkqtp_pow5(sin(t/12.0))),
|
||||||
const double delta=JKQTPSTATISTICS_PI/4.0;
|
cos(t)*(exp(cos(t))-2.0*cos(4.0*t)-jkqtp_pow5(sin(t/12.0)))
|
||||||
return QPointF(sin(a*t+delta), sin(b*t));
|
);
|
||||||
});
|
});
|
||||||
func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
func1->setTRange(0, 12.0*JKQTPSTATISTICS_PI);
|
||||||
func1->setTitle("C++-inline function $[ sin(5{\\cdot}t+\\pi/4), sin(4{\\cdot}t) ]$");
|
func1->setTitle("C++-inline function: \"Butterfly Curve\"");
|
||||||
plot->addGraph(func1);
|
plot->addGraph(func1);
|
||||||
```
|
```
|
||||||
|
|
||||||
Note that here a functor is defined, which calculates the points on a [Lissajous Curve](https://en.wikipedia.org/wiki/Lissajous_curve), i.e. a function mapping a parameter `t` to a two-dimensional point `QPointF` with `x=sin(a*t+delta)`and `y=sin(b*t)`. This function is evaluated on a range of values for `t`, set by
|
Note that here a functor is defined, which calculates the points on a [Butterfly Curve](https://en.wikipedia.org/wiki/Butterfly_curve_(transcendental)), i.e. a function mapping a parameter `t` to a two-dimensional point `QPointF` with complex functions for x and y. This function is evaluated on a range of values for `t`, set by
|
||||||
|
|
||||||
```.cpp
|
```.cpp
|
||||||
func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
||||||
```
|
```
|
||||||
|
|
||||||
The class uses an adaptive algorithm, which determines by the local slope, at how many points (or how close points) the functor has to be evaluated.
|
# Simple C++ inline function with Parameters
|
||||||
|
`JKQTPXYFunctionLineGraph` allows to use more complex functors alternatively: These depend on the variable `t`and a vector of parameters. Here is an example:
|
||||||
|
|
||||||
|
```.cpp
|
||||||
|
func2->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
|
||||||
|
func2->setParamsV(5, 4, JKQTPSTATISTICS_PI/4.0);
|
||||||
|
// and define the range over which to evaluate
|
||||||
|
func2->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
||||||
|
func1->setTitle("C++-inline function $[ sin(5{\\cdot}t+\\pi/4), sin(4{\\cdot}t) ]$");
|
||||||
|
plot->addGraph(func1);
|
||||||
|
```
|
||||||
|
|
||||||
|
In the functor we can use the entries in the vector `param` as function parameters. The values in this vector are defined by
|
||||||
|
|
||||||
|
```.cpp
|
||||||
|
// now we define the 3 parameters of the function
|
||||||
|
func2->setParamsV(5, 4, JKQTPSTATISTICS_PI/4.0);
|
||||||
|
```
|
||||||
|
|
||||||
|
Alternatively they can also be taken from a column in the internal datastore. Then you have to call:
|
||||||
|
|
||||||
|
```.cpp
|
||||||
|
func2->setParameterColumn(ColumnID);
|
||||||
|
```
|
||||||
|
|
||||||
|
instead, where `ColumnID` is the ID of the column with the parameter values.
|
||||||
|
|
||||||
# Screenshot
|
# Screenshot
|
||||||
|
|
||||||
@ -36,7 +69,6 @@ This code snippets above result in a plot like this:
|
|||||||
|
|
||||||
# Notes
|
# Notes
|
||||||
|
|
||||||
Just as shown in [examples/functionplot](https://github.com/jkriege2/JKQtPlotter/tree/master/examples/functionplot) for JKQTPXFunctionLineGraph and JKQTPYFunctionLineGraph, different types of functions can be used to plot. Either simple C++ inline functions, that take a `double t` and return a `QPointF`, or functions that additionally take a parameter vector `void* params`. In that case, the parameters may be provided from different sources, as described in [examples/functionplot](https://github.com/jkriege2/JKQtPlotter/tree/master/examples/functionplot) .
|
This example describes how to draw 2D parametric curves. For (simpler) 1D-functions f(x) or f(y), see [examples/functionplot](https://github.com/jkriege2/JKQtPlotter/tree/master/examples/functionplot) .
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -27,14 +27,34 @@ int main(int argc, char* argv[])
|
|||||||
// the function is defined as C++ inline function
|
// the function is defined as C++ inline function
|
||||||
JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
||||||
func1->setPlotFunctionFunctor([](double t) ->QPointF {
|
func1->setPlotFunctionFunctor([](double t) ->QPointF {
|
||||||
const double a=5;
|
return QPointF(
|
||||||
const double b=4;
|
sin(t)*(exp(cos(t))-2.0*cos(4.0*t)-jkqtp_pow5(sin(t/12.0))),
|
||||||
const double delta=JKQTPSTATISTICS_PI/4.0;
|
cos(t)*(exp(cos(t))-2.0*cos(4.0*t)-jkqtp_pow5(sin(t/12.0)))
|
||||||
return QPointF(sin(a*t+delta), sin(b*t));
|
);
|
||||||
});
|
});
|
||||||
func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
func1->setTRange(0, 12.0*JKQTPSTATISTICS_PI);
|
||||||
func1->setTitle("C++-inline function $[ sin(5{\\cdot}t+\\pi/4), sin(4{\\cdot}t) ]$");
|
func1->setTitle("C++-inline function: \"Butterfly Curve\"");
|
||||||
plot->addGraph(func1);
|
plot->addGraph(func1);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// 3. a second JKQTPXYFunctionLineGraph object shows how to use functions that
|
||||||
|
// also take a parameter vector, in addition to the dependent variable t
|
||||||
|
JKQTPXYFunctionLineGraph* func2=new JKQTPXYFunctionLineGraph(plot);
|
||||||
|
func2->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
|
||||||
|
func2->setParamsV(5, 4, JKQTPSTATISTICS_PI/4.0);
|
||||||
|
// and define the range over which to evaluate
|
||||||
|
func2->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
||||||
|
func2->setTitle("C++-inline function $[ sin(5{\\cdot}t+\\pi/4), sin(4{\\cdot}t) ]$");
|
||||||
|
plot->addGraph(func2);
|
||||||
|
|
||||||
|
|
||||||
// 8. set some axis properties (we use LaTeX for nice equation rendering)
|
// 8. set some axis properties (we use LaTeX for nice equation rendering)
|
||||||
plot->getXAxis()->setAxisLabel(QObject::tr("x-axis"));
|
plot->getXAxis()->setAxisLabel(QObject::tr("x-axis"));
|
||||||
plot->getYAxis()->setAxisLabel(QObject::tr("y-axis"));
|
plot->getYAxis()->setAxisLabel(QObject::tr("y-axis"));
|
||||||
@ -42,7 +62,7 @@ int main(int argc, char* argv[])
|
|||||||
|
|
||||||
|
|
||||||
// 4. scale the plot so the graph is contained
|
// 4. scale the plot so the graph is contained
|
||||||
plot->setXY(-1.1,1.1,-1.1,1.1);
|
plot->setXY(-3,12,-3.2,3.2);
|
||||||
|
|
||||||
// show window and make it a decent size
|
// show window and make it a decent size
|
||||||
mainWin.show();
|
mainWin.show();
|
||||||
|
@ -108,3 +108,6 @@ All examples above use the graph class `JKQTPXFunctionLineGraph`, which plots a
|
|||||||
![functionplot_fy](https://raw.githubusercontent.com/jkriege2/JKQtPlotter/master/screenshots/functionplot_fy.png)
|
![functionplot_fy](https://raw.githubusercontent.com/jkriege2/JKQtPlotter/master/screenshots/functionplot_fy.png)
|
||||||
|
|
||||||
|
|
||||||
|
This example describes how to draw 1D functions. For an example of how to draw 2D parametric curves `[x,y]=f(t)`, see [examples/evalcurve](https://github.com/jkriege2/JKQtPlotter/tree/master/examples/evalcurve) .
|
||||||
|
|
||||||
|
|
||||||
|
@ -290,12 +290,22 @@ inline T jkqtp_sqr(const T& v) {
|
|||||||
\ingroup jkqtptools_math_basic
|
\ingroup jkqtptools_math_basic
|
||||||
|
|
||||||
*/
|
*/
|
||||||
template <class T>
|
template <class T>
|
||||||
inline T jkqtp_pow4(T x) {
|
inline T jkqtp_pow4(T x) {
|
||||||
const T xx=x*x;
|
const T xx=x*x;
|
||||||
return xx*xx;
|
return xx*xx;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*! \brief 5-th power of a number
|
||||||
|
\ingroup jkqtptools_math_basic
|
||||||
|
|
||||||
|
*/
|
||||||
|
template <class T>
|
||||||
|
inline T jkqtp_pow5(T x) {
|
||||||
|
const T xx=x*x;
|
||||||
|
return xx*xx*x;
|
||||||
|
}
|
||||||
|
|
||||||
/*! \brief cube of a number
|
/*! \brief cube of a number
|
||||||
\ingroup jkqtptools_math_basic
|
\ingroup jkqtptools_math_basic
|
||||||
|
|
||||||
|
@ -31,29 +31,22 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase(JKQTBasePlotter* parent):
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphBase(parent),
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTBasePlotter* parent):
|
drawLine(true),
|
||||||
JKQTPEvaluatedFunctionWithParamsGraphBase(parent)
|
fillCurve(false),
|
||||||
|
drawErrorPolygons(false),
|
||||||
|
drawErrorLines(false)
|
||||||
{
|
{
|
||||||
functionType=SpecialFunction::UserFunction;
|
|
||||||
drawLine=true;
|
|
||||||
fillCurve=false;
|
|
||||||
|
|
||||||
initLineStyle(parent, parentPlotStyle);
|
initLineStyle(parent, parentPlotStyle);
|
||||||
initFillStyle(parent, parentPlotStyle);
|
initFillStyle(parent, parentPlotStyle);
|
||||||
|
|
||||||
drawErrorPolygons=false;
|
|
||||||
drawErrorLines=false;
|
|
||||||
errorParams=nullptr;
|
|
||||||
errorColor=getLineColor().lighter();
|
errorColor=getLineColor().lighter();
|
||||||
errorFillColor=getLineColor().lighter();
|
errorFillColor=getLineColor().lighter();
|
||||||
errorStyle=Qt::SolidLine;
|
errorStyle=Qt::SolidLine;
|
||||||
errorLineWidth=1;
|
errorLineWidth=1;
|
||||||
errorFillStyle=Qt::SolidPattern;
|
errorFillStyle=Qt::SolidPattern;
|
||||||
|
|
||||||
errorParameterColumn=-1;
|
|
||||||
|
|
||||||
|
|
||||||
if (parent && parentPlotStyle>=0) { // get style settings from parent object
|
if (parent && parentPlotStyle>=0) { // get style settings from parent object
|
||||||
//std::cout<<"got style settings from parent: "<<parentPlotStyle<<std::endl;
|
//std::cout<<"got style settings from parent: "<<parentPlotStyle<<std::endl;
|
||||||
@ -66,120 +59,29 @@ JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTBasePlotter* parent):
|
|||||||
errorFillColor.setAlphaF(0.5);
|
errorFillColor.setAlphaF(0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTPlotter* parent):
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase(JKQTPlotter* parent):
|
||||||
JKQTPXFunctionLineGraph(parent->getPlotter())
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase(parent->getPlotter())
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title_, JKQTBasePlotter *parent):
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::~JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase()
|
||||||
JKQTPXFunctionLineGraph(parent)
|
|
||||||
{
|
|
||||||
title=title_;
|
|
||||||
plotFunction=jkqtpPlotFunctionType();
|
|
||||||
simplePlotFunction=f;
|
|
||||||
functionType=SpecialFunction::UserFunction;
|
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title_, JKQTPlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(f, title_, parent->getPlotter())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title_, JKQTBasePlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(parent)
|
|
||||||
{
|
|
||||||
title=title_;
|
|
||||||
plotFunction=jkqtpPlotFunctionType();
|
|
||||||
simplePlotFunction=std::move(f);
|
|
||||||
functionType=SpecialFunction::UserFunction;
|
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title_, JKQTPlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(std::move(f), title_, parent->getPlotter())
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTPXFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title_, JKQTBasePlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(parent)
|
|
||||||
{
|
|
||||||
title=title_;
|
|
||||||
functionType=type;
|
|
||||||
setParams(params);
|
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTPXFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title, JKQTPlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(type, params, title, parent->getPlotter())
|
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::~JKQTPXFunctionLineGraph() {
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setDrawLine(bool __value)
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setDrawLine(bool __value)
|
|
||||||
{
|
{
|
||||||
this->drawLine = __value;
|
this->drawLine = __value;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool JKQTPXFunctionLineGraph::getDrawLine() const
|
bool JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getDrawLine() const
|
||||||
{
|
{
|
||||||
return this->drawLine;
|
return this->drawLine;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setPlotFunctionFunctor(const jkqtpPlotFunctionType &__value)
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::drawKeyMarker(JKQTPEnhancedPainter& painter, QRectF& rect) {
|
||||||
{
|
|
||||||
simplePlotFunction=jkqtpSimplePlotFunctionType();
|
|
||||||
plotFunction = __value;
|
|
||||||
functionType=SpecialFunction::UserFunction;
|
|
||||||
|
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setPlotFunctionFunctor(const jkqtpSimplePlotFunctionType &__value)
|
|
||||||
{
|
|
||||||
plotFunction=jkqtpPlotFunctionType();
|
|
||||||
simplePlotFunction=__value;
|
|
||||||
functionType=SpecialFunction::UserFunction;
|
|
||||||
|
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setPlotFunctionFunctor(jkqtpPlotFunctionType &&__value)
|
|
||||||
{
|
|
||||||
simplePlotFunction=jkqtpSimplePlotFunctionType();
|
|
||||||
plotFunction = std::move(__value);
|
|
||||||
functionType=SpecialFunction::UserFunction;
|
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setPlotFunctionFunctor(jkqtpSimplePlotFunctionType &&__value)
|
|
||||||
{
|
|
||||||
plotFunction=jkqtpPlotFunctionType();
|
|
||||||
simplePlotFunction=std::move(__value);
|
|
||||||
functionType=SpecialFunction::UserFunction;
|
|
||||||
|
|
||||||
data.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
jkqtpPlotFunctionType JKQTPXFunctionLineGraph::getPlotFunctionFunctor() const
|
|
||||||
{
|
|
||||||
return plotFunction;
|
|
||||||
}
|
|
||||||
|
|
||||||
jkqtpSimplePlotFunctionType JKQTPXFunctionLineGraph::getSimplePlotFunction() const
|
|
||||||
{
|
|
||||||
return simplePlotFunction;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::drawKeyMarker(JKQTPEnhancedPainter& painter, QRectF& rect) {
|
|
||||||
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
||||||
QPen p=getLinePen(painter, parent);
|
QPen p=getLinePen(painter, parent);
|
||||||
p.setJoinStyle(Qt::RoundJoin);
|
p.setJoinStyle(Qt::RoundJoin);
|
||||||
@ -195,73 +97,110 @@ void JKQTPXFunctionLineGraph::drawKeyMarker(JKQTPEnhancedPainter& painter, QRect
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QColor JKQTPXFunctionLineGraph::getKeyLabelColor() const {
|
QColor JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getKeyLabelColor() const {
|
||||||
return getLineColor();
|
return getLineColor();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool JKQTPXFunctionLineGraph::getXMinMax(double &minx, double &maxx, double &smallestGreaterZero)
|
|
||||||
{
|
QBrush JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getErrorBrush(JKQTPEnhancedPainter& /*painter*/) const {
|
||||||
smallestGreaterZero=minx=maxx=0; return false;
|
QBrush b;
|
||||||
|
b.setColor(errorFillColor);
|
||||||
|
b.setStyle(errorFillStyle);
|
||||||
|
return b;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool JKQTPXFunctionLineGraph::getYMinMax(double &miny, double &maxy, double &smallestGreaterZero)
|
QPen JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getErrorLinePen(JKQTPEnhancedPainter& painter) const {
|
||||||
{
|
QPen p;
|
||||||
smallestGreaterZero=miny=maxy=0; return false;
|
p.setColor(errorColor);
|
||||||
|
p.setWidthF(qMax(JKQTPlotterDrawingTools::ABS_MIN_LINEWIDTH,parent->pt2px(painter, parent->getLineWidthMultiplier()*errorLineWidth)));
|
||||||
|
p.setStyle(errorStyle);
|
||||||
|
p.setJoinStyle(Qt::RoundJoin);
|
||||||
|
p.setCapStyle(Qt::RoundCap);
|
||||||
|
|
||||||
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::createPlotData(bool collectParams) {
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setErrorLineColor(const QColor &__value)
|
||||||
|
{
|
||||||
|
this->errorColor = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
QColor JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getErrorLineColor() const
|
||||||
|
{
|
||||||
|
return this->errorColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setErrorFillColor(const QColor &__value)
|
||||||
|
{
|
||||||
|
this->errorFillColor = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
QColor JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getErrorFillColor() const
|
||||||
|
{
|
||||||
|
return this->errorFillColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setErrorFillStyle(Qt::BrushStyle __value)
|
||||||
|
{
|
||||||
|
this->errorFillStyle = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
Qt::BrushStyle JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getErrorFillStyle() const
|
||||||
|
{
|
||||||
|
return this->errorFillStyle;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setErrorLineStyle(Qt::PenStyle __value)
|
||||||
|
{
|
||||||
|
this->errorStyle = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
Qt::PenStyle JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getErrorLineStyle() const
|
||||||
|
{
|
||||||
|
return this->errorStyle;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setErrorLineWidth(double __value)
|
||||||
|
{
|
||||||
|
this->errorLineWidth = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
double JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getErrorLineWidth() const
|
||||||
|
{
|
||||||
|
return this->errorLineWidth;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setColor(QColor c)
|
||||||
|
{
|
||||||
|
setLineColor(c);
|
||||||
|
c.setAlphaF(0.5);
|
||||||
|
setHighlightingLineColor(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setDrawErrorPolygons(bool __value)
|
||||||
|
{
|
||||||
|
this->drawErrorPolygons = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getDrawErrorPolygons() const
|
||||||
|
{
|
||||||
|
return this->drawErrorPolygons;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::setDrawErrorLines(bool __value)
|
||||||
|
{
|
||||||
|
this->drawErrorLines = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::getDrawErrorLines() const
|
||||||
|
{
|
||||||
|
return this->drawErrorLines;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::drawXGraph(JKQTPEnhancedPainter& painter) {
|
||||||
#ifdef JKQTBP_AUTOTIMER
|
#ifdef JKQTBP_AUTOTIMER
|
||||||
JKQTPAutoOutputTimer jkaat(QString("JKQTPXFunctionLineGraph[%1]::createPlotData()").arg(title));
|
JKQTPAutoOutputTimer jkaaot("JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::drawXGraph");
|
||||||
#endif
|
|
||||||
data.clear();
|
|
||||||
if (collectParams) collectParameters();
|
|
||||||
|
|
||||||
if (parent==nullptr) return;
|
|
||||||
if (!plotFunction && !simplePlotFunction) return;
|
|
||||||
|
|
||||||
jkqtpSimplePlotFunctionType func;
|
|
||||||
if (plotFunction) func=std::bind(plotFunction, std::placeholders::_1, getInternalParams());
|
|
||||||
else if (simplePlotFunction) func=simplePlotFunction;
|
|
||||||
|
|
||||||
const double xmin=parent->getXMin();
|
|
||||||
const double xmax=parent->getXMax();
|
|
||||||
|
|
||||||
std::function<QPointF(double)> fTransformedFunc= std::bind([&](const JKQTPPlotElement* plot, double x) -> QPointF { return plot->transform(x, func(x)); }, this, std::placeholders::_1);
|
|
||||||
|
|
||||||
JKQTPAdaptiveFunctionGraphEvaluator evaluator(fTransformedFunc, minSamples, maxRefinementDegree, slopeTolerance, minPixelPerSample);
|
|
||||||
data=evaluator.evaluate(xmin, xmax);
|
|
||||||
data=JKQTPSimplyfyLineSegemnts(data, dataCleanupMaxAllowedAngleDegree);
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::collectParameters()
|
|
||||||
{
|
|
||||||
JKQTPEvaluatedFunctionWithParamsGraphBase::collectParameters();
|
|
||||||
|
|
||||||
if (parent && errorParameterColumn>=0) {
|
|
||||||
ierrorparams.clear();
|
|
||||||
JKQTPDatastore* datastore=parent->getDatastore();
|
|
||||||
int imin=0;
|
|
||||||
int imax= static_cast<int>(datastore->getRows(errorParameterColumn));
|
|
||||||
|
|
||||||
for (int i=imin; i<imax; i++) {
|
|
||||||
double xv=datastore->get(errorParameterColumn,i);
|
|
||||||
ierrorparams<<xv;
|
|
||||||
}
|
|
||||||
int i=ierrorparams.size()-1;
|
|
||||||
while (i>=0 && !JKQTPIsOKFloat(ierrorparams[i])) {
|
|
||||||
ierrorparams.remove(i,1);
|
|
||||||
i--;
|
|
||||||
}
|
|
||||||
|
|
||||||
errorParams=&ierrorparams;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|
||||||
#ifdef JKQTBP_AUTOTIMER
|
|
||||||
JKQTPAutoOutputTimer jkaaot("JKQTPXFunctionLineGraph::draw");
|
|
||||||
#endif
|
#endif
|
||||||
if (parent==nullptr) return;
|
if (parent==nullptr) return;
|
||||||
JKQTPDatastore* datastore=parent->getDatastore();
|
JKQTPDatastore* datastore=parent->getDatastore();
|
||||||
@ -271,6 +210,8 @@ void JKQTPXFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|||||||
createPlotData();
|
createPlotData();
|
||||||
//qDebug()<<"plot data created\n";
|
//qDebug()<<"plot data created\n";
|
||||||
|
|
||||||
|
auto errorPlotFunction=buildErrorFunctorSpec();
|
||||||
|
|
||||||
drawErrorsBefore(painter);
|
drawErrorsBefore(painter);
|
||||||
{
|
{
|
||||||
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
||||||
@ -308,11 +249,11 @@ void JKQTPXFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|||||||
|
|
||||||
//std::cout<<"(xv, yv) = ( "<<xv<<", "<<yv<<" )\n";
|
//std::cout<<"(xv, yv) = ( "<<xv<<", "<<yv<<" )\n";
|
||||||
if (JKQTPIsOKFloat(x) && JKQTPIsOKFloat(y)) {
|
if (JKQTPIsOKFloat(x) && JKQTPIsOKFloat(y)) {
|
||||||
double xv=backtransformX(x);
|
const double xv=backtransformX(x);
|
||||||
double yv=backtransformY(y);
|
const double yv=backtransformY(y);
|
||||||
double ype=0, yme=0;
|
double ype=0, yme=0;
|
||||||
if ((drawErrorLines || drawErrorPolygons) && (static_cast<bool>(errorPlotFunction))) {
|
if ((drawErrorLines || drawErrorPolygons) && (static_cast<bool>(errorPlotFunction))) {
|
||||||
double e=errorPlotFunction(xv, getInternalErrorParams());
|
const double e=errorPlotFunction(xv).y();
|
||||||
ype=transformY(yv+e);
|
ype=transformY(yv+e);
|
||||||
yme=transformY(yv-e);
|
yme=transformY(yv-e);
|
||||||
ype=qBound(yami, ype, yama);
|
ype=qBound(yami, ype, yama);
|
||||||
@ -384,73 +325,9 @@ void JKQTPXFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|||||||
//std::cout<<"plot done\n";
|
//std::cout<<"plot done\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase::drawYGraph(JKQTPEnhancedPainter& painter) {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTBasePlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(parent)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTPlotter *parent):
|
|
||||||
JKQTPYFunctionLineGraph(parent->getPlotter())
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title, JKQTBasePlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(f, title, parent)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title, JKQTPlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(f, title, parent)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title, JKQTBasePlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(std::move(f), title, parent)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title, JKQTPlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(std::move(f), title, parent)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTPYFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title_, JKQTBasePlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(type, params, title_, parent)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTPYFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title_, JKQTPlotter *parent):
|
|
||||||
JKQTPXFunctionLineGraph(type, params, title_, parent->getPlotter())
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPYFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|
||||||
#ifdef JKQTBP_AUTOTIMER
|
#ifdef JKQTBP_AUTOTIMER
|
||||||
JKQTPAutoOutputTimer jkaaot("JKQTPYFunctionLineGraph::draw");
|
JKQTPAutoOutputTimer jkaaot("JKQTPYFunctionLineGraph::drawYGraph");
|
||||||
#endif
|
#endif
|
||||||
if (parent==nullptr) return;
|
if (parent==nullptr) return;
|
||||||
JKQTPDatastore* datastore=parent->getDatastore();
|
JKQTPDatastore* datastore=parent->getDatastore();
|
||||||
@ -460,6 +337,8 @@ void JKQTPYFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|||||||
createPlotData();
|
createPlotData();
|
||||||
//std::cout<<"plot data created\n";
|
//std::cout<<"plot data created\n";
|
||||||
|
|
||||||
|
auto errorPlotFunction=buildErrorFunctorSpec();
|
||||||
|
|
||||||
drawErrorsBefore(painter);
|
drawErrorsBefore(painter);
|
||||||
{
|
{
|
||||||
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
||||||
@ -497,11 +376,11 @@ void JKQTPYFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|||||||
|
|
||||||
//std::cout<<"(xv, yv) = ( "<<xv<<", "<<yv<<" )\n";
|
//std::cout<<"(xv, yv) = ( "<<xv<<", "<<yv<<" )\n";
|
||||||
if (JKQTPIsOKFloat(x) && JKQTPIsOKFloat(y)) {
|
if (JKQTPIsOKFloat(x) && JKQTPIsOKFloat(y)) {
|
||||||
double xv=backtransformX(x);
|
const double xv=backtransformX(x);
|
||||||
double yv=backtransformY(y);
|
const double yv=backtransformY(y);
|
||||||
double xpe=0, xme=0;
|
double xpe=0, xme=0;
|
||||||
if ((drawErrorLines || drawErrorPolygons) && (static_cast<bool>(errorPlotFunction))) {
|
if ((drawErrorLines || drawErrorPolygons) && (static_cast<bool>(errorPlotFunction))) {
|
||||||
double e=errorPlotFunction(yv, getInternalErrorParams());
|
const double e=errorPlotFunction(yv).x();
|
||||||
xpe=transformX(xv+e);
|
xpe=transformX(xv+e);
|
||||||
xme=transformX(xv-e);
|
xme=transformX(xv-e);
|
||||||
xpe=qBound(xami, xpe, xama);
|
xpe=qBound(xami, xpe, xama);
|
||||||
@ -573,124 +452,80 @@ void JKQTPYFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void JKQTPYFunctionLineGraph::createPlotData(bool collectParams) {
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(JKQTBasePlotter* parent):
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase(parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(JKQTPlotter* parent):
|
||||||
|
JKQTPFunctorLineGraphBase(parent->getPlotter())
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(const jkqtpSimplePlotFunctionType &f, const QString &title_, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(parent)
|
||||||
|
{
|
||||||
|
title=title_;
|
||||||
|
plotFunction=jkqtpPlotFunctionType();
|
||||||
|
simplePlotFunction=f;
|
||||||
data.clear();
|
data.clear();
|
||||||
if (collectParams) collectParameters();
|
|
||||||
|
|
||||||
if (parent==nullptr) return;
|
|
||||||
if (!plotFunction && !simplePlotFunction) return;
|
|
||||||
|
|
||||||
jkqtpSimplePlotFunctionType func;
|
|
||||||
if (plotFunction) func=std::bind(plotFunction, std::placeholders::_1, getInternalParams());
|
|
||||||
else if (simplePlotFunction) func=simplePlotFunction;
|
|
||||||
|
|
||||||
const double ymin=parent->getYMin();
|
|
||||||
const double ymax=parent->getYMax();
|
|
||||||
|
|
||||||
std::function<QPointF(double)> fTransformedFunc= std::bind([&](const JKQTPPlotElement* plot, double y) -> QPointF { return plot->transform(func(y), y); }, this, std::placeholders::_1);
|
|
||||||
|
|
||||||
JKQTPAdaptiveFunctionGraphEvaluator evaluator(fTransformedFunc, minSamples, maxRefinementDegree, slopeTolerance, minPixelPerSample);
|
|
||||||
data=evaluator.evaluate(ymin, ymax);
|
|
||||||
data=JKQTPSimplyfyLineSegemnts(data, dataCleanupMaxAllowedAngleDegree);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(const jkqtpSimplePlotFunctionType &f, const QString &title_, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(f, title_, parent->getPlotter())
|
||||||
|
|
||||||
QBrush JKQTPXFunctionLineGraph::getErrorBrush(JKQTPEnhancedPainter& /*painter*/) const {
|
|
||||||
QBrush b;
|
|
||||||
b.setColor(errorFillColor);
|
|
||||||
b.setStyle(errorFillStyle);
|
|
||||||
return b;
|
|
||||||
}
|
|
||||||
|
|
||||||
QPen JKQTPXFunctionLineGraph::getErrorLinePen(JKQTPEnhancedPainter& painter) const {
|
|
||||||
QPen p;
|
|
||||||
p.setColor(errorColor);
|
|
||||||
p.setWidthF(qMax(JKQTPlotterDrawingTools::ABS_MIN_LINEWIDTH,parent->pt2px(painter, parent->getLineWidthMultiplier()*errorLineWidth)));
|
|
||||||
p.setStyle(errorStyle);
|
|
||||||
p.setJoinStyle(Qt::RoundJoin);
|
|
||||||
p.setCapStyle(Qt::RoundCap);
|
|
||||||
|
|
||||||
return p;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorParams(const QVector<double> &errorParams)
|
|
||||||
{
|
{
|
||||||
ierrorparams=errorParams;
|
|
||||||
setErrorParams(&ierrorparams);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorParameterColumn(int __value)
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(jkqtpSimplePlotFunctionType &&f, const QString &title_, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(parent)
|
||||||
{
|
{
|
||||||
this->errorParameterColumn = __value;
|
title=title_;
|
||||||
|
plotFunction=jkqtpPlotFunctionType();
|
||||||
|
simplePlotFunction=std::move(f);
|
||||||
|
data.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
int JKQTPXFunctionLineGraph::getErrorParameterColumn() const
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(jkqtpSimplePlotFunctionType &&f, const QString &title_, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(std::move(f), title_, parent->getPlotter())
|
||||||
{
|
{
|
||||||
return this->errorParameterColumn;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorParameterColumn(size_t __value) {
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(JKQTPFunctorLineGraphBase::SpecialFunction type, const QVector<double> ¶ms, const QString &title_, JKQTBasePlotter *parent):
|
||||||
this->errorParameterColumn = static_cast<int>(__value);
|
JKQTPFunctorLineGraphBase(parent)
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorLineColor(const QColor &__value)
|
|
||||||
{
|
{
|
||||||
this->errorColor = __value;
|
title=title_;
|
||||||
|
setSpecialFunction(type);
|
||||||
|
setParams(params);
|
||||||
|
data.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
QColor JKQTPXFunctionLineGraph::getErrorLineColor() const
|
JKQTPFunctorLineGraphBase::JKQTPFunctorLineGraphBase(JKQTPFunctorLineGraphBase::SpecialFunction type, const QVector<double> ¶ms, const QString &title, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(type, params, title, parent->getPlotter())
|
||||||
{
|
{
|
||||||
return this->errorColor;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorFillColor(const QColor &__value)
|
|
||||||
{
|
JKQTPFunctorLineGraphBase::~JKQTPFunctorLineGraphBase() {
|
||||||
this->errorFillColor = __value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QColor JKQTPXFunctionLineGraph::getErrorFillColor() const
|
void JKQTPFunctorLineGraphBase::setSpecialFunction(JKQTPFunctorLineGraphBase::SpecialFunction function)
|
||||||
{
|
{
|
||||||
return this->errorFillColor;
|
if (function==JKQTPFunctorLineGraphBase::Polynomial) {
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorFillStyle(Qt::BrushStyle __value)
|
|
||||||
{
|
|
||||||
this->errorFillStyle = __value;
|
|
||||||
}
|
|
||||||
|
|
||||||
Qt::BrushStyle JKQTPXFunctionLineGraph::getErrorFillStyle() const
|
|
||||||
{
|
|
||||||
return this->errorFillStyle;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorLineStyle(Qt::PenStyle __value)
|
|
||||||
{
|
|
||||||
this->errorStyle = __value;
|
|
||||||
}
|
|
||||||
|
|
||||||
Qt::PenStyle JKQTPXFunctionLineGraph::getErrorLineStyle() const
|
|
||||||
{
|
|
||||||
return this->errorStyle;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorLineWidth(double __value)
|
|
||||||
{
|
|
||||||
this->errorLineWidth = __value;
|
|
||||||
}
|
|
||||||
|
|
||||||
double JKQTPXFunctionLineGraph::getErrorLineWidth() const
|
|
||||||
{
|
|
||||||
return this->errorLineWidth;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setSpecialFunction(JKQTPXFunctionLineGraph::SpecialFunction function)
|
|
||||||
{
|
|
||||||
if (function==JKQTPXFunctionLineGraph::Polynomial) {
|
|
||||||
setPlotFunctionFunctor([](double x, const QVector<double>& param) {
|
setPlotFunctionFunctor([](double x, const QVector<double>& param) {
|
||||||
double res=0;
|
double res=0;
|
||||||
if (param.size()>0) {
|
if (param.size()>0) {
|
||||||
@ -705,7 +540,7 @@ void JKQTPXFunctionLineGraph::setSpecialFunction(JKQTPXFunctionLineGraph::Specia
|
|||||||
return res;
|
return res;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else if (function==JKQTPXFunctionLineGraph::Exponential) setPlotFunctionFunctor([](double x, const QVector<double>& param) {
|
else if (function==JKQTPFunctorLineGraphBase::Exponential) setPlotFunctionFunctor([](double x, const QVector<double>& param) {
|
||||||
double res=0;
|
double res=0;
|
||||||
if (param.size()>=3) {
|
if (param.size()>=3) {
|
||||||
res=param.value(0,0)+param.value(1,0)*exp(x/param.value(2,0));
|
res=param.value(0,0)+param.value(1,0)*exp(x/param.value(2,0));
|
||||||
@ -714,7 +549,7 @@ void JKQTPXFunctionLineGraph::setSpecialFunction(JKQTPXFunctionLineGraph::Specia
|
|||||||
}
|
}
|
||||||
return res;
|
return res;
|
||||||
});
|
});
|
||||||
else if (function==JKQTPXFunctionLineGraph::PowerLaw) setPlotFunctionFunctor([](double x, const QVector<double>& param) {
|
else if (function==JKQTPFunctorLineGraphBase::PowerLaw) setPlotFunctionFunctor([](double x, const QVector<double>& param) {
|
||||||
double res=0;
|
double res=0;
|
||||||
if (param.size()>=3) {
|
if (param.size()>=3) {
|
||||||
res=param.value(0,0)+param.value(1,0)*pow(x, param.value(2,1));
|
res=param.value(0,0)+param.value(1,0)*pow(x, param.value(2,1));
|
||||||
@ -728,84 +563,279 @@ void JKQTPXFunctionLineGraph::setSpecialFunction(JKQTPXFunctionLineGraph::Specia
|
|||||||
else throw std::runtime_error("unknown special function type");
|
else throw std::runtime_error("unknown special function type");
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXFunctionLineGraph::SpecialFunction JKQTPXFunctionLineGraph::getFunctionType() const
|
|
||||||
{
|
|
||||||
return functionType;
|
|
||||||
}
|
|
||||||
|
|
||||||
QVector<double> JKQTPXFunctionLineGraph::getInternalErrorParams() const {
|
void JKQTPFunctorLineGraphBase::setErrorPlotFunction(const jkqtpPlotFunctionType &__value)
|
||||||
return ierrorparams;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setDrawErrorPolygons(bool __value)
|
|
||||||
{
|
|
||||||
this->drawErrorPolygons = __value;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool JKQTPXFunctionLineGraph::getDrawErrorPolygons() const
|
|
||||||
{
|
|
||||||
return this->drawErrorPolygons;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setDrawErrorLines(bool __value)
|
|
||||||
{
|
|
||||||
this->drawErrorLines = __value;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool JKQTPXFunctionLineGraph::getDrawErrorLines() const
|
|
||||||
{
|
|
||||||
return this->drawErrorLines;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorPlotFunction(const jkqtpPlotFunctionType &__value)
|
|
||||||
{
|
{
|
||||||
errorSimplePlotFunction=jkqtpSimplePlotFunctionType();
|
errorSimplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||||
errorPlotFunction=__value;
|
errorPlotFunction=__value;
|
||||||
data.clear();
|
data.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorPlotFunction(jkqtpPlotFunctionType &&__value)
|
void JKQTPFunctorLineGraphBase::setErrorPlotFunction(jkqtpPlotFunctionType &&__value)
|
||||||
{
|
{
|
||||||
errorSimplePlotFunction=jkqtpSimplePlotFunctionType();
|
errorSimplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||||
errorPlotFunction = std::move(__value);
|
errorPlotFunction = std::move(__value);
|
||||||
data.clear();
|
data.clear();
|
||||||
}
|
}
|
||||||
jkqtpPlotFunctionType JKQTPXFunctionLineGraph::getErrorPlotFunction() const
|
jkqtpPlotFunctionType JKQTPFunctorLineGraphBase::getErrorPlotFunction() const
|
||||||
{
|
{
|
||||||
return errorPlotFunction;
|
return errorPlotFunction;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorPlotFunction(const jkqtpSimplePlotFunctionType &__value)
|
void JKQTPFunctorLineGraphBase::setErrorPlotFunction(const jkqtpSimplePlotFunctionType &__value)
|
||||||
{
|
{
|
||||||
errorPlotFunction=jkqtpPlotFunctionType();
|
errorPlotFunction=jkqtpPlotFunctionType();
|
||||||
errorSimplePlotFunction=__value;
|
errorSimplePlotFunction=__value;
|
||||||
data.clear();
|
data.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorPlotFunction(jkqtpSimplePlotFunctionType &&__value)
|
void JKQTPFunctorLineGraphBase::setErrorPlotFunction(jkqtpSimplePlotFunctionType &&__value)
|
||||||
{
|
{
|
||||||
errorPlotFunction=jkqtpPlotFunctionType();
|
errorPlotFunction=jkqtpPlotFunctionType();
|
||||||
errorSimplePlotFunction = std::move(__value);
|
errorSimplePlotFunction = std::move(__value);
|
||||||
data.clear();
|
data.clear();
|
||||||
}
|
}
|
||||||
jkqtpSimplePlotFunctionType JKQTPXFunctionLineGraph::getErrorSimplePlotFunction() const
|
|
||||||
|
jkqtpSimplePlotFunctionType JKQTPFunctorLineGraphBase::getErrorSimplePlotFunction() const
|
||||||
{
|
{
|
||||||
return errorSimplePlotFunction;
|
return errorSimplePlotFunction;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXFunctionLineGraph::setErrorParams(void *__value)
|
void JKQTPFunctorLineGraphBase::setPlotFunctionFunctor(const jkqtpPlotFunctionType &__value)
|
||||||
{
|
{
|
||||||
this->errorParams = __value;
|
simplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||||
|
plotFunction = __value;
|
||||||
|
data.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void *JKQTPXFunctionLineGraph::getErrorParams() const
|
void JKQTPFunctorLineGraphBase::setPlotFunctionFunctor(const jkqtpSimplePlotFunctionType &__value)
|
||||||
{
|
{
|
||||||
return this->errorParams;
|
plotFunction=jkqtpPlotFunctionType();
|
||||||
|
simplePlotFunction=__value;
|
||||||
|
data.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPFunctorLineGraphBase::setPlotFunctionFunctor(jkqtpPlotFunctionType &&__value)
|
||||||
|
{
|
||||||
|
simplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||||
|
plotFunction = std::move(__value);
|
||||||
|
data.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPFunctorLineGraphBase::setPlotFunctionFunctor(jkqtpSimplePlotFunctionType &&__value)
|
||||||
|
{
|
||||||
|
plotFunction=jkqtpPlotFunctionType();
|
||||||
|
simplePlotFunction=std::move(__value);
|
||||||
|
data.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
jkqtpPlotFunctionType JKQTPFunctorLineGraphBase::getPlotFunctionFunctor() const
|
||||||
|
{
|
||||||
|
return plotFunction;
|
||||||
|
}
|
||||||
|
|
||||||
|
jkqtpSimplePlotFunctionType JKQTPFunctorLineGraphBase::getSimplePlotFunction() const
|
||||||
|
{
|
||||||
|
return simplePlotFunction;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JKQTPFunctorLineGraphBase::isSimplePlotFunction() const
|
||||||
|
{
|
||||||
|
return !static_cast<bool>(plotFunction) && static_cast<bool>(simplePlotFunction);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool JKQTPXFunctionLineGraph::usesColumn(int c) const
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTBasePlotter* parent):
|
||||||
|
JKQTPFunctorLineGraphBase(parent)
|
||||||
{
|
{
|
||||||
return JKQTPEvaluatedFunctionWithParamsGraphBase::usesColumn(c)||(c==errorParameterColumn);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTPlotter* parent):
|
||||||
|
JKQTPFunctorLineGraphBase(parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title_, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(f, title_, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title_, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(f, title_, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title_, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(std::move(f), title_, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title_, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(std::move(f), title_, parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTPXFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title_, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(type, params, title_, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::JKQTPXFunctionLineGraph(JKQTPXFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(type, params, title, parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPXFunctionLineGraph::~JKQTPXFunctionLineGraph() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPXFunctionLineGraph::draw(JKQTPEnhancedPainter &painter)
|
||||||
|
{
|
||||||
|
drawXGraph(painter);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec JKQTPXFunctionLineGraph::buildPlotFunctorSpec()
|
||||||
|
{
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec spec;
|
||||||
|
|
||||||
|
if (parent==nullptr) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
if (!plotFunction && !simplePlotFunction) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
|
||||||
|
// range over which to evaluate func
|
||||||
|
spec.range_start=parent->getXMin();
|
||||||
|
spec.range_end=parent->getXMax();
|
||||||
|
|
||||||
|
// the actual function to use
|
||||||
|
if (plotFunction) spec.func=std::bind([&](double x, const QVector<double>& p) -> QPointF { return QPointF(x, plotFunction(x, p)); }, std::placeholders::_1, getInternalParams());
|
||||||
|
else if (simplePlotFunction) spec.func=[&](double x) -> QPointF { return QPointF(x, simplePlotFunction(x)); };
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::function<QPointF (double)> JKQTPXFunctionLineGraph::buildErrorFunctorSpec()
|
||||||
|
{
|
||||||
|
std::function<QPointF (double)> spec;
|
||||||
|
if (parent==nullptr) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
if (!plotFunction && !simplePlotFunction) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
|
||||||
|
// the actual function to use
|
||||||
|
if (errorPlotFunction) spec=std::bind([&](double x, const QVector<double>& p) -> QPointF { return QPointF(0, errorPlotFunction(x, p)); }, std::placeholders::_1, getInternalParams());
|
||||||
|
else if (errorSimplePlotFunction) spec=[&](double x) -> QPointF { return QPointF(0, errorSimplePlotFunction(x)); };
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTPlotter *parent):
|
||||||
|
JKQTPYFunctionLineGraph(parent->getPlotter())
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(f, title, parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(const jkqtpSimplePlotFunctionType &f, const QString &title, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(f, title, parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(std::move(f), title, parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(jkqtpSimplePlotFunctionType &&f, const QString &title, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(std::move(f), title, parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTPYFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title_, JKQTBasePlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(type, params, title_, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPYFunctionLineGraph::JKQTPYFunctionLineGraph(JKQTPYFunctionLineGraph::SpecialFunction type, const QVector<double> ¶ms, const QString &title_, JKQTPlotter *parent):
|
||||||
|
JKQTPFunctorLineGraphBase(type, params, title_, parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPYFunctionLineGraph::draw(JKQTPEnhancedPainter &painter)
|
||||||
|
{
|
||||||
|
drawYGraph(painter);
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec JKQTPYFunctionLineGraph::buildPlotFunctorSpec()
|
||||||
|
{
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec spec;
|
||||||
|
|
||||||
|
if (parent==nullptr) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
if (!plotFunction && !simplePlotFunction) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
|
||||||
|
// range over which to evaluate func
|
||||||
|
spec.range_start=parent->getYMin();
|
||||||
|
spec.range_end=parent->getYMax();
|
||||||
|
|
||||||
|
// the actual function to use
|
||||||
|
if (plotFunction) spec.func=std::bind([&](double y, const QVector<double>& p) -> QPointF { return QPointF(plotFunction(y, p), y); }, std::placeholders::_1, getInternalParams());
|
||||||
|
else if (simplePlotFunction) spec.func=[&](double y) -> QPointF { return QPointF(simplePlotFunction(y), y); };
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::function<QPointF (double)> JKQTPYFunctionLineGraph::buildErrorFunctorSpec()
|
||||||
|
{
|
||||||
|
std::function<QPointF (double)> spec;
|
||||||
|
if (parent==nullptr) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
if (!plotFunction && !simplePlotFunction) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
|
||||||
|
// the actual function to use
|
||||||
|
if (errorPlotFunction) spec=std::bind([&](double y, const QVector<double>& p) -> QPointF { return QPointF(errorPlotFunction(y, p), 0); }, std::placeholders::_1, getInternalParams());
|
||||||
|
else if (errorSimplePlotFunction) spec=[&](double y) -> QPointF { return QPointF(errorSimplePlotFunction(y), 0); };
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -33,6 +33,118 @@
|
|||||||
#define jkqtpgraphsevaluatedfunction_H
|
#define jkqtpgraphsevaluatedfunction_H
|
||||||
|
|
||||||
|
|
||||||
|
/** \brief This class extends JKQTPEvaluatedFunctionWithErrorsGraphBase with functions to draw the graphs and
|
||||||
|
* set the drawing style
|
||||||
|
* \ingroup jkqtplotter_functiongraphs
|
||||||
|
*
|
||||||
|
* \note Since this class is meant as a base for both f(x)- and f(y)-functions, it cannot
|
||||||
|
* implememt JKQTPGraph::draw(). Therefore it provides two implementations drawXGraph()
|
||||||
|
* and drawYGraph() and the user has to decide in the concrete graph class, which one to call
|
||||||
|
* (or whether to do the drawing completely different).
|
||||||
|
*
|
||||||
|
* \see e.g. JKQTPXFunctionLineGraph for a concrete implementation
|
||||||
|
*/
|
||||||
|
class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase: public JKQTPEvaluatedFunctionWithErrorsGraphBase, public JKQTPGraphLineStyleMixin, public JKQTPGraphFillStyleMixin {
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase(JKQTBasePlotter* parent=nullptr);
|
||||||
|
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase(JKQTPlotter* parent);
|
||||||
|
|
||||||
|
/** \brief class destructor */
|
||||||
|
virtual ~JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase() override;
|
||||||
|
|
||||||
|
/** \brief plots a key marker inside the specified rectangle \a rect */
|
||||||
|
virtual void drawKeyMarker(JKQTPEnhancedPainter& painter, QRectF& rect) override;
|
||||||
|
/** \brief returns the color to be used for the key label */
|
||||||
|
virtual QColor getKeyLabelColor() const override;
|
||||||
|
|
||||||
|
/*! \copydoc drawLine */
|
||||||
|
bool getDrawLine() const;
|
||||||
|
|
||||||
|
|
||||||
|
/*! \copydoc drawErrorPolygons */
|
||||||
|
bool getDrawErrorPolygons() const;
|
||||||
|
/*! \copydoc drawErrorLines */
|
||||||
|
bool getDrawErrorLines() const;
|
||||||
|
/*! \copydoc errorColor */
|
||||||
|
virtual QColor getErrorLineColor() const;
|
||||||
|
/*! \copydoc errorFillColor */
|
||||||
|
virtual QColor getErrorFillColor() const;
|
||||||
|
/*! \copydoc errorFillStyle */
|
||||||
|
virtual Qt::BrushStyle getErrorFillStyle() const;
|
||||||
|
/*! \copydoc errorStyle */
|
||||||
|
virtual Qt::PenStyle getErrorLineStyle() const;
|
||||||
|
/*! \copydoc errorLineWidth */
|
||||||
|
virtual double getErrorLineWidth() const;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
/*! \brief set color, fill color and error color at the same time */
|
||||||
|
void setColor(QColor c);
|
||||||
|
|
||||||
|
/*! \copydoc drawLine */
|
||||||
|
void setDrawLine(bool __value);
|
||||||
|
/*! \copydoc drawErrorPolygons */
|
||||||
|
void setDrawErrorPolygons(bool __value);
|
||||||
|
/*! \copydoc drawErrorLines */
|
||||||
|
void setDrawErrorLines(bool __value);
|
||||||
|
/*! \copydoc errorColor */
|
||||||
|
virtual void setErrorLineColor(const QColor & __value);
|
||||||
|
/*! \copydoc errorFillColor */
|
||||||
|
virtual void setErrorFillColor(const QColor & __value);
|
||||||
|
/*! \copydoc errorFillStyle */
|
||||||
|
virtual void setErrorFillStyle(Qt::BrushStyle __value);
|
||||||
|
/*! \copydoc errorStyle */
|
||||||
|
virtual void setErrorLineStyle(Qt::PenStyle __value);
|
||||||
|
/*! \copydoc errorLineWidth */
|
||||||
|
virtual void setErrorLineWidth(double __value);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/** \brief plots the graph to the plotter object specified as parent */
|
||||||
|
void drawXGraph(JKQTPEnhancedPainter& painter);
|
||||||
|
/** \brief plots the graph to the plotter object specified as parent */
|
||||||
|
void drawYGraph(JKQTPEnhancedPainter& painter);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/** \brief indicates whether to draw a line or not */
|
||||||
|
bool drawLine;
|
||||||
|
/** \brief indicates whether to fill the space between the curve and the x-axis */
|
||||||
|
bool fillCurve;
|
||||||
|
|
||||||
|
|
||||||
|
/** \brief indicates whether an error polygon should be drawn */
|
||||||
|
bool drawErrorPolygons;
|
||||||
|
/** \brief indicates whether error lines should be drawn */
|
||||||
|
bool drawErrorLines;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/** \brief color of the error graph */
|
||||||
|
QColor errorColor;
|
||||||
|
/** \brief color of the error graph fill */
|
||||||
|
QColor errorFillColor;
|
||||||
|
/** \brief linestyle of the error graph lines */
|
||||||
|
Qt::PenStyle errorStyle;
|
||||||
|
/** \brief width (pixels) of the error graph */
|
||||||
|
double errorLineWidth;
|
||||||
|
/** \brief fill style, if the error curve should be filled */
|
||||||
|
Qt::BrushStyle errorFillStyle;
|
||||||
|
|
||||||
|
|
||||||
|
QBrush getErrorBrush(JKQTPEnhancedPainter& painter) const;
|
||||||
|
QPen getErrorLinePen(JKQTPEnhancedPainter &painter) const;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*! \brief type of functions that may be plotted by JKQTPXFunctionLineGraph and JKQTPYFunctionLineGraph
|
/*! \brief type of functions that may be plotted by JKQTPXFunctionLineGraph and JKQTPYFunctionLineGraph
|
||||||
@ -54,84 +166,43 @@ typedef std::function<double(double, const QVector<double>&)> jkqtpPlotFunctionT
|
|||||||
typedef std::function<double(double)> jkqtpSimplePlotFunctionType;
|
typedef std::function<double(double)> jkqtpSimplePlotFunctionType;
|
||||||
|
|
||||||
|
|
||||||
/*! \brief This implements line plots where the data is taken from a user supplied function \f$ y=f(x) \f$
|
|
||||||
\ingroup jkqtplotter_functiongraphs
|
|
||||||
|
|
||||||
This class uses the intelligent plotting algorithm for functions, implemented in JKQTPAdaptiveFunctionGraphEvaluator.
|
/** \brief extends JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase with the capabilities to define functions from C++-functors
|
||||||
It starts by sampling the function at minSamples positions. Then each function interval is bisected recursively if
|
* of type jkqtpSimplePlotFunctionType or jkqtpPlotFunctionType
|
||||||
necessary. To do so the function is evaluated at the mid point and the slopes \f$ \alpha_{\mbox{left}} \f$
|
* \ingroup jkqtplotter_functiongraphs
|
||||||
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]
|
* \see JKQTPXFunctionLineGraph and JKQTPYFunctionLineGraph for a concrete implementation
|
||||||
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.
|
|
||||||
|
|
||||||
The following image shows some example graphs:
|
|
||||||
|
|
||||||
\image html plot_functionplots.png
|
|
||||||
|
|
||||||
\see \ref JKQTPlotterFunctionPlots, JKQTPAdaptiveFunctionGraphEvaluator, JKQTPYFunctionLineGraph, JKQTPXYFunctionLineGraph, jkqtpstatAddPolyFit(), jkqtpstatAddWeightedRegression(), jkqtpstatAddRobustIRLSRegression(), jkqtpstatAddRegression(), jkqtpstatAddLinearWeightedRegression(), jkqtpstatAddRobustIRLSLinearRegression(), jkqtpstatAddLinearRegression()
|
|
||||||
*/
|
*/
|
||||||
class JKQTPLOTTER_LIB_EXPORT JKQTPXFunctionLineGraph: public JKQTPEvaluatedFunctionWithParamsGraphBase, public JKQTPGraphLineStyleMixin, public JKQTPGraphFillStyleMixin {
|
class JKQTPLOTTER_LIB_EXPORT JKQTPFunctorLineGraphBase: public JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
|
|
||||||
enum SpecialFunction {
|
enum SpecialFunction {
|
||||||
Polynomial, /*!< \brief a polynomial \f$ f(x)=p_0+p_1x+p_2x^2+p_3x^3+... \f$ The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
Polynomial, /*!< \brief a polynomial \f$ f(x)=p_0+p_1x+p_2x^2+p_3x^3+... \f$ The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
||||||
Line=Polynomial, /*!< \brief a polynomial \f$ f(x)=p_0+p_1x \f$ The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
Line=Polynomial, /*!< \brief a polynomial \f$ f(x)=p_0+p_1x \f$ The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
||||||
Exponential, /*!< \brief an exponential function \f$ f(x)=p_0+p_1\cdot\exp(x/p_2) \f$ or \f$ f(x)=p_0\cdot\exp(x/p_1) \f$ (depending on the number of parameters). The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
Exponential, /*!< \brief an exponential function \f$ f(x)=p_0+p_1\cdot\exp(x/p_2) \f$ or \f$ f(x)=p_0\cdot\exp(x/p_1) \f$ (depending on the number of parameters). The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
||||||
PowerLaw, /*!< \brief an exponential function \f$ f(x)=p_0+p_1\cdot x^{p_3} \f$ or \f$ f(x)=p_0\cdot x^{p_1} \f$ or \f$ f(x)= x^{p_0} \f$ (depending on the number of parameters) The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
PowerLaw, /*!< \brief an exponential function \f$ f(x)=p_0+p_1\cdot x^{p_3} \f$ or \f$ f(x)=p_0\cdot x^{p_1} \f$ or \f$ f(x)= x^{p_0} \f$ (depending on the number of parameters) The parameters \a params have to be point to a QVector<double> and contain the parameters \f$ p_0, p_1, ... \f$ */
|
||||||
|
|
||||||
UserFunction, /*!< \brief no special function but the function is provided by the user */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(JKQTBasePlotter* parent=nullptr);
|
JKQTPFunctorLineGraphBase(JKQTBasePlotter* parent=nullptr);
|
||||||
|
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(JKQTPlotter* parent);
|
JKQTPFunctorLineGraphBase(JKQTPlotter* parent);
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType & f, const QString& title, JKQTBasePlotter* parent=nullptr);
|
JKQTPFunctorLineGraphBase(const jkqtpSimplePlotFunctionType & f, const QString& title, JKQTBasePlotter* parent=nullptr);
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType & f, const QString& title, JKQTPlotter* parent);
|
JKQTPFunctorLineGraphBase(const jkqtpSimplePlotFunctionType & f, const QString& title, JKQTPlotter* parent);
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType && f, const QString& title, JKQTBasePlotter* parent=nullptr);
|
JKQTPFunctorLineGraphBase(jkqtpSimplePlotFunctionType && f, const QString& title, JKQTBasePlotter* parent=nullptr);
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType && f, const QString& title, JKQTPlotter* parent);
|
JKQTPFunctorLineGraphBase(jkqtpSimplePlotFunctionType && f, const QString& title, JKQTPlotter* parent);
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(SpecialFunction type, const QVector<double>& params, const QString& title, JKQTBasePlotter* parent);
|
JKQTPFunctorLineGraphBase(SpecialFunction type, const QVector<double>& params, const QString& title, JKQTBasePlotter* parent);
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXFunctionLineGraph(SpecialFunction type, const QVector<double>& params, const QString& title, JKQTPlotter* parent);
|
JKQTPFunctorLineGraphBase(SpecialFunction type, const QVector<double>& params, const QString& title, JKQTPlotter* parent);
|
||||||
|
|
||||||
/** \brief class destructor */
|
/** \brief class destructor */
|
||||||
virtual ~JKQTPXFunctionLineGraph() override;
|
virtual ~JKQTPFunctorLineGraphBase() 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, QRectF& rect) override;
|
|
||||||
/** \brief returns the color to be used for the key label */
|
|
||||||
virtual QColor getKeyLabelColor() const override;
|
|
||||||
|
|
||||||
/** \brief get the maximum and minimum x-value of the graph
|
|
||||||
*
|
|
||||||
* This functions returns 0 for both parameters, so that the plotter uses the predefined
|
|
||||||
* min and max values.
|
|
||||||
*/
|
|
||||||
virtual bool getXMinMax(double& minx, double& maxx, double& smallestGreaterZero) override;
|
|
||||||
/** \brief get the maximum and minimum y-value of the graph
|
|
||||||
*/
|
|
||||||
virtual bool getYMinMax(double& miny, double& maxy, double& smallestGreaterZero) override;
|
|
||||||
|
|
||||||
/*! \brief set color, fill color and error color at the same time */
|
|
||||||
void setColor(QColor c);
|
|
||||||
|
|
||||||
/*! \copydoc drawLine */
|
|
||||||
void setDrawLine(bool __value);
|
|
||||||
/*! \copydoc drawLine */
|
|
||||||
bool getDrawLine() const;
|
|
||||||
|
|
||||||
/** \brief sets a functor to be plotted
|
/** \brief sets a functor to be plotted
|
||||||
*
|
*
|
||||||
@ -153,23 +224,20 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPXFunctionLineGraph: public JKQTPEvaluatedFunct
|
|||||||
* \see simplePlotFunction
|
* \see simplePlotFunction
|
||||||
*/
|
*/
|
||||||
virtual void setPlotFunctionFunctor (const jkqtpSimplePlotFunctionType & __value);
|
virtual void setPlotFunctionFunctor (const jkqtpSimplePlotFunctionType & __value);
|
||||||
/*! \copydoc plotFunction */ \
|
/*! \copydoc plotFunction
|
||||||
|
*
|
||||||
|
* \see isSimplePlotFunction() */ \
|
||||||
virtual jkqtpPlotFunctionType getPlotFunctionFunctor () const;
|
virtual jkqtpPlotFunctionType getPlotFunctionFunctor () const;
|
||||||
/*! \copydoc simplePlotFunction */ \
|
/*! \copydoc simplePlotFunction
|
||||||
|
*
|
||||||
|
* \see isSimplePlotFunction() */ \
|
||||||
virtual jkqtpSimplePlotFunctionType getSimplePlotFunction () const;
|
virtual jkqtpSimplePlotFunctionType getSimplePlotFunction () const;
|
||||||
|
|
||||||
|
/** \brief returns whether the plot function was defined as a jkqtpSimpleParametricCurveFunctionType (\c true ) or
|
||||||
|
* a jkqtpParametricCurveFunctionType (\c false ) */
|
||||||
|
bool isSimplePlotFunction() const;
|
||||||
|
|
||||||
/** \brief returns the currently set internal parameter vector */
|
|
||||||
QVector<double> getInternalErrorParams() const;
|
|
||||||
|
|
||||||
/*! \copydoc drawErrorPolygons */
|
|
||||||
void setDrawErrorPolygons(bool __value);
|
|
||||||
/*! \copydoc drawErrorPolygons */
|
|
||||||
bool getDrawErrorPolygons() const;
|
|
||||||
/*! \copydoc drawErrorLines */
|
|
||||||
void setDrawErrorLines(bool __value);
|
|
||||||
/*! \copydoc drawErrorLines */
|
|
||||||
bool getDrawErrorLines() const;
|
|
||||||
/** \brief sets a functor to be used for calculating errors
|
/** \brief sets a functor to be used for calculating errors
|
||||||
*
|
*
|
||||||
* \see errorPlotFunction
|
* \see errorPlotFunction
|
||||||
@ -194,100 +262,73 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPXFunctionLineGraph: public JKQTPEvaluatedFunct
|
|||||||
virtual void setErrorPlotFunction (const jkqtpSimplePlotFunctionType & __value);
|
virtual void setErrorPlotFunction (const jkqtpSimplePlotFunctionType & __value);
|
||||||
/*! \copydoc errorSimplePlotFunction */ \
|
/*! \copydoc errorSimplePlotFunction */ \
|
||||||
virtual jkqtpSimplePlotFunctionType getErrorSimplePlotFunction () const;
|
virtual jkqtpSimplePlotFunctionType getErrorSimplePlotFunction () const;
|
||||||
/*! \copydoc errorParams */
|
|
||||||
virtual void setErrorParams(void* __value);
|
|
||||||
/*! \copydoc errorParams */
|
|
||||||
void *getErrorParams() const;
|
|
||||||
/** \brief sets the error params as a pointer to an internal COPY of the given vector (not the data of the vector, as then the size would be unknown!!!) */
|
|
||||||
void setErrorParams(const QVector<double>& errorParams);
|
|
||||||
/*! \copydoc errorParameterColumn */
|
|
||||||
void setErrorParameterColumn(int __value);
|
|
||||||
/*! \copydoc errorParameterColumn */
|
|
||||||
int getErrorParameterColumn() const;
|
|
||||||
/*! \copydoc errorParameterColumn */
|
|
||||||
void setErrorParameterColumn (size_t __value);
|
|
||||||
|
|
||||||
/*! \copydoc errorColor */
|
|
||||||
virtual void setErrorLineColor(const QColor & __value);
|
|
||||||
/*! \copydoc errorColor */
|
|
||||||
virtual QColor getErrorLineColor() const;
|
|
||||||
/*! \copydoc errorFillColor */
|
|
||||||
virtual void setErrorFillColor(const QColor & __value);
|
|
||||||
/*! \copydoc errorFillColor */
|
|
||||||
virtual QColor getErrorFillColor() const;
|
|
||||||
/*! \copydoc errorFillStyle */
|
|
||||||
virtual void setErrorFillStyle(Qt::BrushStyle __value);
|
|
||||||
/*! \copydoc errorFillStyle */
|
|
||||||
virtual Qt::BrushStyle getErrorFillStyle() const;
|
|
||||||
/*! \copydoc errorStyle */
|
|
||||||
virtual void setErrorLineStyle(Qt::PenStyle __value);
|
|
||||||
/*! \copydoc errorStyle */
|
|
||||||
virtual Qt::PenStyle getErrorLineStyle() const;
|
|
||||||
/*! \copydoc errorLineWidth */
|
|
||||||
virtual void setErrorLineWidth(double __value);
|
|
||||||
/*! \copydoc errorLineWidth */
|
|
||||||
virtual double getErrorLineWidth() const;
|
|
||||||
|
|
||||||
/** \copydoc JKQTPGraph::usesColumn() */
|
|
||||||
virtual bool usesColumn(int c) const override;
|
|
||||||
|
|
||||||
|
|
||||||
/** \brief sets function to the given special function */
|
/** \brief sets function to the given special function */
|
||||||
void setSpecialFunction(SpecialFunction function);
|
void setSpecialFunction(SpecialFunction function);
|
||||||
/** \brief returns, which special function is set (or if any is set) */
|
|
||||||
SpecialFunction getFunctionType() const;
|
|
||||||
protected:
|
protected:
|
||||||
/** \brief ensure that current function parameters for plotFunction (which may stem from different sources, as direct data, a datastore column ...) are stored in iparams and ierrorparams */
|
|
||||||
virtual void collectParameters() override;
|
|
||||||
|
|
||||||
|
|
||||||
/** \brief fill the data array with data from the function plotFunction */
|
|
||||||
virtual void createPlotData( bool collectParams=true) override;
|
|
||||||
|
|
||||||
|
|
||||||
/** \brief if set, the values from this datatsore column are used for the parameters \c p1 , \c p2 , \c p3 , ... of the error plot function */
|
|
||||||
int errorParameterColumn;
|
|
||||||
|
|
||||||
/** \brief indicates whether to draw a line or not */
|
|
||||||
bool drawLine;
|
|
||||||
/** \brief indicates whether to fill the space between the curve and the x-axis */
|
|
||||||
bool fillCurve;
|
|
||||||
/** \brief the function to be plotted */
|
/** \brief the function to be plotted */
|
||||||
jkqtpPlotFunctionType plotFunction;
|
jkqtpPlotFunctionType plotFunction;
|
||||||
/** \brief a simple function to be plotted, simplified form without parameters */
|
/** \brief a simple function to be plotted, simplified form without parameters */
|
||||||
jkqtpSimplePlotFunctionType simplePlotFunction;
|
jkqtpSimplePlotFunctionType simplePlotFunction;
|
||||||
/** \brief indicates whether a special function is set (and if so, which one), or a user-supplied function */
|
|
||||||
SpecialFunction functionType;
|
|
||||||
|
|
||||||
|
|
||||||
/** \brief indicates whether an error polygon should be drawn */
|
|
||||||
bool drawErrorPolygons;
|
|
||||||
/** \brief indicates whether error lines should be drawn */
|
|
||||||
bool drawErrorLines;
|
|
||||||
/** \brief this function calculates the error at a given position */
|
/** \brief this function calculates the error at a given position */
|
||||||
jkqtpPlotFunctionType errorPlotFunction;
|
jkqtpPlotFunctionType errorPlotFunction;
|
||||||
/** \brief this function calculates the error at a given position, simplified form without parameters */
|
/** \brief this function calculates the error at a given position, simplified form without parameters */
|
||||||
jkqtpSimplePlotFunctionType errorSimplePlotFunction;
|
jkqtpSimplePlotFunctionType errorSimplePlotFunction;
|
||||||
/** \brief parameters for errorFunction */
|
};
|
||||||
void* errorParams;
|
|
||||||
|
|
||||||
|
|
||||||
/** \brief color of the error graph */
|
/*! \brief This implements line plots where the data is taken from a user supplied function \f$ y=f(x) \f$
|
||||||
QColor errorColor;
|
\ingroup jkqtplotter_functiongraphs
|
||||||
/** \brief color of the error graph fill */
|
|
||||||
QColor errorFillColor;
|
This class uses the intelligent plotting algorithm for functions, implemented in JKQTPAdaptiveFunctionGraphEvaluator.
|
||||||
/** \brief linestyle of the error graph lines */
|
|
||||||
Qt::PenStyle errorStyle;
|
The following image shows some example graphs:
|
||||||
/** \brief width (pixels) of the error graph */
|
|
||||||
double errorLineWidth;
|
\image html plot_functionplots.png
|
||||||
/** \brief fill style, if the error curve should be filled */
|
|
||||||
Qt::BrushStyle errorFillStyle;
|
\see \ref JKQTPlotterFunctionPlots, JKQTPAdaptiveFunctionGraphEvaluator, JKQTPYFunctionLineGraph, JKQTPXYFunctionLineGraph, jkqtpstatAddPolyFit(), jkqtpstatAddWeightedRegression(), jkqtpstatAddRobustIRLSRegression(), jkqtpstatAddRegression(), jkqtpstatAddLinearWeightedRegression(), jkqtpstatAddRobustIRLSLinearRegression(), jkqtpstatAddLinearRegression()
|
||||||
|
*/
|
||||||
|
class JKQTPLOTTER_LIB_EXPORT JKQTPXFunctionLineGraph: public JKQTPFunctorLineGraphBase {
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
|
||||||
|
|
||||||
QBrush getErrorBrush(JKQTPEnhancedPainter& painter) const;
|
/** \brief class constructor */
|
||||||
QPen getErrorLinePen(JKQTPEnhancedPainter &painter) const;
|
JKQTPXFunctionLineGraph(JKQTBasePlotter* parent=nullptr);
|
||||||
/** \brief internal storage for the current error function parameters for errorPlotFunction (which may stem from different sources, as direct data, a datastore column ...) */
|
|
||||||
QVector<double> ierrorparams;
|
/** \brief class constructor */
|
||||||
|
JKQTPXFunctionLineGraph(JKQTPlotter* parent);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType & f, const QString& title, JKQTBasePlotter* parent=nullptr);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXFunctionLineGraph(const jkqtpSimplePlotFunctionType & f, const QString& title, JKQTPlotter* parent);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType && f, const QString& title, JKQTBasePlotter* parent=nullptr);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXFunctionLineGraph(jkqtpSimplePlotFunctionType && f, const QString& title, JKQTPlotter* parent);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXFunctionLineGraph(SpecialFunction type, const QVector<double>& params, const QString& title, JKQTBasePlotter* parent);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXFunctionLineGraph(SpecialFunction type, const QVector<double>& params, const QString& title, JKQTPlotter* parent);
|
||||||
|
|
||||||
|
/** \brief class destructor */
|
||||||
|
virtual ~JKQTPXFunctionLineGraph() override;
|
||||||
|
|
||||||
|
/** \brief plots the graph to the plotter object specified as parent */
|
||||||
|
virtual void draw(JKQTPEnhancedPainter& painter) override;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
/** \copydoc JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec() */
|
||||||
|
virtual PlotFunctorSpec buildPlotFunctorSpec() override;
|
||||||
|
|
||||||
|
/** \copydoc JKQTPEvaluatedFunctionWithErrorsGraphBase::buildPlotFunctorSpec() */
|
||||||
|
virtual std::function<QPointF(double)> buildErrorFunctorSpec() override;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*! \brief This implements line plots where the data is taken from a user supplied function \f$ x=f(y) \f$
|
/*! \brief This implements line plots where the data is taken from a user supplied function \f$ x=f(y) \f$
|
||||||
@ -299,7 +340,7 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPXFunctionLineGraph: public JKQTPEvaluatedFunct
|
|||||||
|
|
||||||
\see \ref JKQTPlotterFunctionPlots , JKQTPXFunctionLineGraph, JKQTPXYFunctionLineGraph
|
\see \ref JKQTPlotterFunctionPlots , JKQTPXFunctionLineGraph, JKQTPXYFunctionLineGraph
|
||||||
*/
|
*/
|
||||||
class JKQTPLOTTER_LIB_EXPORT JKQTPYFunctionLineGraph: public JKQTPXFunctionLineGraph {
|
class JKQTPLOTTER_LIB_EXPORT JKQTPYFunctionLineGraph: public JKQTPFunctorLineGraphBase {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
@ -323,8 +364,11 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPYFunctionLineGraph: public JKQTPXFunctionLineG
|
|||||||
virtual void draw(JKQTPEnhancedPainter& painter) override;
|
virtual void draw(JKQTPEnhancedPainter& painter) override;
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
/** \brief fill the data array with data from the function plotFunction */
|
/** \copydoc JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec() */
|
||||||
virtual void createPlotData( bool collectParams=true) override;
|
virtual PlotFunctorSpec buildPlotFunctorSpec() override;
|
||||||
|
|
||||||
|
/** \copydoc JKQTPEvaluatedFunctionWithErrorsGraphBase::buildPlotFunctorSpec() */
|
||||||
|
virtual std::function<QPointF(double)> buildErrorFunctorSpec() override;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -35,16 +35,17 @@
|
|||||||
|
|
||||||
|
|
||||||
JKQTPEvaluatedFunctionGraphBase::JKQTPEvaluatedFunctionGraphBase(JKQTBasePlotter* parent):
|
JKQTPEvaluatedFunctionGraphBase::JKQTPEvaluatedFunctionGraphBase(JKQTBasePlotter* parent):
|
||||||
JKQTPGraph(parent)
|
JKQTPGraph(parent),
|
||||||
|
parameterColumn(-1),
|
||||||
|
minSamples(50),
|
||||||
|
maxRefinementDegree(5),
|
||||||
|
slopeTolerance(0.005),
|
||||||
|
minPixelPerSample(32),
|
||||||
|
dataCleanupMaxAllowedAngleDegree(0.2),
|
||||||
|
displaySamplePoints(false)
|
||||||
{
|
{
|
||||||
minSamples=50;
|
|
||||||
maxRefinementDegree=5;
|
|
||||||
slopeTolerance=0.005;
|
|
||||||
minPixelPerSample=32;
|
|
||||||
dataCleanupMaxAllowedAngleDegree=0.2;
|
|
||||||
displaySamplePoints=false;
|
|
||||||
data.clear();
|
data.clear();
|
||||||
|
iparams.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPEvaluatedFunctionGraphBase::JKQTPEvaluatedFunctionGraphBase(JKQTPlotter* parent):
|
JKQTPEvaluatedFunctionGraphBase::JKQTPEvaluatedFunctionGraphBase(JKQTPlotter* parent):
|
||||||
@ -127,6 +128,24 @@ void JKQTPEvaluatedFunctionGraphBase::setDisplaySamplePoints(bool __value)
|
|||||||
this->displaySamplePoints = __value;
|
this->displaySamplePoints = __value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionGraphBase::createPlotData(bool collectParams)
|
||||||
|
{
|
||||||
|
#ifdef JKQTBP_AUTOTIMER
|
||||||
|
JKQTPAutoOutputTimer jkaat(QString("JKQTPEvaluatedFunctionWithErrorsGraphBase[%1]::createPlotData()").arg(title));
|
||||||
|
#endif
|
||||||
|
data.clear();
|
||||||
|
if (collectParams) collectParameters();
|
||||||
|
PlotFunctorSpec plotfunc= buildPlotFunctorSpec();
|
||||||
|
|
||||||
|
if (plotfunc.isValid()) {
|
||||||
|
std::function<QPointF(double)> fTransformedFunc= std::bind([plotfunc](const JKQTPPlotElement* plot, double t) -> QPointF { return plot->transform(plotfunc.func(t)); }, this, std::placeholders::_1);
|
||||||
|
|
||||||
|
JKQTPAdaptiveFunctionGraphEvaluator evaluator(fTransformedFunc, minSamples, maxRefinementDegree, slopeTolerance, minPixelPerSample);
|
||||||
|
data=evaluator.evaluate(plotfunc.range_start, plotfunc.range_end);
|
||||||
|
data=JKQTPSimplyfyLineSegemnts(data, dataCleanupMaxAllowedAngleDegree);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool JKQTPEvaluatedFunctionGraphBase::getDisplaySamplePoints() const
|
bool JKQTPEvaluatedFunctionGraphBase::getDisplaySamplePoints() const
|
||||||
{
|
{
|
||||||
return this->displaySamplePoints;
|
return this->displaySamplePoints;
|
||||||
@ -181,25 +200,7 @@ bool JKQTPEvaluatedFunctionGraphBase::getYMinMax(double &miny, double &maxy, dou
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionGraphBase::collectParameters()
|
||||||
JKQTPEvaluatedFunctionWithParamsGraphBase::JKQTPEvaluatedFunctionWithParamsGraphBase(JKQTBasePlotter *parent):
|
|
||||||
parameterColumn(-1)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPEvaluatedFunctionWithParamsGraphBase::JKQTPEvaluatedFunctionWithParamsGraphBase(JKQTPlotter *parent):
|
|
||||||
JKQTPEvaluatedFunctionWithParamsGraphBase(parent->getPlotter())
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
JKQTPEvaluatedFunctionWithParamsGraphBase::~JKQTPEvaluatedFunctionWithParamsGraphBase()
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::collectParameters()
|
|
||||||
{
|
{
|
||||||
if (parent && parameterColumn>=0) {
|
if (parent && parameterColumn>=0) {
|
||||||
iparams.clear();
|
iparams.clear();
|
||||||
@ -221,67 +222,195 @@ void JKQTPEvaluatedFunctionWithParamsGraphBase::collectParameters()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParams(const QVector<double> ¶ms)
|
void JKQTPEvaluatedFunctionGraphBase::setParams(const QVector<double> ¶ms)
|
||||||
{
|
{
|
||||||
iparams=params;
|
iparams=params;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setCopiedParams(const double *params, int N)
|
void JKQTPEvaluatedFunctionGraphBase::setCopiedParams(const double *params, int N)
|
||||||
{
|
{
|
||||||
QVector<double> v;
|
QVector<double> v;
|
||||||
for (int i=0; i<N; i++) { v<<params[i]; }
|
for (int i=0; i<N; i++) { v<<params[i]; }
|
||||||
setParams(v);
|
setParams(v);
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParamsV(double p1) {
|
void JKQTPEvaluatedFunctionGraphBase::setParamsV(double p1) {
|
||||||
QVector<double> p;
|
QVector<double> p;
|
||||||
p<<p1;
|
p<<p1;
|
||||||
setParams(p);
|
setParams(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParamsV(double p1, double p2) {
|
void JKQTPEvaluatedFunctionGraphBase::setParamsV(double p1, double p2) {
|
||||||
QVector<double> p;
|
QVector<double> p;
|
||||||
p<<p1<<p2;
|
p<<p1<<p2;
|
||||||
setParams(p);
|
setParams(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParamsV(double p1, double p2, double p3) {
|
void JKQTPEvaluatedFunctionGraphBase::setParamsV(double p1, double p2, double p3) {
|
||||||
QVector<double> p;
|
QVector<double> p;
|
||||||
p<<p1<<p2<<p3;
|
p<<p1<<p2<<p3;
|
||||||
setParams(p);
|
setParams(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParamsV(double p1, double p2, double p3, double p4) {
|
void JKQTPEvaluatedFunctionGraphBase::setParamsV(double p1, double p2, double p3, double p4) {
|
||||||
QVector<double> p;
|
QVector<double> p;
|
||||||
p<<p1<<p2<<p3<<p4;
|
p<<p1<<p2<<p3<<p4;
|
||||||
setParams(p);
|
setParams(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParamsV(double p1, double p2, double p3, double p4, double p5) {
|
void JKQTPEvaluatedFunctionGraphBase::setParamsV(double p1, double p2, double p3, double p4, double p5) {
|
||||||
QVector<double> p;
|
QVector<double> p;
|
||||||
p<<p1<<p2<<p3<<p4<<p5;
|
p<<p1<<p2<<p3<<p4<<p5;
|
||||||
setParams(p);
|
setParams(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParameterColumn(int __value)
|
void JKQTPEvaluatedFunctionGraphBase::setParameterColumn(int __value)
|
||||||
{
|
{
|
||||||
this->parameterColumn = __value;
|
this->parameterColumn = __value;
|
||||||
}
|
}
|
||||||
|
|
||||||
int JKQTPEvaluatedFunctionWithParamsGraphBase::getParameterColumn() const
|
int JKQTPEvaluatedFunctionGraphBase::getParameterColumn() const
|
||||||
{
|
{
|
||||||
return this->parameterColumn;
|
return this->parameterColumn;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPEvaluatedFunctionWithParamsGraphBase::setParameterColumn(size_t __value) {
|
void JKQTPEvaluatedFunctionGraphBase::setParameterColumn(size_t __value) {
|
||||||
this->parameterColumn = static_cast<int>(__value);
|
this->parameterColumn = static_cast<int>(__value);
|
||||||
}
|
}
|
||||||
|
|
||||||
QVector<double> JKQTPEvaluatedFunctionWithParamsGraphBase::getInternalParams() const {
|
const QVector<double>& JKQTPEvaluatedFunctionGraphBase::getInternalParams() const {
|
||||||
return iparams;
|
return iparams;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool JKQTPEvaluatedFunctionWithParamsGraphBase::usesColumn(int c) const
|
QVector<double> &JKQTPEvaluatedFunctionGraphBase::getInternalParams()
|
||||||
|
{
|
||||||
|
return iparams;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JKQTPEvaluatedFunctionGraphBase::usesColumn(int c) const
|
||||||
{
|
{
|
||||||
return (c==parameterColumn);
|
return (c==parameterColumn);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec::PlotFunctorSpec():
|
||||||
|
func(),
|
||||||
|
range_start(0),
|
||||||
|
range_end(0)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec::isValid() const
|
||||||
|
{
|
||||||
|
return static_cast<bool>(func) && (fabs(range_end-range_start)>0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphBase::JKQTPEvaluatedFunctionWithErrorsGraphBase(JKQTBasePlotter *parent):
|
||||||
|
JKQTPEvaluatedFunctionGraphBase(parent),
|
||||||
|
errorParameterColumn(-1)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphBase::JKQTPEvaluatedFunctionWithErrorsGraphBase(JKQTPlotter *parent):
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphBase(parent->getPlotter())
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionWithErrorsGraphBase::~JKQTPEvaluatedFunctionWithErrorsGraphBase()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
const QVector<double>& JKQTPEvaluatedFunctionWithErrorsGraphBase::getInternalErrorParams() const {
|
||||||
|
return ierrorparams;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVector<double>& JKQTPEvaluatedFunctionWithErrorsGraphBase::getInternalErrorParams() {
|
||||||
|
return ierrorparams;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JKQTPEvaluatedFunctionWithErrorsGraphBase::usesColumn(int c) const
|
||||||
|
{
|
||||||
|
return JKQTPEvaluatedFunctionGraphBase::usesColumn(c)||(c==errorParameterColumn);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParams(const QVector<double> &errorParams)
|
||||||
|
{
|
||||||
|
ierrorparams=errorParams;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParameterColumn(int __value)
|
||||||
|
{
|
||||||
|
this->errorParameterColumn = __value;
|
||||||
|
}
|
||||||
|
|
||||||
|
int JKQTPEvaluatedFunctionWithErrorsGraphBase::getErrorParameterColumn() const
|
||||||
|
{
|
||||||
|
return this->errorParameterColumn;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParameterColumn(size_t __value) {
|
||||||
|
this->errorParameterColumn = static_cast<int>(__value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParamsV(double p1)
|
||||||
|
{
|
||||||
|
QVector<double> p;
|
||||||
|
p<<p1;
|
||||||
|
setErrorParams(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParamsV(double p1, double p2)
|
||||||
|
{
|
||||||
|
QVector<double> p;
|
||||||
|
p<<p1<<p2;
|
||||||
|
setErrorParams(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParamsV(double p1, double p2, double p3)
|
||||||
|
{
|
||||||
|
QVector<double> p;
|
||||||
|
p<<p1<<p2<<p3;
|
||||||
|
setErrorParams(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParamsV(double p1, double p2, double p3, double p4)
|
||||||
|
{
|
||||||
|
QVector<double> p;
|
||||||
|
p<<p1<<p2<<p3<<p4;
|
||||||
|
setErrorParams(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::setErrorParamsV(double p1, double p2, double p3, double p4, double p5)
|
||||||
|
{
|
||||||
|
QVector<double> p;
|
||||||
|
p<<p1<<p2<<p3<<p4<<p5;
|
||||||
|
setErrorParams(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPEvaluatedFunctionWithErrorsGraphBase::collectParameters()
|
||||||
|
{
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::collectParameters();
|
||||||
|
|
||||||
|
if (parent && errorParameterColumn>=0) {
|
||||||
|
ierrorparams.clear();
|
||||||
|
JKQTPDatastore* datastore=parent->getDatastore();
|
||||||
|
int imin=0;
|
||||||
|
int imax= static_cast<int>(datastore->getRows(errorParameterColumn));
|
||||||
|
|
||||||
|
for (int i=imin; i<imax; i++) {
|
||||||
|
double xv=datastore->get(errorParameterColumn,i);
|
||||||
|
ierrorparams<<xv;
|
||||||
|
}
|
||||||
|
int i=ierrorparams.size()-1;
|
||||||
|
while (i>=0 && !JKQTPIsOKFloat(ierrorparams[i])) {
|
||||||
|
ierrorparams.remove(i,1);
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -34,10 +34,26 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
/** \brief Base class for graph classes that evaluate a mathematical function (e.g. defined as a C-fucntion),
|
/** \brief Base class for graph classes that evaluate a mathematical function (e.g. defined as a C-function),
|
||||||
* using an adaptive plotting algorithm from JKQTPAdaptiveFunctionGraphEvaluator
|
* using an adaptive plotting algorithm from JKQTPAdaptiveFunctionGraphEvaluator
|
||||||
* \ingroup jkqtplotter_functiongraphs
|
* \ingroup jkqtplotter_functiongraphs
|
||||||
*
|
*
|
||||||
|
* This class always plots a general 2D-graph \f$ [x,y]=\vec{f}(t) \f$ , which is calculated in dependence of
|
||||||
|
* a parameter \f$ t \f$ . This parametrization is general enough to cover the cases of parametric function, as well as
|
||||||
|
* x- and y-dependent function graphs:
|
||||||
|
* - plot a function \f$ f(x) \f$ i.e. the plot points will be \f$ [x, f(x)] \f$
|
||||||
|
* and the value rage will be the x-axis range. This is implemented by e.g. JKQTPXFunctionLineGraph.
|
||||||
|
* - plot a function \f$ f(y) \f$ i.e. the plot points will be \f$ [f(y), y] \f$
|
||||||
|
* and the value rage will be the y-axis range. This is implemented by e.g. JKQTPYFunctionLineGraph.
|
||||||
|
* - plot a function \f$ [x,y]=\vec{f}(t) \f$ i.e. the plot points will be \f$ \vec{f}(t) \f$
|
||||||
|
* and the value rage will be a user-defined range for \f$ gt \f$.
|
||||||
|
* This is implemented by e.g. JKQTPXYFunctionLineGraph.
|
||||||
|
* .
|
||||||
|
*
|
||||||
|
* In order to implement a special cas, one has to override/implement buildPlotFunctorSpec(), which
|
||||||
|
* returns a functor and a value-range that can represent the cases above.
|
||||||
|
*
|
||||||
|
*
|
||||||
* This class uses the intelligent plotting algorithm for functions, implemented in JKQTPAdaptiveFunctionGraphEvaluator.
|
* 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
|
* 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$
|
* necessary. To do so the function is evaluated at the mid point and the slopes \f$ \alpha_{\mbox{left}} \f$
|
||||||
@ -50,6 +66,9 @@
|
|||||||
* angle between consecutive line-segments of less than dataCleanupMaxAllowedAngleDegree.
|
* angle between consecutive line-segments of less than dataCleanupMaxAllowedAngleDegree.
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
|
*
|
||||||
|
*
|
||||||
|
*
|
||||||
* \see JKQTPAdaptiveFunctionGraphEvaluator, JKQTPXFunctionLineGraph, JKQTPYFunctionLineGraph, JKQTPXYFunctionLineGraph
|
* \see JKQTPAdaptiveFunctionGraphEvaluator, JKQTPXFunctionLineGraph, JKQTPYFunctionLineGraph, JKQTPXYFunctionLineGraph
|
||||||
*/
|
*/
|
||||||
class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionGraphBase: public JKQTPGraph {
|
class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionGraphBase: public JKQTPGraph {
|
||||||
@ -75,6 +94,21 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionGraphBase: public JKQTPGraph
|
|||||||
*/
|
*/
|
||||||
virtual bool getYMinMax(double& miny, double& maxy, double& smallestGreaterZero) override;
|
virtual bool getYMinMax(double& miny, double& maxy, double& smallestGreaterZero) override;
|
||||||
|
|
||||||
|
/** \brief sets the params as a pointer to an internal COPY of the given vector (not the data of the vector, as then the size would be unknown!!!) */
|
||||||
|
virtual void setParams(const QVector<double>& params);
|
||||||
|
/** \brief sets the params from a copy of the given array of length \a N */
|
||||||
|
void setCopiedParams(const double* params, int N);
|
||||||
|
|
||||||
|
/** \brief returns the currently set internal parameter vector */
|
||||||
|
const QVector<double>& getInternalParams() const;
|
||||||
|
/** \brief returns the currently set internal parameter vector */
|
||||||
|
QVector<double>& getInternalParams();
|
||||||
|
/** \copydoc parameterColumn */
|
||||||
|
int getParameterColumn() const;
|
||||||
|
|
||||||
|
/** \copydoc JKQTPGraph::usesColumn() */
|
||||||
|
virtual bool usesColumn(int c) const override;
|
||||||
|
|
||||||
|
|
||||||
/** \copydoc minSamples */
|
/** \copydoc minSamples */
|
||||||
unsigned int getMinSamples() const;
|
unsigned int getMinSamples() const;
|
||||||
@ -102,12 +136,73 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionGraphBase: public JKQTPGraph
|
|||||||
/** \copydoc displaySamplePoints */
|
/** \copydoc displaySamplePoints */
|
||||||
void setDisplaySamplePoints(bool __value);
|
void setDisplaySamplePoints(bool __value);
|
||||||
|
|
||||||
|
/** \brief set an internal parameter vector as function parameters, initialized with {p1} */
|
||||||
|
void setParamsV(double p1);
|
||||||
|
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2} */
|
||||||
|
void setParamsV(double p1, double p2);
|
||||||
|
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3} */
|
||||||
|
void setParamsV(double p1, double p2, double p3);
|
||||||
|
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3,p4} */
|
||||||
|
void setParamsV(double p1, double p2, double p3, double p4);
|
||||||
|
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3,p4,p5} */
|
||||||
|
void setParamsV(double p1, double p2, double p3, double p4, double p5);
|
||||||
|
|
||||||
|
/** \copydoc parameterColumn */
|
||||||
|
void setParameterColumn(int __value);
|
||||||
|
/** \copydoc parameterColumn */
|
||||||
|
void setParameterColumn (size_t __value);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/** \brief plot data calculated by createPlotData(), i.e. the datapoints \f$ \mbox{transform}\left(x, y=f(x, \vec{p})\right) \f$ to be plotted */
|
/** \brief specifies an internal plot functor \see buildPlotFunctor() */
|
||||||
QVector<QPointF> data;
|
struct PlotFunctorSpec {
|
||||||
|
/** brief construct an in-valid PlotFunctorSpec ... will become valid, by assigning a non-zero range and a plot-function */
|
||||||
|
PlotFunctorSpec();
|
||||||
|
/** \brief calculates the points \f$ [x,y] \f$ on the function graph, in dependence on
|
||||||
|
* a dependent parameter variable, could be e.g. \f$ [x, f(x)] \f$ for plotting
|
||||||
|
* a function \f$ f(x) \f$ over the x-axis. */
|
||||||
|
std::function<QPointF(double)> func;
|
||||||
|
/** \brief lower bound for the dependent parameter variable of func */
|
||||||
|
double range_start;
|
||||||
|
/** \brief upper bound for the dependent parameter variable of func */
|
||||||
|
double range_end;
|
||||||
|
|
||||||
|
bool isValid() const;
|
||||||
|
};
|
||||||
|
/** \brief this function returns a functor that is used to generate the plot data
|
||||||
|
* in coordinate space, based on a range of the dependent variable in coordinate space.
|
||||||
|
* In addition it also contains the value range over which to evaluate the functor PlotFunctorSpec::func
|
||||||
|
*
|
||||||
|
* This function has to be overridden by each class. Depending on the way that class defines
|
||||||
|
* the actual plot function, this function has to compose its return type in different ways.
|
||||||
|
* The three most common ways are:
|
||||||
|
* - plot a function \f$ f(x) \f$ i.e. the plot points will be \f$ [x, f(x)] \f$
|
||||||
|
* and the value rage will be the x-axis range. This is implemented by e.g. JKQTPXFunctionLineGraph.
|
||||||
|
* - plot a function \f$ f(y) \f$ i.e. the plot points will be \f$ [f(y), y] \f$
|
||||||
|
* and the value rage will be the y-axis range. This is implemented by e.g. JKQTPYFunctionLineGraph.
|
||||||
|
* - plot a function \f$ [x,y]=\vec{f}(t) \f$ i.e. the plot points will be \f$ \vec{f}(t) \f$
|
||||||
|
* and the value rage will be a user-defined range for \f$ gt \f$.
|
||||||
|
* This is implemented by e.g. JKQTPXYFunctionLineGraph.
|
||||||
|
* .
|
||||||
|
*/
|
||||||
|
virtual PlotFunctorSpec buildPlotFunctorSpec() =0;
|
||||||
|
|
||||||
|
/** \brief ensure that current function parameters for a plot function (which may stem from different sources, as direct data, a datastore column ...) are stored in iparams */
|
||||||
|
virtual void collectParameters();
|
||||||
|
|
||||||
|
/** \brief draw all the sample points in data as small symbols */
|
||||||
|
void drawSamplePoints(JKQTPEnhancedPainter &painter, QColor graphColor);
|
||||||
|
|
||||||
/** \brief fill the data array with data from the function plotFunction */
|
/** \brief fill the data array with data from the function plotFunction */
|
||||||
virtual void createPlotData( bool collectParams=true) =0;
|
virtual void createPlotData( bool collectParams=true) ;
|
||||||
|
|
||||||
|
/** \brief if set, the values from this datatsore column are used for the parameters \c p1 , \c p2 , \c p3 , ... of the plot function */
|
||||||
|
int parameterColumn;
|
||||||
|
|
||||||
|
/** \brief internal storage for the current function parameters for plotFunction (which may stem from different sources, as direct data, a datastore column ...) */
|
||||||
|
QVector<double> iparams;
|
||||||
|
|
||||||
|
/** \brief plot data calculated by createPlotData(), i.e. the datapoints \f$ \mbox{transform}\left(x, y=f(x, \vec{p})\right) \f$ to be plotted */
|
||||||
|
QVector<QPointF> data;
|
||||||
|
|
||||||
/** \brief the minimum number of points to evaluate the function at */
|
/** \brief the minimum number of points to evaluate the function at */
|
||||||
unsigned int minSamples;
|
unsigned int minSamples;
|
||||||
@ -127,66 +222,75 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionGraphBase: public JKQTPGraph
|
|||||||
double dataCleanupMaxAllowedAngleDegree;
|
double dataCleanupMaxAllowedAngleDegree;
|
||||||
/** \brief if true [default: off] display the points where the function has been sampled */
|
/** \brief if true [default: off] display the points where the function has been sampled */
|
||||||
bool displaySamplePoints;
|
bool displaySamplePoints;
|
||||||
|
|
||||||
/** \brief draw all the sample points in data as small symbols */
|
|
||||||
void drawSamplePoints(JKQTPEnhancedPainter &painter, QColor graphColor);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
/** \brief extends JKQTPEvaluatedFunctionGraphBase with a set of functions that support function parameters
|
/** \brief extends JKQTPEvaluatedFunctionGraphBase with some basic properties (e.g. function parameters)
|
||||||
|
* for a second function that calculates an error (for drawing error indicators)
|
||||||
* \ingroup jkqtplotter_functiongraphs
|
* \ingroup jkqtplotter_functiongraphs
|
||||||
*
|
*
|
||||||
|
* When implementing this, you will have to implement buildErrorFunctorSpec() in addition to
|
||||||
|
* JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec()!
|
||||||
|
*
|
||||||
* \see JKQTPEvaluatedFunctionGraphBase
|
* \see JKQTPEvaluatedFunctionGraphBase
|
||||||
*/
|
*/
|
||||||
class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionWithParamsGraphBase: public JKQTPEvaluatedFunctionGraphBase {
|
class JKQTPLOTTER_LIB_EXPORT JKQTPEvaluatedFunctionWithErrorsGraphBase: public JKQTPEvaluatedFunctionGraphBase {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
explicit JKQTPEvaluatedFunctionWithParamsGraphBase(JKQTBasePlotter* parent=nullptr);
|
explicit JKQTPEvaluatedFunctionWithErrorsGraphBase(JKQTBasePlotter* parent=nullptr);
|
||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
explicit JKQTPEvaluatedFunctionWithParamsGraphBase(JKQTPlotter* parent);
|
explicit JKQTPEvaluatedFunctionWithErrorsGraphBase(JKQTPlotter* parent);
|
||||||
/** \brief class destructor */
|
/** \brief class destructor */
|
||||||
virtual ~JKQTPEvaluatedFunctionWithParamsGraphBase();
|
virtual ~JKQTPEvaluatedFunctionWithErrorsGraphBase();
|
||||||
|
|
||||||
|
/** \copydoc errorParameterColumn */
|
||||||
/** \brief sets the params as a pointer to an internal COPY of the given vector (not the data of the vector, as then the size would be unknown!!!) */
|
int getErrorParameterColumn() const;
|
||||||
virtual void setParams(const QVector<double>& params);
|
|
||||||
/** \brief sets the params from a copy of the given array of length \a N */
|
|
||||||
void setCopiedParams(const double* params, int N);
|
|
||||||
|
|
||||||
/** \brief returns the currently set internal parameter vector */
|
/** \brief returns the currently set internal parameter vector */
|
||||||
QVector<double> getInternalParams() const;
|
const QVector<double>& getInternalErrorParams() const;
|
||||||
/** \copydoc parameterColumn */
|
/** \brief returns the currently set internal parameter vector */
|
||||||
int getParameterColumn() const;
|
QVector<double>& getInternalErrorParams();
|
||||||
|
|
||||||
/** \copydoc JKQTPGraph::usesColumn() */
|
/** \copydoc JKQTPGraph::usesColumn() */
|
||||||
virtual bool usesColumn(int c) const override;
|
virtual bool usesColumn(int c) const override;
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1} */
|
/** \brief sets the error params as a pointer to an internal COPY of the given vector (not the data of the vector, as then the size would be unknown!!!) */
|
||||||
void setParamsV(double p1);
|
void setErrorParams(const QVector<double>& errorParams);
|
||||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2} */
|
/** \copydoc errorParameterColumn */
|
||||||
void setParamsV(double p1, double p2);
|
void setErrorParameterColumn(int __value);
|
||||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3} */
|
/** \copydoc errorParameterColumn */
|
||||||
void setParamsV(double p1, double p2, double p3);
|
void setErrorParameterColumn (size_t __value);
|
||||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3,p4} */
|
|
||||||
void setParamsV(double p1, double p2, double p3, double p4);
|
/** \brief set the internal error function parameters to {p1} */
|
||||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3,p4,p5} */
|
void setErrorParamsV(double p1);
|
||||||
void setParamsV(double p1, double p2, double p3, double p4, double p5);
|
/** \brief set the internal error function parameters to {p1,p2} */
|
||||||
|
void setErrorParamsV(double p1, double p2);
|
||||||
|
/** \brief set the internal error function parameters to {p1,p2,p3} */
|
||||||
|
void setErrorParamsV(double p1, double p2, double p3);
|
||||||
|
/** \brief set the internal error function parameters to {p1,p2,p3,p4} */
|
||||||
|
void setErrorParamsV(double p1, double p2, double p3, double p4);
|
||||||
|
/** \brief set the internal error function parameters to {p1,p2,p3,p4,p5} */
|
||||||
|
void setErrorParamsV(double p1, double p2, double p3, double p4, double p5);
|
||||||
|
|
||||||
/** \copydoc parameterColumn */
|
|
||||||
void setParameterColumn(int __value);
|
|
||||||
/** \copydoc parameterColumn */
|
|
||||||
void setParameterColumn (size_t __value);
|
|
||||||
protected:
|
protected:
|
||||||
/** \brief ensure that current function parameters for a plot function (which may stem from different sources, as direct data, a datastore column ...) are stored in iparams */
|
/** \brief ensure that current function parameters for plotFunction (which may stem from different sources, as direct data, a datastore column ...) are stored in iparams and ierrorparams */
|
||||||
virtual void collectParameters();
|
virtual void collectParameters() override;
|
||||||
|
/** \brief same as JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec(), but for error functions.
|
||||||
|
*
|
||||||
|
* The functor, returned by this function should calculate the error of the function (in x- and y-direction)
|
||||||
|
* for every value \f$ t \f$ of the actual function.
|
||||||
|
*
|
||||||
|
* The parameter range is the same as for JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec()
|
||||||
|
*
|
||||||
|
* \see JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec()
|
||||||
|
*/
|
||||||
|
virtual std::function<QPointF(double)> buildErrorFunctorSpec() =0;
|
||||||
|
|
||||||
/** \brief if set, the values from this datatsore column are used for the parameters \c p1 , \c p2 , \c p3 , ... of the plot function */
|
/** \brief if set, the values from this datatsore column are used for the parameters \c p1 , \c p2 , \c p3 , ... of the error plot function */
|
||||||
int parameterColumn;
|
int errorParameterColumn;
|
||||||
|
/** \brief internal storage for the current error function parameters for errorPlotFunction (which may stem from different sources, as direct data, a datastore column ...) */
|
||||||
/** \brief internal storage for the current function parameters for plotFunction (which may stem from different sources, as direct data, a datastore column ...) */
|
QVector<double> ierrorparams;
|
||||||
QVector<double> iparams;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -33,14 +33,125 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(JKQTBasePlotter* parent):
|
JKQTPXYFunctionLineGraphBase::JKQTPXYFunctionLineGraphBase(JKQTBasePlotter* parent):
|
||||||
JKQTPEvaluatedFunctionWithParamsGraphBase(parent)
|
JKQTPEvaluatedFunctionGraphBase(parent),
|
||||||
|
tmin(0.0),
|
||||||
|
tmax(1.0)
|
||||||
{
|
{
|
||||||
tmin=0.0;
|
|
||||||
tmax=1.0;
|
|
||||||
|
|
||||||
initLineStyle(parent, parentPlotStyle);
|
initLineStyle(parent, parentPlotStyle);
|
||||||
|
setMaxRefinementDegree(8);
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXYFunctionLineGraphBase::JKQTPXYFunctionLineGraphBase(JKQTPlotter* parent):
|
||||||
|
JKQTPXYFunctionLineGraphBase(parent->getPlotter())
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXYFunctionLineGraphBase::~JKQTPXYFunctionLineGraphBase()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void JKQTPXYFunctionLineGraphBase::drawKeyMarker(JKQTPEnhancedPainter& painter, QRectF& rect) {
|
||||||
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
||||||
|
QPen p=getLinePen(painter, parent);
|
||||||
|
p.setJoinStyle(Qt::RoundJoin);
|
||||||
|
p.setCapStyle(Qt::RoundCap);
|
||||||
|
QPen np(Qt::NoPen);
|
||||||
|
const double y=rect.top()+rect.height()/2.0;
|
||||||
|
painter.setPen(np);
|
||||||
|
painter.setPen(p);
|
||||||
|
painter.drawLine(QLineF(rect.left(), y, rect.right(), y));
|
||||||
|
}
|
||||||
|
|
||||||
|
QColor JKQTPXYFunctionLineGraphBase::getKeyLabelColor() const {
|
||||||
|
return getLineColor();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void JKQTPXYFunctionLineGraphBase::draw(JKQTPEnhancedPainter& painter) {
|
||||||
|
#ifdef JKQTBP_AUTOTIMER
|
||||||
|
JKQTPAutoOutputTimer jkaaot("JKQTPXYFunctionLineGraph::draw");
|
||||||
|
#endif
|
||||||
|
if (parent==nullptr) return;
|
||||||
|
JKQTPDatastore* datastore=parent->getDatastore();
|
||||||
|
if (datastore==nullptr) return;
|
||||||
|
|
||||||
|
//qDebug()<<"start plot\n";
|
||||||
|
createPlotData();
|
||||||
|
//qDebug()<<"plot data created\n";
|
||||||
|
|
||||||
|
drawErrorsBefore(painter);
|
||||||
|
{
|
||||||
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
||||||
|
|
||||||
|
QPen p=getLinePen(painter, parent);
|
||||||
|
QPen np(Qt::NoPen);
|
||||||
|
|
||||||
|
|
||||||
|
{
|
||||||
|
painter.save(); auto __finalpaintline=JKQTPFinally([&painter]() {painter.restore();});
|
||||||
|
painter.setPen(p);
|
||||||
|
painter.drawPolyline(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (displaySamplePoints) drawSamplePoints(painter, getLineColor());
|
||||||
|
}
|
||||||
|
drawErrorsAfter(painter);
|
||||||
|
//std::cout<<"plot done\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
double JKQTPXYFunctionLineGraphBase::getTMin() const
|
||||||
|
{
|
||||||
|
return tmin;
|
||||||
|
}
|
||||||
|
|
||||||
|
double JKQTPXYFunctionLineGraphBase::getTMax() const
|
||||||
|
{
|
||||||
|
return tmax;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPXYFunctionLineGraphBase::setTMin(double val)
|
||||||
|
{
|
||||||
|
tmin=val;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPXYFunctionLineGraphBase::setTMax(double val)
|
||||||
|
{
|
||||||
|
tmax=val;
|
||||||
|
}
|
||||||
|
|
||||||
|
QPair<double, double> JKQTPXYFunctionLineGraphBase::getTRange() const
|
||||||
|
{
|
||||||
|
return QPair<double, double>(tmin,tmax);
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPXYFunctionLineGraphBase::setTRange(double tmin_, double tmax_)
|
||||||
|
{
|
||||||
|
tmin=tmin_;
|
||||||
|
tmax=tmax_;
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPXYFunctionLineGraphBase::setTRange(const QPair<double, double> &range)
|
||||||
|
{
|
||||||
|
tmin=range.first;
|
||||||
|
tmax=range.second;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(JKQTBasePlotter* parent):
|
||||||
|
JKQTPXYFunctionLineGraphBase(parent)
|
||||||
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(JKQTPlotter* parent):
|
JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(JKQTPlotter* parent):
|
||||||
@ -54,10 +165,9 @@ JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(const jkqtpSimpleParametricCu
|
|||||||
{
|
{
|
||||||
tmin=tmin_;
|
tmin=tmin_;
|
||||||
tmax=tmax_;
|
tmax=tmax_;
|
||||||
title=title_;
|
setTitle(title_);
|
||||||
plotFunction=jkqtpParametricCurveFunctionType();
|
plotFunction=jkqtpParametricCurveFunctionType();
|
||||||
simplePlotFunction=f;
|
simplePlotFunction=f;
|
||||||
data.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(const jkqtpSimpleParametricCurveFunctionType &f, const QString &title_, double tmin_, double tmax_, JKQTPlotter *parent):
|
JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(const jkqtpSimpleParametricCurveFunctionType &f, const QString &title_, double tmin_, double tmax_, JKQTPlotter *parent):
|
||||||
@ -72,7 +182,7 @@ JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(jkqtpSimpleParametricCurveFun
|
|||||||
{
|
{
|
||||||
tmin=tmin_;
|
tmin=tmin_;
|
||||||
tmax=tmax_;
|
tmax=tmax_;
|
||||||
title=title_;
|
setTitle(title_);
|
||||||
plotFunction=jkqtpParametricCurveFunctionType();
|
plotFunction=jkqtpParametricCurveFunctionType();
|
||||||
simplePlotFunction=std::move(f);
|
simplePlotFunction=std::move(f);
|
||||||
data.clear();
|
data.clear();
|
||||||
@ -89,7 +199,7 @@ JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(jkqtpParametricCurveFunctionT
|
|||||||
{
|
{
|
||||||
tmin=tmin_;
|
tmin=tmin_;
|
||||||
tmax=tmax_;
|
tmax=tmax_;
|
||||||
title=title_;
|
setTitle(title_);
|
||||||
simplePlotFunction=jkqtpSimpleParametricCurveFunctionType();
|
simplePlotFunction=jkqtpSimpleParametricCurveFunctionType();
|
||||||
plotFunction=std::move(f);
|
plotFunction=std::move(f);
|
||||||
data.clear();
|
data.clear();
|
||||||
@ -106,7 +216,7 @@ JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(const jkqtpParametricCurveFun
|
|||||||
{
|
{
|
||||||
tmin=tmin_;
|
tmin=tmin_;
|
||||||
tmax=tmax_;
|
tmax=tmax_;
|
||||||
title=title_;
|
setTitle(title_);
|
||||||
simplePlotFunction=jkqtpSimpleParametricCurveFunctionType();
|
simplePlotFunction=jkqtpSimpleParametricCurveFunctionType();
|
||||||
plotFunction=std::move(f);
|
plotFunction=std::move(f);
|
||||||
data.clear();
|
data.clear();
|
||||||
@ -119,7 +229,7 @@ JKQTPXYFunctionLineGraph::JKQTPXYFunctionLineGraph(const jkqtpParametricCurveFun
|
|||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXYFunctionLineGraph::~JKQTPXYFunctionLineGraph() {
|
JKQTPXYFunctionLineGraph::~JKQTPXYFunctionLineGraph() {
|
||||||
data.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -162,115 +272,26 @@ jkqtpSimpleParametricCurveFunctionType JKQTPXYFunctionLineGraph::getSimplePlotFu
|
|||||||
return simplePlotFunction;
|
return simplePlotFunction;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool JKQTPXYFunctionLineGraph::isSimplePlotFunction() const
|
||||||
void JKQTPXYFunctionLineGraph::drawKeyMarker(JKQTPEnhancedPainter& painter, QRectF& rect) {
|
|
||||||
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
||||||
QPen p=getLinePen(painter, parent);
|
|
||||||
p.setJoinStyle(Qt::RoundJoin);
|
|
||||||
p.setCapStyle(Qt::RoundCap);
|
|
||||||
QPen np(Qt::NoPen);
|
|
||||||
const double y=rect.top()+rect.height()/2.0;
|
|
||||||
painter.setPen(np);
|
|
||||||
painter.setPen(p);
|
|
||||||
painter.drawLine(QLineF(rect.left(), y, rect.right(), y));
|
|
||||||
}
|
|
||||||
|
|
||||||
QColor JKQTPXYFunctionLineGraph::getKeyLabelColor() const {
|
|
||||||
return getLineColor();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void JKQTPXYFunctionLineGraph::createPlotData(bool collectParams) {
|
|
||||||
#ifdef JKQTBP_AUTOTIMER
|
|
||||||
JKQTPAutoOutputTimer jkaat(QString("JKQTPXYFunctionLineGraph[%1]::createPlotData()").arg(title));
|
|
||||||
#endif
|
|
||||||
data.clear();
|
|
||||||
if (collectParams) collectParameters();
|
|
||||||
|
|
||||||
if (parent==nullptr) return;
|
|
||||||
if (!plotFunction && !simplePlotFunction) return;
|
|
||||||
|
|
||||||
jkqtpSimpleParametricCurveFunctionType func;
|
|
||||||
if (plotFunction) func=std::bind(plotFunction, std::placeholders::_1, getInternalParams());
|
|
||||||
else if (simplePlotFunction) func=simplePlotFunction;
|
|
||||||
|
|
||||||
jkqtpSimpleParametricCurveFunctionType fTransformedFunc= std::bind([&](const JKQTPPlotElement* plot, double t) -> QPointF { return plot->transform(func(t)); }, this, std::placeholders::_1);
|
|
||||||
|
|
||||||
JKQTPAdaptiveFunctionGraphEvaluator evaluator(fTransformedFunc, minSamples, maxRefinementDegree, slopeTolerance, minPixelPerSample);
|
|
||||||
data=evaluator.evaluate(tmin, tmax);
|
|
||||||
data=JKQTPSimplyfyLineSegemnts(data, dataCleanupMaxAllowedAngleDegree);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void JKQTPXYFunctionLineGraph::draw(JKQTPEnhancedPainter& painter) {
|
|
||||||
#ifdef JKQTBP_AUTOTIMER
|
|
||||||
JKQTPAutoOutputTimer jkaaot("JKQTPXYFunctionLineGraph::draw");
|
|
||||||
#endif
|
|
||||||
if (parent==nullptr) return;
|
|
||||||
JKQTPDatastore* datastore=parent->getDatastore();
|
|
||||||
if (datastore==nullptr) return;
|
|
||||||
|
|
||||||
//qDebug()<<"start plot\n";
|
|
||||||
createPlotData();
|
|
||||||
//qDebug()<<"plot data created\n";
|
|
||||||
|
|
||||||
drawErrorsBefore(painter);
|
|
||||||
{
|
|
||||||
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
||||||
|
|
||||||
QPen p=getLinePen(painter, parent);
|
|
||||||
QPen np(Qt::NoPen);
|
|
||||||
|
|
||||||
|
|
||||||
{
|
|
||||||
painter.save(); auto __finalpaintline=JKQTPFinally([&painter]() {painter.restore();});
|
|
||||||
painter.setPen(p);
|
|
||||||
painter.drawPolyline(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (displaySamplePoints) drawSamplePoints(painter, getLineColor());
|
|
||||||
}
|
|
||||||
drawErrorsAfter(painter);
|
|
||||||
//std::cout<<"plot done\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
double JKQTPXYFunctionLineGraph::getTMin() const
|
|
||||||
{
|
{
|
||||||
return tmin;
|
return !static_cast<bool>(plotFunction) && static_cast<bool>(simplePlotFunction);
|
||||||
}
|
}
|
||||||
|
|
||||||
double JKQTPXYFunctionLineGraph::getTMax() const
|
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec JKQTPXYFunctionLineGraph::buildPlotFunctorSpec()
|
||||||
{
|
{
|
||||||
return tmax;
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec spec;
|
||||||
}
|
|
||||||
|
if (!plotFunction && !simplePlotFunction) return spec; // return invalid spec!
|
||||||
void JKQTPXYFunctionLineGraph::setTMin(double val)
|
|
||||||
{
|
// range over which to evaluate func
|
||||||
tmin=val;
|
spec.range_start=tmin;
|
||||||
}
|
spec.range_end=tmax;
|
||||||
|
|
||||||
void JKQTPXYFunctionLineGraph::setTMax(double val)
|
// the actual function to use
|
||||||
{
|
if (plotFunction) spec.func=std::bind(plotFunction, std::placeholders::_1, getInternalParams());
|
||||||
tmax=val;
|
else if (simplePlotFunction) spec.func=simplePlotFunction;
|
||||||
}
|
|
||||||
|
return spec;
|
||||||
QPair<double, double> JKQTPXYFunctionLineGraph::getTRange() const
|
|
||||||
{
|
|
||||||
return QPair<double, double>(tmin,tmax);
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXYFunctionLineGraph::setTRange(double tmin_, double tmax_)
|
|
||||||
{
|
|
||||||
tmin=tmin_;
|
|
||||||
tmax=tmax_;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPXYFunctionLineGraph::setTRange(const QPair<double, double> &range)
|
|
||||||
{
|
|
||||||
tmin=range.first;
|
|
||||||
tmax=range.second;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,62 +33,136 @@
|
|||||||
#define jkqtpevaluatedparametriccurve_H
|
#define jkqtpevaluatedparametriccurve_H
|
||||||
|
|
||||||
|
|
||||||
|
/** \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 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, QRectF& rect) override;
|
||||||
|
/** \brief returns the color to be used for the key label */
|
||||||
|
virtual QColor getKeyLabelColor() const override;
|
||||||
|
|
||||||
|
|
||||||
/*! \brief type of functions that may be plotted by JKQTPXYFunctionLineGraph
|
/** \brief returns the t-value range for \f$ [x,y]=f(t), t \in \left[t_\text{min}, t_\text{max}\right] \f$ */
|
||||||
\ingroup jkqtplotter_functiongraphs
|
QPair<double,double> getTRange() const;
|
||||||
|
/** \copydoc tmin */
|
||||||
|
double getTMin() const;
|
||||||
|
/** \copydoc tmax */
|
||||||
|
double getTMax() const;
|
||||||
|
|
||||||
This is the type of functions \f$ [x,y]=f(t, \vec{p}) \f$ that may be plottet by JKQTPXYFunctionLineGraph.
|
public slots:
|
||||||
It is possible to supply parameters \f$ \vec{p} \f$ to the function that
|
/** \copydoc tmin */
|
||||||
influence its result. Parameters are given as a pointer to some memory location. The function has to
|
void setTMin(double val);
|
||||||
know on its own how to interpret these.
|
/** \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;
|
typedef std::function<QPointF(double, const QVector<double>)> jkqtpParametricCurveFunctionType;
|
||||||
|
|
||||||
/*! \brief simplified type of functions (without parameters) that may be plotted by JKQTPXYFunctionLineGraph
|
/** \brief simplified type of functions (without parameters) that may be plotted by JKQTPXYFunctionLineGraph
|
||||||
\ingroup jkqtplotter_functiongraphs
|
* \ingroup jkqtplotter_functiongraphs
|
||||||
|
*
|
||||||
This is the type of functions \f$ [x,y]=f(t) \f$ that may be plottet by JKQTPXYFunctionLineGraph
|
* This is the type of functions \f$ [x,y]=f(t) \f$ that may be plottet by JKQTPXYFunctionLineGraph
|
||||||
and JKQTPYFunctionLineGraph.
|
* and JKQTPYFunctionLineGraph.
|
||||||
*/
|
*/
|
||||||
typedef std::function<QPointF(double)> jkqtpSimpleParametricCurveFunctionType;
|
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$
|
/** \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$
|
* The function is evaluated on a user-specified range \f$ t \in \left[t_\text{min}, t_\text{max}\right] \f$
|
||||||
\ingroup jkqtplotter_functiongraphs
|
* \ingroup jkqtplotter_functiongraphs
|
||||||
|
*
|
||||||
This class uses the intelligent plotting algorithm for functions, implemented in JKQTPAdaptiveFunctionGraphEvaluator.
|
* \see JKQTPXYFunctionLineGraphBase for details on the used plotting algorithm
|
||||||
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$
|
* The following image shows a Lissajou's fugure drawn with this function
|
||||||
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]
|
* \image html plot_evalcurve.png
|
||||||
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.
|
* The source code for this example is:
|
||||||
|
* \code
|
||||||
Finally the obtained data is cleaned up to reduce the amount of points, by deleting a point, when it leads to an
|
* JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
||||||
angle between consecutive line-segments of less than dataCleanupMaxAllowedAngleDegree.
|
* // here we define the C++-functor for [x,y]=f(t)
|
||||||
|
* func1->setPlotFunctionFunctor([](double t) -> QPointF {
|
||||||
the following image shows a Lissajou's fugure drawn with this function
|
* const double a=5;
|
||||||
|
* const double b=4;
|
||||||
\image html plot_evalcurve.png
|
* const double delta=JKQTPSTATISTICS_PI/4.0;
|
||||||
|
* return QPointF(sin(a*t+delta), sin(b*t));
|
||||||
The source code for this example is:
|
* });
|
||||||
\code
|
* // and define the range over which to evaluate
|
||||||
JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
* func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
||||||
func1->setPlotFunctionFunctor([](double t) -> QPointF {
|
* \endcode
|
||||||
const double a=5;
|
*
|
||||||
const double b=4;
|
* The source code for the same example, but using functions that also get a parameter vector:
|
||||||
const double delta=JKQTPSTATISTICS_PI/4.0;
|
* \code
|
||||||
return QPointF(sin(a*t+delta), sin(b*t));
|
* JKQTPXYFunctionLineGraph* func1=new JKQTPXYFunctionLineGraph(plot);
|
||||||
});
|
* // here we define the C++-functor for [x,y]=f(t)
|
||||||
func1->setTRange(0, 2.0*JKQTPSTATISTICS_PI);
|
* func1->setPlotFunctionFunctor([](double t, const QVector<double>& params) -> QPointF {
|
||||||
func1->setTitle("C++-inline function $[ sin(5{\\cdot}t+\\pi/4), sin(4{\\cdot}t) ]$");
|
* return QPointF(3.0*sin(params[0]*t+params[2])+8.0, 3.0*sin(params[1]*t));
|
||||||
\endcode
|
* });
|
||||||
|
* // now we define the 3 parameters of the function
|
||||||
\see \ref JKQTPlotterEvalCurves , JKQTPAdaptiveFunctionGraphEvaluator, JKQTPXFunctionLineGraph, JKQTPYFunctionLineGraph
|
* // 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 JKQTPEvaluatedFunctionWithParamsGraphBase, public JKQTPGraphLineStyleMixin {
|
class JKQTPLOTTER_LIB_EXPORT JKQTPXYFunctionLineGraph: public JKQTPXYFunctionLineGraphBase {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
|
|
||||||
@ -117,60 +191,44 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPXYFunctionLineGraph: public JKQTPEvaluatedFunc
|
|||||||
/** \brief class destructor */
|
/** \brief class destructor */
|
||||||
virtual ~JKQTPXYFunctionLineGraph() override;
|
virtual ~JKQTPXYFunctionLineGraph() 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, QRectF& rect) override;
|
|
||||||
/** \brief returns the color to be used for the key label */
|
|
||||||
virtual QColor getKeyLabelColor() const override;
|
|
||||||
|
|
||||||
/** \brief sets a functor to be plotted
|
/** \brief sets a functor to be plotted
|
||||||
*
|
*
|
||||||
* \see plotFunction
|
* \see plotFunction
|
||||||
*/
|
*/
|
||||||
virtual void setPlotFunctionFunctor (jkqtpParametricCurveFunctionType && __value);
|
void setPlotFunctionFunctor (jkqtpParametricCurveFunctionType && __value);
|
||||||
/** \brief sets a functor to be plotted
|
/** \brief sets a functor to be plotted
|
||||||
*
|
*
|
||||||
* \see plotFunction
|
* \see plotFunction
|
||||||
*/
|
*/
|
||||||
virtual void setPlotFunctionFunctor (const jkqtpParametricCurveFunctionType & __value);
|
void setPlotFunctionFunctor (const jkqtpParametricCurveFunctionType & __value);
|
||||||
/** \brief sets a functor to be plotted
|
/** \brief sets a functor to be plotted
|
||||||
*
|
*
|
||||||
* \see simplePlotFunction
|
* \see simplePlotFunction
|
||||||
*/
|
*/
|
||||||
virtual void setPlotFunctionFunctor (jkqtpSimpleParametricCurveFunctionType && __value);
|
void setPlotFunctionFunctor (jkqtpSimpleParametricCurveFunctionType && __value);
|
||||||
/** \brief sets a functor to be plotted
|
/** \brief sets a functor to be plotted
|
||||||
*
|
*
|
||||||
* \see simplePlotFunction
|
* \see simplePlotFunction
|
||||||
*/
|
*/
|
||||||
virtual void setPlotFunctionFunctor (const jkqtpSimpleParametricCurveFunctionType & __value);
|
void setPlotFunctionFunctor (const jkqtpSimpleParametricCurveFunctionType & __value);
|
||||||
/*! \copydoc plotFunction */ \
|
/** \copydoc plotFunction
|
||||||
virtual jkqtpParametricCurveFunctionType getPlotFunctionFunctor () const;
|
*
|
||||||
/*! \copydoc simplePlotFunction */ \
|
* \see isSimplePlotFunction() */
|
||||||
virtual jkqtpSimpleParametricCurveFunctionType getSimplePlotFunction () const;
|
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;
|
||||||
|
|
||||||
|
|
||||||
/*! \copydoc tmin */
|
|
||||||
double getTMin() const;
|
|
||||||
/*! \copydoc tmax */
|
|
||||||
double getTMax() const;
|
|
||||||
/*! \copydoc tmin */
|
|
||||||
void setTMin(double val);
|
|
||||||
/*! \copydoc tmax */
|
|
||||||
void setTMax(double val);
|
|
||||||
/*! \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;
|
|
||||||
/*! \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:
|
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 */
|
/** \copydoc JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec() */
|
||||||
double tmin;
|
virtual PlotFunctorSpec buildPlotFunctorSpec() override;
|
||||||
/** \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 */
|
|
||||||
double tmax;
|
|
||||||
/** \brief fill the data array with data from the function plotFunction */
|
|
||||||
virtual void createPlotData( bool collectParams=true) override;
|
|
||||||
|
|
||||||
|
|
||||||
/** \brief the function to be plotted */
|
/** \brief the function to be plotted */
|
||||||
|
@ -27,131 +27,73 @@
|
|||||||
#include <QElapsedTimer>
|
#include <QElapsedTimer>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
|
JKQTPParsedFunctionLineGraphBase::JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName_, const QString& function_, JKQTBasePlotter *parent):
|
||||||
JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraph(JKQTBasePlotter *parent):
|
JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase(parent),
|
||||||
JKQTPXFunctionLineGraph(parent)
|
dependentVariableName(dependentVariableName_),
|
||||||
|
function(function_)
|
||||||
{
|
{
|
||||||
fdata.parser=new JKQTPMathParser();
|
fdata.parser=std::make_shared<JKQTPMathParser>();
|
||||||
fdata.node=nullptr;
|
fdata.node=nullptr;
|
||||||
fdata.varcount=0;
|
fdata.varcount=0;
|
||||||
function="";
|
|
||||||
parameterColumn=-1;
|
|
||||||
setPlotFunctionFunctor(jkqtpPlotFunctionType(std::bind(&JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &fdata)));
|
|
||||||
|
|
||||||
efdata.parser=new JKQTPMathParser();
|
|
||||||
|
efdata.parser=std::make_shared<JKQTPMathParser>();
|
||||||
efdata.node=nullptr;
|
efdata.node=nullptr;
|
||||||
efdata.varcount=0;
|
efdata.varcount=0;
|
||||||
errorFunction="";
|
|
||||||
errorParameterColumn=-1;
|
|
||||||
setErrorPlotFunction(jkqtpPlotFunctionType(std::bind(&JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &efdata)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraph(JKQTPlotter *parent):
|
JKQTPParsedFunctionLineGraphBase::JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName, const QString& function, JKQTPlotter *parent):
|
||||||
JKQTPXParsedFunctionLineGraph(parent->getPlotter())
|
JKQTPParsedFunctionLineGraphBase(dependentVariableName, function, parent->getPlotter())
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPXParsedFunctionLineGraph::~JKQTPXParsedFunctionLineGraph()
|
JKQTPParsedFunctionLineGraphBase::JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName, JKQTBasePlotter *parent):
|
||||||
|
JKQTPParsedFunctionLineGraphBase(dependentVariableName, QString(), parent)
|
||||||
{
|
{
|
||||||
if (fdata.node) delete fdata.node;
|
|
||||||
delete fdata.parser;
|
|
||||||
if (efdata.node) delete efdata.node;
|
|
||||||
delete efdata.parser;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXParsedFunctionLineGraph::setFunction(const QString &__value)
|
JKQTPParsedFunctionLineGraphBase::JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName, JKQTPlotter *parent):
|
||||||
|
JKQTPParsedFunctionLineGraphBase(dependentVariableName, QString(), parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPParsedFunctionLineGraphBase::~JKQTPParsedFunctionLineGraphBase()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPParsedFunctionLineGraphBase::setFunction(const QString &__value)
|
||||||
{
|
{
|
||||||
this->function = __value;
|
this->function = __value;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString JKQTPXParsedFunctionLineGraph::getFunction() const
|
QString JKQTPParsedFunctionLineGraphBase::getFunction() const
|
||||||
{
|
{
|
||||||
return this->function;
|
return this->function;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPXParsedFunctionLineGraph::setErrorFunction(const QString &__value)
|
void JKQTPParsedFunctionLineGraphBase::setErrorFunction(const QString &__value)
|
||||||
{
|
{
|
||||||
this->errorFunction = __value;
|
this->errorFunction = __value;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString JKQTPXParsedFunctionLineGraph::getErrorFunction() const
|
QString JKQTPParsedFunctionLineGraphBase::getErrorFunction() const
|
||||||
{
|
{
|
||||||
return this->errorFunction;
|
return this->errorFunction;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void JKQTPXParsedFunctionLineGraph::createPlotData(bool /*collectParams*/)
|
QString JKQTPParsedFunctionLineGraphBase::getDependentVariableName() const
|
||||||
{
|
{
|
||||||
collectParameters();
|
return dependentVariableName;
|
||||||
|
|
||||||
//QElapsedTimer timer;
|
|
||||||
//timer.start();
|
|
||||||
for (int i=0; i<fdata.varcount; i++) {
|
|
||||||
fdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
|
||||||
}
|
|
||||||
fdata.varcount=0;
|
|
||||||
try {
|
|
||||||
for (const auto& p: getInternalParams()) {
|
|
||||||
fdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(fdata.varcount+1), p);
|
|
||||||
fdata.varcount=fdata.varcount+1;
|
|
||||||
}
|
|
||||||
fdata.parser->addVariableDouble(std::string("x"), 0.0);
|
|
||||||
if (fdata.node) delete fdata.node;
|
|
||||||
//qint64 t=timer.elapsed();
|
|
||||||
|
|
||||||
|
|
||||||
//qDebug()<<"createPlotData(): adding variables: "<<t<<"ms";
|
|
||||||
fdata.node=fdata.parser->parse(function.toStdString());
|
|
||||||
//qDebug()<<"createPlotData(): parsing: "<<timer.elapsed()-t<<"ms";
|
|
||||||
} catch(std::exception& E) {
|
|
||||||
qDebug()<<QString("parser error: %1").arg(E.what());
|
|
||||||
}
|
|
||||||
|
|
||||||
//qint64 t0=timer.elapsed();
|
|
||||||
for (int i=0; i<efdata.varcount; i++) {
|
|
||||||
efdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
|
||||||
}
|
|
||||||
efdata.varcount=0;
|
|
||||||
try {
|
|
||||||
QVector<double>* errorParameters=static_cast<QVector<double>*>(errorParams);
|
|
||||||
if (errorParameters) {
|
|
||||||
for (int i=0; i<errorParameters->size(); i++) {
|
|
||||||
efdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(efdata.varcount+1), errorParameters->at(i));
|
|
||||||
efdata.varcount=efdata.varcount+1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
efdata.parser->addVariableDouble(std::string("x"), 0.0);
|
|
||||||
if (efdata.node) delete efdata.node;
|
|
||||||
//qint64 t=timer.elapsed();
|
|
||||||
//qDebug()<<"createPlotData(): adding variables: "<<t-t0<<"ms";
|
|
||||||
efdata.node=efdata.parser->parse(errorFunction.toStdString());
|
|
||||||
//qDebug()<<"createPlotData(): parsing: "<<timer.elapsed()-t<<"ms";
|
|
||||||
} catch(std::exception& /*E*/) {
|
|
||||||
//qDebug()<<QString("parser error: %1").arg(E.what());
|
|
||||||
}
|
|
||||||
|
|
||||||
setPlotFunctionFunctor(jkqtpPlotFunctionType(std::bind(&JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &fdata)));
|
|
||||||
setErrorPlotFunction(jkqtpPlotFunctionType(std::bind(&JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &efdata)));
|
|
||||||
|
|
||||||
//qint64 t=timer.elapsed();
|
|
||||||
JKQTPXFunctionLineGraph::createPlotData(false);
|
|
||||||
//qDebug()<<"createPlotData(): JKQTPXFunctionLineGraph::createPlotData(): "<<timer.elapsed()-t<<"ms";
|
|
||||||
|
|
||||||
/*int count=0;
|
|
||||||
doublePair* d=data;
|
|
||||||
while (d!=nullptr) {
|
|
||||||
count++;
|
|
||||||
d=d->next;
|
|
||||||
}
|
|
||||||
qDebug()<<"refined to "<<count<<" daatapoints";*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
double JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunction(double x, const QVector<double>& /*data*/, JKQTPXParsedFunctionLineGraphFunctionData *fdata) {
|
double JKQTPParsedFunctionLineGraphBase::evaluateParsedFunction(double t, ParsedFunctionLineGraphFunctionData *fdata) {
|
||||||
JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunctionData* d=fdata;//static_cast<JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunctionData*>(data);
|
JKQTPParsedFunctionLineGraphBase::ParsedFunctionLineGraphFunctionData* d=fdata;//static_cast<JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunctionData*>(data);
|
||||||
if (d && d->parser && d->node) {
|
if (d && d->parser && d->node) {
|
||||||
try {
|
try {
|
||||||
d->parser->addVariableDouble("x", x);
|
d->parser->addVariableDouble(d->dependentVariableName.toStdString(), t);
|
||||||
JKQTPMathParser::jkmpResult r=d->node->evaluate();
|
JKQTPMathParser::jkmpResult r=d->node->evaluate();
|
||||||
|
|
||||||
if (r.isValid) {
|
if (r.isValid) {
|
||||||
@ -162,8 +104,8 @@ double JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunction(doub
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch(std::exception& E) {
|
} catch(std::exception& E) {
|
||||||
qDebug()<<QString("parser error: %1").arg(E.what());
|
qDebug()<<QString("parser error: %1").arg(E.what());
|
||||||
/*ok= QMessageBox::critical(this, tr("QuickFit-table"),
|
/*ok= QMessageBox::critical(this, tr("QuickFit-table"),
|
||||||
tr("An error occured while parsing the expression '%1' in cell (row, column)=(%3, %4):\n%2\n\n\"OK\" will still go on evaluating\n\"Cancel\" will cancel evaluation for the rest of the cells.").arg(dlgMathExpression->getExpression()).arg(E.what()).arg(row).arg(column),
|
tr("An error occured while parsing the expression '%1' in cell (row, column)=(%3, %4):\n%2\n\n\"OK\" will still go on evaluating\n\"Cancel\" will cancel evaluation for the rest of the cells.").arg(dlgMathExpression->getExpression()).arg(E.what()).arg(row).arg(column),
|
||||||
QMessageBox::Ok|QMessageBox::Cancel, QMessageBox::Ok)==QMessageBox::Ok;*/
|
QMessageBox::Ok|QMessageBox::Cancel, QMessageBox::Ok)==QMessageBox::Ok;*/
|
||||||
|
|
||||||
@ -186,79 +128,152 @@ double JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraphFunction(doub
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraph(JKQTBasePlotter *parent):
|
||||||
|
JKQTPParsedFunctionLineGraphBase(QString("x"), parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraph(JKQTPlotter *parent):
|
||||||
|
JKQTPParsedFunctionLineGraphBase(QString("x"), parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraph(const QString& function, JKQTBasePlotter *parent):
|
||||||
|
JKQTPParsedFunctionLineGraphBase(QString("x"), function, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXParsedFunctionLineGraph::JKQTPXParsedFunctionLineGraph(const QString& function, JKQTPlotter *parent):
|
||||||
|
JKQTPParsedFunctionLineGraphBase(QString("x"), function, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
JKQTPXParsedFunctionLineGraph::~JKQTPXParsedFunctionLineGraph()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void JKQTPXParsedFunctionLineGraph::draw(JKQTPEnhancedPainter &painter)
|
||||||
|
{
|
||||||
|
drawXGraph(painter);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec JKQTPXParsedFunctionLineGraph::buildPlotFunctorSpec()
|
||||||
|
{
|
||||||
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec spec;
|
||||||
|
|
||||||
|
if (parent==nullptr) return spec; // return an invalid PlotFunctorSpec
|
||||||
|
|
||||||
|
for (int i=0; i<fdata.varcount; i++) {
|
||||||
|
fdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
||||||
|
}
|
||||||
|
fdata.varcount=0;
|
||||||
|
try {
|
||||||
|
for (const auto& p: getInternalParams()) {
|
||||||
|
fdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(fdata.varcount+1), p);
|
||||||
|
fdata.varcount=fdata.varcount+1;
|
||||||
|
}
|
||||||
|
fdata.dependentVariableName=getDependentVariableName();
|
||||||
|
fdata.parser->addVariableDouble(getDependentVariableName().toStdString(), 0.0);
|
||||||
|
fdata.node=std::shared_ptr<JKQTPMathParser::jkmpNode>(fdata.parser->parse(function.toStdString()));
|
||||||
|
} catch(std::exception& E) {
|
||||||
|
qDebug()<<QString("parser error: %1").arg(E.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
jkqtpSimplePlotFunctionType plotFunction=std::bind(&JKQTPXParsedFunctionLineGraph::evaluateParsedFunction, std::placeholders::_1, &fdata);
|
||||||
|
// the actual function to use
|
||||||
|
spec.func=std::bind([=](double x) -> QPointF { return QPointF(0, plotFunction(x)); }, std::placeholders::_1);
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::function<QPointF (double)> JKQTPXParsedFunctionLineGraph::buildErrorFunctorSpec()
|
||||||
|
{
|
||||||
|
std::function<QPointF (double)> spec;
|
||||||
|
|
||||||
|
for (int i=0; i<efdata.varcount; i++) {
|
||||||
|
efdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
||||||
|
}
|
||||||
|
efdata.varcount=0;
|
||||||
|
try {
|
||||||
|
for (const auto& p: getInternalErrorParams()) {
|
||||||
|
efdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(efdata.varcount+1), p);
|
||||||
|
efdata.varcount=efdata.varcount+1;
|
||||||
|
}
|
||||||
|
efdata.dependentVariableName=getDependentVariableName();
|
||||||
|
efdata.parser->addVariableDouble(getDependentVariableName().toStdString(), 0.0);
|
||||||
|
efdata.node=std::shared_ptr<JKQTPMathParser::jkmpNode>(efdata.parser->parse(errorFunction.toStdString()));
|
||||||
|
} catch(std::exception& /*E*/) {
|
||||||
|
//qDebug()<<QString("parser error: %1").arg(E.what());
|
||||||
|
}
|
||||||
|
|
||||||
|
jkqtpSimplePlotFunctionType errorPlotFunction=std::bind(&JKQTPXParsedFunctionLineGraph::evaluateParsedFunction, std::placeholders::_1, &efdata);
|
||||||
|
// the actual function to use
|
||||||
|
spec=std::bind([=](double x) -> QPointF { return QPointF(0, errorPlotFunction(x)); }, std::placeholders::_1);
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraph(JKQTBasePlotter *parent):
|
JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraph(JKQTBasePlotter *parent):
|
||||||
JKQTPYFunctionLineGraph(parent)
|
JKQTPParsedFunctionLineGraphBase(QString("y"), parent)
|
||||||
{
|
{
|
||||||
fdata.parser=new JKQTPMathParser();
|
|
||||||
fdata.node=nullptr;
|
|
||||||
fdata.varcount=0;
|
|
||||||
function="";
|
|
||||||
parameterColumn=-1;
|
|
||||||
setPlotFunctionFunctor(jkqtpPlotFunctionType(std::bind(&JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &fdata)));
|
|
||||||
|
|
||||||
efdata.parser=new JKQTPMathParser();
|
|
||||||
efdata.node=nullptr;
|
|
||||||
efdata.varcount=0;
|
|
||||||
errorFunction="";
|
|
||||||
errorParameterColumn=-1;
|
|
||||||
setErrorPlotFunction(jkqtpPlotFunctionType(std::bind(&JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &efdata)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraph(JKQTPlotter *parent):
|
JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraph(JKQTPlotter *parent):
|
||||||
JKQTPYFunctionLineGraph(parent)
|
JKQTPParsedFunctionLineGraphBase(QString("y"), parent)
|
||||||
{
|
{
|
||||||
fdata.parser=new JKQTPMathParser();
|
}
|
||||||
fdata.node=nullptr;
|
|
||||||
fdata.varcount=0;
|
|
||||||
function="";
|
|
||||||
parameterColumn=-1;
|
|
||||||
setPlotFunctionFunctor(jkqtpPlotFunctionType(std::bind(&JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &fdata)));
|
|
||||||
|
|
||||||
efdata.parser=new JKQTPMathParser();
|
JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraph(const QString& function, JKQTBasePlotter *parent):
|
||||||
efdata.node=nullptr;
|
JKQTPParsedFunctionLineGraphBase(QString("y"), function, parent)
|
||||||
efdata.varcount=0;
|
{
|
||||||
errorFunction="";
|
}
|
||||||
errorParameterColumn=-1;
|
|
||||||
setErrorPlotFunction(jkqtpPlotFunctionType(std::bind(&JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &efdata)));
|
JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraph(const QString& function, JKQTPlotter *parent):
|
||||||
|
JKQTPParsedFunctionLineGraphBase(QString("y"), function, parent)
|
||||||
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
JKQTPYParsedFunctionLineGraph::~JKQTPYParsedFunctionLineGraph()
|
JKQTPYParsedFunctionLineGraph::~JKQTPYParsedFunctionLineGraph()
|
||||||
{
|
{
|
||||||
if (fdata.node) delete fdata.node;
|
|
||||||
delete fdata.parser;
|
|
||||||
if (efdata.node) delete efdata.node;
|
|
||||||
delete efdata.parser;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void JKQTPYParsedFunctionLineGraph::setFunction(const QString &__value)
|
void JKQTPYParsedFunctionLineGraph::draw(JKQTPEnhancedPainter &painter)
|
||||||
{
|
{
|
||||||
this->function = __value;
|
drawYGraph(painter);
|
||||||
}
|
}
|
||||||
|
|
||||||
QString JKQTPYParsedFunctionLineGraph::getFunction() const
|
|
||||||
{
|
|
||||||
return this->function;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPYParsedFunctionLineGraph::setErrorFunction(const QString &__value)
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec JKQTPYParsedFunctionLineGraph::buildPlotFunctorSpec()
|
||||||
{
|
{
|
||||||
this->errorFunction = __value;
|
JKQTPEvaluatedFunctionGraphBase::PlotFunctorSpec spec;
|
||||||
}
|
|
||||||
|
|
||||||
QString JKQTPYParsedFunctionLineGraph::getErrorFunction() const
|
if (parent==nullptr) return spec; // return an invalid PlotFunctorSpec
|
||||||
{
|
|
||||||
return this->errorFunction;
|
|
||||||
}
|
|
||||||
|
|
||||||
void JKQTPYParsedFunctionLineGraph::createPlotData(bool /*collectParams*/)
|
|
||||||
{
|
|
||||||
collectParameters();
|
|
||||||
|
|
||||||
//QElapsedTimer timer;
|
|
||||||
//timer.start();
|
|
||||||
for (int i=0; i<fdata.varcount; i++) {
|
for (int i=0; i<fdata.varcount; i++) {
|
||||||
fdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
fdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
||||||
}
|
}
|
||||||
@ -268,82 +283,47 @@ void JKQTPYParsedFunctionLineGraph::createPlotData(bool /*collectParams*/)
|
|||||||
fdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(fdata.varcount+1), p);
|
fdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(fdata.varcount+1), p);
|
||||||
fdata.varcount=fdata.varcount+1;
|
fdata.varcount=fdata.varcount+1;
|
||||||
}
|
}
|
||||||
fdata.parser->addVariableDouble(std::string("x"), 0.0);
|
fdata.dependentVariableName=getDependentVariableName();
|
||||||
fdata.parser->addVariableDouble(std::string("y"), 0.0);
|
fdata.parser->addVariableDouble(getDependentVariableName().toStdString(), 0.0);
|
||||||
if (fdata.node) delete fdata.node;
|
fdata.node=std::shared_ptr<JKQTPMathParser::jkmpNode>(fdata.parser->parse(function.toStdString()));
|
||||||
//qint64 t=timer.elapsed();
|
|
||||||
|
|
||||||
|
|
||||||
//qDebug()<<"createPlotData(): adding variables: "<<t<<"ms";
|
|
||||||
fdata.node=fdata.parser->parse(function.toStdString());
|
|
||||||
//qDebug()<<"createPlotData(): parsing: "<<timer.elapsed()-t<<"ms";
|
|
||||||
} catch(std::exception& E) {
|
} catch(std::exception& E) {
|
||||||
qDebug()<<QString("parser error: %1").arg(E.what());
|
qDebug()<<QString("parser error: %1").arg(E.what());
|
||||||
}
|
}
|
||||||
|
|
||||||
//qint64 t0=timer.elapsed();
|
jkqtpSimplePlotFunctionType plotFunction=std::bind(&JKQTPXParsedFunctionLineGraph::evaluateParsedFunction, std::placeholders::_1, &fdata);
|
||||||
|
// the actual function to use
|
||||||
|
spec.func=std::bind([=](double y) -> QPointF { return QPointF(plotFunction(y), 0); }, std::placeholders::_1);
|
||||||
|
|
||||||
|
// range over which to evaluate func
|
||||||
|
spec.range_start=parent->getXMin();
|
||||||
|
spec.range_end=parent->getXMax();
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::function<QPointF (double)> JKQTPYParsedFunctionLineGraph::buildErrorFunctorSpec()
|
||||||
|
{
|
||||||
|
std::function<QPointF (double)> spec;
|
||||||
|
|
||||||
for (int i=0; i<efdata.varcount; i++) {
|
for (int i=0; i<efdata.varcount; i++) {
|
||||||
efdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
efdata.parser->deleteVariable(std::string("p")+jkqtp_inttostr(i+1));
|
||||||
}
|
}
|
||||||
efdata.varcount=0;
|
efdata.varcount=0;
|
||||||
try {
|
try {
|
||||||
QVector<double>* errorParameters=static_cast<QVector<double>*>(errorParams);
|
for (const auto& p: getInternalErrorParams()) {
|
||||||
if (errorParameters) {
|
efdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(efdata.varcount+1), p);
|
||||||
for (int i=0; i<errorParameters->size(); i++) {
|
efdata.varcount=efdata.varcount+1;
|
||||||
efdata.parser->addVariableDouble(std::string("p")+jkqtp_inttostr(efdata.varcount+1), errorParameters->at(i));
|
|
||||||
efdata.varcount=efdata.varcount+1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
efdata.parser->addVariableDouble(std::string("x"), 0.0);
|
efdata.dependentVariableName=getDependentVariableName();
|
||||||
efdata.parser->addVariableDouble(std::string("y"), 0.0);
|
efdata.parser->addVariableDouble(getDependentVariableName().toStdString(), 0.0);
|
||||||
if (efdata.node) delete efdata.node;
|
efdata.node=std::shared_ptr<JKQTPMathParser::jkmpNode>(efdata.parser->parse(errorFunction.toStdString()));
|
||||||
//qint64 t=timer.elapsed();
|
} catch(std::exception& /*E*/) {
|
||||||
//qDebug()<<"createPlotData(): adding variables: "<<t-t0<<"ms";
|
//qDebug()<<QString("parser error: %1").arg(E.what());
|
||||||
efdata.node=efdata.parser->parse(errorFunction.toStdString());
|
|
||||||
//qDebug()<<"createPlotData(): parsing: "<<timer.elapsed()-t<<"ms";
|
|
||||||
} catch(std::exception& E) {
|
|
||||||
qDebug()<<QString("parser error: %1").arg(E.what());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
setPlotFunctionFunctor(jkqtpPlotFunctionType(std::bind(&JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &fdata)));
|
jkqtpSimplePlotFunctionType errorPlotFunction=std::bind(&JKQTPXParsedFunctionLineGraph::evaluateParsedFunction, std::placeholders::_1, &efdata);
|
||||||
setErrorPlotFunction(jkqtpPlotFunctionType(std::bind(&JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunction, std::placeholders::_1, std::placeholders::_2, &efdata)));
|
// the actual function to use
|
||||||
|
spec=std::bind([=](double y) -> QPointF { return QPointF(errorPlotFunction(y), 0); }, std::placeholders::_1);
|
||||||
//qint64 t=timer.elapsed();
|
|
||||||
JKQTPYFunctionLineGraph::createPlotData(false);
|
|
||||||
//qDebug()<<"createPlotData(): JKQTPYFunctionLineGraph::createPlotData(): "<<timer.elapsed()-t<<"ms";
|
|
||||||
|
|
||||||
/*int count=0;
|
|
||||||
doublePair* d=data;
|
|
||||||
while (d!=nullptr) {
|
|
||||||
count++;
|
|
||||||
d=d->next;
|
|
||||||
}
|
|
||||||
qDebug()<<"refined to "<<count<<" daatapoints";*/
|
|
||||||
}
|
|
||||||
|
|
||||||
double JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunction(double x, const QVector<double> & /*data*/, JKQTPYParsedFunctionLineGraphFunctionData *fdata) {
|
|
||||||
JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunctionData* d=fdata;//static_cast<JKQTPYParsedFunctionLineGraph::JKQTPYParsedFunctionLineGraphFunctionData*>(data);
|
|
||||||
if (d && d->parser && d->node) {
|
|
||||||
try {
|
|
||||||
d->parser->addVariableDouble("x", x);
|
|
||||||
d->parser->addVariableDouble("y", x);
|
|
||||||
JKQTPMathParser::jkmpResult r=d->node->evaluate();
|
|
||||||
|
|
||||||
if (r.isValid) {
|
|
||||||
if (r.type==JKQTPMathParser::jkmpBool) {
|
|
||||||
return r.boolean?1.0:0.0;
|
|
||||||
} else if (r.type==JKQTPMathParser::jkmpDouble) {
|
|
||||||
return r.num;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch(std::exception& /*E*/) {
|
|
||||||
//qDebug()<<QString("parser error: %1").arg(E.what());
|
|
||||||
/*ok= QMessageBox::critical(this, tr("QuickFit-table"),
|
|
||||||
tr("An error occured while parsing the expression '%1' in cell (row, column)=(%3, %4):\n%2\n\n\"OK\" will still go on evaluating\n\"Cancel\" will cancel evaluation for the rest of the cells.").arg(dlgMathExpression->getExpression()).arg(E.what()).arg(row).arg(column),
|
|
||||||
QMessageBox::Ok|QMessageBox::Cancel, QMessageBox::Ok)==QMessageBox::Ok;*/
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return NAN;
|
|
||||||
|
|
||||||
|
return spec;
|
||||||
}
|
}
|
||||||
|
@ -34,6 +34,74 @@ class JKQTBasePlotter;
|
|||||||
class JKQTPlotter;
|
class JKQTPlotter;
|
||||||
|
|
||||||
|
|
||||||
|
/** \brief extends JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase with the capabilities to define functions from strings
|
||||||
|
* that are parsed by JKQTPMathParser
|
||||||
|
* \ingroup jkqtplotter_functiongraphs
|
||||||
|
*
|
||||||
|
* \see JKQTPXParsedFunctionLineGraph and JKQTPYParsedFunctionLineGraph for a concrete implementation, see also JKQTPMathParser
|
||||||
|
*/
|
||||||
|
class JKQTPLOTTER_LIB_EXPORT JKQTPParsedFunctionLineGraphBase: public JKQTPEvaluatedFunctionWithErrorsGraphDrawingBase {
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
|
||||||
|
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName, JKQTBasePlotter* parent=nullptr);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName, JKQTPlotter* parent);
|
||||||
|
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName, const QString& function, JKQTBasePlotter* parent=nullptr);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPParsedFunctionLineGraphBase(const QString& dependentVariableName, const QString& function, JKQTPlotter* parent);
|
||||||
|
|
||||||
|
|
||||||
|
/** \brief class destructor */
|
||||||
|
virtual ~JKQTPParsedFunctionLineGraphBase() override;
|
||||||
|
|
||||||
|
/** \copydoc function */
|
||||||
|
QString getFunction() const;
|
||||||
|
|
||||||
|
/** \copydoc errorFunction */
|
||||||
|
QString getErrorFunction() const;
|
||||||
|
/** \copydoc dependentVariableName */
|
||||||
|
QString getDependentVariableName() const;
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
/** \copydoc errorFunction */
|
||||||
|
void setErrorFunction(const QString & __value);
|
||||||
|
/** \copydoc function */
|
||||||
|
void setFunction(const QString & __value);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
/** \brief INTERNAL data structure combining a JKQTPMathParser and a JKQTPMathParser::jkmpNode
|
||||||
|
*/
|
||||||
|
struct ParsedFunctionLineGraphFunctionData {
|
||||||
|
std::shared_ptr<JKQTPMathParser> parser;
|
||||||
|
std::shared_ptr<JKQTPMathParser::jkmpNode> node;
|
||||||
|
int varcount;
|
||||||
|
QString dependentVariableName;
|
||||||
|
};
|
||||||
|
|
||||||
|
/** \brief nache of the dependent variable (e.g. x for a function f(x) ) */
|
||||||
|
QString dependentVariableName;
|
||||||
|
|
||||||
|
/** \brief the function to be evaluated for the plot. Use \c x as the free variable, e.g. \c "x^2+2" */
|
||||||
|
QString function;
|
||||||
|
/** \brief parser data structure for function */
|
||||||
|
ParsedFunctionLineGraphFunctionData fdata;
|
||||||
|
|
||||||
|
/** \brief the function to be evaluated to add error indicators to the graph. This function is evaluated to an error for every x. Use \c x as the free variable, e.g. \c "x^2+2". */
|
||||||
|
QString errorFunction;
|
||||||
|
/** \brief parser data structure for errorFunction */
|
||||||
|
ParsedFunctionLineGraphFunctionData efdata;
|
||||||
|
|
||||||
|
/** \brief implements the actual plot function */
|
||||||
|
static double evaluateParsedFunction(double x, ParsedFunctionLineGraphFunctionData* fdata) ;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*! \brief This implements line plots where the data is taken from a user supplied function \f$ y=f(x) \f$ The function is defined as a string and parsed by JKMathParser
|
/*! \brief This implements line plots where the data is taken from a user supplied function \f$ y=f(x) \f$ The function is defined as a string and parsed by JKMathParser
|
||||||
\ingroup jkqtplotter_parsedFgraphs
|
\ingroup jkqtplotter_parsedFgraphs
|
||||||
@ -45,7 +113,7 @@ class JKQTPlotter;
|
|||||||
|
|
||||||
\see \ref JKQTPlotterParsedFunctionPlot, JKQTPMathParser
|
\see \ref JKQTPlotterParsedFunctionPlot, JKQTPMathParser
|
||||||
*/
|
*/
|
||||||
class JKQTPLOTTER_LIB_EXPORT JKQTPXParsedFunctionLineGraph: public JKQTPXFunctionLineGraph {
|
class JKQTPLOTTER_LIB_EXPORT JKQTPXParsedFunctionLineGraph: public JKQTPParsedFunctionLineGraphBase {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
|
|
||||||
@ -55,49 +123,26 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPXParsedFunctionLineGraph: public JKQTPXFunctio
|
|||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPXParsedFunctionLineGraph(JKQTPlotter* parent);
|
JKQTPXParsedFunctionLineGraph(JKQTPlotter* parent);
|
||||||
|
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXParsedFunctionLineGraph(const QString& function, JKQTBasePlotter* parent=nullptr);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPXParsedFunctionLineGraph(const QString& function, JKQTPlotter* parent);
|
||||||
|
|
||||||
|
|
||||||
/** \brief class destructor */
|
/** \brief class destructor */
|
||||||
virtual ~JKQTPXParsedFunctionLineGraph() override;
|
virtual ~JKQTPXParsedFunctionLineGraph() override;
|
||||||
|
|
||||||
/*! \copydoc function */
|
/** \brief plots the graph to the plotter object specified as parent */
|
||||||
void setFunction(const QString & __value);
|
virtual void draw(JKQTPEnhancedPainter& painter) override;
|
||||||
/*! \copydoc function */
|
|
||||||
QString getFunction() const;
|
|
||||||
|
|
||||||
/*! \copydoc errorFunction */
|
|
||||||
void setErrorFunction(const QString & __value);
|
|
||||||
/*! \copydoc errorFunction */
|
|
||||||
QString getErrorFunction() const;
|
|
||||||
|
|
||||||
/** \brief INTERNAL data structure
|
|
||||||
* \internal
|
|
||||||
*/
|
|
||||||
struct JKQTPXParsedFunctionLineGraphFunctionData {
|
|
||||||
JKQTPMathParser* parser;
|
|
||||||
JKQTPMathParser::jkmpNode* node;
|
|
||||||
int varcount;
|
|
||||||
};
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/** \brief the function to be evaluated for the plot. Use \c x as the free variable, e.g. \c "x^2+2" */
|
|
||||||
QString function;
|
|
||||||
JKQTPXParsedFunctionLineGraphFunctionData fdata;
|
|
||||||
|
|
||||||
/** \brief the function to be evaluated to add error indicators to the graph. This function is evaluated to an error for every x. Use \c x as the free variable, e.g. \c "x^2+2". */
|
|
||||||
QString errorFunction;
|
|
||||||
JKQTPXParsedFunctionLineGraphFunctionData efdata;
|
|
||||||
|
|
||||||
// hide functions that should not be used in this class!
|
/** \copydoc JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec() */
|
||||||
using JKQTPXFunctionLineGraph::setPlotFunctionFunctor;
|
virtual PlotFunctorSpec buildPlotFunctorSpec() override;
|
||||||
using JKQTPXFunctionLineGraph::setParams;
|
|
||||||
using JKQTPXFunctionLineGraph::setErrorPlotFunction;
|
|
||||||
using JKQTPXFunctionLineGraph::setErrorParams;
|
|
||||||
|
|
||||||
/** \brief fill the data array with data from the function plotFunction */
|
/** \copydoc JKQTPEvaluatedFunctionWithErrorsGraphBase::buildPlotFunctorSpec() */
|
||||||
virtual void createPlotData(bool collectParams=true) override;
|
virtual std::function<QPointF(double)> buildErrorFunctorSpec() override;
|
||||||
|
|
||||||
/** \brief implements the actual plot function */
|
|
||||||
static double JKQTPXParsedFunctionLineGraphFunction(double x, const QVector<double> &data, JKQTPXParsedFunctionLineGraphFunctionData* fdata) ;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -114,7 +159,7 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPXParsedFunctionLineGraph: public JKQTPXFunctio
|
|||||||
\see \ref JKQTPlotterParsedFunctionPlot, JKQTPMathParser
|
\see \ref JKQTPlotterParsedFunctionPlot, JKQTPMathParser
|
||||||
|
|
||||||
*/
|
*/
|
||||||
class JKQTPLOTTER_LIB_EXPORT JKQTPYParsedFunctionLineGraph: public JKQTPYFunctionLineGraph {
|
class JKQTPLOTTER_LIB_EXPORT JKQTPYParsedFunctionLineGraph: public JKQTPParsedFunctionLineGraphBase {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
|
|
||||||
@ -124,47 +169,25 @@ class JKQTPLOTTER_LIB_EXPORT JKQTPYParsedFunctionLineGraph: public JKQTPYFunctio
|
|||||||
/** \brief class constructor */
|
/** \brief class constructor */
|
||||||
JKQTPYParsedFunctionLineGraph(JKQTPlotter* parent);
|
JKQTPYParsedFunctionLineGraph(JKQTPlotter* parent);
|
||||||
|
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPYParsedFunctionLineGraph(const QString& function, JKQTBasePlotter* parent=nullptr);
|
||||||
|
/** \brief class constructor */
|
||||||
|
JKQTPYParsedFunctionLineGraph(const QString& function, JKQTPlotter* parent);
|
||||||
|
|
||||||
|
|
||||||
/** \brief class destructor */
|
/** \brief class destructor */
|
||||||
virtual ~JKQTPYParsedFunctionLineGraph() override;
|
virtual ~JKQTPYParsedFunctionLineGraph() override;
|
||||||
|
|
||||||
/*! \copydoc function */
|
/** \brief plots the graph to the plotter object specified as parent */
|
||||||
void setFunction(const QString & __value);
|
virtual void draw(JKQTPEnhancedPainter& painter) override;
|
||||||
/*! \copydoc function */
|
|
||||||
QString getFunction() const;
|
|
||||||
|
|
||||||
/*! \copydoc errorFunction */
|
|
||||||
void setErrorFunction(const QString & __value);
|
|
||||||
/*! \copydoc errorFunction */
|
|
||||||
QString getErrorFunction() const;
|
|
||||||
|
|
||||||
/** \brief INTERNAL data structure
|
|
||||||
* \internal
|
|
||||||
*/
|
|
||||||
struct JKQTPYParsedFunctionLineGraphFunctionData {
|
|
||||||
JKQTPMathParser* parser;
|
|
||||||
JKQTPMathParser::jkmpNode* node;
|
|
||||||
int varcount;
|
|
||||||
};
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/** \brief the function to be evaluated for the plot. Use \c x as the free variable, e.g. \c "x^2+2" */
|
|
||||||
QString function;
|
|
||||||
JKQTPYParsedFunctionLineGraphFunctionData fdata;
|
|
||||||
|
|
||||||
/** \brief the function to be evaluated to add error indicators to the graph. This function is evaluated to an error for every x. Use \c x as the free variable, e.g. \c "x^2+2". */
|
/** \copydoc JKQTPEvaluatedFunctionGraphBase::buildPlotFunctorSpec() */
|
||||||
QString errorFunction;
|
virtual PlotFunctorSpec buildPlotFunctorSpec() override;
|
||||||
JKQTPYParsedFunctionLineGraphFunctionData efdata;
|
|
||||||
|
|
||||||
// hide functions that should not be used in this class!
|
/** \copydoc JKQTPEvaluatedFunctionWithErrorsGraphBase::buildPlotFunctorSpec() */
|
||||||
using JKQTPXFunctionLineGraph::setPlotFunctionFunctor;
|
virtual std::function<QPointF(double)> buildErrorFunctorSpec() override;
|
||||||
using JKQTPXFunctionLineGraph::setParams;
|
|
||||||
using JKQTPXFunctionLineGraph::setErrorPlotFunction;
|
|
||||||
using JKQTPXFunctionLineGraph::setErrorParams;
|
|
||||||
|
|
||||||
/** \brief fill the data array with data from the function plotFunction */
|
|
||||||
virtual void createPlotData(bool collectParams=true) override;
|
|
||||||
/** \brief implements the actual plot function */
|
|
||||||
static double JKQTPYParsedFunctionLineGraphFunction(double x, const QVector<double>& data, JKQTPYParsedFunctionLineGraphFunctionData* fdata);
|
|
||||||
};
|
};
|
||||||
#endif // jkqtpgraphsparsedfunction_H
|
#endif // jkqtpgraphsparsedfunction_H
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 127 KiB |
Binary file not shown.
Before Width: | Height: | Size: 21 KiB After Width: | Height: | Size: 22 KiB |
Loading…
Reference in New Issue
Block a user