mirror of
https://github.com/jkriege2/JKQtPlotter.git
synced 2025-01-24 22:49:13 +08:00
1838 lines
66 KiB
C++
1838 lines
66 KiB
C++
/*
|
|
Copyright (c) 2008-2022 Jan W. Krieger (<jan@jkrieger.de>)
|
|
|
|
|
|
|
|
This software is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License (LGPL) as published by
|
|
the Free Software Foundation, either version 2.1 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License (LGPL) for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License (LGPL)
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
|
|
#include "jkqtfastplotter.h"
|
|
#include "jkqtcommon/jkqtpcodestructuring.h"
|
|
#include <QLocale>
|
|
#include <QPainter>
|
|
#include <QPaintEvent>
|
|
#include <QResizeEvent>
|
|
#include <QFontMetricsF>
|
|
#include <QAction>
|
|
#include <QApplication>
|
|
#include <QClipboard>
|
|
#include <QPainterPath>
|
|
|
|
/**
|
|
* \brief saves the given property (for which also a default_property exists) into the given settings object
|
|
* \ingroup jkqtfastplotter
|
|
* \internal
|
|
*/
|
|
#define JKQTFPPROPERTYsave(settings, group, var, varname) \
|
|
if ((var)!=default_##var) (settings).setValue((group)+(varname), var);
|
|
/**
|
|
* \brief loads the given property from the given settings object
|
|
* \ingroup jkqtfastplotter
|
|
* \internal
|
|
*/
|
|
#define JKQTFPPROPERTYload(settings, group, var, varname, varconvert) \
|
|
var=(settings).value((group)+(varname), var).varconvert;
|
|
|
|
const double JKQTFastPlotter::ABS_MIN_LINEWIDTH=0.05;
|
|
const size_t JKQTFastPlotter::LUTSIZE=256;
|
|
|
|
JKQTFPPlot::JKQTFPPlot(JKQTFastPlotter* parent):
|
|
QObject(parent)
|
|
|
|
{
|
|
this->parent=parent;
|
|
this->visible=true;
|
|
}
|
|
|
|
JKQTFPPlot::~JKQTFPPlot()
|
|
= default;;
|
|
|
|
void JKQTFPPlot::replot() {
|
|
if (parent) parent->updateData();
|
|
};
|
|
|
|
void JKQTFPPlot::paint(QPainter& painter) {
|
|
if (visible) drawGraph(painter);
|
|
}
|
|
|
|
JKQTFastPlotter::JKQTFastPlotter(QWidget *parent) :
|
|
#if (QT_VERSION>=QT_VERSION_CHECK(6, 0, 0))
|
|
JKQTFASTPLOTTER_BASE(parent)
|
|
#else
|
|
JKQTFASTPLOTTER_BASE(QGLFormat(QGL::StencilBuffer | QGL::AlphaChannel | QGL::Rgba), parent)//, mutexRepaint(QMutex::Recursive), mutexRepaintData(QMutex::Recursive), mutexRepaintSystem(QMutex::Recursive)
|
|
#endif
|
|
{
|
|
mouseDragStart=QPoint(0,0);
|
|
mouseDragEnd=QPoint(0,0);
|
|
dragging=true;
|
|
|
|
dragShapePen=QPen(QColor("red"));
|
|
dragShapePen.setStyle(Qt::DashLine);
|
|
dragLine=false;
|
|
doFullRepaint=false;
|
|
plotBorderLeft=default_plotBorderLeft=50;
|
|
plotBorderRight=default_plotBorderRight=30;
|
|
plotBorderTop=default_plotBorderTop=5;
|
|
plotBorderBottom=default_plotBorderBottom=30;
|
|
|
|
synchronizeX=nullptr;
|
|
synchronizeY=nullptr;
|
|
|
|
backgroundColor=default_backgroundColor=palette().color(QPalette::Window);
|
|
plotBackgroundColor=default_plotBackgroundColor=QColor("white");
|
|
drawGrid=default_drawGrid=true;
|
|
gridColor=default_gridColor=QColor("lightgrey");
|
|
gridStyle=default_gridStyle=Qt::DashLine;
|
|
gridWidth=default_gridWidth=1;
|
|
labelFontSize=default_labelFontSize=font().pointSizeF();
|
|
labelFontName=default_labelFontName=font().family();
|
|
tickFontSize=default_tickFontSize=font().pointSizeF();
|
|
tickFontName=default_tickFontName=font().family();
|
|
tickLength=default_tickLength=3;
|
|
drawSystemBox=default_drawSystemBox=true;
|
|
drawZeroAxes=default_drawZeroAxes=true;
|
|
systemColor=default_systemColor=QColor("black");
|
|
systemWidth=default_systemWidth=2;
|
|
xZeroTick=0;
|
|
yZeroTick=0;
|
|
xTickDistance=1;
|
|
yTickDistance=1;
|
|
xAxisLabel="x";
|
|
yAxisLabel="y";
|
|
xAxisLabelVisible=true;
|
|
yAxisLabelVisible=true;
|
|
xAxisLog=false;
|
|
yAxisLog=false;
|
|
aspectRatio=1;
|
|
maintainAspectRatio=false;
|
|
doDrawing=false;
|
|
xMin=0;
|
|
xMax=10;
|
|
yMin=0;
|
|
yMax=10;
|
|
resize(400,200);
|
|
setXRange(0, 10);
|
|
setYRange(0,10);
|
|
doDrawing=true;
|
|
|
|
actCopyImage=new QAction(tr("copy image to clipboard"), this);
|
|
connect(actCopyImage, SIGNAL(triggered()), this, SLOT(copyImage()));
|
|
addAction(actCopyImage);
|
|
setContextMenuPolicy(Qt::ActionsContextMenu);
|
|
redrawPlot();
|
|
setMouseTracking(true);
|
|
}
|
|
|
|
void JKQTFastPlotter::clearPlots(bool remove) {
|
|
if (remove) {
|
|
for (int i=0; i<plots.size(); i++) {
|
|
delete plots[i];
|
|
}
|
|
}
|
|
plots.clear();
|
|
redrawPlot();
|
|
}
|
|
|
|
|
|
void JKQTFastPlotter::addPlot(JKQTFPPlot* g) {
|
|
g->setParent(this);
|
|
plots.append(g);
|
|
redrawPlot();
|
|
}
|
|
|
|
|
|
void JKQTFastPlotter::deletePlot(int i, bool remove) {
|
|
if (i>=0 && i<plots.size()) {
|
|
if (remove) delete plots[i];
|
|
plots.remove(i);
|
|
redrawPlot();
|
|
}
|
|
}
|
|
|
|
|
|
void JKQTFastPlotter::deletePlot(JKQTFPPlot* g, bool remove) {
|
|
int i=plots.indexOf(g);
|
|
if (i>=0 && i<plots.size()) {
|
|
if (remove) delete plots[i];
|
|
plots.remove(i);
|
|
redrawPlot();
|
|
}
|
|
}
|
|
|
|
int JKQTFastPlotter::getPlotNum(JKQTFPPlot* g) {
|
|
return plots.indexOf(g);
|
|
}
|
|
|
|
void JKQTFastPlotter::paintEvent(QPaintEvent *event) {
|
|
#ifdef DEBUG
|
|
qDebug()<<"JKQTFastPlotter::paintEvent";
|
|
#endif
|
|
// draw the saved image of the plot
|
|
mutexRepaint.lock();
|
|
QPainter* painter = new QPainter(this);
|
|
//painter->setRenderHints(QPainter::HighQualityAntialiasing|QPainter::Antialiasing|QPainter::SmoothPixmapTransform);
|
|
if (painter->isActive()) {
|
|
//qDebug()<<"JKQTFastPlotter::paintEvent: image: "<<image.width()<<" x "<<image.height()<<": "<<image.isNull();
|
|
//qDebug()<<"image.format: "<<image.format()<<" painter.componsitionMode: "<<painter->compositionMode();
|
|
//image.save(objectName()+"_jkqtfastplotter.png");
|
|
painter->drawImage(QPoint(0, 0), image);
|
|
if (dragging) {
|
|
if (dragLine) {
|
|
painter->setPen(dragShapePen);
|
|
painter->drawLine(mouseDragStart, mouseDragEnd);
|
|
}
|
|
}
|
|
}
|
|
delete painter;
|
|
event->accept();
|
|
mutexRepaint.unlock();
|
|
#ifdef DEBUG
|
|
qDebug()<<"JKQTFastPlotter::paintEvent ... done";
|
|
#endif
|
|
}
|
|
|
|
void JKQTFastPlotter::copyImage()
|
|
{
|
|
mutexRepaint.lock();
|
|
QClipboard *clipboard = QApplication::clipboard();
|
|
clipboard->setImage(image);
|
|
mutexRepaint.unlock();
|
|
}
|
|
|
|
|
|
void JKQTFastPlotter::mouseDoubleClickEvent ( QMouseEvent * event ) {
|
|
if (event->button()==Qt::LeftButton) {
|
|
double x=p2x(event->pos().x());
|
|
double y=p2y(event->pos().y());
|
|
emit doubleClicked(x, y);
|
|
emit doubleClicked(x, y, event->modifiers());
|
|
event->accept();
|
|
}
|
|
}
|
|
|
|
void JKQTFastPlotter::mouseMoveEvent ( QMouseEvent * event ) {
|
|
double x=p2x(event->pos().x());
|
|
double y=p2y(event->pos().y());
|
|
emit mouseMoved(x, y);
|
|
//qDebug()<<"JKQTFastPlotter::mouseMoveEvent "<<x<<y;
|
|
if (event->buttons()&Qt::LeftButton) {
|
|
dragging=true;
|
|
double xd=p2x(mouseDragStart.x());
|
|
double yd=p2y(mouseDragStart.y());
|
|
mouseDragEnd=event->pos();
|
|
//qDebug()<<"JKQTFastPlotter::mouseMoveEvent: dragged: "<<xd<<yd<<x<<y;
|
|
emit mouseDragged(xd, yd, x, y, event->modifiers());
|
|
if (dragLine) update();
|
|
}
|
|
event->accept();
|
|
}
|
|
|
|
void JKQTFastPlotter::mousePressEvent ( QMouseEvent * event ) {
|
|
if (event->button()==Qt::LeftButton) {
|
|
double x=p2x(event->pos().x());
|
|
double y=p2y(event->pos().y());
|
|
emit clicked(x, y);
|
|
emit clicked(x, y, event->modifiers());
|
|
mouseDragStart=event->pos();
|
|
event->accept();
|
|
}
|
|
}
|
|
|
|
void JKQTFastPlotter::mouseReleaseEvent(QMouseEvent *event)
|
|
{
|
|
if (event->button()==Qt::LeftButton) {
|
|
double xd=p2x(mouseDragStart.x());
|
|
double yd=p2y(mouseDragStart.y());
|
|
double x=p2x(event->pos().x());
|
|
double y=p2y(event->pos().y());
|
|
emit mouseDragged(xd, yd, x, y, event->modifiers());
|
|
emit mouseDragFinished(xd, yd, x, y, event->modifiers());
|
|
dragging=false;
|
|
if (dragLine) update();
|
|
}
|
|
event->accept();
|
|
}
|
|
|
|
void JKQTFastPlotter::resizeEvent(QResizeEvent *event) {
|
|
JKQTFASTPLOTTER_BASE::resizeEvent(event);
|
|
if (width() > image.width() || height() > image.height()) {
|
|
QImage newImage(QSize(width(), height()), QImage::Format_ARGB32);
|
|
image=newImage;
|
|
systemImage=newImage;
|
|
}
|
|
//std::cout<<"replotting after resize to ("<<width()<<" x "<<height()<<")...\n";
|
|
redrawPlot();
|
|
event->accept();
|
|
updateGeometry();
|
|
}
|
|
|
|
QSize JKQTFastPlotter::minimumSizeHint() const
|
|
{
|
|
if (isVisible()) return QSize(static_cast<int>(round((plotBorderLeft+plotBorderRight)*2.0)), static_cast<int>(round((plotBorderTop+plotBorderBottom)*2.0)));
|
|
else return QSize(0,0);
|
|
}
|
|
|
|
void JKQTFastPlotter::plotSystem(QPainter& painter) {
|
|
#ifdef DEBUG_TIMING
|
|
JKQTPHighResTimer timer;
|
|
double time_sum=0;
|
|
double time_gt=0;
|
|
std::cout<<"timing plotSystem():\n";
|
|
timer.start();
|
|
#endif
|
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
///////////////////////////////////////////////////////////////////
|
|
// FILL WIDGET BACKGROUND
|
|
///////////////////////////////////////////////////////////////////
|
|
if (backgroundColor!=QColor(Qt::transparent)) painter.fillRect(this->rect(), QBrush(backgroundColor));
|
|
///////////////////////////////////////////////////////////////////
|
|
// FILL PLOT BACKGROUND
|
|
///////////////////////////////////////////////////////////////////
|
|
//qDebug()<<plotBackgroundColor;
|
|
painter.fillRect(QRectF(internalPlotBorderLeft, internalPlotBorderTop, plotWidth, plotHeight), QBrush(plotBackgroundColor));
|
|
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" filling backgrounds: "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
// CREATE DIFFERENT QPENs, QFONTs ...
|
|
///////////////////////////////////////////////////////////////////
|
|
QPen p=painter.pen();
|
|
QPen pSystem(systemColor);
|
|
pSystem.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, systemWidth));
|
|
pSystem.setJoinStyle(Qt::MiterJoin);
|
|
pSystem.setCapStyle(Qt::SquareCap);
|
|
QPen pGrid(gridColor);
|
|
pGrid.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, gridWidth));
|
|
pGrid.setStyle(gridStyle);
|
|
pGrid.setJoinStyle(Qt::MiterJoin);
|
|
pGrid.setCapStyle(Qt::SquareCap);
|
|
QFont fTicks(tickFontName);
|
|
fTicks.setPointSizeF(tickFontSize);
|
|
QFontMetricsF fmTicks(fTicks, &systemImage);
|
|
QFont fLabels(labelFontName);
|
|
fLabels.setPointSizeF(labelFontSize);
|
|
QFontMetricsF fmLabels(fLabels, &systemImage);
|
|
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" creating pens ...: "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
// DRAW GRID AND TICKS into QPainterPaths
|
|
///////////////////////////////////////////////////////////////////
|
|
painter.setFont(fTicks);
|
|
painter.setPen(pSystem);
|
|
QPainterPath gridPath, systemPath;
|
|
double x=xZeroTick;
|
|
if (xAxisLog && xZeroTick<=0) x=xMin;
|
|
int cntr=0; // used to stop plotting after 1000 labels (both axes summed up)
|
|
while (x<=xMax && cntr<1000) {
|
|
if (xMin<=x && x<=xMax) {
|
|
gridPath.moveTo(x2p(x), internalPlotBorderTop);
|
|
gridPath.lineTo(x2p(x), internalPlotBorderTop+plotHeight);
|
|
systemPath.moveTo(x2p(x), internalPlotBorderTop+plotHeight+tickLength);
|
|
systemPath.lineTo(x2p(x), internalPlotBorderTop+plotHeight-tickLength);
|
|
QString text=QLocale::system().toString(x);
|
|
painter.drawText(QPointF(x2p(x)-fmTicks.boundingRect(text).width()/2.0, internalPlotBorderTop+plotHeight+fmTicks.ascent()+fmTicks.boundingRect("x").width()/2.0+tickLength), text);
|
|
}
|
|
if (xAxisLog) {
|
|
x=x*10.0;
|
|
} else {
|
|
x=x+xTickDistance;
|
|
}
|
|
cntr++;
|
|
}
|
|
x=xZeroTick;
|
|
if (xAxisLog && xZeroTick<=0) x=xMin;
|
|
while (x>=xMin && cntr<1000) {
|
|
if (xMin<=x && x<=xMax) {
|
|
gridPath.moveTo(x2p(x), internalPlotBorderTop);
|
|
gridPath.lineTo(x2p(x), internalPlotBorderTop+plotHeight);
|
|
systemPath.moveTo(x2p(x), internalPlotBorderTop+plotHeight+tickLength);
|
|
systemPath.lineTo(x2p(x), internalPlotBorderTop+plotHeight-tickLength);
|
|
QString text=QLocale::system().toString(x);
|
|
painter.drawText(QPointF(x2p(x)-fmTicks.boundingRect(text).width()/2.0, internalPlotBorderTop+plotHeight+fmTicks.ascent()+fmTicks.boundingRect("x").width()/2.0+tickLength), text);
|
|
}
|
|
if (xAxisLog) {
|
|
x=x/10.0;
|
|
} else {
|
|
x=x-xTickDistance;
|
|
}
|
|
cntr++;
|
|
}
|
|
|
|
double y=yZeroTick;
|
|
if (yAxisLog && yZeroTick<=0) y=yMin;
|
|
while (y<=yMax && cntr<1000) {
|
|
if (yMin<=y && y<=yMax) {
|
|
gridPath.moveTo(internalPlotBorderLeft, y2p(y));
|
|
gridPath.lineTo(internalPlotBorderLeft+plotWidth, y2p(y));
|
|
systemPath.moveTo(internalPlotBorderLeft-tickLength, y2p(y));
|
|
systemPath.lineTo(internalPlotBorderLeft+tickLength, y2p(y));
|
|
QString text=QLocale::system().toString(y);
|
|
painter.drawText(QPointF(internalPlotBorderLeft-fmTicks.boundingRect("x").width()/2.0-fmTicks.boundingRect(text).width()-tickLength, y2p(y)+fmTicks.ascent()/2.0), text);
|
|
}
|
|
if (yAxisLog) {
|
|
y=y*10.0;
|
|
} else {
|
|
y=y+yTickDistance;
|
|
}
|
|
cntr++;
|
|
}
|
|
y=yZeroTick;
|
|
if (yAxisLog && yZeroTick<=0) y=yMin;
|
|
while (y>=yMin && cntr<1000) {
|
|
if (yMin<=y && y<=yMax) {
|
|
gridPath.moveTo(internalPlotBorderLeft, y2p(y));
|
|
gridPath.lineTo(internalPlotBorderLeft+plotWidth, y2p(y));
|
|
systemPath.moveTo(internalPlotBorderLeft-tickLength, y2p(y));
|
|
systemPath.lineTo(internalPlotBorderLeft+tickLength, y2p(y));
|
|
QString text=QLocale::system().toString(y);
|
|
painter.drawText(QPointF(internalPlotBorderLeft-fmTicks.boundingRect("x").width()/2.0-fmTicks.boundingRect(text).width()-tickLength, y2p(y)+fmTicks.ascent()/2.0), text);
|
|
}
|
|
if (yAxisLog) {
|
|
y=y/10.0;
|
|
} else {
|
|
y=y-yTickDistance;
|
|
}
|
|
cntr++;
|
|
}
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" calcing system & grid: "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
// DRAW GRID
|
|
///////////////////////////////////////////////////////////////////
|
|
if (drawGrid) {
|
|
painter.setPen(pGrid);
|
|
painter.drawPath(gridPath);
|
|
}
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" drawing grid: "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
///////////////////////////////////////////////////////////////////
|
|
// DRAW SYSTEM TICKS
|
|
///////////////////////////////////////////////////////////////////
|
|
painter.setPen(pSystem);
|
|
painter.drawPath(systemPath);
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" drawing system: "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
// DRAW AXIS LABELS
|
|
///////////////////////////////////////////////////////////////////
|
|
if (xAxisLabelVisible) {
|
|
painter.setPen(pSystem);
|
|
painter.setFont(fLabels);
|
|
painter.drawText(QPointF(internalPlotBorderLeft+plotWidth-fmLabels.boundingRect(xAxisLabel).width(), internalPlotBorderTop+plotHeight+fmTicks.height()+fmTicks.boundingRect("x").width()/2.0+fmLabels.ascent()+tickLength), xAxisLabel);
|
|
}
|
|
if (yAxisLabelVisible) {
|
|
painter.save(); auto __finalpaintinner=JKQTPFinally([&painter]() {painter.restore();});
|
|
painter.translate(fmLabels.ascent(), internalPlotBorderTop+fmLabels.boundingRect(yAxisLabel).width());
|
|
painter.rotate(-90);
|
|
painter.drawText(QPointF(0, 0), yAxisLabel);
|
|
|
|
}
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" drawing labels: "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
// DRAW SYSTEM BOX
|
|
///////////////////////////////////////////////////////////////////
|
|
if (drawSystemBox) {
|
|
painter.setPen(pSystem);
|
|
painter.drawRect(QRectF(internalPlotBorderLeft, internalPlotBorderTop, plotWidth, plotHeight));
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
// DRAW AXES at x==0 / y==0
|
|
///////////////////////////////////////////////////////////////////
|
|
if (drawZeroAxes) {
|
|
painter.setPen(pSystem);
|
|
if (xMin<=0 && 0<=xMax) painter.drawLine(QLineF(x2p(0), internalPlotBorderTop, x2p(0), internalPlotBorderTop+plotHeight));
|
|
if (yMin<=0 && 0<=yMax) painter.drawLine(QLineF(internalPlotBorderLeft, y2p(0), internalPlotBorderLeft+plotWidth, y2p(0)));
|
|
}
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" drawing boxes and axes: "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
|
|
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" sum: "<<time_sum/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
}
|
|
|
|
void JKQTFastPlotter::plotGraphs(QPainter& painter) {
|
|
#ifdef DEBUG_TIMING
|
|
JKQTPHighResTimer timer;
|
|
double time_sum=0;
|
|
double time_gt=0;
|
|
std::cout<<"timing plotGraphs():\n";
|
|
timer.start();
|
|
#endif
|
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
for (int i=0; i<plots.size(); i++) {
|
|
#ifdef DEBUG_TIMING
|
|
timer.start();
|
|
#endif
|
|
plots[i]->paint(painter);
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" drawing graph "<<i<<": "<<time_gt/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
}
|
|
|
|
#ifdef DEBUG_TIMING
|
|
time_gt=timer.getTime();
|
|
time_sum+=time_gt;
|
|
std::cout<<" sum: "<<time_sum/1000.0<<" ms\n";
|
|
timer.start();
|
|
#endif
|
|
}
|
|
|
|
void JKQTFastPlotter::draw(QPainter* painter, QSize* size) {
|
|
calcPlotScaling();
|
|
if (size) *size=QSize(width(), height());
|
|
painter->save();
|
|
plotSystem(*painter);
|
|
painter->setClipRect(QRectF(internalPlotBorderLeft+systemWidth/2.0, internalPlotBorderTop+systemWidth/2.0, plotWidth-systemWidth, plotHeight-systemWidth));
|
|
plotGraphs(*painter);
|
|
painter->restore();
|
|
}
|
|
|
|
void JKQTFastPlotter::draw(QPainter* painter, QColor background, QSize* size) {
|
|
QColor oldb=backgroundColor;
|
|
backgroundColor=background;
|
|
calcPlotScaling();
|
|
if (size) *size=QSize(width(), height());
|
|
painter->save();
|
|
plotSystem(*painter);
|
|
//painter->setClipRect(QRectF(internalPlotBorderLeft+systemWidth/2.0, internalPlotBorderTop+systemWidth/2.0, plotWidth-systemWidth, plotHeight-systemWidth));
|
|
plotGraphs(*painter);
|
|
painter->restore();
|
|
backgroundColor=oldb;
|
|
}
|
|
|
|
void JKQTFastPlotter::redrawPlot() {
|
|
calcPlotScaling();
|
|
if (!doDrawing) {
|
|
doFullRepaint=true;
|
|
return;
|
|
}
|
|
if (systemImage.isNull()) return;
|
|
mutexRepaintSystem.lock();
|
|
QPainter* painter=new QPainter(&systemImage);
|
|
//painter->setRenderHints(QPainter::HighQualityAntialiasing|QPainter::Antialiasing|QPainter::SmoothPixmapTransform);
|
|
if (painter->isActive()) {
|
|
plotSystem(*painter);
|
|
}
|
|
delete painter;
|
|
mutexRepaintSystem.unlock();
|
|
doFullRepaint=false;
|
|
updateData();
|
|
}
|
|
|
|
void JKQTFastPlotter::redrawPlotImmediate() {
|
|
calcPlotScaling();
|
|
if (!doDrawing) {
|
|
doFullRepaint=true;
|
|
return;
|
|
}
|
|
if (systemImage.isNull()) return;
|
|
mutexRepaintSystem.lock();
|
|
QPainter* painter=new QPainter(&systemImage);
|
|
if (painter->isActive()) {
|
|
plotSystem(*painter);
|
|
}
|
|
delete painter;
|
|
mutexRepaintSystem.unlock();
|
|
doFullRepaint=false;
|
|
updateDataImmediate();
|
|
}
|
|
|
|
void JKQTFastPlotter::updateData() {
|
|
if (doFullRepaint) {
|
|
// as doFullRepaint is set false in updatePlot() this is NO endles loop!!!!!!!
|
|
redrawPlot();
|
|
return;
|
|
}
|
|
if (!doDrawing) return;
|
|
if (image.isNull()) return;
|
|
mutexRepaintData.lock();
|
|
emit replotting();
|
|
image=systemImage;
|
|
QPainter* painter=new QPainter(&image);
|
|
//painter->setRenderHints(QPainter::HighQualityAntialiasing|QPainter::Antialiasing|QPainter::SmoothPixmapTransform);
|
|
if (painter->isActive()) {
|
|
/*qDebug()<<"image.format: "<<image.format();
|
|
qDebug()<<"painter.compositionMode: "<<painter->compositionMode();
|
|
qDebug()<<"painter.renderHints: "<<painter->renderHints();*/
|
|
//painter->setClipRect(QRectF(internalPlotBorderLeft+systemWidth/2.0, internalPlotBorderTop+systemWidth/2.0, plotWidth-systemWidth, plotHeight-systemWidth));
|
|
plotGraphs(*painter);
|
|
}
|
|
delete painter;
|
|
mutexRepaintData.unlock();
|
|
update();
|
|
}
|
|
|
|
void JKQTFastPlotter::updateDataImmediate() {
|
|
if (doFullRepaint) {
|
|
// as doFullRepaint is set false in updatePlot() this is NO endles loop!!!!!!!
|
|
redrawPlotImmediate();
|
|
return;
|
|
}
|
|
if (!doDrawing) return;
|
|
if (image.isNull()) return;
|
|
mutexRepaintData.lock();
|
|
emit replotting();
|
|
image=systemImage;
|
|
QPainter* painter=new QPainter(&image);
|
|
//painter->setRenderHints(QPainter::HighQualityAntialiasing|QPainter::Antialiasing|QPainter::SmoothPixmapTransform);
|
|
if (painter->isActive()) {
|
|
//painter->setClipRect(QRectF(internalPlotBorderLeft+systemWidth/2.0, internalPlotBorderTop+systemWidth/2.0, plotWidth-systemWidth, plotHeight-systemWidth));
|
|
plotGraphs(*painter);
|
|
}
|
|
delete painter;
|
|
mutexRepaintData.unlock();
|
|
repaint();
|
|
}
|
|
|
|
void JKQTFastPlotter::setXMin(double value){
|
|
setPlotUpdateEnabled(false);
|
|
setXRange(value,xMax,xAxisLog);
|
|
setPlotUpdateEnabled(true);
|
|
}
|
|
|
|
void JKQTFastPlotter::setXMax(double value){
|
|
setPlotUpdateEnabled(false);
|
|
setXRange(xMin,value,xAxisLog);
|
|
setPlotUpdateEnabled(true);
|
|
}
|
|
|
|
void JKQTFastPlotter::setYMin(double value){
|
|
setPlotUpdateEnabled(false);
|
|
setYRange(value,yMax,yAxisLog);
|
|
setPlotUpdateEnabled(true);
|
|
}
|
|
|
|
void JKQTFastPlotter::setYMax(double value){
|
|
setPlotUpdateEnabled(false);
|
|
setYRange(yMin,value,yAxisLog);
|
|
setPlotUpdateEnabled(true);
|
|
}
|
|
|
|
void JKQTFastPlotter::calcPlotScaling() {
|
|
internalPlotBorderBottom=plotBorderBottom;
|
|
internalPlotBorderTop=plotBorderTop;
|
|
internalPlotBorderLeft=plotBorderLeft;
|
|
internalPlotBorderRight=plotBorderRight;
|
|
|
|
plotWidth=width()-internalPlotBorderLeft-internalPlotBorderRight;
|
|
plotHeight=height()-internalPlotBorderBottom-internalPlotBorderTop;
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
// X AXIS, PRESETTINGS
|
|
////////////////////////////////////////////////////////////////////
|
|
if (xAxisLog) {
|
|
if (xMin<=0) xMin=1e-30;
|
|
if (xMax<=0) xMax=xMin+pow(10.0, static_cast<int>(log10(xMin)));
|
|
if (xMax==xMin) {
|
|
xMax=xMin+pow(10.0, static_cast<int>(log10(xMin))+1);
|
|
}
|
|
}
|
|
if (xMin>xMax) {
|
|
double h=xMax;
|
|
xMax=xMin;
|
|
xMin=h;
|
|
} else if (xMax==xMin) {
|
|
xMax=xMin+1.0;
|
|
}
|
|
|
|
// this is the x- and y-range spanned by the plot
|
|
xWidth=fabs(xMax-xMin);
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
// Y AXIS, PRESETTING
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
if (yAxisLog) {
|
|
if (yMin<=0) yMin=1e-30;
|
|
if (yMax<=0) yMax=yMin+pow(10.0, static_cast<int>(log10(yMin)));
|
|
if (yMax==xMin) {
|
|
yMax=yMin+pow(10.0, static_cast<int>(log10(yMin))+1);
|
|
}
|
|
}
|
|
if (yMin>yMax) {
|
|
double h=yMax;
|
|
yMax=yMin;
|
|
yMin=h;
|
|
} else if (yMax==yMin) {
|
|
yMax=yMin+1.0;
|
|
}
|
|
|
|
|
|
// this is the x- and y-range spanned by the plot
|
|
yWidth=fabs(yMax-yMin);
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
// ENSURE ASPECT RATIO (if activated)
|
|
////////////////////////////////////////////////////////////////////
|
|
if (maintainAspectRatio && (!xAxisLog) && (!yAxisLog)) {
|
|
double currRatio=static_cast<double>(plotWidth)/static_cast<double>(plotHeight);
|
|
double newPlotWidth=plotWidth;
|
|
double newPlotHeight=plotHeight;
|
|
if (currRatio!=aspectRatio) {
|
|
if (aspectRatio<=currRatio) {
|
|
newPlotWidth=aspectRatio*static_cast<double>(plotHeight);
|
|
} else {
|
|
newPlotHeight=static_cast<double>(plotWidth)/aspectRatio;
|
|
}
|
|
}
|
|
double dx=plotWidth-newPlotWidth;
|
|
double dy=plotHeight-newPlotHeight;
|
|
internalPlotBorderBottom+=dy/2.0;
|
|
internalPlotBorderTop+=dy/2.0;
|
|
internalPlotBorderLeft+=dx/2.0;
|
|
internalPlotBorderRight+=dx/2.0;
|
|
}
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
// PLOTTER SYNCHRONIZATION
|
|
////////////////////////////////////////////////////////////////////
|
|
if (synchronizeX!=nullptr) {
|
|
internalPlotBorderLeft=synchronizeX->getInternalPlotBorderLeft();
|
|
internalPlotBorderRight=synchronizeX->getInternalPlotBorderRight();
|
|
xMin=synchronizeX->getXMin();
|
|
xMax=synchronizeX->getXMax();
|
|
xAxisLog=synchronizeX->getXAxisLog();
|
|
xAxisLabel=synchronizeX->getXAxisLabel();
|
|
}
|
|
if (synchronizeY!=nullptr) {
|
|
internalPlotBorderTop=synchronizeY->getInternalPlotBorderTop();
|
|
internalPlotBorderBottom=synchronizeY->getInternalPlotBorderBottom();
|
|
yMin=synchronizeY->getYMin();
|
|
yMax=synchronizeY->getYMax();
|
|
yAxisLog=synchronizeY->getYAxisLog();
|
|
yAxisLabel=synchronizeY->getYAxisLabel();
|
|
}
|
|
|
|
|
|
|
|
plotWidth=width()-internalPlotBorderLeft-internalPlotBorderRight;
|
|
plotHeight=height()-internalPlotBorderBottom-internalPlotBorderTop;
|
|
xWidth=fabs(xMax-xMin);
|
|
yWidth=fabs(yMax-yMin);
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
// X AXIS, CALC PARAMETERS
|
|
////////////////////////////////////////////////////////////////////
|
|
if (xAxisLog) {
|
|
xScale=plotWidth/(log10(xMax)-log10(xMin));
|
|
xOffset=internalPlotBorderLeft-log10(xMin)*xScale;
|
|
} else {
|
|
xScale=plotWidth/xWidth;
|
|
xOffset=internalPlotBorderLeft-xMin*xScale;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
// Y AXIS, CALC PARAMETERS
|
|
////////////////////////////////////////////////////////////////////
|
|
if (yAxisLog) {
|
|
yScale=plotHeight/(log10(yMax)-log10(yMin));
|
|
yOffset=internalPlotBorderTop+log10(yMax)*yScale;
|
|
} else {
|
|
yScale=plotHeight/yWidth;
|
|
yOffset=internalPlotBorderTop+yMax*yScale;
|
|
|
|
}
|
|
|
|
emit plotterSizesChanged();
|
|
}
|
|
|
|
void JKQTFastPlotter::setXRange(double min, double max, bool logarithmic) {
|
|
xMin=min;
|
|
xMax=max;
|
|
xAxisLog=logarithmic;
|
|
calcPlotScaling();
|
|
redrawPlot();
|
|
}
|
|
|
|
void JKQTFastPlotter::setYRange(double min, double max, bool logarithmic) {
|
|
yMin=min;
|
|
yMax=max;
|
|
yAxisLog=logarithmic;
|
|
calcPlotScaling();
|
|
redrawPlot();
|
|
}
|
|
|
|
void JKQTFastPlotter::setXYRange(double xmin, double xmax, double ymin, double ymax, bool xlogarithmic, bool ylogarithmic) {
|
|
xMin=xmin;
|
|
xMax=xmax;
|
|
xAxisLog=xlogarithmic;
|
|
yMin=ymin;
|
|
yMax=ymax;
|
|
yAxisLog=ylogarithmic;
|
|
calcPlotScaling();
|
|
redrawPlot();
|
|
}
|
|
|
|
/*
|
|
void JKQTFastPlotter::synchronizeX(JKQTFastPlotter* toWhere) {
|
|
bool olddo=doDrawing;
|
|
doDrawing=false;
|
|
xMin=toWhere->getXMin();
|
|
xMax=toWhere->getXMax();
|
|
xAxisLog=toWhere->getXAxisLog();
|
|
plotBorderLeft=toWhere->getInternalPlotBorderLeft();
|
|
plotBorderRight=toWhere->getInternalPlotBorderRight();
|
|
std::cout<<"syncX: xMin="<<xMin<<" xMax="<<xMax<<" plotBorderLeft="<<plotBorderLeft<<" plotBorderLeft="<<plotBorderRight<<"\n";
|
|
doDrawing=olddo;
|
|
redrawPlot();
|
|
}
|
|
|
|
void JKQTFastPlotter::synchronizeY(JKQTFastPlotter* toWhere) {
|
|
bool olddo=doDrawing;
|
|
doDrawing=false;
|
|
yMin=toWhere->getYMin();
|
|
yMax=toWhere->getYMax();
|
|
yAxisLog=toWhere->getYAxisLog();
|
|
plotBorderTop=toWhere->getInternalPlotBorderTop();
|
|
plotBorderBottom=toWhere->getInternalPlotBorderBottom();
|
|
std::cout<<"syncY: yMin="<<yMin<<" yMax="<<yMax<<" plotBorderTop="<<plotBorderTop<<" plotBorderBottom="<<plotBorderBottom<<"\n";
|
|
doDrawing=olddo;
|
|
redrawPlot();
|
|
}
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
JKQTFPLinePlot::JKQTFPLinePlot(JKQTFastPlotter* parent, int N, double* x, double* y, QColor color, Qt::PenStyle style, double width):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->N=N;
|
|
this->x=x;
|
|
this->y=y;
|
|
this->yerr=nullptr;
|
|
this->xv=nullptr;
|
|
this->yv=nullptr;
|
|
this->yerrv=nullptr;
|
|
this->color=color;
|
|
this->errorColor=color.lighter();
|
|
this->errorStyle=this->style=style;
|
|
this->errorWidth=this->width=width;
|
|
datatype=JKQTFPLPPointer;
|
|
}
|
|
|
|
|
|
JKQTFPLinePlot::JKQTFPLinePlot(JKQTFastPlotter* parent, QVector<double>* x, QVector<double>* y, QColor color, Qt::PenStyle style, double width):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->N=x->size();
|
|
this->x=nullptr;
|
|
this->y=nullptr;
|
|
this->yerr=nullptr;
|
|
this->xv=x;
|
|
this->yv=y;
|
|
this->yerrv=nullptr;
|
|
this->color=color;
|
|
this->errorColor=color.lighter();
|
|
this->errorStyle=this->style=style;
|
|
this->errorWidth=this->width=width;
|
|
datatype=JKQTFPLPVector;
|
|
}
|
|
|
|
|
|
void JKQTFPLinePlot::drawGraph(QPainter& painter) {
|
|
if (datatype==JKQTFPLPPointer) if (N<=0 || !x || !y) return;
|
|
if (datatype==JKQTFPLPVector) if (!xv || !yv) return;
|
|
//std::cout<<"JKQTFPLinePlot::drawGraph()\n";
|
|
QPen p(color);
|
|
p.setStyle(style);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, width));
|
|
p.setCapStyle(Qt::RoundCap);
|
|
p.setJoinStyle(Qt::RoundJoin);
|
|
QPen pe(errorColor);
|
|
pe.setStyle(errorStyle);
|
|
pe.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, errorWidth));
|
|
pe.setCapStyle(Qt::RoundCap);
|
|
pe.setJoinStyle(Qt::RoundJoin);
|
|
QPainterPath path, epath;
|
|
if (datatype==JKQTFPLPPointer) {
|
|
if (N>0){
|
|
path.moveTo(parent->x2p(x[0]), parent->y2p(y[0]));
|
|
//std::cout<<"plotting graph, starting @ ("<<parent->x2p(x[0])<<", "<<parent->y2p(y[0])<<")\n";
|
|
}
|
|
for (int i=1; i<N; i++) {
|
|
path.lineTo(parent->x2p(x[i]), parent->y2p(y[i]));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
if (yerr!=nullptr) {
|
|
if (N>0){
|
|
epath.moveTo(parent->x2p(x[0]), parent->y2p(y[0]+yerr[0]));
|
|
}
|
|
for (int i=1; i<N; i++) {
|
|
epath.lineTo(parent->x2p(x[i]), parent->y2p(y[i]+yerr[i]));
|
|
}
|
|
if (N>0){
|
|
epath.moveTo(parent->x2p(x[0]), parent->y2p(y[0]-yerr[0]));
|
|
}
|
|
for (int i=1; i<N; i++) {
|
|
epath.lineTo(parent->x2p(x[i]), parent->y2p(y[i]-yerr[i]));
|
|
}
|
|
}
|
|
} else if (datatype==JKQTFPLPVector) {
|
|
if (xv->size()>1 && (yv->size()>=xv->size())){
|
|
path.moveTo(parent->x2p((*xv)[0]), parent->y2p((*yv)[0]));
|
|
//std::cout<<"plotting graph, starting @ ("<<parent->x2p(x[0])<<", "<<parent->y2p(y[0])<<")\n";
|
|
for ( int i=1; i<xv->size(); i++) {
|
|
path.lineTo(parent->x2p((*xv)[i]), parent->y2p((*yv)[i]));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
}
|
|
if (yerrv!=nullptr) {
|
|
if (yerrv->size()>1 && (xv->size()>=yerrv->size()) && (yv->size()>=yerrv->size())){
|
|
epath.moveTo(parent->x2p((*xv)[0]), parent->y2p((*yv)[0]+(*yerrv)[0]));
|
|
//std::cout<<"plotting graph, starting @ ("<<parent->x2p(x[0])<<", "<<parent->y2p(y[0])<<")\n";
|
|
for ( int i=1; i<yerrv->size(); i++) {
|
|
epath.lineTo(parent->x2p((*xv)[i]), parent->y2p((*yv)[i]+(*yerrv)[i]));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
epath.moveTo(parent->x2p((*xv)[0]), parent->y2p((*yv)[0]-(*yerrv)[0]));
|
|
//std::cout<<"plotting graph, starting @ ("<<parent->x2p(x[0])<<", "<<parent->y2p(y[0])<<")\n";
|
|
for ( int i=1; i<yerrv->size(); i++) {
|
|
epath.lineTo(parent->x2p((*xv)[i]), parent->y2p((*yv)[i]-(*yerrv)[i]));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
painter.setPen(p);
|
|
painter.drawPath(path);
|
|
painter.setPen(pe);
|
|
painter.drawPath(epath);
|
|
//std::cout<<"JKQTFPLinePlot::drawGraph() ... done\n";
|
|
}
|
|
|
|
|
|
|
|
|
|
JKQTFPVCrossPlot::JKQTFPVCrossPlot(JKQTFastPlotter* parent, int N, double* x, double* y, QColor color, Qt::PenStyle style, double width):
|
|
JKQTFPLinePlot(parent, N, x, y, color, style, width)
|
|
{
|
|
crossWidth=5;
|
|
}
|
|
|
|
JKQTFPVCrossPlot::JKQTFPVCrossPlot(JKQTFastPlotter* parent, QVector<double>* x, QVector<double>* y, QColor color, Qt::PenStyle style, double width):
|
|
JKQTFPLinePlot(parent, x, y, color, style, width)
|
|
{
|
|
crossWidth=5;
|
|
}
|
|
|
|
void JKQTFPVCrossPlot::drawGraph(QPainter& painter) {
|
|
if (datatype==JKQTFPLPPointer) if (N<=0 || !x || !y) return;
|
|
if (datatype==JKQTFPLPVector) if (!xv || !yv) return;
|
|
//std::cout<<"JKQTFPVCrossPlot::drawGraph()\n";
|
|
QPen p(color);
|
|
p.setStyle(style);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, width));
|
|
p.setCapStyle(Qt::RoundCap);
|
|
p.setJoinStyle(Qt::RoundJoin);
|
|
QPen pe(errorColor);
|
|
pe.setStyle(errorStyle);
|
|
pe.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, errorWidth));
|
|
pe.setCapStyle(Qt::RoundCap);
|
|
pe.setJoinStyle(Qt::RoundJoin);
|
|
QPainterPath path, epath;
|
|
if (datatype==JKQTFPLPPointer) {
|
|
for (int i=0; i<N; i++) {
|
|
path.moveTo(QPointF(parent->x2p(x[i])-crossWidth/2.0, parent->y2p(y[i])));
|
|
path.lineTo(QPointF(parent->x2p(x[i])+crossWidth/2.0, parent->y2p(y[i])));
|
|
path.moveTo(QPointF(parent->x2p(x[i]), parent->y2p(y[i])-crossWidth/2.0));
|
|
path.lineTo(QPointF(parent->x2p(x[i]), parent->y2p(y[i])+crossWidth/2.0));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
if (yerr!=nullptr) {
|
|
if (N>0){
|
|
epath.moveTo(parent->x2p(x[0]), parent->y2p(y[0]+yerr[0]));
|
|
}
|
|
for (int i=1; i<N; i++) {
|
|
epath.lineTo(parent->x2p(x[i]), parent->y2p(y[i]+yerr[i]));
|
|
}
|
|
if (N>0){
|
|
epath.moveTo(parent->x2p(x[0]), parent->y2p(y[0]-yerr[0]));
|
|
}
|
|
for (int i=1; i<N; i++) {
|
|
epath.lineTo(parent->x2p(x[i]), parent->y2p(y[i]-yerr[i]));
|
|
}
|
|
}
|
|
} else if (datatype==JKQTFPLPVector) {
|
|
if (xv->size()>0 && (yv->size()>=xv->size())){
|
|
for ( int i=0; i<xv->size(); i++) {
|
|
path.moveTo(QPointF(parent->x2p((*xv)[i])-crossWidth/2.0, parent->y2p((*yv)[i])));
|
|
path.lineTo(QPointF(parent->x2p((*xv)[i])+crossWidth/2.0, parent->y2p((*yv)[i])));
|
|
path.moveTo(QPointF(parent->x2p((*xv)[i]), parent->y2p((*yv)[i])-crossWidth/2.0));
|
|
path.lineTo(QPointF(parent->x2p((*xv)[i]), parent->y2p((*yv)[i])+crossWidth/2.0));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
}
|
|
if (yerrv!=nullptr) {
|
|
if (yerrv->size()>0 && (xv->size()>=yerrv->size()) && (yv->size()>=yerrv->size())){
|
|
epath.moveTo(parent->x2p((*xv)[0]), parent->y2p((*yv)[0]+(*yerrv)[0]));
|
|
//std::cout<<"plotting graph, starting @ ("<<parent->x2p(x[0])<<", "<<parent->y2p(y[0])<<")\n";
|
|
for ( int i=1; i<yerrv->size(); i++) {
|
|
epath.lineTo(parent->x2p((*xv)[i]), parent->y2p((*yv)[i]+(*yerrv)[i]));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
epath.moveTo(parent->x2p((*xv)[0]), parent->y2p((*yv)[0]-(*yerrv)[0]));
|
|
//std::cout<<"plotting graph, starting @ ("<<parent->x2p(x[0])<<", "<<parent->y2p(y[0])<<")\n";
|
|
for ( int i=1; i<yerrv->size(); i++) {
|
|
epath.lineTo(parent->x2p((*xv)[i]), parent->y2p((*yv)[i]-(*yerrv)[i]));
|
|
//std::cout<<"-> ("<<parent->x2p(x[i])<<", "<<parent->y2p(y[i])<<")\n";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
painter.setPen(p);
|
|
painter.drawPath(path);
|
|
painter.setPen(pe);
|
|
painter.drawPath(epath);
|
|
//std::cout<<"JKQTFPVCrossPlot::drawGraph() ... done\n";
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
JKQTFPVBarPlot::JKQTFPVBarPlot(JKQTFastPlotter* parent, int N, double* x, double* y, QColor color, Qt::PenStyle style, double width):
|
|
JKQTFPLinePlot(parent, N, x, y, color, style, width)
|
|
{
|
|
}
|
|
|
|
JKQTFPVBarPlot::JKQTFPVBarPlot(JKQTFastPlotter* parent, QVector<double>* x, QVector<double>* y, QColor color, Qt::PenStyle style, double width):
|
|
JKQTFPLinePlot(parent, x, y, color, style, width)
|
|
{
|
|
}
|
|
|
|
void JKQTFPVBarPlot::drawGraph(QPainter& painter) {
|
|
if (datatype==JKQTFPLPPointer) if (N<=0 || !x || !y) return;
|
|
if (datatype==JKQTFPLPVector) if (!xv || !yv) return;
|
|
|
|
QPen p(color);
|
|
p.setStyle(style);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, width));
|
|
p.setCapStyle(Qt::RoundCap);
|
|
p.setJoinStyle(Qt::RoundJoin);
|
|
|
|
painter.setPen(p);
|
|
if (!parent->getYAxisLog()) {
|
|
if (datatype==JKQTFPLPPointer) {
|
|
for (int i=0; i<N; i++) {
|
|
double xval=parent->x2p(x[i]);
|
|
QLineF l(xval, parent->y2p(0), xval, parent->y2p(y[i]));
|
|
if (l.length()>0) painter.drawLine(l);
|
|
}
|
|
} else if (datatype==JKQTFPLPVector) {
|
|
for ( int i=0; i<xv->size(); i++) {
|
|
double xval=parent->x2p((*xv)[i]);
|
|
QLineF l(xval, parent->y2p(0), xval, parent->y2p((*yv)[i]));
|
|
if (l.length()>0) painter.drawLine(l);
|
|
}
|
|
}
|
|
} else {
|
|
double starty=parent->getInternalPlotBorderTop()+parent->getPlotHeight();
|
|
if (datatype==JKQTFPLPPointer) {
|
|
for (int i=0; i<N; i++) {
|
|
double v=parent->y2p(y[i]);
|
|
if (!std::isnan(v) && !std::isinf(v)) {
|
|
double xval=parent->x2p(x[i]);
|
|
QLineF l(xval, starty, xval, v);
|
|
if (l.length()>0) painter.drawLine(l);
|
|
}
|
|
}
|
|
} else if (datatype==JKQTFPLPVector) {
|
|
for ( int i=0; i<xv->size(); i++) {
|
|
double v=parent->y2p((*yv)[i]);
|
|
if (!std::isnan(v) && !std::isinf(v)) {
|
|
double xval=parent->x2p((*xv)[i]);
|
|
QLineF l(xval, starty, xval, v);
|
|
if (l.length()>0) painter.drawLine(l);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
JKQTFPXRangePlot::JKQTFPXRangePlot(JKQTFastPlotter* parent, double xmin, double xmax, QColor color, Qt::PenStyle style, double width, Qt::BrushStyle fillStyle):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->centerline=(xmax+xmin)/2.0;
|
|
this->showCenterline=false;
|
|
this->color=color;
|
|
this->style=style;
|
|
this->width=width;
|
|
this->fillColor=color;
|
|
this->fillColor.setAlphaF(0.5);
|
|
this->fillStyle=fillStyle;
|
|
}
|
|
|
|
void JKQTFPXRangePlot::drawGraph(QPainter& painter) {
|
|
QRectF r(QPointF(parent->x2p(xmin), parent->getInternalPlotBorderTop()), QPointF(parent->x2p(xmax), parent->getInternalPlotBorderTop()+parent->getPlotHeight()));
|
|
QBrush b(fillStyle);
|
|
b.setColor(fillColor);
|
|
QPen p(color);
|
|
p.setStyle(style);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, width));
|
|
painter.setPen(p);
|
|
painter.fillRect(r, b);
|
|
if (showCenterline) {
|
|
painter.drawLine(QLineF(parent->x2p(centerline), parent->getInternalPlotBorderTop(), parent->x2p(centerline), parent->getInternalPlotBorderTop()+parent->getPlotHeight()));
|
|
}
|
|
painter.drawRect(r);
|
|
}
|
|
|
|
JKQTFPQImagePlot::JKQTFPQImagePlot(JKQTFastPlotter* parent, QImage* image, double xmin, double xmax, double ymin, double ymax):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->image=image;
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->ymin=ymin;
|
|
this->ymax=ymax;
|
|
}
|
|
|
|
JKQTFPQImagePlot::JKQTFPQImagePlot(JKQTFastPlotter* parent, QImage* image):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->image=image;
|
|
this->xmin=0;
|
|
this->xmax=image->width();
|
|
this->ymin=0;
|
|
this->ymax=image->height();
|
|
}
|
|
|
|
void JKQTFPQImagePlot::drawGraph(QPainter& painter) {
|
|
if (!image) return;
|
|
if (image->isNull()) return;
|
|
double pxmin=parent->x2p(xmin);
|
|
double pxmax=parent->x2p(xmax);
|
|
double dx=fabs(pxmax-pxmin);
|
|
double pymin=parent->y2p(ymin);
|
|
double pymax=parent->y2p(ymax);
|
|
double dy=fabs(pymax-pymin);
|
|
if (dx>0 && dy>0)
|
|
painter.drawImage(QRectF(pxmin, pymax, dx, dy), *image);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
JKQTFPimagePlot::JKQTFPimagePlot(JKQTFastPlotter* parent, void* image, JKQTFPImageFormat imageFormat, int width, int height, double xmin, double xmax, double ymin, double ymax, JKQTFPColorPalette palette):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->image=image;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->ymin=ymin;
|
|
this->ymax=ymax;
|
|
colorMin=0;
|
|
colorMax=0;
|
|
this->palette=palette;
|
|
drawColorBar=true;
|
|
colorBarWidth=15;
|
|
this->imageFormat=imageFormat;
|
|
this->rotation=0;
|
|
}
|
|
|
|
JKQTFPimagePlot::JKQTFPimagePlot(JKQTFastPlotter* parent, void* image, JKQTFPImageFormat imageFormat, int width, int height, JKQTFPColorPalette palette):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->image=image;
|
|
this->width=width;
|
|
this->height=height;
|
|
colorMin=0;
|
|
colorMax=0;
|
|
this->palette=palette;
|
|
drawColorBar=true;
|
|
colorBarWidth=15;
|
|
xmin=0;
|
|
ymin=0;
|
|
xmax=width;
|
|
ymax=height;
|
|
this->imageFormat=imageFormat;
|
|
this->rotation=0;
|
|
}
|
|
|
|
|
|
QStringList JKQTFPimagePlot_getPalettes() {
|
|
QStringList sl;
|
|
sl<<QObject::tr("red");
|
|
sl<<QObject::tr("inverted red");
|
|
sl<<QObject::tr("green");
|
|
sl<<QObject::tr("inverted green");
|
|
sl<<QObject::tr("blue");
|
|
sl<<QObject::tr("inverted blue");
|
|
sl<<QObject::tr("gray");
|
|
sl<<QObject::tr("inverted gray");
|
|
sl<<QObject::tr("Matlab");
|
|
sl<<QObject::tr("RYGB");
|
|
sl<<QObject::tr("HSV");
|
|
sl<<QObject::tr("inverted HSV");
|
|
sl<<QObject::tr("rainbow");
|
|
sl<<QObject::tr("AFM hot");
|
|
sl<<QObject::tr("ocean");
|
|
sl<<QObject::tr("trafficlight");
|
|
sl<<QObject::tr("blue-magenta-yellow");
|
|
sl<<QObject::tr("blue-yellow");
|
|
sl<<QObject::tr("cyan");
|
|
//sl<<QObject::tr("");
|
|
return sl;
|
|
}
|
|
|
|
QIcon JKQTFPimagePlot_getPaletteIcon(int i) {
|
|
QImage img;
|
|
double pic[32];
|
|
for (int j=0; j<32; j++) {
|
|
pic[j]=j;
|
|
}
|
|
JKQTFPimagePlot_array2image<double>(pic, 32, 1, img, static_cast<JKQTFPColorPalette>(i), 0, 31);
|
|
QPixmap pix(16,8);
|
|
QRect r(0,0,15,7);
|
|
QPainter p(&pix);
|
|
p.drawImage(r, img);
|
|
p.setPen(QPen(QColor("black")));
|
|
p.drawRect(r);
|
|
p.end();
|
|
return QIcon(pix);
|
|
|
|
}
|
|
|
|
void JKQTFPimagePlot::drawGraph(QPainter& painter) {
|
|
if (!image) return;
|
|
if (width<=0 && height<=0) return;
|
|
|
|
//QPainter::CompositionMode cm=painter.compositionMode();
|
|
painter.setCompositionMode(QPainter::CompositionMode_SourceOver);
|
|
QImage img(width, height, QImage::Format_ARGB32);
|
|
switch(imageFormat) {
|
|
case JKQTFP_uint8: JKQTFPimagePlot_array2image<uint8_t>(static_cast<uint8_t*>(image), width, height, img, palette, colorMin, colorMax); break;
|
|
case JKQTFP_uint16: JKQTFPimagePlot_array2image<uint16_t>(static_cast<uint16_t*>(image), width, height, img, palette, colorMin, colorMax); break;
|
|
case JKQTFP_uint32: JKQTFPimagePlot_array2image<uint32_t>(static_cast<uint32_t*>(image), width, height, img, palette, colorMin, colorMax); break;
|
|
case JKQTFP_int64: JKQTFPimagePlot_array2image<int64_t>(static_cast<int64_t*>(image), width, height, img, palette, colorMin, colorMax); break;
|
|
case JKQTFP_float: JKQTFPimagePlot_array2image<float>(static_cast<float*>(image), width, height, img, palette, colorMin, colorMax); break;
|
|
case JKQTFP_double: JKQTFPimagePlot_array2image<double>(static_cast<double*>(image), width, height, img, palette, colorMin, colorMax); break;
|
|
}
|
|
if (img.isNull()) return;
|
|
double pxmin=parent->x2p(xmin);
|
|
double pxmax=parent->x2p(xmax);
|
|
double dx=fabs(pxmax-pxmin);
|
|
double pymin=parent->y2p(ymin);
|
|
double pymax=parent->y2p(ymax);
|
|
double dy=fabs(pymax-pymin);
|
|
QTransform trans;
|
|
trans.reset();
|
|
if (rotation==1) trans.rotate(90);
|
|
else if (rotation==2) trans.rotate(180);
|
|
else if (rotation==3) trans.rotate(270);
|
|
|
|
if (dx>0 && dy>0)
|
|
painter.drawImage(QRectF(pxmin, pymax, dx, dy), img.transformed(trans));
|
|
if (drawColorBar && parent->getPlotHeight()>3) {
|
|
uint8_t d[200];
|
|
for (uint8_t i=0; i<200; i++) d[i]=i;
|
|
QImage b(1, 200, QImage::Format_ARGB32);
|
|
JKQTFPimagePlot_array2image<uint8_t>(d, 1, 200, b, palette, 0, 199);
|
|
//std::cout<<"bar.width="<<b.width()<<" bar.height="<<b.height()<<"\n";
|
|
QRectF r(parent->getInternalPlotBorderLeft()+parent->getPlotWidth()+parent->getTickLength()*2, parent->getInternalPlotBorderTop()+parent->getPlotHeight()*0.15, colorBarWidth, parent->getPlotHeight()*0.8);
|
|
//std::cout<<"r.left="<<r.left()<<" r.top="<<r.top()<<" r.width="<<r.width()<<" r.height="<<r.height()<<"\n";
|
|
bool c=painter.hasClipping();
|
|
painter.setClipping(false);
|
|
painter.drawImage(r, b);
|
|
painter.drawRect(r);
|
|
painter.setClipping(c);
|
|
}
|
|
//painter.setCompositionMode(cm);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JKQTFPimageOverlayPlot::JKQTFPimageOverlayPlot(JKQTFastPlotter* parent, bool* image, int width, int height, double xmin, double xmax, double ymin, double ymax, QColor color):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->image=image;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->ymin=ymin;
|
|
this->ymax=ymax;
|
|
this->color=color;
|
|
this->rotation=0;
|
|
this->showAsSymbols=false;
|
|
this->linewidth=2;
|
|
this->symboltype=stCircle;
|
|
}
|
|
|
|
|
|
JKQTFPimageOverlayPlot::JKQTFPimageOverlayPlot(JKQTFastPlotter* parent, bool* image, int width, int height, QColor color):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->image=image;
|
|
this->width=width;
|
|
this->height=height;
|
|
xmin=0;
|
|
ymin=0;
|
|
xmax=width;
|
|
ymax=height;
|
|
this->color=color;
|
|
this->linewidth=2;
|
|
this->rotation=0;
|
|
this->showAsSymbols=false;
|
|
this->symboltype=stCircle;
|
|
}
|
|
|
|
|
|
void JKQTFPimageOverlayPlot::drawGraph(QPainter& painter) {
|
|
if (!image) return;
|
|
if (width<=0 && height<=0) return;
|
|
|
|
QTransform trans;
|
|
trans.reset();
|
|
if (rotation==1) trans.rotate(90);
|
|
else if (rotation==2) trans.rotate(180);
|
|
else if (rotation==3) trans.rotate(270);
|
|
|
|
double pxmin=parent->x2p(xmin);
|
|
double pxmax=parent->x2p(xmax);
|
|
double dx=fabs(pxmax-pxmin);
|
|
double pymin=parent->y2p(ymin);
|
|
double pymax=parent->y2p(ymax);
|
|
double dy=fabs(pymax-pymin);
|
|
double ddx=dx/double(width);
|
|
double ddy=dy/double(height);
|
|
|
|
if (showAsSymbols) {
|
|
QList<double> x,y;
|
|
for (int j=0; j<height; ++j) {
|
|
for (int i=0; i<width; ++i) {
|
|
if (image[j*width+i]) {
|
|
x<<xmin+(double(i)+0.5)/double(width)*(xmax-xmin);
|
|
y<<ymin+(double(j)+0.5)/double(height)*(ymax-ymin);
|
|
}
|
|
}
|
|
}
|
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
painter.setTransform(trans, true);
|
|
QPen p=painter.pen();
|
|
p.setColor(color);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH,linewidth));
|
|
painter.setPen(p);
|
|
for (int i=0; i<qMin(x.size(), y.size()); i++) {
|
|
double px=parent->x2p(x[i]);
|
|
double py=parent->y2p(y[i]);
|
|
QRectF r(px-ddx/2.0, py-ddy/2.0, ddx, ddy);
|
|
if (this->symboltype==stCircle) painter.drawEllipse(r);
|
|
else painter.drawRect(r);
|
|
}
|
|
|
|
} else {
|
|
QImage img(width, height, QImage::Format_ARGB32);
|
|
QColor tc(Qt::transparent);
|
|
for (int j=0; j<height; ++j) {
|
|
QRgb* line=reinterpret_cast<QRgb *>(img.scanLine(height-1-j));
|
|
for (int i=0; i<width; ++i) {
|
|
if (!image[j*width+i]) line[i]=tc.rgba();
|
|
else line[i]=color.rgba();
|
|
}
|
|
}
|
|
|
|
if (img.isNull()) return;
|
|
|
|
|
|
if (dx>0 && dy>0) {
|
|
//qDebug()<<painter.compositionMode()<<img.transformed(trans).format();
|
|
painter.drawImage(QRectF(pxmin, pymax, dx, dy), img.transformed(trans, Qt::SmoothTransformation));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
JKQTFPQScaleBarXPlot::JKQTFPQScaleBarXPlot(JKQTFastPlotter* parent, double width, const QString& label, QColor color, double lineWidth):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->width=width;
|
|
this->label=label;
|
|
this->lineWidth=lineWidth;
|
|
this->color=color;
|
|
this->position=JKQTFPQScaleBarXPlot::TopRight;
|
|
if (parent) {
|
|
this->font.setFamily(parent->getLabelFontName());
|
|
this->font.setPointSizeF(parent->getLabelFontSize());
|
|
}
|
|
}
|
|
|
|
void JKQTFPQScaleBarXPlot::drawGraph(QPainter& painter) {
|
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
|
|
|
|
QRectF r(QPointF(parent->getInternalPlotBorderLeft(), parent->getInternalPlotBorderTop()), QPointF(parent->getInternalPlotBorderLeft()+parent->getPlotWidth(), parent->getInternalPlotBorderTop()+parent->getPlotHeight()));
|
|
|
|
double borderfraction=0.1;
|
|
int yDistance=static_cast<int>(static_cast<double>(parent->getPlotHeight())*borderfraction);
|
|
|
|
QPen p(color);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, lineWidth));
|
|
p.setColor(color);
|
|
QString s=label.arg(width);
|
|
|
|
double yy1, yy2, xx1, xx2, x1, x2;//, y2, y1;
|
|
|
|
if (position==JKQTFPQScaleBarXPlot::TopRight) {
|
|
yy1=parent->getInternalPlotBorderTop()+yDistance;
|
|
yy2=yy1;
|
|
x2=parent->getXMax()-borderfraction*(parent->getXMax()-parent->getXMin());
|
|
x1=x2-width;
|
|
xx1=parent->x2p(x1);
|
|
xx2=parent->x2p(x2);
|
|
painter.drawLine(QLineF(xx1, yy1, xx2, yy2));
|
|
painter.setFont(font);
|
|
QFontMetrics fm=painter.fontMetrics();
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,11,0)
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.horizontalAdvance(s)/2), static_cast<int>(yy1+3*lineWidth+fm.ascent()), s);
|
|
#else
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.width(s)/2), static_cast<int>(yy1+3*lineWidth+fm.ascent()), s);
|
|
#endif
|
|
} else if (position==JKQTFPQScaleBarXPlot::TopLeft) {
|
|
yy1=parent->getInternalPlotBorderTop()+yDistance;
|
|
yy2=yy1;
|
|
x1=parent->getXMin()+borderfraction*(parent->getXMax()-parent->getXMin());
|
|
x2=x1+width;
|
|
xx1=parent->x2p(x1);
|
|
xx2=parent->x2p(x2);
|
|
painter.drawLine(QLineF(xx1, yy1, xx2, yy2));
|
|
painter.setFont(font);
|
|
QFontMetrics fm=painter.fontMetrics();
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,11,0)
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.horizontalAdvance(s)/2), static_cast<int>(yy1+3*lineWidth+fm.ascent()), s);
|
|
#else
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.width(s)/2), static_cast<int>(yy1+3*lineWidth+fm.ascent()), s);
|
|
#endif
|
|
} else if (position==JKQTFPQScaleBarXPlot::BottomLeft) {
|
|
yy1=parent->getInternalPlotBorderTop()+parent->getPlotHeight()-yDistance;
|
|
yy2=yy1;
|
|
x1=parent->getXMin()+borderfraction*(parent->getXMax()-parent->getXMin());
|
|
x2=x1+width;
|
|
xx1=parent->x2p(x1);
|
|
xx2=parent->x2p(x2);
|
|
painter.drawLine(QLineF(xx1, yy1, xx2, yy2));
|
|
painter.setFont(font);
|
|
QFontMetrics fm=painter.fontMetrics();
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,11,0)
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.horizontalAdvance(s)/2), static_cast<int>(yy1-3*lineWidth-fm.descent()), s);
|
|
#else
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.width(s)/2), static_cast<int>(yy1-3*lineWidth-fm.descent()), s);
|
|
#endif
|
|
} else if (position==JKQTFPQScaleBarXPlot::BottomRight) {
|
|
yy1=parent->getInternalPlotBorderTop()+parent->getPlotHeight()-yDistance;
|
|
yy2=yy1;
|
|
x2=parent->getXMax()-borderfraction*(parent->getXMax()-parent->getXMin());
|
|
x1=x2-width;
|
|
xx1=parent->x2p(x1);
|
|
xx2=parent->x2p(x2);
|
|
painter.drawLine(QLineF(xx1, yy1, xx2, yy2));
|
|
painter.setFont(font);
|
|
QFontMetrics fm=painter.fontMetrics();
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,11,0)
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.horizontalAdvance(s)/2), static_cast<int>(yy1-3*lineWidth-fm.descent()), s);
|
|
#else
|
|
painter.drawText(static_cast<int>(xx1+(xx2-xx1)/2-fm.width(s)/2), static_cast<int>(yy1-3*lineWidth-fm.descent()), s);
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
QIcon JKQTFPimagePlot_getPaletteIcon(JKQTFPColorPalette palette) {
|
|
return JKQTFPimagePlot_getPaletteIcon(static_cast<int>(palette));
|
|
};
|
|
|
|
|
|
JKQTFPQOverlayLinearGridPlot::JKQTFPQOverlayLinearGridPlot(JKQTFastPlotter* parent, double width, QColor color, double lineWidth, Qt::PenStyle style):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->width=width;
|
|
this->color=color;
|
|
this->lineWidth=lineWidth;
|
|
this->style=style;
|
|
}
|
|
|
|
void JKQTFPQOverlayLinearGridPlot::drawGraph(QPainter& painter) {
|
|
painter.save(); auto __finalpaint=JKQTPFinally([&painter]() {painter.restore();});
|
|
|
|
QPen p(color);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, lineWidth));
|
|
p.setColor(color);
|
|
p.setStyle(style);
|
|
|
|
double xmin=parent->getXMin();
|
|
double xmax=parent->getXMax();
|
|
double ymin=parent->getYMin();
|
|
double ymax=parent->getYMax();
|
|
|
|
double xstart=(floor(xmin/width)-1.0)*width;
|
|
double ystart=(floor(ymin/width)-1.0)*width;
|
|
|
|
QPainterPath gridPath;
|
|
double x=xstart;
|
|
int cntr=0; // used to stop plotting after 1000 grid lines
|
|
while (x<=xmax && cntr<1000) {
|
|
if (xmin<=x && x<=xmax) {
|
|
gridPath.moveTo(parent->x2p(x), parent->getInternalPlotBorderTop());
|
|
gridPath.lineTo(parent->x2p(x), parent->getInternalPlotBorderTop()+parent->getPlotHeight());
|
|
}
|
|
x+=width;
|
|
cntr++;
|
|
}
|
|
double y=ystart;
|
|
cntr=0;
|
|
while (y<=ymax && cntr<1000) {
|
|
if (ymin<=y && y<=ymax) {
|
|
gridPath.moveTo(parent->getInternalPlotBorderLeft(), parent->y2p(y));
|
|
gridPath.lineTo(parent->getInternalPlotBorderLeft()+parent->getPlotWidth(), parent->y2p(y));
|
|
}
|
|
y+=width;
|
|
cntr++;
|
|
}
|
|
|
|
painter.setPen(p);
|
|
painter.drawPath(gridPath);
|
|
|
|
}
|
|
|
|
|
|
|
|
JKQTFPRGBImageOverlayPlot::JKQTFPRGBImageOverlayPlot(JKQTFastPlotter *parent, void *imageRed, JKQTFPImageFormat imageFormat, int width, int height, double xmin, double xmax, double ymin, double ymax):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->ymin=ymin;
|
|
this->ymax=ymax;
|
|
colorMinRed=0;
|
|
colorMaxRed=0;
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormat;
|
|
colorMinGreen=0;
|
|
colorMaxGreen=0;
|
|
this->imageGreen=nullptr;
|
|
this->imageFormatGreen=JKQTFP_uint16;
|
|
colorMinBlue=0;
|
|
colorMaxBlue=0;
|
|
this->imageBlue=nullptr;
|
|
this->imageFormatBlue=JKQTFP_uint16;
|
|
this->rotation=0;
|
|
}
|
|
|
|
JKQTFPRGBImageOverlayPlot::JKQTFPRGBImageOverlayPlot(JKQTFastPlotter *parent, void *imageRed, JKQTFPImageFormat imageFormat, int width, int height):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=0;
|
|
this->xmax=width-1;
|
|
this->ymin=0;
|
|
this->ymax=height-1;
|
|
colorMinRed=0;
|
|
colorMaxRed=0;
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormat;
|
|
colorMinGreen=0;
|
|
colorMaxGreen=0;
|
|
this->imageGreen=nullptr;
|
|
this->imageFormatGreen=JKQTFP_uint16;
|
|
colorMinBlue=0;
|
|
colorMaxBlue=0;
|
|
this->imageBlue=nullptr;
|
|
this->imageFormatBlue=JKQTFP_uint16;
|
|
this->rotation=0;
|
|
}
|
|
|
|
JKQTFPRGBImageOverlayPlot::JKQTFPRGBImageOverlayPlot(JKQTFastPlotter *parent):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->width=0;
|
|
this->height=0;
|
|
this->xmin=0;
|
|
this->xmax=11;
|
|
this->ymin=0;
|
|
this->ymax=11;
|
|
colorMinRed=0;
|
|
colorMaxRed=0;
|
|
this->imageRed=nullptr;
|
|
this->imageFormatRed=JKQTFP_uint16;
|
|
colorMinGreen=0;
|
|
colorMaxGreen=0;
|
|
this->imageGreen=nullptr;
|
|
this->imageFormatGreen=JKQTFP_uint16;
|
|
colorMinBlue=0;
|
|
colorMaxBlue=0;
|
|
this->imageBlue=nullptr;
|
|
this->imageFormatBlue=JKQTFP_uint16;
|
|
this->rotation=0;
|
|
}
|
|
|
|
void JKQTFPRGBImageOverlayPlot::drawGraph(QPainter &painter) {
|
|
//qDebug()<<"JKQTFPRGBImageOverlayPlot::drawGraph() "<<imageRed<<imageGreen<<imageBlue<<width<<height;
|
|
if (!imageRed && !imageGreen && !imageBlue) return;
|
|
if (width<=0 && height<=0) return;
|
|
|
|
QImage img(width, height, QImage::Format_ARGB32);
|
|
img.fill(0);
|
|
switch(imageFormatRed) {
|
|
case JKQTFP_uint8: JKQTFPRGBImageOverlayPlot_array2image<uint8_t>(static_cast<uint8_t*>(imageRed), width, height, img, 0, colorMinRed, colorMaxRed); break;
|
|
case JKQTFP_uint16: JKQTFPRGBImageOverlayPlot_array2image<uint16_t>(static_cast<uint16_t*>(imageRed), width, height, img, 0, colorMinRed, colorMaxRed); break;
|
|
case JKQTFP_uint32: JKQTFPRGBImageOverlayPlot_array2image<uint32_t>(static_cast<uint32_t*>(imageRed), width, height, img, 0, colorMinRed, colorMaxRed); break;
|
|
case JKQTFP_int64: JKQTFPRGBImageOverlayPlot_array2image<int64_t>(static_cast<int64_t*>(imageRed), width, height, img, 0, colorMinRed, colorMaxRed); break;
|
|
case JKQTFP_float: JKQTFPRGBImageOverlayPlot_array2image<float>(static_cast<float*>(imageRed), width, height, img, 0, colorMinRed, colorMaxRed); break;
|
|
case JKQTFP_double: JKQTFPRGBImageOverlayPlot_array2image<double>(static_cast<double*>(imageRed), width, height, img, 0, colorMinRed, colorMaxRed); break;
|
|
}
|
|
switch(imageFormatGreen) {
|
|
case JKQTFP_uint8: JKQTFPRGBImageOverlayPlot_array2image<uint8_t>(static_cast<uint8_t*>(imageGreen), width, height, img, 1, colorMinGreen, colorMaxGreen); break;
|
|
case JKQTFP_uint16: JKQTFPRGBImageOverlayPlot_array2image<uint16_t>(static_cast<uint16_t*>(imageGreen), width, height, img, 1, colorMinGreen, colorMaxGreen); break;
|
|
case JKQTFP_uint32: JKQTFPRGBImageOverlayPlot_array2image<uint32_t>(static_cast<uint32_t*>(imageGreen), width, height, img, 1, colorMinGreen, colorMaxGreen); break;
|
|
case JKQTFP_int64: JKQTFPRGBImageOverlayPlot_array2image<int64_t>(static_cast<int64_t*>(imageGreen), width, height, img, 1, colorMinGreen, colorMaxGreen); break;
|
|
case JKQTFP_float: JKQTFPRGBImageOverlayPlot_array2image<float>(static_cast<float*>(imageGreen), width, height, img, 1, colorMinGreen, colorMaxGreen); break;
|
|
case JKQTFP_double: JKQTFPRGBImageOverlayPlot_array2image<double>(static_cast<double*>(imageGreen), width, height, img, 1, colorMinGreen, colorMaxGreen); break;
|
|
}
|
|
switch(imageFormatBlue) {
|
|
case JKQTFP_uint8: JKQTFPRGBImageOverlayPlot_array2image<uint8_t>(static_cast<uint8_t*>(imageBlue), width, height, img, 2, colorMinBlue, colorMaxBlue); break;
|
|
case JKQTFP_uint16: JKQTFPRGBImageOverlayPlot_array2image<uint16_t>(static_cast<uint16_t*>(imageBlue), width, height, img, 2, colorMinBlue, colorMaxBlue); break;
|
|
case JKQTFP_uint32: JKQTFPRGBImageOverlayPlot_array2image<uint32_t>(static_cast<uint32_t*>(imageBlue), width, height, img, 2, colorMinBlue, colorMaxBlue); break;
|
|
case JKQTFP_int64: JKQTFPRGBImageOverlayPlot_array2image<int64_t>(static_cast<int64_t*>(imageBlue), width, height, img, 2, colorMinBlue, colorMaxBlue); break;
|
|
case JKQTFP_float: JKQTFPRGBImageOverlayPlot_array2image<float>(static_cast<float*>(imageBlue), width, height, img, 2, colorMinBlue, colorMaxBlue); break;
|
|
case JKQTFP_double: JKQTFPRGBImageOverlayPlot_array2image<double>(static_cast<double*>(imageBlue), width, height, img, 2, colorMinBlue, colorMaxBlue); break;
|
|
}
|
|
if (img.isNull()) return;
|
|
double pxmin=parent->x2p(xmin);
|
|
double pxmax=parent->x2p(xmax);
|
|
double dx=fabs(pxmax-pxmin);
|
|
double pymin=parent->y2p(ymin);
|
|
double pymax=parent->y2p(ymax);
|
|
double dy=fabs(pymax-pymin);
|
|
QTransform trans;
|
|
trans.reset();
|
|
if (rotation==1) trans.rotate(90);
|
|
else if (rotation==2) trans.rotate(180);
|
|
else if (rotation==3) trans.rotate(270);
|
|
|
|
if (dx>0 && dy>0)
|
|
painter.drawImage(QRectF(pxmin, pymax, dx, dy), img.transformed(trans));
|
|
}
|
|
|
|
|
|
void JKQTFPRGBImageOverlayPlot::setImage(void* imageRed, JKQTFPImageFormat imageFormatRed, int width, int height){
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormatRed;
|
|
this->imageGreen=nullptr;
|
|
this->imageBlue=nullptr;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=0;
|
|
this->xmax=width-1;
|
|
this->ymin=0;
|
|
this->ymax=height-1;
|
|
|
|
replot();
|
|
}
|
|
void JKQTFPRGBImageOverlayPlot::setImage(void* imageRed, JKQTFPImageFormat imageFormatRed, void* imageGreen, JKQTFPImageFormat imageFormatGreen, int width, int height) {
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormatRed;
|
|
this->imageGreen=imageGreen;
|
|
this->imageFormatGreen=imageFormatGreen;
|
|
this->imageBlue=nullptr;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=0;
|
|
this->xmax=width-1;
|
|
this->ymin=0;
|
|
this->ymax=height-1;
|
|
|
|
replot();
|
|
}
|
|
void JKQTFPRGBImageOverlayPlot::setImage(void* imageRed, JKQTFPImageFormat imageFormatRed, void* imageGreen, JKQTFPImageFormat imageFormatGreen, void* imageBlue, JKQTFPImageFormat imageFormatBlue, int width, int height) {
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormatRed;
|
|
this->imageGreen=imageGreen;
|
|
this->imageFormatGreen=imageFormatGreen;
|
|
this->imageBlue=imageBlue;
|
|
this->imageFormatBlue=imageFormatBlue;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=0;
|
|
this->xmax=width-1;
|
|
this->ymin=0;
|
|
this->ymax=height-1;
|
|
|
|
replot();
|
|
}
|
|
|
|
|
|
void JKQTFPRGBImageOverlayPlot::setImage(void* imageRed, JKQTFPImageFormat imageFormatRed, int width, int height, double xmin, double xmax, double ymin, double ymax){
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormatRed;
|
|
this->imageRed=nullptr;
|
|
this->imageBlue=nullptr;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->ymin=ymin;
|
|
this->ymax=ymax;
|
|
|
|
replot();
|
|
}
|
|
void JKQTFPRGBImageOverlayPlot::setImage(void* imageRed, JKQTFPImageFormat imageFormatRed, void* imageGreen, JKQTFPImageFormat imageFormatGreen, int width, int height, double xmin, double xmax, double ymin, double ymax) {
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormatRed;
|
|
this->imageGreen=imageGreen;
|
|
this->imageFormatGreen=imageFormatGreen;
|
|
this->imageBlue=nullptr;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->ymin=ymin;
|
|
this->ymax=ymax;
|
|
|
|
replot();
|
|
}
|
|
void JKQTFPRGBImageOverlayPlot::setImage(void* imageRed, JKQTFPImageFormat imageFormatRed, void* imageGreen, JKQTFPImageFormat imageFormatGreen, void* imageBlue, JKQTFPImageFormat imageFormatBlue, int width, int height, double xmin, double xmax, double ymin, double ymax) {
|
|
this->imageRed=imageRed;
|
|
this->imageFormatRed=imageFormatRed;
|
|
this->imageGreen=imageGreen;
|
|
this->imageFormatGreen=imageFormatGreen;
|
|
this->imageBlue=imageBlue;
|
|
this->imageFormatBlue=imageFormatBlue;
|
|
this->width=width;
|
|
this->height=height;
|
|
this->xmin=xmin;
|
|
this->xmax=xmax;
|
|
this->ymin=ymin;
|
|
this->ymax=ymax;
|
|
|
|
replot();
|
|
}
|
|
|
|
JKQTFPYRangePlot::JKQTFPYRangePlot(JKQTFastPlotter *parent, double xmin, double xmax, QColor color, Qt::PenStyle style, double width, Qt::BrushStyle fillStyle):
|
|
JKQTFPPlot(parent)
|
|
{
|
|
this->ymin=xmin;
|
|
this->ymax=xmax;
|
|
this->centerline=(xmax+xmin)/2.0;
|
|
this->showCenterline=false;
|
|
this->color=color;
|
|
this->style=style;
|
|
this->width=width;
|
|
this->fillColor=color;
|
|
this->fillColor.setAlphaF(0.5);
|
|
this->fillStyle=fillStyle;
|
|
}
|
|
|
|
void JKQTFPYRangePlot::drawGraph(QPainter& painter) {
|
|
QRectF r(QPointF(parent->getInternalPlotBorderLeft(), parent->y2p(ymin)), QPointF(parent->getInternalPlotBorderLeft()+parent->getPlotWidth(), parent->y2p(ymax)));
|
|
QBrush b(fillStyle);
|
|
b.setColor(fillColor);
|
|
QPen p(color);
|
|
p.setStyle(style);
|
|
p.setWidthF(qMax(JKQTFastPlotter::ABS_MIN_LINEWIDTH, width));
|
|
painter.setPen(p);
|
|
painter.fillRect(r, b);
|
|
if (showCenterline) {
|
|
painter.drawLine(parent->getInternalPlotBorderLeft(), parent->y2p(centerline), parent->getInternalPlotBorderLeft()+parent->getPlotWidth(), parent->y2p(centerline));
|
|
}
|
|
painter.drawRect(r);
|
|
}
|
|
|