/* Copyright (c) 2008-2018 Jan W. Krieger () 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 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 . */ /** \file jkqtplotter.cpp * \ingroup jkqtpplotterclasses */ #include #if (QT_VERSION >= QT_VERSION_CHECK(5, 0, 0)) #include #include #else #include #endif #include "jkqtplotter/jkqtplotter.h" #include "jkqtplotter/jkqtpgraphs.h" #define JKQTP_RESIZE_DELAY 100 /************************************************************************************************************************** * JKQtPlotter **************************************************************************************************************************/ JKQtPlotter::JKQtPlotter(bool datastore_internal, QWidget* parent, JKQTPdatastore* datast): QWidget(parent, Qt::Widget) { initJKQtPlotterResources(); init(datastore_internal, parent, datast); } JKQtPlotter::JKQtPlotter(QWidget *parent): QWidget(parent, Qt::Widget) { initJKQtPlotterResources(); init(true, parent, nullptr); } void JKQtPlotter::init(bool datastore_internal, QWidget* parent, JKQTPdatastore* datast) { leftDoubleClickAction=LeftDoubleClickDefault; menuSpecialContextMenu=nullptr; mouseContextX=0; mouseContextY=0; setParent(parent); connect(&resizeTimer, SIGNAL(timeout()), this, SLOT(delayedResizeEvent())); doDrawing=false; magnification=1; plotter=new JKQtBasePlotter(datastore_internal, this, datast); plotter->set_emitSignals(false); plotter->set_backgroundColor(palette().color(QPalette::Window));//QPalette::Window plotter->set_def_backgroundColor(palette().color(QPalette::Window));//QPalette::Window //plotter->set_plotBackgroundColor(palette().color(QPalette::Base)); //plotter->set_def_plotBackgroundColor((palette().color(QPalette::Base)); mousePosX=0; mousePosY=0; rightMouseButtonAction=JKQtPlotter::RightMouseButtonContextMenu; connect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); connect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); 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=QImage(width(), height(), QImage::Format_ARGB32); oldImage=image; imageNoOverlays=image; toolbarIconSize=16; mouseDragingRectangle=false; def_userActionColor=QColor("steelblue"); userActionColor=def_userActionColor; def_userActionCompositionMode=QPainter::CompositionMode_SourceOver; userActionCompositionMode=def_userActionCompositionMode; def_mousePositionTemplate=QString("(%1; %2)"); mousePositionTemplate=def_mousePositionTemplate; displayMousePosition=true; displayCustomMousePosition=false; customMousePositiontext=""; displayToolbar=true; toolbarAlwaysOn=false; mouseActionMode=JKQtPlotter::ZoomRectangle; //zoomByDoubleAndRightMouseClick=true; zoomByMouseWheel=true; setMouseTracking(true); setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); toolbar=new JKVanishQToolBar(this); toolbar->clear(); toolbar->move(1,1); toolbar->hide(); toolbar->setAutoFillBackground(true); toolbar->addSeparator(); toolbar->addSeparator(); populateToolbar(toolbar); contextMenu=new QMenu(this); QSize s=QSize(toolbarIconSize, toolbarIconSize); toolbar->setIconSize(s); //move(32,32); resize(400,300); doDrawing=true; plotter->set_emitSignals(true); update_plot(); } JKQtPlotter::~JKQtPlotter() { disconnect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); disconnect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); 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::updateToolbarActions() { toolbar->clear(); toolbar->setAutoFillBackground(true); toolbar->addSeparator(); toolbar->addSeparator(); populateToolbar(toolbar); } void JKQtPlotter::loadSettings(QSettings& settings, QString group) { plotter->loadSettings(settings, group); set_toolbarIconSize(settings.value(group+"toolbar_icon_size", toolbarIconSize).toInt()); mousePositionTemplate=settings.value(group+"mouse_position_template", mousePositionTemplate).toString(); userActionColor=QColor(settings.value(group+"zoomrect_color", JKQTP_QColor2String(userActionColor)).toString()); update_plot(); } void JKQtPlotter::saveSettings(QSettings& settings, QString group) { plotter->saveSettings(settings, group); if (userActionColor!=def_userActionColor) settings.setValue(group+"zoomrect_color", JKQTP_QColor2String(userActionColor)); if (toolbarIconSize!=def_toolbarIconSize) settings.setValue(group+"toolbar_icon_size", toolbarIconSize); if (mousePositionTemplate!=def_mousePositionTemplate) settings.setValue(group+"mouse_position_template", mousePositionTemplate); } void JKQtPlotter::paintUserAction() { if (mouseDragingRectangle) { image=oldImage; if (image.width()>0 && image.height()>0 && !image.isNull()) { JKQTPEnhancedPainter painter(&image); QPen p=painter.pen(); p.setWidthF(1); p.setColor(QColor("black")); p.setStyle(Qt::DashLine); painter.setPen(p); QPainter::CompositionMode oldCMode=painter.compositionMode(); painter.setCompositionMode(userActionCompositionMode); if ((mouseDragRectXEnd!=mouseDragRectXStart) && (mouseDragRectYEnd!=mouseDragRectYStart)) { double x1=plotter->x2p(mouseDragRectXStart)*magnification; double y1=plotter->y2p(mouseDragRectYStart)*magnification; double x2=plotter->x2p(mouseDragRectXEnd)*magnification; double y2=plotter->y2p(mouseDragRectYEnd)*magnification; double dx=x2-x1; double dy=y2-y1; if ((mouseActionMode==JKQtPlotter::ZoomRectangle) || (mouseActionMode==JKQtPlotter::RectangleEvents)) { painter.setOpacity(0.2); painter.fillRect(QRectF(x1, y1, x2-x1, y2-y1), QBrush(userActionColor)); painter.setOpacity(1.0); painter.drawRect(QRectF(x1, y1, x2-x1, y2-y1)); } else if (mouseActionMode==JKQtPlotter::CircleEvents) { QColor zc=userActionColor; zc.setAlphaF(0.2); painter.setBrush(QBrush(zc)); painter.drawEllipse(QPointF(x1, y1), qMin(fabs(dx), fabs(dy)), qMin(fabs(dx), fabs(dy))); } else if (mouseActionMode==JKQtPlotter::EllipseEvents) { QColor zc=userActionColor; zc.setAlphaF(0.2); painter.setBrush(QBrush(zc)); painter.drawEllipse(QPointF(x1, y1), fabs(dx), fabs(dy)); } else if (mouseActionMode==JKQtPlotter::LineEvents) { QPen pp=p; pp.setColor(userActionColor); painter.setPen(pp); painter.drawLine(QPointF(x1,y1), QPointF(x2,y2)); } } painter.setCompositionMode(oldCMode); } update(); } } void JKQtPlotter::mouseMoveEvent ( QMouseEvent * event ) { if (displayMousePosition) { mousePosX=plotter->p2x(event->x()/magnification); mousePosY=plotter->p2y((event->y()-getPlotYOffset())/magnification); update(); } if (displayToolbar && (!toolbarAlwaysOn) && (!toolbar->isVisible())) { // decide whether to display toolbar int y1=10; if (event->y()/magnification>=0 && event->y()/magnification<=y1) { toolbar->show(); toolbar->move(1,1); } } if (!displayToolbar) { toolbar->hide(); } if (( (mouseActionMode==JKQtPlotter::ZoomRectangle) || (mouseActionMode==JKQtPlotter::RectangleEvents) || (mouseActionMode==JKQtPlotter::CircleEvents) || (mouseActionMode==JKQtPlotter::EllipseEvents) || (mouseActionMode==JKQtPlotter::ScribbleEvents) || (mouseActionMode==JKQtPlotter::LineEvents) ) && mouseDragingRectangle && (event->buttons() & Qt::LeftButton)) { if (mouseActionMode==JKQtPlotter::ScribbleEvents) { mouseDragRectXStart=mouseDragRectXEnd; mouseDragRectYStart=mouseDragRectYEnd; } mouseDragRectXEnd=plotter->p2x(event->x()/magnification); mouseDragRectYEnd=plotter->p2y((event->y()-getPlotYOffset())/magnification); paintUserAction(); event->accept(); //std::cout<modifiers(), false, false); } } else { event->accept(); /*if (emitSignals)*/ //emit plotMouseMove(x, y); } // emit clicked signal, if event occured inside plot only if ( (event->x()/magnification>=plotter->get_iplotBorderLeft()) && (event->x()/magnification<=plotter->get_plotWidth()+plotter->get_iplotBorderLeft()) && ((event->y()-getPlotYOffset())/magnification>=plotter->get_iplotBorderTop()) && ((event->y()-getPlotYOffset())/magnification<=plotter->get_plotHeight()+plotter->get_iplotBorderTop()) ) { emit plotMouseMove(plotter->p2x(event->x()/magnification), plotter->p2y((event->y()-getPlotYOffset())/magnification)); } } void JKQtPlotter::mousePressEvent ( QMouseEvent * event ){ if (event->button()==Qt::LeftButton) { mouseLastClickX=event->x(); mouseLastClickY=event->y(); if ( (mouseActionMode!=JKQtPlotter::NoMouseAction)) { mouseDragRectXStart=plotter->p2x(event->x()/magnification); mouseDragRectYStart=plotter->p2y((event->y()-getPlotYOffset())/magnification); mouseDragingRectangle=true; oldImage=image; event->accept(); if (mouseActionMode==JKQtPlotter::ScribbleEvents) emit userScribbleClick(mouseDragRectXStart, mouseDragRectYStart, event->modifiers(), true, false); if (mouseActionMode==JKQtPlotter::ClickEvents) emit userClickFinished(mouseDragRectXStart, mouseDragRectYStart, event->modifiers()); } } else if (event->button()==Qt::RightButton) { mouseLastClickX=event->x(); mouseLastClickY=event->y(); if (rightMouseButtonAction==JKQtPlotter::RightMouseButtonZoom) { double xmin=plotter->p2x((long)round((double)plotter->get_iplotBorderLeft()-(double)plotter->get_plotWidth()/2.0)); double xmax=plotter->p2x((long)round((double)plotter->get_iplotBorderLeft()+1.5*(double)plotter->get_plotWidth())); double ymin=plotter->p2y((long)round((double)plotter->get_iplotBorderTop()+1.5*(double)plotter->get_plotHeight())); double ymax=plotter->p2y((long)round((double)plotter->get_iplotBorderTop()-(double)plotter->get_plotHeight()/2.0)); if (plotter->get_xAxis()->isLogAxis()) { if (xmin<=0) xmin=1; if (xmax<=0) xmax=10; } if (plotter->get_yAxis()->isLogAxis()) { if (ymin<=0) ymin=1; if (ymax<=0) ymax=10; } /*plotter->get_xAxis->setRange(xmin, xmax); plotter->get_yAxis->setRange(ymin, ymax);*/ //update_plot(); /*if (plotter->get_emitSignals())*/ emit zoomChangedLocally(xmin, xmax, ymin, ymax, this); plotter->setXY(xmin, xmax, ymin, ymax); //update(); event->accept(); } else if (rightMouseButtonAction==JKQtPlotter::RightMouseButtonContextMenu) { openContextMenu(event->x(), event->y()); event->accept(); } } // emit clicked signal, if event occured inside plot only if ( (event->x()/magnification>=plotter->get_iplotBorderLeft()) && (event->x()/magnification<=plotter->get_plotWidth()+plotter->get_iplotBorderLeft()) && ((event->y()-getPlotYOffset())/magnification>=plotter->get_iplotBorderTop()) && ((event->y()-getPlotYOffset())/magnification<=plotter->get_plotHeight()+plotter->get_iplotBorderTop()) ) { emit plotMouseClicked(plotter->p2x(event->x()/magnification), plotter->p2y((event->y()-getPlotYOffset())/magnification), event->modifiers(), event->button()); event->accept(); } } void JKQtPlotter::mouseReleaseEvent ( QMouseEvent * event ){ if ((event->flags()&Qt::MouseEventCreatedDoubleClick)==Qt::MouseEventCreatedDoubleClick) { return; } if (mouseDragingRectangle && event->button()==Qt::LeftButton) { mouseDragRectXEnd=plotter->p2x(event->x()/magnification); mouseDragRectYEnd=plotter->p2y((event->y()-getPlotYOffset())/magnification); image=oldImage; //update(); mouseDragingRectangle=false; double x1=mouseDragRectXStart; double y1=mouseDragRectYStart; double x2=mouseDragRectXEnd; double y2=mouseDragRectYEnd; if ((mouseDragRectXStart!=mouseDragRectXEnd) && (mouseDragRectYStart!=mouseDragRectYEnd)) { if (mouseActionMode==JKQtPlotter::ZoomRectangle) { double xmin=mouseDragRectXStart; double xmax=mouseDragRectXEnd; double ymin=mouseDragRectYStart; double ymax=mouseDragRectYEnd; emit zoomChangedLocally(xmin, xmax, ymin, ymax, this); plotter->setXY(xmin, xmax, ymin, ymax); } else if (mouseActionMode==JKQtPlotter::RectangleEvents) { emit userRectangleFinished(x1, y1, x2-x1, y2-y1, event->modifiers()); } else if (mouseActionMode==JKQtPlotter::CircleEvents) { emit userCircleFinished(x1, y1, qMin(fabs(x2-x1), fabs(y2-y1)), event->modifiers()); } else if (mouseActionMode==JKQtPlotter::EllipseEvents) { emit userEllipseFinished(x1, y1, fabs(x2-x1), fabs(y2-y1), event->modifiers()); } else if (mouseActionMode==JKQtPlotter::LineEvents) { emit userLineFinished(x1, y1, x2, y2, event->modifiers()); } else if (mouseActionMode==JKQtPlotter::ScribbleEvents) { emit userScribbleClick(x1, y1, event->modifiers(), false, true); } } if (mouseActionMode!=JKQtPlotter::ZoomRectangle) update(); event->accept(); } } void JKQtPlotter::mouseDoubleClickEvent ( QMouseEvent * event ){ // only react on double clicks inside the widget if ( (event->x()/magnification>=plotter->get_iplotBorderLeft()) && (event->x()/magnification<=plotter->get_plotWidth()+plotter->get_iplotBorderLeft()) && ((event->y()-getPlotYOffset())/magnification>=plotter->get_iplotBorderTop()) && ((event->y()-getPlotYOffset())/magnification<=plotter->get_plotHeight()+plotter->get_iplotBorderTop()) ) { mouseLastClickX=event->x(); mouseLastClickY=event->y(); if (event->button()==Qt::LeftButton) { if (leftDoubleClickAction==LeftDoubleClickContextMenu) { openContextMenu(event->x(), event->y()); event->accept(); } else if (leftDoubleClickAction==LeftDoubleClickSpecialContextMenu) { openSpecialContextMenu(event->x(), event->y()); event->accept(); } } if (rightMouseButtonAction==JKQtPlotter::RightMouseButtonZoom && event->button()==Qt::RightButton) { double factor=4.0; if (event->button()==Qt::RightButton) factor=1; double xmin=plotter->p2x((long)round((double)event->x()/magnification-(double)plotter->get_plotWidth()/factor)); double xmax=plotter->p2x((long)round((double)event->x()/magnification+(double)plotter->get_plotWidth()/factor)); double ymin=plotter->p2y((long)round((double)event->y()/magnification-(double)getPlotYOffset()+(double)plotter->get_plotHeight()/factor)); double ymax=plotter->p2y((long)round((double)event->y()/magnification-(double)getPlotYOffset()-(double)plotter->get_plotHeight()/factor)); event->accept(); //xAxis->setRange(xmin, xmax); //yAxis->setRange(ymin, ymax); //update_plot(); /*if (plotter->get_emitSignals())*/ emit zoomChangedLocally(xmin, xmax, ymin, ymax, this); plotter->setXY(xmin, xmax, ymin, ymax); update(); } emit plotMouseDoubleClicked(plotter->p2x(event->x()/magnification), plotter->p2y((event->y()-getPlotYOffset())/magnification), event->modifiers(), event->button()); } else { event->ignore(); } } void JKQtPlotter::keyReleaseEvent(QKeyEvent *event) { QWidget::keyPressEvent(event); if (event->key()==Qt::Key_Escape && event->modifiers()==Qt::NoModifier) { if (mouseDragingRectangle) { mouseDragingRectangle=false; image=oldImage; update(); event->accept(); } } } void JKQtPlotter::wheelEvent ( QWheelEvent * event ) { // only react on wheel turns inside the widget, turning forward will zoom out and turning backwards will zoom in by a factor of 2 if ( (event->x()/magnification>=plotter->get_iplotBorderLeft()) && (event->x()/magnification<=plotter->get_plotWidth()+plotter->get_iplotBorderLeft()) && ((event->y()-getPlotYOffset())/magnification>=plotter->get_iplotBorderTop()) && ((event->y()-getPlotYOffset())/magnification<=plotter->get_plotHeight()+plotter->get_iplotBorderTop()) ) { double factor=pow(2.0, 1.0*(double)event->delta()/120.0)*2.0; //std::cout<<(double)event->delta()/120.0<<": "<p2x((long)round((double)event->x()/magnification-(double)plotter->get_plotWidth()/factor)); double xmax=plotter->p2x((long)round((double)event->x()/magnification+(double)plotter->get_plotWidth()/factor)); double ymin=plotter->p2y((long)round((double)event->y()/magnification-(double)getPlotYOffset()+(double)plotter->get_plotHeight()/factor)); double ymax=plotter->p2y((long)round((double)event->y()/magnification-(double)getPlotYOffset()-(double)plotter->get_plotHeight()/factor)); //std::cout<<"t="< "<accept(); /*if (plotter->get_emitSignals())*/ emit zoomChangedLocally(xmin, xmax, ymin, ymax, this); plotter->setXY(xmin, xmax, ymin, ymax); //update(); /*xAxis->setRange(xmin, xmax); yAxis->setRange(ymin, ymax); //update_plot(); if (emitSignals) emit zoomChangedLocally(xAxis->getMin(), xAxis->getMax(), yAxis->getMin(), yAxis->getMax(), this);*/ } else { event->ignore(); } } int JKQtPlotter::getPlotYOffset() { int plotYOffset=0; if (displayMousePosition) { plotYOffset=plotYOffset+QFontMetrics(font()).height()+2; } if (displayToolbar && toolbarAlwaysOn) { plotYOffset=plotYOffset+toolbar->height(); } return plotYOffset; } void JKQtPlotter::initContextMenu() { contextMenu->clear(); qDeleteAll(contextSubMenus); contextSubMenus.clear(); contextMenu->addAction(plotter->get_actSaveData()); contextMenu->addAction(plotter->get_actSavePlot()); contextMenu->addAction(plotter->get_actPrint()); contextMenu->addSeparator(); contextMenu->addAction(plotter->get_actCopyPixelImage()); contextMenu->addAction(plotter->get_actCopyData()); contextMenu->addAction(plotter->get_actCopyMatlab()); contextMenu->addSeparator(); contextMenu->addAction(plotter->get_actShowPlotData()); contextMenu->addSeparator(); contextMenu->addAction(plotter->get_actZoomAll()); contextMenu->addAction(plotter->get_actZoomIn()); contextMenu->addAction(plotter->get_actZoomOut()); contextMenu->addSeparator(); QMenu* menVisibleGroup=new QMenu(tr("Graph Visibility"), contextMenu); for (size_t i=0; igetGraphCount(); i++) { QString tit=get_plotter()->getGraph(i)->get_title(); if (tit.isEmpty()) tit=tr("Graph %1").arg(static_cast(i)); QAction* act=new QAction(tit, menVisibleGroup); act->setCheckable(true); act->setChecked(get_plotter()->getGraph(i)->get_visible()); act->setIcon(QIcon(QPixmap::fromImage(get_plotter()->getGraph(i)->generateKeyMarker(QSize(16,16))))); act->setData(static_cast(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=get_plotter()->get_lstAdditionalPlotterActions(); JKQtBasePlotter::AdditionalActionsMapIterator it(lst); while (it.hasNext()) { it.next(); if (it.value().size()>0) { bool hasMenu=false; for (int i=0; iaddSeparator(); hasSep=true; } hasMenu=true; } contextSubMenus.last()->addAction(it.value().at(i)); } } if (hasMenu) { contextMenu->addMenu(contextSubMenus.last()); } } } modifyContextMenu(contextMenu); } void JKQtPlotter::updateCursor() { if (mouseActionMode==JKQtPlotter::ZoomRectangle) { QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_zoom.png"); QBitmap mask(":/JKQTPlotter/jkqtp_cursor_zoom_mask.png"); setCursor(QCursor(cursor, mask, 9, 14)); } else if (mouseActionMode==JKQtPlotter::RectangleEvents) { QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_rectangle.png"); QBitmap mask(":/JKQTPlotter/jkqtp_cursor_rectangle_mask.png"); setCursor(QCursor(cursor, mask, 9, 14)); } else if (mouseActionMode==JKQtPlotter::CircleEvents) { QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_circle.png"); QBitmap mask(":/JKQTPlotter/jkqtp_cursor_circle_mask.png"); setCursor(QCursor(cursor, mask, 9, 14)); } else if (mouseActionMode==JKQtPlotter::EllipseEvents) { QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_ellipse.png"); QBitmap mask(":/JKQTPlotter/jkqtp_cursor_ellipse_mask.png"); setCursor(QCursor(cursor, mask, 9, 14)); } else if (mouseActionMode==JKQtPlotter::LineEvents) { QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_line.png"); QBitmap mask(":/JKQTPlotter/jkqtp_cursor_line_mask.png"); setCursor(QCursor(cursor, mask, 9, 14)); } else if (mouseActionMode==JKQtPlotter::ClickEvents) { setCursor(QCursor(Qt::CrossCursor)); } else if (mouseActionMode==JKQtPlotter::ScribbleEvents) { QBitmap cursor(":/JKQTPlotter/jkqtp_cursor_scribble.png"); QBitmap mask(":/JKQTPlotter/jkqtp_cursor_scribble_mask.png"); setCursor(QCursor(cursor, mask, 9, 14)); } else { setCursor(QCursor(Qt::ArrowCursor)); } } void JKQtPlotter::update_overlays() { if (!doDrawing) return; disconnect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); disconnect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); image=imageNoOverlays; JKQTPEnhancedPainter painter(&image); if (painter.isActive()) { painter.scale(magnification, magnification); plotter->drawNonGridOverlays(painter); } oldImage=image; connect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); connect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); update(); } 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::replot_overlays() { #ifdef JKQTBP_AUTOTIMER JKQTPAutoOutputTimer jkaaot(QString("JKQtPlotter::replot_overlays()")); #endif if (!doDrawing) return; disconnect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); disconnect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); image=imageNoOverlays; JKQTPEnhancedPainter painter(&image); if (painter.isActive()) { painter.scale(magnification, magnification); plotter->drawNonGridOverlays(painter); } oldImage=image; connect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); connect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); repaint(); } void JKQtPlotter::update_plot() { #ifdef JKQTBP_AUTOTIMER JKQTPAutoOutputTimer jkaaot(QString("JKQtPlotter::update_plot()")); #endif if (!doDrawing) return; disconnect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); disconnect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); plotter->resize(width()/magnification, height()/magnification-getPlotYOffset()); JKQTPEnhancedPainter painter(&image); if (painter.isActive()) { painter.scale(magnification, magnification); //QTime t; //t.start(); plotter->drawNonGrid(painter, QPoint(0,0), false);//, QRect(QPoint(0,0), QSize(plotter->get_plotWidth(), plotter->get_plotHeight()))); //qDebug()<<"drawNonGrid"<drawNonGridOverlays(painter); } oldImage=image; connect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); connect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); update(); } void JKQtPlotter::replot_plot() { #ifdef JKQTBP_AUTOTIMER JKQTPAutoOutputTimer jkaaot(QString("JKQtPlotter::replot_plot()")); #endif if (!doDrawing) return; disconnect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); disconnect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); plotter->resize(width()/magnification, height()/magnification-getPlotYOffset()); JKQTPEnhancedPainter painter(&image); if (painter.isActive()) { painter.scale(magnification, magnification); plotter->drawNonGrid(painter);//, QRect(QPoint(0,0), QSize(plotter->get_plotWidth(), plotter->get_plotHeight()))); } oldImage=image; connect(plotter, SIGNAL(plotUpdated()), this, SLOT(update_plot())); connect(plotter, SIGNAL(overlaysUpdated()), this, SLOT(replot_overlays())); repaint(); } 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()="<save(); p->setBrush(palette().brush(QPalette::Window)); p->setPen(palette().color(QPalette::Window)); p->drawRect(geometry()); p->restore(); if (displayMousePosition ) { if (displayCustomMousePosition) { p->drawText(plotter->get_iplotBorderLeft(), getPlotYOffset()-1, customMousePositiontext); } else { p->drawText(plotter->get_iplotBorderLeft(), getPlotYOffset()-1, mousePositionTemplate.arg(mousePosX).arg(mousePosY)); } } int plotImageWidth=width(); int plotImageHeight=height(); plotImageHeight=plotImageHeight-getPlotYOffset(); if (image.width()!=plotImageWidth || image.height()!=plotImageHeight) { p->drawImage(QRectF(0, getPlotYOffset(), plotImageWidth, plotImageHeight), image); } else { p->drawImage(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="<hide(); } void JKQtPlotter::updateToolbar() { if (displayToolbar) { if (toolbarAlwaysOn) { toolbar->set_toolbarVanishes(false); toolbar->show(); toolbar->move(1,1); } else { toolbar->set_toolbarVanishes(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((int)round((plotter->get_iplotBorderLeft()+plotter->get_iplotBorderRight())*1.5), (int)round((plotter->get_iplotBorderTop()+plotter->get_iplotBorderBottom())*1.5)); } QSize JKQtPlotter::sizeHint() const { return QWidget::sizeHint(); //minimumSizeHint()+QSize(200,100); } void JKQtPlotter::masterPlotScalingRecalculated() { if (masterPlotter) { if (plotter->get_masterSynchronizeHeight()||plotter->get_masterSynchronizeWidth()) { update_plot(); } } } void JKQtPlotter::synchronizeToMaster(JKQtPlotter* master, bool synchronizeWidth, bool synchronizeHeight) { if (!master) { resetMasterSynchronization(); } plotter->synchronizeToMaster(master->get_plotter(), synchronizeWidth, synchronizeHeight); masterPlotter=master; if (masterPlotter) connect(masterPlotter->get_plotter(), SIGNAL(plotScalingRecalculated()), this, SLOT(masterPlotScalingRecalculated())); update_plot(); } void JKQtPlotter::resetMasterSynchronization() { if (masterPlotter) disconnect(masterPlotter->get_plotter(), SIGNAL(plotScalingRecalculated()), this, SLOT(masterPlotScalingRecalculated())); plotter->resetMasterSynchronization(); update_plot(); } void JKQtPlotter::pzoomChangedLocally(double newxmin, double newxmax, double newymin, double newymax, JKQtBasePlotter* /*sender*/) { emit zoomChangedLocally(newxmin, newxmax, newymin, newymax, this); minSize=QSize(plotter->get_iplotBorderLeft()+plotter->get_iplotBorderRight()+10, plotter->get_iplotBorderTop()+plotter->get_iplotBorderBottom()+10); } void JKQtPlotter::intBeforePlotScalingRecalculate() { emit beforePlotScalingRecalculate(); } void JKQtPlotter::reactGraphVisible(bool visible) { QAction* act=dynamic_cast(sender()); if (act) { get_plotter()->setGraphVisible(act->data().toInt(), visible); } } void JKQtPlotter::set_zoomByMouseRectangle(bool zomByrectangle) { if (zomByrectangle) mouseActionMode=JKQtPlotter::ZoomRectangle; else mouseActionMode=JKQtPlotter::NoMouseAction; } void JKQtPlotter::set_menuSpecialContextMenu(QMenu *menu) { menuSpecialContextMenu=menu; if (menuSpecialContextMenu) { menuSpecialContextMenu->setParent(this); menuSpecialContextMenu->close(); } } void JKQtPlotter::setMagnification(double m) { magnification=m; update_plot(); } bool JKQtPlotter::get_zoomByMouseRectangle() const { return (mouseActionMode==JKQtPlotter::ZoomRectangle); } void JKQtPlotter::modifyContextMenu(QMenu * /*menu*/) { } void JKQtPlotter::populateToolbar(QToolBar *toolbar) const { toolbar->addAction(plotter->get_actSaveData()); toolbar->addAction(plotter->get_actSavePlot()); toolbar->addAction(plotter->get_actPrint()); toolbar->addSeparator(); toolbar->addAction(plotter->get_actCopyPixelImage()); toolbar->addAction(plotter->get_actCopyData()); toolbar->addAction(plotter->get_actCopyMatlab()); toolbar->addSeparator(); toolbar->addAction(plotter->get_actShowPlotData()); toolbar->addSeparator(); toolbar->addAction(plotter->get_actZoomAll()); toolbar->addAction(plotter->get_actZoomIn()); toolbar->addAction(plotter->get_actZoomOut()); if (actions().size()>0) { toolbar->addSeparator(); toolbar->addActions(actions()); } } void JKQtPlotter::setMousePositionLabel(const QString &text) { customMousePositiontext=text; if (displayCustomMousePosition) update(); } void JKQtPlotter::openContextMenu() { openContextMenu(mouseLastClickX, mouseLastClickY); } void JKQtPlotter::openContextMenu(int x, int y) { //qDebug()<<"openContextMenu("<p2x(x/magnification); mouseContextY=plotter->p2y((y-getPlotYOffset())/magnification); contextMenu->close(); initContextMenu(); contextMenu->popup(mapToGlobal(QPoint(x,y))); //qDebug()<<" -> "<size()<pos()<parent(); emit contextMenuOpened(mouseContextX, mouseContextY, contextMenu); //qDebug()<<"openContextMenu("<actions().size(); i++) { //qDebug()<<" - "<actions().at(i)->text(); } mouseContextX=plotter->p2x(x/magnification); mouseContextY=plotter->p2y((y-getPlotYOffset())/magnification); menuSpecialContextMenu->close(); menuSpecialContextMenu->popup(mapToGlobal(QPoint(x,y))); menuSpecialContextMenu->resize(menuSpecialContextMenu->sizeHint()); //qDebug()<<" -> "<size()<pos()<parent(); emit contextMenuOpened(mouseContextX, mouseContextY, menuSpecialContextMenu); //qDebug()<<"openSpecialContextMenu("<set_emitSignals(enable); //qDebug()<