/* gBarChart Implementation Copyright (c)2011 Mark Watkins License: GPL */ #include #include #include #include "gYAxis.h" #include "gSummaryChart.h" extern QLabel * qstatus2; SummaryChart::SummaryChart(QString label,GraphType type) :Layer(""),m_label(label),m_graphtype(type) { //QColor color=Qt::black; addGLBuf(quads=new GLShortBuffer(20000,GL_QUADS)); addGLBuf(lines=new GLShortBuffer(20000,GL_LINES)); quads->forceAntiAlias(true); lines->setSize(2); lines->forceAntiAlias(false); m_empty=true; hl_day=-1; m_machinetype=MT_CPAP; QDateTime d1=QDateTime::currentDateTime(); QDateTime d2=d1; d1.setTimeSpec(Qt::UTC); tz_offset=d2.secsTo(d1); tz_hours=tz_offset/3600.0; } SummaryChart::~SummaryChart() { } void SummaryChart::SetDay(Day * nullday) { Day * day=nullday; Layer::SetDay(day); m_values.clear(); m_times.clear(); m_days.clear(); m_hours.clear(); m_goodcodes.clear(); m_miny=999999999; m_maxy=-999999999; m_minx=0; m_maxx=0; int dn; EventDataType tmp,tmp2,total; ChannelID code; m_goodcodes.resize(m_codes.size()); for (int i=0;i >::iterator d=PROFILE.daylist.begin();d!=PROFILE.daylist.end();d++) { tt=QDateTime(d.key(),QTime(0,0,0),Qt::UTC).toTime_t(); dn=tt/86400; tt*=1000L; if (!m_minx || ttm_maxx) m_maxx=tt; total=0; bool fnd=false; if (m_graphtype==GT_SESSIONS) { for (int i=0;imachine_type()!=m_machinetype) continue; int ft=qint64(day->first())/1000L; ft+=tz_offset; // convert to local time int dz2=ft/86400; dz2*=86400; // ft = first sessions time, rounded back to midnight.. for (int s=0;ssize();s++) { tmp=(*day)[s]->hours(); m_values[dn][s]=tmp; total+=tmp; zt=qint64((*day)[s]->first())/1000L; zt+=tz_offset; tmp2=zt-dn*86400; //zt %= 86400; tmp2/=3600.0; //tmp2-=24.0; //if (zdn>dn2) { //if (tmp2<12) { // tmp2-=24; //m_times[dn][s]=(tmp2+12); //}/* else { // tmp2-=12; //m_times[dn][s]=(tmp2)-12; //}*/ m_times[dn][s]=tmp2; if (tmp2 < m_miny) m_miny=tmp2; if (tmp2+tmp > m_maxy) m_maxy=tmp2+tmp; } if (total>0) { m_days[dn]=day; m_hours[dn]=total; m_empty=false; } } } else { for (int j=0;jmachine_type()!=m_machinetype) continue; //m_values[dn][j+1]=0; bool hascode=//day->channelHasData(code) || type==ST_HOURS || type==ST_SESSIONS || day->settingExists(code) || day->hasData(code,type); if (hascode) { m_days[dn]=day; switch(m_type[j]) { case ST_AVG: tmp=day->avg(code); break; case ST_SUM: tmp=day->sum(code); break; case ST_WAVG: tmp=day->wavg(code); break; case ST_90P: tmp=day->p90(code); break; case ST_MIN: tmp=day->Min(code); break; case ST_MAX: tmp=day->Max(code); break; case ST_CNT: tmp=day->count(code); break; case ST_CPH: tmp=day->cph(code); break; case ST_SPH: tmp=day->sph(code); break; case ST_HOURS: tmp=day->hours(); break; case ST_SESSIONS: tmp=day->size(); break; case ST_SETMIN: tmp=day->settings_min(code); break; case ST_SETMAX: tmp=day->settings_max(code); break; case ST_SETAVG: tmp=day->settings_avg(code); break; case ST_SETWAVG: tmp=day->settings_wavg(code); break; case ST_SETSUM: tmp=day->settings_sum(code); break; default: break; } if (suboffset>0) { tmp-=suboffset; if (tmp<0) tmp=0; } total+=tmp; m_values[dn][j+1]=tmp; if (tmpm_maxy) m_maxy=tmp; m_goodcodes[j]=true; fnd=true; break; } } } if (fnd) { if (!m_fday) m_fday=dn; m_values[dn][0]=total; m_hours[dn]=day->hours(); if (m_graphtype==GT_BAR) { if (totalm_maxy) m_maxy=total; } //m_empty=false; } else m_hours[dn]=0; } } if (m_graphtype!=GT_SESSIONS) for (int j=0;j >::iterator d=PROFILE.daylist.begin();d!=PROFILE.daylist.end();d++) { tt=QDateTime(d.key(),QTime(0,0,0),Qt::UTC).toTime_t(); dn=tt/86400; for (int i=0;imachine_type()!=m_machinetype) continue; if (!m_values[dn].contains(j+1)) { m_days[dn]=day; m_values[dn][j+1]=0; if (!m_values[dn].contains(0)) { m_values[dn][0]=0; } if (0m_maxy) m_maxy=0; m_hours[dn]=day->hours(); } break; } } } m_empty=true; for (int i=0;i255) cr=255; if (cg>255) cg=255; if (cb>255) cb=255; return QColor(cr,cg,cb,255); } void SummaryChart::paint(gGraph & w,int left, int top, int width, int height) { if (!m_visible) return; rtop=top; GLShortBuffer *outlines=w.lines(); QColor blk=Qt::black; outlines->add(left, top, left, top+height, left, top+height, left+width,top+height, blk); outlines->add(left+width,top+height, left+width, top, left+width, top, left, top, blk); //if (outlines->full()) qDebug() << "WTF??? Outlines full in SummaryChart::paint()"; qint64 minx=w.min_x, maxx=w.max_x; //qint64 minx=m_minx, maxx=m_maxx; qint64 xx=maxx - minx; float days=double(xx)/86400000.0; EventDataType maxy=m_maxy; EventDataType miny=m_miny; w.roundY(miny,maxy); EventDataType yy=maxy-miny; EventDataType ymult=float(height-2)/yy; barw=(float(width)/float(days)); //qint64 ts; graph=&w; float px=left; l_left=w.marginLeft()+gYAxis::Margin; l_top=w.marginTop(); l_width=width; l_height=height; float py; EventDataType total; int daynum=0; EventDataType h,tmp,tmp2; l_offset=(minx) % 86400000L; offset=float(l_offset)/86400000.0; offset*=barw; px=left-offset; l_minx=minx; l_maxx=maxx+86400000L; //QHash lastvalues; int total_days=0; double total_val=0; //qint64 lastQ=0; bool lastdaygood=false; QVector totalcounts; QVector totalvalues; //QVector lastvalues; QVector lastX; QVector lastY; int numcodes=m_codes.size(); totalcounts.resize(numcodes); totalvalues.resize(numcodes); //lastvalues.resize(numcodes); lastX.resize(numcodes); lastY.resize(numcodes); int zd=minx/86400000L; zd--; QHash >::iterator d=m_values.find(zd); // if (d==m_values.end()) { // d=m_values.find(zd--); // } lastdaygood=true; for (int i=0;ileft+width) x2=left+width; if (x2 >::iterator times=m_times.find(zd); QColor col=m_colors[0]; //if (hoursadd(x1,py,x1,py-h,col); quads->add(x2,py-h,x2,py,col2); if (h>0 && barw>2) { outlines->add(x1,py,x1,py-h,x1,py-h,x2,py-h,blk); outlines->add(x1,py,x2,py,x2,py,x2,py-h,blk); } // if (bar //py-=h; totalvalues[0]+=tmp; } totalcounts[0]++; totalvalues[1]+=j; totalcounts[1]++; total_val+=hours; total_days++; } else { total=d.value()[0]; //if (total>0) { if (day) { total_val+=total; total_days++; } py=top+height; //} for (QHash::iterator g=d.value().begin();g!=d.value().end();g++) { short j=g.key(); if (!j) continue; j--; if (!m_goodcodes[j]) continue; tmp=g.value(); QColor col=m_colors[j]; if (m_type[j]==ST_HOURS) { if (tmptotalvalues[j]) totalvalues[j]=tmp; } else if (m_type[j]==ST_MIN) { if (tmpadd(x1,py,x1,py-h,col); quads->add(x2,py-h,x2,py,col2); if (h>0 && barw>2) { outlines->add(x1,py,x1,py-h,x1,py-h,x2,py-h,blk); outlines->add(x1,py,x2,py,x2,py,x2,py-h,blk); if (outlines->full()) qDebug() << "WTF??? Outlines full in SummaryChart::paint()"; } // if (bar py-=h; } else if (m_graphtype==GT_LINE) { // if (m_graphtype==GT_BAR col.setAlpha(128); short px2=px+barw; short py2=top+height-1-h; py2+=j; if (lastdaygood) { lines->add(lastX[j],lastY[j],px,py2,m_colors[j]); lines->add(px,py2,px2,py2,col); } else { lines->add(x1,py2,x2,py2,col); } lastX[j]=px2; lastY[j]=py2; //} } } // for(QHashmaxx+extra) break; } else { if (Qscissor(left,w.flipY(top+height+2),width+1,height+1); // Draw Ledgend px=left+width-3; py=top-5; QString a,b; int x,y; bool ishours=false; for (int j=0;j0) { if ((m_type[j]==ST_MIN) || (m_type[j]==ST_MAX)) { f=totalvalues[j]; } else { f=totalvalues[j]/totalcounts[j]; } } if (m_type[j]==ST_HOURS) { int h=f; int m=int(f*60) % 60; val.sprintf("%02i:%02i",h,m); ishours=true; } else { val=QString::number(f,'f',2); } a+="="+val; GetTextExtent(a,x,y); float wt=20*w.printScaleX(); px-=wt+x; w.renderText(a,px+wt,py+1); quads->add(px+wt-y/4-y,py-y,px+wt-y/4,py-y,px+wt-y/4,py+1,px+wt-y/4-y,py+1,m_colors[j]); //lines->add(px,py,px+20,py,m_colors[j]); //lines->add(px,py+1,px+20,py+1,m_colors[j]); } if (m_graphtype==GT_BAR) { if (m_type.size()>1) { float val=total_val/float(total_days); a=m_label+"="+QString::number(val,'f',2)+" "; GetTextExtent(a,x,y); px-=20+x; w.renderText(a,px+24,py+1); // } } a=""; /*if (m_graphtype==GT_BAR) { if (m_type.size()>1) { float val=total_val/float(total_days); a+=m_label+"="+QString::number(val,'f',2)+" "; // } }*/ a+="Days="+QString::number(total_days,'f',0); if (PROFILE.ExistsAndTrue("ShowCompliance")) { if (ishours && incompliant>0) { a+=" Low Usage Days="+QString::number(incompliant,'f',0)+" (%"+QString::number((1.0/daynum)*(total_days-incompliant)*100.0,'f',2)+" compliant, defined as >"+QString::number(compliance_hours,'f',1)+" hours)"; } } GetTextExtent(a,x,y); px-=30+x; //w.renderText(a,px+24,py+5); w.renderText(a,left,py+1); } QString formatTime(EventDataType v, bool show_seconds=false, bool duration=false,bool show_12hr=false) { int h=int(v); if (!duration) { h%=24; } else show_12hr=false; int m=int(v*60) % 60; int s=int(v*3600) % 60; char pm[3]={"am"}; if (show_12hr) { h>=12 ? pm[0]='p' : pm[0]='a'; h %= 12; if (h==0) h=12; } else { pm[0]=0; } if (show_seconds) return QString().sprintf("%i:%02i:%02i%s",h,m,s,pm); else return QString().sprintf("%i:%02i%s",h,m,pm); } bool SummaryChart::mouseMoveEvent(QMouseEvent *event) { int x=event->x()-l_left; int y=event->y()-l_top; if ((x<0 || y<0 || x>l_width || y>l_height)) { hl_day=-1; //graph->timedRedraw(2000); return false; } double xx=l_maxx-l_minx; double xmult=xx/double(l_width+barw); qint64 mx=ceil(xmult*double(x-offset)); mx+=l_minx; mx=mx+l_offset;//-86400000L; int zd=mx/86400000L; Day * day; //if (hl_day!=zd) // This line is an optimization { hl_day=zd; graph->Trigger(2000); QHash >::iterator d=m_values.find(hl_day); x+=gYAxis::Margin+gGraphView::titleWidth; //graph->m_marginleft+ int y=event->y()+rtop-15; //QDateTime dt1=QDateTime::fromTime_t(hl_day*86400).toLocalTime(); QDateTime dt2=QDateTime::fromTime_t(hl_day*86400).toUTC(); //QTime t1=dt1.time(); //QTime t2=dt2.time(); QDate dt=dt2.date(); if (d!=m_values.end()) { day=m_days[zd]; QString z=dt.toString(Qt::SystemLocaleShortDate); // Day * day=m_days[hl_day]; //EventDataType val; QString val; if (m_graphtype==GT_SESSIONS) { if (m_type[0]==ST_HOURS) { int t=day->hours()*3600.0; int h=t/3600; int m=(t / 60) % 60; //int s=t % 60; val.sprintf("%02i:%02i",h,m); } else val=QString::number(d.value()[0],'f',2); z+="\r\n"+m_label+"="+val; if (m_type[1]==ST_SESSIONS){ z+=" (Sess="+QString::number(day->size(),'f',0)+")"; } EventDataType v=m_times[zd][0]; int lastt=m_times[zd].size()-1; if (lastt<0) lastt=0; z+="\r\nBedtime="+formatTime(v,false,false,true); v=m_times[zd][lastt]+m_values[zd][lastt]; z+="\r\nWaketime="+formatTime(v,false,false,true); } else if (m_graphtype==GT_BAR){ if (m_type[0]==ST_HOURS) { int t=d.value()[0]*3600.0; int h=t/3600; int m=(t / 60) % 60; //int s=t % 60; val.sprintf("%02i:%02i",h,m); } else val=QString::number(d.value()[0],'f',2); z+="\r\n"+m_label+"="+val; //z+="\r\nMode="+QString::number(day->settings_min("FlexSet"),'f',0); } else { QString a; for (int i=0;iToolTip(z,x,y-15,2200); return true; } } return false; } bool SummaryChart::mousePressEvent(QMouseEvent * event) { if (event->modifiers() & Qt::ShiftModifier) { //qDebug() << "Jump to daily view?"; return true; } Q_UNUSED(event) return false; } bool SummaryChart::keyPressEvent(QKeyEvent * event) { Q_UNUSED(event) //qDebug() << "Summarychart Keypress"; return false; } #include "mainwindow.h" extern MainWindow *mainwin; bool SummaryChart::mouseReleaseEvent(QMouseEvent * event) { if (event->modifiers() & Qt::ShiftModifier) { if (hl_day<0) { mouseMoveEvent(event); } if (hl_day>0) { QDateTime d=QDateTime::fromTime_t(hl_day*86400).toUTC(); mainwin->getDaily()->LoadDate(d.date()); mainwin->JumpDaily(); //qDebug() << "Jump to daily view?" << d; return true; } } Q_UNUSED(event) hl_day=-1; graph->timedRedraw(2000); return false; }