/******************************************************************** gLineChart Implementation Copyright (c)2011 Mark Watkins License: GPL *********************************************************************/ #include #include #include #include "gLineChart.h" #define EXTRA_ASSERTS 1 gLineChart::gLineChart(gPointData *d,QColor col,int dlsize,bool _accelerate,bool _hide_axes,bool _square_plot) :gLayer(d),m_accelerate(_accelerate),m_drawlist_size(dlsize),m_hide_axes(_hide_axes),m_square_plot(_square_plot) { m_drawlist=new QPointD [dlsize]; color.clear(); color.push_back(col); m_report_empty=false; } gLineChart::~gLineChart() { delete [] m_drawlist; } // Time Domain Line Chart void gLineChart::Plot(gGraphWindow & w,float scrx,float scry) { if (!m_visible) return; if (!data) return; if (!data->IsReady()) return; int start_px=w.GetLeftMargin(), start_py=w.GetBottomMargin(); int width=scrx-(w.GetLeftMargin()+w.GetRightMargin()); int height=scry-(w.GetTopMargin()+w.GetBottomMargin()); double minx=w.min_x, miny=w.min_y, maxx=w.max_x, maxy=w.max_y; double xx=maxx-minx, yy=maxy-miny; float xmult=width/xx, ymult=height/yy; // time to pixel conversion multiplier // Return on screwy min/max conditions if ((xx<0) || (yy<0)) return; if ((yy==0) && (miny==0)) return; int num_points=0; for (int z=0;zVC();z++) num_points+=data->np[z]; // how many points all up? // Draw bounding box if something else will be drawn. if (!(!m_report_empty && !num_points)) { glColor3f (0.1F, 0.1F, 0.1F); glLineWidth (1); glBegin (GL_LINE_LOOP); glVertex2f (start_px, start_py); glVertex2f (start_px, start_py+height); glVertex2f (start_px+width,start_py+height); glVertex2f (start_px+width, start_py); glEnd (); } width--; if (!num_points) { // No Data? if (m_report_empty) { QString msg="No Waveform Available"; float x,y; //TextMarkup... GetTextExtent(msg,x,y,bigfont); //w.renderText(start_px+(width/2.0)-(x/2.0),start_py+(height/2.0)-(y/2.0),msg,*bigfont); DrawText(w,msg,start_px+(width/2.0)-(x/2.0),scry-w.GetBottomMargin()-height/2.0+y/2.0,0,Qt::gray,bigfont);//-(y/2.0) } return; } bool accel=m_accelerate; double sfit,sr; int dp,sam; QColor & col=color[0]; // Selected the plot line color qint32 vertcnt=0; GLshort * vertarray=vertex_array[0]; assert(vertarray!=NULL); float lastpx,lastpy; float px,py; int idx,idxend,np; bool done,first; double x0,x1,xL; int visible_points=0; for (int n=0;nVC();n++) { // for each segment int siz=data->np[n]; if (siz<=1) continue; // Don't bother drawing 1 point or less. QPointD * point=data->point[n]; x0=point[0].x(); xL=point[siz-1].x(); if (maxxxL) { if (siz==2) { // this happens on CPAP QPointD t=point[0]; point[0]=point[siz-1]; point[siz-1]=t; x0=point[0].x(); } else { qDebug("Reversed order sample fed to gLineChart - ignored."); continue; //assert(x10); double qx=xL-x0; // Full time range of this segment double gx=xx/qx; // ratio of how much of the whole data set this represents double segwidth=width*gx; double XR=xx/sr; double Z1=MAX(x0,minx); double Z2=MIN(xL,maxx); double ZD=Z2-Z1; double ZR=ZD/sr; double ZQ=ZR/XR; double ZW=ZR/(width*ZQ); const int num_averages=15; // Max n umber of samples taken from samples per pixel for better min/max values visible_points+=ZR*ZQ; if (accel && n>0) { sam=1; } if (ZWx1) { double j=minx-x0; // == starting min of first sample in this segment idx=floor(j/sr); // Loose the precision idx-=idx % sam; } // else just start from the beginning idxend=floor(xx/sr); idxend/=sam; // devide by number of samples skips np=(idxend-idx)+sam; np /= sam; } else { np=siz; } bool watch_verts_carefully=false; // better to do it here than in the main loop. np <<= 2; if (!accel && m_square_plot) np <<= 1; // double it again if (np>=maxverts) { watch_verts_carefully=true; //assert(np=sam) i-=sam; // Start with the previous sample (which will be in clipping area) } if (point[i].x() > maxx) done=true; // Let this iteration finish.. (This point will be in far clipping) px=1+((point[i].x() - minx) * xmult); // Scale the time scale X to pixel scale X if (!accel) { py=1+((point[i].y() - miny) * ymult); // Same for Y scale if (firstpx) { firstpx=false; } else { if (m_square_plot) { vertarray[vertcnt++]=lastpx; vertarray[vertcnt++]=lastpy; vertarray[vertcnt++]=start_px+px; vertarray[vertcnt++]=lastpy; vertarray[vertcnt++]=start_px+px; vertarray[vertcnt++]=lastpy; } else { vertarray[vertcnt++]=lastpx; vertarray[vertcnt++]=lastpy; } vertarray[vertcnt++]=start_px+px; vertarray[vertcnt++]=start_py+py; #if defined(EXTRA_ASSERTS) assert(vertcntmaxz) maxz=z; // maxz=Last pixel // Update the Y pixel bounds. if (y1m_drawlist[z].y()) m_drawlist[z].setY(y1); } if (done) break; } if (accel) { dp=0; // Plot compressed accelerated vertex list for (int i=minz;i>1); glDisableClientState(GL_VERTEX_ARRAY); if (antialias) { glDisable(GL_LINE_SMOOTH); glDisable(GL_BLEND); } glDisable(GL_SCISSOR_TEST); }