/******************************************************************** gGraphWindow Implementation Copyright (c)2011 Mark Watkins License: GPL *********************************************************************/ #include #include #include #include #include "SleepLib/profiles.h" #include "graphwindow.h" #include "gTitle.h" #include "gXAxis.h" #include "gYAxis.h" #include "gFooBar.h" extern QLabel *qstatus2; gGraphWindow::gGraphWindow(QWidget *parent, const QString & title, QGLWidget * shared,Qt::WindowFlags f) : QGLWidget(parent,shared, f ) { splitter=NULL; m_scrX = m_scrY = 100; m_title=title; m_mouseRDown=m_mouseLDown=false; m_block_zoom=false; m_drag_foobar=false; m_dragGraph=false; m_gradient_background=true; m_foobar_pos=0; m_foobar_moved=0; SetMargins(10, 15, 0, 0); lastlayer=NULL; ti=QDateTime::currentDateTime(); gtitle=foobar=xaxis=yaxis=NULL; if (!title.isEmpty()) { AddLayer(new gTitle(title)); } //setAcceptDrops(true); setMouseTracking(true); setFocusPolicy(Qt::StrongFocus); rmin_x=rmax_x=0; min_x=max_x=0; } gGraphWindow::gGraphWindow(QWidget *parent, const QString & title, QGLContext * context,Qt::WindowFlags f) : QGLWidget((QGLContext *)context, parent, 0, f ) { gl_context=context; m_scrX = m_scrY = 100; m_title=title; m_mouseRDown=m_mouseLDown=false; SetMargins(10, 15, 0, 0); m_block_zoom=false; m_drag_foobar=false; m_dragGraph=false; m_gradient_background=false; m_foobar_pos=0; m_foobar_moved=0; lastlayer=NULL; ti=QDateTime::currentDateTime(); gtitle=foobar=xaxis=yaxis=NULL; if (!title.isEmpty()) { AddLayer(new gTitle(title)); } //setAcceptDrops(true); setMouseTracking(true); setFocusPolicy(Qt::StrongFocus); } gGraphWindow::~gGraphWindow() { for (list::iterator l=layers.begin();l!=layers.end();l++) delete (*l); layers.clear(); } bool gGraphWindow::isEmpty() { bool empty=true; for (list::iterator l=layers.begin();l!=layers.end();l++) { if (!(*l)->isEmpty()) { empty=false; break; } } return empty; } void gGraphWindow::AddLayer(gLayer *l) { if (l) { if (dynamic_cast(l)) { if (xaxis) { qWarning("Can only have one gXAxis per graph"); return; } //if (m_marginBottom(l)) { if (foobar) { qWarning("Can only have one gFooBar per graph"); return; } //if (m_marginBottom(l)) { if (yaxis) { qWarning("Can only have one gYAxis per graph"); return; } //if (m_marginLeft(l)) { if (gtitle) { qWarning("Can only have one gGraphTitle per graph"); return; } //if (m_marginLeftNotifyGraphWindow(this); layers.push_back(l); } }; // Sets a new Min & Max X clipping, refreshing the graph and all it's layers. void gGraphWindow::SetXBounds(qint64 minx, qint64 maxx) { min_x=minx; max_x=maxx; updateGL(); } void gGraphWindow::ResetBounds() { min_x=MinX(); max_x=MaxX(); min_y=MinY(); max_y=MaxY(); } void gGraphWindow::ZoomXPixels(int x1, int x2) { qint64 rx1=0,rx2=0; ZoomXPixels(x1,x2,rx1,rx2); if (pref["LinkGraphMovement"].toBool()) { for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(rx1,rx2); } } SetXBounds(rx1,rx2); } void gGraphWindow::ZoomXPixels(int x1,int x2,qint64 &rx1,qint64 &rx2) { x1-=GetLeftMargin(); x2-=GetLeftMargin(); if (x1<0) x1=0; if (x2<0) x2=0; if (x1>Width()) x1=Width(); if (x2>Width()) x2=Width(); qint64 min; qint64 max; if (!m_block_zoom) { min=min_x; max=max_x; } else { min=rmin_x; max=rmax_x; } double q=max-min; rx1=min+(double(x1)/Width()) * q; rx2=min+(double(x2)/Width()) * q; } // Move x-axis by the amount of space represented by integer i Pixels (negative values moves backwards) void gGraphWindow::MoveX(int i,qint64 &min, qint64 & max) { //if (i==0) return; min=min_x; max=max_x; double q=max-min; double rx1=(double(i)/Width()) * q; min-=rx1; max-=rx1; // Keep bounds when hitting hard edges if (minrmax_x) { max=rmax_x; min=max-q; } } void gGraphWindow::MoveX(int i) { qint64 min,max; MoveX(i,min,max); /* for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(min,max); } */ //if (!m_block_zoom) { SetXBounds(min,max); //} } void gGraphWindow::ZoomX(double mult,int origin_px) { if (origin_px==0) origin_px=(Width()/2); else origin_px-=GetLeftMargin(); if (origin_px<0) origin_px=0; if (origin_px>Width()) origin_px=Width(); // Okay, I want it to zoom in centered on the mouse click area.. // Find X graph position of mouse click // find current zoom width // apply zoom // center on point found in step 1. qint64 min=min_x; qint64 max=max_x; double hardspan=rmax_x-rmin_x; double span=max-min; double ww=double(origin_px) / double(Width()); double origin=ww * span; //double center=0.5*span; //double dist=(origin-center); double q=span*mult; if (q>hardspan) q=hardspan; if (qrmax_x) { max=rmax_x; min=max-q; } SetXBounds(min,max); updateSelectionTime(max-min); } void gGraphWindow::updateSelectionTime(qint64 span) // in milliseconds { qint64 time=span % 86400000; int days,hours,minutes,seconds,milli; milli=time % 1000; time/=1000; seconds=time%60; time/=60; minutes=time%60; time/=60; hours=time; time/=24; days=time; QString s; s.sprintf("%02i:%02i:%02i:%04i",hours,minutes,seconds,milli); qstatus2->setText(s); } gGraphWindow *LastGraphLDown=NULL; gGraphWindow *LastGraphRDown=NULL; gGraphWindow *currentWidget=NULL; void gGraphWindow::mouseMoveEvent(QMouseEvent * event) { // static bool first=true; static QRect last; // grabbed if (m_dragGraph) { if (LastGraphLDown!=this) currentWidget=this; if (splitter) { if (event->y()>m_scrY) { //qDebug() << "Swap Down"; int i=splitter->indexOf(this); splitter->insertWidget(i+1,this); } else if (event->y()<0) { //qDebug() << "Swap up"; int i=splitter->indexOf(this); if (i>0) { splitter->insertWidget(i-1,this); } } } return; } if (m_mouseLDown && LastGraphLDown && (LastGraphLDown!=this)) { LastGraphLDown->mouseMoveEvent(event); return; } if (m_mouseRDown && LastGraphRDown && (LastGraphRDown!=this)) { LastGraphRDown->mouseMoveEvent(event); return; } int y=event->y(); int x=event->x(); if (foobar && m_drag_foobar) { if (xm_scrX-GetRightMargin()) return; double mx=double(x1)/double(width); qint64 rminx=rmin_x; qint64 rmaxx=rmax_x; double rx=rmaxx-rminx; double qx=rx*mx; qx+=rminx; // qx is centerpoint of new zoom area. qint64 minx=min_x; double dx=max_x-minx; // zoom rect width; // Could smarten this up by remembering where the mouse was clicked on the foobar double gx=dx*m_foobar_pos; qx-=gx; if (qxrmaxx) { ex=rmaxx; qx=ex-dx; } if (m_mouseRDown) m_foobar_moved+=fabs((qx-minx))+fabs((m_mouseRClick.x()-x))+fabs((m_mouseRClick.y()-y)); if (m_mouseLDown) m_foobar_moved+=fabs((qx-minx))+fabs((m_mouseLClick.x()-x))+fabs((m_mouseLClick.y()-y)); //else SetXBounds(qx,ex); if (pref["LinkGraphMovement"].toBool()) { for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { if (*g==this) { qWarning() << "mouseMoveEvent *g should not equal this"; continue; } (*g)->SetXBounds(qx,ex); } } } else if (event->buttons() & Qt::RightButton) { MoveX(x - m_mouseRClick.x()); m_mouseRClick.setX(x); if (pref["LinkGraphMovement"].toBool()) { for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(min_x,max_x); } } } else if (event->buttons() & Qt::LeftButton) { int x1=m_mouseLClick.x(); int x2=x; int t1=MIN(x1,x2); int t2=MAX(x1,x2); if (t1(m_scrX-m_marginRight)) t2=m_scrX-m_marginRight; QRect r(t1-1, m_marginBottom, t2-t1, m_scrY-m_marginBottom-m_marginTop); double z; if (m_block_zoom) { z=rmax_x-rmin_x; } else { z=max_x-min_x; } double q=double(t2-t1)/Width(); this->updateSelectionTime(q*z); m_mouseRBlast=m_mouseRBrect; m_mouseRBrect=r; // TODO: Only the rect needs clearing, however OpenGL & wx have reversed coordinate systems. updateGL(); //r.Union(m_mouseRBlast),true); } } void gGraphWindow::mouseDoubleClickEvent(QMouseEvent * event) { // TODO: Retest.. QT might not be so retarded if (event->buttons() & Qt::LeftButton) OnMouseLeftDown(event); else if (event->buttons() & Qt::RightButton) OnMouseRightDown(event); } void gGraphWindow::mousePressEvent(QMouseEvent * event) { if (event->button()==Qt::LeftButton) OnMouseLeftDown(event); else if (event->button()==Qt::RightButton) OnMouseRightDown(event); } void gGraphWindow::mouseReleaseEvent(QMouseEvent * event) { if (event->button()==Qt::LeftButton) OnMouseLeftRelease(event); else if (event->button()==Qt::RightButton) OnMouseRightRelease(event); } void gGraphWindow::keyPressEvent(QKeyEvent * event) { bool moved=false; int accel=1; if (event->key()==Qt::Key_Left) { if (event->modifiers() & Qt::ControlModifier) accel=4; MoveX(40*accel); moved=true; } else if (event->key()==Qt::Key_Right) { if (event->modifiers() & Qt::ControlModifier) accel=4; MoveX(-40*accel); moved=true; } else if (event->key()==Qt::Key_Up) { double zoom_fact=2; if (event->modifiers() & Qt::ControlModifier) zoom_fact=5; ZoomX(zoom_fact,0); moved=true; } else if (event->key()==Qt::Key_Down) { double zoom_fact=.5; if (event->modifiers() & Qt::ControlModifier) zoom_fact=.2; ZoomX(zoom_fact,0); moved=true; } if (moved) { if (pref["LinkGraphMovement"].toBool()) { for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(min_x,max_x); } } event->accept(); } else { event->ignore(); } } void gGraphWindow::OnMouseRightDown(QMouseEvent * event) { int y=event->y(); int x=event->x(); if (ym_scrY-GetBottomMargin()) { // after top margin // return; //} // inside the margin area.. int width=m_scrX-GetRightMargin()-GetLeftMargin(); int height=m_scrY-GetBottomMargin()-GetTopMargin(); QRect hot1(GetLeftMargin(),GetTopMargin(),width,height); // Graph data area. m_mouseRClick.setX(x); m_mouseRClick.setY(y); m_mouseRClick_start=m_mouseRClick; m_mouseRDown=true; // m_mouseRDown=true; if (foobar && m_block_zoom && (((y>GetTopMargin())) && (yx1) && (xrmax_x) { x2=rmax_x; x1=x2-z; } // SetXBounds(x1,x2); LastGraphRDown=this; } } } void gGraphWindow::OnMouseRightRelease(QMouseEvent * event) { if (LastGraphRDown && (LastGraphRDown!=this)) { // Same graph that initiated the click?? LastGraphRDown->OnMouseLeftRelease(event); // Nope.. Give it the event. return; } LastGraphRDown=NULL; if (!m_mouseRDown) return; // Do this properly with real hotspots later.. int y=event->y(); int x=event->x(); int width=m_scrX-GetRightMargin()-GetLeftMargin(); int height=m_scrY-GetBottomMargin()-GetTopMargin(); QRect hot1(GetLeftMargin(),GetTopMargin(),width,height); // Graph data area. if (m_block_zoom) { // && hot1.Contains(x,y)) { //bool zoom_in=false; bool did_draw=false; // Finished Dragging the FooBar? if (foobar && m_drag_foobar) { if (m_foobar_moved<5) { double zoom_fact=2; if (event->modifiers() & Qt::ControlModifier) zoom_fact=5; //if (!m_block_zoom) { ZoomX(zoom_fact,x); did_draw=true; //} m_foobar_moved=0; } } m_drag_foobar=false; if (did_draw) { if (pref["LinkGraphMovement"].toBool()) { for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(min_x,max_x); } } } return; } double zoom_fact=2; if (ym_scrY-GetBottomMargin()) { if (!foobar) return; } if (event->modifiers() & Qt::ControlModifier) zoom_fact=5.0; if (abs(x-m_mouseRClick_start.x())<3 && abs(y-m_mouseRClick_start.y())<3) { // for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { //(*g)->ZoomX(zoom_fact,0); //} //if (!m_block_zoom) { ZoomX(zoom_fact,x); //event.GetX()); // adds origin to zoom out.. Doesn't look that cool. if (pref["LinkGraphMovement"].toBool()) { for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(min_x,max_x); } } //} }/* else { double min=MinX(); double max=MaxX(); for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(min,max); } }*/ m_mouseRDown=false; } void gGraphWindow::OnMouseLeftDown(QMouseEvent * event) { int y=event->y(); int x=event->x(); if (x(m_scrY-GetBottomMargin())) && (y<(m_scrY)))) { //-GetBottomMargin())+20))) { double rx=rmax_x-rmin_x; double qx=double(width)/rx; double minx=min_x-rmin_x; double maxx=max_x-rmin_x; int x1=(qx*minx); int x2=(qx*maxx); // length in pixels int xw=x2-x1; x1+=GetLeftMargin(); x2+=GetLeftMargin(); if ((x>x1-4) && (xindexOf(currentWidget); //int idx=splitter->indexOf(LastGraphLDown); splitter->insertWidget(newidx,LastGraphLDown); return; } } m_dragGraph=false; return; } if (LastGraphLDown && (LastGraphLDown!=this)) { // Same graph that initiated the click?? LastGraphLDown->OnMouseLeftRelease(event); // Nope.. Give it the event. return; } int y=event->y(); int x=event->x(); int width=m_scrX-GetRightMargin()-GetLeftMargin(); int height=m_scrY-GetBottomMargin()-GetTopMargin(); QRect hot1(GetLeftMargin(),GetTopMargin(),width,height); // Graph data area. //bool was_dragging_foo=false; bool did_draw=false; // Finished Dragging the FooBar? if (foobar && m_drag_foobar) { m_drag_foobar=false; //was_dragging_foo=true; if (m_foobar_moved<5) { double zoom_fact=0.5; if (event->modifiers() & Qt::ControlModifier) zoom_fact=0.25; //if (!m_block_zoom) { ZoomX(zoom_fact,0); did_draw=true; //} m_foobar_moved=0; } else m_mouseLDown=false; } // The user selected a range with the left mouse button if (!did_draw && m_mouseLDown) { //QPoint release(x, m_scrY-m_marginBottom); //QPoint press(m_mouseLClick.x(), m_marginTop); int x1=m_mouseRBrect.x(); int x2=x1+m_mouseRBrect.width(); int t1=MIN(x1,x2); int t2=MAX(x1,x2); if ((t2-t1)>4) { // Range Selected less than threshold m_mouseLDown=false; ZoomXPixels(t1,t2); did_draw=true; } } if (!did_draw && hot1.contains(x,y) && !m_drag_foobar && m_mouseLDown) { if (m_block_zoom) { x-=GetLeftMargin(); double rx=rmax_x-rmin_x; double mx=max_x-min_x; double qx=(rx/width)*double(x); if (mx>=rx) { mx=300000;//1.0/(24.0*15.0); } qx+=rmin_x; qx-=mx/2.0; if (qxrmax_x) { qx=rmax_x-mx; } SetXBounds(qx,qx+mx); did_draw=true; } else { int xp=x; //xp=0; double zoom_fact=0.5; if (event->modifiers() & Qt::ControlModifier) zoom_fact=0.25; ZoomX(zoom_fact,xp); //event.GetX()); // adds origin to zoom in.. Doesn't look that cool. did_draw=true; } //} } m_drag_foobar=false; QRect r=m_mouseRBrect; m_mouseRBrect=QRect(0, 0, 0, 0); m_mouseLDown=false; m_drag_foobar=false; if (!did_draw) { if (r!=m_mouseRBrect) updateGL(); } else { if (pref["LinkGraphMovement"].toBool()) { for (list::iterator g=link_zoom.begin();g!=link_zoom.end();g++) { (*g)->SetXBounds(min_x,max_x); } } } //} LastGraphLDown=NULL; } void gGraphWindow::SetMargins(int top, int right, int bottom, int left) { m_marginTop=top; m_marginBottom=bottom; m_marginLeft=left; m_marginRight=right; } void gGraphWindow::initializeGL() { setAutoFillBackground(false); setAutoBufferSwap(false); glDisable(GL_LIGHTING); glDisable(GL_DEPTH_TEST); glEnable(GL_TEXTURE_2D); m_scrX=width(); m_scrY=height(); } void gGraphWindow::resizeGL(int w, int h) { m_scrX=w; m_scrY=h; glViewport(0, 0, w, h); glMatrixMode(GL_PROJECTION); glLoadIdentity(); glOrtho(0, w, 0, h, -1, 1); glMatrixMode(GL_MODELVIEW); glLoadIdentity(); // Only needs doing on a mac updateGL(); } void gGraphWindow::Render(int w, int h) { if (m_gradient_background) { glBegin(GL_QUADS); glColor4f(1.0,1.0,1.0,.5); // Gradient start glVertex2f(0, h); glVertex2f(0, 0); glColor4f(0.8,0.8,1.0,.5); // Gradient End glVertex2f(w, 0); glVertex2f(w, h); glEnd(); } else { glClearColor(255,255,255,0); glClear(GL_COLOR_BUFFER_BIT ); //| GL_DEPTH_BUFFER_BIT } for (list::iterator l=layers.begin();l!=layers.end();l++) { (*l)->Plot(*this,w,h); } DrawTextQueue(*this); } void gGraphWindow::paintGL() { if (m_scrX<=0) m_scrX=width(); if (m_scrY<=0) m_scrY=height(); if (m_scrX<=0) return; if (m_scrY<=0) return; InitGraphs(); glDisable(GL_DEPTH_TEST); Render(m_scrX,m_scrY); if (m_mouseLDown) { if (m_mouseRBrect.width()>0) glDisable(GL_DEPTH_TEST); glColor4ub(50,50,200,64); glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glBegin(GL_QUADS); glVertex2f(m_mouseRBrect.x(),m_mouseRBrect.y()); glVertex2f(m_mouseRBrect.x()+m_mouseRBrect.width(),m_mouseRBrect.y()); glVertex2f(m_mouseRBrect.x()+m_mouseRBrect.width(),m_mouseRBrect.y()+m_mouseRBrect.height()); glVertex2f(m_mouseRBrect.x(),m_mouseRBrect.y()+m_mouseRBrect.height()); glEnd(); glDisable(GL_BLEND); //RoundedRectangle(m_mouseRBrect.x(),m_mouseRBrect.y(),m_mouseRBrect.width(),m_mouseRBrect.height(),5,QColor(50,50,200,64)); //glEnable(GL_DEPTH_TEST); } glEnable(GL_DEPTH_TEST); swapBuffers(); // Dump to screen. } qint64 gGraphWindow::MinX() { bool first=true; qint64 val=0,tmp; for (list::iterator l=layers.begin();l!=layers.end();l++) { if ((*l)->isEmpty()) continue; tmp=(*l)->Minx(); if (!tmp) continue; if (first) { val=tmp; first=false; } else { if (tmp < val) val = tmp; } } if (val) rmin_x=val; return val; } qint64 gGraphWindow::MaxX() { bool first=true; qint64 val=0,tmp; for (list::iterator l=layers.begin();l!=layers.end();l++) { if ((*l)->isEmpty()) continue; tmp=(*l)->Maxx(); if (!tmp) continue; if (first) { val=tmp; first=false; } else { if (tmp > val) val = tmp; } } if (val) rmax_x=val; return val; } EventDataType gGraphWindow::MinY() { bool first=true; EventDataType val=0,tmp; for (list::iterator l=layers.begin();l!=layers.end();l++) { if ((*l)->isEmpty()) continue; tmp=(*l)->Miny(); if (tmp==(*l)->Maxy()) continue; if (first) { val=tmp; first=false; } else { if (tmp < val) val = tmp; } } return rmin_y=val; } EventDataType gGraphWindow::MaxY() { bool first=true; EventDataType val=0,tmp; for (list::iterator l=layers.begin();l!=layers.end();l++) { if ((*l)->isEmpty()) continue; tmp=(*l)->Maxy(); if (tmp==(*l)->Miny()) continue; if (first) { val=tmp; first=false; } else { if (tmp > val) val = tmp; } } return rmax_y=val; } void gGraphWindow::SetMinX(qint64 v) { min_x=v; } void gGraphWindow::SetMaxX(qint64 v) { max_x=v; } void gGraphWindow::SetMinY(EventDataType v) { min_y=v; } void gGraphWindow::SetMaxY(EventDataType v) { max_y=v; }