/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- * vim: set ts=8 sts=4 et sw=4 tw=99: * * gLineOverlayBar Implementation * * Copyright (c) 2011-2014 Mark Watkins * * This file is subject to the terms and conditions of the GNU General Public * License. See the file COPYING in the main directory of the Linux * distribution for more details. */ #include #include "SleepLib/profiles.h" #include "gLineOverlay.h" gLineOverlayBar::gLineOverlayBar(ChannelID code,QColor color,QString label,FlagType flt) :Layer(code),m_flag_color(color),m_label(label),m_flt(flt) { addVertexBuffer(points=new gVertexBuffer(2048,GL_POINTS)); points->setSize(4); points->setColor(m_flag_color); addVertexBuffer(quads=new gVertexBuffer(2048,GL_QUADS)); //addGLBuf(lines=new GLBuffer(color,1024,GL_LINES)); points->setAntiAlias(true); quads->setAntiAlias(true); quads->setColor(m_flag_color); //lines->setAntiAlias(true); } gLineOverlayBar::~gLineOverlayBar() { //delete lines; //delete quads; //delete points; } void gLineOverlayBar::paint(gGraph & w, int left, int topp, int width, int height) { if (!m_visible) return; if (!m_day) return; gVertexBuffer * lines=w.lines(); int start_py=topp; double xx=w.max_x-w.min_x; double yy=w.max_y-w.min_y; if (xx<=0) return; float x1,x2; int x,y; float bottom=start_py+height-25*w.printScaleY(), top=start_py+25*w.printScaleY(); double X; double Y; bool verts_exceeded=false; m_count=0; m_sum=0; m_flag_color=schema::channel[m_code].defaultColor(); lines->setColor(m_flag_color); points->setColor(m_flag_color); if (m_flt==FT_Span) { m_flag_color.setAlpha(128); } EventStoreType raw; quint32 * tptr; EventStoreType * dptr, *eptr; qint64 stime; OverlayDisplayType odt=PROFILE.appearance->overlayType(); QHash >::iterator cei; int count; qint64 clockdrift=qint64(PROFILE.cpap->clockDrift()) * 1000L; qint64 drift=0; // For each session, process it's eventlist for (QList::iterator s=m_day->begin();s!=m_day->end(); s++) { if (!(*s)->enabled()) continue; cei=(*s)->eventlist.find(m_code); if (cei==(*s)->eventlist.end()) continue; QVector & evlist=cei.value(); if (evlist.size()==0) continue; drift=((*s)->machine()->GetType()==MT_CPAP) ? clockdrift : 0; // Could loop through here, but nowhere uses more than one yet.. for (int k=0;k= w.min_x) break; tptr++; } if (m_flt==FT_Span) { //////////////////////////////////////////////////////////////////////////// // FT_Span //////////////////////////////////////////////////////////////////////////// for (;dptr < eptr; dptr++) { X=stime + *tptr++; raw=*dptr; Y=X-(qint64(raw)*1000.0L); // duration if (Y > w.max_x) break; x1=double(width)/double(xx)*double(X-w.min_x)+left; m_count++; m_sum+=raw; x2=double(width)/double(xx)*double(Y-w.min_x)+left; if (int(x1)==int(x2)) x2+=1; if (x2width+left) x1=width+left; quads->add(x2,start_py, x1,start_py, x1,start_py+height, x2,start_py+height,m_flag_color.rgba()); if (quads->full()) { verts_exceeded=true; break; } } } else if (m_flt==FT_Dot) { //////////////////////////////////////////////////////////////////////////// // FT_Dot //////////////////////////////////////////////////////////////////////////// for (; dptr < eptr; dptr++) { X=stime + *tptr++; //el.time(i); raw=*dptr; //el.data(i); if (X > w.max_x) break; x1=double(width)/double(xx)*double(X-w.min_x)+left; m_count++; m_sum+=raw; if ((odt==ODT_Bars) || (xx<3600000)) { // show the fat dots in the middle points->add(x1,double(height)/double(yy)*double(-20-w.min_y)+topp); if (points->full()) { verts_exceeded=true; break; } } else { // thin lines down the bottom lines->add(x1,start_py+1,x1,start_py+1+12); if (lines->full()) { verts_exceeded=true; break; } } } } else if (m_flt==FT_Bar) { //////////////////////////////////////////////////////////////////////////// // FT_Bar //////////////////////////////////////////////////////////////////////////// for (; dptr < eptr; dptr++) { X=stime + *tptr++; raw=*dptr; if (X > w.max_x) break; x1=double(width)/double(xx)*double(X-w.min_x)+left; m_count++; m_sum+=raw; int z=start_py+height; if ((odt==ODT_Bars) || (xx<3600000)) { z=top; points->add(x1,top); lines->add(x1,top,x1,bottom); if (points->full()) { verts_exceeded=true; break; } } else { lines->add(x1,z,x1,z-12); } if (lines->full()) { verts_exceeded=true; break; } if (xx<(1800000)) { GetTextExtent(m_label,x,y); w.renderText(m_label,x1-(x/2),top-y+(3*w.printScaleY())); } } } if (verts_exceeded) break; } if (verts_exceeded) break; } if (verts_exceeded) { qWarning() << "exceeded maxverts in gLineOverlay::Plot()"; } } gLineOverlaySummary::gLineOverlaySummary(QString text, int x, int y) :Layer(CPAP_Obstructive),m_text(text),m_x(x),m_y(y) // The Layer code is a dummy here. { } gLineOverlaySummary::~gLineOverlaySummary() { } void gLineOverlaySummary::paint(gGraph & w,int left, int top, int width, int height) { if (!m_visible) return; if (!m_day) return; Q_UNUSED(width); Q_UNUSED(height); float cnt=0; double sum=0; bool isSpan=false; for (int i=0;icount(); sum+=m_overlays[i]->sum(); if (m_overlays[i]->flagtype()==FT_Span) isSpan=true; } double val,first,last; double time=0; // Calculate the session time. for (QList::iterator s=m_day->begin();s!=m_day->end(); s++) { if (!(*s)->enabled()) continue; first=(*s)->first(); last=(*s)->last(); if (last < w.min_x) continue; if (first > w.max_x) continue; if (first < w.min_x) first=w.min_x; if (last > w.max_x) last=w.max_x; time+=last-first; } val=0; time/=1000; int h=time/3600; int m=int(time/60) % 60; int s=int(time) % 60; time/=3600; //if (time<1) time=1; if (time>0) val=cnt/time; QString a=QObject::tr("Events")+"="+QString::number(cnt)+" "+QObject::tr("Duration")+" "+QString().sprintf("%02i:%02i:%02i",h,m,s)+", "+m_text+"="+QString::number(val,'f',2); if (isSpan) { float sph; if (!time) sph=0; else { sph=(100.0/float(time))*(sum/3600.0); if (sph>100) sph=100; } a+=" "+QObject::tr("(\%%1 in events)").arg(sph,0,'f',2); // eg: %num of time in a span, like Periodic Breathing } w.renderText(a,left+m_x,top+m_y); }