#include #include "gGraphView.h" Layer::Layer(ChannelID code) { m_code = code; m_visible = true; m_movable = false; m_day=NULL; m_miny=m_maxy=0; m_minx=m_maxx=0; m_order=0; m_width=m_height=0; m_X=m_Y=0; m_position=LayerCenter; } Layer::~Layer() { } void Layer::SetDay(Day * d) { m_day=d; if (!d) return; m_minx=d->first(m_code); m_maxx=d->last(m_code); m_miny=d->min(m_code); m_maxy=d->max(m_code); } bool Layer::isEmpty() { if (m_day && (m_day->count(m_code)!=0)) return false; return true; } void Layer::setLayout(LayerPosition position, short width, short height, short order) { m_position=position; m_width=width; m_height=height; m_order=order; } LayerGroup::LayerGroup() : Layer(EmptyChannel) { } LayerGroup::~LayerGroup() { } bool LayerGroup::isEmpty() { if (!m_day) return true; bool empty=true; for (int i=0;iisEmpty()) { empty=false; break; } } return empty; } void LayerGroup::SetDay(Day * d) { for (int i=0;iSetDay(d); } m_day=d; } void LayerGroup::AddLayer(Layer *l) { layers.push_back(l); } qint64 LayerGroup::Minx() { bool first=true; qint64 m=0,t; for (int i=0;iMinx(); if (!t) continue; if (first) { m=t; first=false; } else if (m>t) m=t; } return m; } qint64 LayerGroup::Maxx() { bool first=true; qint64 m=0,t; for (int i=0;iMaxx(); if (!t) continue; if (first) { m=t; first=false; } else if (mMiny(); if (t==layers[i]->Minx()) continue; if (first) { m=t; first=false; } else { if (m>t) m=t; } } return m; } EventDataType LayerGroup::Maxy() { bool first=true; EventDataType m=0,t; for (int i=0;iMaxy(); if (t==layers[i]->Miny()) continue; if (first) { m=t; first=false; } else if (mAddGraph(this); } else { qWarning() << "gGraph created without a gGraphView container.. Naughty programmer!! Bad!!!"; } m_margintop=10; m_marginbottom=10; m_marginleft=5; m_marginright=10; m_selecting_area=m_blockzoom=false; } gGraph::~gGraph() { } bool gGraph::isEmpty() { bool empty=true; for (QVector::iterator l=m_layers.begin();l!=m_layers.end();l++) { if (!(*l)->isEmpty()) { empty=false; break; } } return empty; } void gGraph::invalidate() { // this may not be necessary, as scrollbar & resize issues a full redraw.. //m_lastbounds.setWidth(m_graphview->width()); m_lastbounds.setY(m_graphview->findTop(this)); m_lastbounds.setX(gGraphView::titleWidth); m_lastbounds.setHeight(m_height * m_graphview->scaleY()); m_lastbounds.setWidth(m_graphview->width()-gGraphView::titleWidth); int i=0; //m_lastbounds.setHeight(0); } void gGraph::repaint() { if (m_lastbounds.height()>0) { //glScissor(0,m_lastbounds.y(),m_lastbounds.width(),m_lastbounds.height()); // m_graphview->swapBuffers(); // how fast is this?? //glEnable(GL_SCISSOR_BOX); glBegin(GL_QUADS); glColor4f(1,1,1,1.0); // Gradient End glVertex2i(0,m_lastbounds.y()); glVertex2i(gGraphView::titleWidth,m_lastbounds.y()); glVertex2i(gGraphView::titleWidth,m_lastbounds.y()+height()); glVertex2i(0,m_lastbounds.y()+height()); glEnd(); paint(m_lastbounds.x(),m_lastbounds.y(),m_lastbounds.width(),m_lastbounds.height()); m_graphview->swapBuffers(); //glDisable(GL_SCISSOR_BOX); } else { qDebug() << "Wanted to redraw graph" << m_title << "but previous bounds were invalid.. Issuing a slower full redraw instead. Todo: Find out why."; m_graphview->updateGL(); } } void gGraph::qglColor(QColor col) { m_graphview->qglColor(col); } void gGraph::renderText(QString text, int x,int y, float angle, QColor color, QFont *font) { // GL Font drawing is ass in Qt.. :( // I tried queuing this but got crappy memory leaks.. for now I don't give a crap if this is slow. QPainter *painter=m_graphview->painter; painter->endNativePainting(); QBrush b(color); painter->setBrush(b); painter->setFont(*font); if (angle==0) { painter->drawText(x,y,text); } else { float w,h; GetTextExtent(text, w, h, font); painter->translate(x,y); painter->rotate(-angle); painter->drawText(floor(-w/2.0),floor(-h/2.0),text); painter->rotate(+angle); painter->translate(-x,-y); } painter->beginNativePainting(); } void gGraph::paint(int originX, int originY, int width, int height) { m_lastbounds=QRect(originX,originY,width,height); glBegin(GL_QUADS); glColor4f(1,1,1,1.0); // Gradient End glVertex2i(originX,originY); glVertex2i(originX+width,originY); glColor4f(1,1,1.0,1.0); // Gradient End glVertex2i(originX+width,originY+height); glVertex2i(originX,originY+height); glEnd(); glColor4f(0,0,0,1); renderText(title(),20,originY+height/2,90); left=0,right=0,top=0,bottom=0; int tmp; originX+=m_marginleft; originY+=m_margintop; width-=m_marginleft+m_marginright; height-=m_margintop+m_marginbottom; for (int i=0;iHeight(); if (ll->position()==LayerTop) top+=tmp; if (ll->position()==LayerBottom) bottom+=tmp; } for (int i=0;iWidth(); if (ll->position()==LayerLeft) { ll->paint(*this,originX+left,originY+top,tmp,height-top-bottom); left+=tmp; } if (ll->position()==LayerRight) { right+=tmp; ll->paint(*this,originX+width-right,originY+top,tmp,height-top-bottom); } } bottom=0; top=0; for (int i=0;iHeight(); if (ll->position()==LayerTop) { ll->paint(*this,originX+left,originY+top,width-left-right,tmp); top+=tmp; } if (ll->position()==LayerBottom) { bottom+=tmp; ll->paint(*this,originX+left,originY+height-bottom,width-left-right,tmp); } } for (int i=0;iposition()==LayerCenter) { ll->paint(*this,originX+left,originY+top,width-left-right,height-top-bottom); } } if (m_selection.width()>0 && m_selecting_area) { glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glBegin(GL_QUADS); glColor4ub(128,128,128,128); glVertex2i(originX+m_selection.x(),originY+top); glVertex2i(originX+m_selection.x()+m_selection.width(),originY+top); glColor4ub(128,128,255,128); glVertex2i(originX+m_selection.x()+m_selection.width(),originY+height-top-bottom); glVertex2i(originX+m_selection.x(),originY+height-top-bottom); glEnd(); glDisable(GL_BLEND); } } void gGraph::AddLayer(Layer * l,LayerPosition position, short width, short height, short order, bool movable, short x, short y) { l->setLayout(position,width,height,order); l->setMovable(movable); l->setPos(x,y); m_layers.push_back(l); } void gGraph::mouseMoveEvent(QMouseEvent * event) { // qDebug() << m_title << "Move" << event->pos() << m_graphview->pointClicked(); int y=event->pos().y(); int x=event->pos().x(); int x2=m_graphview->pointClicked().x(),y2=m_graphview->pointClicked().y(); int w=m_width-(m_graphview->titleWidth+right+m_marginright); int h=m_height-(bottom+m_marginbottom); double xx=max_x-min_x; double xmult=xx/w; if ((event->buttons() & Qt::LeftButton) && (m_graphview->m_selected_graph==this)) { qDebug() << m_title << "Moved" << x << y << left << right << top << bottom << m_width << m_height; int a1=MIN(x,x2); int a2=MAX(x,x2); m_selecting_area=true; m_selection=QRect(a1-m_marginleft,0,a2-a1,m_height); m_graphview->updateGL(); //repaint(); } else m_selecting_area=false; if (x>left+m_marginleft && xtitleWidth+right+m_marginright) && y>top+m_margintop && ypos().y(); int x=event->pos().x(); int w=m_width-(m_graphview->titleWidth+right+m_marginright); int h=m_height-(bottom+m_marginbottom); int x2,y2; double xx=max_x-min_x; double xmult=xx/w; if (x>left+m_marginleft && xtitleWidth+right+m_marginright) && y>top+m_margintop && ypos().y(); int x=event->pos().x(); int w=m_width-(m_graphview->titleWidth+m_marginleft+left+right+m_marginright); int h=m_height-(bottom+m_marginbottom); int x2=m_graphview->pointClicked().x(),y2=m_graphview->pointClicked().y(); m_selecting_area=false; m_selection.setWidth(0); if (x>left+m_marginleft && xtop+m_margintop && ybutton() & Qt::RightButton) { ZoomX(2,x); // zoom out. } else if (event->button() & Qt::LeftButton) { if (abs(x-x2)<4) { ZoomX(0.5,x); } else { x-=left+m_marginleft; y-=top+m_margintop; x2-=left+m_marginleft; y2-=top+m_margintop; if (!m_blockzoom) { double xx=max_x-min_x; double xmult=xx/double(w); qint64 j1=min_x+xmult*x; qint64 j2=min_x+xmult*x2; qint64 a1=MIN(j1,j2) qint64 a2=MAX(j1,j2) m_graphview->SetXBounds(a1,a2); } else { double xx=rmax_x-rmin_x; double xmult=xx/double(w); qint64 j1=rmin_x+xmult*x; qint64 j2=rmin_x+xmult*x2; qint64 a1=MIN(j1,j2) qint64 a2=MAX(j1,j2) m_graphview->SetXBounds(a1,a2); } } } qDebug() << m_title << "Released" << min_x << max_x << x << y << x2 << y2 << left << right << top << bottom << m_width << m_height; // qDebug() << m_title << "Released" << event->pos() << m_graphview->pointClicked() << left << top; } //m_graphview->updateGL(); } void gGraph::mouseWheelEvent(QMouseEvent * event) { qDebug() << m_title << "Wheel" << event->x() << event->y(); } void gGraph::mouseDoubleClickEvent(QMouseEvent * event) { mousePressEvent(event); qDebug() << m_title << "Double Clicked" << event->x() << event->y(); } void gGraph::keyPressEvent(QKeyEvent * event) { qDebug() << m_title << "Key Pressed" << event->key(); } void gGraph::ZoomX(double mult,int origin_px) { int width=m_width-(m_graphview->titleWidth+m_marginleft+left+right+m_marginright); if (origin_px==0) origin_px=(width/2); else origin_px-=m_marginleft+left; 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; } m_graphview->SetXBounds(min,max); //updateSelectionTime(max-min); } // margin recalcs.. void gGraph::resize(int width, int height) { m_height=height; m_width=width; } qint64 gGraph::MinX() { bool first=true; qint64 val=0,tmp; for (QVector::iterator l=m_layers.begin();l!=m_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 gGraph::MaxX() { bool first=true; qint64 val=0,tmp; for (QVector::iterator l=m_layers.begin();l!=m_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 gGraph::MinY() { bool first=true; EventDataType val=0,tmp; for (QVector::iterator l=m_layers.begin();l!=m_layers.end();l++) { if ((*l)->isEmpty()) continue; tmp=(*l)->Miny(); if (tmp==0 && tmp==(*l)->Maxy()) continue; if (first) { val=tmp; first=false; } else { if (tmp < val) val = tmp; } } return rmin_y=val; } EventDataType gGraph::MaxY() { bool first=true; EventDataType val=0,tmp; for (QVector::iterator l=m_layers.begin();l!=m_layers.end();l++) { if ((*l)->isEmpty()) continue; tmp=(*l)->Maxy(); if (tmp==0 && tmp==(*l)->Miny()) continue; if (first) { val=tmp; first=false; } else { if (tmp > val) val = tmp; } } return rmax_y=val; } void gGraph::SetMinX(qint64 v) { min_x=v; } void gGraph::SetMaxX(qint64 v) { max_x=v; } void gGraph::SetMinY(EventDataType v) { min_y=v; } void gGraph::SetMaxY(EventDataType v) { max_y=v; } // Sets a new Min & Max X clipping, refreshing the graph and all it's layers. void gGraph::SetXBounds(qint64 minx, qint64 maxx) { min_x=minx; max_x=maxx; //repaint(); //m_graphview->updateGL(); } int gGraph::flipY(int y) { return m_graphview->height()-y; } void gGraph::ResetBounds() { min_x=MinX(); max_x=MaxX(); min_y=MinY(); max_y=MaxY(); } gGraphView::gGraphView(QWidget *parent) : QGLWidget(parent), m_offsetY(0),m_offsetX(0),m_scaleY(1.0),m_scrollbar(NULL) { m_sizer_index=m_graph_index=0; m_button_down=m_graph_dragging=m_sizer_dragging=false; this->setMouseTracking(true); InitGraphs(); } gGraphView::~gGraphView() { for (int i=0;idisconnect(SIGNAL(sliderMoved(int)),this); } } void gGraphView::AddGraph(gGraph *g) { if (!m_graphs.contains(g)) { m_graphs.push_back(g); updateScrollBar(); } } float gGraphView::totalHeight() { float th=0; for (int i=0;iisEmpty()) continue; th += m_graphs[i]->height() + graphSpacer; } return ceil(th); } float gGraphView::findTop(gGraph * graph) { float th=-m_offsetY; for (int i=0;iisEmpty()) continue; th += m_graphs[i]->height()*m_scaleY + graphSpacer; } //th-=m_offsetY; return ceil(th); } float gGraphView::scaleHeight() { float th=0; for (int i=0;iisEmpty()) continue; th += m_graphs[i]->height() * m_scaleY + graphSpacer; } return ceil(th); } void gGraphView::resizeEvent(QResizeEvent *e) { QGLWidget::resizeEvent(e); // This ques a redraw event.. updateScale(); for (int i=0;iresize(e->size().width(),m_graphs[i]->height()*m_scaleY); } } void gGraphView::scrollbarValueChanged(int val) { //qDebug() << "Scrollbar Changed" << val; m_offsetY=val; updateGL(); // do this on a timer? } void gGraphView::ResetBounds() { for (int i=0;iResetBounds(); } updateScale(); } void gGraphView::SetXBounds(qint64 minx, qint64 maxx) { for (int i=0;iSetXBounds(minx,maxx); } updateGL(); } void gGraphView::updateScale() { float th=totalHeight(); // height of all graphs float h=height(); // height of main widget if (th < h) { m_scaleY=h / th; // less graphs than fits on screen, so scale to fit } else { m_scaleY=1.0; } updateScrollBar(); } void gGraphView::updateScrollBar() { if (!m_scrollbar) return; if (!m_graphs.size()) return; float th=scaleHeight(); // height of all graphs float h=height(); // height of main widget float vis=0; for (int i=0;iisEmpty() ? 0 : 1; if (thsetMaximum(0); // turn scrollbar off. } else { // more graphs than fit on screen //m_scaleY=1.0; float avgheight=th/vis; m_scrollbar->setPageStep(avgheight); m_scrollbar->setSingleStep(avgheight/8.0); m_scrollbar->setMaximum(th-height()); if (m_offsetY>th-height()) { m_offsetY=th-height(); } } } void gGraphView::setScrollBar(MyScrollBar *sb) { m_scrollbar=sb; m_scrollbar->setMinimum(0); updateScrollBar(); this->connect(m_scrollbar,SIGNAL(valueChanged(int)),SLOT(scrollbarValueChanged(int))); } void gGraphView::initializeGL() { setAutoFillBackground(false); setAutoBufferSwap(false); glDisable(GL_LIGHTING); glDisable(GL_DEPTH_TEST); glDisable(GL_TEXTURE_2D); } void gGraphView::resizeGL(int w, int h) { glViewport(0, 0, w, h); glMatrixMode(GL_PROJECTION); glLoadIdentity(); glOrtho(0, w, h, 0, -1, 1); glMatrixMode(GL_MODELVIEW); glLoadIdentity(); } void gGraphView::paintGL() { if (width()<=0) return; if (height()<=0) return; QPainter qp(this); qp.beginNativePainting(); painter=&qp; glClearColor(255,255,255,255); //glClearDepth(1); glClear(GL_COLOR_BUFFER_BIT);// | GL_DEPTH_BUFFER_BIT); //glEnable(GL_BLEND); //glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); glBegin(GL_QUADS); glColor4f(1.0,1.0,1.0,1.0); // Gradient start glVertex2f(0, height()); glVertex2f(0, 0); glColor4f(0.8,0.8,1.0,1.0); // Gradient End glVertex2f(width(), 0); glVertex2f(width(), height()); glEnd(); float px=titleWidth-m_offsetX; float py=-m_offsetY; int numgraphs=m_graphs.size(); float h,w; //ax=px;//-m_offsetX; if (!numgraphs) { QString ts="No Data"; } else for (int i=0;iisEmpty()) continue; h=m_graphs[i]->height() * m_scaleY; // set clipping? if (py > height()) break; // we are done.. can't draw anymore if ((py + h + graphSpacer) >= 0) { w=width(); m_graphs[i]->paint(px,py,width()-titleWidth,h); glColor4f(0,0,0,1); //if (ix(); int y=event->y(); if (m_sizer_dragging) { // Resize handle being dragged float my=y - m_sizer_point.y(); //qDebug() << "Sizer moved vertically" << m_sizer_index << my*m_scaleY; float h=m_graphs[m_sizer_index]->height(); h+=my / m_scaleY; if (h > m_graphs[m_sizer_index]->minHeight()) { m_graphs[m_sizer_index]->setHeight(h); m_sizer_point.setX(x); m_sizer_point.setY(y); updateScrollBar(); updateGL(); } return; } if (m_graph_dragging) { // Title bar being dragged to reorder gGraph *p; int yy=m_sizer_point.y(); bool empty; if (y < yy) { for (int i=m_graph_index-1;i>=0;i--) { empty=m_graphs[i]->isEmpty(); // swapping upwards. int yy2=yy-graphSpacer-m_graphs[i]->height()*m_scaleY; yy2+=m_graphs[m_graph_index]->height()*m_scaleY; if (yheight()*m_scaleY); updateGL(); } m_graph_index--; } if (!empty) break; } } else if (y > yy+graphSpacer+m_graphs[m_graph_index]->height()*m_scaleY) { // swapping downwards qDebug() << "Graph Reorder" << m_graph_index; for (int i=m_graph_index+1;iisEmpty(); p=m_graphs[m_graph_index]; m_graphs[m_graph_index]=m_graphs[i]; m_graphs[i]=p; if (!empty) { m_sizer_point.setY(yy+graphSpacer+m_graphs[m_graph_index]->height()*m_scaleY); updateGL(); } m_graph_index++; if (!empty) break; } } return; } float py = -m_offsetY; float h; for (int i=0; i < m_graphs.size(); i++) { if (m_graphs[i]->isEmpty()) continue; h=m_graphs[i]->height() * m_scaleY; if (py > height()) break; // we are done.. can't draw anymore if ((py + h + graphSpacer) >= 0) { if ((y >= py) && (y < py + h)) { if (x >= titleWidth) { this->setCursor(Qt::ArrowCursor); QPoint p(x-titleWidth,y-py); QMouseEvent e(event->type(),p,event->button(),event->buttons(),event->modifiers()); m_graphs[i]->mouseMoveEvent(&e); } else { this->setCursor(Qt::OpenHandCursor); } } else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) { this->setCursor(Qt::SplitVCursor); } } py+=h; py+=graphSpacer; // do we want the extra spacer down the bottom? } } void gGraphView::mousePressEvent(QMouseEvent * event) { int x=event->x(); int y=event->y(); float py=-m_offsetY; float h; for (int i=0;iisEmpty()) continue; h=m_graphs[i]->height()*m_scaleY; if (py>height()) break; // we are done.. can't draw anymore if ((py + h + graphSpacer) >= 0) { if ((y >= py) && (y < py + h)) { //qDebug() << "Clicked" << i; if (x < titleWidth) { // clicked on title to drag graph.. m_graph_dragging=true; m_graph_index=i; m_sizer_point.setX(x); m_sizer_point.setY(py); // point at top of graph.. this->setCursor(Qt::ClosedHandCursor); } else { // send event to graph.. m_global_point_clicked=QPoint(x,y); m_point_clicked=QPoint (x-titleWidth,y-py); m_selected_graph=m_graphs[i]; QMouseEvent e(event->type(),m_point_clicked,event->button(),event->buttons(),event->modifiers()); m_graphs[i]->mousePressEvent(&e); m_graph_index=i; m_button_down=true; //m_graphs[i]->mousePressEvent(event); } } else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) { this->setCursor(Qt::SplitVCursor); m_sizer_dragging=true; m_sizer_index=i; m_sizer_point.setX(x); m_sizer_point.setY(y); //qDebug() << "Sizer clicked" << i; } } py+=h; py+=graphSpacer; // do we want the extra spacer down the bottom? } } void gGraphView::mouseReleaseEvent(QMouseEvent * event) { this->setCursor(Qt::ArrowCursor); if (m_sizer_dragging) { m_sizer_dragging=false; return; } if (m_graph_dragging) { m_graph_dragging=false; return; } if (m_button_down) { m_button_down=false; int x1=m_global_point_clicked.x()-event->x(); int y1=m_global_point_clicked.y()-event->y(); QPoint p(m_point_clicked.x()-x1,m_point_clicked.y()-y1); QMouseEvent e(event->type(),p,event->button(),event->buttons(),event->modifiers()); m_graphs[m_graph_index]->mouseReleaseEvent(&e); } int x=event->x(); int y=event->y(); } void gGraphView::mouseDoubleClickEvent(QMouseEvent * event) { // A single click gets sent first.. Need to set a timer to check for this event.. int x=event->x(); int y=event->y(); float py=-m_offsetY; float h; for (int i=0;iisEmpty()) continue; h=m_graphs[i]->height()*m_scaleY; if (py>height()) break; // we are done.. can't draw anymore if ((py + h + graphSpacer) >= 0) { if ((y >= py) && (y <= py + h)) { if (x < titleWidth) { // What to do when double clicked on the graph title ?? } else { // send event to graph.. m_graphs[i]->mouseDoubleClickEvent(event); } } else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) { // What to do when double clicked on the resize handle? } } py+=h; py+=graphSpacer; // do we want the extra spacer down the bottom? } } void gGraphView::wheelEvent(QWheelEvent * event) { // Hmm.. I could optionalize this to change mousewheel behaviour without affecting the scrollbar now.. m_scrollbar->SendWheelEvent(event); // Just forwarding the event to scrollbar for now.. } void gGraphView::keyPressEvent(QKeyEvent * event) { } MyScrollBar::MyScrollBar(QWidget * parent) :QScrollBar(parent) { } void MyScrollBar::SendWheelEvent(QWheelEvent * e) { wheelEvent(e); }