JKQtPlotter/lib/jkqtplotter/jkqtplotter.cpp
Snolandia a275fb73c3
Couple bug fixes (#147)
* BugFix

* Resolve several issues

1: Allow mouse scroll events to pass through to parent if the plot has not mouse scroll events to do. i.e. if scroll to zoom is disabled, and the graph is part of a scroll area, this allows scrolling the scroll area while the mouse is over the plot.

2: Fixed bug with resize timer resetting itself every time, rather than actually performing the countdown and calling the function.

3. Fixed bug with margins being calculated incorrectly, preventing the complete removal of margins from a plot.
2025-01-15 21:21:39 +01:00

2007 lines
89 KiB
C++

/*
Copyright (c) 2008-2024 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 <QFileInfo>
#if (QT_VERSION >= QT_VERSION_CHECK(5, 0, 0))
#include <QtGlobal>
#include <QtWidgets>
#else
#include <QtGui>
#endif
#include "jkqtplotter/jkqtplotter.h"
#include "jkqtplotter.h"
std::atomic<int> JKQTPlotter::jkqtp_RESIZE_DELAY(100);
void JKQTPlotter::setGlobalResizeDelay(int delayMS)
{
jkqtp_RESIZE_DELAY=delayMS;
}
int JKQTPlotter::getGlobalResizeDelay()
{
return jkqtp_RESIZE_DELAY;
}
/**************************************************************************************************************************
* JKQTPlotter
**************************************************************************************************************************/
JKQTPlotter::JKQTPlotter(bool datastore_internal, QWidget* parent, JKQTPDatastore* datast):
QWidget(parent, Qt::Widget),
currentMouseDragAction(),
lastWheelActionType(WheelActionType::None),
lastWheelActionTimestamp(0),
maxWheelEventSeriesTimestampDifference(300),
doDrawing(false),
plotter(nullptr),
mouseDragingRectangle(false),
mouseDragRectXStart(0), mouseDragRectXStartPixel(0), mouseDragRectXEndPixel(0),
mouseDragRectYEndPixel(0), mouseDragRectXEnd(0), mouseDragRectYStart(0),
mouseDragRectYStartPixel(0), mouseDragRectYEnd(0),
mouseDragMarkers(),
image(), oldImage(),
menuSpecialContextMenu(nullptr),toolbar(nullptr), masterPlotterX(nullptr), masterPlotterY(nullptr),
mousePosX(0), mousePosY(0),
magnification(1),
minSize(),
contextMenu(nullptr),
contextMenuMode(JKQTPContextMenuModes::jkqtpcmmStandardContextMenu),
mouseContextX(0), mouseContextY(0), mouseLastClickX(0), mouseLastClickY(0),
contextSubMenus(),
plotterStyle(JKQTPGetSystemDefaultStyle()),
resizeTimer(), registeredOverrideMouseDragActionModes(),
actgrpMouseLeft(nullptr), actMouseLeftAsDefault(nullptr), actMouseLeftAsRuler(nullptr), actMouseMoveToolTip(nullptr), actMouseLeftAsZoomRect(nullptr), actMouseLeftAsPanView(nullptr)
{
initJKQTPlotterResources();
setParent(parent);
connect(&resizeTimer, SIGNAL(timeout()), this, SLOT(delayedResizeEvent()));
plotter=new JKQTBasePlotter(datastore_internal, this, datast);
plotter->setEmittingSignalsEnabled(false);
fixBasePlotterSettings();
connect(plotter, SIGNAL(plotUpdated()), this, SLOT(redrawPlot()));
connect(plotter, SIGNAL(beforePlotScalingRecalculate()), this, SLOT(intBeforePlotScalingRecalculate()));
connect(plotter, SIGNAL(zoomChangedLocally(double, double, double, double, JKQTBasePlotter*)), this, SLOT(pzoomChangedLocally(double, double, double, double, JKQTBasePlotter*)));
image=createImageBuffer();
oldImage=image;
// enable mouse-tracking, so mouseMoved-Events can be caught
setMouseTracking(true);
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
actgrpMouseLeft=new QActionGroup(this);
actMouseLeftAsDefault=actgrpMouseLeft->addAction(QIcon(":/JKQTPlotter/jkqtp_mouseact_default.png"), tr("Default Tool"));
actMouseLeftAsDefault->setToolTip(tr("switch back to the default mouse action/tool (left button, no modifiers)"));
actMouseLeftAsDefault->setCheckable(true);
actMouseLeftAsPanView=actgrpMouseLeft->addAction(QIcon(":/JKQTPlotter/jkqtp_mouseact_drag.png"), tr("Pan View Tool"));
actMouseLeftAsPanView->setToolTip(tr("switch on the pan view tool (left button, no modifiers)"));
actMouseLeftAsPanView->setCheckable(true);
actMouseLeftAsZoomRect=actgrpMouseLeft->addAction(QIcon(":/JKQTPlotter/jkqtp_mouseact_zoomrect.png"), tr("Zoom Rectangle Tool"));
actMouseLeftAsZoomRect->setToolTip(tr("switch on the zoom rectangle tool (left button, no modifiers)"));
actMouseLeftAsZoomRect->setCheckable(true);
actMouseLeftAsRuler=actgrpMouseLeft->addAction(QIcon(":/JKQTPlotter/jkqtp_mouseact_ruler.png"), tr("Ruler Tool"));
actMouseLeftAsRuler->setToolTip(tr("switch on the ruler tool (left button, no modifiers)"));
actMouseLeftAsRuler->setCheckable(true);
//actMouseLeftAsToolTip=actgrpMouseLeft->addAction(QIcon(":/JKQTPlotter/jkqtp_mouseact_tooltip.png"), tr("Data Tooltip Tool"));
//actMouseLeftAsToolTip->setToolTip(tr("switch on the data tooltip tool when moving the mouse (no modifiers)"));
//actMouseLeftAsToolTip->setCheckable(true);
actgrpMouseLeft->setExclusive(true);
actMouseLeftAsDefault->setChecked(true);
actMouseMoveToolTip=new QAction(QIcon(":/JKQTPlotter/jkqtp_mouseact_tooltip.png"), tr("Data Tooltip Tool"));
actMouseMoveToolTip->setToolTip(tr("switch on the data tooltip tool when moving the mouse (no modifiers)"));
actMouseMoveToolTip->setCheckable(true);
actMouseMoveToolTip->setChecked(false);
connect(actMouseLeftAsDefault, SIGNAL(triggered()), this, SLOT(resetMouseLeftAction()));
connect(actMouseLeftAsRuler, SIGNAL(triggered()), this, SLOT(setMouseLeftActionAsRuler()));
connect(actMouseLeftAsPanView, SIGNAL(triggered()), this, SLOT(setMouseLeftActionAsPanView()));
connect(actMouseLeftAsZoomRect, SIGNAL(triggered()), this, SLOT(setMouseLeftActionAsZoomRect()));
connect(actMouseMoveToolTip, SIGNAL(toggled(bool)), this, SLOT(setMouseMoveActionAsToolTip(bool)));
toolbar=new JKVanishQToolBar(this);
toolbar->clear();
toolbar->move(1,1);
toolbar->hide();
toolbar->setAutoFillBackground(true);
toolbar->addSeparator();
toolbar->addSeparator();
populateToolbar(toolbar);
QSize s=QSize(plotterStyle.toolbarIconSize, plotterStyle.toolbarIconSize);
toolbar->setIconSize(s);
contextMenu=nullptr;//new QMenu(this);
//move(32,32);
resize(400,300);
doDrawing=true;
plotter->setEmittingSignalsEnabled(true);
redrawPlot();
}
JKQTPlotter::JKQTPlotter(JKQTPDatastore *datast, QWidget *parent):
JKQTPlotter(false, parent, datast)
{
}
JKQTPlotter::JKQTPlotter(QWidget *parent):
JKQTPlotter(true, parent, nullptr)
{
}
JKQTPlotter::~JKQTPlotter() {
resetContextMenu(false);
disconnect(plotter, SIGNAL(plotUpdated()), this, SLOT(redrawPlot()));
disconnect(plotter, SIGNAL(beforePlotScalingRecalculate()), this, SLOT(intBeforePlotScalingRecalculate()));
disconnect(plotter, SIGNAL(zoomChangedLocally(double, double, double, double, JKQTBasePlotter*)), this, SLOT(pzoomChangedLocally(double, double, double, double, JKQTBasePlotter*)));
delete plotter;
}
void JKQTPlotter::fixBasePlotterSettings()
{
if (plotterStyle.usePaletteColors) {
plotter->setBackgroundColor(palette().color(QPalette::Window));
}
}
JKQTPlotter::InternalBufferImageType JKQTPlotter::createImageBuffer() const
{
float scale=1.0;
auto img=InternalBufferImageType(getImageBufferSize(&scale));
img.setDevicePixelRatio(scale);
return img;
}
QSize JKQTPlotter::getImageBufferSize(float* scale_out) const
{
float imagewidth=width();
float imageheight=height()-getPlotYOffset();
/*
* The following code is to avoid low quality blur pixmap created via
* QPainter, sample code is found in QAbstractItemViewPrivate::renderToPixmap method
*/
qreal scale = 1.0;
const QWidget *window = this->window();
if (window) {
const QWindow *windowHandle = window->windowHandle();
if (windowHandle) {
scale = windowHandle->devicePixelRatio();
}
}
//qDebug()<<"scale="<<scale<<", dprF="<<devicePixelRatioF()<<", dpr="<<devicePixelRatio()<<" widgetsize="<<QSize(width(),height())<<" img:deviceindependensize="<<QSize(imagewidth,imageheight)<<" img.size.raw="<<QSize(width()*scale, height()*scale);
if (scale_out) *scale_out=scale;
return QSize(imagewidth*scale, imageheight*scale);
}
void JKQTPlotter::updateToolbarActions()
{
toolbar->clear();
toolbar->setAutoFillBackground(true);
toolbar->addSeparator();
toolbar->addSeparator();
populateToolbar(toolbar);
}
void JKQTPlotter::setToolbarIconSize(int value) {
plotterStyle.toolbarIconSize=value;
QSize s=QSize(plotterStyle.toolbarIconSize, plotterStyle.toolbarIconSize);
toolbar->setIconSize(s);
}
int JKQTPlotter::getToolbarIconSize() const {
return plotterStyle.toolbarIconSize;
}
void JKQTPlotter::setToolbarEnabled(bool __value)
{
if (plotterStyle.toolbarEnabled != __value) {
plotterStyle.toolbarEnabled = __value;
updateToolbar();
}
}
bool JKQTPlotter::isToolbarEnabled() const
{
return plotterStyle.toolbarEnabled;
}
void JKQTPlotter::setToolbarAlwaysOn(bool __value)
{
if (plotterStyle.toolbarAlwaysOn != __value) {
plotterStyle.toolbarAlwaysOn = __value;
updateToolbar();
}
}
bool JKQTPlotter::isToolbarAlwaysOn() const
{
return plotterStyle.toolbarAlwaysOn;
}
void JKQTPlotter::setMousePositionShown(bool __value)
{
plotterStyle.displayMousePosition = __value;
}
bool JKQTPlotter::isMousePositionShown() const
{
return plotterStyle.displayMousePosition;
}
void JKQTPlotter::setUserActionOverlayPen(const QPen &__value)
{
if (plotterStyle.userActionOverlayPen != __value) {
plotterStyle.userActionOverlayPen = __value;
update();
}
}
QPen JKQTPlotter::getUserActionOverlayPen() const
{
return plotterStyle.userActionOverlayPen;
}
void JKQTPlotter::setUserActionOverlayBrush(const QBrush &__value)
{
if (plotterStyle.userActionOverlayBrush != __value) {
plotterStyle.userActionOverlayBrush = __value;
update();
}
}
QBrush JKQTPlotter::getUserActionOverlayBrush() const
{
return plotterStyle.userActionOverlayBrush;
}
void JKQTPlotter::setUserActionOpaquePen(const QPen &__value)
{
if (plotterStyle.userActionOpaquePen != __value) {
plotterStyle.userActionOpaquePen = __value;
update();
}
}
QPen JKQTPlotter::getUserActionOpaquePen() const
{
return plotterStyle.userActionOpaquePen;
}
void JKQTPlotter::setUserActionOpaqueBrush(const QBrush &__value)
{
if (plotterStyle.userActionOpaqueBrush != __value) {
plotterStyle.userActionOpaqueBrush = __value;
update();
}
}
QBrush JKQTPlotter::getUserActionOpaqueBrush() const
{
return plotterStyle.userActionOpaqueBrush;
}
void JKQTPlotter::setUserActionMarkerPen(const QPen &__value)
{
if (plotterStyle.userActionMarkerPen != __value) {
plotterStyle.userActionMarkerPen = __value;
update();
}
}
QPen JKQTPlotter::getUserActionMarkerPen() const
{
return plotterStyle.userActionMarkerPen;
}
void JKQTPlotter::setUserActionMarkerBrush(const QBrush &__value)
{
if (plotterStyle.userActionMarkerBrush != __value) {
plotterStyle.userActionMarkerBrush = __value;
update();
}
}
QBrush JKQTPlotter::getUserActionMarkerBrush() const
{
return plotterStyle.userActionMarkerBrush;
}
void JKQTPlotter::setMaxTooltipDistance(int v)
{
plotterStyle.maxTooltipDistance=v;
}
int JKQTPlotter::getMaxTooltipDistance() const
{
return plotterStyle.maxTooltipDistance;
}
void JKQTPlotter::setUserActionCatchSensitivity(int v)
{
plotterStyle.userActionCatchSensitivity=v;
}
int JKQTPlotter::getUserActionCatchSensitivity() const
{
return plotterStyle.userActionCatchSensitivity;
}
void JKQTPlotter::setUserActionMarkerDiameter(int v)
{
plotterStyle.userActionMarkerDiameter=v;
}
void JKQTPlotter::setUserActionMarkerType(JKQTPUserActionMarkerType v)
{
plotterStyle.userActionMarkerType=v;
}
int JKQTPlotter::getUserActionMarkerDiameter() const
{
return plotterStyle.userActionMarkerDiameter;
}
JKQTPUserActionMarkerType JKQTPlotter::getUserActionMarkerType() const
{
return plotterStyle.userActionMarkerType;
}
void JKQTPlotter::registerMouseWheelAction(Qt::KeyboardModifiers modifier, JKQTPMouseWheelActions action)
{
//qDebug()<<"registerMouseWheelAction("<<modifier<<","<<action<<"): "<<plotterStyle.registeredMouseWheelActions;
plotterStyle.registeredMouseWheelActions[modifier]=action;
//qDebug()<<"registerMouseWheelAction("<<modifier<<","<<action<<"): "<<plotterStyle.registeredMouseWheelActions;
}
void JKQTPlotter::deregisterMouseWheelAction(Qt::KeyboardModifiers modifier)
{
//qDebug()<<"deregisterMouseWheelAction("<<modifier<<"): "<<plotterStyle.registeredMouseWheelActions;
plotterStyle.registeredMouseWheelActions.remove(modifier);
//qDebug()<<"deregisterMouseWheelAction("<<modifier<<"): "<<plotterStyle.registeredMouseWheelActions;
}
void JKQTPlotter::clearAllMouseWheelActions()
{
//qDebug()<<"clearAllMouseWheelActions(): "<<plotterStyle.registeredMouseWheelActions;
plotterStyle.registeredMouseWheelActions.clear();
//qDebug()<<"clearAllMouseWheelActions(): "<<plotterStyle.registeredMouseWheelActions;
}
void JKQTPlotter::registerMouseMoveAction(Qt::KeyboardModifiers modifier, JKQTPMouseMoveActions action)
{
//qDebug()<<"registerMouseMoveAction("<<modifier<<","<<action<<"): "<<plotterStyle.registeredMouseMoveActions;
plotterStyle.registeredMouseMoveActions[modifier]=action;
//qDebug()<<"registerMouseMoveAction("<<modifier<<","<<action<<"): "<<plotterStyle.registeredMouseMoveActions;
}
void JKQTPlotter::deregisterMouseMoveAction(Qt::KeyboardModifiers modifier)
{
//qDebug()<<"deregisterMouseMoveAction("<<modifier<<"): "<<plotterStyle.registeredMouseMoveActions;
plotterStyle.registeredMouseMoveActions.remove(modifier);
//qDebug()<<"deregisterMouseMoveAction("<<modifier<<"): "<<plotterStyle.registeredMouseMoveActions;
}
void JKQTPlotter::clearAllMouseMoveActions()
{
//qDebug()<<"clearAllMouseMoveActions(): "<<plotterStyle.registeredMouseMoveActions;
plotterStyle.registeredMouseMoveActions.clear();
//qDebug()<<"clearAllMouseMoveActions(): "<<plotterStyle.registeredMouseMoveActions;
}
void JKQTPlotter::loadSettings(const QSettings& settings, const QString& group) {
plotter->loadSettings(settings, group);
plotterStyle.loadSettings(settings, group, JKQTPGetSystemDefaultStyle());
updateToolbar();
redrawPlot();
}
void JKQTPlotter::saveSettings(QSettings& settings, const QString& group) const {
plotterStyle.saveSettings(settings, group);
plotter->saveSettings(settings, group);
}
void JKQTPlotter::fillInternalStructForToolTipOfClosestDataPoint(double x0, double y0, bool emitEvent) {
bool foundAll=false;
QList<QString> labels, titles;
QList<QColor> colors;
QList<QPointF> posSyss;
QList<QImage> keymarkers;
QList<JKQTPPlotElement*> graphs;
QList<double> ds;
double dbest=0;
for (size_t i=0; i<getPlotter()->getGraphCount(); i++) {
JKQTPPlotElement* g=getPlotter()->getGraph(i);
bool found=false;
QString label;
QPointF posSys;
double d=0;
if (g && g->isVisible()) {
QPointF pp;
QString ll;
double dist=g->hitTest(QPointF(x0, y0), &pp, &ll, JKQTPPlotElement::HitTestXY);
if (JKQTPIsOKFloat(dist)) {
if (!found || dist<d) {
d=dist;
posSys=pp;
label=ll;
found=true;
}
}
}
if (found && g) {
if (!foundAll) {
dbest=d;
labels.clear();
labels<<label;
titles.clear();
titles<<g->getTitle();
colors.clear();
colors<<g->getKeyLabelColor();
posSyss.clear();
posSyss<<posSys;
ds.clear();
ds<<d;
keymarkers.clear();
keymarkers<<g->generateKeyMarker();
graphs.clear();
graphs<<g;
} else {
if (fabs(dbest-d)<=plotterStyle.userActionCatchSensitivity) {
labels<<label;
titles<<g->getTitle();
colors<<g->getKeyLabelColor();
posSyss<<posSys;
ds<<d;
keymarkers<<g->generateKeyMarker();
graphs<<g;
} else {
if (d<dbest) {
labels.clear();
labels<<label;
titles.clear();
titles<<g->getTitle();
colors.clear();
colors<<g->getKeyLabelColor();
posSyss.clear();
posSyss<<posSys;
ds.clear();
ds<<d;
keymarkers.clear();
keymarkers<<g->generateKeyMarker();
graphs.clear();
graphs<<g;
}
}
if (d<dbest) dbest=d;
}
foundAll=true;
}
}
mouseDragMarkers.clear();
if (foundAll && dbest<getCurrentPlotterStyle().maxTooltipDistance) {
QStringList entries;
for (int i=0; i<posSyss.size(); i++) {
const QPoint posPix=QPointF(getXAxis()->x2p(posSyss[i].x())*magnification, getYAxis()->x2p(posSyss[i].y())*magnification).toPoint();
entries<<(titles[i]+": "+labels[i]);
mouseDragMarkers<<MouseDragMarker(posPix, labels[i], titles[i], colors[i], keymarkers[i], graphs[i]);
}
//const QPoint globalPosPix=mapToGlobal(posPix);
//const int rectsize=24;
//qDebug()<<graph->getTitle()<<" clicked("<<event->pos()<<") ==> posSys="<<posSys<<" / posPix="<<posPix<<" globalPosPix="<<globalPosPix<<", label="<<label<<", distance="<<d;
//QToolTip::showText(globalPosPix, graph->getTitle()+":\n"+label, this);//, QRect(posPix.x()-rectsize/2, posPix.y()-rectsize/2, rectsize, rectsize));
if (emitEvent) {
if (mouseDragMarkers.size()>0) {
emit tooltipDisplayed(posSyss[0].x(), posSyss[0].y(), entries, graphs);
}
}
}
}
void JKQTPlotter::paintUserAction() {
if ((currentMouseDragAction.isValid() && mouseDragingRectangle) || (!currentMouseMoveAction.isEmpty())) {
image=oldImage;
if (image.width()>0 && image.height()>0 && !image.isNull()) {
JKQTPEnhancedPainter painter(&image);
#if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
painter.setRenderHint(JKQTPEnhancedPainter::NonCosmeticDefaultPen, true);
#endif
painter.setRenderHint(JKQTPEnhancedPainter::Antialiasing, true);
painter.setRenderHint(JKQTPEnhancedPainter::TextAntialiasing, true);
painter.setPen(plotterStyle.userActionOverlayPen);
if (jkqtp_approximatelyUnequal(mouseDragRectXEnd,mouseDragRectXStart) && jkqtp_approximatelyUnequal(mouseDragRectYEnd,mouseDragRectYStart)) {
const double x1=plotter->x2p(mouseDragRectXStart)*magnification;
const double y1=plotter->y2p(mouseDragRectYStart)*magnification;
const double x2=plotter->x2p(mouseDragRectXEnd)*magnification;
const double y2=plotter->y2p(mouseDragRectYEnd)*magnification;
const double dx=x2-x1;
const double dy=y2-y1;
if (currentMouseDragAction.isValid() && mouseDragingRectangle) {
if (currentMouseDragAction.mode==jkqtpmdaDrawRectangleForEvent) {
painter.fillRect(QRectF(x1, y1, x2-x1, y2-y1), plotterStyle.userActionOverlayBrush);
painter.setPen(plotterStyle.userActionOverlayPen);
painter.drawRect(QRectF(x1, y1, x2-x1, y2-y1));
} else if (currentMouseDragAction.mode==jkqtpmdaZoomByRectangle) {
double xmin=mouseDragRectXStart;
double xmax=mouseDragRectXEnd;
double ymin=mouseDragRectYStart;
double ymax=mouseDragRectYEnd;
plotter->correctXYRangeForAspectRatio(xmin,xmax,ymin,ymax);
const double xz1=plotter->x2p(xmin)*magnification;
const double yz1=plotter->y2p(ymin)*magnification;
const double xz2=plotter->x2p(xmax)*magnification;
const double yz2=plotter->y2p(ymax)*magnification;
painter.fillRect(QRectF(xz1, yz1, xz2-xz1, yz2-yz1), plotterStyle.userActionOverlayBrush);
painter.setPen(plotterStyle.userActionOverlayPen);
painter.drawRect(QRectF(xz1, yz1, xz2-xz1, yz2-yz1));
} else if (currentMouseDragAction.mode==jkqtpmdaDrawCircleForEvent) {
painter.setPen(plotterStyle.userActionOverlayPen);
painter.setBrush(plotterStyle.userActionOverlayBrush);
painter.drawEllipse(QPointF(x1, y1), qMin(fabs(dx), fabs(dy)), qMin(fabs(dx), fabs(dy)));
} else if (currentMouseDragAction.mode==jkqtpmdaDrawEllipseForEvent) {
painter.setPen(plotterStyle.userActionOverlayPen);
painter.setBrush(plotterStyle.userActionOverlayBrush);
painter.drawEllipse(QPointF(x1, y1), fabs(dx), fabs(dy));
} else if (currentMouseDragAction.mode==jkqtpmdaDrawLineForEvent) {
painter.setPen(plotterStyle.userActionOverlayPen);
painter.setBrush(plotterStyle.userActionOverlayBrush);
painter.drawLine(QPointF(x1,y1), QPointF(x2,y2));
} else if (currentMouseDragAction.mode==jkqtpmdaRuler) {
painter.setPen(plotterStyle.userActionOpaquePen);
painter.setBrush(plotterStyle.userActionOpaqueBrush);
painter.drawLine(QPointF(x1,y1), QPointF(x2,y1));
painter.drawLine(QPointF(x2,y1), QPointF(x2,y2));
painter.drawLine(QPointF(x1,y1), QPointF(x2,y2));
const double dxy=sqrt(jkqtp_sqr(mouseDragRectXEnd-mouseDragRectXStart)+jkqtp_sqr(mouseDragRectYEnd-mouseDragRectYStart));
const double alpha=atan2((mouseDragRectYEnd-mouseDragRectYStart), (mouseDragRectXEnd-mouseDragRectXStart))/JKQTPSTATISTICS_PI*180.0;
const double dx=fabs(mouseDragRectXEnd-mouseDragRectXStart);
const double dy=fabs(mouseDragRectYEnd-mouseDragRectYStart);
painter.setBrush(plotterStyle.userActionOpaqueBrush);
QString txt;
double a=0,d=0,so=0,w=0;
getPlotter()->getMathText()->setFontSize(plotterStyle.userActionFontSize);
getPlotter()->getMathText()->setFontSpecial(plotterStyle.userActionFontName);
txt=QString::fromStdString("\\delta_{x}="+jkqtp_floattolatexstr(dx, 3));
getPlotter()->getMathText()->parse(txt);
getPlotter()->getMathText()->getSizeDetail(painter, w, a, d, so);
if (y1>y2) {
QRectF rec((x1+x2)/2.0-w/2.0, y1+2, w, a+d);
painter.fillRect(rec, plotterStyle.userActionOpaqueBrush);
getPlotter()->getMathText()->draw(painter, Qt::AlignTop, rec, getPlotter()->getCurrentPlotterStyle().debugShowTextBoxes);
} else {
QRectF rec((x1+x2)/2.0-w/2.0, y1-2-a-d, w, a+d);
painter.fillRect(rec, plotterStyle.userActionOpaqueBrush);
getPlotter()->getMathText()->draw(painter, Qt::AlignBottom, rec, getPlotter()->getCurrentPlotterStyle().debugShowTextBoxes);
}
txt=jkqtp_floattolatexqstr(dy, 3);
getPlotter()->getMathText()->parse("\\delta_{y}="+txt);
getPlotter()->getMathText()->getSizeDetail(painter, w, a, d, so);
//double dyh=a+d;
if (x2>x1) {
QRectF rec(x2+2, (y1+y2)/2.0-(a+d)/2.0, w, a+d);
painter.fillRect(rec, plotterStyle.userActionOpaqueBrush);
getPlotter()->getMathText()->draw(painter, Qt::AlignVCenter|Qt::AlignLeft, rec, getPlotter()->getCurrentPlotterStyle().debugShowTextBoxes);
} else {
QRectF rec(x2-2-w, (y1+y2)/2.0-(a+d)/2.0, w, a+d);
painter.fillRect(rec, plotterStyle.userActionOpaqueBrush);
getPlotter()->getMathText()->draw(painter, Qt::AlignVCenter|Qt::AlignRight, rec, getPlotter()->getCurrentPlotterStyle().debugShowTextBoxes);
}
txt=QString::fromStdString("\\delta_{x,y}="+jkqtp_floattolatexstr(dxy)+", \\alpha="+jkqtp_floattolatexstr(alpha, 1)+"\\degree, \\stfrac{\\mathrm{d}y}{\\mathrm{d}x}="+jkqtp_floattolatexstr(dy/dx, 1));
getPlotter()->getMathText()->parse(txt);
getPlotter()->getMathText()->getSizeDetail(painter, w, a, d, so);
if (x2<x1) {
QRectF rec((x1+x2)/2.0, (y1+y2)/2.0-(a+d)/2.0, w, a+d);
painter.fillRect(rec, plotterStyle.userActionOpaqueBrush);
getPlotter()->getMathText()->draw(painter,Qt::AlignTop|Qt::AlignLeft, rec, getPlotter()->getCurrentPlotterStyle().debugShowTextBoxes);
} else {
QRectF rec((x1+x2)/2.0-w, (y1+y2)/2.0-(a+d)/2.0, w, a+d);
painter.fillRect(rec, plotterStyle.userActionOpaqueBrush);
getPlotter()->getMathText()->draw(painter,Qt::AlignTop|Qt::AlignRight, rec, getPlotter()->getCurrentPlotterStyle().debugShowTextBoxes);
}
}
}
if ((currentMouseDragAction.isValid() && currentMouseDragAction.mode==jkqtpmdaToolTipForClosestDataPoint) || currentMouseMoveAction.contains(jkqtpmmaToolTipForClosestDataPoint)) {
if (mouseDragMarkers.size()>0) {
const int markerD=getCurrentPlotterStyle().userActionMarkerDiameter;
if (getCurrentPlotterStyle().userActionMarkerType==jkqtpuamtCircle) {
painter.setBrush(plotterStyle.userActionMarkerBrush);
painter.setPen(plotterStyle.userActionMarkerPen);
painter.drawEllipse(QRectF(mouseDragMarkers[0].pos.x()-markerD/2, mouseDragMarkers[0].pos.y()-markerD/2, markerD, markerD));
} else if (getCurrentPlotterStyle().userActionMarkerType==jkqtpuamtCrossHair) {
painter.setBrush(plotterStyle.userActionMarkerBrush);
painter.setPen(plotterStyle.userActionMarkerPen);
painter.drawLine(QLineF(plotter->getInternalPlotBorderLeft(), mouseDragMarkers[0].pos.y(), image.width()-plotter->getInternalPlotBorderRight(), mouseDragMarkers[0].pos.y()));
painter.drawLine(QLineF(mouseDragMarkers[0].pos.x(), plotter->getInternalPlotBorderTop(), mouseDragMarkers[0].pos.x(), image.height()-plotter->getInternalPlotBorderBottom()));
} else if (getCurrentPlotterStyle().userActionMarkerType==jkqtpuamtCircleAndCrossHair) {
painter.setBrush(plotterStyle.userActionMarkerBrush);
painter.setPen(plotterStyle.userActionMarkerPen);
painter.drawEllipse(QRectF(mouseDragMarkers[0].pos.x()-markerD/2, mouseDragMarkers[0].pos.y()-markerD/2, markerD, markerD));
painter.drawLine(QLineF(plotter->getInternalPlotBorderLeft(), mouseDragMarkers[0].pos.y(), mouseDragMarkers[0].pos.x()-markerD/2, mouseDragMarkers[0].pos.y()));
painter.drawLine(QLineF(image.width()-plotter->getInternalPlotBorderRight(), mouseDragMarkers[0].pos.y(), mouseDragMarkers[0].pos.x()+markerD/2, mouseDragMarkers[0].pos.y()));
painter.drawLine(QLineF(mouseDragMarkers[0].pos.x(), plotter->getInternalPlotBorderTop(), mouseDragMarkers[0].pos.x(), mouseDragMarkers[0].pos.y()-markerD/2));
painter.drawLine(QLineF(mouseDragMarkers[0].pos.x(), image.height()-plotter->getInternalPlotBorderBottom(), mouseDragMarkers[0].pos.x(), mouseDragMarkers[0].pos.y()+markerD/2));
}
const int xoffset=qMax<int>(markerD*3/2,16);
const int x0=mouseDragMarkers[0].pos.x()+xoffset;
QRectF rall(0,0,0,0);
double y=mouseDragMarkers[0].pos.y();
QVector<QRectF> rects;
QVector<QString> txts;
QVector<QColor> colors;
const int txtoffset=4;
double ascent=0,descent=0,strikeout=0,width=0;
getPlotter()->getMathText()->setFontSize(plotterStyle.userActionFontSize);
getPlotter()->getMathText()->setFontSpecial(plotterStyle.userActionFontName);
for (auto& m: mouseDragMarkers) {
QString txt="\\textcolor{"+jkqtp_QColor2String(m.color)+"}{"+m.title+": "+m.label+"}";
getPlotter()->getMathText()->parse(txt);
getPlotter()->getMathText()->getSizeDetail(painter, width, ascent, descent, strikeout);
double txtheight=ascent+descent;
QRectF rec(x0+txtoffset, y, width+1, txtheight+1);
if (rall.width()*rall.height()<=0) {
rall=rec.toRect();
} else {
rall.setBottom(rall.bottom()+txtheight+ascent/4);
rall.setRight(qMax(rall.right(), rec.right()));
}
rects << rec;
txts<<txt;
colors<<m.color;
//qDebug()<<y<<txt<<rall;
y=y+txtheight+ascent/4;
}
rall.setLeft(rall.left()-txtoffset);
rall.setRight(rall.right()+txtoffset);
rall.setTop(rall.top()-txtoffset);
rall.setBottom(rall.bottom()+txtoffset);
if (rall.right()>image.width()) {
rall.moveLeft(mouseDragMarkers[0].pos.x()-rall.width()-xoffset);
for (auto& r: rects) {
r.moveLeft(rall.left()+txtoffset);
}
}
if (rall.bottom()>image.height()) {
double oldY=rall.y();
rall.moveTop(image.height()-rall.height());
for (auto& r: rects) {
r.moveTop(r.y()-(oldY-rall.y()));
}
}
//qDebug()<<"rall="<<rall;
painter.setBrush(plotterStyle.userActionOpaqueBrush);
painter.setPen(plotterStyle.userActionOpaquePen);
JKQTPDrawTooltip(painter, mouseDragMarkers[0].pos.x(), mouseDragMarkers[0].pos.y(), rall);
for (int i=0; i<txts.size(); i++) {
getPlotter()->getMathText()->parse(txts[i]);
getPlotter()->getMathText()->draw(painter, Qt::AlignTop|Qt::AlignLeft, rects[i], getPlotter()->getCurrentPlotterStyle().debugShowTextBoxes);
QStringList el=getPlotter()->getMathText()->getErrorList();
if (el.size()>0) {
qDebug()<<"error parsing LaTeX string fo tooltip ("<<txts[i]<<"):\n - "<<el.join("\n - ");
// } else {
// qDebug()<<txts[i];
}
}
}
}
}
}
update();
}
}
void JKQTPlotter::correctZoomRectForPanning(QRectF &zoomRect) const
{
zoomRect=zoomRect.normalized();
// this code corrects for cases, where you pan over getAbsoluteXY().
// without this correction, the zoom rectangle size will be changed, instead of jus stopping at the absolute size
auto correctForAbsoluteLeft=[](double pos, double absPos) {
if (!JKQTPIsOKFloat(absPos)) return pos;
if (pos<absPos) return absPos;
return pos;
};
auto correctForAbsoluteRight=[](double pos, double absPos) {
if (!JKQTPIsOKFloat(absPos)) return pos;
if (pos>absPos) return absPos;
return pos;
};
const double absLeft=plotter->x2p((plotter->getXAxis()->getInverted())? plotter->getAbsoluteXMax() : plotter->getAbsoluteXMin());
const double absTop=plotter->y2p((plotter->getYAxis()->getInverted())? plotter->getAbsoluteYMin() : plotter->getAbsoluteYMax());
const double absRight=plotter->x2p((plotter->getXAxis()->getInverted())? plotter->getAbsoluteXMin() : plotter->getAbsoluteXMax());
const double absBottom=plotter->y2p((plotter->getYAxis()->getInverted())? plotter->getAbsoluteYMax() : plotter->getAbsoluteYMin());
if (JKQTPIsOKFloat(absLeft)) zoomRect.moveLeft(correctForAbsoluteLeft(zoomRect.left(), absLeft));
if (JKQTPIsOKFloat(absTop)) zoomRect.moveTop(correctForAbsoluteLeft(zoomRect.top(), absTop));
if (JKQTPIsOKFloat(absRight)) zoomRect.moveRight(correctForAbsoluteRight(zoomRect.right(), absRight));
if (JKQTPIsOKFloat(absBottom)) zoomRect.moveBottom(correctForAbsoluteRight(zoomRect.bottom(), absBottom));
}
void JKQTPlotter::mouseMoveEvent ( QMouseEvent * event ) {
if (plotterStyle.displayMousePosition) {
mousePosX=plotter->p2x(event->pos().x()/magnification);
mousePosY=plotter->p2y((event->pos().y()-getPlotYOffset())/magnification);
update();
}
if (plotterStyle.toolbarEnabled && (!plotterStyle.toolbarAlwaysOn) && (!toolbar->isVisible())) { // decide whether to display toolbar
int y1=10;
if (event->pos().y()/magnification>=0 && event->pos().y()/magnification<=y1) {
toolbar->show();
toolbar->move(1,1);
}
}
if (!plotterStyle.toolbarEnabled) {
toolbar->hide();
}
currentMouseMoveAction.clear();
if (currentMouseDragAction.isValid()) {
if (( (currentMouseDragAction.mode==jkqtpmdaZoomByRectangle) ||
(currentMouseDragAction.mode==jkqtpmdaDrawRectangleForEvent) ||
(currentMouseDragAction.mode==jkqtpmdaDrawCircleForEvent) ||
(currentMouseDragAction.mode==jkqtpmdaDrawEllipseForEvent) ||
(currentMouseDragAction.mode==jkqtpmdaScribbleForEvents) ||
(currentMouseDragAction.mode==jkqtpmdaToolTipForClosestDataPoint) ||
(currentMouseDragAction.mode==jkqtpmdaRuler) ||
(currentMouseDragAction.mode==jkqtpmdaPanPlotOnMove) ||
(currentMouseDragAction.mode==jkqtpmdaPanPlotOnRelease) ||
(currentMouseDragAction.mode==jkqtpmdaDrawLineForEvent) ) &&
mouseDragingRectangle)
{
if (currentMouseDragAction.mode==jkqtpmdaScribbleForEvents || currentMouseDragAction.mode==jkqtpmdaPanPlotOnMove) {
// start is last event position
mouseDragRectXStart=mouseDragRectXEnd;
mouseDragRectYStart=mouseDragRectYEnd;
mouseDragRectXStartPixel=mouseDragRectXEndPixel;
mouseDragRectYStartPixel=mouseDragRectYEndPixel;
}
mouseDragRectXEnd=plotter->p2x(event->pos().x()/magnification);
mouseDragRectYEnd=plotter->p2y((event->pos().y()-getPlotYOffset())/magnification);
mouseDragRectXEndPixel=event->pos().x();
mouseDragRectYEndPixel=event->pos().y();
if (currentMouseDragAction.mode==jkqtpmdaToolTipForClosestDataPoint) {
fillInternalStructForToolTipOfClosestDataPoint(mouseDragRectXEnd, mouseDragRectYEnd);
}
if (currentMouseDragAction.mode==jkqtpmdaRuler) {
emit rulerDisplayed(mouseDragRectXStart, mouseDragRectYStart, mouseDragRectXEnd, mouseDragRectYEnd, event->modifiers());
}
paintUserAction();
event->accept();
//std::cout<<mouseZoomingTStart<<" -- "<<mouseZoomingTEnd<<std::endl;
if (currentMouseDragAction.mode==jkqtpmdaZoomByRectangle) {
double xmin=mouseDragRectXStart;
double xmax=mouseDragRectXEnd;
double ymin=mouseDragRectYStart;
double ymax=mouseDragRectYEnd;
plotter->correctXYRangeForAspectRatio(xmin,xmax,ymin,ymax);
emit plotNewZoomRectangle(xmin,xmax,ymin,ymax, event->modifiers());
//emit plotNewZoomRectangle(mouseDragRectXStart, mouseDragRectXEnd, mouseDragRectYStart, mouseDragRectYEnd, event->modifiers());
}
if ((currentMouseDragAction.mode==jkqtpmdaScribbleForEvents) && (jkqtp_approximatelyUnequal(mouseDragRectXStart,mouseDragRectXEnd) || jkqtp_approximatelyUnequal(mouseDragRectYStart,mouseDragRectYEnd)) ) {
emit userScribbleClick(mouseDragRectXEnd, mouseDragRectYEnd, event->modifiers(), false, false);
}
if ((currentMouseDragAction.mode==jkqtpmdaPanPlotOnMove) && (jkqtp_approximatelyUnequal(mouseDragRectXStart,mouseDragRectXEnd) || jkqtp_approximatelyUnequal(mouseDragRectYStart,mouseDragRectYEnd)) ) {
QRectF zoomRect= QRectF(QPointF(plotter->x2p(getXAxis()->getMin()),plotter->y2p(getYAxis()->getMax())), QPointF(plotter->x2p(getXAxis()->getMax()),plotter->y2p(getYAxis()->getMin())));
if ( (mouseLastClickX/magnification<plotter->getInternalPlotBorderLeft()) || (mouseLastClickX/magnification>plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) ) {
zoomRect.translate(0, mouseDragRectYStartPixel-mouseDragRectYEndPixel);
} else if (((mouseLastClickY-getPlotYOffset())/magnification<plotter->getInternalPlotBorderTop()) || ((mouseLastClickY-getPlotYOffset())/magnification>plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
zoomRect.translate(mouseDragRectXStartPixel-mouseDragRectXEndPixel, 0);
} else {
zoomRect.translate(mouseDragRectXStartPixel-mouseDragRectXEndPixel, mouseDragRectYStartPixel-mouseDragRectYEndPixel);
}
correctZoomRectForPanning(zoomRect);
setXY(plotter->p2x(zoomRect.left()), plotter->p2x(zoomRect.right()), plotter->p2y(zoomRect.bottom()), plotter->p2y(zoomRect.top()), true);
}
} else {
event->accept();
/*if (emitSignals)*/ //emit plotMouseMove(x, y);
}
} else if (event->buttons()==Qt::NoButton) {
bool foundIT=false;
auto actionIT=findMatchingMouseMoveAction(event->modifiers(), &foundIT);
if (foundIT) {
// we found a matching action
currentMouseMoveAction.insert(actionIT.value());
if(actionIT.value()==jkqtpmmaToolTipForClosestDataPoint){
mouseDragRectXStart=plotter->p2x(event->pos().x()/magnification);
mouseDragRectYStart=plotter->p2y((event->pos().y()-getPlotYOffset())/magnification);
mouseDragRectXStartPixel=event->pos().x();
mouseDragRectYStartPixel=event->pos().y();
fillInternalStructForToolTipOfClosestDataPoint(plotter->p2x(event->pos().x()/magnification), plotter->p2y((event->pos().y()-getPlotYOffset())/magnification));
}
paintUserAction();
}
}
// emit move signal, if event occured inside plot only
if ( (event->pos().x()/magnification>=plotter->getInternalPlotBorderLeft()) && (event->pos().x()/magnification<=plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) &&
((event->pos().y()-getPlotYOffset())/magnification>=plotter->getInternalPlotBorderTop()) && ((event->pos().y()-getPlotYOffset())/magnification<=plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
emit plotMouseMove(plotter->p2x(event->pos().x()/magnification), plotter->p2y((event->pos().y()-getPlotYOffset())/magnification));
}
updateCursor();
}
void JKQTPlotter::mousePressEvent ( QMouseEvent * event ){
MouseDragAction prevAction = currentMouseDragAction;
currentMouseDragAction.clear();
mouseDragMarkers.clear();
bool foundIT=false;
auto actionIT=findMatchingMouseDragAction(event->button(), event->modifiers(), &foundIT);
if (foundIT) {
// we found a matching action
mouseLastClickX=event->pos().x();
mouseLastClickY=event->pos().y();
mouseDragRectXStart=plotter->p2x(event->pos().x()/magnification);
mouseDragRectYStart=plotter->p2y((event->pos().y()-getPlotYOffset())/magnification);
mouseDragRectXEndPixel=mouseDragRectXStartPixel=event->pos().x();
mouseDragRectYEndPixel=mouseDragRectYStartPixel=event->pos().y();
currentMouseDragAction=MouseDragAction(actionIT.key().first, actionIT.key().second, actionIT.value());
mouseDragingRectangle=true;
oldImage=image;
if (currentMouseDragAction.mode==jkqtpmdaScribbleForEvents) emit userScribbleClick(mouseDragRectXStart, mouseDragRectYStart, event->modifiers(), true, false);
event->accept();
} else if (event->button()==Qt::RightButton) {
if (prevAction.mode==jkqtpmdaScribbleForEvents){
mouseDragRectXStart=plotter->p2x(event->pos().x()/magnification);
mouseDragRectYStart=plotter->p2y((event->pos().y()-getPlotYOffset())/magnification);
emit userScribbleClick(mouseDragRectXStart, mouseDragRectYStart, event->modifiers(), false, true);
}
if(event->modifiers()==Qt::NoModifier && contextMenuMode!=jkqtpcmmNoContextMenu){
mouseLastClickX=event->pos().x();
mouseLastClickY=event->pos().y();
openContextMenu(event->pos().x(), event->pos().y());
event->accept();
}
}
// emit clicked signal, if event occured inside plot only
if ( (event->pos().x()/magnification>=plotter->getInternalPlotBorderLeft()) && (event->pos().x()/magnification<=plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) &&
((event->pos().y()-getPlotYOffset())/magnification>=plotter->getInternalPlotBorderTop()) && ((event->pos().y()-getPlotYOffset())/magnification<=plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
emit plotMouseClicked(plotter->p2x(event->pos().x()/magnification), plotter->p2y((event->pos().y()-getPlotYOffset())/magnification), event->modifiers(), event->button());
event->accept();
}
updateCursor();
if (foundIT && actionIT.value()==jkqtpmdaToolTipForClosestDataPoint) {
fillInternalStructForToolTipOfClosestDataPoint(mouseDragRectXStart, mouseDragRectYStart);
paintUserAction();
}
}
void JKQTPlotter::mouseReleaseEvent ( QMouseEvent * event ){
if ((event->flags()&Qt::MouseEventCreatedDoubleClick)==Qt::MouseEventCreatedDoubleClick) {
return;
}
if (currentMouseDragAction.isValid()) {
mouseDragRectXEnd=plotter->p2x(event->pos().x()/magnification);
mouseDragRectYEnd=plotter->p2y((event->pos().y()-getPlotYOffset())/magnification);
mouseDragRectXEndPixel=event->pos().x();
mouseDragRectYEndPixel=event->pos().y();
image=oldImage;
//update();
mouseDragingRectangle=false;
double x1=mouseDragRectXStart;
double y1=mouseDragRectYStart;
double x2=mouseDragRectXEnd;
double y2=mouseDragRectYEnd;
if (jkqtp_approximatelyUnequal(mouseDragRectXStart,mouseDragRectXEnd) && jkqtp_approximatelyUnequal(mouseDragRectYStart,mouseDragRectYEnd)) {
if (currentMouseDragAction.mode==jkqtpmdaZoomByRectangle) {
double xmin=mouseDragRectXStart;
double xmax=mouseDragRectXEnd;
double ymin=mouseDragRectYStart;
double ymax=mouseDragRectYEnd;
plotter->correctXYRangeForAspectRatio(xmin,xmax,ymin,ymax);
emit zoomChangedLocally(xmin, xmax, ymin, ymax, this);
plotter->setXY(xmin, xmax, ymin, ymax, true);
} else if (currentMouseDragAction.mode==jkqtpmdaPanPlotOnRelease) {
QRectF zoomRect= QRectF(QPointF(plotter->x2p(getXAxis()->getMin()),plotter->y2p(getYAxis()->getMax())), QPointF(plotter->x2p(getXAxis()->getMax()),plotter->y2p(getYAxis()->getMin())));
if ( (mouseLastClickX/magnification<plotter->getInternalPlotBorderLeft()) || (mouseLastClickX/magnification>plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) ) {
zoomRect.translate(0, mouseDragRectYStartPixel-mouseDragRectYEndPixel);
} else if (((mouseLastClickY-getPlotYOffset())/magnification<plotter->getInternalPlotBorderTop()) || ((mouseLastClickY-getPlotYOffset())/magnification>plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
zoomRect.translate(mouseDragRectXStartPixel-mouseDragRectXEndPixel, 0);
} else {
zoomRect.translate(mouseDragRectXStartPixel-mouseDragRectXEndPixel, mouseDragRectYStartPixel-mouseDragRectYEndPixel);
}
correctZoomRectForPanning(zoomRect);
setXY(plotter->p2x(zoomRect.left()), plotter->p2x(zoomRect.right()), plotter->p2y(zoomRect.bottom()), plotter->p2y(zoomRect.top()), true);
} else if (currentMouseDragAction.mode==jkqtpmdaDrawRectangleForEvent) {
emit userRectangleFinished(x1, y1, x2-x1, y2-y1, event->modifiers());
} else if (currentMouseDragAction.mode==jkqtpmdaDrawCircleForEvent) {
emit userCircleFinished(x1, y1, qMin(fabs(x2-x1), fabs(y2-y1)), event->modifiers());
} else if (currentMouseDragAction.mode==jkqtpmdaDrawEllipseForEvent) {
emit userEllipseFinished(x1, y1, fabs(x2-x1), fabs(y2-y1), event->modifiers());
} else if (currentMouseDragAction.mode==jkqtpmdaDrawLineForEvent) {
emit userLineFinished(x1, y1, x2, y2, event->modifiers());
}
}
if (currentMouseDragAction.mode!=jkqtpmdaZoomByRectangle && currentMouseDragAction.mode!=jkqtpmdaRuler && currentMouseDragAction.mode!=jkqtpmdaToolTipForClosestDataPoint) {
update();
}
if (currentMouseDragAction.mode==jkqtpmdaScribbleForEvents) {
emit userScribbleClick(x1, y1, event->modifiers(), false, true);
}
event->accept();
}
resetCurrentMouseDragAction();
}
void JKQTPlotter::resetCurrentMouseDragAction () {
mouseDragingRectangle=false;
currentMouseDragAction.clear();
updateCursor();
}
void JKQTPlotter::mouseDoubleClickEvent ( QMouseEvent * event ){
bool foundIT=false;
JKQTPMouseDoubleClickActionsHashMapIterator itAction=findMatchingMouseDoubleClickAction(event->button(), event->modifiers(), &foundIT);
if (foundIT) {
// we found an action to perform on this double-click
if (itAction.value()==JKQTPMouseDoubleClickActions::jkqtpdcaClickOpensContextMenu) {
openStandardContextMenu(event->pos().x(), event->pos().y());
} else if (itAction.value()==JKQTPMouseDoubleClickActions::jkqtpdcaClickOpensSpecialContextMenu) {
openSpecialContextMenu(event->pos().x(), event->pos().y());
} else if (itAction.value()==JKQTPMouseDoubleClickActions::jkqtpdcaClickZoomsIn || itAction.value()==JKQTPMouseDoubleClickActions::jkqtpdcaClickZoomsOut) {
double factor=4.0;
if (itAction.value()==JKQTPMouseDoubleClickActions::jkqtpdcaClickZoomsOut) factor=1;
double xmin=plotter->p2x(static_cast<double>(event->pos().x())/magnification-static_cast<double>(plotter->getPlotWidth())/factor);
double xmax=plotter->p2x(static_cast<double>(event->pos().x())/magnification+static_cast<double>(plotter->getPlotWidth())/factor);
double ymin=plotter->p2y(static_cast<double>(event->pos().y())/magnification-static_cast<double>(getPlotYOffset())+static_cast<double>(plotter->getPlotHeight())/factor);
double ymax=plotter->p2y(static_cast<double>(event->pos().y())/magnification-static_cast<double>(getPlotYOffset())-static_cast<double>(plotter->getPlotHeight())/factor);
if ( (event->pos().x()/magnification<plotter->getInternalPlotBorderLeft()) || (event->pos().x()/magnification>plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) ) {
xmin=getXMin();
xmax=getXMax();
} else if (((event->pos().y()-getPlotYOffset())/magnification<plotter->getInternalPlotBorderTop()) || ((event->pos().y()-getPlotYOffset())/magnification>plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
ymin=getYMin();
ymax=getYMax();
}
plotter->setXY(xmin, xmax, ymin, ymax, true);
update();
} else if (itAction.value()==JKQTPMouseDoubleClickActions::jkqtpdcaClickMovesViewport) {
QRectF zoomRect= QRectF(QPointF(plotter->x2p(getXAxis()->getMin()),plotter->y2p(getYAxis()->getMax())), QPointF(plotter->x2p(getXAxis()->getMax()),plotter->y2p(getYAxis()->getMin())));
if ( (event->pos().x()/magnification<plotter->getInternalPlotBorderLeft()) || (event->pos().x()/magnification>plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) ) {
zoomRect.moveCenter(QPointF(zoomRect.center().x(), event->pos().y()));
} else if (((event->pos().y()-getPlotYOffset())/magnification<plotter->getInternalPlotBorderTop()) || ((event->pos().y()-getPlotYOffset())/magnification>plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
zoomRect.moveCenter(QPointF(event->pos().x(), zoomRect.center().y()));
} else {
zoomRect.moveCenter(QPointF(event->pos().x(), event->pos().y()));
}
setXY(plotter->p2x(zoomRect.left()), plotter->p2x(zoomRect.right()), plotter->p2y(zoomRect.bottom()), plotter->p2y(zoomRect.top()), true);
}
}
// only react on double clicks inside the widget
if ( (event->pos().x()/magnification>=plotter->getInternalPlotBorderLeft()) && (event->pos().x()/magnification<=plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) &&
((event->pos().y()-getPlotYOffset())/magnification>=plotter->getInternalPlotBorderTop()) && ((event->pos().y()-getPlotYOffset())/magnification<=plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
mouseLastClickX=event->pos().x();
mouseLastClickY=event->pos().y();
emit plotMouseDoubleClicked(plotter->p2x(event->pos().x()/magnification), plotter->p2y((event->pos().y()-getPlotYOffset())/magnification), event->modifiers(), event->button());
}
event->accept();
updateCursor();
currentMouseDragAction.clear();
}
void JKQTPlotter::keyReleaseEvent(QKeyEvent *event) {
QWidget::keyPressEvent(event);
if (event->key()==Qt::Key_Escape && event->modifiers()==Qt::NoModifier) {
if (mouseDragingRectangle || currentMouseDragAction.isValid()) {
mouseDragingRectangle=false;
image=oldImage;
currentMouseDragAction.clear();
update();
event->accept();
}
}
updateCursor();
}
void JKQTPlotter::wheelEvent ( QWheelEvent * event ) {
#if (QT_VERSION >= QT_VERSION_CHECK(5, 15, 0))
const double wheel_x=event->position().x();
const double wheel_y=event->position().y();
#else
const double wheel_x=event->pos().x();
const double wheel_y=event->pos().y();
#endif
const QPoint angleDelta=event->angleDelta();
QRectF zoomRect= QRectF(QPointF(plotter->x2p(getXAxis()->getMin()),plotter->y2p(getYAxis()->getMax())), QPointF(plotter->x2p(getXAxis()->getMax()),plotter->y2p(getYAxis()->getMin())));
#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
//qDebug()<<"wheelEvent: "<<event->type()<<"["<<event->deviceType()<<", "<<event->device()->name()<<", "<<event->device()->seatName()<<", "<<event->device()->capabilities()<<", "<<event->pointingDevice()->pointerType()<<", "<<event->timestamp()<<"], "<<event->phase()<<", "<<event->modifiers()<<", "<<angleDelta<<", "<<event->pixelDelta()<<", "<<event->inverted()<<", "<<event->source()<<", "<<event->buttons();
#else
//qDebug()<<"wheelEvent: "<<event->type()<<", "<<event->timestamp()<<", "<<event->phase()<<", "<<event->modifiers()<<", "<<angleDelta<<", "<<event->pixelDelta()<<", "<<event->inverted()<<", "<<event->source()<<", "<<event->buttons();
#endif
bool foundIT=false;
auto itAction=findMatchingMouseWheelAction(event->modifiers(), &foundIT);
WheelActionType acTodo=WheelActionType::None;
QPointF d(0,0); // for acPan, the shift of the ROI
double factor=1; // for zooming, the zoom-factor
if (foundIT) {
if (itAction.value()==JKQTPMouseWheelActions::jkqtpmwaZoomByWheel) {
acTodo=WheelActionType::Zoom;
if (abs(angleDelta.y())>30 && angleDelta.x()==0) {
factor=pow(2.0, 1.0*static_cast<double>(angleDelta.y())/120.0);
} else if (abs(angleDelta.x())>30 && angleDelta.y()==0) {
factor=pow(2.0, 1.0*static_cast<double>(angleDelta.x())/120.0);
}
} else if (itAction.value()==JKQTPMouseWheelActions::jkqtpmwaZoomByWheelAndTrackpadPan) {
// this heuristics recognizes pan-gestures on a track-pad.
// These are converted by Qt to wheelEvents with small angleDelta()-Values
// typical angleDelta values are 120 (1/10 degree) or above, here we use 30
// to accomodate for finer resolved mouse-wheels
//
// unfortunately there is no other way to distinguish these cases, as Qt does not transport
// the source of the QWheelEvent!
const bool doPan=(abs(angleDelta.x())<30 && abs(angleDelta.y())<30);
// in addition when the last wheelEvent was a pan-Event and the current is close in time (less than maxWheelEventSeriesTimestampDifference Milliseconds),
// but now we classify a zoom-event we assume there was an acceleration in panning, that leads to angleDelta() larger than the
// limit of 30 above ... in this we still force another PANNING!
const bool forcePan=(!doPan && lastWheelActionType==WheelActionType::Pan
&& (event->timestamp()-lastWheelActionTimestamp)<maxWheelEventSeriesTimestampDifference);
if (forcePan || doPan) {
acTodo=WheelActionType::Pan;
d=angleDelta;
} else {
acTodo=WheelActionType::Zoom;
if (abs(angleDelta.y())>30 && angleDelta.x()==0) {
factor=pow(2.0, 1.0*static_cast<double>(angleDelta.y())/120.0);
} else if (abs(angleDelta.x())>30 && angleDelta.y()==0) {
factor=pow(2.0, 1.0*static_cast<double>(angleDelta.x())/120.0);
}
}
} else if (itAction.value()==JKQTPMouseWheelActions::jkqtpmwaPanByWheel) {
acTodo=WheelActionType::Pan;
d=QPointF(angleDelta.x()/120.0*zoomRect.width()/10.0,
angleDelta.y()/120.0*zoomRect.height()/10.0);
// maximum shift is 100 Pixels in either direction
d.setX(jkqtp_bounded<double>(-100, d.x(), 100));
d.setY(jkqtp_bounded<double>(-100, d.y(), 100));
// minimmum shift is 10 pixels, unles |shift|<1
if (d.x()>=1 && d.x()<10) d.setX(10);
if (d.x()<=-1 && d.x()>-10) d.setX(-10);
if (d.y()>=1 && d.y()<10) d.setY(10);
if (d.y()<=-1 && d.y()>-10) d.setY(-10);
}
}else{
event->ignore();
return;
}
//qDebug()<<" action: "<<acTodo<<" factor="<<factor<<" d="<<d;
if (acTodo==WheelActionType::Zoom && factor!=1.0) {
double xmin=plotter->p2x(static_cast<double>(wheel_x)/magnification-static_cast<double>(plotter->getPlotWidth()*0.5)/factor);
double xmax=plotter->p2x(static_cast<double>(wheel_x)/magnification+static_cast<double>(plotter->getPlotWidth()*0.5)/factor);
double ymin=plotter->p2y(static_cast<double>(wheel_y)/magnification-static_cast<double>(getPlotYOffset())+static_cast<double>(plotter->getPlotHeight()*0.5)/factor);
double ymax=plotter->p2y(static_cast<double>(wheel_y)/magnification-static_cast<double>(getPlotYOffset())-static_cast<double>(plotter->getPlotHeight()*0.5)/factor);
if ( (wheel_x/magnification<plotter->getInternalPlotBorderLeft()) || (wheel_x/magnification>plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) ) {
xmin=getXMin();
xmax=getXMax();
} else if (((wheel_y-getPlotYOffset())/magnification<plotter->getInternalPlotBorderTop()) || ((wheel_y-getPlotYOffset())/magnification>plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
ymin=getYMin();
ymax=getYMax();
}
//qDebug()<<" zoom: factor="<<factor;
plotter->setXY(xmin, xmax, ymin, ymax, true);
} else if (acTodo==WheelActionType::Pan && (d.x()!=0 || d.y()!=0)) {
//qDebug()<<" pan: d="<<d;
if ( (wheel_x/magnification<plotter->getInternalPlotBorderLeft()) || (wheel_x/magnification>plotter->getPlotWidth()+plotter->getInternalPlotBorderLeft()) ) {
zoomRect.translate(0, d.y());
} else if (((wheel_y-getPlotYOffset())/magnification<plotter->getInternalPlotBorderTop()) || ((wheel_y-getPlotYOffset())/magnification>plotter->getPlotHeight()+plotter->getInternalPlotBorderTop()) ) {
zoomRect.translate(d.x(), 0);
} else {
zoomRect.translate(d.x(), d.y());
}
correctZoomRectForPanning(zoomRect);
// now apply the new range
setXY(plotter->p2x(zoomRect.left()), plotter->p2x(zoomRect.right()), plotter->p2y(zoomRect.bottom()), plotter->p2y(zoomRect.top()), true);
}
event->accept();
emit plotMouseWheelOperated(plotter->p2x(wheel_x/magnification), plotter->p2y((wheel_y-getPlotYOffset())/magnification), event->modifiers(), angleDelta.x(), angleDelta.y());
updateCursor();
currentMouseDragAction.clear();
lastWheelActionType=acTodo;
lastWheelActionTimestamp=event->timestamp();
}
bool JKQTPlotter::event(QEvent* event) {
//qDebug()<<"event: "<<event->type()<<", "<<event->isAccepted();
return QWidget::event(event);
}
int JKQTPlotter::getPlotYOffset() const {
int plotYOffset=0;
if (plotterStyle.displayMousePosition) {
plotYOffset=plotYOffset+QFontMetrics(font()).height()+2;
}
if (plotterStyle.toolbarEnabled && plotterStyle.toolbarAlwaysOn) {
plotYOffset=plotYOffset+toolbar->height();
}
return plotYOffset;
}
void JKQTPlotter::initContextMenu()
{
resetContextMenu(true);
contextMenu->addAction(plotter->getActionSaveData());
contextMenu->addAction(plotter->getActionSavePlot());
#ifndef JKQTPLOTTER_COMPILE_WITHOUT_PRINTSUPPORT
contextMenu->addAction(plotter->getActionPrint());
#endif
contextMenu->addSeparator();
contextMenu->addAction(plotter->getActionCopyPixelImage());
contextMenu->addAction(plotter->getActionCopyData());
contextMenu->addAction(plotter->getActionCopyMatlab());
contextMenu->addSeparator();
contextMenu->addAction(plotter->getActionShowPlotData());
contextMenu->addSection(tr("Zooming"));
contextMenu->addAction(plotter->getActionZoomAll());
contextMenu->addAction(plotter->getActionZoomIn());
contextMenu->addAction(plotter->getActionZoomOut());
contextMenu->addSeparator();
contextMenu->addAction(actMouseMoveToolTip);
contextMenu->addSection(tr("left mouse button tool"));
contextMenu->addAction(actMouseLeftAsDefault);
contextMenu->addAction(actMouseLeftAsPanView);
contextMenu->addAction(actMouseLeftAsZoomRect);
contextMenu->addAction(actMouseLeftAsRuler);
contextMenu->addSeparator();
QMenu* menVisibleGroup=new QMenu(tr("Graph Visibility"), contextMenu);
for (size_t i=0; i<getPlotter()->getGraphCount(); i++) {
QString tit=getPlotter()->getGraph(i)->getTitle();
if (tit.isEmpty()) tit=tr("Graph %1").arg(static_cast<int>(i));
QAction* act=new QAction(tit, menVisibleGroup);
act->setCheckable(true);
act->setChecked(getPlotter()->getGraph(i)->isVisible());
const qreal dpr = devicePixelRatioF();
QPixmap pix=QPixmap::fromImage(getPlotter()->getGraph(i)->generateKeyMarker(QSize(16,16)*dpr));
pix.setDevicePixelRatio(dpr);
act->setIcon(QIcon(pix));
act->setData(static_cast<int>(i));
connect(act, SIGNAL(toggled(bool)), this, SLOT(reactGraphVisible(bool)));
menVisibleGroup->addAction(act);
}
contextMenu->addMenu(menVisibleGroup);
if (actions().size()>0) {
contextMenu->addSeparator();
contextMenu->addActions(actions());
}
bool hasSep=false;
JKQTBasePlotter::AdditionalActionsMap lst=getPlotter()->getLstAdditionalPlotterActions();
JKQTBasePlotter::AdditionalActionsMapIterator it(lst);
while (it.hasNext()) {
it.next();
if (it.value().size()>0) {
bool hasMenu=false;
for (int i=0; i<it.value().size(); i++) {
if (it.value().at(i)) {
if (!hasMenu) {
contextSubMenus.append(new QMenu(it.key(), this));
if (!hasSep) {
contextMenu->addSeparator();
hasSep=true;
}
hasMenu=true;
}
contextSubMenus.last()->addAction(it.value().at(i));
}
}
if (hasMenu) {
contextMenu->addMenu(contextSubMenus.last());
}
}
}
modifyContextMenu(contextMenu);
}
void JKQTPlotter::updateCursor() {
if (!currentMouseDragAction.isValid()) {
if (currentMouseMoveAction.contains(jkqtpmmaToolTipForClosestDataPoint)) {
setCursor(QCursor(Qt::CrossCursor));
} else {
setCursor(QCursor(Qt::ArrowCursor));
}
} else {
if (currentMouseDragAction.mode==jkqtpmdaZoomByRectangle) {
static QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_zoom.png");
static QBitmap mask(":/JKQTPlotter/jkqtp_cursor_zoom_mask.png");
setCursor(QCursor(cursor, mask, 9, 14));
} else if (currentMouseDragAction.mode==jkqtpmdaDrawRectangleForEvent) {
static QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_rectangle.png");
static QBitmap mask(":/JKQTPlotter/jkqtp_cursor_rectangle_mask.png");
setCursor(QCursor(cursor, mask, 9, 14));
} else if (currentMouseDragAction.mode==jkqtpmdaPanPlotOnMove || currentMouseDragAction.mode==jkqtpmdaPanPlotOnRelease) {
setCursor(QCursor(Qt::ClosedHandCursor));
} else if (currentMouseDragAction.mode==jkqtpmdaDrawCircleForEvent) {
static QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_circle.png");
static QBitmap mask(":/JKQTPlotter/jkqtp_cursor_circle_mask.png");
setCursor(QCursor(cursor, mask, 9, 14));
} else if (currentMouseDragAction.mode==jkqtpmdaDrawEllipseForEvent) {
static QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_ellipse.png");
static QBitmap mask(":/JKQTPlotter/jkqtp_cursor_ellipse_mask.png");
setCursor(QCursor(cursor, mask, 9, 14));
} else if (currentMouseDragAction.mode==jkqtpmdaDrawLineForEvent) {
static QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_line.png");
static QBitmap mask(":/JKQTPlotter/jkqtp_cursor_line_mask.png");
setCursor(QCursor(cursor, mask, 9, 14));
} else if (currentMouseDragAction.mode==jkqtpmdaScribbleForEvents) {
static QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_scribble.png");
static QBitmap mask(":/JKQTPlotter/jkqtp_cursor_scribble_mask.png");
setCursor(QCursor(cursor, mask, 9, 14));
} else if (currentMouseDragAction.mode==jkqtpmdaToolTipForClosestDataPoint) {
setCursor(QCursor(Qt::CrossCursor));
} else if (currentMouseDragAction.mode==jkqtpmdaRuler) {
static QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_line.png");
static QBitmap mask(":/JKQTPlotter/jkqtp_cursor_line_mask.png");
setCursor(QCursor(cursor, mask, 9, 14));
} else {
setCursor(QCursor(Qt::ArrowCursor));
}
}
}
void JKQTPlotter::synchronizeXAxis(double newxmin, double newxmax, double /*newymin*/, double /*newymax*/, JKQTPlotter * /*sender*/) {
setX(newxmin, newxmax);
}
void JKQTPlotter::synchronizeYAxis(double /*newxmin*/, double /*newxmax*/, double newymin, double newymax, JKQTPlotter * /*sender*/) {
setY(newymin, newymax);
}
void JKQTPlotter::synchronizeXYAxis(double newxmin, double newxmax, double newymin, double newymax, JKQTPlotter * /*sender*/) {
setXY(newxmin, newxmax, newymin, newymax);
}
void JKQTPlotter::redrawPlot() {
#ifdef JKQTBP_AUTOTIMER
JKQTPAutoOutputTimer jkaaot(QString("JKQTPlotter::redrawPlot()"));
#endif
if (!doDrawing) return;
disconnect(plotter, SIGNAL(plotUpdated()), this, SLOT(redrawPlot()));
plotter->setWidgetSize(jkqtp_roundTo<int>(width()/magnification), jkqtp_roundTo<int>(height()/magnification-getPlotYOffset()));
JKQTPEnhancedPainter painter(&image);
if (painter.isActive()) {
painter.scale(magnification, magnification);
//QTime t;
//t.start();
plotter->drawNonGrid(painter, QPoint(0,0));//, QRect(QPoint(0,0), QSize(plotter->getPlotWidth(), plotter->getPlotHeight())));
//qDebug()<<"drawNonGrid"<<objectName()<<": "<<t.elapsed()<<"ms";
}
oldImage=image;
connect(plotter, SIGNAL(plotUpdated()), this, SLOT(redrawPlot()));
update();
}
void JKQTPlotter::paintEvent(QPaintEvent *event){
#ifdef JKQTBP_AUTOTIMER
JKQTPAutoOutputTimer jkaaot(QString("JKQTPlotter::paintEvent()"));
#endif
// draw the saved image of the plot
JKQTPEnhancedPainter* p = new JKQTPEnhancedPainter(this);
if (p->isActive()) {
//std::cout<<"getPlotYOffset()="<<getPlotYOffset()<<std::endl;
p->save();
p->setBrush(palette().brush(QPalette::Window));
p->setPen(palette().color(QPalette::Window));
p->drawRect(geometry());
p->restore();
if (plotterStyle.displayMousePosition ) {
p->drawText(QPointF(plotter->getInternalPlotBorderLeft(), getPlotYOffset()-1), plotterStyle.mousePositionTemplate.arg(mousePosX).arg(mousePosY));
}
const QSize plotImageSize=getImageBufferSize();
if (image.size() != plotImageSize) {
p->drawPixmap(QRect(QPoint(0, getPlotYOffset()), plotImageSize), image);
} else {
p->drawPixmap(QPoint(0, getPlotYOffset()), image);
}
}
delete p;
event->accept();
}
void JKQTPlotter::resizeEvent(QResizeEvent *event) {
#ifdef JKQTBP_AUTOTIMER
JKQTPAutoOutputTimer jkaaot(QString("JKQTPlotter::resizeEvent()"));
#endif
//qDebug()<<"resizeEvent old="<<size()<<" new="<<event->size();
QWidget::resizeEvent(event);
event->accept();
bool sizeChanged=false;
int plotImageWidth=width();
int plotImageHeight=height();
plotImageHeight=plotImageHeight-getPlotYOffset();
if (plotImageWidth != image.width() || plotImageHeight != image.height()) {
sizeChanged=true;
}
if (sizeChanged) {
if (jkqtp_RESIZE_DELAY == 0) {
// Do this now
delayedResizeEvent();
} else {
if(!resizeTimer.isActive()){
resizeTimer.setSingleShot(true);
resizeTimer.start(jkqtp_RESIZE_DELAY);
}
}
}
//updateGeometry();
//qDebug()<<"resize DONE width()="<<width()<<" height()="<<height()<<" plotImageWidth="<<plotImageWidth<<" plotImageHeight="<<plotImageHeight<<" sizeChanged="<<sizeChanged<<"\n";
}
void JKQTPlotter::delayedResizeEvent()
{
if (getImageBufferSize() != image.size()) {
image=createImageBuffer();
emit widgetResized(width(), height(), this);
redrawPlot();
}
}
void JKQTPlotter::leaveEvent(QEvent * /*event*/) {
if (!plotterStyle.toolbarAlwaysOn) toolbar->hide();
}
void JKQTPlotter::updateToolbar()
{
toolbar->setIconSize(QSize(plotterStyle.toolbarIconSize, plotterStyle.toolbarIconSize));
if (plotterStyle.toolbarEnabled) {
if (plotterStyle.toolbarAlwaysOn) {
toolbar->setToolbarVanishesEnabled(false);
toolbar->show();
toolbar->move(1,1);
} else {
toolbar->setToolbarVanishesEnabled(true);
}
} else {
toolbar->hide();
}
update();
}
QSize JKQTPlotter::minimumSizeHint() const {
QSize m=QWidget::minimumSizeHint();
if (minSize.width()>m.width()) m.setWidth(minSize.width());
if (minSize.height()>m.height()) m.setHeight(minSize.height());
return m;
}
QSize JKQTPlotter::sizeHint() const {
return QWidget::sizeHint();
}
bool JKQTPlotter::isResizeTimerRunning() const
{
return resizeTimer.isActive();
}
void JKQTPlotter::synchronizeToMaster(JKQTPlotter *master, JKQTBasePlotter::SynchronizationDirection synchronizeDirection, bool synchronizeAxisLength, bool synchronizeZoomingMasterToSlave, bool synchronizeZoomingSlaveToMaster)
{
if (masterPlotterX) disconnect(masterPlotterX->getPlotter(), SIGNAL(plotScalingRecalculated()), this, SLOT(masterPlotScalingRecalculated()));
if (masterPlotterY) disconnect(masterPlotterY->getPlotter(), SIGNAL(plotScalingRecalculated()), this, SLOT(masterPlotScalingRecalculated()));
if (master) plotter->synchronizeToMaster(master->getPlotter(), synchronizeDirection, synchronizeAxisLength, synchronizeZoomingMasterToSlave, synchronizeZoomingSlaveToMaster);
if (synchronizeDirection==JKQTBasePlotter::sdXAxis || synchronizeDirection==JKQTBasePlotter::sdXYAxes) {
masterPlotterX=master;
if (masterPlotterX) connect(masterPlotterX->getPlotter(), SIGNAL(plotScalingRecalculated()), this, SLOT(masterPlotScalingRecalculated()));
}
if (synchronizeDirection==JKQTBasePlotter::sdYAxis || synchronizeDirection==JKQTBasePlotter::sdXYAxes) {
masterPlotterY=master;
if (masterPlotterY) connect(masterPlotterY->getPlotter(), SIGNAL(plotScalingRecalculated()), this, SLOT(masterPlotScalingRecalculated()));
}
redrawPlot();
}
void JKQTPlotter::synchronizeXToMaster(JKQTPlotter *master, bool synchronizeAxisLength, bool synchronizeZoomingMasterToSlave, bool synchronizeZoomingSlaveToMaster)
{
synchronizeToMaster(master, JKQTBasePlotter::sdXAxis, synchronizeAxisLength, synchronizeZoomingMasterToSlave, synchronizeZoomingSlaveToMaster);
}
void JKQTPlotter::synchronizeYToMaster(JKQTPlotter *master, bool synchronizeAxisLength, bool synchronizeZoomingMasterToSlave, bool synchronizeZoomingSlaveToMaster)
{
synchronizeToMaster(master, JKQTBasePlotter::sdYAxis, synchronizeAxisLength, synchronizeZoomingMasterToSlave, synchronizeZoomingSlaveToMaster);
}
void JKQTPlotter::resetMasterSynchronization(JKQTBasePlotter::SynchronizationDirection synchronizeDirection)
{
synchronizeToMaster(nullptr, synchronizeDirection, false, false, false);
}
void JKQTPlotter::masterPlotScalingRecalculated() {
//qDebug()<<"this="<<this<<", sender="<<sender()<<" --> masterPlotScalingRecalculated()";
redrawPlot();
}
void JKQTPlotter::setGridPrinting(bool enabled)
{
plotter->setGridPrinting(enabled);
}
void JKQTPlotter::addGridPrintingPlotter(size_t x, size_t y, JKQTPlotter *plotterOther)
{
plotter->addGridPrintingPlotter(x,y,plotterOther->getPlotter());
}
void JKQTPlotter::clearGridPrintingPlotters()
{
plotter->clearGridPrintingPlotters();
}
void JKQTPlotter::setGridPrintingCurrentX(size_t x)
{
plotter->setGridPrintingCurrentX(x);
}
void JKQTPlotter::setGridPrintingCurrentY(size_t y)
{
plotter->setGridPrintingCurrentY(y);
}
void JKQTPlotter::setGridPrintingCurrentPos(size_t x, size_t y)
{
plotter->setGridPrintingCurrentPos(x,y);
}
bool JKQTPlotter::isPlotUpdateEnabled() const {
return doDrawing;
}
void JKQTPlotter::pzoomChangedLocally(double newxmin, double newxmax, double newymin, double newymax, JKQTBasePlotter* /*sender*/) {
emit zoomChangedLocally(newxmin, newxmax, newymin, newymax, this);
minSize=QSizeF(plotter->getInternalPlotBorderLeft()+plotter->getInternalPlotBorderRight()+10, plotter->getInternalPlotBorderTop()+plotter->getInternalPlotBorderBottom()+10).toSize();
}
void JKQTPlotter::intBeforePlotScalingRecalculate() {
emit beforePlotScalingRecalculate();
}
void JKQTPlotter::reactGraphVisible(bool visible)
{
QAction* act=dynamic_cast<QAction*>(sender());
if (act) {
getPlotter()->setGraphVisible(act->data().toInt(), visible);
}
}
void JKQTPlotter::setMouseLeftActionAsPanView()
{
setOverrideMouseDragAction(Qt::LeftButton, Qt::NoModifier, JKQTPMouseDragActions::jkqtpmdaPanPlotOnMove);
}
void JKQTPlotter::setMouseLeftActionAsZoomRect()
{
setOverrideMouseDragAction(Qt::LeftButton, Qt::NoModifier, JKQTPMouseDragActions::jkqtpmdaZoomByRectangle);
}
void JKQTPlotter::setMouseLeftActionAsRuler()
{
setOverrideMouseDragAction(Qt::LeftButton, Qt::NoModifier, JKQTPMouseDragActions::jkqtpmdaRuler);
}
void JKQTPlotter::setMouseLeftDragActionAsToolTip()
{
setOverrideMouseDragAction(Qt::LeftButton, Qt::NoModifier, JKQTPMouseDragActions::jkqtpmdaToolTipForClosestDataPoint);
}
void JKQTPlotter::resetMouseLeftAction()
{
resetOverrideMouseDragAction(Qt::LeftButton, Qt::NoModifier);
}
void JKQTPlotter::setMouseMoveActionAsToolTip(bool enabled)
{
if (enabled) {
registerMouseMoveAction(Qt::NoModifier, JKQTPMouseMoveActions::jkqtpmmaToolTipForClosestDataPoint);
} else {
deregisterMouseMoveAction(Qt::NoModifier);
}
}
void JKQTPlotter::setMouseActionToolbarActionsActive(bool __value)
{
actgrpMouseLeft->setVisible(__value);
actMouseMoveToolTip->setVisible(__value);
}
QAction* JKQTPlotter::getActMouseLeftAsDefault() {
return actMouseLeftAsDefault;
}
QAction *JKQTPlotter::getActMouseLeftAsZoomRect()
{
return actMouseLeftAsZoomRect;
}
QAction *JKQTPlotter::getActMouseLeftAsPanView()
{
return actMouseLeftAsPanView;
}
QAction* JKQTPlotter::getActMouseLeftAsRuler() {
return actMouseLeftAsRuler;
}
const QAction* JKQTPlotter::getActMouseLeftAsDefault() const {
return actMouseLeftAsDefault;
}
const QAction *JKQTPlotter::getActMouseLeftAsZoomRect() const
{
return actMouseLeftAsZoomRect;
}
const QAction *JKQTPlotter::getActMouseLeftAsPanView() const
{
return actMouseLeftAsPanView;
}
const QAction* JKQTPlotter::getActMouseLeftAsRuler() const {
return actMouseLeftAsRuler;
}
const QAction* JKQTPlotter::getActMouseMoveToolTip() const {
return actMouseMoveToolTip;
}
QAction* JKQTPlotter::getActMouseMoveToolTip() {
return actMouseMoveToolTip;
}
void JKQTPlotter::setOverrideMouseDragAction(Qt::MouseButton button, Qt::KeyboardModifiers modifier, JKQTPMouseDragActions action)
{
registeredOverrideMouseDragActionModes.insert(QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifier), action);
}
void JKQTPlotter::resetOverrideMouseDragAction(Qt::MouseButton button, Qt::KeyboardModifiers modifier)
{
registeredOverrideMouseDragActionModes.remove(QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifier));
}
void JKQTPlotter::setContextMenuMode(JKQTPContextMenuModes mode) {
contextMenuMode=mode;
}
QMenu *JKQTPlotter::getSpecialContextMenu() const {
return this->menuSpecialContextMenu;
}
void JKQTPlotter::setSpecialContextMenu(QMenu *menu)
{
menuSpecialContextMenu=menu;
if (menuSpecialContextMenu) {
menuSpecialContextMenu->setParent(this);
menuSpecialContextMenu->close();
}
}
double JKQTPlotter::getMouseContextX() const {
return this->mouseContextX;
}
double JKQTPlotter::getMouseContextY() const {
return this->mouseContextY;
}
int JKQTPlotter::getMouseLastClickX() const {
return this->mouseLastClickX;
}
int JKQTPlotter::getMouseLastClickY() const {
return this->mouseLastClickY;
}
JKQTPContextMenuModes JKQTPlotter::getContextMenuMode() const {
return contextMenuMode;
}
const JKQTPlotterStyle &JKQTPlotter::getCurrentPlotterStyle() const
{
return plotterStyle;
}
void JKQTPlotter::setCurrentPlotterStyle(const JKQTPlotterStyle &style)
{
plotterStyle=style;
updateToolbar();
redrawPlot();
}
void JKQTPlotter::setCurrentPlotterStyle(const JKQTPlotterStyle &style, const JKQTBasePlotterStyle &baseStyle)
{
plotterStyle=style;
plotter->setCurrentPlotterStyle(baseStyle);
updateToolbar();
redrawPlot();
}
void JKQTPlotter::loadCurrentPlotterStyle(const QSettings &settings, const QString &group, bool alsoLoadBaseStyle)
{
plotterStyle.loadSettings(settings, group);
if (alsoLoadBaseStyle) plotter->loadCurrentPlotterStyle(settings, group);
updateToolbar();
redrawPlot();
}
void JKQTPlotter::saveCurrentPlotterStyle(QSettings &settings, const QString &group, bool alsoSaveBaseStyle) const
{
plotterStyle.saveSettings(settings, group);
if (alsoSaveBaseStyle) plotter->saveCurrentPlotterStyle(settings, group);
}
void JKQTPlotter::setMagnification(double m)
{
magnification=m;
redrawPlot();
}
void JKQTPlotter::modifyContextMenu(QMenu * /*menu*/)
{
}
void JKQTPlotter::populateToolbar(QToolBar *toolbar) const
{
toolbar->addAction(plotter->getActionSaveData());
toolbar->addAction(plotter->getActionSavePlot());
#ifndef JKQTPLOTTER_COMPILE_WITHOUT_PRINTSUPPORT
toolbar->addAction(plotter->getActionPrint());
#endif
toolbar->addSeparator();
toolbar->addAction(plotter->getActionCopyPixelImage());
toolbar->addAction(plotter->getActionCopyData());
toolbar->addAction(plotter->getActionCopyMatlab());
toolbar->addSeparator();
toolbar->addAction(plotter->getActionShowPlotData());
toolbar->addSeparator();
toolbar->addAction(plotter->getActionZoomAll());
toolbar->addAction(plotter->getActionZoomIn());
toolbar->addAction(plotter->getActionZoomOut());
toolbar->addSeparator();
toolbar->addAction(actMouseMoveToolTip);
toolbar->addSeparator();
toolbar->addAction(actMouseLeftAsDefault);
toolbar->addAction(actMouseLeftAsPanView);
toolbar->addAction(actMouseLeftAsZoomRect);
toolbar->addAction(actMouseLeftAsRuler);
if (actions().size()>0) {
toolbar->addSeparator();
toolbar->addActions(actions());
}
}
void JKQTPlotter::openContextMenu()
{
openContextMenu(mouseLastClickX, mouseLastClickY);
}
void JKQTPlotter::openContextMenu(int x, int y)
{
if (contextMenuMode==JKQTPContextMenuModes::jkqtpcmmStandardContextMenu) {
openStandardContextMenu(x,y);
} else if (contextMenuMode==JKQTPContextMenuModes::jkqtpcmmSpecialContextMenu) {
openSpecialContextMenu(x,y);
} else if (contextMenuMode==JKQTPContextMenuModes::jkqtpcmmStandardAndSpecialContextMenu) {
openStandardAndSpecialContextMenu(x,y);
}
}
void JKQTPlotter::openStandardContextMenu()
{
openStandardContextMenu(mouseLastClickX, mouseLastClickY);
}
void JKQTPlotter::resetContextMenu(bool createnew)
{
if (contextMenu) {
contextMenu->close();
contextMenu->clear();
qDeleteAll(contextSubMenus);
contextSubMenus.clear();
delete contextMenu;
}
if (createnew) {
contextMenu=new QMenu(this);
} else {
contextMenu=nullptr;
}
}
void JKQTPlotter::openStandardContextMenu(int x, int y)
{
//qDebug()<<"openContextMenu("<<x<<y<<contextMenu<<")";
mouseContextX=plotter->p2x(x/magnification);
mouseContextY=plotter->p2y((y-getPlotYOffset())/magnification);
initContextMenu();
contextMenu->popup(mapToGlobal(QPoint(x,y)));
//qDebug()<<" -> "<<mapToGlobal(QPoint(x,y))<<contextMenu->size()<<contextMenu->pos()<<contextMenu->parent();
emit contextMenuOpened(mouseContextX, mouseContextY, contextMenu);
//qDebug()<<"openContextMenu("<<x<<y<<contextMenu<<") ... DONE";
}
void JKQTPlotter::openSpecialContextMenu()
{
openSpecialContextMenu(mouseLastClickX, mouseLastClickY);
}
void JKQTPlotter::openSpecialContextMenu(int x, int y)
{
//qDebug()<<"openSpecialContextMenu("<<x<<y<<menuSpecialContextMenu<<")";
if (menuSpecialContextMenu) {
mouseContextX=plotter->p2x(x/magnification);
mouseContextY=plotter->p2y((y-getPlotYOffset())/magnification);
menuSpecialContextMenu->close();
menuSpecialContextMenu->popup(mapToGlobal(QPoint(x,y)));
menuSpecialContextMenu->resize(menuSpecialContextMenu->sizeHint());
//qDebug()<<" -> "<<mapToGlobal(QPoint(x,y))<<menuSpecialContextMenu->size()<<menuSpecialContextMenu->pos()<<menuSpecialContextMenu->parent();
emit contextMenuOpened(mouseContextX, mouseContextY, menuSpecialContextMenu);
//qDebug()<<"openSpecialContextMenu("<<x<<y<<menuSpecialContextMenu<<") ... DONE";
}
}
void JKQTPlotter::openStandardAndSpecialContextMenu()
{
openStandardAndSpecialContextMenu(mouseLastClickX, mouseLastClickY);
}
void JKQTPlotter::openStandardAndSpecialContextMenu(int x, int y)
{
//qDebug()<<"openContextMenu("<<x<<y<<contextMenu<<")";
mouseContextX=plotter->p2x(x/magnification);
mouseContextY=plotter->p2y((y-getPlotYOffset())/magnification);
initContextMenu();
if (menuSpecialContextMenu) {
contextMenu->addSeparator();
for (QAction* act: menuSpecialContextMenu->actions()) {
contextMenu->addAction(act);
}
}
contextMenu->popup(mapToGlobal(QPoint(x,y)));
//qDebug()<<" -> "<<mapToGlobal(QPoint(x,y))<<contextMenu->size()<<contextMenu->pos()<<contextMenu->parent();
emit contextMenuOpened(mouseContextX, mouseContextY, contextMenu);
//qDebug()<<"openContextMenu("<<x<<y<<contextMenu<<") ... DONE";
}
JKQTPMouseDragActionsHashMapIterator JKQTPlotter::findMatchingMouseDragAction(Qt::MouseButton button, Qt::KeyboardModifiers modifiers, bool* found) const
{
if (found) *found=false;
JKQTPMouseDragActionsHashMapIterator it=registeredOverrideMouseDragActionModes.cbegin();
while (it!=registeredOverrideMouseDragActionModes.cend() ) {
if (it.key()==QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifiers)) {
if (found) *found=true;
return it;
}
++it;
}
it=plotterStyle.registeredMouseDragActionModes.cbegin();
while (it!=plotterStyle.registeredMouseDragActionModes.cend() ) {
if (it.key()==QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifiers)) {
if (found) *found=true;
return it;
}
++it;
}
return plotterStyle.registeredMouseDragActionModes.cend();
}
JKQTPMouseDoubleClickActionsHashMapIterator JKQTPlotter::findMatchingMouseDoubleClickAction(Qt::MouseButton button, Qt::KeyboardModifiers modifiers, bool*found) const
{
if (found) *found=false;
for (JKQTPMouseDoubleClickActionsHashMapIterator it=plotterStyle.registeredMouseDoubleClickActions.cbegin(); it!=plotterStyle.registeredMouseDoubleClickActions.cend(); ++it) {
if (it.key()==QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifiers)) {
if (found) *found=true;
return it;
}
}
return plotterStyle.registeredMouseDoubleClickActions.cend();
}
JKQTPMouseWheelActionsHashMapIterator JKQTPlotter::findMatchingMouseWheelAction(Qt::KeyboardModifiers modifiers, bool *found) const
{
if (found) *found=false;
for (JKQTPMouseWheelActionsHashMapIterator it=plotterStyle.registeredMouseWheelActions.begin(); it!=plotterStyle.registeredMouseWheelActions.end(); ++it) {
if (it.key()==modifiers) {
if (found) *found=true;
return it;
}
}
return plotterStyle.registeredMouseWheelActions.end();
//return plotterStyle.registeredMouseWheelActions.find(modifiers);
}
JKQTPMouseMoveActionsHashMapIterator JKQTPlotter::findMatchingMouseMoveAction(Qt::KeyboardModifiers modifiers, bool *found) const
{
if (found) *found=false;
for (JKQTPMouseMoveActionsHashMapIterator it=plotterStyle.registeredMouseMoveActions.begin(); it!=plotterStyle.registeredMouseMoveActions.end(); ++it) {
if (it.key()==modifiers) {
if (found) *found=true;
return it;
}
}
return plotterStyle.registeredMouseMoveActions.end();
//return plotterStyle.registeredMouseMoveActions.find(modifiers);
}
void JKQTPlotter::setPlotUpdateEnabled(bool enable)
{
doDrawing=enable;
plotter->setEmittingSignalsEnabled(enable);
//qDebug()<<objectName()<<" doDrawing="<<doDrawing;
}
void JKQTPlotter::registerMouseDragAction(Qt::MouseButton button, Qt::KeyboardModifiers modifier, JKQTPMouseDragActions action)
{
plotterStyle.registeredMouseDragActionModes[QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifier)]=action;
if (button==Qt::LeftButton && modifier==Qt::NoModifier) {
actMouseLeftAsDefault->setChecked(true);
resetMouseLeftAction();
}
}
void JKQTPlotter::deregisterMouseDragAction(Qt::MouseButton button, Qt::KeyboardModifiers modifier)
{
plotterStyle.registeredMouseDragActionModes.remove(QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifier));
}
void JKQTPlotter::clearAllRegisteredMouseDragActions()
{
plotterStyle.registeredMouseDragActionModes.clear();
}
void JKQTPlotter::registerMouseDoubleClickAction(Qt::MouseButton button, Qt::KeyboardModifiers modifier, JKQTPMouseDoubleClickActions action)
{
plotterStyle.registeredMouseDoubleClickActions[QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifier)]=action;
}
void JKQTPlotter::deregisterMouseDoubleClickAction(Qt::MouseButton button, Qt::KeyboardModifiers modifier)
{
plotterStyle.registeredMouseDoubleClickActions.remove(QPair<Qt::MouseButton, Qt::KeyboardModifiers>(button, modifier));
}
void JKQTPlotter::clearAllRegisteredMouseDoubleClickActions()
{
plotterStyle.registeredMouseDoubleClickActions.clear();
}
void initJKQTPlotterResources()
{
initJKQTBasePlotterResources();
}
JKQTPlotter::MouseDragAction::MouseDragAction():
mode(jkqtpmdaZoomByRectangle), modifier(Qt::NoModifier), mouseButton(Qt::LeftButton), valid(false)
{
}
JKQTPlotter::MouseDragAction::MouseDragAction(Qt::MouseButton _mouseButton, Qt::KeyboardModifiers _modifier, JKQTPMouseDragActions _mode):
mode(_mode), modifier(_modifier), mouseButton(_mouseButton), valid(true)
{
}
bool JKQTPlotter::MouseDragAction::isValid() const {
return valid;
}
void JKQTPlotter::MouseDragAction::clear()
{
valid=false;
mode=jkqtpmdaZoomByRectangle;
modifier=Qt::NoModifier;
mouseButton=Qt::LeftButton;
}