mirror of
https://github.com/jkriege2/JKQtPlotter.git
synced 2024-11-15 18:15:52 +08:00
JKQTPxFunctionLineGraph and JKQTPyFunctionLineGraph now allow for simplified functions without parameters + improved handling of special functions, improved example for parsed functions (with possibility to display sample points
This commit is contained in:
parent
936ae5b410
commit
94acc1b7d5
@ -28,57 +28,16 @@
|
||||
#include "jkqtplotter/jkqtpgraphsimage.h"
|
||||
#include "jkqtplotter/jkqtpbaseelements.h"
|
||||
#include "jkqtplotter/jkqtplotter.h"
|
||||
#include "jkqtpgraphsevaluatedfunction.h"
|
||||
|
||||
|
||||
|
||||
double JKQTPxFunctionLineGraphPolynomial(double x, void* param) {
|
||||
double res=0;
|
||||
QVector<double>* d=static_cast<QVector<double>*>(param);
|
||||
if (d && d->size()>0) {
|
||||
res=d->value(0,0);
|
||||
double xx=x;
|
||||
for (int i=1; i<d->size(); i++) {
|
||||
res=res+d->value(i,0)*xx;
|
||||
xx=xx*x;
|
||||
}
|
||||
}
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
double JKQTPxFunctionLineGraphExponential(double x, void* param) {
|
||||
double res=0;
|
||||
QVector<double>* d=static_cast<QVector<double>*>(param);
|
||||
if (d) {
|
||||
if (d->size()>=3) {
|
||||
res=d->value(0,0)+d->value(1,0)*exp(x/d->value(2,0));
|
||||
} else if (d->size()>=2) {
|
||||
res=d->value(0,0)*exp(x/d->value(1,0));
|
||||
}
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
double JKQTPxFunctionLineGraphPowerLaw(double x, void* param) {
|
||||
double res=0;
|
||||
QVector<double>* d=static_cast<QVector<double>*>(param);
|
||||
if (d) {
|
||||
if (d->size()>=3) {
|
||||
res=d->value(0,0)+d->value(1,0)*pow(x, d->value(2,1));
|
||||
} else if (d->size()>=2) {
|
||||
res=d->value(0,0)*pow(x, d->value(1,1));
|
||||
} else if (d->size()>=1) {
|
||||
res=pow(x, d->value(0,1));
|
||||
}
|
||||
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
|
||||
JKQTPxFunctionLineGraph::JKQTPxFunctionLineGraph(JKQtBasePlotter* parent):
|
||||
JKQTPgraph(parent)
|
||||
{
|
||||
functionType=SpecialFunction::UserFunction;
|
||||
color=QColor("red");
|
||||
fillColor=color.lighter();
|
||||
style=Qt::SolidLine;
|
||||
@ -86,7 +45,6 @@ JKQTPxFunctionLineGraph::JKQTPxFunctionLineGraph(JKQtBasePlotter* parent):
|
||||
fillStyle=Qt::SolidPattern;
|
||||
drawLine=true;
|
||||
fillCurve=false;
|
||||
plotFunction=nullptr;
|
||||
params=nullptr;
|
||||
minSamples=10;
|
||||
maxRefinementDegree=7;
|
||||
@ -98,7 +56,6 @@ JKQTPxFunctionLineGraph::JKQTPxFunctionLineGraph(JKQtBasePlotter* parent):
|
||||
|
||||
drawErrorPolygons=false;
|
||||
drawErrorLines=false;
|
||||
errorPlotFunction=nullptr;
|
||||
errorParams=nullptr;
|
||||
errorColor=color.lighter();
|
||||
errorFillColor=color.lighter();
|
||||
@ -127,6 +84,7 @@ JKQTPxFunctionLineGraph::JKQTPxFunctionLineGraph(JKQtBasePlotter* parent):
|
||||
JKQTPxFunctionLineGraph::JKQTPxFunctionLineGraph(JKQtPlotter* parent):
|
||||
JKQTPgraph(parent)
|
||||
{
|
||||
functionType=SpecialFunction::UserFunction;
|
||||
color=QColor("red");
|
||||
fillColor=color.lighter();
|
||||
style=Qt::SolidLine;
|
||||
@ -134,7 +92,6 @@ JKQTPxFunctionLineGraph::JKQTPxFunctionLineGraph(JKQtPlotter* parent):
|
||||
fillStyle=Qt::SolidPattern;
|
||||
drawLine=true;
|
||||
fillCurve=false;
|
||||
plotFunction=nullptr;
|
||||
params=nullptr;
|
||||
minSamples=10;
|
||||
maxRefinementDegree=7;
|
||||
@ -146,7 +103,6 @@ JKQTPxFunctionLineGraph::JKQTPxFunctionLineGraph(JKQtPlotter* parent):
|
||||
|
||||
drawErrorPolygons=false;
|
||||
drawErrorLines=false;
|
||||
errorPlotFunction=nullptr;
|
||||
errorParams=nullptr;
|
||||
errorColor=color.lighter();
|
||||
errorFillColor=color.lighter();
|
||||
@ -188,19 +144,47 @@ void JKQTPxFunctionLineGraph::clearData() {
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_plotFunction(const jkqtpPlotFunctionType &__value)
|
||||
{
|
||||
this->plotFunction = __value;
|
||||
simplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||
plotFunction = __value;
|
||||
functionType=SpecialFunction::UserFunction;
|
||||
|
||||
clearData();
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_plotFunction(const jkqtpSimplePlotFunctionType &__value)
|
||||
{
|
||||
plotFunction=jkqtpPlotFunctionType();
|
||||
simplePlotFunction=__value;
|
||||
functionType=SpecialFunction::UserFunction;
|
||||
|
||||
clearData();
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_plotFunction(jkqtpPlotFunctionType &&__value)
|
||||
{
|
||||
this->plotFunction = std::move(__value);
|
||||
simplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||
plotFunction = std::move(__value);
|
||||
functionType=SpecialFunction::UserFunction;
|
||||
clearData();
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_plotFunction(jkqtpSimplePlotFunctionType &&__value)
|
||||
{
|
||||
plotFunction=jkqtpPlotFunctionType();
|
||||
simplePlotFunction=std::move(__value);
|
||||
functionType=SpecialFunction::UserFunction;
|
||||
|
||||
clearData();
|
||||
}
|
||||
|
||||
jkqtpPlotFunctionType JKQTPxFunctionLineGraph::get_plotFunction() const
|
||||
{
|
||||
return this->plotFunction;
|
||||
return plotFunction;
|
||||
}
|
||||
|
||||
jkqtpSimplePlotFunctionType JKQTPxFunctionLineGraph::get_simplePlotFunction() const
|
||||
{
|
||||
return simplePlotFunction;
|
||||
}
|
||||
|
||||
|
||||
@ -215,7 +199,7 @@ void JKQTPxFunctionLineGraph::drawKeyMarker(JKQTPEnhancedPainter& painter, QRect
|
||||
QBrush b=painter.brush();
|
||||
b.setColor(fillColor);
|
||||
b.setStyle(fillStyle);
|
||||
int y=rect.top()+rect.height()/2.0;
|
||||
const double y=rect.top()+rect.height()/2.0;
|
||||
painter.setPen(np);
|
||||
if (drawLine) painter.setPen(p);
|
||||
painter.setBrush(b);
|
||||
@ -246,7 +230,11 @@ void JKQTPxFunctionLineGraph::createPlotData(bool collectParams) {
|
||||
if (collectParams) collectParameters();
|
||||
|
||||
if (parent==nullptr) return;
|
||||
if (plotFunction==nullptr) return;
|
||||
if (!plotFunction && !simplePlotFunction) return;
|
||||
|
||||
jkqtpSimplePlotFunctionType func;
|
||||
if (plotFunction) func=std::bind(plotFunction, std::placeholders::_1, params);
|
||||
else if (simplePlotFunction) func=simplePlotFunction;
|
||||
|
||||
double xmin=parent->getXMin();
|
||||
double xmax=parent->getXMax();
|
||||
@ -258,14 +246,14 @@ void JKQTPxFunctionLineGraph::createPlotData(bool collectParams) {
|
||||
// initially sample function
|
||||
doublePair* d=new doublePair;
|
||||
d->x=xmin;
|
||||
d->f=plotFunction(xmin, params);
|
||||
d->f=func(xmin);
|
||||
d->next=nullptr;
|
||||
data=d;
|
||||
/*if (parent && parent->getXAxis()->isLogAxis()) {
|
||||
for (double x=log(xmin)+logdelta0; x<log(xmax); x=x+logdelta0) {
|
||||
d->next = new doublePair;
|
||||
d->next->x=exp(x+((double)rand()/(double)RAND_MAX-0.5)*delta0/2.0);
|
||||
d->next->f=plotFunction(d->next->x, params);
|
||||
d->next->f=func(d->next->x,);
|
||||
d->next->next=nullptr;
|
||||
doublePair* dd=d;
|
||||
d=d->next;
|
||||
@ -273,20 +261,24 @@ void JKQTPxFunctionLineGraph::createPlotData(bool collectParams) {
|
||||
}
|
||||
} else {*/
|
||||
QVector<double>* dv=static_cast<QVector<double>*>(params);
|
||||
for (double x=pxmin+delta0; x<pxmax; x=x+delta0) {
|
||||
d->next = new doublePair;
|
||||
d->next->x=parent->p2x(x+((double)rand()/(double)RAND_MAX-0.5)*delta0/2.0);
|
||||
d->next->f=plotFunction(d->next->x, params);
|
||||
d->next->next=nullptr;
|
||||
doublePair* dd=d;
|
||||
d=d->next;
|
||||
refine(dd, d);
|
||||
if (functionType==Polynomial && dv && dv->size()<=2) {
|
||||
// we only need the first and last datapoint
|
||||
} else {
|
||||
for (double x=pxmin+delta0; x<pxmax; x=x+delta0) {
|
||||
d->next = new doublePair;
|
||||
d->next->x=parent->p2x(x+((double)rand()/(double)RAND_MAX-0.5)*delta0/2.0);
|
||||
d->next->f=func(d->next->x);
|
||||
d->next->next=nullptr;
|
||||
doublePair* dd=d;
|
||||
d=d->next;
|
||||
refine(dd, d);
|
||||
}
|
||||
}
|
||||
|
||||
//}
|
||||
d->next = new doublePair;
|
||||
d->next->x=xmax;
|
||||
d->next->f=plotFunction(xmax, params);
|
||||
d->next->f=func(xmax);
|
||||
d->next->next=nullptr;
|
||||
refine(d, d->next);
|
||||
|
||||
@ -355,7 +347,9 @@ void JKQTPxFunctionLineGraph::refine(doublePair* a, doublePair* b, unsigned int
|
||||
xmid=xmid+((double)rand()/(double)RAND_MAX-0.5)*delta/5.0; // shake by 10%
|
||||
//}
|
||||
double realxmid=parent->p2x(xmid);
|
||||
double realfmid=plotFunction(realxmid, params);
|
||||
double realfmid;
|
||||
if (plotFunction) realfmid=plotFunction(realxmid, params);
|
||||
else if (simplePlotFunction) realfmid=simplePlotFunction(realxmid);
|
||||
double fmid=yAxis->x2p(realfmid);
|
||||
double a1=(fmid - af)/(xmid - ax);
|
||||
double a2=(bf - fmid)/(bx - xmid);
|
||||
@ -721,7 +715,11 @@ void JKQTPyFunctionLineGraph::createPlotData(bool /*collectParams*/) {
|
||||
clearData();
|
||||
|
||||
if (parent==nullptr) return;
|
||||
if (plotFunction==nullptr) return;
|
||||
if (!plotFunction && !simplePlotFunction) return;
|
||||
|
||||
jkqtpSimplePlotFunctionType func;
|
||||
if (plotFunction) func=std::bind(plotFunction, std::placeholders::_1, params);
|
||||
else if (simplePlotFunction) func=simplePlotFunction;
|
||||
|
||||
double ymin=parent->getYMin();
|
||||
double ymax=parent->getYMax();
|
||||
@ -730,13 +728,13 @@ void JKQTPyFunctionLineGraph::createPlotData(bool /*collectParams*/) {
|
||||
// initially sample function
|
||||
doublePair* d=new doublePair;
|
||||
d->x=ymin;
|
||||
d->f=plotFunction(ymin, params);
|
||||
d->f=func(ymin);
|
||||
d->next=nullptr;
|
||||
data=d;
|
||||
for (double y=ymin+delta0; y<ymax; y=y+delta0) {
|
||||
d->next = new doublePair;
|
||||
d->next->x=y+((double)rand()/(double)RAND_MAX-0.5)*delta0/2.0;
|
||||
d->next->f=plotFunction(d->next->x, params);
|
||||
d->next->f=func(d->next->x);
|
||||
d->next->next=nullptr;
|
||||
doublePair* dd=d;
|
||||
d=d->next;
|
||||
@ -744,7 +742,7 @@ void JKQTPyFunctionLineGraph::createPlotData(bool /*collectParams*/) {
|
||||
}
|
||||
d->next = new doublePair;
|
||||
d->next->x=ymax;
|
||||
d->next->f=plotFunction(ymax, params);
|
||||
d->next->f=func(ymax);
|
||||
d->next->next=nullptr;
|
||||
refine(d, d->next);
|
||||
|
||||
@ -802,18 +800,93 @@ void JKQTPxFunctionLineGraph::set_copiedParams(const double *params, int N)
|
||||
set_params(v);
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_paramsV(double p1) {
|
||||
QVector<double> p;
|
||||
p<<p1;
|
||||
set_params(p);
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_paramsV(double p1, double p2) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2;
|
||||
set_params(p);
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_paramsV(double p1, double p2, double p3) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2<<p3;
|
||||
set_params(p);
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_paramsV(double p1, double p2, double p3, double p4) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2<<p3<<p4;
|
||||
set_params(p);
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_paramsV(double p1, double p2, double p3, double p4, double p5) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2<<p3<<p4<<p5;
|
||||
set_params(p);
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_errorParams(const QVector<double> &errorParams)
|
||||
{
|
||||
ierrorparams=errorParams;
|
||||
set_errorParams(&ierrorparams);
|
||||
}
|
||||
|
||||
|
||||
void JKQTPxFunctionLineGraph::setSpecialFunction(JKQTPxFunctionLineGraph::SpecialFunction function)
|
||||
{
|
||||
if (function==JKQTPxFunctionLineGraph::Polynomial) set_plotFunction(JKQTPxFunctionLineGraphPolynomial);
|
||||
else if (function==JKQTPxFunctionLineGraph::Exponential) set_plotFunction(JKQTPxFunctionLineGraphExponential);
|
||||
else if (function==JKQTPxFunctionLineGraph::PowerLaw) set_plotFunction(JKQTPxFunctionLineGraphPowerLaw);
|
||||
if (function==JKQTPxFunctionLineGraph::Polynomial) {
|
||||
set_plotFunction([](double x, void* param) {
|
||||
double res=0;
|
||||
QVector<double>* d=static_cast<QVector<double>*>(param);
|
||||
if (d && d->size()>0) {
|
||||
res=d->value(0,0);
|
||||
double xx=x;
|
||||
for (int i=1; i<d->size(); i++) {
|
||||
res=res+d->value(i,0)*xx;
|
||||
xx=xx*x;
|
||||
}
|
||||
}
|
||||
|
||||
return res;
|
||||
});
|
||||
}
|
||||
else if (function==JKQTPxFunctionLineGraph::Exponential) set_plotFunction([](double x, void* param) {
|
||||
double res=0;
|
||||
QVector<double>* d=static_cast<QVector<double>*>(param);
|
||||
if (d) {
|
||||
if (d->size()>=3) {
|
||||
res=d->value(0,0)+d->value(1,0)*exp(x/d->value(2,0));
|
||||
} else if (d->size()>=2) {
|
||||
res=d->value(0,0)*exp(x/d->value(1,0));
|
||||
}
|
||||
}
|
||||
return res;
|
||||
});
|
||||
else if (function==JKQTPxFunctionLineGraph::PowerLaw) set_plotFunction([](double x, void* param) {
|
||||
double res=0;
|
||||
QVector<double>* d=static_cast<QVector<double>*>(param);
|
||||
if (d) {
|
||||
if (d->size()>=3) {
|
||||
res=d->value(0,0)+d->value(1,0)*pow(x, d->value(2,1));
|
||||
} else if (d->size()>=2) {
|
||||
res=d->value(0,0)*pow(x, d->value(1,1));
|
||||
} else if (d->size()>=1) {
|
||||
res=pow(x, d->value(0,1));
|
||||
}
|
||||
|
||||
}
|
||||
return res;
|
||||
});
|
||||
else throw std::runtime_error("unknown special function type");
|
||||
}
|
||||
|
||||
JKQTPxFunctionLineGraph::SpecialFunction JKQTPxFunctionLineGraph::getFunctionType() const
|
||||
{
|
||||
return functionType;
|
||||
}
|
||||
|
||||
QVector<double> JKQTPxFunctionLineGraph::get_internalParams() const {
|
||||
@ -825,12 +898,15 @@ QVector<double> JKQTPxFunctionLineGraph::get_internalErrorParams() const {
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_errorPlotFunction(const jkqtpPlotFunctionType &__value)
|
||||
{
|
||||
errorSimplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||
errorPlotFunction=__value;
|
||||
clearData();
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_errorPlotFunction(jkqtpPlotFunctionType &&__value)
|
||||
{
|
||||
this->errorPlotFunction = std::move(__value);
|
||||
errorSimplePlotFunction=jkqtpSimplePlotFunctionType();
|
||||
errorPlotFunction = std::move(__value);
|
||||
clearData();
|
||||
}
|
||||
jkqtpPlotFunctionType JKQTPxFunctionLineGraph::get_errorPlotFunction() const
|
||||
@ -838,6 +914,24 @@ jkqtpPlotFunctionType JKQTPxFunctionLineGraph::get_errorPlotFunction() const
|
||||
return errorPlotFunction;
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_errorPlotFunction(const jkqtpSimplePlotFunctionType &__value)
|
||||
{
|
||||
errorPlotFunction=jkqtpPlotFunctionType();
|
||||
errorSimplePlotFunction=__value;
|
||||
clearData();
|
||||
}
|
||||
|
||||
void JKQTPxFunctionLineGraph::set_errorPlotFunction(jkqtpSimplePlotFunctionType &&__value)
|
||||
{
|
||||
errorPlotFunction=jkqtpPlotFunctionType();
|
||||
errorSimplePlotFunction = std::move(__value);
|
||||
clearData();
|
||||
}
|
||||
jkqtpSimplePlotFunctionType JKQTPxFunctionLineGraph::get_errorSimplePlotFunction() const
|
||||
{
|
||||
return errorSimplePlotFunction;
|
||||
}
|
||||
|
||||
|
||||
bool JKQTPxFunctionLineGraph::usesColumn(int c)
|
||||
{
|
||||
|
@ -53,6 +53,14 @@
|
||||
*/
|
||||
typedef std::function<double(double, void*)> jkqtpPlotFunctionType;
|
||||
|
||||
/*! \brief simplified type of functions (without parameters) that may be plottet
|
||||
\ingroup jkqtplotter_plots
|
||||
|
||||
This is the type of functions \f$ y=f(x) \f$ that may be plottet by JKQTPxFunctionLineGraph
|
||||
and JKQTPyFunctionLineGraph.
|
||||
*/
|
||||
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_plots
|
||||
@ -76,7 +84,9 @@ class LIB_EXPORT JKQTPxFunctionLineGraph: public JKQTPgraph {
|
||||
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$ */
|
||||
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 */
|
||||
@ -125,41 +135,40 @@ class LIB_EXPORT JKQTPxFunctionLineGraph: public JKQTPgraph {
|
||||
* \details Description of the parameter varname is: <CENTER>\copybrief plotFunction.</CENTER>
|
||||
* \see plotFunction for more information */
|
||||
virtual void set_plotFunction (const jkqtpPlotFunctionType & __value);
|
||||
/** \brief returns the property varname. \see varname for more information */ \
|
||||
/** \brief sets the property plotFunction to the specified \a __value.
|
||||
*
|
||||
* \details Description of the parameter plotFunction is: <CENTER>\copybrief plotFunction.</CENTER>
|
||||
* \see plotFunction for more information */
|
||||
virtual void set_plotFunction (jkqtpSimplePlotFunctionType && __value);
|
||||
/** \brief sets the property plotFunction to the specified \a __value.
|
||||
*
|
||||
* \details Description of the parameter plotFunction is: <CENTER>\copybrief plotFunction.</CENTER>
|
||||
* \see plotFunction for more information */
|
||||
virtual void set_plotFunction (const jkqtpSimplePlotFunctionType & __value);
|
||||
/** \brief returns the property plotFunction. \see plotFunction for more information */ \
|
||||
virtual jkqtpPlotFunctionType get_plotFunction () const;
|
||||
/** \brief returns the property simplePlotFunction. \see simplePlotFunction for more information */ \
|
||||
virtual jkqtpSimplePlotFunctionType get_simplePlotFunction () const;
|
||||
|
||||
JKQTPGET_SET_MACRO_I(void*, params, clearData())
|
||||
/** \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!!!) */
|
||||
void set_params(const QVector<double>& params);
|
||||
/** \brief sets the params from a copy of the given array of length \a N */
|
||||
void set_copiedParams(const double* params, int N);
|
||||
inline void set_paramsV(double p1) {
|
||||
QVector<double> p;
|
||||
p<<p1;
|
||||
set_params(p);
|
||||
}
|
||||
inline void set_paramsV(double p1, double p2) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2;
|
||||
set_params(p);
|
||||
}
|
||||
inline void set_paramsV(double p1, double p2, double p3) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2<<p3;
|
||||
set_params(p);
|
||||
}
|
||||
inline void set_paramsV(double p1, double p2, double p3, double p4) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2<<p3<<p4;
|
||||
set_params(p);
|
||||
}
|
||||
inline void set_paramsV(double p1, double p2, double p3, double p4, double p5) {
|
||||
QVector<double> p;
|
||||
p<<p1<<p2<<p3<<p4<<p5;
|
||||
set_params(p);
|
||||
}
|
||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1} */
|
||||
void set_paramsV(double p1);
|
||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2} */
|
||||
void set_paramsV(double p1, double p2);
|
||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3} */
|
||||
void set_paramsV(double p1, double p2, double p3);
|
||||
/** \brief set an internal parameter vector as function parameters, initialized with {p1,p2,p3,p4} */
|
||||
void set_paramsV(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 set_paramsV(double p1, double p2, double p3, double p4, double p5);
|
||||
|
||||
/** \brief returns the currently set internal parameter vector */
|
||||
QVector<double> get_internalParams() const;
|
||||
/** \brief returns the currently set internal parameter vector */
|
||||
QVector<double> get_internalErrorParams() const;
|
||||
JKQTPGET_SET_MACRO(unsigned int, minSamples)
|
||||
JKQTPGET_SET_MACRO(unsigned int, maxRefinementDegree)
|
||||
@ -181,6 +190,18 @@ class LIB_EXPORT JKQTPxFunctionLineGraph: public JKQTPgraph {
|
||||
virtual void set_errorPlotFunction (const jkqtpPlotFunctionType & __value);
|
||||
/** \brief returns the property varname. \see varname for more information */ \
|
||||
virtual jkqtpPlotFunctionType get_errorPlotFunction () const;
|
||||
/** \brief sets the property errorPlotFunction to the specified \a __value.
|
||||
*
|
||||
* \details Description of the parameter varname is: <CENTER>\copybrief errorPlotFunction.</CENTER>
|
||||
* \see errorPlotFunction for more information */
|
||||
virtual void set_errorPlotFunction (jkqtpSimplePlotFunctionType && __value);
|
||||
/** \brief sets the property errorPlotFunction to the specified \a __value.
|
||||
*
|
||||
* \details Description of the parameter varname is: <CENTER>\copybrief errorPlotFunction.</CENTER>
|
||||
* \see errorPlotFunction for more information */
|
||||
virtual void set_errorPlotFunction (const jkqtpSimplePlotFunctionType & __value);
|
||||
/** \brief returns the property varname. \see varname for more information */ \
|
||||
virtual jkqtpSimplePlotFunctionType get_errorSimplePlotFunction () const;
|
||||
JKQTPGET_SET_MACRO(void*, errorParams)
|
||||
/** \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 set_errorParams(const QVector<double>& errorParams);
|
||||
@ -202,6 +223,8 @@ class LIB_EXPORT JKQTPxFunctionLineGraph: public JKQTPgraph {
|
||||
|
||||
/** \brief sets function to the given special function */
|
||||
void setSpecialFunction(SpecialFunction function);
|
||||
/** \brief returns, which special function is set (or if any is set) */
|
||||
SpecialFunction getFunctionType() const;
|
||||
protected:
|
||||
/** \brief which plot style to use from the parent plotter (via JKQtPlotterBase::getPlotStyle() and JKQtPlotterBase::getNextStyle() ) */
|
||||
int parentPlotStyle;
|
||||
@ -218,7 +241,7 @@ class LIB_EXPORT JKQTPxFunctionLineGraph: public JKQTPgraph {
|
||||
virtual void createPlotData( bool collectParams=true);
|
||||
|
||||
virtual void collectParameters();
|
||||
|
||||
/** \brief refine datapoints on the function graph between two evaluations \a a and \a b */
|
||||
void refine(doublePair* a, doublePair* b, unsigned int degree=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 */
|
||||
@ -242,6 +265,10 @@ class LIB_EXPORT JKQTPxFunctionLineGraph: public JKQTPgraph {
|
||||
bool fillCurve;
|
||||
/** \brief the function to be plotted */
|
||||
jkqtpPlotFunctionType plotFunction;
|
||||
/** \brief a simple function to be plotted, simplified form without parameters */
|
||||
jkqtpSimplePlotFunctionType simplePlotFunction;
|
||||
/** \brief indicates whether a special function is set (and if so, which one), or a user-supplied function */
|
||||
SpecialFunction functionType;
|
||||
/** \brief pointer to the parameters supplied to the plotting funtion */
|
||||
void* params;
|
||||
/** \brief the minimum number of points to evaluate the function at */
|
||||
@ -267,6 +294,8 @@ class LIB_EXPORT JKQTPxFunctionLineGraph: public JKQTPgraph {
|
||||
bool drawErrorLines;
|
||||
/** \brief this function calculates the error at a given position */
|
||||
jkqtpPlotFunctionType errorPlotFunction;
|
||||
/** \brief this function calculates the error at a given position, simplified form without parameters */
|
||||
jkqtpSimplePlotFunctionType errorSimplePlotFunction;
|
||||
/** \brief parameters for errorFunction */
|
||||
void* errorParams;
|
||||
/** \brief color of the error graph */
|
||||
|
Binary file not shown.
After Width: | Height: | Size: 31 KiB |
@ -8,7 +8,7 @@ This project (see `./test/simpletest_functionplot/`) demonstrates how to plot ma
|
||||
The first example shows how to plot a C++ inline function:
|
||||
```c++
|
||||
JKQTPxFunctionLineGraph* func1=new JKQTPxFunctionLineGraph(plot);
|
||||
func1->set_plotFunction([](double x, void* /*params*/) { return 0.2*x*x-0.015*x*x*x; });
|
||||
func1->set_plotFunction([](double x) { return 0.2*x*x-0.015*x*x*x; });
|
||||
func1->set_title("C++-inline function $0.2x^2-0.015x^3$");
|
||||
plot->addGraph(func1);
|
||||
```
|
||||
@ -47,7 +47,7 @@ You can also use C++ functors (or function objects):
|
||||
struct SincSqr {
|
||||
public:
|
||||
inline SincSqr(double amplitude): a(amplitude) {}
|
||||
inline double operator()(double x, void* /*params*/) {
|
||||
inline double operator()(double x) {
|
||||
return a*sin(x)*sin(x)/x/x;
|
||||
}
|
||||
private:
|
||||
@ -64,7 +64,7 @@ You can also use C++ functors (or function objects):
|
||||
|
||||
... or simple static C functions:
|
||||
```c++
|
||||
double sinc(double x, void* /*params*/) {
|
||||
double sinc(double x) {
|
||||
return 10.0*sin(x)/x;
|
||||
}
|
||||
|
||||
|
@ -4,14 +4,14 @@
|
||||
#include "jkqtplotter/jkqtplotter.h"
|
||||
#include "jkqtplotter/jkqtpgraphsevaluatedfunction.h"
|
||||
|
||||
double sinc(double x, void* /*params*/) {
|
||||
double sinc(double x) {
|
||||
return 10.0*sin(x)/x;
|
||||
}
|
||||
|
||||
struct SincSqr {
|
||||
public:
|
||||
inline SincSqr(double amplitude): a(amplitude) {}
|
||||
inline double operator()(double x, void* /*params*/) {
|
||||
inline double operator()(double x) {
|
||||
return a*sin(x)*sin(x)/x/x;
|
||||
}
|
||||
private:
|
||||
@ -33,7 +33,7 @@ int main(int argc, char* argv[])
|
||||
// 2. now we add a JKQTPxFunctionLineGraph object, which will draw a simple function
|
||||
// the function is defined as C++ inline function
|
||||
JKQTPxFunctionLineGraph* func1=new JKQTPxFunctionLineGraph(plot);
|
||||
func1->set_plotFunction([](double x, void* /*params*/) { return 0.2*x*x-0.015*x*x*x; });
|
||||
func1->set_plotFunction([](double x) { return 0.2*x*x-0.015*x*x*x; });
|
||||
func1->set_title("C++-inline function $0.2x^2-0.015x^3$");
|
||||
plot->addGraph(func1);
|
||||
|
||||
|
@ -45,9 +45,13 @@ This code snippet results in a plot like this:
|
||||
|
||||
![jkqtplotter_simpletest_parsedfunctionplot](https://raw.githubusercontent.com/jkriege2/JKQtPlotter/master/screenshots/jkqtplotter_simpletest_parsedfunctionplot.png)
|
||||
|
||||
the adaptive capabilities of the rendering algorithm can be seen, when plotting e.g. `2/x`, which is drawn smoothely, even around the undefined value at `x=0`:
|
||||
The adaptive capabilities of the rendering algorithm can be seen, when plotting e.g. `2/x`, which is drawn smoothely, even around the undefined value at `x=0`:
|
||||
|
||||
![jkqtplotter_simpletest_parsedfunctionplot_2overx.png](https://raw.githubusercontent.com/jkriege2/JKQtPlotter/master/screenshots/jkqtplotter_simpletest_parsedfunctionplot_2overx.png)
|
||||
|
||||
With an additional checkbox in this example, you can switch drawing the actual sample points of the drawing algorithm on and off, by calling `parsedFunc->set_displaySamplePoints(...)`. This can be used to debug the drawing algorithm and explore its parameters (which you can set with `set_minSamples()`, `set_maxRefinementDegree()`, `set_slopeTolerance()`, `set_minPixelPerSample()`). Here is an example of a 2/x function with shown sample points:
|
||||
|
||||
![jkqtplotter_simpletest_parsedfunctionplot_2overx_samplepoints.png](https://raw.githubusercontent.com/jkriege2/JKQtPlotter/master/screenshots/jkqtplotter_simpletest_parsedfunctionplot_2overx_samplepoints.png)
|
||||
|
||||
|
||||
[Back to JKQTPlotter main page](https://github.com/jkriege2/JKQtPlotter/)
|
@ -1,5 +1,6 @@
|
||||
#include <QApplication>
|
||||
#include <QLineEdit>
|
||||
#include <QCheckBox>
|
||||
#include "jkqtplotter/jkqtplotter.h"
|
||||
#include "jkqtplotter/jkqtpgraphsparsedfunction.h"
|
||||
|
||||
@ -14,10 +15,12 @@ int main(int argc, char* argv[])
|
||||
QWidget mainWin;
|
||||
QLineEdit* edit=new QLineEdit(&mainWin);
|
||||
edit->setToolTip("enter a function in dependence of the variable <tt>x</tt> and press ENTER to update the graph");
|
||||
QCheckBox* check=new QCheckBox("display sample points");
|
||||
JKQtPlotter* plot=new JKQtPlotter(&mainWin);
|
||||
QVBoxLayout* layout=new QVBoxLayout;
|
||||
mainWin.setLayout(layout);
|
||||
layout->addWidget(edit);
|
||||
layout->addWidget(check);
|
||||
layout->addWidget(plot);
|
||||
|
||||
// 2. now we add a JKQTPxParsedFunctionLineGraph object, which will draw the function from
|
||||
@ -30,10 +33,12 @@ int main(int argc, char* argv[])
|
||||
[=]() {
|
||||
parsedFunc->set_title("user function: \\verb{"+edit->text()+"}");
|
||||
parsedFunc->set_function(edit->text());
|
||||
parsedFunc->set_displaySamplePoints(check->isChecked());
|
||||
plot->update_plot();
|
||||
};
|
||||
QObject::connect(edit, &QLineEdit::returnPressed, updateGraphFunctor);
|
||||
QObject::connect(edit, &QLineEdit::editingFinished, updateGraphFunctor);
|
||||
QObject::connect(check, &QCheckBox::toggled, updateGraphFunctor);
|
||||
edit->setText("sin(x*8)*exp(-x/4)");
|
||||
updateGraphFunctor();
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user